加载中...

LogicFlow工作流在React和Vue3中的使用

LogicFlow工作流在React和Vue3中的使用

LogicFlow 是一款流程图编辑框架,提供了一系列流程图交互、编辑所必需的功能和简单灵活的节点自定义、插件等拓展机制,方便我们快速在业务系统内满足类流程图的需求。

核心能力

  • 可视化模型:通过 LogicFlow 提供的直观可视化界面,用户可以轻松创建、编辑和管理复杂的逻辑流程图。
  • 高可定制性:用户可以根据自己的需要定制节点、连接器和样式,创建符合特定用例的定制逻辑流程图。
  • 灵活易拓展: 内置提供丰富的插件,用户也可根据自身需求定制复杂插件实现业务需求。
  • 自执行引擎: 执行引擎支持浏览器端执行流程图逻辑,为无代码执行提供新思路。
  • 数据可转换:支持 LogicFlow 数据与 BPMN、Turbo 等各种后端执行引擎数据结构转换能力。

添加 logic-flow 基础代码

$ npm install @logicflow/core --save

在 App.vue 文件中, 添加 logic-flow 核心代码;

  1. <script setup lang="ts">
  2. import { onMounted, ref } from "vue";
  3. import LogicFlow from "@logicflow/core";
  4. import "@logicflow/core/es/index.css";
  5. const container = ref();
  6. onMounted(() => {
  7. const lf = new LogicFlow({
  8. container: container.value,
  9. grid: true,
  10. });
  11. lf.render();
  12. });
  13. </script>
  14. <template>
  15. <div class="container" ref="container"></div>
  16. </template>
  17. <style scoped>
  18. .container {
  19. width: 100%;
  20. height: 100%;
  21. }
  22. </style>

使用内置拖拽面板

安装 @logicflow/extension 扩展依赖, 先看一下内置拖拽面板如何使用;

npm install @logicflow/extension --save

再次修改 App.vue 文件内容, 导入 DndPanel 对象及扩展所需要的样式模块;

  1. import { DndPanel } from "@logicflow/extension";
  2. import "@logicflow/extension/lib/style/index.css";

 在实例化 LogicFlow 对象时, 通过选项 plugins 配置 DndPanel 对象;

  1. const lf = new LogicFlow({
  2. ...
  3. plugins: [DndPanel],
  4. });

在实例化 LogicFlow 对象后, 通过实例对象 lf.extension.dndPanel 中的 setPatternItems 方法设置拖拽面板的内容:

  1. // icons 是一组图标对象(Base64字符串)
  2. import { icons } from "./icons";
  3. lf.extension.dndPanel.setPatternItems([
  4. {
  5. label: "选区",
  6. icon: icons.select,
  7. },
  8. {
  9. type: "circle",
  10. text: "开始",
  11. label: "开始节点",
  12. icon: icons.start,
  13. },
  14. {
  15. type: "rect",
  16. label: "用户任务",
  17. icon: icons.task,
  18. },
  19. {
  20. type: "rect",
  21. label: "系统任务",
  22. icon: icons.task,
  23. },
  24. {
  25. type: "diamond",
  26. label: "条件判断",
  27. icon: icons.condition,
  28. },
  29. {
  30. type: "circle",
  31. text: "结束",
  32. label: "结束节点",
  33. icon: icons.end,
  34. },
  35. ]);

 重新预览效果, 可以看到内置拖拽面板已经生效;

New Image

React中的使用示例(使用CRA搭建开发环境)

package.json

  1. {
  2. "name": "logicflow-react",
  3. "version": "0.1.0",
  4. "private": true,
  5. "dependencies": {
  6. "@logicflow/core": "^2.0.0",
  7. "@logicflow/extension": "^2.0.0",
  8. "@testing-library/jest-dom": "^5.17.0",
  9. "@testing-library/react": "^13.4.0",
  10. "@testing-library/user-event": "^13.5.0",
  11. "antd": "^5.20.1",
  12. "insert-css": "^2.0.0",
  13. "react": "^18.3.1",
  14. "react-dom": "^18.3.1",
  15. "react-scripts": "5.0.1",
  16. "web-vitals": "^2.1.4"
  17. },
  18. "scripts": {
  19. "start": "react-scripts start",
  20. "build": "react-scripts build",
  21. "test": "react-scripts test",
  22. "eject": "react-scripts eject"
  23. },
  24. "eslintConfig": {
  25. "extends": [
  26. "react-app",
  27. "react-app/jest"
  28. ]
  29. },
  30. "browserslist": {
  31. "production": [
  32. ">0.2%",
  33. "not dead",
  34. "not op_mini all"
  35. ],
  36. "development": [
  37. "last 1 chrome version",
  38. "last 1 firefox version",
  39. "last 1 safari version"
  40. ]
  41. }
  42. }

