LogicFlow工作流在React和Vue3中的使用
LogicFlow 是一款流程图编辑框架,提供了一系列流程图交互、编辑所必需的功能和简单灵活的节点自定义、插件等拓展机制,方便我们快速在业务系统内满足类流程图的需求。
核心能力
- 可视化模型:通过 LogicFlow 提供的直观可视化界面,用户可以轻松创建、编辑和管理复杂的逻辑流程图。
- 高可定制性:用户可以根据自己的需要定制节点、连接器和样式,创建符合特定用例的定制逻辑流程图。
- 灵活易拓展: 内置提供丰富的插件,用户也可根据自身需求定制复杂插件实现业务需求。
- 自执行引擎: 执行引擎支持浏览器端执行流程图逻辑,为无代码执行提供新思路。
- 数据可转换:支持 LogicFlow 数据与 BPMN、Turbo 等各种后端执行引擎数据结构转换能力。
添加 logic-flow 基础代码
$ npm install @logicflow/core --save
在 App.vue 文件中, 添加 logic-flow 核心代码;
<script setup lang="ts">
import { onMounted, ref } from "vue";
import LogicFlow from "@logicflow/core";
import "@logicflow/core/es/index.css";const container = ref();onMounted(() => {const lf = new LogicFlow({container: container.value,grid: true,});lf.render();
});
</script><template><div class="container" ref="container"></div>
</template><style scoped>
.container {width: 100%;height: 100%;
}
</style>
使用内置拖拽面板
安装 @logicflow/extension 扩展依赖, 先看一下内置拖拽面板如何使用;
npm install @logicflow/extension --save
再次修改 App.vue 文件内容, 导入 DndPanel 对象及扩展所需要的样式模块;
import { DndPanel } from "@logicflow/extension";
import "@logicflow/extension/lib/style/index.css";
在实例化 LogicFlow 对象时, 通过选项 plugins 配置 DndPanel 对象;
const lf = new LogicFlow({...plugins: [DndPanel],
});
在实例化 LogicFlow 对象后, 通过实例对象 lf.extension.dndPanel 中的 setPatternItems 方法设置拖拽面板的内容:
// icons 是一组图标对象(Base64字符串)
import { icons } from "./icons";lf.extension.dndPanel.setPatternItems([{label: "选区",icon: icons.select,},{type: "circle",text: "开始",label: "开始节点",icon: icons.start,},{type: "rect",label: "用户任务",icon: icons.task,},{type: "rect",label: "系统任务",icon: icons.task,},{type: "diamond",label: "条件判断",icon: icons.condition,},{type: "circle",text: "结束",label: "结束节点",icon: icons.end,},
]);
重新预览效果, 可以看到内置拖拽面板已经生效;

