Browse Source

feat: FlowChart 流程图组件 & test

lanceJiang 1 year ago
parent
commit
121d3e9494

+ 154 - 0
src/components/FlowChart/FlowChartToolbar.vue

@@ -0,0 +1,154 @@
+<template>
+	<div class="le-flow-chart_toolbar">
+		<template v-for="item in toolbarItemList" :key="item.type">
+			<el-tooltip placement="bottom" v-bind="item.disabled ? { visible: false } : {}">
+				<template #content>{{ item.tooltip }}</template>
+				<span v-if="item.icon" class="le-flow-chart_toolbar__icon" @click="onControl(item)">
+					<PickerIcon :icon-class="item.icon" :class="item.disabled ? 'cursor-not-allowed disabled' : 'cursor-pointer'"/>
+<!--					<LeIcon :icon="item.icon" :class="item.disabled ? 'cursor-not-allowed disabled' : 'cursor-pointer'" />-->
+				</span>
+			</el-tooltip>
+		</template>
+	</div>
+</template>
+<script lang="ts" setup>
+import { ToolbarConfig, ToolbarTypeEnum } from './types'
+import { ref, onUnmounted, unref, nextTick, watchEffect } from 'vue'
+import { useFlowChartContext } from './useFlowContext'
+import PickerIcon from '@/components/IconPicker/PickerIcon.vue'
+defineOptions({ name: 'FlowChartToolbar' })
+
+defineProps({})
+
+const emit = defineEmits(['view-data'])
+
+const toolbarItemList = ref<ToolbarConfig[]>([
+	{
+		type: ToolbarTypeEnum.ZOOM_IN,
+		icon: 'zoom-out',
+		tooltip: '缩小'
+	},
+	{
+		type: ToolbarTypeEnum.ZOOM_OUT,
+		icon: 'zoom-in',
+		tooltip: '放大'
+	},
+	{
+		type: ToolbarTypeEnum.RESET_ZOOM,
+		icon: 'le-suoxiao1',
+		tooltip: '重置比例'
+	},
+	{ separate: true },
+	{
+		type: ToolbarTypeEnum.UNDO,
+		icon: 'DArrowLeft',
+		tooltip: '后退',
+		disabled: true
+	},
+	{
+		type: ToolbarTypeEnum.REDO,
+		icon: 'DArrowRight',
+		tooltip: '前进',
+		disabled: true
+	},
+	{ separate: true },
+	{
+		type: ToolbarTypeEnum.SNAPSHOT,
+		icon: 'Download',
+		tooltip: '下载'
+	},
+	{
+		type: ToolbarTypeEnum.VIEW_DATA,
+		icon: 'Document',
+		tooltip: '查看数据'
+	}
+])
+
+const { logicFlow } = useFlowChartContext()
+
+function onHistoryChange({ data: { undoAble, redoAble } }) {
+	const itemsList = unref(toolbarItemList)
+	const undoIndex = itemsList.findIndex(item => item.type === ToolbarTypeEnum.UNDO)
+	const redoIndex = itemsList.findIndex(item => item.type === ToolbarTypeEnum.REDO)
+	if (undoIndex !== -1) {
+		unref(toolbarItemList)[undoIndex].disabled = !undoAble
+	}
+	if (redoIndex !== -1) {
+		unref(toolbarItemList)[redoIndex].disabled = !redoAble
+	}
+}
+
+const onControl = item => {
+	const lf = unref(logicFlow)
+	if (!lf) {
+		return
+	}
+	switch (item.type) {
+		case ToolbarTypeEnum.ZOOM_IN:
+			lf.zoom()
+			break
+		case ToolbarTypeEnum.ZOOM_OUT:
+			lf.zoom(true)
+			break
+		case ToolbarTypeEnum.RESET_ZOOM:
+			lf.resetZoom()
+			break
+		case ToolbarTypeEnum.UNDO:
+			lf.undo()
+			break
+		case ToolbarTypeEnum.REDO:
+			lf.redo()
+			break
+		case ToolbarTypeEnum.SNAPSHOT:
+			lf.getSnapshot()
+			break
+		case ToolbarTypeEnum.VIEW_DATA:
+			emit('view-data')
+			break
+	}
+}
+
+watchEffect(async () => {
+	if (unref(logicFlow)) {
+		await nextTick()
+		unref(logicFlow)?.on('history:change', onHistoryChange)
+	}
+})
+
+onUnmounted(() => {
+	unref(logicFlow)?.off('history:change', onHistoryChange)
+})
+</script>
+<style lang="scss">
+//@prefix-cls: ~'@{namespace}-flow-chart_toolbar';
+.#{$prefix}flow-chart_toolbar {
+	display: flex;
+	align-items: center;
+	padding: 0.25rem 0.5rem;
+	//padding-bottom: 0.25rem;
+	height: 36px;
+	border-bottom: 1px solid var(--el-border-color);
+	background-color: var(--el-bg-color-page);
+
+	.disabeld {
+		color: var(--el-text-color-disabled);
+	}
+
+	&__icon {
+		display: inline-block;
+		margin-right: 10px;
+		padding: 2px 4px;
+		font-size: 14px;
+		color: var(--el-text-color-primary);
+		cursor: pointer;
+		&:hover {
+			color: var(--el-color-primary);
+		}
+	}
+}
+/*html[data-theme='dark'] {
+	.lf-dnd {
+		background: #080808;
+	}
+}*/
+</style>

