LogicFlow 来绘制兼容 BPMN2.0 规范的流程 (React hooks版)

Ⅰ- 壹 - 功能展示和使用需求

需求描述

使用LogicFlow来绘制兼容 BPMN2.0 规范的流程,使用react hooks 根据官方示例改的

下载图片功能只能在谷歌浏览器中使用(官方不支持其他的)

功能展示

在这里插入图片描述
在这里插入图片描述

目录结构

在这里插入图片描述

Ⅱ - 贰 - 封装代码

img文件

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

index.tsx

import LogicFlow from '@logicflow/core';
import {
    
    
  BpmnElement,
  BpmnXmlAdapter,
  Snapshot,
  Control,
  Menu,
  SelectionSelect,
} from '@logicflow/extension';
import './index.css';
import '@logicflow/extension/lib/style/index.css';
import '@logicflow/core/dist/style/index.css';
import {
    
     useEffect, useRef, useState } from 'react';
import BpmnPattern from './pattern';
import BpmnIo from './io';
const config = {
    
    
  stopScrollGraph: true,
  stopZoomGraph: true,
  metaKeyMultipleSelected: true,
  grid: {
    
    
    size: 10,
    type: 'dot',
  },
  keyboard: {
    
    
    enabled: true,
  },
  snapline: true,
  width: 1500,
  height: 800,
};

const index = () => {
    
    
  const refContainer = useRef<any>();
  const [lfData, setLfData] = useState<any>(null);
  useEffect(() => {
    
    
    LogicFlow.use(BpmnElement);
    LogicFlow.use(BpmnXmlAdapter);
    LogicFlow.use(Snapshot);
    LogicFlow.use(Control);
    LogicFlow.use(Menu);
    LogicFlow.use(SelectionSelect);
    // use 必须放上面
    const lf = new LogicFlow({
    
    
      container: document.querySelector('#graph') as HTMLElement,
      // container: refContainer.current,
      ...config,
    });

    lf.render();
    setLfData(() => lf);
  }, []);
  useEffect(() => {
    
    
    if (!lfData) return;
    // setIState(() => true);
  }, [lfData]);

  return (
    <div className="bpmn-example-container">
      <div id="graph" ref={
    
    refContainer} className="viewport"></div>
      {
    
    lfData && (
        <>
          <BpmnPattern lf={
    
    lfData} />
          <BpmnIo lf={
    
    lfData} />
        </>
      )}
    </div>
  );
};
export default index;

index.css

.bpmn-example-container {
    
    
  position: relative;
  height: 100%;
  overflow: hidden;
}
.pattern {
    
    
  position: absolute;
  top: 10px;
  left: 10px;
  z-index: 111;
  display: flex;
  flex-direction: column;
  align-items: center;
  width: 60px;
  padding: 10px 0;
  color: #676768;
  font-size: 12px;
  background: rgba(255, 255, 255, 0.8);
  box-shadow: 0 1px 4px rgba(0, 0, 0, 0.3);
  -ms-user-select: none;
  -webkit-user-select: none;
  user-select: node;
}
.pattern-selection {
    
    
  width: 36px;
  height: 36px;
  background: url('')
    center center no-repeat;
  cursor: pointer;
  opacity: 0.99;
}
.pattern-start {
    
    
  width: 36px;
  height: 36px;
  background: url('')
    center center no-repeat;
  cursor: grab;
  opacity: 0.99;
}
.pattern-end {
    
    
  width: 36px;
  height: 36px;
  background: url('')
    center center no-repeat;
  cursor: grab;
  opacity: 0.99;
}
.pattern-user {
    
    
  width: 36px;
  height: 36px;
  background: url('')
    center center no-repeat;
  cursor: grab;
  opacity: 0.99;
}
.pattern-condition {
    
    
  width: 36px;
  height: 36px;
  background: url('')
    center center no-repeat;
  cursor: grab;
  opacity: 0.99;
}
.pattern-circle {
    
    
  width: 36px;
  height: 36px;
  background: url('')
    center center no-repeat;
  cursor: grab;
  opacity: 0.99;
}

.pattern-rect {
    
    
  width: 36px;
  height: 36px;
  background: url('')
    center center no-repeat;
  cursor: grab;
  opacity: 0.99;
}
.graph-io {
    
    
  position: absolute;
  bottom: 10px;
  left: 10px;
  z-index: 9999;
  display: flex;
  padding: 10px;
  background: rgba(255, 255, 255, 0.8);
  box-shadow: 0 1px 4px rgba(0, 0, 0, 0.3);
}
.graph-io > span {
    
    
  margin: 0 5px;
  cursor: pointer;
}
#upload-xml {
    
    
  position: relative;
  display: inline-block;
  overflow: hidden;
  cursor: pointer;
}
.upload {
    
    
  position: absolute;
  top: 0;
  left: 0;
  z-index: 99;
  cursor: pointer;
  opacity: 0;
}
.upload::-webkit-file-upload-button {
    
    
  cursor: pointer;
}

io.tsx


import React from 'react';
import LogicFlow from '@logicflow/core';
import {
    
     Input, Image } from 'antd';

const downloadImg = require('./img/download.png');
const photo = require('./img/img.png');
const uploadImg = require('./img/upload.png');

type IProps = {
    
    
  lf: LogicFlow;
};

function download(filename: string, text: string) {
    
    
  var element = document.createElement('a');
  element.setAttribute('href', 'data:text/plain;charset=utf-8,' + encodeURIComponent(text));
  element.setAttribute('download', filename);

  element.style.display = 'none';
  document.body.appendChild(element);

  element.click();

  document.body.removeChild(element);
}

type FileEventTarget = EventTarget & {
    
     files: FileList };

