This commit is contained in:
dengqichen 2024-12-19 16:37:03 +08:00
parent c47a63f620
commit dd57d2d8c8

View File

@ -1,16 +1,16 @@
import React, { useEffect, useState } from 'react'; import React, { useEffect, useState } from 'react';
import { PageContainer } from '@ant-design/pro-components'; import { PageContainer } from '@ant-design/pro-components';
import { Row, Col, Menu, Tabs } from 'antd'; import { Row, Col, Menu, Tabs, Card } from 'antd';
import FormRender, { useForm } from 'form-render'; import { BetaSchemaForm, ProForm, ProFormGroup } from '@ant-design/pro-form';
import type { NodeDefinitionData } from './types'; import type { NodeDefinitionData } from './types';
import * as service from './service'; import * as service from './service';
// Tab 配置 // Tab 配置
const TAB_CONFIG = [ const TAB_CONFIG = [
{ key: 'panel', label: '属性(预览)', schemaKey: 'panelVariablesSchema' }, { key: 'panel', label: '属性(预览)', schemaKey: 'panelVariablesSchema', readonly: true },
{ key: 'local', label: '环境变量(预览)', schemaKey: 'localVariablesSchema' }, { key: 'local', label: '环境变量(预览)', schemaKey: 'localVariablesSchema', readonly: true },
{ key: 'form', label: '表单(预览)', schemaKey: 'formVariablesSchema' }, { key: 'form', label: '表单(预览)', schemaKey: 'formVariablesSchema', readonly: true },
{ key: 'ui', label: 'UI配置', schemaKey: 'uiVariables' } { key: 'ui', label: 'UI配置', schemaKey: 'uiVariables', readonly: false }
]; ];
const NodeDesignForm: React.FC = () => { const NodeDesignForm: React.FC = () => {
@ -21,8 +21,6 @@ const NodeDesignForm: React.FC = () => {
const [selectedNode, setSelectedNode] = useState<NodeDefinitionData | null>(null); const [selectedNode, setSelectedNode] = useState<NodeDefinitionData | null>(null);
// 当前选中的 tab // 当前选中的 tab
const [activeTab, setActiveTab] = useState<string>('panel'); const [activeTab, setActiveTab] = useState<string>('panel');
// form 实例
const form = useForm();
// 加载节点定义数据 // 加载节点定义数据
useEffect(() => { useEffect(() => {
@ -65,127 +63,166 @@ const NodeDesignForm: React.FC = () => {
if (availableTabs.length > 0 && !availableTabs.find(tab => tab.key === activeTab)) { if (availableTabs.length > 0 && !availableTabs.find(tab => tab.key === activeTab)) {
setActiveTab(availableTabs[0].key); setActiveTab(availableTabs[0].key);
} }
// 重置表单
form.resetFields();
} }
}; };
// 处理 Tab 切换 // 处理 Tab 切换
const handleTabChange = (newTab: string) => { const handleTabChange = (newTab: string) => {
setActiveTab(newTab); setActiveTab(newTab);
// 重置表单
form.resetFields();
}; };
// 处理 schema 的 title // 处理表单提交
const processSchemaTitle = (schema: any) => { const handleFormSubmit = async (values: any) => {
if (!schema || typeof schema !== 'object') return schema; console.log('表单提交:', values);
// TODO: 调用接口保存数据
};
const newSchema = { ...schema }; // 递归获取默认值
const getDefaultValues = (schema: any) => {
if (!schema || !schema.properties) return {};
// 如果有 properties处理每个属性 return Object.entries(schema.properties).reduce((acc, [key, value]: [string, any]) => {
if (newSchema.properties) { if (value.type === 'object') {
newSchema.properties = Object.entries(newSchema.properties).reduce((acc, [key, value]: [string, any]) => { acc[key] = getDefaultValues(value);
const newValue = { ...value }; } else if (value.type === 'array' && value.items) {
// 如果有 title添加 key acc[key] = value.default || [];
if (newValue.title) { } else {
newValue.title = `${newValue.title}-${key}`; acc[key] = value.default;
} }
// 递归处理嵌套的 properties return acc;
if (newValue.properties) {
newValue.properties = processSchemaTitle(newValue).properties;
}
return { ...acc, [key]: newValue };
}, {}); }, {});
}
return newSchema;
}; };
// 获取表单 schema // 将 JSON Schema 转换为 ProForm Schema
const getFormSchema = (tabKey: string) => { const convertToProFormSchema = (schema: any, parentKey = '', level = 0) => {
if (!selectedNode) return null; if (!schema || !schema.properties) return [];
const content = selectedNode[tabKey as keyof NodeDefinitionData]; return Object.entries(schema.properties).map(([key, value]: [string, any]) => {
if (!content) return null; const fullKey = parentKey ? `${parentKey}.${key}` : key;
const baseField = {
// 如果内容本身就是 schema 格式,处理 title 后返回 title: value.title || key,
if (typeof content === 'object' && 'type' in content && 'properties' in content) { dataIndex: fullKey,
return processSchemaTitle(content); tooltip: value.description,
fieldProps: {
style: { width: '100%' }
} }
};
// 否则,构建一个 schema // 处理枚举类型
if (value.enum) {
return { return {
type: 'object', ...baseField,
properties: Object.entries(content).reduce((acc, [key, value]) => { valueType: 'select',
let fieldSchema: any = { valueEnum: value.enum.reduce((acc: any, item: string, index: number) => {
title: key, acc[item] = { text: value.enumNames?.[index] || item };
}; return acc;
// 根据值的类型设置不同的 schema
if (typeof value === 'string') {
fieldSchema = {
...fieldSchema,
type: 'string',
default: value
};
} else if (typeof value === 'number') {
fieldSchema = {
...fieldSchema,
type: 'number',
default: value
};
} else if (typeof value === 'boolean') {
fieldSchema = {
...fieldSchema,
type: 'boolean',
default: value
};
} else if (Array.isArray(value)) {
fieldSchema = {
...fieldSchema,
type: 'array',
items: {
type: typeof value[0] || 'string'
},
default: value
};
} else if (typeof value === 'object' && value !== null) {
fieldSchema = {
...fieldSchema,
type: 'object',
properties: this.getFormSchema(value)?.properties || {},
default: value
};
}
return {
...acc,
[key]: fieldSchema
};
}, {}) }, {})
}; };
}
// 处理不同类型的字段
if (value.type === 'string') {
return {
...baseField,
valueType: 'text'
};
} else if (value.type === 'number' || value.type === 'integer') {
return {
...baseField,
valueType: 'digit'
};
} else if (value.type === 'boolean') {
return {
...baseField,
valueType: 'switch'
};
} else if (value.type === 'array') {
const itemColumns = value.items ? convertToProFormSchema(
{ properties: { item: value.items } },
`${fullKey}.item`,
level + 1
) : [];
return {
...baseField,
valueType: 'formList',
columns: itemColumns
};
} else if (value.type === 'object') {
const childColumns = value.properties ? convertToProFormSchema(value, fullKey, level + 1) : [];
return {
...baseField,
valueType: 'group',
columns: childColumns,
colProps: {
span: 24
},
fieldProps: {
title: value.title,
style: {
padding: level === 0 ? '16px' : '8px',
marginBottom: '16px',
border: '1px solid #f0f0f0',
borderRadius: '4px',
background: level === 0 ? '#fafafa' : 'transparent'
}
}
};
}
return baseField;
});
}; };
// 渲染 Tab 内容 // 渲染 Tab 内容
const renderTabContent = (tabKey: string) => { const renderTabContent = (tabKey: string) => {
if (!selectedNode) return null; if (!selectedNode) return null;
const schema = getFormSchema(tabKey); const content = selectedNode[tabKey as keyof NodeDefinitionData];
if (!schema) return null; if (!content) return null;
const currentTab = TAB_CONFIG.find(tab => tab.schemaKey === tabKey);
const isReadOnly = currentTab?.readonly ?? true;
// 判断是否是 schema 格式
const isSchema = typeof content === 'object' && 'type' in content && 'properties' in content;
const schema = isSchema ? content : {
type: 'object',
properties: Object.entries(content).reduce((acc, [key, value]) => ({
...acc,
[key]: {
title: key,
type: typeof value,
default: value
}
}), {})
};
// 获取默认值
const defaultValues = getDefaultValues(schema);
return ( return (
<div style={{ padding: '16px' }}> <Card bordered={false}>
<FormRender <BetaSchemaForm
form={form} layoutType="Form"
schema={schema} readonly={isReadOnly}
disabled={true} onFinish={handleFormSubmit}
displayType="column" columns={convertToProFormSchema(schema)}
column={1} initialValues={defaultValues}
readOnly={true} submitter={!isReadOnly}
watch={{}} grid={false}
layout="vertical"
style={{ maxWidth: '100%' }}
formItemProps={{
labelCol: { span: 24 },
wrapperCol: { span: 24 },
style: {
marginBottom: 16
}
}}
/> />
</div> </Card>
); );
}; };