+ 75 - 0
src/components/FlowChart/adpterForTurbo.ts

@@ -0,0 +1,75 @@
+const TurboType = {
+	SEQUENCE_FLOW: 1,
+	START_EVENT: 2,
+	END_EVENT: 3,
+	USER_TASK: 4,
+	SERVICE_TASK: 5,
+	EXCLUSIVE_GATEWAY: 6
+}
+
+function convertFlowElementToEdge(element: any) {
+	const { incoming, outgoing, properties, key } = element
+	const { text, startPoint, endPoint, pointsList, logicFlowType } = properties
+	const edge = {
+		id: key,
+		type: logicFlowType,
+		sourceNodeId: incoming[0],
+		targetNodeId: outgoing[0],
+		text,
+		startPoint,
+		endPoint,
+		pointsList,
+		properties: {} as any
+	}
+	const excludeProperties = ['startPoint', 'endPoint', 'pointsList', 'text', 'logicFlowType']
+	Object.keys(element.properties).forEach(property => {
+		if (excludeProperties.indexOf(property) === -1) {
+			edge.properties[property] = element.properties[property]
+		}
+	})
+	return edge
+}
+
+function convertFlowElementToNode(element: any) {
+	const { properties, key } = element
+	const { x, y, text, logicFlowType } = properties
+	const node = {
+		id: key,
+		type: logicFlowType,
+		x,
+		y,
+		text,
+		properties: {} as any
+	}
+	const excludeProperties = ['x', 'y', 'text', 'logicFlowType']
+	Object.keys(element.properties).forEach(property => {
+		if (excludeProperties.indexOf(property) === -1) {
+			node.properties[property] = element.properties[property]
+		}
+	})
+	return node
+}
+
+export function toLogicFlowData(data: any) {
+	const lfData: {
+		// TODO type
+		nodes: any[]
+		edges: any[]
+	} = {
+		nodes: [],
+		edges: []
+	}
+	const list = data.flowElementList
+	list &&
+		list.length > 0 &&
+		list.forEach((element: any) => {
+			if (element.type === TurboType.SEQUENCE_FLOW) {
+				const edge = convertFlowElementToEdge(element)
+				lfData.edges.push(edge)
+			} else {
+				const node = convertFlowElementToNode(element)
+				lfData.nodes.push(node)
+			}
+		})
+	return lfData
+}

+ 96 - 0
src/components/FlowChart/config.ts

@@ -0,0 +1,96 @@
+export const nodeList = [
+	{
+		text: '开始',
+		type: 'start',
+		class: 'node-start'
+	},
+	{
+		text: '矩形',
+		type: 'rect',
+		class: 'node-rect'
+	},
+	{
+		type: 'user',
+		text: '用户',
+		class: 'node-user'
+	},
+	{
+		type: 'push',
+		text: '推送',
+		class: 'node-push'
+	},
+	{
+		type: 'download',
+		text: '位置',
+		class: 'node-download'
+	},
+	{
+		type: 'end',
+		text: '结束',
+		class: 'node-end'
+	}
+]
+
+export const BpmnNode = [
+	{
+		type: 'bpmn:startEvent',
+		text: '开始',
+		class: 'bpmn-start'
+	},
+	{
+		type: 'bpmn:endEvent',
+		text: '结束',
+		class: 'bpmn-end'
+	},
+	{
+		type: 'bpmn:exclusiveGateway',
+		text: '网关',
+		class: 'bpmn-exclusiveGateway'
+	},
+	{
+		type: 'bpmn:userTask',
+		text: '用户',
+		class: 'bpmn-user'
+	}
+]
+
+export function configDefaultDndPanel(lf: any) {
+	return [
+		{
+			text: '选区',
+			icon: '',
+			callback: () => {
+				lf.updateEditConfig({
+					stopMoveGraph: true
+				})
+			}
+		},
+		{
+			type: 'circle',
+			text: '开始',
+			icon: ''
+		},
+		{
+			type: 'rect',
+			text: '用户任务',
+			icon: '',
+			cls: 'important-node'
+		},
+		{
+			type: 'rect',
+			text: '系统任务',
+			icon: '',
+			cls: 'import_icon'
+		},
+		{
+			type: 'diamond',
+			text: '条件判断',
+			icon: ''
+		},
+		{
+			type: 'circle',
+			text: '结束',
+			icon: ''
+		}
+	]
+}

