mirror of
https://github.com/Tencent/tmagic-editor.git
synced 2025-04-06 03:57:56 +08:00
refactor(core): 完善事件逻辑
This commit is contained in:
parent
88c04c6dac
commit
5074c9e68b
@ -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配置转换成css,主要是将数值转成rem为单位的样式值,例如100将被转换成1rem
|
|
||||||
* @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`;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -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;
|
||||||
|
@ -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配置转换成css,主要是将数值转成rem为单位的样式值,例如100将被转换成1rem
|
||||||
|
* @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`;
|
||||||
|
};
|
||||||
|
@ -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;
|
||||||
|
Loading…
x
Reference in New Issue
Block a user