使用logicFlow完成一个简单的流程图

2,378 阅读10分钟

LogicFlow 是什么

LogicFlow 是一款流程图编辑框架,提供了一系列流程图交互、编辑所必需的功能和灵活的节点自定义、插件等拓展机制。LogicFlow 支持前端研发自定义开发各种逻辑编排场景,如流程图、ER 图、BPMN 流程等。在工作审批配置、机器人逻辑编排、无代码平台流程配置都有较好的应用。

logicFlow 文档地址: site.logic-flow.cn/docs/#/zh/g…

本人学习记录。不一定准确仅供参考。

1 要使用BPMNAdapter 而不是使用Bpmndapter 。转化xml的lf.adapterOut()方法有区别。
2 通过修改参数方法 setProperties() 对节点和线进行参数处理。使用固定字段,转化xml时会转成bpmn标准格式。

image.png

image.png

import引入

// 引入 logicFlow
import LogicFlow from '@logicflow/core'
import '@logicflow/core/dist/style/index.css';
import '@logicflow/extension/lib/style/index.css';
import { Control, Menu, DndPanel,Snapshot, BpmnElement, SelectionSelect,BPMNAdapter,BPMNElements ,BpmnXmlAdapter  } from '@logicflow/extension';
import {  lfXml2Json,lfJson2Xml } from "@logicflow/extension";

LogicFlow.use(Control); // 控制面板 (放大缩小之类的)
LogicFlow.use(Menu); // 右键菜单 (右键删除)
LogicFlow.use(DndPanel); // 拖拽面板
LogicFlow.use(Snapshot);// 导出图片
LogicFlow.use(BPMNElements);// 带s的  是符合B pmn2.0标准的新的
LogicFlow.use(SelectionSelect);
LogicFlow.use(BPMNAdapter); // 内置的数据转换工具 (大写!)

lf面板初始化

onMounted(() => {
    const lf = new LogicFlow({
        container: document.querySelector('#container'),
        grid:{size: 20, visible: true, type: "dot", config: {color: "#c6c6c6", thickness: 1,},},
    });
    lf?.setPatternItems(configDefaultDndPanel(lf));
    lf.render();
    // 监听node的点击事件
    lf.on("node:click", (val) => {
        // 记录参数
        nodeClickData.value = val.data;
        openNodeFormModal.value =  true;

    });
     // 监听空白处的点击事件
    lf.on("blank:click", (val) => {
      openNodeFormModal.value =  false;
    });
      // 监听线的点击事件
    lf.on("edge:click", ({ data }) => {
        edgeClickData.value = data;
    });
    // 颜色主题
    lf.setTheme(themeConfig);

    // 赋值。
    newLf.value = lf;
});

页面:

<template>
  <BasicModal defaultFullscreen  :title="title"  :forceRender="true"  :visible="visible" @ok="handleOk" :okButtonProps="{ class: { 'jee-hidden': disableSubmit } }" @cancel="handleCancel" cancelText="关闭">
<!--      操作栏按钮-->
    <div style="position: absolute;z-index: 90;left: 100px;width: 500px">
      <div class="lf-control">
        <div class="lf-control-item" @click="setLineProperties"><SaveOutlined class="lf-icon-zm" /><span class="lf-control-text" title="保存">setLineProperties</span></div>
        <div class="lf-control-item" @click="saveLf"><SaveOutlined class="lf-icon-zm" /><span class="lf-control-text" title="保存">保存</span></div>
        <div class="lf-control-item" @click="downloadImg"><PictureOutlined class="lf-icon-zm" /><span class="lf-control-text" title="下载图片">下载图片</span></div>
        <div class="lf-control-item" @click="showXml"><FileSearchOutlined class="lf-icon-zm"/><span class="lf-control-text" title="预览xml数据">预览xml</span></div>
        <div class="lf-control-item" @click="showJson"><FileSearchOutlined class="lf-icon-zm"/><span class="lf-control-text" title="预览json数据">预览json</span></div>
        <div class="lf-control-item" @click="clearLf"><ClearOutlined class="lf-icon-zm"/><span class="lf-control-text" title="清空画布">清空</span></div>
      </div>
    </div>
