一、实现效果
前端使用LogicFlow框架绘制流程图,可以导出为xml工作流标准格式数据,通过xml文件传递到后端进行Flowable流程注册,并保存到数据库中。
二、BPM传输文件格式(.xml)
如需添加承办人的话,需要在LogicFlow导出文件的基础上手动添加xmlns:flowable="http://flowable.org/bpmn"
flowable插件,不然后台无法识别flowable:candidateUsers
。
<bpmn:definitions xmlns:flowable="http://flowable.org/bpmn" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:bpmn="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:dc="http://www.omg.org/spec/DD/20100524/DC" xmlns:di="http://www.omg.org/spec/DD/20100524/DI" id="Definitions_5588fe5_6" targetNamespace="http://logic-flow.org" exporter="logicflow" exporterVersion="1.2.0">
<bpmn:process isExecutable="true" id="Process_2a9c067_6">
<bpmn:startEvent id="Event_14efe0e" name="开始节点" flowable:candidateUsers="admin,admin1,admin2">
<bpmn:outgoing>Flow_a3e7d0c</bpmn:outgoing>
</bpmn:startEvent>
<bpmn:userTask id="Activity_602107f" name="普通节点" flowable:candidateUsers="uesr,uesr1,uesr2">
<bpmn:incoming>Flow_a3e7d0c</bpmn:incoming>
<bpmn:outgoing>Flow_3f9a386</bpmn:outgoing>
</bpmn:userTask>
<bpmn:endEvent id="Event_49a11b4" name="结束节点">
<bpmn:incoming>Flow_3f9a386</bpmn:incoming>
</bpmn:endEvent>
<bpmn:sequenceFlow id="Flow_a3e7d0c" sourceRef="Event_14efe0e" targetRef="Activity_602107f"/>
<bpmn:sequenceFlow id="Flow_3f9a386" sourceRef="Activity_602107f" targetRef="Event_49a11b4"/>
</bpmn:process>
<bpmndi:BPMNDiagram id="BPMNDiagram_1">
<bpmndi:BPMNPlane id="BPMNPlane_1" bpmnElement="Process_2a9c067">
<bpmndi:BPMNEdge id="Flow_a3e7d0c_di" bpmnElement="Flow_a3e7d0c">
<di:waypoint x="343" y="164"/>
<di:waypoint x="448" y="164"/>
</bpmndi:BPMNEdge>
<bpmndi:BPMNEdge id="Flow_3f9a386_di" bpmnElement="Flow_3f9a386">
<di:waypoint x="548" y="164"/>
<di:waypoint x="665" y="164"/>
</bpmndi:BPMNEdge>
<bpmndi:BPMNShape id="Event_14efe0e_di" bpmnElement="Event_14efe0e">
<dc:Bounds x="305" y="144" width="40" height="40"/>
<bpmndi:BPMNLabel>
<dc:Bounds x="305" y="197" width="40" height="14"/>
</bpmndi:BPMNLabel>
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="Activity_602107f_di" bpmnElement="Activity_602107f">
<dc:Bounds x="448" y="124" width="100" height="80"/>
<bpmndi:BPMNLabel>
<dc:Bounds x="478" y="157" width="40" height="14"/>
</bpmndi:BPMNLabel>
</bpmndi:BPMNShape>
<bpmndi:BPMNShape id="Event_49a11b4_di" bpmnElement="Event_49a11b4">
<dc:Bounds x="663" y="144" width="40" height="40"/>
<bpmndi:BPMNLabel>
<dc:Bounds x="663" y="197" width="40" height="14"/>
</bpmndi:BPMNLabel>
</bpmndi:BPMNShape>
</bpmndi:BPMNPlane>
</bpmndi:BPMNDiagram>
</bpmn:definitions>
三、前端框架(LogicFlow)
- LogicFlow.vue
<template>
<div class="container" ref="container"></div>
<BpmnNodePanel :lf="lf"></BpmnNodePanel>
<div class="node-item">
<div class="node-item-icon bpmn-save" @click="saveNode()"></div>
<span class="node-label">保存</span>
<div class="node-item-icon bpmn-save" @click="cleanNode()"></div>
<span class="node-label">清空</span>
<div class="node-item-icon bpmn-save" @click="reloadNode()"></div>
<span class="node-label">加载</span>
<div class="node-item-icon bpmn-save" @click="deleteNode()"></div>
<span class="node-label">删除</span>
<div class="node-item-icon bpmn-save" @click="getList()"></div>
<span class="node-label">获取定义流程</span>
<div class="node-item-icon bpmn-save" @click="flowRun()"></div>
<span class="node-label">工作流实例</span>
</div>
</template>
<script>
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)
})
},
},
}
</script>
<style scoped>
.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;
}
</style>
2.BpmnNodePanel.vue
<template>
<div class="node-panel">
<!-- <div class="node-item" @mousedown="openSelection()">
<div class="node-item-icon bpmn-selection"></div>
<span class="node-label">选区</span>
</div> -->
<div class="node-item" @mousedown="addStartNode()">
<div class="node-item-icon bpmn-start"></div>
<span class="node-label">开始节点</span>
</div>
<div class="node-item" @mousedown="addUserTask()">
<div class="node-item-icon bpmn-user"></div>
<span class="node-label">普通节点</span>
</div>
<!-- <div class="node-item" @mousedown="addServiceTask()">
<div class="node-item-icon bpmn-service"></div>
<span class="node-label">系统</span>
</div> -->
<!-- <div class="node-item" @mousedown="addGateWay()">
<div class="node-item-icon bpmn-gateway"></div>
<span class="node-label">判断</span>
</div> -->
<div class="node-item" @mousedown="addEndNode()">
<div class="node-item-icon bpmn-end"></div>
<span class="node-label">结束节点</span>
</div>
</div>
</template>
<script>
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: "结束节点",
});
},
},
};
</script>
<style>
.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;
}
</style>
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; i < userTaskElement.length; i++) {
if (!userTaskElement[i].attributes[key]) {
userTaskElement[i].setAttribute(key, value);
}
}
// 将修改后的XML文档转换回字符串
const xmlSerializer = new XMLSerializer();
const modifiedXmlString = xmlSerializer.serializeToString(xmlDoc);
return modifiedXmlString;
}
/**
*根据标签名称返回xml内容,标签名必须唯一,若不满足,修改方法
* @param {*} xmlstr xml字符串
* @param {element} tagName 标签名称,如<Name>
*/
export function getTypeNameByTag(xmlstr, tagName) {
const parser = new DOMParser()
const xmlDoc = parser.parseFromString(xmlstr, 'application/xml')
const nameNodes = xmlDoc.getElementsByTagName(tagName)[0]
if (nameNodes) {
return nameNodes
} else {
return false
}
}
/**
* 根据标签名称修改内容
* @param {*} xmlstr xml字符串
* @param {element} tagName 标签名称,如<Name>
* @param {*} key 修改对应key
* @param {*} value 修改值value
* @returns
*/
export function setValueByTag(xmlstr, tagName, key, value) {
// 字符串转xml
const parser = new DOMParser()
const xmlDoc = parser.parseFromString(xmlstr, 'application/xml')
const nameNodes = xmlDoc.getElementsByTagName(tagName)[0]
nameNodes.setAttribute(key, value);
// xml转字符串
const s = new XMLSerializer();
const xml = s.serializeToString(xmlDoc);
return xml;
}
4.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<String,String> 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<String,String> map) {
return flowService.InfoFlow(map.get("id"), map.get("name"));
}
// 工作流删除
@CrossOrigin
@PostMapping("/flow/deleteFlow")
public Result deleteFlow(@RequestBody Map<String,String> 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<String,String> map) {
return flowService.DeploymentRun(map.get("userId"), map.get("key"));
}
// 流程查询
@CrossOrigin
@PostMapping("/flow/infoTask")
public Result InfoTask(@RequestBody Map<String,String> map) {
return flowService.InfoTask(map.get("userId"));
}
// 流程执行
@CrossOrigin
@PostMapping("/flow/makeTask")
public Result MakeTask(@RequestBody Map<String,String> 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<ProcessDefinition> 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<String, Object> variables = new HashMap<String, Object>();
// 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<Task> 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<Task> 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<HistoricActivityInstance> 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("失败");
}
}
}