refactor(core): 完善事件逻辑

This commit is contained in:
roymondchen 2024-05-14 14:05:01 +08:00
parent 88c04c6dac
commit 5074c9e68b
4 changed files with 100 additions and 94 deletions

View File

@ -20,7 +20,7 @@ import { EventEmitter } from 'events';
import { has, isEmpty } from 'lodash-es'; import { has, isEmpty } from 'lodash-es';
import { createDataSourceManager, DataSourceManager, ObservedDataClass } from '@tmagic/data-source'; import { createDataSourceManager, DataSource, DataSourceManager, ObservedDataClass } from '@tmagic/data-source';
import { import {
ActionType, ActionType,
type AppCore, type AppCore,
@ -28,7 +28,7 @@ import {
type CodeItemConfig, type CodeItemConfig,
type CompItemConfig, type CompItemConfig,
type DataSourceItemConfig, type DataSourceItemConfig,
type DeprecatedEventConfig, type EventActionItem,
type EventConfig, type EventConfig,
type Id, type Id,
type JsEngine, type JsEngine,
@ -41,7 +41,7 @@ import Env from './Env';
import { bindCommonEventListener, isCommonMethod, triggerCommonMethod } from './events'; import { bindCommonEventListener, isCommonMethod, triggerCommonMethod } from './events';
import Node from './Node'; import Node from './Node';
import Page from './Page'; import Page from './Page';
import { fillBackgroundImage, isNumber, style2Obj } from './utils'; import { calcFontsize, transformStyle as defaultTransformStyle } from './utils';
interface AppOptionsConfig { interface AppOptionsConfig {
ua?: string; ua?: string;
@ -57,7 +57,7 @@ interface AppOptionsConfig {
} }
interface EventCache { interface EventCache {
eventConfig: CompItemConfig | DeprecatedEventConfig; eventConfig: CompItemConfig;
fromCpt: any; fromCpt: any;
args: any[]; args: any[];
} }
@ -72,13 +72,14 @@ class App extends EventEmitter implements AppCore {
public useMock = false; public useMock = false;
public platform = 'mobile'; public platform = 'mobile';
public jsEngine = 'browser'; public jsEngine: JsEngine = 'browser';
public designWidth = 375; public designWidth = 375;
public request?: RequestFunction; public request?: RequestFunction;
public components = new Map(); public components = new Map();
public eventQueueMap: Record<string, EventCache[]> = {}; public eventQueueMap: Record<string, EventCache[]> = {};
public transformStyle: (style: Record<string, any>) => Record<string, any>;
private eventList = new Map<(fromCpt: Node, ...args: any[]) => void, string>(); private eventList = new Map<(fromCpt: Node, ...args: any[]) => void, string>();
private dataSourceEventList = new Map<string, Map<string, (...args: any[]) => void>>(); private dataSourceEventList = new Map<string, Map<string, (...args: any[]) => void>>();
@ -100,9 +101,8 @@ class App extends EventEmitter implements AppCore {
this.setDesignWidth(options.designWidth); this.setDesignWidth(options.designWidth);
} }
if (options.transformStyle) { this.transformStyle =
this.transformStyle = options.transformStyle; options.transformStyle || ((style: Record<string, any>) => defaultTransformStyle(style, this.jsEngine));
}
if (options.request) { if (options.request) {
this.request = options.request; this.request = options.request;
@ -129,45 +129,6 @@ class App extends EventEmitter implements AppCore {
} }
} }
/**
* dsl中的style配置转换成cssrem为单位的样式值1001rem
* @param style Object
* @returns Object
*/
public transformStyle(style: Record<string, any> | string) {
if (!style) {
return {};
}
let styleObj: Record<string, any> = {};
const results: Record<string, any> = {};
if (typeof style === 'string') {
styleObj = style2Obj(style);
} else {
styleObj = { ...style };
}
const isHippy = this.jsEngine === 'hippy';
const whiteList = ['zIndex', 'opacity', 'fontWeight'];
Object.entries(styleObj).forEach(([key, value]) => {
if (key === 'scale' && !results.transform && isHippy) {
results.transform = [{ scale: value }];
} else if (key === 'backgroundImage' && !isHippy) {
value && (results[key] = fillBackgroundImage(value));
} else if (key === 'transform' && typeof value !== 'string') {
results[key] = this.getTransform(value);
} else if (!whiteList.includes(key) && value && /^[-]?[0-9]*[.]?[0-9]*$/.test(value)) {
results[key] = isHippy ? value : `${value / 100}rem`;
} else {
results[key] = value;
}
});
return results;
}
/** /**
* dsl * dsl
* @param config dsl跟节点 * @param config dsl跟节点
@ -305,7 +266,7 @@ class App extends EventEmitter implements AppCore {
* @param eventConfig * @param eventConfig
* @returns void * @returns void
*/ */
public async compActionHandler(eventConfig: CompItemConfig | DeprecatedEventConfig, fromCpt: any, args: any[]) { public async compActionHandler(eventConfig: CompItemConfig, fromCpt: Node | DataSource, args: any[]) {
if (!this.page) throw new Error('当前没有页面'); if (!this.page) throw new Error('当前没有页面');
const { method: methodName, to } = eventConfig; const { method: methodName, to } = eventConfig;
@ -397,33 +358,41 @@ class App extends EventEmitter implements AppCore {
}); });
} }
private async actionHandler(actionItem: EventActionItem, fromCpt: Node | DataSource, args: any[]) {
if (actionItem.actionType === ActionType.COMP) {
// 组件动作
await this.compActionHandler(actionItem as CompItemConfig, fromCpt, args);
} else if (actionItem.actionType === ActionType.CODE) {
// 执行代码块
await this.codeActionHandler(actionItem as CodeItemConfig, args);
} else if (actionItem.actionType === ActionType.DATA_SOURCE) {
await this.dataSourceActionHandler(actionItem as DataSourceItemConfig, args);
}
}
/** /**
* *
* @param eventsConfigIndex node.event中获取最新事件配置 * @param eventsConfigIndex node.event中获取最新事件配置
* @param fromCpt * @param fromCpt
* @param args * @param args
*/ */
private async eventHandler(eventsConfigIndex: number, fromCpt: Node, args: any[]) { private async eventHandler(config: EventConfig | number, fromCpt: Node | DataSource | undefined, args: any[]) {
const eventConfig = fromCpt.events[eventsConfigIndex] as EventConfig | DeprecatedEventConfig; const eventConfig = typeof config === 'number' ? (fromCpt as Node).events[config] : config;
if (has(eventConfig, 'actions')) { if (has(eventConfig, 'actions')) {
// EventConfig类型 // EventConfig类型
const { actions } = eventConfig as EventConfig; const { actions } = eventConfig as EventConfig;
for (let i = 0; i < actions.length; i++) { for (let i = 0; i < actions.length; i++) {
// 事件响应中可能会有修改数据源数据的会更新dsl所以这里需要重新获取 if (typeof config === 'number') {
const actionItem = (fromCpt.events[eventsConfigIndex] as EventConfig).actions[i]; // 事件响应中可能会有修改数据源数据的会更新dsl所以这里需要重新获取
if (actionItem.actionType === ActionType.COMP) { const actionItem = ((fromCpt as Node).events[config] as EventConfig).actions[i];
// 组件动作 this.actionHandler(actionItem, fromCpt as Node, args);
await this.compActionHandler(actionItem as CompItemConfig, fromCpt, args); } else {
} else if (actionItem.actionType === ActionType.CODE) { this.actionHandler(actions[i], fromCpt as DataSource, args);
// 执行代码块
await this.codeActionHandler(actionItem as CodeItemConfig, args);
} else if (actionItem.actionType === ActionType.DATA_SOURCE) {
await this.dataSourceActionHandler(actionItem as DataSourceItemConfig, args);
} }
} }
} else { } else {
// 兼容DeprecatedEventConfig类型 组件动作 // 兼容DeprecatedEventConfig类型 组件动作
await this.compActionHandler(eventConfig as DeprecatedEventConfig, fromCpt, args); await this.compActionHandler(eventConfig as unknown as CompItemConfig, fromCpt as Node, args);
} }
} }
@ -435,29 +404,8 @@ class App extends EventEmitter implements AppCore {
} }
} }
private getTransform(value: Record<string, string>) {
if (!value) return [];
const transform = Object.entries(value).map(([transformKey, transformValue]) => {
if (!transformValue.trim()) return '';
if (transformKey === 'rotate' && isNumber(transformValue)) {
transformValue = `${transformValue}deg`;
}
return this.jsEngine !== 'hippy' ? `${transformKey}(${transformValue})` : { [transformKey]: transformValue };
});
if (this.jsEngine === 'hippy') {
return transform;
}
const values = transform.join(' ');
return !values.trim() ? 'none' : values;
}
private calcFontsize() { private calcFontsize() {
const { width } = document.documentElement.getBoundingClientRect(); calcFontsize(this.designWidth);
const fontSize = width / (this.designWidth / 100);
document.documentElement.style.fontSize = `${fontSize}px`;
} }
} }

