/* * 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 dayjs from 'dayjs'; import utc from 'dayjs/plugin/utc'; import { cloneDeep, set as objectSet } from 'lodash-es'; import type { DataSchema, DataSourceDeps, Id, MComponent, MNode } from '@tmagic/schema'; import { NodeType } from '@tmagic/schema'; export * from './dom'; export const sleep = (ms: number): Promise => new Promise((resolve) => { const timer = setTimeout(() => { clearTimeout(timer); resolve(); }, ms); }); export const datetimeFormatter = ( v: string | Date, defaultValue = '-', format = 'YYYY-MM-DD HH:mm:ss', ): string | number => { if (v) { let time: string | number; if (['x', 'timestamp'].includes(format)) { time = dayjs(v).valueOf(); } else if ((typeof v === 'string' && v.includes('Z')) || v.constructor === Date) { dayjs.extend(utc); // UTC字符串时间或Date对象格式化为北京时间 time = dayjs(v).utcOffset(8).format(format); } else { time = dayjs(v).format(format); } if (time !== 'Invalid Date') { return time; } return defaultValue; } return defaultValue; }; // 驼峰转换横线 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 filterXSS = (str: string) => str.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 = {}, ): 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 = {}): 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()), ); 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()), ); /** * 将新节点更新到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 = {}; const defaultValue: Record = { 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 (obj: T) => Array; 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, 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;