public/index.html

  1. <!DOCTYPE html>
  2. <html lang="en">
  3. <head>
  4. <meta charset="utf-8" />
  5. <link rel="icon" href="%PUBLIC_URL%/favicon.ico" />
  6. <meta name="viewport" content="width=device-width, initial-scale=1" />
  7. <meta name="theme-color" content="#000000" />
  8. <meta
  9. name="description"
  10. content="Web site created using create-react-app"
  11. />
  12. <link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" />
  13. <link rel="manifest" href="%PUBLIC_URL%/manifest.json" />
  14. <title>React App</title>
  15. <style>
  16. html, body, #root{
  17. margin: 0;
  18. padding: 0;
  19. height: 100%;
  20. }
  21. </style>
  22. </head>
  23. <body>
  24. <noscript>You need to enable JavaScript to run this app.</noscript>
  25. <div id="root"></div>
  26. </body>
  27. </html>

src/index.js

  1. import ReactDOM from ''react-dom/client'';
  2. import React, { useEffect, useRef } from "react";
  3. import { Button } from "antd";
  4. import LogicFlow from "@logicflow/core";
  5. import {
  6. BpmnElement,
  7. BpmnXmlAdapter,
  8. Control,
  9. Menu,
  10. SelectionSelect,
  11. DndPanel,
  12. } from "@logicflow/extension";
  13. import "@logicflow/core/es/index.css";
  14. import "@logicflow/extension/lib/style/index.css";
  15. import insertCss from ''insert-css'';
  16. const root = ReactDOM.createRoot(document.getElementById(''root''));
  17. const App = () => {
  18. const container = useRef(null);
  19. const lfRef = useRef(null);
  20. useEffect(() => {
  21. lfRef.current = new LogicFlow({
  22. container: container.current,
  23. stopZoomGraph: true,
  24. metaKeyMultipleSelected: true,
  25. grid: true,
  26. keyboard: {
  27. enabled: true,
  28. },
  29. snapline: true,
  30. plugins: [
  31. BpmnElement,
  32. BpmnXmlAdapter,
  33. Control,
  34. Menu,
  35. SelectionSelect,
  36. DndPanel,
  37. ],
  38. });
  39. lfRef.current.setPatternItems([
  40. {
  41. label: "选区",
  42. icon: "",
  43. callback: () => {
  44. lfRef.current.openSelectionSelect();
  45. lfRef.current.once("selection:selected", () => {
  46. lfRef.current.closeSelectionSelect();
  47. });
  48. },
  49. },
  50. {
  51. type: "bpmn:startEvent",
  52. label: "开始",
  53. text: "开始",
  54. icon: "",
  55. },
  56. {
  57. type: "bpmn:userTask",
  58. label: "用户任务",
  59. text: "用户任务",
  60. icon: "",
  61. },
  62. {
  63. type: "bpmn:serviceTask",
  64. label: "系统任务",
  65. text: "系统任务",
  66. icon: "",
  67. },
  68. {
  69. type: "bpmn:exclusiveGateway",
  70. label: "条件判断",
  71. text: "条件判断",
  72. icon: "",
  73. },
  74. {
  75. type: "bpmn:endEvent",
  76. label: "结束",
  77. text: "结束",
  78. icon: "",
  79. },
  80. ]);
  81. lfRef.current.render({});
  82. }, []);
  83. const download = (filename, text) => {
  84. const element = document.createElement("a");
  85. element.setAttribute(
  86. "href",
  87. "data:text/plain;charset=utf-8," + encodeURIComponent(text)
  88. );
  89. element.setAttribute("download", filename);
  90. element.style.display = "none";
  91. document.body.appendChild(element);
  92. element.click();
  93. document.body.removeChild(element);
  94. };
  95. const downloadXml = () => {
  96. const data = lfRef.current.getGraphData();
  97. download("logic-flow.xml", data);
  98. };
  99. const uploadXml = (ev) => {
  100. const file = ev.target.files[0];
  101. const reader = new FileReader();
  102. reader.onload = (event) => {
  103. if (event.target) {
  104. const xml = event.target.result;
  105. lfRef.current.render(xml);
  106. }
  107. };
  108. reader.readAsText(file);
  109. };
  110. return (
  111. <>
  112. <div className="explain">
  113. 点击左下角下载 XML,将文件上传到
  114. <Button type="link" href="https://demo.bpmn.io/" target="_blank">
  115. BPMN Demo
  116. </Button>
  117. 即可使用
  118. </div>
  119. <div id="graph" ref={container}></div>
  120. <div className="graph-io">
  121. <span title="下载 XML" onMouseDown={() => downloadXml()}>
  122. <img
  123. src=""
  124. alt="下载XML"
  125. />
  126. </span>
  127. <span id="upload-xml" title="上传 XML">
  128. <input
  129. type="file"
  130. className="upload"
  131. onChange={(ev) => uploadXml(ev)}
  132. />
  133. <img
  134. className="upload-img"
  135. src=""
  136. alt="上传XML"
  137. />
  138. </span>
  139. </div>
  140. </>
  141. );
  142. };
  143. root.render(<App></App>);
  144. insertCss(`
  145. .explain {
  146. height: 30px;
  147. }
  148. #graph {
  149. width: 100%;
  150. height: calc(100% - 30px);
  151. }
  152. .graph-io {
  153. position: absolute;
  154. left: 10px;
  155. bottom: 10px;
  156. z-index: 9999;
  157. background:rgba(255,255,255,0.8);
  158. box-shadow: 0 1px 4px rgba(0,0,0,.3);
  159. padding: 10px;
  160. display: flex;
  161. }
  162. .graph-io > span {
  163. margin: 0 5px;
  164. cursor: pointer;
  165. }
  166. #upload-xml {
  167. position: relative;
  168. overflow: hidden;
  169. display: inline-block;
  170. cursor: pointer;
  171. }
  172. .upload {
  173. position: absolute;
  174. z-index: 99;
  175. left: 0;
  176. top: 0;
  177. opacity: 0;
  178. cursor: pointer;
  179. }
  180. .upload::-webkit-file-upload-button {
  181. cursor: pointer;
  182. }
  183. .upload-img {
  184. width: 26px;
  185. }
  186. `);

