This commit is contained in:
GGJ
2025-01-09 19:02:44 +08:00
commit 92e7a7a5eb
2943 changed files with 1152283 additions and 0 deletions

View File

@@ -0,0 +1,427 @@
<template>
<div
v-loading="isView"
class="flow-containers"
:class="{ 'view-mode': isView }"
>
<el-container style="height: 100%">
<el-header
style="
border-bottom: 1px solid rgb(218 218 218);
height: auto;
padding-left: 0px;
"
>
<div
style="
display: flex;
padding: 10px 0px;
justify-content: space-between;
"
>
<div>
<el-upload
action=""
:before-upload="openBpmn"
style="margin-right: 10px; display: inline-block"
>
<el-tooltip effect="dark" content="加载xml" placement="bottom">
<el-button size="mini" icon="el-icon-folder-opened" />
</el-tooltip>
</el-upload>
<el-tooltip effect="dark" content="新建" placement="bottom">
<el-button
size="mini"
icon="el-icon-circle-plus"
@click="newDiagram"
/>
</el-tooltip>
<el-tooltip effect="dark" content="自适应屏幕" placement="bottom">
<el-button size="mini" icon="el-icon-rank" @click="fitViewport" />
</el-tooltip>
<el-tooltip effect="dark" content="放大" placement="bottom">
<el-button
size="mini"
icon="el-icon-zoom-in"
@click="zoomViewport(true)"
/>
</el-tooltip>
<el-tooltip effect="dark" content="缩小" placement="bottom">
<el-button
size="mini"
icon="el-icon-zoom-out"
@click="zoomViewport(false)"
/>
</el-tooltip>
<el-tooltip effect="dark" content="后退" placement="bottom">
<el-button
size="mini"
icon="el-icon-back"
@click="modeler.get('commandStack').undo()"
/>
</el-tooltip>
<el-tooltip effect="dark" content="前进" placement="bottom">
<el-button
size="mini"
icon="el-icon-right"
@click="modeler.get('commandStack').redo()"
/>
</el-tooltip>
</div>
<div>
<el-button size="mini" icon="el-icon-s-check" @click="verifyXML"
>校验xml</el-button
>
<el-button size="mini" icon="el-icon-view" @click="showXML"
>查看xml</el-button
>
<el-button
size="mini"
icon="el-icon-download"
@click="saveXML(true)"
>下载xml</el-button
>
<el-button
size="mini"
icon="el-icon-picture"
@click="saveImg('svg', true)"
>下载svg</el-button
>
<el-button size="mini" type="primary" @click="save"
>保存模型</el-button
>
</div>
</div>
</el-header>
<el-container style="align-items: stretch">
<el-main style="padding: 0">
<div ref="canvas" class="canvas" />
</el-main>
<el-aside
style="width: 400px; min-height: 650px; background-color: #f0f2f5"
>
<!-- <panel
v-if="modeler"
:modeler="modeler"
:users="users"
:groups="groups"
:exps="exps"
:categorys="categorys"
/> -->
</el-aside>
</el-container>
</el-container>
</div>
</template>
<script>
// 汉化
import customTranslate from "./common/customTranslate";
// import lintModule from "bpmn-js-bpmnlint";
import Modeler from "bpmn-js/lib/Modeler";
import bpmnlintConfig from "./.bpmnlintrc.json";
// import panel from './PropertyPanel'
import getInitStr from "./flowable/init";
// 引入flowable的节点文件
import FlowableModule from "./flowable/flowable.json";
import customControlsModule from "./customPanel";
export default {
components: {},
props: {
xml: {
type: String,
default: "",
},
users: {
type: Array,
default: () => [],
},
groups: {
type: Array,
default: () => [],
},
categorys: {
type: Array,
default: () => [],
},
exps: {
type: Array,
default: () => [],
},
isView: {
type: Boolean,
default: false,
},
},
data() {
return {};
},
created() {},
mounted() {
// 生成实例
this.modeler = new Modeler({
container: this.$refs.canvas,
additionalModules: [
// lintModule,
customControlsModule,
{
//汉化
translate: ["value", customTranslate],
},
],
linting: {
bpmnlint: bpmnlintConfig,
},
moddleExtensions: {
flowable: FlowableModule,
},
});
// 新增流程定义
if (!this.xml) {
this.newDiagram();
} else {
this.createNewDiagram(this.xml);
}
},
methods: {
newDiagram() {
this.createNewDiagram(getInitStr());
},
// 让图能自适应屏幕
fitViewport() {
this.zoom = this.modeler.get("canvas").zoom("fit-viewport");
const bbox = document
.querySelector(".flow-containers .viewport")
.getBBox();
const currentViewbox = this.modeler.get("canvas").viewbox();
const elementMid = {
x: bbox.x + bbox.width / 2 - 65,
y: bbox.y + bbox.height / 2,
};
this.modeler.get("canvas").viewbox({
x: elementMid.x - currentViewbox.width / 2,
y: elementMid.y - currentViewbox.height / 2,
width: currentViewbox.width,
height: currentViewbox.height,
});
this.zoom = (bbox.width / currentViewbox.width) * 1.8;
},
// 放大缩小
zoomViewport(zoomIn = true) {
this.zoom = this.modeler.get("canvas").zoom();
this.zoom += zoomIn ? 0.1 : -0.1;
this.modeler.get("canvas").zoom(this.zoom);
},
async createNewDiagram(data) {
// 将字符串转换成图显示出来
// data = data.replace(/<!\[CDATA\[(.+?)]]>/g, '&lt;![CDATA[$1]]&gt;')
data = data.replace(/<!\[CDATA\[(.+?)]]>/g, function (match, str) {
return str.replace(/</g, "&lt;");
});
try {
await this.modeler.importXML(data);
// this.adjustPalette()
this.fitViewport();
} catch (err) {
// console.error(err.message, err.warnings);
}
},
// 对外 api
getProcess() {
const element = this.getProcessElement();
return {
id: element.id,
name: element.name,
category: element.$attrs["flowable:processCategory"],
};
},
getProcessElement() {
const rootElements = this.modeler.getDefinitions().rootElements;
for (let i = 0; i < rootElements.length; i++) {
if (rootElements[i].$type === "bpmn:Process") return rootElements[i];
}
},
async verifyXML() {
const linting = this.modeler.get("linting");
linting.toggle();
},
async saveXML(download = false) {
try {
const { xml } = await this.modeler.saveXML({ format: true });
if (download) {
this.downloadFile(
`${this.getProcessElement().name}.bpmn20.xml`,
xml,
"application/xml"
);
}
return xml;
} catch (err) {
// console.log(err);
}
},
async showXML() {
try {
const xml = await this.saveXML();
this.$emit("showXML", xml);
} catch (err) {
// console.log(err);
}
},
async saveImg(type = "svg", download = false) {
try {
const { svg } = await this.modeler.saveSVG({ format: true });
if (download) {
this.downloadFile(
this.getProcessElement().name,
svg,
"image/svg+xml"
);
}
return svg;
} catch (err) {
// console.log(err);
}
},
async save() {
const process = this.getProcess();
const xml = await this.saveXML();
const svg = await this.saveImg();
const result = { process, xml, svg };
this.$emit("save", result);
window.parent.postMessage(result, "*");
},
openBpmn(file) {
const reader = new FileReader();
reader.readAsText(file, "utf-8");
reader.onload = () => {
this.createNewDiagram(reader.result);
};
return false;
},
downloadFile(filename, data, type) {
const a = document.createElement("a");
const url = window.URL.createObjectURL(new Blob([data], { type: type }));
a.href = url;
a.download = filename;
a.click();
window.URL.revokeObjectURL(url);
},
},
computed: {},
watch: {},
};
</script>
<style lang="scss">
/*左边工具栏以及编辑节点的样式*/
@import "~bpmn-js/dist/assets/diagram-js.css";
@import "~bpmn-js/dist/assets/bpmn-font/css/bpmn.css";
@import "~bpmn-js/dist/assets/bpmn-font/css/bpmn-codes.css";
@import "~bpmn-js/dist/assets/bpmn-font/css/bpmn-embedded.css";
@import "~bpmn-js-bpmnlint/dist/assets/css/bpmn-js-bpmnlint.css";
.view-mode {
.el-header,
.el-aside,
.djs-palette,
.bjs-powered-by {
display: none;
}
.el-loading-mask {
background-color: initial;
}
.el-loading-spinner {
display: none;
}
}
.flow-containers {
// background-color: #ffffff;
width: 100%;
height: 100%;
.canvas {
width: 100%;
height: 100%;
//flex: 1;
//position: relative;
//background: url("")
//repeat !important;
//div.toggle-mode {
// display: none;
//}
}
.panel {
position: absolute;
right: 0;
top: 50px;
width: 300px;
}
.load {
margin-right: 10px;
}
.el-form-item__label {
font-size: 13px;
}
.djs-palette {
left: 0px !important;
top: 0px;
border-top: none;
}
.djs-container svg {
//min-height: 650px;
}
.highlight.djs-shape .djs-visual > :nth-child(1) {
fill: green !important;
stroke: green !important;
fill-opacity: 0.2 !important;
}
.highlight.djs-shape .djs-visual > :nth-child(2) {
fill: green !important;
}
.highlight.djs-shape .djs-visual > path {
fill: green !important;
fill-opacity: 0.2 !important;
stroke: green !important;
}
.highlight.djs-connection > .djs-visual > path {
stroke: green !important;
}
// .djs-connection > .djs-visual > path {
// stroke: orange !important;
// stroke-dasharray: 4px !important;
// fill-opacity: 0.2 !important;
// }
// .djs-shape .djs-visual > :nth-child(1) {
// fill: orange !important;
// stroke: orange !important;
// stroke-dasharray: 4px !important;
// fill-opacity: 0.2 !important;
// }
.highlight-todo.djs-connection > .djs-visual > path {
stroke: orange !important;
stroke-dasharray: 4px !important;
fill-opacity: 0.2 !important;
}
.highlight-todo.djs-shape .djs-visual > :nth-child(1) {
fill: orange !important;
stroke: orange !important;
stroke-dasharray: 4px !important;
fill-opacity: 0.2 !important;
}
.overlays-div {
font-size: 10px;
color: red;
width: 100px;
top: -20px !important;
}
}
</style>
<style lang="less" scoped>
@import url("../../../styles/comStyle.less");
</style>