赞
踩
使用LogicFlow来绘制兼容 BPMN2.0 规范的流程,使用react hooks 根据官方示例改的
下载图片功能只能在谷歌浏览器中使用(官方不支持其他的)
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;
.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; }
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;
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;
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。