在Vue3中使用(使用Vite搭建开发环境)

package.json:

  1. {
  2. "name": "logicflow",
  3. "version": "0.0.0",
  4. "private": true,
  5. "type": "module",
  6. "scripts": {
  7. "dev": "vite",
  8. "build": "vite build",
  9. "preview": "vite preview"
  10. },
  11. "dependencies": {
  12. "@logicflow/core": "^2.0.0",
  13. "@logicflow/extension": "^2.0.0",
  14. "vue": "^3.4.29"
  15. },
  16. "devDependencies": {
  17. "@vitejs/plugin-vue": "^5.0.5",
  18. "unplugin-auto-import": "^0.18.2",
  19. "unplugin-vue-components": "^0.27.3",
  20. "vite": "^5.3.1"
  21. }
  22. }

src/App.vue:

  1. <template>
  2. <div class="root" ref="rootRef"></div>
  3. <div class="graph"></div>
  4. <div class="graph-io">
  5. <span title="下载 XML" @mousedown="downloadXml">
  6. <img src=""
  7. alt="下载XML" />
  8. </span>
  9. <span id="upload-xml" title="上传 XML">
  10. <input type="file" class="upload" @change="uploadXml" />
  11. <img class="upload-img"
  12. src=""
  13. alt="上传XML" />
  14. </span>
  15. </div>
  16. </template>
  17. <script setup>
  18. import { ref, onMounted } from "vue";
  19. import LogicFlow from "@logicflow/core";
  20. import {
  21. BpmnElement,
  22. BpmnXmlAdapter,
  23. Control,
  24. Menu,
  25. SelectionSelect,
  26. DndPanel,
  27. } from "@logicflow/extension";
  28. import "@logicflow/core/es/index.css";
  29. import "@logicflow/extension/lib/style/index.css";
  30. const rootRef = ref(null)
  31. const lfRef = ref(null);
  32. const download = (filename, text) => {
  33. const element = document.createElement("a");
  34. element.setAttribute(
  35. "href",
  36. "data:text/plain;charset=utf-8," + encodeURIComponent(text)
  37. );
  38. element.setAttribute("download", filename);
  39. element.style.display = "none";
  40. document.body.appendChild(element);
  41. element.click();
  42. document.body.removeChild(element);
  43. };
  44. const downloadXml = () => {
  45. const data = lfRef.value.getGraphData();
  46. download("logic-flow.xml", data);
  47. };
  48. const uploadXml = (ev) => {
  49. const file = ev.target.files[0];
  50. const reader = new FileReader();
  51. reader.onload = (event) => {
  52. if (event.target) {
  53. const xml = event.target.result;
  54. lfRef.value.render(xml);
  55. }
  56. };
  57. reader.readAsText(file);
  58. };
  59. onMounted(() => {
  60. lfRef.value = new LogicFlow({
  61. container: rootRef.value,
  62. stopZoomGraph: true,
  63. metaKeyMultipleSelected: true,
  64. grid: true,
  65. keyboard: {
  66. enabled: true,
  67. },
  68. snapline: true,
  69. plugins: [
  70. BpmnElement,
  71. BpmnXmlAdapter,
  72. Control,
  73. Menu,
  74. SelectionSelect,
  75. DndPanel,
  76. ],
  77. });
  78. lfRef.value.setPatternItems([
  79. {
  80. label: "选区",
  81. icon: "",
  82. callback: () => {
  83. lfRef.current.openSelectionSelect();
  84. lfRef.current.once("selection:selected", () => {
  85. lfRef.current.closeSelectionSelect();
  86. });
  87. },
  88. },
  89. {
  90. type: "bpmn:startEvent",
  91. label: "开始",
  92. text: "开始",
  93. icon: "",
  94. },
  95. {
  96. type: "bpmn:userTask",
  97. label: "用户任务",
  98. text: "用户任务",
  99. icon: "",
  100. },
  101. {
  102. type: "bpmn:serviceTask",
  103. label: "系统任务",
  104. text: "系统任务",
  105. icon: "",
  106. },
  107. {
  108. type: "bpmn:exclusiveGateway",
  109. label: "条件判断",
  110. text: "条件判断",
  111. icon: "",
  112. },
  113. {
  114. type: "bpmn:endEvent",
  115. label: "结束",
  116. text: "结束",
  117. icon: "",
  118. },
  119. ]);
  120. lfRef.value.render({});
  121. });
  122. </script>
  123. <style scoped>
  124. .root {
  125. width: 100%;
  126. height: 100%;
  127. }
  128. .explain {
  129. height: 30px;
  130. }
  131. .graph {
  132. width: 100%;
  133. height: calc(100% - 30px);
  134. }
  135. .graph-io {
  136. position: absolute;
  137. left: 10px;
  138. bottom: 10px;
  139. z-index: 9999;
  140. background: rgba(255, 255, 255, 0.8);
  141. box-shadow: 0 1px 4px rgba(0, 0, 0, 0.3);
  142. padding: 10px;
  143. display: flex;
  144. }
  145. .graph-io>span {
  146. margin: 0 5px;
  147. cursor: pointer;
  148. }
  149. #upload-xml {
  150. position: relative;
  151. overflow: hidden;
  152. display: inline-block;
  153. cursor: pointer;
  154. }
  155. .upload {
  156. position: absolute;
  157. z-index: 99;
  158. left: 0;
  159. top: 0;
  160. opacity: 0;
  161. cursor: pointer;
  162. }
  163. .upload::-webkit-file-upload-button {
  164. cursor: pointer;
  165. }
  166. .upload-img {
  167. width: 26px;
  168. }
  169. </style>

效果:

New Image

官网:https://site.logic-flow.cn/

www.hhzai.top