vue整合logicflow

LogicFlow是什么

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

特性

  • 可视化模型:通过 LogicFlow 提供的直观可视化界面,用户可以轻松创建、编辑和管理复杂的逻辑流程图。
  • 高可定制性:用户可以根据自己的需要定制节点、连接器和样式,创建符合特定用例的定制逻辑流程图。
  • 自执行引擎:执行引擎支持浏览器端执行流程图逻辑,为无代码执行提供新思路。

 创建logicflow的vue项目

创建vue项目

vue create mylogicflow

我选择vue2

 

idea打开项目 

执行安装依赖命令

npm install

执行启动命令

npm run serve

打开如上页面证明vue项目创建成功

安装logicflow依赖

npm install @logicflow/core --save

 插件包(不使用插件时不需要引入)推荐引入

npm install @logicflow/extension --save

创建draw.vue

创建pages文件夹,并在pages下面创建draw.vue

内容如下

<template>
  <div class="container" ref="container"></div>
</template>

<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
// import "@logicflow/core/dist/style/index.css"; // 2.0版本前的引入方式

export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  mounted() {
    this.lf = new LogicFlow({
      container: this.$refs.container,
      grid: true,
    });
    this.lf.render();
  },
};
</script>

<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}
</style>

安装router

安装router

npm install vue-router@3.1.3 --force

配置 router

src 文件夹下创建一个 router 文件夹,并在其中创建一个 index.js 文件:

在index.js中添加如下代码

// 引入Vue核心库
import Vue from 'vue'
// 引入vue-router
import VueRouter from 'vue-router'
// 应用vue-router插件
Vue.use(VueRouter)

// 引入组件(方式一)
import Home from '../pages/draw.vue'

// 配置路由
const routes = [
    {
        // 路径,/ 表示首页
        path: '/',
        // 使用组件(方式一)
        component: Home
    }
    // ,
    // {
    //     // 路径
    //     path: '/about',
    //     // 引入并使用组件(方式二)
    //     component: () => import('../views/About.vue')
    // }
]

// 创建路由器
const router = new VueRouter({
    routes
})

// 导出路由器
export default router

在 main.js 中引入并使用 Vue Router

import Vue from 'vue'
import App from './App.vue'
import router from './router';  // 引入 router 配置
Vue.config.productionTip = false

new Vue({
  router,
  render: h => h(App),
}).$mount('#app')

修改app.vue

<template>
  <router-view></router-view>
</template>

<script>
export default {
  name: 'App',
  components: {
  }
}
</script>

<style>
#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;
}
</style>

安装element ui

npm install element-ui@2.15.7

 在mian.js中使用

import Vue from 'vue'
import App from './App.vue'
import router from './router';  // 引入 router 配置
Vue.config.productionTip = false
import Element from 'element-ui';
import 'element-ui/lib/theme-chalk/index.css';
Vue.use(Element)
new Vue({
  router,
  render: h => h(App),
}).$mount('#app')

启动项目

logicflow实操

添加控件(render之前)

系统默认控件

<template>
  <div class="container" ref="container"></div>
</template>

<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  mounted() {
    this.lf = new LogicFlow({
      container: this.$refs.container,
      plugins: [Control],
      grid: true,
    });
    this.lf.render();
  },
};
</script>

<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}
</style>

为什么会有上面默认控件,查看源码

node_modules/@logicflow/extension/src/components/control/index.ts

自定义控件 

<template>
  <el-container>
     <div class="container" ref="container"></div>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  mounted() {
    this.lf = new LogicFlow({
      container: this.$refs.container,
      plugins: [Control],
      grid: true,
    });
    this.init()
  },
  methods: {
    init(){//el-icon-delete是element图标 icon-delete是自定义属性
      this.lf.extension.control.addItem({
        key: 'clear',
        iconClass: 'el-icon-delete icon-delete',
        title: "清空画板",
        text: "清空画板",
        onClick: () => {
          console.log('清空画板')
        }
      })
      this.lf.render();
    }
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}
.icon-delete {
  font-size: 30px;
}
</style>

删除控件 (render之前)

<template>
  <el-container>
     <div class="container" ref="container"></div>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  mounted() {
    this.lf = new LogicFlow({
      container: this.$refs.container,
      plugins: [Control],
      grid: true,
    });
    this.init()
  },
  methods: {
    init(){//el-icon-delete是element图标 icon-delete是自定义属性
      this.lf.extension.control.addItem({
        key: 'clear',
        iconClass: 'el-icon-delete icon-delete',
        title: "清空画板",
        text: "清空画板",
        onClick: () => {
          console.log('清空画板')
        }
      })
      //删除缩小按钮
      this.lf.extension.control.removeItem('zoom-out')
      this.lf.render();
    }
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}
.icon-delete {
  font-size: 30px;
}
</style>

 添加拖拽面板

