Skip to content

LogicFlow 实例

一、概念

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

An image

二、具体操作步骤

1)VUE 项目引入

bash
npm install @logicflow/core
npm install @logicflow/extension

2).vue文件编辑

vue
<template>
  <div class="container" ref="container"></div>
</template>
<script>
import LogicFlow from '@logicflow/core'
import '@logicflow/core/dist/style/index.css'
export default {
  mounted() {
    this.lf = new LogicFlow({
      container: this.$refs.container,
      grid: true
    })
    this.lf.render({
      nodes: [
        {
          id: '1',
          type: 'rect',
          x: 100,
          y: 100,
          text: { x: 100, y: 100, value: '节点1' }
        },
        {
          id: '2',
          type: 'circle',
          x: 300,
          y: 200,
          text: { x: 300, y: 200, value: '节点2' }
        }
      ],
      edges: [
        {
          sourceNodeId: '1',
          targetNodeId: '2',
          type: 'polyline'
        }
      ]
    })
  }
}
</script>
<style scoped>
.container {
  width: 1000px;
  height: 500px;
}
</style>

效果如下所示:

An image

3)引入操作控件等

vue
<template>
  <div class="container" ref="container"></div>
</template>
<script>
import LogicFlow from '@logicflow/core'
import { Control, Menu, DndPanel } from '@logicflow/extension'
import '@logicflow/core/dist/style/index.css'
import '@logicflow/extension/lib/style/index.css'

LogicFlow.use(Control) // 控制面板
LogicFlow.use(Menu) // 右键菜单
LogicFlow.use(DndPanel) // 拖拽面板

export default {
  mounted() {
    let that = this
    this.lf = new LogicFlow({
      container: this.$refs.container,
      grid: true
    })

    // 拖拽面板 DndPanel
    this.lf.extension.dndPanel.setPatternItems([
      {
        label: '选区',
        icon: '',
        callback: () => {
          that.lf.extension.selectionSelect.openSelectionSelect()
          that.lf.once('selection:selected', () => {
            that.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({
      nodes: [
        {
          id: '1',
          type: 'rect',
          x: 100,
          y: 100,
          text: { x: 100, y: 100, value: '节点1' }
        },
        {
          id: '2',
          type: 'circle',
          x: 300,
          y: 200,
          text: { x: 300, y: 200, value: '节点2' }
        }
      ],
      edges: [
        {
          sourceNodeId: '1',
          targetNodeId: '2',
          type: 'polyline'
        }
      ]
    })

    // 打印流程数据
    const xData = this.lf.getGraphData()
    console.log('🚀 ~ mounted ~ xData:', xData)
  }
}
</script>
<style scoped>
.container {
  width: 100%;
  height: 100vh;
}
</style>

效果如下所示:

An image

An image

4)代码整理优化

改成 setup 语法糖形式

vue
<template>
  <div class="container" id="container" ref="containerRef"></div>
</template>
<script setup>
import { onMounted, ref, reactive } from 'vue'
import LogicFlow from '@logicflow/core'
import { Control, Menu, DndPanel, MiniMap } from '@logicflow/extension'
import '@logicflow/core/dist/style/index.css'
import '@logicflow/extension/lib/style/index.css'

LogicFlow.use(Control) // 控制面板
LogicFlow.use(Menu) // 右键菜单
LogicFlow.use(DndPanel) // 拖拽面板
LogicFlow.use(MiniMap) // 导航小地图

const containerRef = ref(null)

const state = reactive({
  lf: {}
})

const init = () => {
  state.lf = new LogicFlow({
    container: containerRef.value,
    // 也可以用下面的方式直接获取(通过 getElementById 方式)
    // container: document.getElementById('container'),
    grid: true
  })

  // 拖拽面板
  state.lf.extension.dndPanel.setPatternItems([
    {
      label: '选区',
      icon: '',
      callback: () => {
        that.lf.extension.selectionSelect.openSelectionSelect()
        that.lf.once('selection:selected', () => {
          that.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: ''
    }
  ])

  // 新增control
  state.lf.extension.control.addItem({
    key: 'mini-map',
    iconClass: 'custom-minimap',
    title: '',
    text: '导航',
    onMouseEnter: (lf, ev) => {
      const position = lf.getPointByClient(ev.x, ev.y)
      lf.extension.miniMap.show(
        position.domOverlayPosition.x - 120,
        position.domOverlayPosition.y + 35
      )
    },
    onClick: (lf, ev) => {
      const position = lf.getPointByClient(ev.x, ev.y)
      lf.extension.miniMap.show(
        position.domOverlayPosition.x - 120,
        position.domOverlayPosition.y + 35
      )
    }
  })

  // 渲染
  state.lf.render({
    nodes: [
      {
        id: '1',
        type: 'rect',
        x: 300,
        y: 100,
        text: { x: 300, y: 100, value: '节点1' }
      },
      {
        id: '2',
        type: 'circle',
        x: 500,
        y: 200,
        text: { x: 500, y: 200, value: '节点2' }
      }
    ],
    edges: [
      {
        sourceNodeId: '1',
        targetNodeId: '2',
        type: 'polyline'
      }
    ]
  })
}

onMounted(() => {
  // 初始化操作
  init()
})
</script>
<style scoped>
.container {
  width: 100%;
  height: 100vh;
}
</style>

An image

三、注意事项

TODO