vue.js

关注公众号 jb51net

关闭
首页 > 网络编程 > JavaScript > javascript类库 > vue.js > Vue 工作流

Vue+LogicFlow+Flowable实现工作流

作者:王八八。

本文主要介绍了Vue+LogicFlow+Flowable实现工作流,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

一、实现效果

前端使用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)

<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("失败");
        }
    }
}

到此这篇关于Vue+LogicFlow+Flowable实现工作流的文章就介绍到这了,更多相关Vue 工作流内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

您可能感兴趣的文章:
阅读全文