<template>
  <el-container>
     <div class="container" ref="container"></div>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control,DndPanel, SelectionSelect } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  mounted() {
    this.lf = new LogicFlow({
      container: this.$refs.container,
      plugins: [Control,DndPanel, SelectionSelect],
      grid: true,
    });
    this.init()
  },
  methods: {
    init(){
      var lf = this.lf
      lf.extension.dndPanel.setPatternItems([
        {
          label: '选区',
          icon: '',
          callback: () => {
            lf.extension.selectionSelect.openSelectionSelect();
            lf.once('selection:selected', () => {
              lf.extension.selectionSelect.closeSelectionSelect();
            });
          }
        },
        {
          type: 'circle',
          text: '开始',
          label: '开始节点',
          icon: '',
        },
        {
          type: 'rect',
          label: '用户任务',
          icon: '',
          className: 'important-node'
        },
        {
          type: 'rect',
          label: '系统任务',
          icon: '',
          className: 'import_icon'
        },
        {
          type: 'diamond',
          label: '条件判断',
          icon: '',
        },
        {
          type: 'circle',
          text: '结束',
          label: '结束节点',
          icon: '',
        }
      ]);
      
      this.lf.render();
    },
    
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}

</style>

 自定义拖拽节点

自定义五个节点

CustomAnd.js

import {HtmlNode, HtmlNodeModel} from '@logicflow/core'
//自定义and节点
// 提供节点 svg dom
class CustomAndNode extends HtmlNode {
  constructor(props){
    props.model.text.y = props.model.y + 20
    super(props)
  }
  setHtml(rootEl) {
    const { properties } = this.props.model;

    const el = document.createElement("div");
    el.className = "uml-wrapper";
    el.innerHTML = `
    <img width="58" src='${properties.status === 'COMPLETED' ? require('./img/and1.png') : (properties.status === 'FAILED' ? require('./img/and0.png') : require('./img/and.png'))}' alt=""/>
    `;
    rootEl.innerHTML = "";
    rootEl.appendChild(el);
  }
}

// 提供节点的样式
class CustomAndModel extends HtmlNodeModel {
  setAttributes() {
    const width = 58;
    const height = 80;
    this.width = width;
    this.height = height;
    this.anchorsOffset = [
      [width / 2, -15],
      [0, height / 2 - 30],
      [-width / 2, -15],
      [0, -height / 2],
    ];
  }
}

export default {
  type: 'CustomAnd',
  view: CustomAndNode,
  model: CustomAndModel
}

 CustomBusiness.js

import {HtmlNode, HtmlNodeModel} from '@logicflow/core'
//自定义业务节点
// 提供节点
class CustomBusinessNode extends HtmlNode {
  constructor(props){
    props.model.text.y = props.model.y + 20
    super(props)
  }
    
  setHtml(rootEl) {
    const { properties } = this.props.model;

    const el = document.createElement("div");
    el.className = "uml-wrapper";
    el.innerHTML = `
    <img width="58" src='${properties.status === 'COMPLETED' ? require('./img/node1.png') : (properties.status === 'FAILED' ? require('./img/node0.png') : require('./img/node.png'))}'  alt=""/>
    `;
    rootEl.innerHTML = "";
    rootEl.appendChild(el);
  }
}

// 提供节点的属性
class CustomBusinessModel extends HtmlNodeModel {
  setAttributes() {
    const width = 58;
    const height = 80;
    this.width = width;
    this.height = height;
    this.anchorsOffset = [
      [width / 2, -15],
      [0, height / 2 - 30],
      [-width / 2, -15],
      [0, -height / 2],
    ];
    // 业务节点文本不可编辑
    this.text.editable=false
  }
}

export default {
  type: 'CustomBusiness',
  view: CustomBusinessNode,
  model: CustomBusinessModel
}

CustomEnd.js

// CustomRectNode.js
import {HtmlNode, HtmlNodeModel} from '@logicflow/core'
//自定义结束节点
// 提供节点 svg dom
class CustomEndNode extends HtmlNode {
  constructor(props){
    props.model.text.y = props.model.y + 20
    super(props)
  }
    
