Vue2.0+AntvX6
Vue2.0+AntvX6
·
Vue2.0+AntvX6
前篇文章实现了AntvX6流程图 这篇是另一种效果,但都是Vue2.0+AntvX6,antv/x6版本都是"@antv/x6": “~1.16.0”,其实现的效果图如下:
流程线也可以配置信息
其实现的代码如下:
<template>
<div>
<el-button style="float: right; margin-top: -75px; margin-right: 80px" size="mini" type="primary" @click="saveFlow">保存</el-button>
<el-container class="process-edit-container">
<!-- 顶部功能区域 -->
<!-- <el-header>-->
<!-- </el-header>-->
<el-container>
<!-- 左侧组件区域 -->
<el-aside width="200px">
<div class="process-component-list">
<img
src=""
data-type="node"
data-shape="circle"
data-size="72*72"
data-color="#FA8C16"
data-label="开始节点"
@mousedown="addNodeToGraph">
<img
draggable="false"
src=""
data-type="node"
data-shape="rect"
data-size="80*48"
data-color="#1890FF"
data-label="审批节点"
@mousedown="addNodeToGraph">
<img
draggable="false"
src=""
data-type="node"
data-shape="polygon"
data-size="80*72"
data-color="#13C2C2"
data-label="处理节点"
@mousedown="addNodeToGraph">
<img
draggable="false"
src=""
data-type="node"
data-shape="ellipse"
data-size="80*48"
data-color="#722ED1"
data-label="结束节点"
@mousedown="addNodeToGraph">
</div>
</el-aside>
<el-main>
<div class="process-canvas" ref="processCanvas"></div>
</el-main>
<!-- 右侧配置区域 -->
<el-aside>
<el-tabs v-model="activeTab" type="card">
<el-tab-pane label="流程信息" name="processInfo">
<el-form :model="workflow" :rules="rules" ref="workflow" label-width="80px" style="margin-right: 8px">
<el-form-item label="名称" prop="name">
<el-input v-model="workflow.name" size="mini"></el-input>
</el-form-item>
<el-form-item label="分类" prop="projectId">
<el-select
clearable
filterable
placeholder="请选择分类"
size="mini"
style="width: 100%"
v-model="workflow.projectId"
>
<el-option
:key="index"
:label="option.name"
:value="option.id"
v-for="(option, index) in projectListTmp"
></el-option>
</el-select>
</el-form-item>
<el-form-item label="模版" prop="tpl">
<el-select
size="mini"
v-model="workflow.tpl"
multiple
filterable
clearable
placeholder="请选择"
style="width: 100%">
<el-option
v-for="item in templateList"
:key="item.id"
:label="item.name"
:value="item.id">
</el-option>
</el-select>
</el-form-item>
<el-form-item label="描述" prop="description">
<el-input type="textarea" v-model="workflow.description" size="mini"></el-input>
</el-form-item>
</el-form>
</el-tab-pane>
<el-tab-pane label="配置信息" name="cellInfo">
<el-form :model="node" :rules="rules" ref="node" label-width="80px" v-if="selectCell && selectCell.shape !== 'edge'">
<el-form-item label="名称" prop="name">
<el-input v-model="node.name" size="mini" @input="updateNodeName" placeholder="请输入节点名称"></el-input>
</el-form-item>
<el-form-item label="是否隐藏" prop="isHidden" style="margin-top: -20px">
<el-select v-model="node.isHidden" placeholder="请选择是否隐藏" size="mini" style="width: 100%" filterable>
<el-option label="是" :value="true"></el-option>
<el-option label="否" :value="false"></el-option>
</el-select>
</el-form-item>
<el-form-item label="顺序" prop="orderId" style="margin-top: -20px">
<el-input v-model.number="node.orderId" size="mini" placeholder="请输入顺序值"></el-input>
</el-form-item>
<el-form-item label="人员类型" prop="participantTypeId" style="margin-top: -20px;">
<el-select filterable v-model.number="node.participantTypeId" placeholder="请选择处理人类型" @change="getParticipantList" size="mini" style="width: 100%">
<el-option
v-for="item in participantTypeOptions"
:key="item.value"
:label="item.label"
:value="item.value">
</el-option>
</el-select>
</el-form-item>
<el-form-item label="处理人" prop="participant" style="margin-top: -20px;" v-if="node.participantTypeId!==0">
<el-select filterable v-model="node.participant" placeholder="请选择处理人" size="mini" style="width: 100%">
<el-option
v-for="item in participantOptions"
:key="item.id"
:label="item.name===undefined?item.nickname:item.name"
:value="item.id">
</el-option>
</el-select>
</el-form-item>
<el-form-item label="分配方式" prop="distributeTypeId" style="margin-top: -20px;">
<el-select filterable v-model.number="node.distributeTypeId" placeholder="请选择分配方式" size="mini" style="width: 100%">
<el-option
v-for="item in distributeTypeOptions"
:key="item.value"
:label="item.label"
:value="item.value">
</el-option>
</el-select>
</el-form-item>
<el-form-item label="可写模版" style="margin-top: -20px;">
<el-select
filterable
multiple
v-model="node.writableTemplate"
placeholder="请选择可写模版"
size="mini"
style="width: 100%"
@change="uniqueTemplate($event, 'writable')"
>
<el-option
v-for="item in currentTemplateList"
:key="item.id"
:label="item.name"
:value="item.id"
>
</el-option>
</el-select>
</el-form-item>
<el-form-item label="隐藏模版" style="margin-top: -20px;">
<el-select
filterable
multiple
v-model="node.hideTemplate"
placeholder="请选择需要隐藏的模版"
size="mini"
style="width: 100%"
@change="uniqueTemplate($event, 'hidden')"
>
<el-option
v-for="item in currentTemplateList"
:key="item.id"
:label="item.name"
:value="item.id"
>
</el-option>
</el-select>
</el-form-item>
<el-form-item label="状态属性" prop="attributeTypeId" style="margin-top: -20px;">
<el-select filterable v-model.number="node.attributeTypeId" placeholder="请选择状态属性" size="mini" style="width: 100%">
<el-option
v-for="item in attributeTypeOptions"
:key="item.value"
:label="item.label"
:value="item.value">
</el-option>
</el-select>
</el-form-item>
<el-form-item label="抄送" style="margin-top: -20px;">
<el-select filterable multiple clearable v-model.number="node.ccEmail" placeholder="请选择抄送人" size="mini" style="width: 100%">
<el-option
v-for="item in participantOptions"
:key="item.id"
:label="item.name===undefined?item.nickname:item.name"
:value="item.email">
</el-option>
</el-select>
</el-form-item>
<el-form-item label="标签" style="margin-top: -20px;">
<el-select multiple clearable filterable v-model="node.label" placeholder="请选择标签" size="mini" style="width: 100%">
<el-option
v-for="item in labelOptions"
:key="item.value"
:label="item.label"
:value="item.value">
</el-option>
</el-select>
</el-form-item>
</el-form>
<el-form :model="edge" :rules="rules" ref="edge" label-width="80px" v-if="selectCell && selectCell.shape === 'edge'">
<el-form-item label="名称" prop="name">
<el-input v-model="edge.name" size="mini" @input="updateEdgeName" placeholder="请输入流转名称"></el-input>
</el-form-item>
<el-form-item label="源节点" prop="sourceStatus" style="margin-top: -20px" v-show="false">
<el-input v-model.number="edge.sourceStatus" size="mini" placeholder="请输入源节点" readonly></el-input>
</el-form-item>
<el-form-item label="目标节点" prop="destinationStatus" style="margin-top: -20px;" v-show="false">
<el-input v-model.number="edge.destinationStatus" size="mini" placeholder="请输入目标节点" readonly></el-input>
</el-form-item>
<el-form-item label="流转属性" prop="attributeTypeId" style="margin-top: -20px;">
<el-select v-model.number="edge.attributeTypeId" placeholder="请选择流转属性" size="mini" style="width: 100%">
<el-option
v-for="item in tranAttributeTypeOptions"
:key="item.value"
:label="item.label"
:value="item.value">
</el-option>
</el-select>
</el-form-item>
<el-form-item label="标签" style="margin-top: -20px;">
<el-select multiple clearable filterable v-model="edge.label" placeholder="请选择标签" size="mini" style="width: 100%">
<el-option
v-for="item in edgeLabelOptions"
:key="item.value"
:label="item.label"
:value="item.value">
</el-option>
</el-select>
</el-form-item>
</el-form>
</el-tab-pane>
</el-tabs>
</el-aside>
</el-container>
</el-container>
</div>
</template>
<script>
// import {
// loadProjectList, loadTemplateList, loadUserList, loadDepartList,
// } from '@/api/processCenter';
import { deepCopy } from '../utils/utils';
import { Graph, Addon } from '@antv/x6';
const { Dnd } = Addon;
export default {
name: 'ProcessEdit',
props: {
data: {
type: Object,
default: () => ({
name: '',
projectId: null,
description: '',
icon: 'el-icon-user',
iconColor: 'blue',
tpl: [],
flowchart: {
nodes: [],
edges: [],
workflow: {},
},
}),
},
},
computed: {
// workflow: {
// get () {
// return this.data;
// },
// set (val) {
// this.$emit('update:data', val);
// }
// },
currentTemplateList () {
return this.templateList.filter((tpl) => {
if (!this.workflow.tpl || !Array.isArray(this.workflow.tpl) || this.workflow.tpl.length === 0) {
return false;
}
return this.workflow.tpl.indexOf(tpl.id) >= 0;
});
},
},
watch: {
data: {
handler (val) {
if (this.internalChange) {
return;
}
// nodeConfigs: {},
// edgeConfigs: {},
this.workflow = {
name: val.name,
projectId: val.projectId,
description: val.description,
icon: val.icon,
iconColor: val.iconColor,
tpl: deepCopy(val.tpl),
nodeConfigs: {},
edgeConfigs: {},
}
const cells = [];
if (val.flowchart.nodes.length > 0) {
val.flowchart.nodes.forEach((node) => {
const temp = deepCopy(node);
this.workflow.nodeConfigs[node.id] = temp.valueData;
delete temp.valueData;
cells.push(temp);
});
}
if (val.flowchart.edges.length > 0) {
val.flowchart.edges.forEach((edge) => {
const temp = deepCopy(edge);
this.workflow.edgeConfigs[edge.id] = temp.valueData;
delete temp.valueData;
cells.push(temp);
});
}
if (this.graph) {
this.graph.fromJSON({ cells });
this.graph.centerContent();
} else {
this.cells = cells;
}
},
deep: true,
immediate: true,
},
},
data: () => ({
cells: [],
workflow: {},
internalChange: false,
graph: null,
dnd: null,
selectCell: null,
activeTab: 'processInfo',
projectListTmp: [],
templateList: [],
node: {},
edge: {},
rules: {
name: [
{ required: true, message: ' ', trigger: 'change' }
],
tpl: [
{ required: true, message: ' ', trigger: 'change' }
],
icon: [
{ required: true, message: ' ', trigger: 'change' }
],
orderId: [
{ required: true, message: ' ', trigger: 'change' }
],
participant: [
{ required: true, message: ' ', trigger: 'change' }
],
participantTypeId: [
{ required: true, message: ' ', trigger: 'change' }
],
distributeTypeId: [
{ required: true, message: ' ', trigger: 'change' }
],
attributeTypeId: [
{ required: true, message: ' ', trigger: 'change' }
],
sourceStatus: [
{ required: true, message: ' ', trigger: 'change' }
],
projectId: [
{ required: true, message: '请选择项目', trigger: 'change' }
],
destinationStatus: [
{ required: true, message: ' ', trigger: 'change' }
]
},
participantTypeOptions: [
{ label: '无处理人', value: 0 },
{ label: '个人', value: 1 },
{ label: '业务组', value: 2 },
// { label: '部门', value: 3 },
{ label: '变量', value: 4 },
],
distributeTypeOptions: [
// { label: '主动接单', value: 1 },
{ label: '直接处理', value: 2 }
// { label: '随机分配', value: 3 },
// { label: '全部处理', value: 4 }
],
attributeTypeOptions: [
{ label: '新建', value: 1 },
{ label: '审批', value: 2 },
{ label: '处理', value: 3 },
{ label: '结束', value: 4 },
{ label: '其他', value: 5 }
],
labelOptions: [
{ label: '抄送HRBP', value: 'ccHrbp' },
{ label: '指定交接人', value: 'handover' },
{ label: '分管负责人', value: 'chargeLeader' }
],
edgeLabelOptions: [
{ label: 'leader申请', value: 'leaderApply' },
{ label: '指定分管领导', value: 'plusSign' }
],
tranAttributeTypeOptions: [
{ label: '同意', value: 1 },
{ label: '拒绝', value: 2 },
{ label: '其他', value: 3 }
],
participantOptions: [],
participantMap: {},
clickCellMap: {},
loadUserPromise: [],
}),
created () {
this.loadProject();
this.loadTemplate();
// this.loadUserListPartial();
},
methods: {
uniqueTemplate (select, type) {
// debugger;
let uniqueList = [];
if (type === 'writable') {
uniqueList = this.node.hideTemplate;
} else {
uniqueList = this.node.writableTemplate;
}
select.forEach(v => {
const index = uniqueList.findIndex(value => value === v);
if (index >= 0) {
uniqueList.splice(index, 1);
}
});
// this.node.name = this.node.hideTemplate.length;
this.$forceUpdate();
},
async loadUserListPartialFunc () {
const params = {
page: 1,
perPage: 100,
sort: 1,
}
// const loadAction = async (params, res) => {
// return await loadUserList(params).then(response => {
// res.push(...response.data);
// return response.data.length === params.perPage;
// }, () => false);
// }
const res = [];
let flag = await loadAction(params, res);
while (flag) {
params.page = params.page + 1;
flag = await loadAction(params, res);
}
this.participantMap.user = res;
while (this.loadUserPromise.length > 0) {
const callback = this.loadUserPromise.shift();
callback[0]();
}
},
loadUserListPartial () {
if (this.loadUserPromise.length === 0) {
this.loadUserListPartialFunc();
}
return new Promise((resolve, reject) => {
this.loadUserPromise.push([resolve, reject]);
});
},
getParticipantList (init = false) {
// this.node.participant = '';
this.participantOptions = [];
const params = {
page: 1,
perPage: 99999,
sort: 1,
}
if (this.node.participantTypeId === 1) {
if (!init || this.node.participant === 0) this.node.participant = '';
this.participantOptions = this.participantMap.user;
if (!this.participantMap.user) {
this.loadUserListPartial().then(() => {
this.participantOptions = this.participantMap.user;
});
// loadUserList(params).then(res => {
// this.participantMap.user = res.data;
// this.participantOptions = res.data;
// }, err => {
// console.error(err);
// this.$message({
// type: 'error',
// message: err.errmsg,
// });
// });
} else {
this.participantOptions = this.participantMap.user;
}
} else if (this.node.participantTypeId === 2) {
if (!this.participantMap.depart) {
// loadDepartList(params).then(res => {
// this.participantMap.depart = res.data;
// this.participantOptions = res.data;
// }, err => {
// console.error(err);
// this.$message({
// type: 'error',
// message: err.errmsg,
// });
// });
} else {
this.participantOptions = this.participantMap.depart;
}
} else if (this.node.participantTypeId === 4) {
// 处理变量
this.participantOptions = [
{ id: 1, name: '创建者' },
{ id: 2, name: '创建者Leader' },
{ id: 3, name: '汇报对象' },
{ id: 4, name: '分管负责人' },
{ id: 5, name: 'HRBP' },
{ id: 6, name: 'VP' }
]
if (!init) this.node.participant = 1;
} else if (this.node.participantTypeId === 0) {
this.node.participant = 0;
}
},
loadTemplate () {
const params = {
page: 1,
perPage: 99999,
sort: 1,
}
// loadTemplateList(params).then((res) => {
// this.templateList = res.data;
// }, (err) => {
// this.$message({
// type: 'error',
// message: err.errmsg,
// });
// console.error(err);
// });
},
loadProject () {
const params = {
page: 1,
perPage: 99999,
sort: 1,
}
// loadProjectList(params).then((res) => {
// this.projectListTmp = res.data;
// }, (err) => {
// this.$message({
// type: 'error',
// message: err.errmsg,
// });
// console.error(err);
// });
},
handleCellClick (cell) {
this.activeTab = 'cellInfo';
this.selectCell = cell;
const id = cell.id;
if (cell.isNode()) {
if (!this.workflow.nodeConfigs[id]) {
let attributeTypeId = 1;
if (cell.shape === 'rect') {
attributeTypeId = 2;
} else if (cell.shape === 'polygon') {
attributeTypeId = 3;
} else if (cell.shape === 'ellipse') {
attributeTypeId = 4;
}
// { label: '新建', value: 1 },
// { label: '审批', value: 2 },
// { label: '处理', value: 3 },
// { label: '结束', value: 4 },
// { label: '其他', value: 5 }
this.workflow.nodeConfigs[id] = {
tmpId: id,
name: cell.attrs.label.text,
isHidden: false,
orderId: null,
participantTypeId: 0,
participant: '',
distributeTypeId: 2,
attributeTypeId,
writableTemplate: [],
hideTemplate: [],
label: [],
ccEmail: [],
};
}
this.node = this.workflow.nodeConfigs[id];
} else {
if (!this.workflow.edgeConfigs[id]) {
this.workflow.edgeConfigs[id] = {
tmpId: id,
label: [],
name: '',
sourceStatus: cell.source.cell,
destinationStatus: cell.target.cell,
attributeTypeId: '',
};
}
this.edge = this.workflow.edgeConfigs[id];
}
this.getParticipantList(true);
},
updateEdgeName (name) {
this.selectCell.setLabels(name);
},
updateNodeName (name) {
this.selectCell.setAttrs({
label: {
text: name,
},
});
},
addNodeToGraph (event) {
const target = event.currentTarget;
const type = target.getAttribute('data-type');
let node = null;
if (type === 'node') {
const shape = target.getAttribute('data-shape');
const size = target.getAttribute('data-size');
const width = Number(size.split('*')[0]);
const height = Number(size.split('*')[1]);
// data-color="#1890FF"
const label = target.getAttribute('data-label');
const color = target.getAttribute('data-color');
const rgbToHex = (r, g, b) => {
const hex = ((r << 16) | (g << 8) | b).toString(16);
return '#' + new Array(Math.abs(hex.length - 7)).join('0') + hex;
}
// hex to rgb
const hexToRgb = (hex) => {
const rgb = [];
for (let i = 1; i < 7; i += 2) {
rgb.push(parseInt('0x' + hex.slice(i, i + 2)));
}
return rgb;
}
// 计算渐变过渡色
const gradient = (startColor, endColor, step) => {
// 将 hex 转换为rgb
const sColor = hexToRgb(startColor);
const eColor = hexToRgb(endColor);
// 计算R\G\B每一步的差值
const rStep = (eColor[0] - sColor[0]) / step;
const gStep = (eColor[1] - sColor[1]) / step;
const bStep = (eColor[2] - sColor[2]) / step;
const gradientColorArr = [];
for (let i = 0; i < step; i++) {
// 计算每一步的hex值
gradientColorArr.push(rgbToHex(parseInt(rStep * i + sColor[0]), parseInt(gStep * i + sColor[1]), parseInt(bStep * i + sColor[2])));
}
return gradientColorArr;
}
const colorList = gradient(color, '#ffffff', 10);
const halfColor = colorList[9];
const fillColor = colorList[6];
const ports = {
groups: {
top: {
position: 'top',
attrs: {
circle: {
r: 4,
magnet: true,
stroke: '#31d0c6',
strokeWidth: 2,
fill: '#fff',
}
}
},
bottom: {
position: 'bottom',
attrs: {
circle: {
r: 4,
magnet: true,
stroke: '#31d0c6',
strokeWidth: 2,
fill: '#fff',
}
}
},
left: {
position: 'left',
attrs: {
circle: {
r: 4,
magnet: true,
stroke: '#31d0c6',
strokeWidth: 2,
fill: '#fff',
}
}
},
right: {
position: 'right',
attrs: {
circle: {
r: 4,
magnet: true,
stroke: '#31d0c6',
strokeWidth: 2,
fill: '#fff',
}
}
}
},
items: [
{
id: 'top',
group: 'top',
},
{
id: 'bottom',
group: 'bottom',
},
{
id: 'left',
group: 'left',
},
{
id: 'right',
group: 'right',
}
],
}
const nodeInfo = {
shape,
width,
height,
attrs: {
label: {
'font-size': 12,
text: label,
fill: 'black',
},
body: {
stroke: fillColor,
fill: halfColor,
strokeWidth: 2,
},
},
ports,
};
if (shape === 'polygon') {
nodeInfo.points = '0,10 10,0 20,10 10,20';
}
node = this.graph.createNode(nodeInfo);
} else {
node = {};
}
this.dnd.start(node, event);
},
initG6 () {
const that = this;
this.graph = new Graph({
container: that.$refs.processCanvas,
autoResize: true,
connecting: {
snap: true, // 自动吸附连接线
allowBlank: false, // 不允许只有一个节点的线
// allowMulti: false, // 不允许在两个节点间创建多个线
highlight: true, // 高亮可被连接的点
},
grid: true,
history: true,
snapline: {
enabled: true, // 对齐线
sharp: true,
},
scroller: {
enabled: true,
pageVisible: false,
pageBreak: false,
pannable: true,
},
// mousewheel: {
// enabled: true,
// modifiers: ['ctrl', 'meta'],
// },
});
// this.graph.fromJSON(data);
this.dnd = new Dnd({
target: this.graph,
scaled: false,
animation: true,
});
this.graph.on('edge:connected', ({ isNew, edge }) => {
if (isNew) {
that.handleCellClick(edge);
}
});
this.graph.on('cell:mouseenter', ({ cell }) => {
if (this.clickCellMap[cell.id]) {
clearTimeout(this.clickCellMap[cell.id]);
delete this.clickCellMap[cell.id];
}
if (cell.isNode()) {
let offset = { x: 10, y: 10 };
if (cell.shape === 'polygon') {
offset = { x: 20, y: 20 };
}
cell.addTools([
{
name: 'boundary',
args: {
attrs: {
fill: '#7c68fc',
stroke: '#333',
'stroke-width': 1,
'fill-opacity': 0.2,
},
},
},
{
name: 'button-remove',
args: {
x: 0,
y: 0,
offset,
},
},
])
} else {
cell.addTools(['button-remove']);
}
});
this.graph.on('cell:mouseleave', ({ cell }) => {
cell.removeTools();
});
this.graph.on('cell:click', ({ cell }) => {
that.handleCellClick(cell);
});
this.graph.on('node:added', ({ cell }) => {
that.handleCellClick(cell);
});
if (this.cells.length > 0) {
this.graph.fromJSON({ cells: this.cells });
this.$nextTick(() => {
this.graph.centerContent();
});
this.cells = [];
}
},
async saveFlow () {
let workflowValid;
await this.$refs.workflow.validate(valid => {
workflowValid = valid;
});
if (!workflowValid) {
this.activeTab = 'processInfo';
this.$message({
type: 'warning',
message: '请完成流程信息的配置',
});
return;
}
const data = this.graph.toJSON();
const { cells } = data;
const workflow = {
name: this.workflow.name,
projectId: this.workflow.projectId,
description: this.workflow.description,
icon: this.workflow.icon,
iconColor: this.workflow.iconColor,
tpl: deepCopy(this.workflow.tpl),
}
const flowchart = {};
flowchart.workflow = deepCopy(workflow);
flowchart.nodes = [];
flowchart.edges = [];
const that = this;
const checkEmpty = (v) => (v === null || v === undefined || v === '');
const checkNode = (form) => {
if (checkEmpty(form.name)) {
return false;
}
if (checkEmpty(form.orderId)) {
return false;
}
if (checkEmpty(form.participantTypeId)) {
return false;
}
if (checkEmpty(form.participant)) {
return false;
}
if (checkEmpty(form.distributeTypeId)) {
return false;
}
return !checkEmpty(form.attributeTypeId);
}
const checkEdge = (form) => {
if (checkEmpty(form.name)) {
return false;
}
return form.attributeTypeId;
}
const cellValid = cells.every((cell) => {
// 线
if (cell.shape === 'edge') {
const { id } = cell;
if (!that.workflow.edgeConfigs[id]) {
const cellObj = that.graph.getCell(id);
that.handleCellClick(cellObj);
that.$nextTick(() => {
that.$refs.edge.validate();
});
return false;
}
if (!checkEdge(that.workflow.edgeConfigs[id])) {
const cellObj = that.graph.getCell(id);
that.handleCellClick(cellObj);
that.$nextTick(() => {
that.$refs.edge.validate();
});
this.$message({
type: 'warning',
message: '请完成流转信息的配置',
});
return false;
}
const temp = deepCopy(cell);
temp.valueData = that.workflow.edgeConfigs[id];
flowchart.edges.push(temp);
return true;
} else {
const { id } = cell;
if (!that.workflow.nodeConfigs[id]) {
const cellObj = that.graph.getCell(id);
that.activeTab = 'cellInfo';
that.handleCellClick(cellObj);
that.$nextTick(() => {
that.$refs.node.validate();
});
return false;
}
if (!checkNode(that.workflow.nodeConfigs[id])) {
const cellObj = that.graph.getCell(id);
that.activeTab = 'cellInfo';
that.handleCellClick(cellObj);
that.$nextTick(() => {
that.$refs.node.validate();
});
this.$message({
type: 'warning',
message: '请完成节点信息的配置',
});
return false;
}
const temp = deepCopy(cell);
temp.valueData = that.workflow.nodeConfigs[id];
flowchart.nodes.push(temp);
return true;
}
});
// workflow.flowchart = flowchart;
if (!cellValid) {
return;
}
this.internalChange = true;
this.$emit('update', flowchart);
this.$nextTick(() => {
this.internalChange = false;
});
},
},
mounted () {
this.initG6();
},
}
</script>
<style scoped>
.process-edit-container {
margin-top: 20px;
/* height: calc(100% - 20px); */
height: 90vh;
}
.process-canvas {
width: 100%;
height: 100%;
}
.process-component-list {
height: 100%;
background-color: #e6e6e6;
display: flex;
flex-wrap: wrap;
align-items: center;
justify-content: space-around;
align-content: flex-start;
}
</style>
上面的代码作为一个公用组件在下面这个组件中导入,根据后端返回的json可以对流程图进行编辑
<template>
<span class="staff-title" style="padding-top: 100px">{{activeTitle}}</span>
<process-edit :data="processInfo" @update="handleSave"/>
</template>
<script>
// 导入的后端服务地址,这个不需要参考
import { loadFlowById, createFlow, updateFlowById } from '@/api/processCenter';
import { deepCopy } from '@/utils/utils'; // 深拷贝(贴到下面了)
import processEdit from '@/components/processEditor/processEdit.vue'; // 上面的代码作为子组件导入
export default {
name: 'editProcess',
components: {
processEdit,
},
props: {
processId: { // 父组件传入的值
type: [Number, String],
default: -1,
},
},
data: () => ({
init: false,
realId: null,
processInfo: {
name: '',
projectId: null,
description: '',
icon: 'el-icon-user',
iconColor: 'blue',
tpl: [],
flowchart: {
nodes: [],
edges: [],
workflow: {},
},
},
// rDong修改
activeTitle: ''
}),
created () {
// 父组件传入processId后传入后端地址请求数据
if (Number(this.processId) !== -1) {
this.init = true;
this.activeTitle = '编辑流程';
this.realId = this.processId;
loadFlowById(this.processId).then(res => {
this.processInfo = res;
this.$nextTick(() => {
this.init = false;
});
}, (err) => {
this.$message({
type: 'error',
message: err.errmsg,
});
console.error(err);
});
} else {
this.activeTitle = '新建流程';
}
},
watch: {
// 将后端返回的数据进行处理转变为子组件可以接收展示的数据,并传给子组件展示并可编辑
// 这边不用参考, 可以将之前子组件传过来的值直接保存到后端,到时候请求将后端保存的值在进行返回展示编辑,这样可以不进行数据格式的处理。
processInfo: {
handler (val) {
if (this.init) {
return;
}
// let func = createFlow;
// if (this.realId) {
// func = (data) => updateFlowById(this.realId, data);
// }
const data = deepCopy(val);
const statusList = [];
const transformList = [];
const statusIDList = [];
const transformIDList = [];
for (const arrayValue of data.nodes) {
if (statusIDList.indexOf(arrayValue.valueData.id) === -1) {
statusIDList.push(arrayValue.valueData.id);
} else {
delete arrayValue.valueData.id;
}
arrayValue.valueData.tmpId = arrayValue.id;
statusList.push(arrayValue.valueData);
arrayValue.label = arrayValue.valueData.name;
}
for (const transformValue of data.edges) {
if (transformIDList.indexOf(transformValue.valueData.id) === -1) {
transformIDList.push(transformValue.valueData.id);
} else {
delete transformValue.valueData.id;
}
transformValue.valueData.tmpId = transformValue.id;
transformList.push(transformValue.valueData);
}
const workflowValue = Object.assign({}, data.workflow);
workflowValue.flowchart = data;
const jsonData = {
workflow: workflowValue,
status: statusList,
transform: transformList,
}
if (this.realId === null) {
createFlow(jsonData).then((res) => {
this.realId = res;
this.$router.replace({ name: 'WorkProcess' });
this.$message({
type: 'success',
message: '保存流程成功',
});
}, err => {
this.$message({
type: 'error',
message: err.errmsg,
});
console.error(err);
});
} else {
jsonData.workflow.id = Number(this.realId);
updateFlowById(Number(this.realId), jsonData).then(() => {
this.$router.replace({ name: 'WorkProcess' });
this.$message({
type: 'success',
message: '保存流程成功',
});
}, err => {
this.$message({
type: 'error',
message: err.errmsg,
});
console.error(err);
});
}
}
},
},
methods: {
// 子组件发过来的值($emit) 将发过来的值进行处理改变为后端需要的数据格式然后发给后端
// 看业务需求。可以不进行处理直接发给后端,后端只是保存数据的作用,编辑的时候再请求过来直接传给子组件赋值就可以
handleSave (val) {
const data = deepCopy(val);
const statusList = [];
const transformList = [];
const statusIDList = [];
const transformIDList = [];
for (const arrayValue of data.nodes) {
if (statusIDList.indexOf(arrayValue.valueData.id) === -1) {
statusIDList.push(arrayValue.valueData.id);
} else {
delete arrayValue.valueData.id;
}
arrayValue.valueData.tmpId = arrayValue.id;
statusList.push(arrayValue.valueData);
arrayValue.label = arrayValue.valueData.name;
}
for (const transformValue of data.edges) {
if (transformIDList.indexOf(transformValue.valueData.id) === -1) {
transformIDList.push(transformValue.valueData.id);
} else {
delete transformValue.valueData.id;
}
transformValue.valueData.tmpId = transformValue.id;
transformList.push(transformValue.valueData);
}
const workflowValue = Object.assign({}, data.workflow);
workflowValue.flowchart = data;
const jsonData = {
workflow: workflowValue,
status: statusList,
transform: transformList,
}
if (this.realId === null) {
createFlow(jsonData).then((res) => {
this.realId = res;
this.$router.replace({ name: 'WorkProcess' });
this.$message({
type: 'success',
message: '保存流程成功',
});
}, err => {
this.$message({
type: 'error',
message: err.errmsg,
});
console.error(err);
});
} else {
jsonData.workflow.id = Number(this.realId);
updateFlowById(Number(this.realId), jsonData).then(() => {
this.$router.replace({ name: 'WorkProcess' });
this.$message({
type: 'success',
message: '保存流程成功',
});
}, err => {
this.$message({
type: 'error',
message: err.errmsg,
});
console.error(err);
});
}
}
},
}
</script>
<style lang="less" scoped>
.staff-title {
display: block;
width: 100%;
height: 40px;
padding: 20px 0 0 30px;
font-size: 16px;
margin-top: 20px;
border-bottom: 1px solid #ccc;
margin-bottom: 30px;
background-color: skyblue;
}
</style>
深拷贝也贴上来吧
// 深拷贝
export function deepCopy(obj, cache = []) {
// just return if obj is immutable value
if (obj === null || typeof obj !== 'object') {
return obj
}
// if obj is hit, it is in circular structure
const hit = find(cache, (c) => c.original === obj)
if (hit) {
return hit.copy
}
const copy = Array.isArray(obj) ? [] : {}
// put the copy into cache at first
// because we want to refer it in recursive deepCopy
cache.push({
original: obj,
copy
})
Object.keys(obj).forEach((key) => {
copy[key] = deepCopy(obj[key], cache)
})
return copy
}
更多推荐
已为社区贡献2条内容
所有评论(0)