在React中的使用示例(使用CRA搭建开发环境)
package.json
{"name": "logicflow-react","version": "0.1.0","private": true,"dependencies": {"@logicflow/core": "^2.0.0","@logicflow/extension": "^2.0.0","@testing-library/jest-dom": "^5.17.0","@testing-library/react": "^13.4.0","@testing-library/user-event": "^13.5.0","antd": "^5.20.1","insert-css": "^2.0.0","react": "^18.3.1","react-dom": "^18.3.1","react-scripts": "5.0.1","web-vitals": "^2.1.4"},"scripts": {"start": "react-scripts start","build": "react-scripts build","test": "react-scripts test","eject": "react-scripts eject"},"eslintConfig": {"extends": ["react-app","react-app/jest"]},"browserslist": {"production": [">0.2%","not dead","not op_mini all"],"development": ["last 1 chrome version","last 1 firefox version","last 1 safari version"]}
}
public/index.html
<!DOCTYPE html>
<html lang="en"><head><meta charset="utf-8" /><link rel="icon" href="%PUBLIC_URL%/favicon.ico" /><meta name="viewport" content="width=device-width, initial-scale=1" /><meta name="theme-color" content="#000000" /><metaname="description"content="Web site created using create-react-app"/><link rel="apple-touch-icon" href="%PUBLIC_URL%/logo192.png" /><link rel="manifest" href="%PUBLIC_URL%/manifest.json" /><title>React App</title><style>html, body, #root{margin: 0;padding: 0;height: 100%;}</style></head><body><noscript>You need to enable JavaScript to run this app.</noscript><div id="root"></div></body>
</html>
src/index.js
import ReactDOM from 'react-dom/client';
import React, { useEffect, useRef } from "react";
import { Button } from "antd";
import LogicFlow from "@logicflow/core";
import {BpmnElement,BpmnXmlAdapter,Control,Menu,SelectionSelect,DndPanel,
} from "@logicflow/extension";
import "@logicflow/core/es/index.css";
import "@logicflow/extension/lib/style/index.css";
import insertCss from 'insert-css';const root = ReactDOM.createRoot(document.getElementById('root'));const App = () => {const container = useRef(null);const lfRef = useRef(null);useEffect(() => {lfRef.current = new LogicFlow({container: container.current,stopZoomGraph: true,metaKeyMultipleSelected: true,grid: true,keyboard: {enabled: true,},snapline: true,plugins: [BpmnElement,BpmnXmlAdapter,Control,Menu,SelectionSelect,DndPanel,],});lfRef.current.setPatternItems([{label: "选区",icon: "",callback: () => {lfRef.current.openSelectionSelect();lfRef.current.once("selection:selected", () => {lfRef.current.closeSelectionSelect();});},},{type: "bpmn:startEvent",label: "开始",text: "开始",icon: "",},{type: "bpmn:userTask",label: "用户任务",text: "用户任务",icon: "",},{type: "bpmn:serviceTask",label: "系统任务",text: "系统任务",icon: "",},{type: "bpmn:exclusiveGateway",label: "条件判断",text: "条件判断",icon: "",},{type: "bpmn:endEvent",label: "结束",text: "结束",icon: "",},]);lfRef.current.render({});}, []);const download = (filename, text) => {const element = document.createElement("a");element.setAttribute("href","data:text/plain;charset=utf-8," + encodeURIComponent(text));element.setAttribute("download", filename);element.style.display = "none";document.body.appendChild(element);element.click();document.body.removeChild(element);};const downloadXml = () => {const data = lfRef.current.getGraphData();download("logic-flow.xml", data);};const uploadXml = (ev) => {const file = ev.target.files[0];const reader = new FileReader();reader.onload = (event) => {if (event.target) {const xml = event.target.result;lfRef.current.render(xml);}};reader.readAsText(file);};return (<><div className="explain">点击左下角下载 XML,将文件上传到<Button type="link" href="https://demo.bpmn.io/" target="_blank">BPMN Demo</Button>即可使用</div><div id="graph" ref={container}></div><div className="graph-io"><span title="下载 XML" onMouseDown={() => downloadXml()}><imgsrc=""alt="下载XML"/></span><span id="upload-xml" title="上传 XML"><inputtype="file"className="upload"onChange={(ev) => uploadXml(ev)}/><imgclassName="upload-img"src=""alt="上传XML"/></span></div></>);
};root.render(<App></App>);insertCss(`.explain {height: 30px; }#graph {width: 100%;height: calc(100% - 30px);}.graph-io {position: absolute;left: 10px;bottom: 10px;z-index: 9999;background:rgba(255,255,255,0.8);box-shadow: 0 1px 4px rgba(0,0,0,.3);padding: 10px;display: flex;}.graph-io > span {margin: 0 5px;cursor: pointer;}#upload-xml {position: relative;overflow: hidden;display: inline-block;cursor: pointer;}.upload {position: absolute;z-index: 99;left: 0;top: 0;opacity: 0;cursor: pointer;}.upload::-webkit-file-upload-button {cursor: pointer;}.upload-img {width: 26px;}`);
在Vue3中使用(使用Vite搭建开发环境)
package.json:
{"name": "logicflow","version": "0.0.0","private": true,"type": "module","scripts": {"dev": "vite","build": "vite build","preview": "vite preview"},"dependencies": {"@logicflow/core": "^2.0.0","@logicflow/extension": "^2.0.0","vue": "^3.4.29"},"devDependencies": {"@vitejs/plugin-vue": "^5.0.5","unplugin-auto-import": "^0.18.2","unplugin-vue-components": "^0.27.3","vite": "^5.3.1"}
}
src/App.vue:
<template><div class="root" ref="rootRef"></div><div class="graph"></div><div class="graph-io"><span title="下载 XML" @mousedown="downloadXml"><img src=""alt="下载XML" /></span><span id="upload-xml" title="上传 XML"><input type="file" class="upload" @change="uploadXml" /><img class="upload-img"src=""alt="上传XML" /></span></div>
</template><script setup>
import { ref, onMounted } from "vue";
import LogicFlow from "@logicflow/core";
import {BpmnElement,BpmnXmlAdapter,Control,Menu,SelectionSelect,DndPanel,
} from "@logicflow/extension";
import "@logicflow/core/es/index.css";
import "@logicflow/extension/lib/style/index.css";const rootRef = ref(null)
const lfRef = ref(null);const download = (filename, text) => {const element = document.createElement("a");element.setAttribute("href","data:text/plain;charset=utf-8," + encodeURIComponent(text));element.setAttribute("download", filename);element.style.display = "none";document.body.appendChild(element);element.click();document.body.removeChild(element);
};const downloadXml = () => {const data = lfRef.value.getGraphData();download("logic-flow.xml", data);
};const uploadXml = (ev) => {const file = ev.target.files[0];const reader = new FileReader();reader.onload = (event) => {if (event.target) {const xml = event.target.result;lfRef.value.render(xml);}};reader.readAsText(file);
};onMounted(() => {lfRef.value = new LogicFlow({container: rootRef.value,stopZoomGraph: true,metaKeyMultipleSelected: true,grid: true,keyboard: {enabled: true,},snapline: true,plugins: [BpmnElement,BpmnXmlAdapter,Control,Menu,SelectionSelect,DndPanel,],});lfRef.value.setPatternItems([{label: "选区",icon: "",callback: () => {lfRef.current.openSelectionSelect();lfRef.current.once("selection:selected", () => {lfRef.current.closeSelectionSelect();});},},{type: "bpmn:startEvent",label: "开始",text: "开始",icon: "",},{type: "bpmn:userTask",label: "用户任务",text: "用户任务",icon: "",},{type: "bpmn:serviceTask",label: "系统任务",text: "系统任务",icon: "",},{type: "bpmn:exclusiveGateway",label: "条件判断",text: "条件判断",icon: "",},{type: "bpmn:endEvent",label: "结束",text: "结束",icon: "",},]);lfRef.value.render({});
});
</script><style scoped>
.root {width: 100%;height: 100%;
}.explain {height: 30px;
}.graph {width: 100%;height: calc(100% - 30px);
}.graph-io {position: absolute;left: 10px;bottom: 10px;z-index: 9999;background: rgba(255, 255, 255, 0.8);box-shadow: 0 1px 4px rgba(0, 0, 0, 0.3);padding: 10px;display: flex;
}.graph-io>span {margin: 0 5px;cursor: pointer;
}#upload-xml {position: relative;overflow: hidden;display: inline-block;cursor: pointer;
}.upload {position: absolute;z-index: 99;left: 0;top: 0;opacity: 0;cursor: pointer;
}.upload::-webkit-file-upload-button {cursor: pointer;
}.upload-img {width: 26px;
}
</style>
效果:

官网:https://site.logic-flow.cn/
相关文章:
LogicFlow工作流在React和Vue3中的使用
LogicFlow 是一款流程图编辑框架,提供了一系列流程图交互、编辑所必需的功能和简单灵活的节点自定义、插件等拓展机制,方便我们快速在业务系统内满足类流程图的需求。 核心能力 可视化模型:通过 LogicFlow 提供的直观可视化界面,…...
Python循环语句:不到长城心不死
Python中的循环语句是编程中非常重要的结构,它们允许你重复执行一段代码多次,直到满足某个条件为止。Python提供了两种主要的循环类型:for循环和while循环。 文章目录 1. for 循环2. while 循环循环控制语句range() 函数结合循环语句和 rang…...
Unity教程(九)角色攻击的改进
Unity开发2D类银河恶魔城游戏学习笔记 Unity教程(零)Unity和VS的使用相关内容 Unity教程(一)开始学习状态机 Unity教程(二)角色移动的实现 Unity教程(三)角色跳跃的实现 Unity教程&…...
宠物空气净化器真的能除毛吗?有哪些选购技巧和品牌推荐修改版
夏日炎炎,有猫超甜。作为一名资深铲屎官,家里养有猫让我倍感幸福,夏天里有空调、有西瓜、有猫,这几个搭配在一起真的是超级爽。但在这么高温的夏天,家里养有宠物还是有不少烦恼的。比如家里的浮毛一直飘,似…...
Qt自定义注释
前言 是谁在Qt中编写代码,函数注释,类注释时,注释符号一个一个的敲? comment注释brief简洁的 Detailed详细的 第一步: 打开Qt 工具->选项->文本编辑器->片段 第二步: 点击添加 然后点击OK…...
【模电笔记】——信号的运算和处理电路(含电压比较器)
tips:本章节的笔记已经打包到word文档里啦,建议大家下载文章顶部资源(有时看不到是在审核中,等等就能下载了。手机端下载后里面的插图可能会乱,建议电脑下载,兼容性更好且易于观看),…...
Java之 equals()与==
目录 运算符用途:用于比较两个引用是否指向同一个对象。比较内容:比较的是内存地址(引用)。适用范围:适用于基本数据类型和对象引用 equals() 方法用途:用于比较两个对象的内容是否相同。比较内容…...
Ubuntu20.04 运行深蓝路径规划hw1
前言 环境: ubuntu 20.04 ; ROS版本: noetic; 问题 1、出现PCL报错:#error PCL requires C14 or above catkin_make 编译时,出现如下错误 解决: 在grid_path_searcher文件夹下面的CMakeLis…...
企业如何组建安全稳定的跨国通信网络
当企业在海外设有分公司时,如何建立一个安全且稳定的跨国通信网络是一个关键问题。为了确保跨国通信的安全和稳定性,可以考虑以下几种方案。 首先,可以在分公司之间搭建虚拟专用网络。虚拟专用网络通过对传输数据进行加密,保护通信…...
WordPress原创插件:Download-block-plugin下载按钮图标美化
WordPress原创插件:Download-block-plugin下载按钮图标美化 https://download.csdn.net/download/huayula/89632743...
前端【详解】缓存
HTTP 缓存 https://blog.csdn.net/weixin_41192489/article/details/136446539 CDN 缓存 CDN 全称 Content Delivery Network,即内容分发网络。 用户在浏览网站的时候,CDN会选择一个离用户最近的CDN边缘节点来响应用户的请求 CDN边缘节点的缓存机制与HTTP 缓存相同…...
P5821 【LK R-03】密码串匹配
[题目通道](【L&K R-03】密码串匹配 - 洛谷) 一道神题。 如果没有修改操作,翻转A数组或B数组后就是裸的FFT了 如果每次操作都暴力修改FFT时间复杂度显然爆炸 如果每次操作都不修改,记下修改序列,询问时加上修改序列的贡献,…...
httpx,一个网络请求的 Python 新宠儿
大家好!我是爱摸鱼的小鸿,关注我,收看每期的编程干货。 一个简单的库,也许能够开启我们的智慧之门, 一个普通的方法,也许能在危急时刻挽救我们于水深火热, 一个新颖的思维方式,也许能…...
计算机网络408考研 2014
1 计算机网络408考研2014年真题解析_哔哩哔哩_bilibili 1 111 1 11 1...
JavaScript 资源大全中文版
目录 JavaScript资源大全中文版 包管理器加载器组件管理器打包工具测试框架QA工具MVC 框架和库基于 Node 的 CMS 框架模板引擎文章和帖子数据可视化 时间轴电子表格 编辑器文档工具 文件函数式编程响应式编程数据结构日期字符串数字存储颜色国际化和本地化控制流路由安全性日志…...
如何获取能直接在浏览器打开的播放地址?
背景:需要在浏览器上直接打开设备的画面,但又不想二次开发 本文介绍一种极简的取流方式,不需要掌握前端开发知识,按照本文档拼接就能得到设备的播放地址 一、准备工作 1.将设备接入到萤石账号下。萤石设备接入指南:h…...
如何用 LangChain 实现一个Zero Shot智能决策器(附源码)
写在前面 最近一直在研究Agent和Tool的使用,今天给大家带来一篇何枝大佬(知乎何枝)的文章《如何用LangChain实现一个Zero Shot智能决策器》,并附上源码。 知乎:https://zhuanlan.zhihu.com/p/627333499LangChain是当…...
读完这本书,我终于搞懂了Transformer、BERT和GPT!【附PDF】
前言 《Transformer、BERT和GPT: 包括ChatGPT和提示工程》 是一本深入浅出地介绍自然语言处理领域前沿技术的专著,全书一共379页PDF,是截止到目前比较系统介绍NLP和GPT融合领域的书籍。 全书共十章,内容丰富,结构清晰,…...
仿RabbitMq简易消息队列基础篇(Muduo库的使用)
TOC Muduo库简介 Muduo由陈硕⼤佬开发,是⼀个基于⾮阻塞IO和事件驱动的C⾼并发TCP⽹络编程库。他是一款基于主从Reactor模型的网络库,其使用的线程模型是one loop per thread, 所谓 one loop per thread 指的是: 一个线程只能有一个事件循…...
.net SqlSugarHelper
NuGet安装: SqlSugarCore using SqlSugar; using System; using System.Collections.Generic; using System.Linq; using System.Text; using System.Threading.Tasks;namespace Namespace {public class SqlSugarHelper{public string _connectionString Custom…...
【入坑系列】TiDB 强制索引在不同库下不生效问题
文章目录 背景SQL 优化情况线上SQL运行情况分析怀疑1:执行计划绑定问题?尝试:SHOW WARNINGS 查看警告探索 TiDB 的 USE_INDEX 写法Hint 不生效问题排查解决参考背景 项目中使用 TiDB 数据库,并对 SQL 进行优化了,添加了强制索引。 UAT 环境已经生效,但 PROD 环境强制索…...
2021-03-15 iview一些问题
1.iview 在使用tree组件时,发现没有set类的方法,只有get,那么要改变tree值,只能遍历treeData,递归修改treeData的checked,发现无法更改,原因在于check模式下,子元素的勾选状态跟父节…...
P3 QT项目----记事本(3.8)
3.8 记事本项目总结 项目源码 1.main.cpp #include "widget.h" #include <QApplication> int main(int argc, char *argv[]) {QApplication a(argc, argv);Widget w;w.show();return a.exec(); } 2.widget.cpp #include "widget.h" #include &q…...
spring:实例工厂方法获取bean
spring处理使用静态工厂方法获取bean实例,也可以通过实例工厂方法获取bean实例。 实例工厂方法步骤如下: 定义实例工厂类(Java代码),定义实例工厂(xml),定义调用实例工厂ÿ…...
全面解析各类VPN技术:GRE、IPsec、L2TP、SSL与MPLS VPN对比
目录 引言 VPN技术概述 GRE VPN 3.1 GRE封装结构 3.2 GRE的应用场景 GRE over IPsec 4.1 GRE over IPsec封装结构 4.2 为什么使用GRE over IPsec? IPsec VPN 5.1 IPsec传输模式(Transport Mode) 5.2 IPsec隧道模式(Tunne…...
什么是Ansible Jinja2
理解 Ansible Jinja2 模板 Ansible 是一款功能强大的开源自动化工具,可让您无缝地管理和配置系统。Ansible 的一大亮点是它使用 Jinja2 模板,允许您根据变量数据动态生成文件、配置设置和脚本。本文将向您介绍 Ansible 中的 Jinja2 模板,并通…...
在Ubuntu24上采用Wine打开SourceInsight
1. 安装wine sudo apt install wine 2. 安装32位库支持,SourceInsight是32位程序 sudo dpkg --add-architecture i386 sudo apt update sudo apt install wine32:i386 3. 验证安装 wine --version 4. 安装必要的字体和库(解决显示问题) sudo apt install fonts-wqy…...
CVE-2020-17519源码分析与漏洞复现(Flink 任意文件读取)
漏洞概览 漏洞名称:Apache Flink REST API 任意文件读取漏洞CVE编号:CVE-2020-17519CVSS评分:7.5影响版本:Apache Flink 1.11.0、1.11.1、1.11.2修复版本:≥ 1.11.3 或 ≥ 1.12.0漏洞类型:路径遍历&#x…...
20个超级好用的 CSS 动画库
分享 20 个最佳 CSS 动画库。 它们中的大多数将生成纯 CSS 代码,而不需要任何外部库。 1.Animate.css 一个开箱即用型的跨浏览器动画库,可供你在项目中使用。 2.Magic Animations CSS3 一组简单的动画,可以包含在你的网页或应用项目中。 3.An…...
作为测试我们应该关注redis哪些方面
1、功能测试 数据结构操作:验证字符串、列表、哈希、集合和有序的基本操作是否正确 持久化:测试aof和aof持久化机制,确保数据在开启后正确恢复。 事务:检查事务的原子性和回滚机制。 发布订阅:确保消息正确传递。 2、性…...