+ 158 - 0
src/components/FlowChart/index.vue

@@ -0,0 +1,158 @@
+<template>
+	<div class="le-flow-chart">
+		<FlowChartToolbar v-if="toolbar" @view-data="handlePreview" />
+		<div ref="lfElRef" class="chart-ref"></div>
+		<el-dialog v-model="visible" class="le-dialog" title="流程数据" width="50%">
+			<div class="test">{{ graphData }}</div>
+			<!--			<JsonPreview :data="graphData" />-->
+		</el-dialog>
+	</div>
+</template>
+<script lang="ts" setup>
+import type { Ref, PropType } from 'vue'
+import type { Definition } from '@logicflow/core'
+import { ref, onMounted, unref, nextTick, computed, watch } from 'vue'
+import FlowChartToolbar from './FlowChartToolbar.vue'
+import LogicFlow from '@logicflow/core'
+import { Snapshot, BpmnElement, Menu, DndPanel, SelectionSelect } from '@logicflow/extension'
+import { createFlowChartContext } from './useFlowContext'
+import { toLogicFlowData } from './adpterForTurbo'
+// import { JsonPreview } from '@/components/CodeEditor'
+import { configDefaultDndPanel } from './config'
+import '@logicflow/core/dist/style/index.css'
+import '@logicflow/extension/lib/style/index.css'
+
+defineOptions({ name: 'LeFlowChart' })
+
+const props = defineProps({
+	flowOptions: {
+		type: Object as PropType<Definition>,
+		default: () => ({})
+	},
+
+	data: {
+		type: Object as PropType<any>,
+		default: () => ({})
+	},
+
+	toolbar: {
+		type: Boolean,
+		default: true
+	},
+	patternItems: {
+		type: Array
+	}
+})
+
+const lfElRef = ref(null)
+const graphData = ref({})
+
+const lfInstance = ref(null) as Ref<LogicFlow | null>
+const visible = ref(false)
+// const appStore = useAppStore()
+createFlowChartContext({
+	logicFlow: lfInstance as unknown as LogicFlow
+})
+
+const getFlowOptions = computed(() => {
+	const { flowOptions } = props
+
+	const defaultOptions: Partial<Definition> = {
+		grid: true,
+		background: {
+			// color: appStore.getDarkMode === 'light' ? '#f7f9ff' : '#151515'
+			color: '#f7f9ff'
+		},
+		keyboard: {
+			enabled: true
+		},
+		...flowOptions
+	}
+	return defaultOptions as Definition
+})
+
+watch(
+	() => props.data,
+	() => {
+		onRender()
+	}
+)
+
+// TODO
+// watch(
+//   () => appStore.getDarkMode,
+//   () => {
+//     init();
+//   }
+// );
+
+watch(
+	() => unref(getFlowOptions),
+	options => {
+		unref(lfInstance)?.updateEditConfig(options)
+	}
+)
+
+// init logicFlow
+async function init() {
+	await nextTick()
+
+	const lfEl = unref(lfElRef)
+	if (!lfEl) {
+		return
+	}
+	LogicFlow.use(DndPanel)
+
+	// Canvas configuration
+	LogicFlow.use(Snapshot)
+	// Use the bpmn plug-in to introduce bpmn elements, which can be used after conversion in turbo
+	LogicFlow.use(BpmnElement)
+	// Start the right-click menu
+	LogicFlow.use(Menu)
+	LogicFlow.use(SelectionSelect)
+
+	lfInstance.value = new LogicFlow({
+		...unref(getFlowOptions),
+		container: lfEl
+	})
+	const lf = unref(lfInstance)!
+	lf?.setDefaultEdgeType('line')
+	onRender()
+	lf?.setPatternItems(props.patternItems || configDefaultDndPanel(lf))
+}
+
+async function onRender() {
+	await nextTick()
+	const lf = unref(lfInstance)
+	if (!lf) {
+		return
+	}
+	const lFData = toLogicFlowData(props.data)
+	lf.render(lFData)
+}
+
+function handlePreview() {
+	const lf = unref(lfInstance)
+	if (!lf) {
+		return
+	}
+	const data = unref(lf).getGraphData()
+	graphData.value = data
+	console.error(data, 'data')
+	visible.value = true
+}
+
+onMounted(init)
+</script>
+<style lang="scss">
+.#{$prefix}flow-chart {
+	height: 100%;
+	.chart-ref {
+		height: 100%;
+	}
+	&_toolbar {
+		&__icon {
+		}
+	}
+}
+</style>