const BpmnIo: any = (props: IProps) => {
    
    
  const {
    
     lf } = props;
  const downloadXml = () => {
    
    
    const data = lf.getGraphData() as string;
    download('logic-flow.xml', data);
  };
  const uploadXml = (ev: React.ChangeEvent<HTMLInputElement>) => {
    
    
    const file = (ev.target as FileEventTarget).files[0];
    const reader = new FileReader();
    reader.onload = (event: ProgressEvent<FileReader>) => {
    
    
      if (event.target) {
    
    
        const xml = event.target.result as string;
        lf.render(xml);
      }
    };
    reader.readAsText(file); // you could also read images and other binaries
  };

  const downloadImage = async () => {
    
    
    const {
    
     lf } = props;
    lf.getSnapshot('', '#fff');
    // lf.extension.snapshot.getSnapshot();
  };
  return (
    <div className="graph-io">
      <span title="下载 XML" onMouseDown={
    
    () => downloadXml()}>
        <Image preview={
    
    false} width={
    
    '10'} height={
    
    '10'} src={
    
    downloadImg} alt="下载XML" />
      </span>
      <span id="download-img" title="下载图片" onMouseDown={
    
    () => downloadImage()}>
        <Image preview={
    
    false} width={
    
    '10'} height={
    
    '10'} src={
    
    photo} alt="下载图片" />
      </span>
      <span id="upload-xml" title="上传 XML">
        <Input type="file" className="upload" onChange={
    
    (ev) => uploadXml(ev)} />
        <Image preview={
    
    false} width={
    
    '10'} height={
    
    '10'} src={
    
    uploadImg} alt="上传XML" />
      </span>
    </div>
  );
};

export default BpmnIo;

pattern.tsx

import React, {
    
     useEffect } from 'react';
import LogicFlow from '@logicflow/core';

type IProps = {
    
    
  lf: LogicFlow;
};

const BpmnPattern = (props: IProps) => {
    
    
  const {
    
     lf } = props;
  useEffect(() => {
    
    
    lf &&
      lf.on('selection:selected', () => {
    
    
        lf.updateEditConfig({
    
    
          stopMoveGraph: false,
        });
      });
  }, [lf]);
  const addStartNode = () => {
    
    
    lf.dnd.startDrag({
    
    
      type: 'bpmn:startEvent',
      text: '开始',
    });
  };
  const addUserTask = () => {
    
    
    lf.dnd.startDrag({
    
    
      type: 'bpmn:userTask',
    });
  };
  const addServiceTask = () => {
    
    
    lf.dnd.startDrag({
    
    
      type: 'bpmn:serviceTask',
    });
  };
  const addGateWay = () => {
    
    
    lf.dnd.startDrag({
    
    
      type: 'bpmn:exclusiveGateway',
    });
  };
  const addEndNode = () => {
    
    
    lf.dnd.startDrag({
    
    
      type: 'bpmn:endEvent',
      text: '结束',
    });
  };
  const openSelection = () => {
    
    
    lf.updateEditConfig({
    
    
      stopMoveGraph: true,
    });
  };

  const onMouseDownFun = (type: any) => {
    
    
    switch (type) {
    
    
      case 'openSelection':
        lf.updateEditConfig({
    
    
          stopMoveGraph: true,
        });
        break;
      case 'addStartNode':
        lf.dnd.startDrag({
    
    
          type: 'bpmn:startEvent',
          text: '开始',
        });
        break;
      case 'addUserTask':
        lf.dnd.startDrag({
    
    
          type: 'bpmn:userTask',
        });
        break;
      case 'addServiceTask':
        lf.dnd.startDrag({
    
    
          type: 'bpmn:serviceTask',
        });
        break;
      case 'addGateWay':
        lf.dnd.startDrag({
    
    
          type: 'bpmn:exclusiveGateway',
        });
        break;
      case 'addEndNode':
        lf.dnd.startDrag({
    
    
          type: 'bpmn:endEvent',
          text: '结束',
        });
        break;
      case 'circle':
        lf.dnd.startDrag({
    
    
          type: 'circle',
          text: '圆形',
        });
        break;
      case 'rect':
        lf.dnd.startDrag({
    
    
          type: 'rect',
          text: '矩形',
        });
        break;

      default:
        break;
    }
  };

  return (
    <div className="pattern">
      <div className="pattern-selection" onMouseDown={
    
    () => onMouseDownFun('openSelection')} />
      <div>选区</div>
      <div className="pattern-start" onMouseDown={
    
    () => onMouseDownFun('addStartNode')} />
      <div>开始</div>
      <div className="pattern-user" onMouseDown={
    
    () => onMouseDownFun('addUserTask')}></div>
      <div>用户任务</div>
      <div className="pattern-user" onMouseDown={
    
    () => onMouseDownFun('addServiceTask')}></div>
      <div>系统任务</div>
      <div className="pattern-circle" onMouseDown={
    
    () => onMouseDownFun('circle')}></div>
      <div>圆形</div>
      <div className="pattern-rect" onMouseDown={
    
    () => onMouseDownFun('rect')}></div>
      <div>矩形</div>
      <div className="pattern-condition" onMouseDown={
    
    () => onMouseDownFun('addGateWay')}></div>
      <div>条件判断</div>
      <div className="pattern-end" onMouseDown={
    
    () => onMouseDownFun('addEndNode')}></div>
      <div>结束</div>
    </div>
  );
};

export default BpmnPattern;

Ⅲ - 叁 - 使用

import Bpmn from './components/bpmn/index';
const Demo = () => {
    
    
  return <Bpmn></Bpmn>;
};

export default Demo;

猜你喜欢

转载自blog.csdn.net/weixin_42863800/article/details/125633693
今日推荐