Vue+LogicFlow+Flowable 前端+后端实现工作流
一、实现效果
前端使用LogicFlow框架绘制流程图,可以导出为xml工作流标准格式数据,通过xml文件传递到后端进行Flowable流程注册,并保存到数据库中。
二、BPM传输文件格式(.xml)
如需添加承办人的话,需要在LogicFlow导出文件的基础上手动添加xmlns:flowable="http://flowable.org/bpmn"flowable插件,不然后台无法识别flowable:candidateUsers。
Flow_a3e7d0c Flow_a3e7d0c Flow_3f9a386 Flow_3f9a386
三、前端框架(LogicFlow)
- LogicFlow.vue
保存 清空 加载 删除 获取定义流程 工作流实例import LogicFlow from '@logicflow/core' import '@logicflow/core/dist/style/index.css' import { BpmnElement, BpmnXmlAdapter, Menu } from '@logicflow/extension' import '@logicflow/extension/lib/style/index.css' import BpmnNodePanel from "./BpmnNodePanel.vue" import { addFlow, infoFlow, deleteFlow, getDeployList, flowRun } from "../api/server" import { addFlowable, addProp, getTypeNameByTag, setValueByTag } from '../utils/xml' export default { data() { return { lf: void 0, xmlData: void 0, definitionsId: void 0, processId: void 0, } }, components: { BpmnNodePanel }, created() {}, mounted() { this.loadFlow() }, methods: { loadFlow() { LogicFlow.use(BpmnElement) LogicFlow.use(BpmnXmlAdapter) LogicFlow.use(Menu) //初始化 this.lf = new LogicFlow({ container: this.$refs.container, stopScrollGraph: true, stopZoomGraph: true, grid: false, keyboard: { enabled: true, }, }); // this.lf.setDefaultEdgeType('bezier') this.lf.render(); }, // 保存 saveNode() { // 获取xml数据 this.xmlData = this.lf.getGraphData(); // 添加flowable扩展 this.xmlData = addFlowable(this.xmlData) // 判断是否为文档编辑 if (this.processId) { // 修改为原流程id this.xmlData = setValueByTag(this.xmlData, 'bpmn:definitions', 'id', this.definitionsId) this.xmlData = setValueByTag(this.xmlData, 'bpmn:process', 'id', this.processId) } else { // 添加节点用户 个人assignee 候选人candidateUsers 候选组候选人candidateGroups 动态设置${employee} // this.xmlData = addProp(this.xmlData, 'bpmn:startEvent', 'flowable:candidateUsers', 'zhao,qian,sun') this.xmlData = addProp(this.xmlData, 'bpmn:userTask', 'flowable:candidateUsers', 'li,zhou,wang') } const data = { name: '测试流程', xml: this.xmlData } // 请求后台接口(添加工作流) addFlow(data) .then(res => { console.log(res) }) .catch(err => { console.log(err) }) }, // 清除 cleanNode() { this.lf.clearData() }, // 重新加载 reloadNode() { // 查询数据 const data = { id: 'ff292b5d-4193-11ee-8e48-502b73dc5fce', name: '测试流程' } infoFlow(data) .then(res => { if (res.result) { // 获取processId const definitionsNodes = getTypeNameByTag(res.result, 'bpmn:definitions') const processNodes = getTypeNameByTag(res.result, 'bpmn:process') this.definitionsId = definitionsNodes.id this.processId = processNodes.id // 渲染数据 this.lf.render(res.result); } console.log(res) }) .catch(err => { console.log(err) }) }, // 删除数据 deleteNode() { // 查询数据 const data = { id: '' } deleteFlow(data) .then(res => { console.log(res) }) .catch(err => { console.log(err) }) }, // 获取数据 getList() { getDeployList() .then(res => { console.log(res) }) .catch(err => { console.log(err) }) }, // 流程实例 flowRun() { flowRun() .then(res => { console.log(res) }) .catch(err => { console.log(err) }) }, }, } .container{ width: 100%; height: 100%; } .node-item { position: absolute; top: 350px; left: 10px; width: 50px; padding: 10px; background-color: white; box-shadow: 0 0 10px 1px rgb(228, 224, 219); border-radius: 6px; text-align: center; z-index: 101; } .node-item-icon { width: 30px; height: 30px; margin-left: 10px; background-size: cover; } .node-label { font-size: 12px; margin-top: 5px; user-select: none; } .bpmn-save { background: url() center center no-repeat; cursor: grab; }
2.BpmnNodePanel.vue
import LogicFlow from '@logicflow/core'; export default { name: "BpmnNodePanel", data() { return {} }, props: { lf: Object, }, mounted() { //选区框选使用的 let lf = this.$props.lf lf && lf.on("selection:selected", () => { lf.updateEditConfig({ stopMoveGraph: false, }); }); }, methods: { openSelection() { (this.$props.lf).updateEditConfig({ stopMoveGraph: true }); }, addStartNode() { (this.$props.lf).dnd.startDrag({ type: "bpmn:startEvent", text: "开始节点", }); }, addUserTask() { (this.$props.lf).dnd.startDrag({ type: "bpmn:userTask", text: "普通节点", }); }, addServiceTask() { (this.$props.lf).dnd.startDrag({ type: "bpmn:serviceTask", text: "系统", }); }, addGateWay() { (this.$props.lf).dnd.startDrag({ type: "bpmn:exclusiveGateway", text: "判断", }); }, addEndNode() { (this.$props.lf).dnd.startDrag({ type: "bpmn:endEvent", text: "结束节点", }); }, }, }; .node-panel { position: absolute; top: 100px; left: 10px; width: 50px; padding: 10px; background-color: white; box-shadow: 0 0 10px 1px rgb(228, 224, 219); border-radius: 6px; text-align: center; z-index: 101; } .node-item { margin-bottom: 10px; } .node-item-icon { width: 30px; height: 30px; margin-left: 10px; background-size: cover; } .node-label { font-size: 12px; margin-top: 5px; user-select: none; } .bpmn-selection { background: url() center center no-repeat; cursor: grab; } .bpmn-start { background: url() center center no-repeat; cursor: grab; } .bpmn-end { background: url() center center no-repeat; cursor: grab; } .bpmn-user { background: url() center center no-repeat; cursor: grab; } .bpmn-gateway { background: url() center center no-repeat; cursor: grab; } .bpmn-service { background: url() center center no-repeat; cursor: grab; }开始节点普通节点结束节点
3.xml.js
/** * 添加flowable扩展 * @param {*} xmlstr * @returns */ export function addFlowable(xmlstr) { const part1 = xmlstr.slice(0, 43); // 从开头到指定位置之前的部分 const part2 = xmlstr.slice(43); // 从指定位置到末尾的部分 const newString = part1 + 'xmlns:flowable="http://flowable.org/bpmn" ' + part2; // 拼接成新的字符串 return newString } /** * 添加属性信息 * @param {*} xmlstr * @returns */ export function addProp(xmlstr, Elements, key, value) { // 创建一个XML文档对象 const parser = new DOMParser(); const xmlDoc = parser.parseFromString(xmlstr, 'application/xml'); // 查找元素 const userTaskElement = xmlDoc.getElementsByTagName(Elements); for (let i = 0; i4.server.js
import request from '../utils/request'; const url = 'http://localhost:8088' /** * 添加(编辑)工作流 * @param {*} data * @returns */ export function addFlow(data) { return request({ url: url + '/flow/addFlow', method: 'post', data, }); } /** * 查询工作流 * @param {*} data * @returns */ export function infoFlow(data) { return request({ url: url + '/flow/infoFlow', method: 'post', data, }); } /** * 删除工作流 * @param {*} data * @returns */ export function deleteFlow(data) { return request({ url: url + '/flow/deleteFlow', method: 'post', data, }); } /** * 获取工作流定义 * @param {*} data * @returns */ export function getDeployList(data) { return request({ url: url + '/flow/getDeployList', method: 'post', data, }); } /** * 工作流实例 * @param {*} data * @returns */ export function flowRun(data) { return request({ url: url + '/flow/deploymentRun', method: 'post', data, }); }5.request.js
import axios from 'axios'; const service = axios.create({}); service.defaults.timeout = 20000; // 请求拦截器 service.interceptors.request.use( (config) => { return config; }, (error) => { console.log(error); return Promise.reject(error); } ); // 响应拦截器 service.interceptors.response.use( (response) => { return response.data; }, (error) => { return Promise.reject(error); } ); export default service;四、后端代码(Flowable)
1.FlowController .java
@RestController public class FlowController { @Autowired FlowService flowService; // 工作流部署(添加、编辑) @CrossOrigin @PostMapping("/flow/addFlow") public Result addFlow(@RequestBody Map map){ return flowService.AddFlow(map.get("id"), map.get("name"), map.get("xml"), map.get("key")); } // 工作流返回 @CrossOrigin @PostMapping("/flow/infoFlow") public Result infoFlow(@RequestBody Map map) { return flowService.InfoFlow(map.get("id"), map.get("name")); } // 工作流删除 @CrossOrigin @PostMapping("/flow/deleteFlow") public Result deleteFlow(@RequestBody Map map) { return flowService.DeleteFlow(map.get("id")); } // 工作流查询 @CrossOrigin @PostMapping("/flow/getDeployList") public Result getDeployList() { return flowService.GetDeployList(); } // 流程实例 @CrossOrigin @PostMapping("/flow/deploymentRun") public Result deploymentRun(@RequestBody Map map) { return flowService.DeploymentRun(map.get("userId"), map.get("key")); } // 流程查询 @CrossOrigin @PostMapping("/flow/infoTask") public Result InfoTask(@RequestBody Map map) { return flowService.InfoTask(map.get("userId")); } // 流程执行 @CrossOrigin @PostMapping("/flow/makeTask") public Result MakeTask(@RequestBody Map map) { return flowService.MakeTask(map.get("userId")); } // 流程历史 @CrossOrigin @PostMapping("/flow/taskHistory") public Result TaskHistory() { return flowService.TaskHistory(); } // 测试 @CrossOrigin @PostMapping("/test") public Result test(@RequestBody AskForLeaveVO test) { return flowService.test(test); } }2.FlowService .java
@Service public class FlowService { @Autowired ProcessEngine processEngine; @Autowired FlowTreeService flowTreeService; /** * 工作流部署(添加、编辑) * @param name 名称 * @param xml xml * @return */ @Transactional public Result AddFlow(String id, String name, String xml, String key){ try { RepositoryService repositoryService = processEngine.getRepositoryService(); // 创建新流程 Deployment deployment = repositoryService.createDeployment() .addString(name + ".bpmn", xml) .deploy(); // 将工作流信息保存到流程树表 flowTreeService.AutoSave(id, deployment.getId(), key); System.out.println("id : " + id); System.out.println("添加id : " + deployment.getId()); System.out.println("添加key : " + key); return Result.ok("添加成功", deployment.getId()); } catch (Exception e) { e.printStackTrace(); return Result.error("添加失败"); } } /** * 工作流返回(返回xml) * @param id 查询id * @param name 名称 * @return */ @Transactional public Result InfoFlow(String id, String name) { try { // 流程查询 RepositoryService repositoryService = processEngine.getRepositoryService(); InputStream resourceAsStream = repositoryService.getResourceAsStream(id, name + ".bpmn"); // Java流转String String resultXml = new BufferedReader(new InputStreamReader(resourceAsStream,"utf-8")) .lines().collect(Collectors.joining(System.lineSeparator())); return Result.ok("查询成功", resultXml); } catch (Exception e) { e.printStackTrace(); } return Result.error("查询失败"); } /** * 工作流删除 * @param id * @return */ @Transactional public Result DeleteFlow(String id){ try { RepositoryService repositoryService = processEngine.getRepositoryService(); // 设置为TRUE 级联删除流程定义,及时流程有实例启动,也可以删除,设置为false 非级联删除操作。 repositoryService.deleteDeployment(id); System.out.println("删除id : " + id); return Result.ok("删除成功", id); } catch (Exception e) { e.printStackTrace(); return Result.error("删除失败"); } } /** * 工作流部署列表查询 * @return */ @Transactional public Result GetDeployList() { try { RepositoryService repositoryService = processEngine.getRepositoryService(); List list = repositoryService.createProcessDefinitionQuery().list(); System.out.println("list : " + list); return Result.ok("查询成功"); } catch (Exception e) { e.printStackTrace(); return Result.error("查询失败"); } } /** * 启动流程实例 * @param userId 发起人 * @param key 流程key * @return */ @Transactional public Result DeploymentRun(String userId, String key){ try { // Map variables = new HashMap(); // variables.put("employee", userId); RuntimeService runtimeService = processEngine.getRuntimeService(); ProcessInstance processInstance = runtimeService.startProcessInstanceByKey(key); //获取流程实例的相关信息 System.out.println("流程定义的id = " + processInstance.getProcessDefinitionId()); System.out.println("流程实例的id = " + processInstance.getId()); return Result.ok("成功", "流程实例id = " + processInstance.getId()); } catch (Exception e) { e.printStackTrace(); return Result.error("失败"); } } /** * 查询流程任务 * @param userId * @return */ @Transactional public Result InfoTask(String userId){ try { TaskService taskService = processEngine.getTaskService(); // 查询多人任务 List taskList = taskService.createTaskQuery().taskCandidateUser(userId).list(); System.out.println("taskList" + taskList); //遍历任务列表 for(Task task:taskList){ System.out.println("流程定义id = " + task.getProcessDefinitionId()); System.out.println("流程实例id = " + task.getProcessInstanceId()); System.out.println("任务id = " + task.getId()); System.out.println("任务名称 = " + task.getName()); } return Result.ok("成功"); } catch (Exception e) { e.printStackTrace(); return Result.error("失败"); } } /** * 执行流程任务 * @param userId * @return */ @Transactional public Result MakeTask(String userId){ try { TaskService taskService = processEngine.getTaskService(); // 查询个人任务 List list = taskService.createTaskQuery().taskCandidateUser(userId).list(); System.out.println("taskList" + list); for (Task task : list) { taskService.complete(task.getId()); System.out.println("task.getId()" + task.getId()); } return Result.ok("成功"); } catch (Exception e) { e.printStackTrace(); return Result.error("失败"); } } /** * 查询历史流程 * @param * @return */ @Transactional public Result TaskHistory(){ try { HistoryService historyService = processEngine.getHistoryService(); List activities = historyService.createHistoricActivityInstanceQuery() // .processInstanceId("12501") // 特定的实例 .finished() // 完成的 // .orderByHistoricActivityInstanceEndTime().asc() // 根据实例完成时间升序排列 .list(); for (HistoricActivityInstance activity : activities) { System.out.println("id:" + activity.getActivityId() + " 任务名:" + activity.getActivityName() + " 类型:" + activity.getActivityType() + " 持续时间:" + activity.getDurationInMillis()); } return Result.ok("成功"); } catch (Exception e) { e.printStackTrace(); return Result.error("失败"); } } }
免责声明:我们致力于保护作者版权,注重分享,被刊用文章因无法核实真实出处,未能及时与作者取得联系,或有版权异议的,请联系管理员,我们会立即处理! 部分文章是来自自研大数据AI进行生成,内容摘自(百度百科,百度知道,头条百科,中国民法典,刑法,牛津词典,新华词典,汉语词典,国家院校,科普平台)等数据,内容仅供学习参考,不准确地方联系删除处理! 图片声明:本站部分配图来自人工智能系统AI生成,觅知网授权图片,PxHere摄影无版权图库和百度,360,搜狗等多加搜索引擎自动关键词搜索配图,如有侵权的图片,请第一时间联系我们。