From e37175099a4971c4cc6a0ccbf4c8b860b23c4819 Mon Sep 17 00:00:00 2001 From: dengqichen Date: Wed, 11 Dec 2024 13:03:12 +0800 Subject: [PATCH] =?UTF-8?q?=E5=A2=9E=E5=8A=A0=E5=B7=A5=E5=85=B7=E6=A0=8F?= =?UTF-8?q?=E6=8F=90=E7=A4=BA=E3=80=82?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- .../Designer/components/NodePanel/index.tsx | 33 ++-- .../Definition/Designer/configs/nodeConfig.ts | 44 ++--- .../Workflow/Definition/Designer/index.tsx | 20 +-- .../pages/Workflow/Definition/Edit/index.tsx | 159 ------------------ .../src/pages/Workflow/Definition/index.tsx | 116 ++++++++----- frontend/src/pages/Workflow/service.ts | 10 +- frontend/src/pages/Workflow/types.ts | 31 ++-- frontend/src/router/index.tsx | 9 - 8 files changed, 130 insertions(+), 292 deletions(-) delete mode 100644 frontend/src/pages/Workflow/Definition/Edit/index.tsx diff --git a/frontend/src/pages/Workflow/Definition/Designer/components/NodePanel/index.tsx b/frontend/src/pages/Workflow/Definition/Designer/components/NodePanel/index.tsx index cd47e31d..8fca8159 100644 --- a/frontend/src/pages/Workflow/Definition/Designer/components/NodePanel/index.tsx +++ b/frontend/src/pages/Workflow/Definition/Designer/components/NodePanel/index.tsx @@ -1,7 +1,7 @@ import React, { useEffect, useState } from 'react'; import { Tabs, Spin } from 'antd'; -import { NodeType, getNodeTypes } from '../../service'; -import { NODE_CONFIG } from '../../configs/nodeConfig'; +import { NodeType, NodeCategory } from '../../../../types'; +import { getNodeTypes } from '../../../../service'; import './index.less'; interface NodePanelProps { @@ -17,10 +17,8 @@ const NodePanel: React.FC = ({ onNodeDragStart }) => { setLoading(true); try { const response = await getNodeTypes({ enabled: true }); - if (response) { - // 只保留有配置的节点类型 - const filteredTypes = response.filter(type => NODE_CONFIG[type.code]); - setNodeTypes(filteredTypes); + if (response?.content) { + setNodeTypes(response.content); } } catch (error) { console.error('获取节点类型失败:', error); @@ -49,39 +47,35 @@ const NodePanel: React.FC = ({ onNodeDragStart }) => { children: (
{types.map((nodeType) => { - const config = NODE_CONFIG[nodeType.code]; - if (!config) return null; - + const graphConfig = JSON.parse(nodeType.graphConfig || '{}'); return (
{ e.dataTransfer.setData('node-type', JSON.stringify({ ...nodeType, - config: config // 添加节点配置 + graphConfig })); onNodeDragStart(nodeType); }} style={{ - borderColor: config.theme.stroke, - backgroundColor: config.theme.fill + borderColor: graphConfig.attrs?.body?.stroke || '#5F95FF', + backgroundColor: graphConfig.attrs?.body?.fill || '#fff' }} >
- {config.extras?.icon ? ( + {graphConfig.attrs?.icon?.xlinkHref && ( {nodeType.name} - ) : ( - )}
{nodeType.name}
@@ -104,7 +98,7 @@ const NodePanel: React.FC = ({ onNodeDragStart }) => {
@@ -114,7 +108,6 @@ const NodePanel: React.FC = ({ onNodeDragStart }) => { // 获取类别标签 const getCategoryLabel = (category: string) => { const labels: Record = { - BASIC: '基础节点', TASK: '任务节点', EVENT: '事件节点', GATEWAY: '网关节点', diff --git a/frontend/src/pages/Workflow/Definition/Designer/configs/nodeConfig.ts b/frontend/src/pages/Workflow/Definition/Designer/configs/nodeConfig.ts index 330df759..4737f65b 100644 --- a/frontend/src/pages/Workflow/Definition/Designer/configs/nodeConfig.ts +++ b/frontend/src/pages/Workflow/Definition/Designer/configs/nodeConfig.ts @@ -1,14 +1,14 @@ import { NodeConfig } from '../types'; export const NODE_CONFIG: Record = { - START: { + startEvent: { size: { width: 80, height: 80 }, shape: 'circle', theme: { fill: '#f6ffed', stroke: '#52c41a' }, - label: '开始', + label: '开始节点', extras: { icon: { 'xlink:href': '', @@ -19,14 +19,14 @@ export const NODE_CONFIG: Record = { } } }, - END: { + endEvent: { size: { width: 80, height: 80 }, shape: 'circle', theme: { fill: '#fff1f0', stroke: '#ff4d4f' }, - label: '结束', + label: '结束节点', extras: { icon: { 'xlink:href': '', @@ -37,7 +37,7 @@ export const NODE_CONFIG: Record = { } } }, - SCRIPT: { + shellTask: { size: { width: 200, height: 80 }, shape: 'rect', theme: { @@ -57,42 +57,22 @@ export const NODE_CONFIG: Record = { } } }, - TIMER: { - size: { width: 200, height: 80 }, - shape: 'rect', - theme: { - fill: '#fff7e6', - stroke: '#fa8c16' - }, - label: '定时器', - extras: { - rx: 4, - ry: 4, - icon: { - 'xlink:href': '', - width: 32, - height: 32, - x: 8, - y: 24 - } - } - }, - GATEWAY: { + exclusiveGateway: { size: { width: 60, height: 60 }, shape: 'polygon', theme: { fill: '#f9f0ff', stroke: '#722ed1' }, - label: '网关', + label: '排他网关', extras: { refPoints: '0,30 30,0 60,30 30,60', icon: { - 'xlink:href': '', - width: 24, - height: 24, - x: 18, - y: 6 + 'xlink:href': '', + width: 32, + height: 32, + x: 14, + y: 14 } } } diff --git a/frontend/src/pages/Workflow/Definition/Designer/index.tsx b/frontend/src/pages/Workflow/Definition/Designer/index.tsx index d9813183..c7f4814f 100644 --- a/frontend/src/pages/Workflow/Definition/Designer/index.tsx +++ b/frontend/src/pages/Workflow/Definition/Designer/index.tsx @@ -105,7 +105,7 @@ const FlowDesigner: React.FC = () => { if (contextMenu.cell && contextMenu.cell.isNode()) { setCurrentNode(contextMenu.cell); const data = contextMenu.cell.getData() as NodeData; - const nodeType = nodeTypes.find(type => type.code === data.type); + const nodeType = nodeTypes.find(type => type.type === data.type); if (nodeType) { setCurrentNodeType(nodeType); const formValues = { @@ -198,7 +198,7 @@ const FlowDesigner: React.FC = () => { console.log('Node clicked, data:', data); if (data) { - const nodeType = nodeTypes.find(type => type.code === data.type); + const nodeType = nodeTypes.find(type => type.type === data.type); if (nodeType) { setCurrentNodeType(nodeType); const formValues = { @@ -273,13 +273,13 @@ const FlowDesigner: React.FC = () => { ); // 获取节点配置 - const position = calculateNodePosition(nodeType.code, dropPosition); - const nodeStyle = generateNodeStyle(nodeType.code); - const ports = generatePorts(nodeType.code); - const config = NODE_CONFIG[nodeType.code]; + const position = calculateNodePosition(nodeType.type, dropPosition); + const nodeStyle = generateNodeStyle(nodeType.type); + const ports = generatePorts(nodeType.type); + const config = NODE_CONFIG[nodeType.type]; if (!config) { - throw new Error(`未找到节点类型 ${nodeType.code} 的配置`); + throw new Error(`未找到节点类型 ${nodeType.type} 的配置`); } // 创建节点 @@ -288,8 +288,8 @@ const FlowDesigner: React.FC = () => { ...nodeStyle, ports, data: { - type: nodeType.code, - name: config.label, // 使用配置中的中文名称 + type: nodeType.type, + name: nodeType.name, config: {} as any, }, }); @@ -299,7 +299,7 @@ const FlowDesigner: React.FC = () => { graphRef.current.select(node); setCurrentNode(node); setCurrentNodeType(nodeType); - form.setFieldsValue({ name: config.label }); // 使用配置中的中文名称 + form.setFieldsValue({ name: nodeType.name }); setConfigVisible(true); message.success('节点创建成功'); diff --git a/frontend/src/pages/Workflow/Definition/Edit/index.tsx b/frontend/src/pages/Workflow/Definition/Edit/index.tsx deleted file mode 100644 index ccd90774..00000000 --- a/frontend/src/pages/Workflow/Definition/Edit/index.tsx +++ /dev/null @@ -1,159 +0,0 @@ -import React, {useEffect, useState} from 'react'; -import {useNavigate, useParams} from 'react-router-dom'; -import {Button, Card, Form, Input, message, Space, Spin} from 'antd'; -import {getDefinition, updateDefinition} from '../../service'; -import {UpdateWorkflowDefinitionRequest, WorkflowDefinition, WorkflowStatus} from '../../types'; -import {ArrowLeftOutlined} from '@ant-design/icons'; -import {WorkflowConfigUtils} from '../../../Workflow/utils'; - -const WorkflowDefinitionEdit: React.FC = () => { - const navigate = useNavigate(); - const {id} = useParams<{ id: string }>(); - const [form] = Form.useForm(); - const [loading, setLoading] = useState(false); - const [detail, setDetail] = useState(); - - // 获取详情 - const fetchDetail = async () => { - if (!id) return; - setLoading(true); - try { - const response = await getDefinition(parseInt(id)); - if (response) { - setDetail(response); - // 设置表单初始值 - form.setFieldsValue({ - code: response.code, - name: response.name, - description: response.description, - }); - } - } finally { - setLoading(false); - } - }; - - // 首次加载 - useEffect(() => { - fetchDetail(); - }, [id]); - - // 处理保存 - const handleSave = async (values: any) => { - if (!id || !detail) return; - try { - // 保留原有配置和状态 - const data: UpdateWorkflowDefinitionRequest = { - ...values, - status: detail.status, - version: detail.version, - enabled: detail.enabled, - nodeConfig: JSON.stringify(WorkflowConfigUtils.parseNodeConfig(detail.nodeConfig)), - transitionConfig: JSON.stringify(WorkflowConfigUtils.parseTransitionConfig(detail.transitionConfig)), - formDefinition: JSON.stringify(WorkflowConfigUtils.parseFormDefinition(detail.formDefinition)), - graphDefinition: JSON.stringify(WorkflowConfigUtils.parseGraphDefinition(detail.graphDefinition)) - }; - await updateDefinition(parseInt(id), data); - message.success('保存成功'); - navigate('/workflow/definition'); - } catch (error) { - // 错误已在请求拦截器中处理 - } - }; - - // 处理取消 - const handleCancel = () => { - navigate('/workflow/definition'); - }; - - if (loading) { - return ( -
- -
- ); - } - - return ( - } onClick={handleCancel}> - 返回 - - } - > -
- {detail?.status !== WorkflowStatus.DRAFT && ( -
- 注意:非草稿状态的流程定义不能修改! -
- )} - -
- - - - - - - - - - - - - - - - -
-
-
- ); -}; - -export default WorkflowDefinitionEdit; \ No newline at end of file diff --git a/frontend/src/pages/Workflow/Definition/index.tsx b/frontend/src/pages/Workflow/Definition/index.tsx index 8df3d45d..511972ac 100644 --- a/frontend/src/pages/Workflow/Definition/index.tsx +++ b/frontend/src/pages/Workflow/Definition/index.tsx @@ -3,6 +3,7 @@ import {Button, Card, Form, Input, message, Modal, Select, Space, Table, Tag} fr import {useNavigate} from 'react-router-dom'; import { createDefinition, + updateDefinition, deleteDefinition, disableDefinition, enableDefinition, @@ -27,6 +28,7 @@ const WorkflowDefinitionList: React.FC = () => { const [current, setCurrent] = useState(1); const [size, setSize] = useState(10); const [createModalVisible, setCreateModalVisible] = useState(false); + const [editingRecord, setEditingRecord] = useState(null); // 获取列表数据 const fetchList = async (page = current, pageSize = size) => { @@ -72,33 +74,44 @@ const WorkflowDefinitionList: React.FC = () => { fetchList(1); }; - // 处理创建 - const handleCreate = async (values: WorkflowDefinitionBase) => { + // 处理创建或更新 + const handleSubmit = async (values: WorkflowDefinitionBase) => { try { - // 设置系统字段的默认值 - const data = { - ...values, - status: WorkflowStatus.DRAFT, - version: 1, - enabled: true, - nodeConfig: JSON.stringify({ - nodes: [] - }), - transitionConfig: JSON.stringify({ - transitions: [] - }), - formDefinition: JSON.stringify({}), - graphDefinition: JSON.stringify({}) - }; - await createDefinition(data); - message.success('创建成功'); + if (editingRecord) { + // 更新流程 + const updateData = { + ...values, + status: editingRecord.status, + flowVersion: editingRecord.flowVersion + }; + await updateDefinition(editingRecord.id, updateData); + message.success('更新成功'); + } else { + // 创建流程 + const data = { + ...values, + status: WorkflowStatus.DRAFT, + flowVersion: 1 + }; + await createDefinition(data); + message.success('创建成功'); + } setCreateModalVisible(false); + setEditingRecord(null); + form.resetFields(); fetchList(); } catch (error) { // 错误已在请求拦截器中处理 } }; + // 处理弹窗关闭 + const handleModalClose = () => { + setCreateModalVisible(false); + setEditingRecord(null); + form.resetFields(); + }; + // 处理删除 const handleDelete = (record: WorkflowDefinition) => { confirm({ @@ -143,6 +156,17 @@ const WorkflowDefinitionList: React.FC = () => { } }; + // 处理编辑 + const handleEdit = (record: WorkflowDefinition) => { + setEditingRecord(record); + form.setFieldsValue({ + key: record.key, + name: record.name, + description: record.description + }); + setCreateModalVisible(true); + }; + // 表格列定义 const columns = [ { @@ -224,13 +248,14 @@ const WorkflowDefinitionList: React.FC = () => { width: 380, render: (_: any, record: WorkflowDefinition) => ( - + {record.status === WorkflowStatus.DRAFT && ( + + )} - diff --git a/frontend/src/pages/Workflow/service.ts b/frontend/src/pages/Workflow/service.ts index 2313a203..56e940c7 100644 --- a/frontend/src/pages/Workflow/service.ts +++ b/frontend/src/pages/Workflow/service.ts @@ -10,7 +10,7 @@ import { } from './types'; const WORKFLOW_DEFINITION_URL = '/api/v1/workflow/definition'; -const NODE_TYPE_URL = '/api/v1/node-types'; +const NODE_DEFINITION_URL = '/api/v1/workflow/node-definition'; // 创建工作流定义 export const createDefinition = (data: CreateWorkflowDefinitionRequest) => @@ -44,13 +44,13 @@ export const disableDefinition = (id: number) => export const enableDefinition = (id: number) => request.post(`${WORKFLOW_DEFINITION_URL}/${id}/enable`); -// 创建新版本 +// ���建新版本 export const createVersion = (id: number) => request.post(`${WORKFLOW_DEFINITION_URL}/${id}/versions`); -// 节点类型相关接口 +// 获取节点类型列表 export const getNodeTypes = (params?: NodeTypeQuery) => - request.get(NODE_TYPE_URL, { params }); + request.get>(`${NODE_DEFINITION_URL}/page`, { params }); export const getNodeTypeExecutors = (code: string) => - request.get(`${NODE_TYPE_URL}/${code}/executors`); \ No newline at end of file + request.get(`${NODE_DEFINITION_URL}/${code}/executors`); \ No newline at end of file diff --git a/frontend/src/pages/Workflow/types.ts b/frontend/src/pages/Workflow/types.ts index c8809f2d..a8ee89f1 100644 --- a/frontend/src/pages/Workflow/types.ts +++ b/frontend/src/pages/Workflow/types.ts @@ -32,18 +32,22 @@ export interface NodeExecutor { // 节点类型 export interface NodeType { id: number; - code: string; + type: string; name: string; - category: NodeCategory; description: string; + category: NodeCategory; + flowableConfig: string; // JSON string + graphConfig: string; // JSON string + formConfig: string; // JSON string + orderNum: number; enabled: boolean; - icon: string; - color: string; - executors: NodeExecutor[]; - configSchema: string; // JSON Schema - defaultConfig: string; createTime: string; updateTime: string; + createBy: string; + updateBy: string; + version: number; + deleted: boolean; + extraData: any; } // 工作流定义查询参数 @@ -77,15 +81,16 @@ export interface TransitionConfig { // 工作流定义基本信息 export interface WorkflowDefinitionBase { - code: string; name: string; + key: string; + flowVersion?: number; + status?: WorkflowStatus; description?: string; } // 工作流定义 export interface WorkflowDefinition extends WorkflowDefinitionBase { id: number; - version: number; status: WorkflowStatus; enabled: boolean; nodeConfig: string; // JSON string of NodeConfig @@ -94,17 +99,13 @@ export interface WorkflowDefinition extends WorkflowDefinitionBase { graphDefinition: string; // JSON string createTime: string; updateTime: string; + createBy: string; + updateBy: string; } // 创建工作流定义请求 export interface CreateWorkflowDefinitionRequest extends WorkflowDefinitionBase { status: WorkflowStatus; - version: number; - enabled: boolean; - nodeConfig: string; // JSON string of NodeConfig - transitionConfig: string; // JSON string of TransitionConfig - formDefinition: string; // JSON string - graphDefinition: string; // JSON string } // 更新工作流定义请求 diff --git a/frontend/src/router/index.tsx b/frontend/src/router/index.tsx index dcbb2d3f..21cc1c8a 100644 --- a/frontend/src/router/index.tsx +++ b/frontend/src/router/index.tsx @@ -33,7 +33,6 @@ const Department = lazy(() => import('../pages/System/Department')); const External = lazy(() => import('../pages/System/External')); const X6Test = lazy(() => import('../pages/X6Test')); const WorkflowDefinition = lazy(() => import('../pages/Workflow/Definition')); -const WorkflowDefinitionEdit = lazy(() => import('../pages/Workflow/Definition/Edit')); const WorkflowInstance = lazy(() => import('../pages/Workflow/Instance')); const WorkflowMonitor = lazy(() => import('../pages/Workflow/Monitor')); const FlowDesigner = lazy(() => import('../pages/Workflow/Definition/Designer')); @@ -132,14 +131,6 @@ const router = createBrowserRouter([ ) }, - { - path: 'edit/:id?', - element: ( - }> - - - ) - }, { path: 'designer/:id?', element: (