SpringBoot+Activiti+bpmn.js+Vue.js+Elementui(OA系统审批流)
引言:OA系统用到请假、加班、调休、离职,需要使用工作流进行流程审批
一:activiti流程设计器的选择(通过学习activiti工作流过程中,发现一款好的流程设计器将会更好的方便的设计好流程(主要介绍BPMN.JS))
1.最开始使用activiti流程设计器是开发工具idea或eclipse中安装的bpmn流程插件(直接安装使用)
缺点:activiti官方已经不再维护这些插件,而且结合企业开发存在很大的缺陷
2.1:通过整合Acitiviti官网的在线流程设计器(Activiti-Modeler )
缺点:需要整合官方项目结构,就是将官方的项目整合到自己项目,在开发前后端分离时,需要使用ifram插件嵌入到我们的前端项目中,不方便结合前端框架element ui使用
<iframe src="http://localhost:8085/modeler.html?modelId=297501" id="show-iframe" name="showHere" scrolling="auto" width="100%" height="901px" frameborder="0"></iframe>
2.2:企业二次开发后的activiti流程设计器(maven依赖)
<properties>
<jeesite-module-core.version>4.0.3-SNAPSHOT</jeesite-module-core.version>
<jeesite-module-bpm.version>4.0-SNAPSHOT</jeesite-module-bpm.version>
</properties>
<!-- 核心模块 -->
<dependency>
<groupId>com.jeesite</groupId>
<artifactId>jeesite-module-core</artifactId>
<version>${jeesite-module-core.version}</version>
</dependency>
<!-- 工作流模块 -->
<dependency>
<groupId>com.jeesite</groupId>
<artifactId>jeesite-module-bpm</artifactId>
<version>${jeesite-module-bpm.version}</version>
</dependency>
3.通过使用bpmn.io中的流程设计器可以很好前后端分离,特别是结合前端vue+element ui 使用,非常的方便
3.1.首先,安装node.js,通过npm安装bpmn.js
npm install bpmn-js
3.2.前端整合bpmn.js,并且改造bpmn流程设计器,可以从公司组织架构选中指定办理人
<!-- 流程设计 -->
<template>
<div>
<!-- 表单区域 -->
<div class="chunk">
<el-link icon="el-icon-arrow-left" @click="Create()">返回</el-link>
<el-form ref="params" :model="params" :rules="rules" :inline="true">
<el-form-item label="流程名称:" prop="modelName">
<el-input v-model="params.modelName" placeholder="请输入" clearable/>
</el-form-item>
<el-form-item label="版本:" prop="modelKey">
<el-input v-model="params.modelKey" placeholder="请输入" clearable/>
</el-form-item>
</el-form>
</div>
<div ref="content" class="containers">
<div ref="canvas" class="canvas">
<div style="margin-bottom: 10px">
<!-- 保存 -->
<el-button icon="el-icon-folder-checked" @click="saveBpmn('params')">保存</el-button>
<!-- 导入 -->
<input id="btn_file" type="file" style="display:none" @change="showBPMN">
<el-button icon="el-icon-folder-opened" @click="Import()">导入</el-button>
<!-- 导出 -->
<el-button icon="el-icon-download" @click="Export()">导出BPMN</el-button>
<!-- 导出 -->
<el-button icon="el-icon-picture-outline" @click="ExportImg()">导出图片</el-button>
<!-- 前进 -->
<el-button icon="el-icon-folder-checked" @click="advance()">前进</el-button>
<!-- 后退 -->
<el-button icon="el-icon-folder-checked" @click="retreat()">后退</el-button>
</div>
</div>
<div id="js-properties-panel" class="panel"/>
</div>
<!--选择考勤人员-->
<el-dialog width="600px" title="选择参与考勤人员" :visible.sync="innerVisible" append-to-body>
<el-row>
<el-col :span="24">
<div class="grid-content bg-purple">
<el-input v-model="filterText" placeholder="输入关键字进行过滤"/>
<el-divider/>
<el-scrollbar style="border: 1px solid #DCDFE6;">
<el-tree
ref="tree"
:data="treeData"
accordion
node-key="id"
:default-expanded-keys="attendancePersonnel"
:default-checked-keys="attendancePersonnel "
highlight-current
:props="{children: 'children',label: 'name'}"
:filter-node-method="filterNode"
style="height: 277px;"
>
<span slot-scope="{ data }" class="custom-tree-node">
<el-radio v-if="data.role" v-model="UserObj" :label="data.userEmployeeId">{{ data.name }}</el-radio>
<span v-else>{{ data.name }}</span>
</span>
</el-tree>
</el-scrollbar>
</div>
</el-col>
</el-row>
<div slot="footer" class="dialog-footer">
<el-button @click="innerVisible = false">取消</el-button>
<el-button type="primary" @click="getCheckedNodes">确定</el-button>
</div>
</el-dialog>
</div>
</template>
<script>
import { getCorpStructure } from '@/api/system-setup'
// 引入相关的依赖
// import BpmnViewer from 'bpmn-js'
import BpmnModeler from 'bpmn-js/lib/Modeler'
import propertiesPanelModule from 'bpmn-js-properties-panel'
import propertiesProviderModule from 'bpmn-js-properties-panel/lib/provider/camunda'
import camundaModdleDescriptor from 'camunda-bpmn-moddle/resources/camunda'
// 图片转换
import canvg from 'canvg'
// api接口
import { saveModel, modelsShows, modifyModels } from '@/api/Approval-manage/approvalManage'
var Vue = {
name: 'Bpmn',
props: ['modelId'],
data() {
return {
// 流程对象
params: {
modelId: '', // 模型编号
modelName: '', // 模型名称
modelKey: '', // 模型key(版本)
modelImage: '', // 模型图片
modelXml: '' // 模型文件
},
rules: {
modelName: [{ required: true, message: '请输入流程名称', trigger: 'blur' }],
modelKey: [{ required: true, message: '请输入版本号', trigger: 'blur' }]
},
// bpmn建模器
bpmnModeler: null,
container: null,
canvas: null,
filterText: '',
treeData: [], // 树控件数据集合(员工信息),
attendancePersonnel: [], // 参与人员
UserObj: '', // 选中的人
e: null,
innerVisible: false // 选择员工弹出层
}
},
watch: {
filterText(val) {
this.$refs.tree.filter(val)
}
},
mounted() {
var than = this
// 流程设计模块的点击事件
window.showUserInfo = function(e) {
than.innerVisible = true
than.e = e
}
// 获取到属性ref为“content”的dom节点
this.container = this.$refs.content
// 获取到属性ref为“canvas”的dom节点
const canvas = this.$refs.canvas
// 建模,官方文档这里讲的很详细
this.bpmnModeler = new BpmnModeler({
container: canvas,
// 添加控制板
propertiesPanel: {
parent: '#js-properties-panel'
},
additionalModules: [
propertiesProviderModule, // 左边工具栏以及节点
propertiesPanelModule // 右边的工具栏
],
// 如果您想在属性面板中维护camunda:XXX属性,则需要该属性
moddleExtensions: {
camunda: camundaModdleDescriptor
}
})
this.getCorpStructure()
if (this.modelId) {
this.params.modelId = this.modelId
modelsShows(this.params.modelId).then(res => {
this.params = res.data
this.createNewDiagram(res.data.modelXml)
})
} else {
this.createNewDiagram('')
}
},
methods: {
// 返回
Create() {
this.$emit('transfer', '') // 将值绑定到transfer上传递过去
// this.$router.push({ path: 'bpmn', query: { approvalSetId: 1, publishedList: 1, create: true }})
},
// 流程设计模板创建
createNewDiagram(bpmnXML) {
if (bpmnXML === '' || bpmnXML === null) {
bpmnXML = '<?xml version="1.0" encoding="UTF-8"?>\n' +
'<definitions xmlns="http://www.omg.org/spec/BPMN/20100524/MODEL" xmlns:camunda="http://camunda.org/schema/1.0/bpmn" xmlns:bpmndi="http://www.omg.org/spec/BPMN/20100524/DI" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:activiti="http://activiti.org/bpmn" xmlns:tns="http://www.activiti.org/testm1568796216967" xmlns:xsd="http://www.w3.org/2001/XMLSchema" id="m1568796216967" name="" targetNamespace="http://www.activiti.org/testm1568796216967">\n' +
' <process id="myProcess_1" processType="None" isClosed="false" isExecutable="true" />\n' +
' <bpmndi:BPMNDiagram id="Diagram-_1" name="New Diagram" documentation="background=#FFFFFF;count=1;horizontalcount=1;orientation=0;width=842.4;height=1195.2;imageableWidth=832.4;imageableHeight=1185.2;imageableX=5.0;imageableY=5.0">\n' +
' <bpmndi:BPMNPlane bpmnElement="myProcess_1" />\n' +
' </bpmndi:BPMNDiagram>\n' +
'</definitions>'
}
// 将字符串转换成图显示出来
this.bpmnModeler.importXML(bpmnXML, function(err) {
if (err) {
return console.error('could not import BPMN 2.0 diagram', err)
}
})
},
// 保存
saveBpmn(formName) {
var than = this
this.$refs[formName].validate((valid) => {
if (valid) {
// 获取XML数据
than.bpmnModeler.saveXML({ format: true }, function(err, xml) {
if (!err) {
than.params.modelXml = xml
// 获取SVG数据(图片)
than.bpmnModeler.saveSVG({ format: true }, (err, data) => {
if (!err) {
var svgXml = data
var canvas = document.createElement('canvas') // 准备空画布
canvas.width = '1000px'
canvas.height = screen.availHeight
canvg(canvas, svgXml)
var imagedata = canvas.toDataURL('image/png')
than.params.modelImage = imagedata
if (than.modelId) {
modifyModels(than.params).then(res => {
if (delop.message(than, res)) {
than.Create()
}
})
} else {
saveModel(than.params).then(res => {
if (delop.message(than, res)) {
than.Create()
}
})
}
}
})
}
})
} else {
return false
}
})
},
// 导入
Import() {
document.getElementById('btn_file').click()
},
showBPMN() {
var than = this
var file = document.getElementById('btn_file').files[0]
var URL = window.URL || window.webkitURL
var imgURL = URL.createObjectURL(file)
$.ajax({
type: 'get',
url: imgURL,
dataType: 'text', // 返回格式
success: function(data) {
than.createNewDiagram(data)
}
})
},
// 导出bpmn文件
Export() {
this.bpmnModeler.saveXML({ format: true }, function(err, xml) {
if (err) {
return console.error('无法保存BPMN 2.0关系图', err)
}
// 如果浏览器支持msSaveOrOpenBlob方法(也就是使用IE浏览器的时候)
if (window.navigator.msSaveOrOpenBlob) {
var blob = new Blob([xml], { type: 'text/plain' })
window.navigator.msSaveOrOpenBlob(blob, '工作流程图BPMN20.bpmn')
} else {
var eleLink = document.createElement('a')
eleLink.download = '工作流程图BPMN20.bpmn'
eleLink.style.display = 'none'
const blob = new Blob([xml]) // 字符内容转变成blob地址
eleLink.href = URL.createObjectURL(blob)
document.body.appendChild(eleLink) // 触发点击
eleLink.click()
document.body.removeChild(eleLink) // 然后移除
}
})
},
// 导出图片
ExportImg() {
if (window.navigator.msSaveOrOpenBlob) {
console.log('IE浏览器无法下载,建议使用谷歌浏览器')
return
}
// 获取SVG数据(图片)
this.bpmnModeler.saveSVG({ format: true }, (err, data) => {
if (err) {
console.log('保存失败')
}
var svgXml = data
var canvas = document.createElement('canvas') // 准备空画布
canvas.width = '1000px'
canvas.height = screen.availHeight
canvg(canvas, svgXml)
var imagedata = canvas.toDataURL('image/png')
// console.log(imagedata)
// 如果浏览器支持msSaveOrOpenBlob方法(也就是使用IE浏览器的时候)
if (window.navigator.msSaveOrOpenBlob) {
var bstr = atob(imagedata.split(',')[1])
var n = bstr.length
var u8arr = new Uint8Array(n)
while (n--) {
u8arr[n] = bstr.charCodeAt(n)
}
var blob = new Blob([u8arr])
window.navigator.msSaveOrOpenBlob(blob, '工作流程图.png')
} else {
var a = document.createElement('a')
a.href = imagedata // 将画布内的信息导出为png图片数据
a.download = '工作流程图' // 设定下载名称
a.click() // 点击触发下载
}
})
},
// 前进
advance() {
this.bpmnModeler.get('commandStack').redo()
},
// 后退
retreat() {
this.bpmnModeler.get('commandStack').undo()
},
// 选择代理人关键字搜索
filterNode(value, data) {
if (!value) return true
return data.name.indexOf(value) !== -1
},
// 查询公司组织树信息
getCorpStructure() {
getCorpStructure().then(res => {
this.treeData = [res.data]
this.option(this.treeData)
})
},
// 查询公司组织用户数据
option(options) {
const than = this
if (options != null && options.length > 0) {
options.forEach(function(obj1, i) {
if (obj1.userInfo && JSON.parse(obj1.userInfo).length > 0) {
var user = JSON.parse(obj1.userInfo)
user.forEach(function(obj, index) {
obj.role = 'user'
})
obj1.children = user
}
than.option(obj1.children)
})
}
}, // 选中的考勤人员
getCheckedNodes() {
var input = this.e.previousElementSibling
$(input).val(this.UserObj)
// 以下代码必须添加,不然文本框内容填充无效。
var changeEvent = document.createEvent('HTMLEvents') // 创建输入框修改事件
changeEvent.initEvent('change', true, true)
$(input)[0].dispatchEvent(changeEvent) // 触发修改事件
this.innerVisible = false
}
}
}
export default Vue
</script>
<style scoped>
/*左边工具栏以及编辑节点的样式*/
@import '~bpmn-js/dist/assets/diagram-js.css';
@import '~bpmn-js/dist/assets/bpmn-font/css/bpmn.css';
@import '~bpmn-js/dist/assets/bpmn-font/css/bpmn-codes.css';
@import '~bpmn-js/dist/assets/bpmn-font/css/bpmn-embedded.css';
/*!*右边工具栏样式*!*/
@import '~bpmn-js-properties-panel/dist/assets/bpmn-js-properties-panel.css';
.containers {
position: absolute;
background-color: #ffffff;
width: 98%;
height: 800px;
}
.canvas {
float: left;
border: 1px solid #eee;
width: 80%;
height: 100%;
}
.panel {
float: right;
width: 20%;
}
</style>
3.2.1相关模块管理
3.3 springboot 整合 activiti 配置类,实现spring容器注入
package edu.nf.project.config;
import org.activiti.engine.*;
import org.activiti.spring.ProcessEngineFactoryBean;
import org.activiti.spring.SpringProcessEngineConfiguration;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.Resource;
import org.springframework.core.io.support.PathMatchingResourcePatternResolver;
import org.springframework.transaction.PlatformTransactionManager;
import javax.sql.DataSource;
import java.io.IOException;
/**
* @author : ywb
* @createdDate : 2019/9/20
* @updatedDate
*/
@Configuration
public class ActivitiConfiguration {
@Autowired
private DataSource dataSource;
@Autowired
private PlatformTransactionManager platformTransactionManager;
/**
* 创建一个流程引擎的配置对象 这里我们使用的是Spring提供的流程引擎对象
* 创建一个流程引擎的配置对象 这里我摸嗯使用的事Spring提供的流程引擎对象
* @return
*/
@Bean
public SpringProcessEngineConfiguration springProcessEngineConfiguration(){
SpringProcessEngineConfiguration spec = new SpringProcessEngineConfiguration();
spec.setDataSource(dataSource);
spec.setTransactionManager(platformTransactionManager);
spec.setDatabaseSchemaUpdate("true");
Resource[] resources = null;
// 启动自动部署流程
try {
resources = new PathMatchingResourcePatternResolver().getResources("classpath*:bpmn/*.bpmn");
} catch (IOException e) {
e.printStackTrace();
}
spec.setDeploymentResources(resources);
return spec;
}
/**
* 创建一个流程引擎bean
* @return
*/
@Bean
public ProcessEngineFactoryBean processEngine(){
ProcessEngineFactoryBean processEngineFactoryBean = new ProcessEngineFactoryBean();
processEngineFactoryBean.setProcessEngineConfiguration(springProcessEngineConfiguration());
return processEngineFactoryBean;
}
/**
* 工作流仓储服务,对所有atc_re开头的表进行操作-
* @return
* @throws Exception
*/
@Bean
public RepositoryService repositoryService() throws Exception{
return processEngine().getObject().getRepositoryService();
}
/**
* 工作流运行服务 对所有act_ru开头的表进行操作
* @return
* @throws Exception
*/
@Bean
public RuntimeService runtimeService() throws Exception{
return processEngine().getObject().getRuntimeService();
}
/**
* 工作流任务服务
* @return
* @throws Exception
*/
@Bean
public TaskService taskService() throws Exception{
return processEngine().getObject().getTaskService();
}
/**
* 工作流历史数据服务 对所有的act_hi开头的表进行操作
* @return
* @throws Exception
*/
@Bean
public HistoryService historyService() throws Exception{
return processEngine().getObject().getHistoryService();
}
/**
* 工作流唯一服务,对所有以act_id开头的表进行增删改查
* @return
* @throws Exception
*/
public IdentityService identityService() throws Exception{
return processEngine().getObject().getIdentityService();
}
/**
* 工作流管理服务
* @return
* @throws Exception
*/
public FormService formService() throws Exception{
return processEngine().getObject().getFormService();
}
/**
* 工作流管理服务
* @return
* @throws Exception
*/
public ManagementService managementService() throws Exception{
return processEngine().getObject().getManagementService();
}
}
3.4建立模型表,用于保存bpmn文件
/*
Navicat Premium Data Transfer
Source Server : mydb
Source Server Type : MySQL
Source Server Version : 50726
Source Host : localhost:3306
Source Schema : mydb
Target Server Type : MySQL
Target Server Version : 50726
File Encoding : 65001
Date: 20/12/2019 17:09:21
*/
SET NAMES utf8mb4;
SET FOREIGN_KEY_CHECKS = 0;
-- ----------------------------
-- Table structure for pms_models
-- ----------------------------
DROP TABLE IF EXISTS `pms_models`;
CREATE TABLE `pms_models` (
`model_id` int(11) NOT NULL AUTO_INCREMENT COMMENT '模型编号',
`model_name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '模型名称',
`model_image` text CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '模型图片',
`model_xml` text CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL COMMENT '模型文件',
`model_key` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '模型key',
`user_name` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NULL DEFAULT NULL COMMENT '创建人',
`create_time` timestamp(0) NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP(0) COMMENT '创建时间',
`update_time` timestamp(0) NULL DEFAULT NULL ON UPDATE CURRENT_TIMESTAMP(0) COMMENT '修改时间',
`model_status` int(1) NULL DEFAULT NULL COMMENT '是否发布(0:发布;1:不发布)',
`deprecated` int(1) NULL DEFAULT NULL COMMENT '是否弃用',
PRIMARY KEY (`model_id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 37 CHARACTER SET = utf8mb4 COLLATE = utf8mb4_general_ci ROW_FORMAT = Dynamic;
-- ----------------------------
-- Records of pms_models
-- ----------------------------
INSERT INTO `pms_models` VALUES (30, '分支流程', '', '<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<definitions xmlns=\"http://www.omg.org/spec/BPMN/20100524/MODEL\" xmlns:camunda=\"http://camunda.org/schema/1.0/bpmn\" xmlns:bpmndi=\"http://www.omg.org/spec/BPMN/20100524/DI\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:dc=\"http://www.omg.org/spec/DD/20100524/DC\" xmlns:di=\"http://www.omg.org/spec/DD/20100524/DI\" xmlns:activiti=\"http://activiti.org/bpmn\" xmlns:tns=\"http://www.activiti.org/testm1568796216967\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" id=\"m1568796216967\" name=\"\" targetNamespace=\"http://www.activiti.org/testm1568796216967\">\n <process id=\"myProcess_1\" processType=\"None\" isClosed=\"false\" isExecutable=\"true\">\n <startEvent id=\"StartEvent_196hi8b\" name=\"开始\">\n <outgoing>SequenceFlow_06ukbb1</outgoing>\n </startEvent>\n <sequenceFlow id=\"SequenceFlow_06ukbb1\" sourceRef=\"StartEvent_196hi8b\" targetRef=\"Task_1m65c6z\" />\n <exclusiveGateway id=\"ExclusiveGateway_1716w5k\">\n <incoming>SequenceFlow_0ict9lf</incoming>\n <outgoing>SequenceFlow_00x90vg</outgoing>\n <outgoing>SequenceFlow_01sftoz</outgoing>\n </exclusiveGateway>\n <sequenceFlow id=\"SequenceFlow_0ict9lf\" sourceRef=\"Task_1m65c6z\" targetRef=\"ExclusiveGateway_1716w5k\" />\n <sequenceFlow id=\"SequenceFlow_00x90vg\" sourceRef=\"ExclusiveGateway_1716w5k\" targetRef=\"Task_1xbbk2a\">\n <conditionExpression xsi:type=\"tFormalExpression\">${condition>1000}</conditionExpression>\n </sequenceFlow>\n <sequenceFlow id=\"SequenceFlow_01sftoz\" sourceRef=\"ExclusiveGateway_1716w5k\" targetRef=\"Task_15bkdk1\">\n <conditionExpression xsi:type=\"tFormalExpression\">${condition<=1000}</conditionExpression>\n </sequenceFlow>\n <endEvent id=\"EndEvent_0lupiok\" name=\"结束\">\n <incoming>SequenceFlow_00cspzq</incoming>\n <incoming>SequenceFlow_12b0itr</incoming>\n </endEvent>\n <sequenceFlow id=\"SequenceFlow_00cspzq\" name=\"通过\" sourceRef=\"Task_1xbbk2a\" targetRef=\"EndEvent_0lupiok\" />\n <sequenceFlow id=\"SequenceFlow_12b0itr\" name=\"通过\" sourceRef=\"Task_15bkdk1\" targetRef=\"EndEvent_0lupiok\" />\n <manualTask id=\"Task_1m65c6z\" name=\"提交申请\">\n <incoming>SequenceFlow_06ukbb1</incoming>\n <outgoing>SequenceFlow_0ict9lf</outgoing>\n </manualTask>\n <userTask id=\"Task_1xbbk2a\" name=\"项目经理审批\" camunda:assignee=\"0003\">\n <incoming>SequenceFlow_00x90vg</incoming>\n <outgoing>SequenceFlow_00cspzq</outgoing>\n </userTask>\n <userTask id=\"Task_15bkdk1\" name=\"产品经理审批\" camunda:assignee=\"0002\">\n <incoming>SequenceFlow_01sftoz</incoming>\n <outgoing>SequenceFlow_12b0itr</outgoing>\n </userTask>\n </process>\n <bpmndi:BPMNDiagram id=\"Diagram-_1\" name=\"New Diagram\" documentation=\"background=#FFFFFF;count=1;horizontalcount=1;orientation=0;width=842.4;height=1195.2;imageableWidth=832.4;imageableHeight=1185.2;imageableX=5.0;imageableY=5.0\">\n <bpmndi:BPMNPlane bpmnElement=\"myProcess_1\">\n <bpmndi:BPMNShape id=\"StartEvent_196hi8b_di\" bpmnElement=\"StartEvent_196hi8b\">\n <dc:Bounds x=\"232\" y=\"352\" width=\"36\" height=\"36\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"239\" y=\"395\" width=\"22\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_06ukbb1_di\" bpmnElement=\"SequenceFlow_06ukbb1\">\n <di:waypoint x=\"268\" y=\"370\" />\n <di:waypoint x=\"320\" y=\"370\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"ExclusiveGateway_1716w5k_di\" bpmnElement=\"ExclusiveGateway_1716w5k\" isMarkerVisible=\"true\">\n <dc:Bounds x=\"475\" y=\"345\" width=\"50\" height=\"50\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_0ict9lf_di\" bpmnElement=\"SequenceFlow_0ict9lf\">\n <di:waypoint x=\"420\" y=\"370\" />\n <di:waypoint x=\"475\" y=\"370\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_00x90vg_di\" bpmnElement=\"SequenceFlow_00x90vg\">\n <di:waypoint x=\"500\" y=\"345\" />\n <di:waypoint x=\"500\" y=\"300\" />\n <di:waypoint x=\"610\" y=\"300\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_01sftoz_di\" bpmnElement=\"SequenceFlow_01sftoz\">\n <di:waypoint x=\"500\" y=\"395\" />\n <di:waypoint x=\"500\" y=\"470\" />\n <di:waypoint x=\"610\" y=\"470\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"EndEvent_0lupiok_di\" bpmnElement=\"EndEvent_0lupiok\">\n <dc:Bounds x=\"792\" y=\"372\" width=\"36\" height=\"36\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"726\" y=\"380\" width=\"22\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_00cspzq_di\" bpmnElement=\"SequenceFlow_00cspzq\">\n <di:waypoint x=\"710\" y=\"300\" />\n <di:waypoint x=\"810\" y=\"300\" />\n <di:waypoint x=\"810\" y=\"372\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"750\" y=\"282\" width=\"22\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_12b0itr_di\" bpmnElement=\"SequenceFlow_12b0itr\">\n <di:waypoint x=\"710\" y=\"470\" />\n <di:waypoint x=\"810\" y=\"470\" />\n <di:waypoint x=\"810\" y=\"408\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"750\" y=\"452\" width=\"22\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"ManualTask_027mo5z_di\" bpmnElement=\"Task_1m65c6z\">\n <dc:Bounds x=\"320\" y=\"330\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNShape id=\"UserTask_0fgmota_di\" bpmnElement=\"Task_1xbbk2a\">\n <dc:Bounds x=\"610\" y=\"260\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNShape id=\"UserTask_1pcwlvh_di\" bpmnElement=\"Task_15bkdk1\">\n <dc:Bounds x=\"610\" y=\"430\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n </bpmndi:BPMNPlane>\n </bpmndi:BPMNDiagram>\n</definitions>\n', '1.0', NULL, '2019-11-22 10:55:03', '2019-11-22 10:55:03', 0, 0);
INSERT INTO `pms_models` VALUES (33, '加班流程', '', '<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<definitions xmlns=\"http://www.omg.org/spec/BPMN/20100524/MODEL\" xmlns:camunda=\"http://camunda.org/schema/1.0/bpmn\" xmlns:bpmndi=\"http://www.omg.org/spec/BPMN/20100524/DI\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:dc=\"http://www.omg.org/spec/DD/20100524/DC\" xmlns:di=\"http://www.omg.org/spec/DD/20100524/DI\" xmlns:activiti=\"http://activiti.org/bpmn\" xmlns:tns=\"http://www.activiti.org/testm1568796216967\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" id=\"m1568796216967\" name=\"\" targetNamespace=\"http://www.activiti.org/testm1568796216967\">\n <process id=\"myProcess_1\" processType=\"None\" isClosed=\"false\" isExecutable=\"true\">\n <startEvent id=\"StartEvent_1103sa0\" name=\"开始\">\n <outgoing>SequenceFlow_1etqq41</outgoing>\n </startEvent>\n <sequenceFlow id=\"SequenceFlow_1etqq41\" sourceRef=\"StartEvent_1103sa0\" targetRef=\"Task_0y7ovyq\" />\n <sequenceFlow id=\"SequenceFlow_11v2do8\" sourceRef=\"Task_0y7ovyq\" targetRef=\"Task_1ravuab\" />\n <sequenceFlow id=\"SequenceFlow_18x1ap2\" sourceRef=\"Task_1ravuab\" targetRef=\"Task_0n0dbcl\" />\n <endEvent id=\"EndEvent_0l75vmy\" name=\"结束\">\n <incoming>SequenceFlow_160u4u8</incoming>\n </endEvent>\n <sequenceFlow id=\"SequenceFlow_160u4u8\" sourceRef=\"Task_0n0dbcl\" targetRef=\"EndEvent_0l75vmy\" />\n <manualTask id=\"Task_0y7ovyq\" name=\"提交申请\">\n <incoming>SequenceFlow_1etqq41</incoming>\n <outgoing>SequenceFlow_11v2do8</outgoing>\n </manualTask>\n <userTask id=\"Task_1ravuab\" name=\"部门组长审批\" camunda:assignee=\"0003\">\n <incoming>SequenceFlow_11v2do8</incoming>\n <outgoing>SequenceFlow_18x1ap2</outgoing>\n </userTask>\n <userTask id=\"Task_0n0dbcl\" name=\"部门经理审批\" camunda:assignee=\"0002\">\n <incoming>SequenceFlow_18x1ap2</incoming>\n <outgoing>SequenceFlow_160u4u8</outgoing>\n </userTask>\n </process>\n <bpmndi:BPMNDiagram id=\"Diagram-_1\" name=\"New Diagram\" documentation=\"background=#FFFFFF;count=1;horizontalcount=1;orientation=0;width=842.4;height=1195.2;imageableWidth=832.4;imageableHeight=1185.2;imageableX=5.0;imageableY=5.0\">\n <bpmndi:BPMNPlane bpmnElement=\"myProcess_1\">\n <bpmndi:BPMNShape id=\"StartEvent_1103sa0_di\" bpmnElement=\"StartEvent_1103sa0\">\n <dc:Bounds x=\"272\" y=\"192\" width=\"36\" height=\"36\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"279\" y=\"235\" width=\"22\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_1etqq41_di\" bpmnElement=\"SequenceFlow_1etqq41\">\n <di:waypoint x=\"308\" y=\"210\" />\n <di:waypoint x=\"360\" y=\"210\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_11v2do8_di\" bpmnElement=\"SequenceFlow_11v2do8\">\n <di:waypoint x=\"460\" y=\"210\" />\n <di:waypoint x=\"520\" y=\"210\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_18x1ap2_di\" bpmnElement=\"SequenceFlow_18x1ap2\">\n <di:waypoint x=\"620\" y=\"210\" />\n <di:waypoint x=\"680\" y=\"210\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"EndEvent_0l75vmy_di\" bpmnElement=\"EndEvent_0l75vmy\">\n <dc:Bounds x=\"842\" y=\"192\" width=\"36\" height=\"36\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"849\" y=\"235\" width=\"22\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_160u4u8_di\" bpmnElement=\"SequenceFlow_160u4u8\">\n <di:waypoint x=\"780\" y=\"210\" />\n <di:waypoint x=\"842\" y=\"210\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"ManualTask_1b6lmz6_di\" bpmnElement=\"Task_0y7ovyq\">\n <dc:Bounds x=\"360\" y=\"170\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNShape id=\"UserTask_0x6mljz_di\" bpmnElement=\"Task_1ravuab\">\n <dc:Bounds x=\"520\" y=\"170\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNShape id=\"UserTask_0o9ubqi_di\" bpmnElement=\"Task_0n0dbcl\">\n <dc:Bounds x=\"680\" y=\"170\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n </bpmndi:BPMNPlane>\n </bpmndi:BPMNDiagram>\n</definitions>\n', '1.0', NULL, '2019-12-06 17:34:20', '2019-12-06 17:34:20', 0, 0);
INSERT INTO `pms_models` VALUES (34, '请假流程', '', '<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<definitions xmlns=\"http://www.omg.org/spec/BPMN/20100524/MODEL\" xmlns:camunda=\"http://camunda.org/schema/1.0/bpmn\" xmlns:bpmndi=\"http://www.omg.org/spec/BPMN/20100524/DI\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:dc=\"http://www.omg.org/spec/DD/20100524/DC\" xmlns:di=\"http://www.omg.org/spec/DD/20100524/DI\" xmlns:activiti=\"http://activiti.org/bpmn\" xmlns:tns=\"http://www.activiti.org/testm1568796216967\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" id=\"m1568796216967\" name=\"\" targetNamespace=\"http://www.activiti.org/testm1568796216967\">\n <process id=\"myProcess_1\" processType=\"None\" isClosed=\"false\" isExecutable=\"true\">\n <startEvent id=\"StartEvent_15id6r0\">\n <outgoing>SequenceFlow_164i0wk</outgoing>\n </startEvent>\n <sequenceFlow id=\"SequenceFlow_164i0wk\" name=\"开始\" sourceRef=\"StartEvent_15id6r0\" targetRef=\"Task_1grs5dm\" />\n <sequenceFlow id=\"SequenceFlow_0ahu08r\" sourceRef=\"Task_1grs5dm\" targetRef=\"Task_07r8yxp\" />\n <endEvent id=\"EndEvent_04ra2c3\" name=\"结束\">\n <incoming>SequenceFlow_1htwxxf</incoming>\n </endEvent>\n <sequenceFlow id=\"SequenceFlow_1htwxxf\" sourceRef=\"Task_03fzwqc\" targetRef=\"EndEvent_04ra2c3\" />\n <manualTask id=\"Task_1grs5dm\" name=\"提交申请\">\n <incoming>SequenceFlow_164i0wk</incoming>\n <incoming>SequenceFlow_17vlcc6</incoming>\n <incoming>SequenceFlow_1eevx3a</incoming>\n <outgoing>SequenceFlow_0ahu08r</outgoing>\n </manualTask>\n <userTask id=\"Task_07r8yxp\" name=\"部门组长审批\" camunda:assignee=\"0003\">\n <incoming>SequenceFlow_0ahu08r</incoming>\n <outgoing>SequenceFlow_08w9avr</outgoing>\n <outgoing>SequenceFlow_17vlcc6</outgoing>\n </userTask>\n <userTask id=\"Task_03fzwqc\" name=\"部门经理审批\" camunda:assignee=\"0002\">\n <incoming>SequenceFlow_08w9avr</incoming>\n <outgoing>SequenceFlow_1htwxxf</outgoing>\n <outgoing>SequenceFlow_1eevx3a</outgoing>\n </userTask>\n <sequenceFlow id=\"SequenceFlow_08w9avr\" sourceRef=\"Task_07r8yxp\" targetRef=\"Task_03fzwqc\" />\n <sequenceFlow id=\"SequenceFlow_17vlcc6\" name=\"驳回\" sourceRef=\"Task_07r8yxp\" targetRef=\"Task_1grs5dm\">\n <conditionExpression xsi:type=\"tFormalExpression\">${outcome==\'驳回\'}</conditionExpression>\n </sequenceFlow>\n <sequenceFlow id=\"SequenceFlow_1eevx3a\" name=\"驳回\" sourceRef=\"Task_03fzwqc\" targetRef=\"Task_1grs5dm\">\n <conditionExpression xsi:type=\"tFormalExpression\">${outcome==\'驳回\'}</conditionExpression>\n </sequenceFlow>\n </process>\n <bpmndi:BPMNDiagram id=\"Diagram-_1\" name=\"New Diagram\" documentation=\"background=#FFFFFF;count=1;horizontalcount=1;orientation=0;width=842.4;height=1195.2;imageableWidth=832.4;imageableHeight=1185.2;imageableX=5.0;imageableY=5.0\">\n <bpmndi:BPMNPlane bpmnElement=\"myProcess_1\">\n <bpmndi:BPMNShape id=\"StartEvent_15id6r0_di\" bpmnElement=\"StartEvent_15id6r0\">\n <dc:Bounds x=\"242\" y=\"222\" width=\"36\" height=\"36\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_164i0wk_di\" bpmnElement=\"SequenceFlow_164i0wk\">\n <di:waypoint x=\"278\" y=\"240\" />\n <di:waypoint x=\"330\" y=\"240\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"294\" y=\"222\" width=\"22\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_0ahu08r_di\" bpmnElement=\"SequenceFlow_0ahu08r\">\n <di:waypoint x=\"430\" y=\"240\" />\n <di:waypoint x=\"490\" y=\"240\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"EndEvent_04ra2c3_di\" bpmnElement=\"EndEvent_04ra2c3\">\n <dc:Bounds x=\"812\" y=\"222\" width=\"36\" height=\"36\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"819\" y=\"265\" width=\"22\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_1htwxxf_di\" bpmnElement=\"SequenceFlow_1htwxxf\">\n <di:waypoint x=\"750\" y=\"240\" />\n <di:waypoint x=\"812\" y=\"240\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"ManualTask_1o8ldgc_di\" bpmnElement=\"Task_1grs5dm\">\n <dc:Bounds x=\"330\" y=\"200\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNShape id=\"UserTask_0td75u5_di\" bpmnElement=\"Task_07r8yxp\">\n <dc:Bounds x=\"490\" y=\"200\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNShape id=\"UserTask_08x6zwu_di\" bpmnElement=\"Task_03fzwqc\">\n <dc:Bounds x=\"650\" y=\"200\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_08w9avr_di\" bpmnElement=\"SequenceFlow_08w9avr\">\n <di:waypoint x=\"590\" y=\"240\" />\n <di:waypoint x=\"650\" y=\"240\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_17vlcc6_di\" bpmnElement=\"SequenceFlow_17vlcc6\">\n <di:waypoint x=\"540\" y=\"200\" />\n <di:waypoint x=\"540\" y=\"160\" />\n <di:waypoint x=\"380\" y=\"160\" />\n <di:waypoint x=\"380\" y=\"200\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"450\" y=\"142\" width=\"21\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_1eevx3a_di\" bpmnElement=\"SequenceFlow_1eevx3a\">\n <di:waypoint x=\"700\" y=\"280\" />\n <di:waypoint x=\"700\" y=\"330\" />\n <di:waypoint x=\"380\" y=\"330\" />\n <di:waypoint x=\"380\" y=\"280\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"531\" y=\"312\" width=\"21\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNEdge>\n </bpmndi:BPMNPlane>\n </bpmndi:BPMNDiagram>\n</definitions>\n', '1.0', NULL, '2019-12-06 17:41:36', '2019-12-06 17:41:36', 0, 0);
INSERT INTO `pms_models` VALUES (36, '离职流程', '', '<?xml version=\"1.0\" encoding=\"UTF-8\"?>\n<definitions xmlns=\"http://www.omg.org/spec/BPMN/20100524/MODEL\" xmlns:camunda=\"http://camunda.org/schema/1.0/bpmn\" xmlns:bpmndi=\"http://www.omg.org/spec/BPMN/20100524/DI\" xmlns:xsi=\"http://www.w3.org/2001/XMLSchema-instance\" xmlns:dc=\"http://www.omg.org/spec/DD/20100524/DC\" xmlns:di=\"http://www.omg.org/spec/DD/20100524/DI\" xmlns:activiti=\"http://activiti.org/bpmn\" xmlns:tns=\"http://www.activiti.org/testm1568796216967\" xmlns:xsd=\"http://www.w3.org/2001/XMLSchema\" id=\"m1568796216967\" name=\"\" targetNamespace=\"http://www.activiti.org/testm1568796216967\">\n <process id=\"myProcess_1\" processType=\"None\" isClosed=\"false\" isExecutable=\"true\">\n <startEvent id=\"StartEvent_15043e5\" name=\"开始\">\n <outgoing>SequenceFlow_0i7p1ba</outgoing>\n </startEvent>\n <sequenceFlow id=\"SequenceFlow_0i7p1ba\" sourceRef=\"StartEvent_15043e5\" targetRef=\"Task_0itd8rd\" />\n <sequenceFlow id=\"SequenceFlow_0xr1nsn\" sourceRef=\"Task_0itd8rd\" targetRef=\"Task_004ii5l\" />\n <sequenceFlow id=\"SequenceFlow_1xzszqp\" sourceRef=\"Task_004ii5l\" targetRef=\"Task_15de8m1\" />\n <endEvent id=\"EndEvent_0j8on5q\" name=\"结束\">\n <incoming>SequenceFlow_1hc35yu</incoming>\n </endEvent>\n <sequenceFlow id=\"SequenceFlow_1hc35yu\" sourceRef=\"Task_15de8m1\" targetRef=\"EndEvent_0j8on5q\" />\n <manualTask id=\"Task_0itd8rd\" name=\"提交申请\">\n <incoming>SequenceFlow_0i7p1ba</incoming>\n <outgoing>SequenceFlow_0xr1nsn</outgoing>\n </manualTask>\n <userTask id=\"Task_004ii5l\" name=\"部门经理审批\" camunda:assignee=\"0003\">\n <incoming>SequenceFlow_0xr1nsn</incoming>\n <outgoing>SequenceFlow_1xzszqp</outgoing>\n </userTask>\n <userTask id=\"Task_15de8m1\" name=\"总经理审批\" camunda:assignee=\"0002\">\n <incoming>SequenceFlow_1xzszqp</incoming>\n <outgoing>SequenceFlow_1hc35yu</outgoing>\n </userTask>\n </process>\n <bpmndi:BPMNDiagram id=\"Diagram-_1\" name=\"New Diagram\" documentation=\"background=#FFFFFF;count=1;horizontalcount=1;orientation=0;width=842.4;height=1195.2;imageableWidth=832.4;imageableHeight=1185.2;imageableX=5.0;imageableY=5.0\">\n <bpmndi:BPMNPlane bpmnElement=\"myProcess_1\">\n <bpmndi:BPMNShape id=\"StartEvent_15043e5_di\" bpmnElement=\"StartEvent_15043e5\">\n <dc:Bounds x=\"292\" y=\"242\" width=\"36\" height=\"36\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"299\" y=\"285\" width=\"23\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_0i7p1ba_di\" bpmnElement=\"SequenceFlow_0i7p1ba\">\n <di:waypoint x=\"328\" y=\"260\" />\n <di:waypoint x=\"380\" y=\"260\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_0xr1nsn_di\" bpmnElement=\"SequenceFlow_0xr1nsn\">\n <di:waypoint x=\"480\" y=\"260\" />\n <di:waypoint x=\"540\" y=\"260\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNEdge id=\"SequenceFlow_1xzszqp_di\" bpmnElement=\"SequenceFlow_1xzszqp\">\n <di:waypoint x=\"640\" y=\"260\" />\n <di:waypoint x=\"710\" y=\"260\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"EndEvent_0j8on5q_di\" bpmnElement=\"EndEvent_0j8on5q\">\n <dc:Bounds x=\"882\" y=\"242\" width=\"36\" height=\"36\" />\n <bpmndi:BPMNLabel>\n <dc:Bounds x=\"889\" y=\"285\" width=\"23\" height=\"14\" />\n </bpmndi:BPMNLabel>\n </bpmndi:BPMNShape>\n <bpmndi:BPMNEdge id=\"SequenceFlow_1hc35yu_di\" bpmnElement=\"SequenceFlow_1hc35yu\">\n <di:waypoint x=\"810\" y=\"260\" />\n <di:waypoint x=\"882\" y=\"260\" />\n </bpmndi:BPMNEdge>\n <bpmndi:BPMNShape id=\"ManualTask_13ynlj0_di\" bpmnElement=\"Task_0itd8rd\">\n <dc:Bounds x=\"380\" y=\"220\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNShape id=\"UserTask_11iqa8j_di\" bpmnElement=\"Task_004ii5l\">\n <dc:Bounds x=\"540\" y=\"220\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n <bpmndi:BPMNShape id=\"UserTask_0hds61y_di\" bpmnElement=\"Task_15de8m1\">\n <dc:Bounds x=\"710\" y=\"220\" width=\"100\" height=\"80\" />\n </bpmndi:BPMNShape>\n </bpmndi:BPMNPlane>\n </bpmndi:BPMNDiagram>\n</definitions>\n', '1.0', NULL, '2019-12-13 14:05:58', '2019-12-13 14:05:58', 0, 0);
SET FOREIGN_KEY_CHECKS = 1;
3.5 查看审批进度
3.6查看审批流程进度图工具类实现代码
/**
* 查看审批流程进度图
* @param processInstanceId 流程实例编号
* @return
*/
public InputStream images(String processInstanceId){
//processInstanceId
// String processInstanceId = taskService.createTaskQuery().taskId(taskId).singleResult().getProcessInstanceId();
//获取历史流程实例
HistoricProcessInstance processInstance = historyService.createHistoricProcessInstanceQuery().processInstanceId(processInstanceId).singleResult();
//获取流程图
BpmnModel bpmnModel = repositoryService.getBpmnModel(processInstance.getProcessDefinitionId());
ProcessEngineConfiguration processEngineConfiguration = processEngine.getProcessEngineConfiguration();
Context.setProcessEngineConfiguration((ProcessEngineConfigurationImpl) processEngineConfiguration);
ProcessDiagramGenerator diagramGenerator = processEngineConfiguration.getProcessDiagramGenerator();
ProcessDefinitionEntity definitionEntity = (ProcessDefinitionEntity)repositoryService.getProcessDefinition(processInstance.getProcessDefinitionId());
List<HistoricActivityInstance> highLightedActivitList = historyService.createHistoricActivityInstanceQuery().processInstanceId(processInstanceId).list();
//高亮环节id集合
List<String> highLightedActivitis = new ArrayList<String>();
//高亮线路id集合
List<String> highLightedFlows = getHighLightedFlows(definitionEntity,highLightedActivitList);
for(HistoricActivityInstance tempActivity : highLightedActivitList){
String activityId = tempActivity.getActivityId();
highLightedActivitis.add(activityId);
}
//中文显示的是乱码,设置字体就好了
InputStream imageStream = diagramGenerator.generateDiagram(bpmnModel, "png", highLightedActivitis,highLightedFlows,"宋体","宋体","宋体",null,1.0);
//单独返回流程图,不高亮显示
return imageStream;
}
/**
* 获取需要高亮的线
* @param processDefinitionEntity
* @param historicActivityInstances
* @return
*/
public List<String> getHighLightedFlows(
ProcessDefinitionEntity processDefinitionEntity,
List<HistoricActivityInstance> historicActivityInstances) {
List<String> highFlows = new ArrayList<String>();// 用以保存高亮的线flowId
for (int i = 0; i < historicActivityInstances.size() - 1; i++) {// 对历史流程节点进行遍历
ActivityImpl activityImpl = processDefinitionEntity
.findActivity(historicActivityInstances.get(i)
.getActivityId());// 得到节点定义的详细信息
List<ActivityImpl> sameStartTimeNodes = new ArrayList<ActivityImpl>();// 用以保存后需开始时间相同的节点
ActivityImpl sameActivityImpl1 = processDefinitionEntity
.findActivity(historicActivityInstances.get(i + 1)
.getActivityId());
// 将后面第一个节点放在时间相同节点的集合里
sameStartTimeNodes.add(sameActivityImpl1);
for (int j = i + 1; j < historicActivityInstances.size() - 1; j++) {
HistoricActivityInstance activityImpl1 = historicActivityInstances
.get(j);// 后续第一个节点
HistoricActivityInstance activityImpl2 = historicActivityInstances
.get(j + 1);// 后续第二个节点
if (activityImpl1.getStartTime().equals(
activityImpl2.getStartTime())) {
// 如果第一个节点和第二个节点开始时间相同保存
ActivityImpl sameActivityImpl2 = processDefinitionEntity
.findActivity(activityImpl2.getActivityId());
sameStartTimeNodes.add(sameActivityImpl2);
} else {
// 有不相同跳出循环
break;
}
}
List<PvmTransition> pvmTransitions = activityImpl
.getOutgoingTransitions();// 取出节点的所有出去的线
for (PvmTransition pvmTransition : pvmTransitions) {
// 对所有的线进行遍历
ActivityImpl pvmActivityImpl = (ActivityImpl) pvmTransition
.getDestination();
// 如果取出的线的目标节点存在时间相同的节点里,保存该线的id,进行高亮显示
if (sameStartTimeNodes.contains(pvmActivityImpl)) {
highFlows.add(pvmTransition.getId());
}
}
}
return highFlows;
}