  setHtml(rootEl) {
    const { properties } = this.props.model;
    const el = document.createElement("div");
    el.className = "uml-wrapper";
    el.innerHTML = `
    <img width="58" src='${properties.status === 'COMPLETED' ? require('./img/end1.png') : (properties.status === 'FAILED' ? require('./img/end0.png') : require('./img/end.png'))}' alt=""/>
    `;
    rootEl.innerHTML = "";
    rootEl.appendChild(el);
  }
}

// 提供节点的样式
class CustomEndModel extends HtmlNodeModel {
  setAttributes() {
    const width = 58;
    const height = 80;
    this.width = width;
    this.height = height;
    this.anchorsOffset = [
      [width / 2, -15],
      [0, height / 2 - 30],
      [-width / 2, -15],
      [0, -height / 2],
    ];
  }
}

export default {
  type: 'CustomEnd',
  view: CustomEndNode,
  model: CustomEndModel
}

CustomIf.js

import {HtmlNode, HtmlNodeModel} from '@logicflow/core'
//自定义条件节点
// 提供节点
class CustomIfNode extends HtmlNode {
  constructor(props){
    props.model.text.y = props.model.y + 20
    super(props)
  }
    
  setHtml(rootEl) {
    const { properties } = this.props.model;
    const el = document.createElement("div");
    el.className = "uml-wrapper";
    el.innerHTML = `
    <img width="58" src='${properties.status === 'COMPLETED' ? require('./img/if1.png') : (properties.status === 'FAILED' ? require('./img/if0.png') : require('./img/if.png'))}' alt=""/>
    `;
    rootEl.innerHTML = "";
    rootEl.appendChild(el);
  }
}

// 提供节点的属性
class CustomIfModel extends HtmlNodeModel {
  setAttributes() {
    const width = 58;
    const height = 80;
    this.width = width;
    this.height = height;
    this.anchorsOffset = [
      [width / 2, -15],
      [0, height / 2 - 30],
      [-width / 2, -15],
      [0, -height / 2],
    ];
  }
}

export default {
  type: 'CustomIf',
  view: CustomIfNode,
  model: CustomIfModel
}

CustomStart.js

import {HtmlNode, HtmlNodeModel} from '@logicflow/core'
//自定义开始节点
// 提供节点 svg dom
class CustomStartNode extends HtmlNode {
  constructor(props){
    props.model.text.y = props.model.y + 20
    super(props)
  }
  setHtml(rootEl) {
    const { properties } = this.props.model;
    const el = document.createElement("div");
    el.className = "start";
    el.innerHTML = `
    <img width="58" src='${properties.status === 'COMPLETED' ? require('./img/start1.png') : (properties.status === 'FAILED' ? require('./img/start0.png') : require('./img/start.png'))}' alt=""/>
    `;
    rootEl.innerHTML = "";
    rootEl.appendChild(el);
  }
}


class CustomStartModel extends HtmlNodeModel {
  setAttributes() {
    const width = 58;
    const height = 80;
    this.width = width;
    this.height = height;
    this.anchorsOffset = [
      [width / 2, -15],
      [0, height / 2 - 30],
      [-width / 2, -15],
      [0, -height / 2],
    ];
  }
}

export default {
  type: 'CustomStart',
  view: CustomStartNode,
  model: CustomStartModel
}

把自定义节点存放到index.js

import CustomStart from './CustomStart'
import CustomEnd from './CustomEnd'
import CustomAnd from './CustomAnd'
import CustomIf from './CustomIf'
import CustomBusiness from './CustomBusiness'
const Nodes = {
    CustomStart,
    CustomEnd,
    CustomAnd,
    CustomIf,
    CustomBusiness,
}
export default Nodes

自定义面板存放节点

CustomNodePanel.vue

<template>
  <div class="node-panel" v-if="showPanel">
    <div class="node-item" @mousedown="addStartNode()" title="开始节点只能有一个出口,不能有入口">
      <img src="../registerNode/img/start.png" width="48px" :draggable="false" alt="" >
      <span class="node-label">开始</span>
    </div>
    <div class="node-item" @mousedown="addBusinessNode()" title="业务节点至少有一个入口,至少有一个出口">
      <img src="../registerNode/img/node.png" width="48px" :draggable="false" alt="" >
      <span class="node-label">业务能力</span>
    </div>
    <div class="node-item" @mousedown="addIfNode()" title="分支条件节点只能有一个入口,可以有多个出口">
      <img src="../registerNode/img/if.png" width="48px" :draggable="false" alt="" >
      <span class="node-label">分支条件</span>
    </div>
    <div class="node-item" @mousedown="addAndNode()" title="并行汇聚节点只能有一个出口,可以有多个出口">
      <img src="../registerNode/img/and.png" width="48px" :draggable="false" alt="" >
      <span class="node-label">并行汇聚</span>
    </div>
    <div class="node-item" @mousedown="addEndNode()" title="结束节点至少有一个入口,不能有出口">
      <img src="../registerNode/img/end.png" width="48px" :draggable="false" alt="" >
      <span class="node-label">结束</span>
    </div>
  </div>
