使用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;