+ 24 - 0
src/components/FlowChart/types.ts

@@ -0,0 +1,24 @@
+import { NodeConfig } from '@logicflow/core'
+export enum ToolbarTypeEnum {
+	ZOOM_IN = 'zoomIn',
+	ZOOM_OUT = 'zoomOut',
+	RESET_ZOOM = 'resetZoom',
+
+	UNDO = 'undo',
+	REDO = 'redo',
+
+	SNAPSHOT = 'snapshot',
+	VIEW_DATA = 'viewData'
+}
+
+export interface NodeItem extends NodeConfig {
+	icon: string
+}
+
+export interface ToolbarConfig {
+	type?: string | ToolbarTypeEnum
+	tooltip?: string | boolean
+	icon?: string
+	disabled?: boolean
+	separate?: boolean
+}

+ 17 - 0
src/components/FlowChart/useFlowContext.ts

@@ -0,0 +1,17 @@
+import type LogicFlow from '@logicflow/core'
+
+import { provide, inject } from 'vue'
+
+const key = Symbol('flow-chart')
+
+type Instance = {
+	logicFlow: LogicFlow
+}
+
+export function createFlowChartContext(instance: Instance) {
+	provide(key, instance)
+}
+
+export function useFlowChartContext(): Instance {
+	return inject(key) as Instance
+}

+ 1 - 1
src/components/Icon.vue

@@ -4,7 +4,7 @@
 	</svg>
 </template>
 