</template>
<script>
// 注册节点
import customNodeTypes from '../registerNode/index'
import CustomPolyline from '../registerLine/CustomPolyline'

export default {
  name: 'CustomNodePanel',
  data () {
    return {
      lf: null,
    }
  },
  props: {
    graphData: Object,
    showPanel: Boolean
  },
  inject: ['drawThis'],
  mounted () {
    // 注册节点
    this.registerNode()
    if (this.graphData) {
      this.lf.renderRawData(this.graphData)
    }
  },
  methods: {
    registerNode () {
      //创建画板
      this.drawThis.lf = this.drawThis.init()
      //注册自定义节点类型
      this.drawThis.lf.register(customNodeTypes.CustomAnd)
      this.drawThis.lf.register(customNodeTypes.CustomEnd)
      this.drawThis.lf.register(customNodeTypes.CustomBusiness)
      this.drawThis.lf.register(customNodeTypes.CustomIf)
      this.drawThis.lf.register(customNodeTypes.CustomStart)
      this.drawThis.lf.register(CustomPolyline)
      this.lf =this.drawThis.lf
      this.drawThis.refreshGraph()
    },
    addStartNode () {
      (this.lf).dnd.startDrag({
        type: 'CustomStart',
        text: '开始',
        properties: { type: 'start' } 
      })
    },
    addBusinessNode () {
      (this.lf).dnd.startDrag({
        type: 'CustomBusiness',
        text: '业务能力',
        properties: { type: 'business' } 
      })
    },
    addIfNode () {
      (this.lf).dnd.startDrag({
        type: 'CustomIf',
        text: '条件',
        properties: { type: 'if' }
      })
    },
    addEndNode () {
      (this.lf).dnd.startDrag({
        type: 'CustomEnd',
        text: '结束',
        properties: { type: 'end', referenceName:'end' }
      })
    },
    addAndNode () {
      (this.lf).dnd.startDrag({
        type: 'CustomAnd',
        text: '合并',
        properties: { type: 'and' }
      })
    }
  }
}
</script>
<style>
.node-panel {
  position: absolute;
  top: 54px;
  left: 24px;
  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-label {
  font-size: 12px;
  margin-top: 5px;
  user-select: none;
}
.lf-html-wrapper .custom-html {
  width:96px;
  height:96px;
  border:2px solid #2987ff;
}
.lf-html-wrapper .custom-head {
  color:red;
  height:26px;
  line-height:26px;
}
.lf-html-wrapper .custom-body {
  color:#2987ff;
  height:50px;
  line-height:50px;
}
</style>

使用自定义节点

<template>
  <el-container>
     <div class="container" ref="container"></div>
    <CustomNodePanel :graphData="graphData" :showPanel="showPanel" ref="drawThis"></CustomNodePanel>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control,DndPanel, SelectionSelect } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
import CustomNodePanel from "./components/CustomNodePanel.vue";
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  components: { CustomNodePanel },
  provide() {
    return {
      drawThis: this
    }
  },
  data() {
    return {
      lf: null,
      graphData: {

      },
      showPanel: true
    }
  },

  methods: {
    init(){
      return  new LogicFlow({
        container: this.$refs.container,
        plugins: [Control,DndPanel, SelectionSelect],
        grid: true,
      });
    },
    refreshGraph(){
      this.lf.render();
    }
    
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}

</style>

系统线条 

系统自带的类型

// 直线
import { LineEdge, PolylineEdgeModel } from "@logicflow/core";
// 折线
import { PolylineEdge, PolylineEdgeModel } from "@logicflow/core";
// 贝塞尔曲线
import { BezierEdge, BezierEdgeModel } from "@logicflow/core";

对应类型名称

export type EdgeType = 'line' | 'polyline' | 'bezier' | string

使用系统线条

<template>
  <el-container>
     <div class="container" ref="container"></div>
    <CustomNodePanel :graphData="graphData" :showPanel="showPanel" ref="drawThis"></CustomNodePanel>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control,DndPanel, SelectionSelect } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
