mirror of
https://github.com/Tencent/tmagic-editor.git
synced 2025-04-04 23:06:31 +08:00
494 lines
13 KiB
TypeScript
494 lines
13 KiB
TypeScript
/*
|
||
* Tencent is pleased to support the open source community by making TMagicEditor available.
|
||
*
|
||
* Copyright (C) 2023 THL A29 Limited, a Tencent company. All rights reserved.
|
||
*
|
||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||
* you may not use this file except in compliance with the License.
|
||
* You may obtain a copy of the License at
|
||
*
|
||
* http://www.apache.org/licenses/LICENSE-2.0
|
||
*
|
||
* Unless required by applicable law or agreed to in writing, software
|
||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
* See the License for the specific language governing permissions and
|
||
* limitations under the License.
|
||
*/
|
||
|
||
import { cloneDeep, set as objectSet } from 'lodash-es';
|
||
|
||
import type {
|
||
DataSchema,
|
||
DataSourceDeps,
|
||
Id,
|
||
MApp,
|
||
MComponent,
|
||
MContainer,
|
||
MNode,
|
||
MNodeInstance,
|
||
MPage,
|
||
MPageFragment,
|
||
} from '@tmagic/schema';
|
||
import { NodeType } from '@tmagic/schema';
|
||
|
||
import type { EditorNodeInfo } from '@editor/type';
|
||
|
||
export * from './dom';
|
||
|
||
export const sleep = (ms: number): Promise<void> =>
|
||
new Promise((resolve) => {
|
||
const timer = setTimeout(() => {
|
||
clearTimeout(timer);
|
||
resolve();
|
||
}, ms);
|
||
});
|
||
|
||
// 驼峰转换横线
|
||
export const toLine = (name = '') => name.replace(/\B([A-Z])/g, '-$1').toLowerCase();
|
||
|
||
export const toHump = (name = ''): string => name.replace(/-(\w)/g, (_all, letter) => letter.toUpperCase());
|
||
|
||
export const emptyFn = (): any => undefined;
|
||
|
||
/**
|
||
* 通过id获取组件在应用的子孙路径
|
||
* @param {number | string} id 组件id
|
||
* @param {Array} data 要查找的根容器节点
|
||
* @return {Array} 组件在data中的子孙路径
|
||
*/
|
||
export const getNodePath = (id: Id, data: MNode[] = []): MNode[] => {
|
||
const path: MNode[] = [];
|
||
|
||
const get = function (id: number | string, data: MNode[]): MNode | null {
|
||
if (!Array.isArray(data)) {
|
||
return null;
|
||
}
|
||
|
||
for (let i = 0, l = data.length; i < l; i++) {
|
||
const item = data[i];
|
||
|
||
path.push(item);
|
||
if (`${item.id}` === `${id}`) {
|
||
return item;
|
||
}
|
||
|
||
if (item.items) {
|
||
const node = get(id, item.items);
|
||
if (node) {
|
||
return node;
|
||
}
|
||
}
|
||
|
||
path.pop();
|
||
}
|
||
|
||
return null;
|
||
};
|
||
|
||
get(id, data);
|
||
|
||
return path;
|
||
};
|
||
|
||
export const getNodeInfo = (id: Id, root: MApp | null) => {
|
||
const info: EditorNodeInfo = {
|
||
node: null,
|
||
parent: null,
|
||
page: null,
|
||
};
|
||
|
||
if (!root) return info;
|
||
|
||
if (id === root.id) {
|
||
info.node = root;
|
||
return info;
|
||
}
|
||
|
||
const path = getNodePath(id, root.items);
|
||
|
||
if (!path.length) return info;
|
||
|
||
path.unshift(root);
|
||
|
||
info.node = path[path.length - 1] as MComponent;
|
||
info.parent = path[path.length - 2] as MContainer;
|
||
|
||
path.forEach((item) => {
|
||
if (isPage(item) || isPageFragment(item)) {
|
||
info.page = item as MPage | MPageFragment;
|
||
return;
|
||
}
|
||
});
|
||
|
||
return info;
|
||
};
|
||
|
||
export const filterXSS = (str: string) =>
|
||
str.replace(/</g, '<').replace(/>/g, '>').replace(/"/g, '"').replace(/'/g, ''');
|
||
|
||
export const getUrlParam = (param: string, url?: string) => {
|
||
const u = url || location.href;
|
||
const reg = new RegExp(`[?&#]${param}=([^&#]+)`, 'gi');
|
||
|
||
const matches = u.match(reg);
|
||
let strArr;
|
||
if (matches && matches.length > 0) {
|
||
strArr = matches[matches.length - 1].split('=');
|
||
if (strArr && strArr.length > 1) {
|
||
// 过滤XSS字符
|
||
return filterXSS(strArr[1]);
|
||
}
|
||
return '';
|
||
}
|
||
return '';
|
||
};
|
||
|
||
export const isObject = (obj: any) => Object.prototype.toString.call(obj) === '[object Object]';
|
||
|
||
export const isPop = (node: MComponent | null): boolean => Boolean(node?.type?.toLowerCase().endsWith('pop'));
|
||
|
||
export const isPage = (node?: MComponent | null): boolean => {
|
||
if (!node) return false;
|
||
return Boolean(node.type?.toLowerCase() === NodeType.PAGE);
|
||
};
|
||
|
||
export const isPageFragment = (node?: MComponent | null): boolean => {
|
||
if (!node) return false;
|
||
return Boolean(node.type?.toLowerCase() === NodeType.PAGE_FRAGMENT);
|
||
};
|
||
|
||
export const isNumber = (value: string) => /^(-?\d+)(\.\d+)?$/.test(value);
|
||
|
||
export const getHost = (targetUrl: string) => targetUrl.match(/\/\/([^/]+)/)?.[1];
|
||
|
||
export const isSameDomain = (targetUrl = '', source = globalThis.location.host) => {
|
||
const isHttpUrl = /^(http[s]?:)?\/\//.test(targetUrl);
|
||
|
||
if (!isHttpUrl) return true;
|
||
|
||
return getHost(targetUrl) === source;
|
||
};
|
||
|
||
/**
|
||
* 生成指定位数的GUID,无【-】格式
|
||
* @param digit 位数,默认值8
|
||
* @returns
|
||
*/
|
||
export const guid = (digit = 8): string =>
|
||
'x'.repeat(digit).replace(/[xy]/g, (c) => {
|
||
const r = (Math.random() * 16) | 0;
|
||
const v = c === 'x' ? r : (r & 0x3) | 0x8;
|
||
return v.toString(16);
|
||
});
|
||
|
||
export const getKeysArray = (keys: string | number) =>
|
||
// 将 array[0] 转成 array.0
|
||
`${keys}`.replaceAll(/\[(\d+)\]/g, '.$1').split('.');
|
||
|
||
export const getValueByKeyPath = (
|
||
keys: number | string | string[] = '',
|
||
data: Record<string | number, any> = {},
|
||
): any => {
|
||
// 将 array[0] 转成 array.0
|
||
const keyArray = Array.isArray(keys) ? keys : getKeysArray(keys);
|
||
return keyArray.reduce((accumulator, currentValue: any) => {
|
||
if (isObject(accumulator)) {
|
||
return accumulator[currentValue];
|
||
}
|
||
|
||
if (Array.isArray(accumulator) && /^\d*$/.test(`${currentValue}`)) {
|
||
return accumulator[currentValue];
|
||
}
|
||
|
||
throw new Error(`${data}中不存在${keys}`);
|
||
}, data);
|
||
};
|
||
|
||
export const setValueByKeyPath = (keys: string | number, value: any, data: Record<string | number, any> = {}): any =>
|
||
objectSet(data, keys, value);
|
||
|
||
export const getNodes = (ids: Id[], data: MNode[] = []): MNode[] => {
|
||
const nodes: MNode[] = [];
|
||
|
||
const get = function (ids: Id[], data: MNode[]) {
|
||
if (!Array.isArray(data)) {
|
||
return;
|
||
}
|
||
|
||
for (let i = 0, l = data.length; i < l; i++) {
|
||
const item = data[i];
|
||
const index = ids.findIndex((id: Id) => `${id}` === `${item.id}`);
|
||
|
||
if (index > -1) {
|
||
ids.slice(index, 1);
|
||
nodes.push(item);
|
||
}
|
||
|
||
if (item.items) {
|
||
get(ids, item.items);
|
||
}
|
||
}
|
||
};
|
||
|
||
get(ids, data);
|
||
|
||
return nodes;
|
||
};
|
||
|
||
export const getDepKeys = (dataSourceDeps: DataSourceDeps = {}, nodeId: Id) =>
|
||
Array.from(
|
||
Object.values(dataSourceDeps).reduce((prev, cur) => {
|
||
(cur[nodeId]?.keys || []).forEach((key) => prev.add(key));
|
||
return prev;
|
||
}, new Set<Id>()),
|
||
);
|
||
|
||
export const getDepNodeIds = (dataSourceDeps: DataSourceDeps = {}) =>
|
||
Array.from(
|
||
Object.values(dataSourceDeps).reduce((prev, cur) => {
|
||
Object.keys(cur).forEach((id) => {
|
||
prev.add(id);
|
||
});
|
||
return prev;
|
||
}, new Set<string>()),
|
||
);
|
||
|
||
/**
|
||
* 将新节点更新到data或者parentId对应的节点的子节点中
|
||
* @param newNode 新节点
|
||
* @param data 需要修改的数据
|
||
* @param parentId 父节点 id
|
||
*/
|
||
export const replaceChildNode = (newNode: MNode, data?: MNode[], parentId?: Id) => {
|
||
const path = getNodePath(newNode.id, data);
|
||
const node = path.pop();
|
||
let parent = path.pop();
|
||
|
||
if (parentId) {
|
||
parent = getNodePath(parentId, data).pop();
|
||
}
|
||
|
||
if (!node) throw new Error('未找到目标节点');
|
||
if (!parent) throw new Error('未找到父节点');
|
||
|
||
const index = parent.items?.findIndex((child: MNode) => child.id === node.id);
|
||
parent.items.splice(index, 1, newNode);
|
||
};
|
||
|
||
export const DSL_NODE_KEY_COPY_PREFIX = '__tmagic__';
|
||
export const IS_DSL_NODE_KEY = '__tmagic__dslNode';
|
||
|
||
export const compiledNode = (
|
||
compile: (value: any) => any,
|
||
node: MNode,
|
||
dataSourceDeps: DataSourceDeps = {},
|
||
sourceId?: Id,
|
||
) => {
|
||
let keys: Id[] = [];
|
||
if (!sourceId) {
|
||
keys = getDepKeys(dataSourceDeps, node.id);
|
||
} else {
|
||
const dep = dataSourceDeps[sourceId];
|
||
keys = dep?.[node.id].keys || [];
|
||
}
|
||
|
||
keys.forEach((key) => {
|
||
const keys = getKeysArray(key);
|
||
|
||
const cacheKey = keys.map((key, index) => {
|
||
if (index < keys.length - 1) {
|
||
return key;
|
||
}
|
||
return `${DSL_NODE_KEY_COPY_PREFIX}${key}`;
|
||
});
|
||
|
||
let templateValue = getValueByKeyPath(cacheKey, node);
|
||
if (typeof templateValue === 'undefined') {
|
||
try {
|
||
const value = getValueByKeyPath(key, node);
|
||
setValueByKeyPath(cacheKey.join('.'), value, node);
|
||
templateValue = value;
|
||
} catch (e) {
|
||
console.warn(e);
|
||
return;
|
||
}
|
||
}
|
||
|
||
let newValue;
|
||
try {
|
||
newValue = compile(templateValue);
|
||
} catch (e) {
|
||
console.error(e);
|
||
newValue = '';
|
||
}
|
||
|
||
setValueByKeyPath(key, newValue, node);
|
||
});
|
||
|
||
return node;
|
||
};
|
||
|
||
export const compiledCond = (op: string, fieldValue: any, inputValue: any, range: number[] = []): boolean => {
|
||
if (typeof fieldValue === 'string' && typeof inputValue === 'undefined') {
|
||
inputValue = '';
|
||
}
|
||
|
||
switch (op) {
|
||
case 'is':
|
||
return fieldValue === inputValue;
|
||
case 'not':
|
||
return fieldValue !== inputValue;
|
||
case '=':
|
||
return fieldValue === inputValue;
|
||
case '!=':
|
||
return fieldValue !== inputValue;
|
||
case '>':
|
||
return fieldValue > inputValue;
|
||
case '>=':
|
||
return fieldValue >= inputValue;
|
||
case '<':
|
||
return fieldValue < inputValue;
|
||
case '<=':
|
||
return fieldValue <= inputValue;
|
||
case 'between':
|
||
return range.length > 1 && fieldValue >= range[0] && fieldValue <= range[1];
|
||
case 'not_between':
|
||
return range.length < 2 || fieldValue < range[0] || fieldValue > range[1];
|
||
case 'include':
|
||
return fieldValue?.includes?.(inputValue);
|
||
case 'not_include':
|
||
return typeof fieldValue === 'undefined' || !fieldValue.includes?.(inputValue);
|
||
default:
|
||
break;
|
||
}
|
||
|
||
return false;
|
||
};
|
||
|
||
export const getDefaultValueFromFields = (fields: DataSchema[]) => {
|
||
const data: Record<string, any> = {};
|
||
|
||
const defaultValue: Record<string, any> = {
|
||
string: undefined,
|
||
object: {},
|
||
array: [],
|
||
boolean: undefined,
|
||
number: undefined,
|
||
null: null,
|
||
any: undefined,
|
||
};
|
||
|
||
fields.forEach((field) => {
|
||
if (typeof field.defaultValue !== 'undefined') {
|
||
if (field.type === 'array' && !Array.isArray(field.defaultValue)) {
|
||
data[field.name] = defaultValue.array;
|
||
return;
|
||
}
|
||
|
||
if (field.type === 'object' && !isObject(field.defaultValue)) {
|
||
if (typeof field.defaultValue === 'string') {
|
||
try {
|
||
data[field.name] = JSON.parse(field.defaultValue);
|
||
} catch (e) {
|
||
data[field.name] = defaultValue.object;
|
||
}
|
||
return;
|
||
}
|
||
|
||
data[field.name] = defaultValue.object;
|
||
return;
|
||
}
|
||
|
||
data[field.name] = cloneDeep(field.defaultValue);
|
||
|
||
return;
|
||
}
|
||
|
||
if (field.type === 'object') {
|
||
data[field.name] = field.fields ? getDefaultValueFromFields(field.fields) : defaultValue.object;
|
||
return;
|
||
}
|
||
|
||
if (field.type) {
|
||
data[field.name] = defaultValue[field.type];
|
||
return;
|
||
}
|
||
|
||
data[field.name] = undefined;
|
||
});
|
||
|
||
return data;
|
||
};
|
||
|
||
export const DATA_SOURCE_FIELDS_SELECT_VALUE_PREFIX = 'ds-field::';
|
||
|
||
export const DATA_SOURCE_FIELDS_CHANGE_EVENT_PREFIX = 'ds-field-changed';
|
||
|
||
export const getKeys = Object.keys as <T extends object>(obj: T) => Array<keyof T>;
|
||
|
||
export const calculatePercentage = (value: number, percentageStr: string) => {
|
||
const percentage = globalThis.parseFloat(percentageStr) / 100; // 先将百分比字符串转换为浮点数,并除以100转换为小数
|
||
const result = value * percentage;
|
||
return result;
|
||
};
|
||
|
||
export const isPercentage = (value: number | string) => /^(\d+)(\.\d+)?%$/.test(`${value}`);
|
||
|
||
export const convertToNumber = (value: number | string, parentValue = 0) => {
|
||
if (typeof value === 'number') {
|
||
return value;
|
||
}
|
||
|
||
if (typeof value === 'string' && isPercentage(value)) {
|
||
return calculatePercentage(parentValue, value);
|
||
}
|
||
|
||
return parseFloat(value);
|
||
};
|
||
|
||
/**
|
||
* 添加参数到URL
|
||
* @param obj 参数对象
|
||
* @param global window对象
|
||
* @param needReload 是否需要刷新
|
||
*/
|
||
export const addParamToUrl = (obj: Record<string, any>, global = globalThis, needReload = true) => {
|
||
const url = new URL(global.location.href);
|
||
const { searchParams } = url;
|
||
for (const [k, v] of Object.entries(obj)) {
|
||
searchParams.set(k, v);
|
||
}
|
||
const newUrl = url.toString();
|
||
if (needReload) {
|
||
global.location.href = newUrl;
|
||
} else {
|
||
global.history.pushState({}, '', url);
|
||
}
|
||
};
|
||
|
||
export const dataSourceTemplateRegExp = /\$\{([\s\S]+?)\}/g;
|
||
|
||
export const isDslNode = (config: MNodeInstance) =>
|
||
typeof config[IS_DSL_NODE_KEY] === 'undefined' || config[IS_DSL_NODE_KEY] === true;
|
||
|
||
export interface NodeItem {
|
||
items?: NodeItem[];
|
||
[key: string]: any;
|
||
}
|
||
|
||
export const traverseNode = <T extends NodeItem = NodeItem>(
|
||
node: T,
|
||
cb: (node: T, parents: T[]) => void,
|
||
parents: T[] = [],
|
||
) => {
|
||
cb(node, parents);
|
||
|
||
if (Array.isArray(node.items) && node.items.length) {
|
||
parents.push(node);
|
||
node.items.forEach((item) => {
|
||
traverseNode(item as T, cb, [...parents]);
|
||
});
|
||
}
|
||
};
|