View File

@ -19,15 +19,7 @@
import { EventEmitter } from 'events'; import { EventEmitter } from 'events';
import { DataSource } from '@tmagic/data-source'; import { DataSource } from '@tmagic/data-source';
import type { import type { AppCore, EventConfig, MComponent, MContainer, MPage, MPageFragment } from '@tmagic/schema';
AppCore,
DeprecatedEventConfig,
EventConfig,
MComponent,
MContainer,
MPage,
MPageFragment,
} from '@tmagic/schema';
import { HookCodeType, HookType } from '@tmagic/schema'; import { HookCodeType, HookType } from '@tmagic/schema';
import type App from './App'; import type App from './App';
@ -45,7 +37,7 @@ class Node extends EventEmitter {
public style?: { public style?: {
[key: string]: any; [key: string]: any;
}; };
public events: DeprecatedEventConfig[] | EventConfig[] = []; public events: EventConfig[] = [];
public instance?: any; public instance?: any;
public page?: Page; public page?: Page;
public parent?: Node; public parent?: Node;

View File

@ -16,6 +16,8 @@
* limitations under the License. * limitations under the License.
*/ */
import { JsEngine } from '@tmagic/schema';
export const style2Obj = (style: string) => { export const style2Obj = (style: string) => {
if (typeof style !== 'string') { if (typeof style !== 'string') {
return style; return style;
@ -55,3 +57,67 @@ export const fillBackgroundImage = (value: string) => {
}; };
export const isNumber = (value: string) => /^(-?\d+)(\.\d+)?$/.test(value); export const isNumber = (value: string) => /^(-?\d+)(\.\d+)?$/.test(value);
export const getTransform = (value: Record<string, string>, jsEngine: JsEngine) => {
if (!value) return [];
const transform = Object.entries(value).map(([transformKey, transformValue]) => {
if (!transformValue.trim()) return '';
if (transformKey === 'rotate' && isNumber(transformValue)) {
transformValue = `${transformValue}deg`;
}
return jsEngine !== 'hippy' ? `${transformKey}(${transformValue})` : { [transformKey]: transformValue };
});
if (jsEngine === 'hippy') {
return transform;
}
const values = transform.join(' ');
return !values.trim() ? 'none' : values;
};
/**
* dsl中的style配置转换成cssrem为单位的样式值1001rem
* @param style Object
* @returns Object
*/
export const transformStyle = (style: Record<string, any> | string, jsEngine: JsEngine) => {
if (!style) {
return {};
}
let styleObj: Record<string, any> = {};
const results: Record<string, any> = {};
if (typeof style === 'string') {
styleObj = style2Obj(style);
} else {
styleObj = { ...style };
}
const isHippy = jsEngine === 'hippy';
const whiteList = ['zIndex', 'opacity', 'fontWeight'];
Object.entries(styleObj).forEach(([key, value]) => {
if (key === 'scale' && !results.transform && isHippy) {
results.transform = [{ scale: value }];
} else if (key === 'backgroundImage' && !isHippy) {
value && (results[key] = fillBackgroundImage(value));
} else if (key === 'transform' && typeof value !== 'string') {
results[key] = getTransform(value, jsEngine);
} else if (!whiteList.includes(key) && value && /^[-]?[0-9]*[.]?[0-9]*$/.test(value)) {
results[key] = isHippy ? value : `${value / 100}rem`;
} else {
results[key] = value;
}
});
return results;
};
export const calcFontsize = (designWidth: number) => {
const { width } = document.documentElement.getBoundingClientRect();
const fontSize = width / (designWidth / 100);
document.documentElement.style.fontSize = `${fontSize}px`;
};

View File

@ -130,7 +130,7 @@ export interface MComponent {
/** 组件根Dom上的class */ /** 组件根Dom上的class */
className?: string; className?: string;
/* 关联事件集合 */ /* 关联事件集合 */
events?: EventConfig[] | DeprecatedEventConfig[]; events?: EventConfig[];
/** 组件根Dom的style */ /** 组件根Dom的style */
style?: { style?: {
[key: string]: any; [key: string]: any;