import CustomNodePanel from "./components/CustomNodePanel.vue";
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  components: { CustomNodePanel },
  provide() {
    return {
      drawThis: this
    }
  },
  data() {
    return {
      lf: null,
      graphData: {

      },
      showPanel: true
    }
  },

  methods: {
    init(){
      return  new LogicFlow({
        container: this.$refs.container,
        plugins: [Control,DndPanel, SelectionSelect],
        grid: true,
        edgeType: 'bezier'
      });
    },
    refreshGraph(){
      this.lf.render();
    }
    
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}

</style>

自定义线条 

CustomPolyline.js

import {PolylineEdge, PolylineEdgeModel} from "@logicflow/core";

class CustomPolylineModel extends PolylineEdgeModel {
    setAttributes() {
        this.offset = 20;
        this.text.editable=false
    }
    //设置线样式
    getEdgeStyle() {
        const style = super.getEdgeStyle();
        const { properties } = this;
        if (properties.isActived) {
            style.strokeDasharray = "4 4";
        }
        style.stroke = "#d51b1b";
        return style;
    }
    //设置文字样色
    getTextStyle() {
        //设置文字样色
        const style = super.getTextStyle();
        // style.color = "#3451F1";
        // style.fontSize = 30;
        // style.background.fill = "#F2F131";
        // console.log(style)
        return style;
    }
    //鼠标悬浮时样式
    getOutlineStyle() {
        const style = super.getOutlineStyle();
        // style.stroke = "";
        // style.hover.stroke = "red";
        return style;
    }
}

export default {
    type: "CustomPolyline",
    view: PolylineEdge,
    model: CustomPolylineModel,
};

注册

import CustomPolyline from '../registerLine/CustomPolyline'
this.drawThis.lf.register(CustomPolyline)

使用

<template>
  <el-container>
     <div class="container" ref="container"></div>
    <CustomNodePanel :graphData="graphData" :showPanel="showPanel" ref="drawThis"></CustomNodePanel>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control,DndPanel, SelectionSelect } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
import CustomNodePanel from "./components/CustomNodePanel.vue";
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  components: { CustomNodePanel },
  provide() {
    return {
      drawThis: this
    }
  },
  data() {
    return {
      lf: null,
      graphData: {

      },
      showPanel: true
    }
  },

  methods: {
    init(){
      return  new LogicFlow({
        container: this.$refs.container,
        plugins: [Control,DndPanel, SelectionSelect],
        grid: true,
        edgeType: 'CustomPolyline'
      });
    },
    refreshGraph(){
      this.lf.render();
    }
    
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}

</style>

菜单 

import LogicFlow from "@logicflow/core";
import { Menu } from "@logicflow/extension";
import "@logicflow/extension/lib/style/index.css";

LogicFlow.use(Menu);

添加默认菜单

<template>
  <el-container>
     <div class="container" ref="container"></div>
    <CustomNodePanel :graphData="graphData" :showPanel="showPanel" ref="drawThis"></CustomNodePanel>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control,DndPanel, SelectionSelect,Menu  } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
import CustomNodePanel from "./components/CustomNodePanel.vue";
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  components: { CustomNodePanel },
  provide() {
    return {
      drawThis: this
    }
  },
  data() {
    return {
      lf: null,
      graphData: {

      },
      showPanel: true
    }
  },

  methods: {
    init(){
      return  new LogicFlow({
        container: this.$refs.container,
        plugins: [Control,DndPanel, SelectionSelect,Menu ],
        grid: true,
        edgeType: 'CustomPolyline'
      });
    },
    refreshGraph(){
      this.lf.render();
    }
    
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}

</style>

添加自定义菜单 

<template>
  <el-container>
     <div class="container" ref="container"></div>
    <CustomNodePanel :graphData="graphData" :showPanel="showPanel" ref="drawThis"></CustomNodePanel>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control,DndPanel, SelectionSelect,Menu  } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
