参考文档:LogicFlow 官方文档

# LogicFlow 是什么

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

# 安装 LogicFlow

通过 npmyarn 来安装 LogicFlow

l
# npm
$ npm install @logicflow/core --save
# yarn
$ yarn add @logicflow/core

安装完成之后,使用 importrequire 进行引用。

s
import LogicFlow from '@logicflow/core';
import '@logicflow/core/dist/style/index.css';

# 快速上手

# 创建容器

在页面中创建一个用于绘图的容器,可以是一个 div 标签。

s
<div id="container"></div>

# 准备数据

通过 JSON 的数据格式,来让 LogicFlow 渲染。该数据中需要有 nodes (节点) 和 edges (边) 字段,分别用数组表示:

s
const data = {
  // 节点
  nodes: [
    {
      id: 50,
      type: 'rect',
      x: 100,
      y: 150,
      text: '你好',
    },
    {
      id: 21,
      type: 'circle',
      x: 300,
      y: 150,
    },
  ],
  // 边
  edges: [
    {
      type: 'polyline',
      sourceNodeId: 50,
      targetNodeId: 21,
    },
  ],
};

# 渲染画布

首先创建一个 LogicFlow 的实例,此时可以传入一些参数来控制画布,比如画布的大小。

s
const lf = new LogicFlow({
  container: document.querySelector('#container'),
  stopScrollGraph: true,
  stopZoomGraph: true,
  width: 500,
  height: 500,
  grid: {
    type: 'dot',
    size: 20,
  },
});

通过刚才创建的实例,来渲染画布。

s
lf.render(data);

到此,我们就创建好了一个最简单的示例。

# 基本使用

LogicFlow 的功能分为核心功能和插件功能。LogicFlow 最初的目标就是支持一个扩展性强的流程绘制工具,用来满足各种业务需求。为了让 LogicFlow 的拓展性足够强,LogicFlow 将所有的非核心功能都使用插件的方式开发,然后将这些插件放到 @logicflow/extension 包中。

核心功能:节点、边、主题、事件、网格、背景、拖拽创建节点、键盘快捷键、对齐线、设置图编辑方式。

插件功能: 菜单、拖拽面板、控制面板、小地图、框选、节点缩放、分组、圆角折线、边上插入节点、数据转换、BPMN 元素、导出、自定义插件。

文档参见:https://site.logic-flow.cn/docs/#/zh/guide/basic/node

# Demo

main.js

t
import  './assets/main.css'
  
import { createApp } from  'vue'
import  App  from  './App.vue'
  
createApp(App).mount('#app')

App.vue

s
<template>
 <Lf />
</template>
  
<script>
import  Lf  from  "./components/Lf.vue";
  
export  default {
 name:  "App",
 components: {
 Lf,
  },
};
</script>
  
<style>
html,
body {
 width: 100%;
 height: 100%;
}
#app {
 font-family: "Avenir", Helvetica, Arial, sans-serif;
 -webkit-font-smoothing: antialiased;
 -moz-osx-font-smoothing: grayscale;
 text-align: center;
 color: #2c3e50;
 margin-top: 60px;
 width: 100%;
 height: 100%;
}
</style>

Lf.vue

s
<template>
 <div  class="container"  ref="container"></div>
</template>
<script>
import  LogicFlow  from  "@logicflow/core";
import { Menu,DndPanel, SelectionSelect} from  "@logicflow/extension";
import  "@logicflow/core/dist/style/index.css";
import  '@logicflow/extension/lib/style/index.css'
import  UserTask  from  "../common/UserTask";
  
export  default {
 mounted() {
 this.lf = new  LogicFlow({
 container:  this.$refs.container,
 grid:  true, // 是否开启网格背景
 keyboard: {
 enabled:  true, //cv 回退 快捷键
            },
 isSilentMode:  false, // 仅浏览不可编辑模式,默认不开启
 snapline:  false, // 关闭对齐线功能
 plugins: [DndPanel, SelectionSelect,Menu],
        });
 this.lf.register(UserTask);
 this.lf.render({
 nodes: [
                {
 id:  "1",
 type:  "rect",
 x:  150,
 y:  100,
                },
                {
 id:  "2",
 type:  "circle",
 x:  350,
 y:  200,
                },
                {
 id:  "3",
 type:  "diamond",
 x:  500,
 y:  300
                },
  
            ],
 edges: [
                {
 sourceNodeId:  "1",
 targetNodeId:  "2",
 type:  "polyline",
                },
                {
 sourceNodeId:  "2",
 targetNodeId:  "3",
 type:  "polyline",
                },
            ],
        });
 // 拖拽面板
 this.lf.extension.dndPanel.setPatternItems([
            {
 label:  '选区',
 icon:  '',
 callback: () => {
 this.lf.extension.selectionSelect.openSelectionSelect();
 this.lf.once('selection:selected', () => {
 this.lf.extension.selectionSelect.closeSelectionSelect();
                    });
                }
            },
            {
 type:  'circle',
 text:  '开始',
 label:  '开始节点',
 icon:  '',
            },
            {
 type:  'UserTask',
 label:  '用户任务',
 icon:  '',
 className:  'important-node'
            },
            {
 type:  'rect',
 label:  '系统任务',
 icon:  '',
 className:  'import_icon'
            },
            {
 type:  'diamond',
 label:  '条件判断',
 icon:  '',
            },
            {
 type:  'circle',
 text:  '结束',
 label:  '结束节点',
 icon:  '',
            }
        ]);
    },
};
</script>
<style  scoped>
.container {
 width: 1000px;
 height: 600px;
}
  
