tmagic-editor/packages/editor/tests/unit/utils/props-config.spec.ts
2026-05-14 15:26:22 +08:00

144 lines
5.7 KiB
TypeScript

/*
* Tencent is pleased to support the open source community by making TMagicEditor available.
*
* Copyright (C) 2025 Tencent.
*/
import { describe, expect, test, vi } from 'vitest';
import { NODE_CONDS_RESULT_KEY } from '@tmagic/core';
import {
advancedTabConfig,
arrayOptions,
displayTabConfig,
eqOptions,
eventTabConfig,
fillConfig,
numberOptions,
styleTabConfig,
} from '@editor/utils/props';
vi.mock('@tmagic/design', () => ({
tMagicMessage: {
success: vi.fn(),
error: vi.fn(),
},
}));
describe('props 选项常量', () => {
test('eqOptions / arrayOptions / numberOptions 内容稳定', () => {
expect(eqOptions.map((o) => o.value)).toEqual(['=', '!=']);
expect(arrayOptions.map((o) => o.value)).toEqual(['include', 'not_include']);
expect(numberOptions.map((o) => o.value)).toEqual(['>', '>=', '<', '<=', 'between', 'not_between']);
});
test('styleTabConfig / eventTabConfig / displayTabConfig 基础结构', () => {
expect(styleTabConfig.title).toBe('样式');
expect(eventTabConfig.title).toBe('事件');
expect(displayTabConfig.title).toBe('显示条件');
});
test('advancedTabConfig 中包含 created/mounted/display', () => {
const names = (advancedTabConfig.items as any[]).map((i) => i.name);
expect(names).toEqual(expect.arrayContaining(['created', 'mounted', 'display']));
});
});
describe('fillConfig', () => {
test('默认会补充 type/id/name 三个基础字段', () => {
const result = fillConfig() as any;
const tab = result[0];
const propsTab = tab.items[0];
const names = propsTab.items.map((i: any) => i.name);
expect(names).toEqual(expect.arrayContaining(['type', 'id', 'name']));
});
test('已有 id/type/name 时不重复添加', () => {
const result = fillConfig([
{ name: 'id', text: 'id', type: 'text' } as any,
{ name: 'type', text: 'type', type: 'hidden' } as any,
{ name: 'name', text: 'name' } as any,
]) as any;
const propsTab = result[0].items[0];
const idItems = propsTab.items.filter((i: any) => i.name === 'id');
expect(idItems).toHaveLength(1);
});
test('disabledDataSource=true 时不追加 displayTab', () => {
const result = fillConfig([], { disabledDataSource: true }) as any;
const titles = result[0].items.map((i: any) => i.title);
expect(titles).not.toContain('显示条件');
});
test('支持自定义 labelWidth', () => {
const result = fillConfig([], { labelWidth: '120px' }) as any;
expect(result[0].labelWidth).toBe('120px');
});
test('styleTabConfig.display 跟随 uiService showStylePanel', () => {
const services = { uiService: { get: () => false } };
expect(styleTabConfig.display!({ services } as any)).toBe(true);
services.uiService.get = () => true;
expect(styleTabConfig.display!({ services } as any)).toBe(false);
services.uiService.get = () => undefined;
expect(styleTabConfig.display!({ services } as any)).toBe(false);
});
test('styleTabConfig 内 transform 项 defaultValue 返回 {}', () => {
const styleField = (styleTabConfig.items as any[])[0];
const transform = styleField.items.find((i: any) => i.name === 'transform');
expect(transform.defaultValue()).toEqual({});
});
test('displayTabConfig.display 当节点为 page 时返回 false', () => {
expect(displayTabConfig.display!({} as any, { model: { type: 'page' } } as any)).toBe(false);
expect(displayTabConfig.display!({} as any, { model: { type: 'text' } } as any)).toBe(true);
});
test('displayTabConfig select 项 extra 文案随 NODE_CONDS_RESULT_KEY 变化', () => {
const selectItem = (displayTabConfig.items as any[]).find((i) => i.type === 'select');
expect(typeof selectItem.extra).toBe('function');
const text1 = selectItem.extra({}, { model: { [NODE_CONDS_RESULT_KEY]: true } });
const text2 = selectItem.extra({}, { model: { [NODE_CONDS_RESULT_KEY]: false } });
expect(text1).toContain('隐藏');
expect(text2).toContain('显示');
});
test('id 字段 append.handler 触发 navigator.clipboard.writeText', async () => {
const result = fillConfig() as any;
const propsTab = result[0].items[0];
const idField = propsTab.items.find((i: any) => i.name === 'id');
const writeText = vi.fn().mockResolvedValue(undefined);
Object.defineProperty(navigator, 'clipboard', { value: { writeText }, configurable: true });
await idField.append.handler({}, { model: { id: 'abc' } });
expect(writeText).toHaveBeenCalledWith('abc');
});
test('id 字段 append.handler 复制失败时调用 error', async () => {
const result = fillConfig() as any;
const propsTab = result[0].items[0];
const idField = propsTab.items.find((i: any) => i.name === 'id');
const writeText = vi.fn().mockRejectedValue(new Error('fail'));
Object.defineProperty(navigator, 'clipboard', { value: { writeText }, configurable: true });
await idField.append.handler({}, { model: { id: 'xx' } });
expect(writeText).toHaveBeenCalled();
});
test('disabledCodeBlock=true 时高级 tab 仅保留非 code-select 项', () => {
const result = fillConfig([], { disabledCodeBlock: true }) as any;
const titles = result[0].items.map((i: any) => i.title);
expect(titles).toContain('高级');
const advanced = result[0].items.find((i: any) => i.title === '高级');
advanced.items.forEach((it: any) => {
expect(it.type).not.toBe('code-select');
});
});
test('advancedTabConfig 中所有 type=code-select 都有 labelPosition=top', () => {
const codeSelects = (advancedTabConfig.items as any[]).filter((i) => i.type === 'code-select');
codeSelects.forEach((cs: any) => {
expect(cs.labelPosition).toBe('top');
});
});
});