import CustomNodePanel from "./components/CustomNodePanel.vue";
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  components: { CustomNodePanel },
  provide() {
    return {
      drawThis: this
    }
  },
  data() {
    return {
      lf: null,
      graphData: {

      },
      showPanel: true
    }
  },

  methods: {
    init(){
      return  new LogicFlow({
        container: this.$refs.container,
        plugins: [Control,DndPanel, SelectionSelect,Menu ],
        grid: true,
        edgeType: 'CustomPolyline'
      });
    },
    refreshGraph(){
      this.addMenu()
      this.lf.render();
    },
    addMenu(){
      this.lf.extension.menu.addMenuConfig({
        nodeMenu: [
          {
            text: "分享",
            callback() {
              alert("分享成功!");
            },
          },
          {
            text: "属性",
            callback(node) {
              alert(`
          节点id:${node.id}
          节点类型:${node.type}
          节点坐标:(x: ${node.x}, y: ${node.y})`);
            },
          },
        ],
        edgeMenu: [
          {
            text: "属性",
            callback(edge) {
              alert(`
          边id:${edge.id}
          边类型:${edge.type}
          边坐标:(x: ${edge.x}, y: ${edge.y})
          源节点id:${edge.sourceNodeId}
          目标节点id:${edge.targetNodeId}`);
            },
          },
        ],
        graphMenu: [
          {
            text: "分享",
            callback() {
              alert("分享成功!");
            },
          },
        ],
      });
    }
    
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}

</style>

事件 

事件列表

添加事件

<template>
  <el-container>
     <div class="container" ref="container"></div>
    <CustomNodePanel :graphData="graphData" :showPanel="showPanel" ref="drawThis"></CustomNodePanel>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control,DndPanel, SelectionSelect,Menu  } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
import CustomNodePanel from "./components/CustomNodePanel.vue";
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  components: { CustomNodePanel },
  provide() {
    return {
      drawThis: this
    }
  },
  data() {
    return {
      lf: null,
      graphData: {

      },
      showPanel: true
    }
  },
  methods: {
    init(){
      return  new LogicFlow({
        container: this.$refs.container,
        plugins: [Control,DndPanel, SelectionSelect,Menu ],
        grid: true,
        edgeType: 'CustomPolyline'
      });
    },
    refreshGraph(){
      this.addEvent()
      this.lf.render();
    },
    addEvent(){
      this.lf.on("node:dnd-add", (data) => {
        console.log(data)
      });
    }
  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}

</style>

线条鼠标悬浮展示文字

创建LineTipsDialog.vue

<template>
    <div
        v-if="isTooltipVisible"
        :style="{ left: tooltipPosition.x + 'px', top: tooltipPosition.y + 'px' }"
        class="tooltip">
      <div v-html="showText"></div>
    </div>
</template>
<script>
export default {
  data() {
    return {
      showText: '',
      isTooltipVisible: false,  // 控制提示框的显示/隐藏
      tooltipPosition: { x: 0, y: 0 },  // 提示框的位置
    };
  },
  methods: {
    // 处理鼠标移动事件
    open(properties,event) {
      this.showText = 'flag=>'+properties.flag+'<br>resultDesc=>'+properties.resultDesc;
      this.tooltipPosition.x = event.pageX + 10;  // 提示框水平位置,略微偏移
      this.tooltipPosition.y = event.pageY + 10;  // 提示框垂直位置,略微偏移
      this.isTooltipVisible = true;  // 鼠标移动时显示提示框
    },
    // 处理鼠标离开事件
    close() {
      this.showText = '';
      this.isTooltipVisible = false;  // 鼠标离开时隐藏提示框
    },
  }
};
</script>
<style scoped>
.tooltip {
  position: absolute;
  padding: 5px;
  color: #6c7375;
  border-radius: 5px;
  pointer-events: none; /* 让鼠标事件透过悬浮框 */
}
</style>

 引用

import LineTipsDialog from './components/LineTipsDialog.vue'
components: {
    LineTipsDialog
  },
    <line-tips-dialog ref="lineTipsDialog"></line-tips-dialog>

 使用

//鼠标进入线条显示文字
      this.lf.on('edge:mouseenter', (data) => {
        if (data.data.properties.flag&&data.data.properties.flag!==''){
          this.$refs.lineTipsDialog.open(data.data.properties,data.e);
        }
      });
      //鼠标离开线条关闭文字
      this.lf.on('edge:mouseleave', (data) => {
        if (data.data.properties.flag&&data.data.properties.flag!==''){
          this.$refs.lineTipsDialog.close();
        }
      });

 

一键美化 

安装插件

npm install @antv/layout@0.3.1

创建工具beauty.js

// 引入布局模块
const { DagreLayout } = require('@antv/layout');

class BeautyUtils {
    static pluginName = 'dagre';
    lf;
    option;
    upIndexMap = {};
    downIndexMap ={};

    constructor(lf) {
        this.lf = lf;
    }

    getBytesLength(word) {
        if (!word) {
            return 0;
        }
        let totalLength = 0;
        for (let i = 0; i < word.length; i++) {
            const c = word.charCodeAt(i);
            if ((word.match(/[A-Z]/))) {
                totalLength += 1.5;
            } else if ((c >= 0x0001 && c <= 0x007e) || (c >= 0xff60 && c <= 0xff9f)) {
                totalLength += 1;
            } else {
                totalLength += 2;
            }
        }
        return totalLength;
    }