-<script lang="ts" setup>
+<script lang="ts" setup name="LeIcon">
 defineOptions({ name: 'LeIcon' })
 const props = defineProps({
 	iconClass: {

+ 240 - 0
src/views/approve/pendingApproval/dataTurbo.json

@@ -0,0 +1,240 @@
+{
+  "flowElementList": [
+    {
+      "incoming": [],
+      "outgoing": ["Flow_33inf2k"],
+      "dockers": [],
+      "type": 2,
+      "properties": {
+        "a": "efrwe",
+        "b": "wewe",
+        "name": "开始",
+        "x": 280,
+        "y": 200,
+        "text": {
+          "x": 280,
+          "y": 200,
+          "value": "开始"
+        },
+        "logicFlowType": "bpmn:startEvent"
+      },
+      "key": "Event_1d42u4p"
+    },
+    {
+      "incoming": ["Flow_379e0o9"],
+      "outgoing": [],
+      "dockers": [],
+      "type": 3,
+      "properties": {
+        "a": "efrwe",
+        "b": "wewe",
+        "name": "结束",
+        "x": 920,
+        "y": 200,
+        "text": {
+          "x": 920,
+          "y": 200,
+          "value": "结束"
+        },
+        "logicFlowType": "bpmn:endEvent"
+      },
+      "key": "Event_08p8i6q"
+    },
+    {
+      "incoming": ["Flow_0pfouf0"],
+      "outgoing": ["Flow_3918lhh"],
+      "dockers": [],
+      "type": 6,
+      "properties": {
+        "a": "efrwe",
+        "b": "wewe",
+        "name": "网关",
+        "x": 580,
+        "y": 200,
+        "text": {
+          "x": 580,
+          "y": 200,
+          "value": "网关"
+        },
+        "logicFlowType": "bpmn:exclusiveGateway"
+      },
+      "key": "Gateway_1fngqgj"
+    },
+    {
+      "incoming": ["Flow_33inf2k"],
+      "outgoing": ["Flow_0pfouf0"],
+      "dockers": [],
+      "type": 4,
+      "properties": {
+        "a": "efrwe",
+        "b": "wewe",
+        "name": "用户",
+        "x": 420,
+        "y": 200,
+        "text": {
+          "x": 420,
+          "y": 200,
+          "value": "用户"
+        },
+        "logicFlowType": "bpmn:userTask"
+      },
+      "key": "Activity_2mgtaia"
+    },
+    {
+      "incoming": ["Flow_3918lhh"],
+      "outgoing": ["Flow_379e0o9"],
+      "dockers": [],
+      "type": 5,
+      "properties": {
+        "a": "efrwe",
+        "b": "wewe",
+        "name": "服务",
+        "x": 760,
+        "y": 200,
+        "text": {
+          "x": 760,
+          "y": 200,
+          "value": "服务"
+        },
+        "logicFlowType": "bpmn:serviceTask"
+      },
+      "key": "Activity_1sp8qc8"
+    },
+    {
+      "incoming": ["Event_1d42u4p"],
+      "outgoing": ["Activity_2mgtaia"],
+      "type": 1,
+      "dockers": [],
+      "properties": {
+        "name": "边",
+        "text": {
+          "x": 331,
+          "y": 200,
+          "value": "边"
+        },
+        "startPoint": {
+          "x": 298,
+          "y": 200
+        },
+        "endPoint": {
+          "x": 370,
+          "y": 200
+        },
+        "pointsList": [
+          {
+            "x": 298,
+            "y": 200
+          },
+          {
+            "x": 370,
+            "y": 200
+          }
+        ],
+        "logicFlowType": "bpmn:sequenceFlow"
+      },
+      "key": "Flow_33inf2k"
+    },
+    {
+      "incoming": ["Activity_2mgtaia"],
+      "outgoing": ["Gateway_1fngqgj"],
+      "type": 1,
+      "dockers": [],
+      "properties": {
+        "name": "边2",
+        "text": {
+          "x": 507,
+          "y": 200,
+          "value": "边2"
+        },
+        "startPoint": {
+          "x": 470,
+          "y": 200
+        },
+        "endPoint": {
+          "x": 555,
+          "y": 200
+        },
+        "pointsList": [
+          {
+            "x": 470,
+            "y": 200
+          },
+          {
+            "x": 555,
+            "y": 200
+          }
+        ],
+        "logicFlowType": "bpmn:sequenceFlow"
+      },
+      "key": "Flow_0pfouf0"
+    },
+    {
+      "incoming": ["Gateway_1fngqgj"],
+      "outgoing": ["Activity_1sp8qc8"],
+      "type": 1,
+      "dockers": [],
+      "properties": {
+        "name": "边3",
+        "text": {
+          "x": 664,
+          "y": 200,
+          "value": "边3"
+        },
+        "startPoint": {
+          "x": 605,
+          "y": 200
+        },
+        "endPoint": {
+          "x": 710,
+          "y": 200
+        },
+        "pointsList": [
+          {
+            "x": 605,
+            "y": 200
+          },
+          {
+            "x": 710,
+            "y": 200
+          }
+        ],
+        "logicFlowType": "bpmn:sequenceFlow"
+      },
+      "key": "Flow_3918lhh"
+    },
+    {
+      "incoming": ["Activity_1sp8qc8"],
+      "outgoing": ["Event_08p8i6q"],
+      "type": 1,
+      "dockers": [],
+      "properties": {
+        "name": "边4",
+        "text": {
+          "x": 871,
+          "y": 200,
+          "value": "边4"
+        },
+        "startPoint": {
+          "x": 810,
+          "y": 200
+        },
+        "endPoint": {
+          "x": 902,
+          "y": 200
+        },
+        "pointsList": [
+          {
+            "x": 810,
+            "y": 200
+          },
+          {
+            "x": 902,
+            "y": 200
+          }
+        ],
+        "logicFlowType": "bpmn:sequenceFlow"
+      },
+      "key": "Flow_379e0o9"
+    }
+  ]
+}

+ 7 - 3
src/views/approve/pendingApproval/index_flowTest.vue → src/views/approve/pendingApproval/index_.vue

@@ -1,13 +1,17 @@
+<!--_flowTest-->
 <template>
-	<div class="test">
-		...
-		<div id="draw-container"></div>
+	<div class="column-page-wrap">
+		<!--		...-->
+		<!--		<div id="draw-container"></div>-->
+		<LeFlowChart :data="demoData" />
 	</div>
 </template>
 
 <script setup lang="tsx">
 import ApprovalIndex from '../components/approvalIndex.vue'
+import LeFlowChart from '@/components/FlowChart'
 import { ref, onMounted } from 'vue'
+import demoData from './dataTurbo.json'
 /**
  * pendingApproval 待审批
  * myApplication 我的申请