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&gt;1000}</conditionExpression>\n    </sequenceFlow>\n    <sequenceFlow id=\"SequenceFlow_01sftoz\" sourceRef=\"ExclusiveGateway_1716w5k\" targetRef=\"Task_15bkdk1\">\n      <conditionExpression xsi:type=\"tFormalExpression\">${condition&lt;=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;
    }
posted @ 2019-12-20 17:25  yiwanbin  阅读(10359)  评论(5编辑  收藏  举报