    /**
     * option: {
     *   rankdir: "TB", // layout 方向, 可选 TB, BT, LR, RL
     *   align: undefined, // 节点对齐方式,可选 UL, UR, DL, DR
     *   nodeSize: undefined, // 节点大小
     *   nodesepFunc: undefined, // 节点水平间距(px)
     *   ranksepFunc: undefined, // 每一层节点之间间距
     *   nodesep: 40, // 节点水平间距(px) 注意:如果有grid,需要保证nodesep为grid的偶数倍
     *   ranksep: 40, // 每一层节点之间间距 注意:如果有grid,需要保证ranksep为grid的偶数倍
     *   controlPoints: false, // 是否保留布局连线的控制点
     *   radial: false, // 是否基于 dagre 进行辐射布局
     *   focusNode: null, // radial 为 true 时生效,关注的节点
     * };
     */
    layout(option = {}) {
        this.upIndexMap = {};
        this.downIndexMap = {};
        const { nodes, edges, gridSize } = this.lf.graphModel;
        // 为了保证生成的节点在girdSize上,需要处理一下。
        let nodesep = 40;
        let ranksep = 40;
        if (gridSize > 20) {
            nodesep = gridSize * 2;
            ranksep = gridSize * 2;
        }
        this.option = {
            type: 'dagre',
            rankdir: 'LR',
            // align: 'UL',
            align: 'UR',
            // align: 'DR',
            nodesep,
            ranksep,
            begin: [120, 120],
            ...option,
        };
        const layoutInstance = new DagreLayout(this.option);
        const layoutData = layoutInstance.layout({
            nodes: nodes.map((node) => ({
                id: node.id,
                size: {
                    width: node.width,
                    height: node.height,
                },
                model: node,
            })),
            edges: edges.map((edge) => ({
                source: edge.sourceNodeId,
                target: edge.targetNodeId,
                model: edge,
            })),
        });
        const newGraphData = {
            nodes: [],
            edges: [],
        };
        layoutData.nodes.forEach(node => {
            const { model } = node;
            const data = model.getData();
            data.x = node.x;
            data.y = node.y;
            newGraphData.nodes.push(data);
        });
        layoutData.edges.forEach(edge => {
            const { model } = edge;
            const data = model.getData();
            data.pointsList = this.calcPointsList(model, newGraphData.nodes);
            if (data.pointsList) {
                let first = data.pointsList[0];
                const second = data.pointsList[1];
                const third = data.pointsList[2];
                let last = data.pointsList[3];
                //调整线的出口和入口位置
                if (first.y===second.y){
                    //如果是水平出线
                    if (first.x>second.x){
                        data.pointsList[0].x = data.pointsList[0].x - 29;
                    }else {
                        data.pointsList[0].x = data.pointsList[0].x + 29;
                    }
                }else {
                    //如果是垂直出线
                    if (first.y>second.y){
                        data.pointsList[0].y = data.pointsList[0].y - 15;
                    }else {
                        data.pointsList[0].y = data.pointsList[0].y + 15;
                    }
                }

                if (third.y===last.y){
                    data.pointsList[2].y = data.pointsList[2].y - 14;
                    data.pointsList[3].y = data.pointsList[3].y - 14;
                    //如果是水平入线
                    if (third.x>last.x){
                        data.pointsList[3].x = data.pointsList[3].x + 29;
                    }else {
                        data.pointsList[3].x = data.pointsList[3].x - 29;
                    }
                }else {
                    //如果是垂直出线
                    if (third.y>last.y){
                        data.pointsList[3].y = data.pointsList[3].y + 15;
                    }else {
                        data.pointsList[3].y = data.pointsList[3].y - 37;
                    }
                }

                first = data.pointsList[0];
                last = data.pointsList[3];

                // 计算文字的位置
                let textX = second.x + (third.x - second.x) / 2;
                if (second.x>=third.x){
                    textX = third.x + (second.x - third.x) / 2;
                }
                let textY = second.y + (third.y - second.y) / 2;
                if (second.y>=third.y){
                    textY = third.y + (second.y - third.y) / 2;
                }
                data.startPoint = { x: first.x, y: first.y };
                data.endPoint = { x: last.x, y: last.y };
                if (data.text && data.text.value) {
                    data.text = {
                        x: textX - this.getBytesLength(data.text.value) * 6 - 10,
                        y: textY,
                        value: data.text.value,
                    };
                }
            } else {
                data.startPoint = undefined;
                data.endPoint = undefined;
                if (data.text && data.text.value) {
                    data.text = data.text.value;
                }
            }
            newGraphData.edges.push(data);
        });
        // 调整节点上文字的位置
        newGraphData.nodes.forEach(node => {
            node.text.x = node.x
            node.text.y = node.y+20
        })
        this.lf.render(newGraphData);
    }


