123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580 |
- <template>
- <div v-loading="loading" class="layout-padding w100 h100">
- <div class="layout-padding-auto layout-padding-view pd20">
- <!-- 外部内容表单 -->
- <el-form :model="form" label-width="80px" ref="ruleFormRef">
- <el-row :gutter="100">
- <el-col :xs="24" :sm="12" :md="12" :lg="8" :xl="8">
- <el-form-item label="模板名称" prop="name" :rules="[{ required: true, message: '请填写模板名称', trigger: 'blur' }]">
- <el-input v-model="form.name" placeholder="请填写模板名称" clearable></el-input>
- </el-form-item>
- </el-col>
- <el-col :xs="24" :sm="12" :md="12" :lg="8" :xl="8">
- <el-form-item label="模板编码" prop="code" :rules="[{ required: true, message: '请填写模板编码', trigger: 'blur' }]">
- <el-input v-model="form.code" placeholder="请填写模板编码" clearable></el-input>
- </el-form-item>
- </el-col>
- <el-col :xs="24" :sm="12" :md="12" :lg="8" :xl="8">
- <el-form-item label="模板类型" prop="flowType" :rules="[{ required: true, message: '请选择模板类型', trigger: 'change' }]">
- <el-select v-model="form.flowType" class="w100" placeholder="请选择模板类型">
- <el-option v-for="item in baseDataResult.flowTypeOptions" :key="item.key" :label="item.value" :value="item.key" />
- </el-select>
- </el-form-item>
- </el-col>
- <el-col :xs="24" :sm="12" :md="12" :lg="8" :xl="8">
- <el-form-item label="展示主办" prop="isMainHandlerShow" :rules="[{ required: false, message: '请选择模板类型', trigger: 'change' }]">
- <el-switch v-model="form.isMainHandlerShow" inline-prompt active-text="展示" inactive-text="隐藏" />
- </el-form-item>
- </el-col>
- <el-col :xs="24" :sm="12" :md="12" :lg="8" :xl="8">
- <el-form-item label="模板描述" prop="description" :rules="[{ required: false, message: '请填写模板描述', trigger: 'blur' }]">
- <el-input v-model="form.description" placeholder="请填写模板描述" clearable></el-input>
- </el-form-item>
- </el-col>
- </el-row>
- </el-form>
- <!-- 流程图画布内容 -->
- <div class="border w100 h100" style="position: relative; flex: 1">
- <div class="w100 h100" ref="lfElRef"></div>
- </div>
- </div>
- <!-- 节点内容弹窗 -->
- <PropertySetting
- ref="propertySettingRef"
- v-model="formData"
- @change="handlePropertyChange"
- @changeOther="handlePropertyChangeOther"
- :baseData="baseDataResult"
- :nodes="nodes"
- />
- </div>
- </template>
- <script setup lang="ts" name="hotlineFlowDesigner">
- import { defineAsyncComponent, nextTick, onMounted, reactive, Ref, ref, unref, watch } from 'vue';
- import { useRoute, useRouter } from 'vue-router';
- import LogicFlow from '@logicflow/core';
- import { Control, DndPanel, Group, InsertNodeInPolyline, Menu, MiniMap, SelectionSelect, Snapshot } from '@logicflow/extension';
- // import '@logicflow/core/dist/style/index.css';
- import '@logicflow/core/lib/style/index.css';
- import '@logicflow/extension/lib/style/index.css';
- import { SnakerFlowAdapter, SnakerFlowElement } from './snakerflow/index';
- import { NodeTypeEnum } from './enums';
- import { baseData, workflowAdd, workflowUpdate } from '@/api/system/workflow';
- import mittBus from '@/utils/mitt';
- import { throttle } from '@/utils/tools';
- import { ElMessage, FormInstance } from 'element-plus';
- // 引入组件
- const PropertySetting = defineAsyncComponent(() => import('./PropertySetting/index.vue')); // 节点属性设置
- const route = useRoute(); // 当前路由信息
- const router = useRouter(); // 路由实例
- const emits = defineEmits(['update:modelValue', 'on-save']); // 定义组件事件
- // 提交表单数据
- let formData = reactive({} as any);
- // 外层表单
- let form = reactive<Record<string, any>>({});
- const ruleFormRef = ref<RefType>();
- // 定义组件接收的参数
- const props = defineProps({
- modelValue: {
- type: [Object, String],
- },
- config: {
- type: Object,
- default() {
- return {
- grid: {
- size: 20,
- visible: true,
- type: 'dot',
- config: {
- color: '#ababab',
- thickness: 1,
- },
- },
- };
- },
- },
- highLight: {
- // 高亮数据
- type: Object,
- default() {
- return {};
- },
- },
- viewer: {
- // 预览模式
- type: Boolean,
- default: false,
- },
- loading: {
- //加载状态
- type: Boolean,
- default: false,
- },
- extendAttrConfig: {
- // 扩展属性配置
- type: Object,
- default: () => {},
- },
- });
- // 监听流程变化
- watch(
- () => props.modelValue,
- () => {
- reRender(props.modelValue);
- },
- {
- deep: true,
- }
- );
- // 初始化
- // 定义挂载元素Ref
- const lfElRef: Ref = ref(null);
- // 定义LogicFlow实例
- const lfInstance = ref(null) as Ref<LogicFlow | null>;
- const init = async () => {
- // 画布配置
- LogicFlow.use(Snapshot);
- LogicFlow.use(DndPanel);
- LogicFlow.use(SelectionSelect);
- LogicFlow.use(Menu);
- LogicFlow.use(Control);
- LogicFlow.use(Group);
- LogicFlow.use(InsertNodeInPolyline);
- LogicFlow.use(MiniMap);
- LogicFlow.use(SnakerFlowElement);
- LogicFlow.use(SnakerFlowAdapter);
- const defaultConfig: any = {};
- lfInstance.value = new LogicFlow({
- container: unref(lfElRef),
- stopScrollGraph: true, // 进禁止鼠标滚动移动画布
- stopZoomGraph: false, // 禁止缩放画布
- autoExpand: false, // 节点拖动靠近画布边缘时是否自动扩充画布,
- keyboard: {
- // 键盘事件
- enabled: true,
- },
- ...props.config,
- ...defaultConfig,
- plugins: [MiniMap],
- pluginsOptions: {
- miniMap: {
- ...miniMapOptions,
- showEdge: true,
- },
- },
- });
- // 初始化操作
- initOp();
- reRender(props.modelValue as any);
- // 初始化事件
- initEvent();
- };
- const miniMapOptions: MiniMap.MiniMapOption = {
- isShowHeader: false,
- isShowCloseIcon: true,
- headerTitle: 'MiniMap',
- width: 200,
- height: 120,
- // leftPosition: 100,
- // topPosition: 100,
- };
- // 初始化操作
- const initOp = () => {
- const lf: any = unref(lfInstance);
- if (!lf) return;
- if (props.viewer) {
- // 预览模式时
- lf.extension.menu.setMenuConfig({
- nodeMenu: [],
- edgeMenu: [],
- });
- // 删除上一步
- lf.extension.control.removeItem('undo');
- // 删除下一步
- lf.extension.control.removeItem('redo');
- return;
- }
- lf.extension.menu.setMenuConfig({
- nodeMenu: [
- {
- text: '删除',
- callback(node: any) {
- lf.deleteNode(node.id);
- },
- },
- ], // 覆盖默认的节点右键菜单
- edgeMenu: [
- {
- text: '删除',
- callback(node: any) {
- lf.deleteEdge(node.id);
- },
- },
- ], // 删除默认的边右键菜单
- graphMenu: [], // 覆盖默认的边右键菜单,与false表现一样
- });
- // 控制面板-清空画布
- lf.extension.control.addItem({
- iconClass: 'lf-control-clear',
- title: '清空当前流程图',
- text: '清空',
- onClick: () => {
- lf.clearData();
- },
- });
- // 导航图
- lf.extension.control.addItem({
- iconClass: 'custom-minimap',
- title: '打开流程图导航',
- text: '导航',
- // onMouseEnter: (lf:any, ev:any) => {
- // const position = lf.getPointByClient(ev.x, ev.y);
- // lf.extension.miniMap.show(
- // position.domOverlayPosition.x - 120,
- // position.domOverlayPosition.y + 35
- // );
- // },
- onClick: (lf: any, ev: any) => {
- const visible = lf.extension.miniMap.isShow;
- const miniMap = lfInstance.value.extension.miniMap as MiniMap;
- if (visible) {
- miniMap.hide();
- } else {
- miniMap.show();
- }
- /* const position = lf.getPointByClient(ev.x, ev.y);
- lf.extension.miniMap.show(position.domOverlayPosition.x - 120, position.domOverlayPosition.y + 35);*/
- },
- });
- // 控制面板-暂存
- lf.extension.control.addItem({
- iconClass: 'lf-control-save',
- title: '保存',
- text: '保存',
- onClick: () => {
- saveOnly(ruleFormRef.value);
- },
- });
- lf.extension.snapshot.useGlobalRules = false;
- lf.extension.snapshot.customCssRules = `
- .lf-canvas-overlay {
- background: white;
- }
- `;
- // 控制面板-导出
- lf.extension.control.addItem({
- iconClass: 'lf-control-release',
- title: '导出为图片',
- text: '导出',
- onClick: () => {
- const exportName = lf.graphModel.name ? lf.graphModel.name : '流程图';
- lf.getSnapshot(exportName + `${new Date().getTime()}`);
- },
- });
- // 控制面板-发布
- // lf.extension.control.addItem({
- // iconClass: 'lf-control-publish',
- // title: '发布流程模板',
- // text: '发布',
- // onClick: () => {
- // publish(ruleFormRef.value);
- // },
- // });
- // 设置默认边
- lf.setDefaultEdgeType('hotline:transition');
- // 设置拖拽面板
- lf.extension.dndPanel.setPatternItems([
- {
- label: '选区',
- icon: '',
- callback: () => {
- lf.extension.selectionSelect.openSelectionSelect();
- lf.once('selection:selected', () => {
- lf.extension.selectionSelect.closeSelectionSelect();
- });
- },
- },
- {
- type: 'hotline:start',
- text: '开始',
- label: '开始节点',
- properties: {
- stepType: 1, // 节点类型(开始1 结束2)
- },
- icon: '',
- },
- {
- type: 'hotline:task',
- text: '流程节点',
- label: '流程节点',
- properties: {},
- icon: '',
- className: 'important-node',
- },
- {
- type: 'hotline:end',
- text: '结束',
- label: '结束节点',
- properties: {
- stepType: 2, // 节点类型(开始1 结束2)
- },
- icon: '',
- },
- ]);
- };
- // 初始化事件
- // 定义表单信息
- const propertySettingRef = ref(null) as any;
- // 当前操作节点/或边id
- const currentOpId = ref('');
- const nodes = ref<EmptyArrayType>([]);
- const initEvent = () => {
- if (props.viewer) return;
- // 初始化事件
- const lf = unref(lfInstance);
- if (!lf) return;
- const eventCenter: any = lf.graphModel.eventCenter;
- // 空白区右键事件-弹出流程属性表单
- // eventCenter.on('blank:contextmenu', () => {
- // propertySettingRef.value.show({
- // name: lf.graphModel.name,
- // code: lf.graphModel.code,
- // version: lf.graphModel.version,
- // description: lf.graphModel.description,
- // moduleName: lf.graphModel.moduleName,
- // type: 'process'
- // })
- // })
- // 节点点击事件
- eventCenter.on('node:click', (args: any, data) => {
- if (['hotline:start', 'hotline:task', 'hotline:end'].includes(args.data.type)) {
- currentOpId.value = args.data.id;
- propertySettingRef.value.show({
- ...args.data.properties,
- code: args.data.id,
- id: args.data.id,
- name: args.data.text?.value,
- type: args.data.type,
- });
- }
- nodes.value = lf.graphModel.nodes;
- });
- // 边点击事件
- // eventCenter.on('edge:click', (args: any) => {
- // console.log(args, 'args')
- // currentOpId.value = args.data.id
- // propertySettingRef.value.show({
- // ...args.data.properties,
- // id: args.data.id,
- // name: args.data.text?.value,
- // type: args.data.type
- // })
- // })
- };
- const closePage = () => {
- // 更新
- ElMessage.success('操作成功');
- // 关闭当前 tagsView
- mittBus.emit('onCurrentContextmenuClick', Object.assign({}, { contextMenuClickId: 1, ...route }));
- mittBus.emit('clearCache', 'systemWorkflow');
- router.push({
- name: 'systemWorkflow',
- state: {
- index: '1',
- },
- });
- };
- // 暂存(保存为草稿)
- const saveOnly = throttle((formEl: FormInstance | undefined) => {
- // lfInstance.value?.getSnapshot(); // 下载为图片
- const lf = unref(lfInstance);
- if (!lf) return;
- if (!formEl) return;
- // 表单验证
- formEl.validate((valid: boolean) => {
- if (!valid) return;
- // 流程模板属性 最外层
- Object.keys(form).forEach((key: string) => {
- // 监听属性变化 并保存
- lf.graphModel[key] = form[key];
- });
- const { submitData } = getGraphData();
- if (submitData.error) {
- //错误提示
- ElMessage.warning(submitData.error);
- return;
- }
- if (route.params.id) {
- workflowUpdate(submitData).then(() => {
- //更新
- closePage();
- });
- } else {
- workflowAdd(submitData).then(() => {
- //保存
- closePage();
- });
- }
- emits('on-save', getGraphData());
- });
- }, 300);
- // 重新渲染
- const reRender = (data: any): void => {
- const lf = unref(lfInstance);
- if (!lf) return;
- lf.render(data);
- if (route.params.id) {
- nextTick(() => {
- // 最外层扩展属性赋值
- form.name = lf.graphModel.name;
- form.code = lf.graphModel.code;
- form.flowType = lf.graphModel.flowType;
- form.description = lf.graphModel.description;
- form.isMainHandlerShow = lf.graphModel.isMainHandlerShow;
- form.id = route.params.id;
- form.name = lf.graphModel.name;
- });
- }
- };
- // 处理当前节点属性值变化事件
- const handlePropertyChange = (e: any) => {
- const lf = unref(lfInstance);
- if (!lf) return;
- if (([NodeTypeEnum.task, NodeTypeEnum.start, NodeTypeEnum.end] as NodeTypeEnum[]).includes(e.type)) {
- // 节点属性
- const nodeId = unref(currentOpId);
- // 节点信息
- if (e.propertyName === 'id') {
- // 更新唯一标识
- if (!lf.getNodeModelById(e.propertyValue)) {
- lf.changeNodeId(nodeId, e.propertyValue);
- currentOpId.value = e.propertyValue;
- }
- } else if (e.propertyName === 'name') {
- // 更新节点文本值
- lf.updateText(nodeId, e.propertyValue);
- } else {
- // 更新基础属性
- lf.setProperties(nodeId, {
- [e.propertyName]: e.propertyValue,
- });
- }
- emits('update:modelValue', getGraphData());
- }
- };
- /* 处理其他节点的属性变化 */
- const handlePropertyChangeOther = (e: any) => {
- const lf = unref(lfInstance);
- if (!lf) return;
- if (([NodeTypeEnum.task, NodeTypeEnum.start, NodeTypeEnum.end] as NodeTypeEnum[]).includes(e.type)) {
- // 节点属性
- const nodeId = unref(e.id);
- // 节点信息
- if (e.propertyName === 'id') {
- // 更新唯一标识
- if (!lf.getNodeModelById(e.propertyValue)) {
- lf.changeNodeId(nodeId, e.propertyValue);
- currentOpId.value = e.propertyValue;
- }
- } else if (e.propertyName === 'name') {
- // 更新节点文本值
- lf.updateText(nodeId, e.propertyValue);
- } else {
- // 更新基础属性
- lf.setProperties(nodeId, {
- [e.propertyName]: e.propertyValue,
- });
- }
- emits('update:modelValue', getGraphData());
- }
- };
- /**
- * 获取流程数据
- */
- const getGraphData = () => {
- const lf = unref(lfInstance);
- if (!lf) return {};
- return lf.getGraphData();
- };
- // 刷新导入下拉数据
- const refreshImport = () => {
- // return unref(importDataRef)?.refresh()
- };
- // 导入json
- const importJson = (data: any) => {
- reRender(data);
- };
- const baseDataResult = ref<EmptyObjectType>({});
- onMounted(async () => {
- // 获取页面基础数据
- const { result } = await baseData();
- baseDataResult.value = result;
- await init();
- });
- // 出提供给外部操作-$refs.xxx
- defineExpose({
- getGraphData,
- refreshImport,
- importJson,
- });
- </script>
- <style lang="scss">
- .border {
- border: var(--el-border);
- border-radius: var(--el-border-radius-base);
- padding: 10px;
- }
- .el-step__icon-inner {
- font-size: var(--el-font-size-base) !important;
- font-weight: 700 !important;
- }
- .lf-menu {
- background-color: var(--el-color-white);
- }
- .lf-menu-item:hover {
- background-color: var(--hotline-color-menu-hover);
- }
- .lf-text-input {
- background-color: var(--el-color-white);
- }
- .lf-element-text {
- color: var(--hotline-color-text-main);
- }
- .lf-graph {
- background-color: var(--el-color-white) !important;
- color: var(--hotline-color-text-main);
- }
- .lf-control {
- background-color: var(--el-color-white) !important;
- }
- .custom-minimap {
- background-image: url('');
- }
- .lf-control-clear {
- background-image: url('');
- }
- .lf-control-save {
- background-image: url('');
- }
- .lf-control-release {
- background-image: url('');
- }
- </style>
|