</style>

common/UserTask.js

t
import { RectNode, RectNodeModel, h } from  "@logicflow/core";
  
class  UserTaskView  extends  RectNode {
 getLabelShape() {
 const { model } = this.props;
 const { x, y, width, height } = model;
 const  style = model.getNodeStyle();
 return  h(
 "svg",
      {
 x:  x - width / 2 + 5,
 y:  y - height / 2 + 5,
 width:  25,
 height:  25,
 viewBox:  "0 0 1274 1024"
      },
      [
 h("path", {
 fill:  style.stroke,
 d:
 "M690.366075 350.568358c0-98.876614-79.937349-179.048571-178.558027-179.048571-98.59935 0-178.515371 80.150629-178.515371 179.048571 0 98.833958 79.916021 178.963259 178.515371 178.963259C610.428726 529.531617 690.366075 449.380988 690.366075 350.568358M376.140632 350.568358c0-75.159877 60.72082-136.072649 135.667416-136.072649 74.989253 0 135.667416 60.912772 135.667416 136.072649 0 75.117221-60.678164 136.029993-135.667416 136.029993C436.861451 486.577022 376.140632 425.664251 376.140632 350.568358M197.284012 762.923936 197.284012 778.472049l15.526785 0 291.255186 0.127968L819.784387 778.472049l15.569441 0 0-15.548113c0-139.783721-136.413897-285.581938-311.026243-273.275681-10.002833 0.703824-24.740482 9.128385-34.658002 9.938849-8.573857 0.74648 13.692577 8.232609 14.396401 16.827793 9.021745-0.789136 6.313088 13.095393 15.505457 13.095393 150.597017 0 263.14488 103.07823 263.14488 224.62651l15.441473-15.590769-285.816546-0.042656-278.991585 1.81288 15.526785 15.612097c0-82.752645 75.095893-152.70849 136.861785-191.824044 7.25152-4.58552 8.659169-17.659585 4.862784-22.906273-6.846288-9.426977-19.877697-8.701825-28.046322-6.014496C285.262018 560.521203 197.284012 667.758394 197.284012 762.923936"
        }),
 h("path", {
 fill:  style.stroke,
 d:
 "M512.31992 1.535616c-282.766642 0-512.021328 228.89211-512.021328 511.210864 0 282.46805 229.254686 511.25352 512.021328 511.25352 117.431975 0 228.828126-39.606098 318.810964-111.204199 10.791969-8.488545 12.540865-24.22861 3.988336-34.99925-8.616513-10.770641-24.356578-12.540865-35.127218-3.94568-81.174373 64.538532-181.586603 100.241606-287.650754 100.241606-255.210864 0-462.028493-206.561693-462.028493-461.367325 0-254.762976 206.817629-461.303341 462.028493-461.303341 255.210864 0 462.092477 206.561693 462.092477 461.303341 0 87.380821-24.33525 171.093227-69.614596 243.651087-7.272848 11.645089-3.668416 27.086562 8.040657 34.35941 11.709073 7.272848 27.10789 3.62576 34.402066-7.976672 50.184787-80.406565 77.143381-173.247355 77.143381-270.055153C1024.383904 230.427726 795.10789 1.535616 512.31992 1.535616z"
        })
      ]
    );
  }
 getShape() {
 const { model } = this.props;
 const { x, y, width, height, radius } = model;
 const  style = model.getNodeStyle();
 return  h("g", {}, [
 h("rect", {
        ...style,
 x:  x - width / 2,
 y:  y - height / 2,
 rx:  radius,
 ry:  radius,
 width,
 height
      }),
 this.getLabelShape()
    ]);
  }
}
  
class  UserTaskModel  extends  RectNodeModel {
 setAttributes() {
 this.width = 120;
 this.height = 80;
  }
 getTextStyle() {
 const  style = super.getTextStyle();
 style.fontSize = 12;
 const  properties = this.properties;
 style.color = properties.disabled ? "red" : "rgb(24, 125, 255)";
 return  style;
  }
 getNodeStyle() {
 const  style = super.getNodeStyle();
 const  properties = this.properties;
 if (properties.disabled) {
 style.stroke = "red";
} else {
 style.stroke = "rgb(24, 125, 255)";
    }
 return  style;
  }
 getAnchorStyle() {
 const  style = super.getAnchorStyle();
 style.stroke = "rgb(24, 125, 255)";
 style.r = 3;
 style.hover.r = 8;
 style.hover.fill = "rgb(24, 125, 255)";
 style.hover.stroke = "rgb(24, 125, 255)";
 return  style;
  }
 getAnchorLineStyle() {
 const  style = super.getAnchorLineStyle();
 style.stroke = "rgb(24, 125, 255)";
 return  style;
  }
 getOutlineStyle() {
 const  style = super.getOutlineStyle();
 style.stroke = "red";
 style.hover.stroke = "red";
 return  style;
  }
}
  
export  default {
 type:  "UserTask",
 view:  UserTaskView,
 model:  UserTaskModel
};

# 实现效果