    // 在节点确认从左向右后,通过计算来保证节点连线清晰。
    //线比点的y少15 x多29或少29
    calcPointsList(model, nodes) {
        const pointsList = [];
        if (this.option.rankdir === 'LR' && model.modelType === 'polyline-edge') {
            const newSourceNodeData = nodes.find(node => node.id === model.sourceNodeId);
            const newTargetNodeData = nodes.find(node => node.id === model.targetNodeId);
            const sourceX = newSourceNodeData.x;
            const sourceY = newSourceNodeData.y;
            const targetX = newTargetNodeData.x;
            const targetY = newTargetNodeData.y;
            //如果节点在同一行或者同一列,则直接连线
            pointsList.push({
                x: sourceX,
                y: sourceY,
            });
            const offset = this.getOffset(targetX, targetY);
            if (sourceY===targetY){
                //偏移量
                pointsList.push({
                    x: sourceX,
                    y: targetY+offset,
                });
                pointsList.push({
                    x: targetX,
                    y: targetY+offset,
                });
            }else {
                pointsList.push({
                    x: targetX+offset,
                    y: sourceY,
                });
                pointsList.push({
                    x: targetX+offset,
                    y: targetY,
                });
            }
            pointsList.push({
                x: targetX,
                y: targetY,
            });
            return pointsList;
        }
        return undefined;
    }

    //参数是目标节点的坐标
    getOffset(x,y){
        let key = x+','+y;
        let offset = 0;
        if (!this.upIndexMap[key]){
            this.upIndexMap[key]=0
        }
        if (!this.downIndexMap[key]){
            this.downIndexMap[key]=0
        }
        if (this.upIndexMap[key]>this.downIndexMap[key]){
            this.downIndexMap[key]++
            offset = -80*this.downIndexMap[key]
        }else {
            this.upIndexMap[key]++
            offset = 80*this.upIndexMap[key]
        }
        return offset;
    }
}
module.exports = BeautyUtils;

使用

<template>
  <el-container>
     <div class="container" ref="container"></div>
    <CustomNodePanel :graphData="graphData" :showPanel="showPanel" ref="drawThis"></CustomNodePanel>
  </el-container>
</template>
<script>
import LogicFlow from "@logicflow/core";
import "@logicflow/core/lib/style/index.css";
import { Control,DndPanel, SelectionSelect,Menu,MiniMap  } from "@logicflow/extension";
import "@logicflow/extension/lib/index.css";
import '@logicflow/extension/lib/style/index.css';
import CustomNodePanel from "./components/CustomNodePanel.vue";
import beauty from './utils/beauty'
export default {
  // eslint-disable-next-line vue/multi-word-component-names
  name: "draw",
  components: { CustomNodePanel },
  provide() {
    return {
      drawThis: this
    }
  },
  data() {
    return {
      lf: null,
      graphData: {
        nodes: [],
        edges: []
      },
      showPanel: true
    }
  },
  methods: {
    init(){
      return  new LogicFlow({
        container: this.$refs.container,
        plugins: [Control,DndPanel, SelectionSelect,Menu,MiniMap  ],
        grid: true,
        edgeType: 'CustomPolyline'
      });
    },
    refreshGraph(){
      this.lf.extension.control.addItem({
        key: 'clear',
        iconClass: 'el-icon-refresh icon-delete',
        title: "一键美化",
        text: "一键美化",
        onClick: () => {
          this.beauty()
        }
      })
      this.lf.render();
    },
    beauty() {
      console.log(this.lf.getGraphData())
      new beauty(this.lf).layout()
    },

  },
};
</script>
<style scoped>
.container {
  display: flex;
  height: 800px;
  flex-grow: 1; /*铺满剩余空间*/
  border: 3px solid #ababab;
  overflow: hidden;
}

</style>

测试

如果启动报错

先查看是否安装了该依赖

npm list @babel/plugin-transform-private-methods

 不存在的话就安装

npm install @babel/plugin-transform-private-methods

 在babel.config.js中添加插件

重新启动即可 

无连线

有连线

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值