<!--    画布-->
    <div style="position: absolute" id="container"></div>
<!--    预览数据弹窗-->
    <BasicModal :visible="codeEditorVisible" title="预览数据" width="50%" @ok="editorHandleOk" @cancel="editorHandleCancel" >
      <CodeEditor :value="graphData"/>
    </BasicModal>
<!--    节点数据弹窗-->
    <nodeFormModal  :open="openNodeFormModal" @setNodeProperties="setNodeProperties"></nodeFormModal>
  </BasicModal>
</template>

方法:

// 根据node的id设置自定义参数 - 表单id绑定到开始节点
function setNodeProperties(formId) {
    // 例子 绑定表单id
    newLf.value.getNodeModelById(nodeClickData.value.id).setProperties({
        'flowable:formKey' : formId
    });
}

// 根据node的id设置自定义参数 - 箭头流转条件绑定
function setLineProperties() {
    newLf.value.getEdgeModelById(edgeClickData.value.id).setProperties({
        'properties': {expressionType: 'cdata',condition: 'foo &gt; bar',}
    });
}

// 根据node的id 删除参数
function deleteNodeProperty(nodeId,property) {
    newLf.value.getNodeModelById(nodeId).deleteProperty(property);
}
// 修改节点内容
function updateNodeText(nodeId,text) {
    const nodeModel = newLf.value.getNodeModelById(nodeId);
    nodeModel.updateText(text);
}

// 下载图片
function downloadImg() {
    newLf.value.getSnapshot();
}
// 导出xml
function exportXml() {
    return lfJson2Xml(newLf.value.getGraphData());
}
// 预览xml
function showXml() {
    graphData.value = newLf.value.adapterOut(newLf.value.getGraphRawData());
    codeEditorVisible.value = true;
}
// 预览 Json
function showJson() {
    graphData.value = newLf.value.getGraphData();
    codeEditorVisible.value = true;
}
// 清空画图图层
function clearLf() {
    newLf.value.clearData();
}

// 保存流程图数据到后台
function saveLf() {

}

配置文件

export function configDefaultDndPanel(lf) {
  return [
    {
      type: 'bpmn:startEvent',
      text: '开始',
      label: '开始',
      icon: '',
      callback: () => {

      },
    },
    {
      type: 'bpmn:userTask',
      text: '用户任务',
      label: '用户任务',
      icon: '',
      cls: 'important-node',
    },
    {
      type: 'bpmn:serviceTask',
      text: '服务任务',
      label: '服务任务',
      icon: '',
      cls: 'import_icon',
    },

    {
      type: 'bpmn:parallelGateway',
      text: '并行网关',
      label: '并行网关',
      icon: '',
    },
    {
      type: 'bpmn:exclusiveGateway',
      text: '排他网关',
      label: '排他网关',
      icon: '',
    },
    {
      type: 'bpmn:inclusiveGateway',
      text: '包容网关',
      label: '包容网关',
      icon: '',
    },
    {
      type: 'bpmn:endEvent',
      text: '结束',
      label: '结束',
      icon: '',
    },
  ];
}

export const themeConfig = { // 设置默认主题样式
  rect: { // 矩形样式
    fill: "#FFFFFF",// 白色填充
      stroke: "#71c8ff",// 边框,黑色 2 。
      strokeWidth: 2,
  },
  circle: { // 圆形样式
    fill: "#FFFFFF",// 白色填充
      stroke: "#a756ff",// 边框,黑色 2 。
      strokeWidth: 2,
  },
  polygon: { // 网关样式
    fill: "#FFFFFF",// 白色填充
      stroke: "#ff6264",// 边框,黑色 2 。
      strokeWidth: 2,
  },

  nodeText: { // 节点文本样式
  },
  edgeText: { // 边文本样式
  },
  anchor: { // 锚点样式
  },
  polyline: {
    stroke: "#4fad33",
      strokeWidth: 2,
  },
}