2024-12-09 20:21:34 +08:00

494 lines
13 KiB
TypeScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

/*
* 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, '&lt;').replace(/>/g, '&gt;').replace(/"/g, '&quot;').replace(/'/g, '&apos;');
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]);
});
}
};