feat: [BPM 工作流] Simple 模型 - 包容节点

pull/123/head
jason 2025-05-31 08:33:34 +08:00
parent a2832f1546
commit 1c2b247cf4
4 changed files with 313 additions and 4 deletions

View File

@ -0,0 +1,282 @@
<script setup lang="ts">
import type { SimpleFlowNode } from '../../consts';
import { getCurrentInstance, inject, ref, watch } from 'vue';
import { IconifyIcon } from '@vben/icons';
import { cloneDeep, buildShortUUID as generateUUID } from '@vben/utils';
import { Button, Input } from 'ant-design-vue';
import {
ConditionType,
DEFAULT_CONDITION_GROUP_VALUE,
NODE_DEFAULT_TEXT,
NodeType,
} from '../../consts';
import {
getDefaultInclusiveConditionNodeName,
useTaskStatusClass,
} from '../../helpers';
// import ConditionNodeConfig from '../nodes-config/ConditionNodeConfig.vue';
import ProcessNodeTree from '../process-node-tree.vue';
import NodeHandler from './node-handler.vue';
defineOptions({
name: 'InclusiveNode',
});
const props = defineProps({
flowNode: {
type: Object as () => SimpleFlowNode,
required: true,
},
});
//
const emits = defineEmits<{
findParentNode: [nodeList: SimpleFlowNode[], nodeType: number];
recursiveFindParentNode: [
nodeList: SimpleFlowNode[],
curentNode: SimpleFlowNode,
nodeType: number,
];
'update:modelValue': [node: SimpleFlowNode | undefined];
}>();
const { proxy } = getCurrentInstance() as any;
//
const readonly = inject<Boolean>('readonly');
const currentNode = ref<SimpleFlowNode>(props.flowNode);
watch(
() => props.flowNode,
(newValue) => {
currentNode.value = newValue;
},
);
const showInputs = ref<boolean[]>([]);
//
const blurEvent = (index: number) => {
showInputs.value[index] = false;
const conditionNode = currentNode.value.conditionNodes?.at(
index,
) as SimpleFlowNode;
conditionNode.name =
conditionNode.name ||
getDefaultInclusiveConditionNodeName(
index,
conditionNode.conditionSetting?.defaultFlow,
);
};
//
const clickEvent = (index: number) => {
showInputs.value[index] = true;
};
const conditionNodeConfig = (nodeId: string) => {
if (readonly) {
return;
}
const conditionNode = proxy.$refs[nodeId][0];
conditionNode.open();
};
//
const addCondition = () => {
const conditionNodes = currentNode.value.conditionNodes;
if (conditionNodes) {
const len = conditionNodes.length;
const lastIndex = len - 1;
const conditionData: SimpleFlowNode = {
id: `Flow_${generateUUID()}`,
name: `包容条件${len}`,
showText: '',
type: NodeType.CONDITION_NODE,
childNode: undefined,
conditionNodes: [],
conditionSetting: {
defaultFlow: false,
conditionType: ConditionType.RULE,
conditionGroups: cloneDeep(DEFAULT_CONDITION_GROUP_VALUE),
},
};
conditionNodes.splice(lastIndex, 0, conditionData);
}
};
//
const deleteCondition = (index: number) => {
const conditionNodes = currentNode.value.conditionNodes;
if (conditionNodes) {
conditionNodes.splice(index, 1);
if (conditionNodes.length === 1) {
const childNode = currentNode.value.childNode;
//
emits('update:modelValue', childNode);
}
}
};
//
const moveNode = (index: number, to: number) => {
// -1 1
if (
currentNode.value.conditionNodes &&
currentNode.value.conditionNodes[index]
) {
currentNode.value.conditionNodes[index] =
currentNode.value.conditionNodes.splice(
index + to,
1,
currentNode.value.conditionNodes[index],
)[0] as SimpleFlowNode;
}
};
//
const recursiveFindParentNode = (
nodeList: SimpleFlowNode[],
node: SimpleFlowNode,
nodeType: number,
) => {
if (!node || node.type === NodeType.START_USER_NODE) {
return;
}
if (node.type === nodeType) {
nodeList.push(node);
}
// (NodeType.CONDITION_NODE) NodeType.INCLUSIVE_BRANCH_NODE)
emits('findParentNode', nodeList, nodeType);
};
</script>
<template>
<div class="branch-node-wrapper">
<div class="branch-node-container">
<div
v-if="readonly"
class="branch-node-readonly"
:class="`${useTaskStatusClass(currentNode?.activityStatus)}`"
>
<span class="iconfont icon-inclusive icon-size inclusive"></span>
</div>
<Button v-else class="branch-node-add" @click="addCondition" plain>
添加条件
</Button>
<div
class="branch-node-item"
v-for="(item, index) in currentNode.conditionNodes"
:key="index"
>
<template v-if="index === 0">
<div class="branch-line-first-top"></div>
<div class="branch-line-first-bottom"></div>
</template>
<template v-if="index + 1 === currentNode.conditionNodes?.length">
<div class="branch-line-last-top"></div>
<div class="branch-line-last-bottom"></div>
</template>
<div class="node-wrapper">
<div class="node-container">
<div
class="node-box"
:class="[
{ 'node-config-error': !item.showText },
`${useTaskStatusClass(item.activityStatus)}`,
]"
>
<div class="branch-node-title-container">
<div v-if="!readonly && showInputs[index]">
<Input
type="text"
class="editable-title-input"
@blur="blurEvent(index)"
v-model="item.name"
/>
</div>
<div v-else class="branch-title" @click="clickEvent(index)">
{{ item.name }}
</div>
</div>
<div
class="branch-node-content"
@click="conditionNodeConfig(item.id)"
>
<div
class="branch-node-text"
:title="item.showText"
v-if="item.showText"
>
{{ item.showText }}
</div>
<div class="branch-node-text" v-else>
{{ NODE_DEFAULT_TEXT.get(NodeType.CONDITION_NODE) }}
</div>
</div>
<div
class="node-toolbar"
v-if="
!readonly && index + 1 !== currentNode.conditionNodes?.length
"
>
<div class="toolbar-icon">
<IconifyIcon
color="#0089ff"
icon="ep:circle-close-filled"
:size="18"
@click="deleteCondition(index)"
/>
</div>
</div>
<div
class="branch-node-move move-node-left"
v-if="
!readonly &&
index !== 0 &&
index + 1 !== currentNode.conditionNodes?.length
"
@click="moveNode(index, -1)"
>
<IconifyIcon icon="ep:arrow-left" />
</div>
<div
class="branch-node-move move-node-right"
v-if="
!readonly &&
currentNode.conditionNodes &&
index < currentNode.conditionNodes.length - 2
"
@click="moveNode(index, 1)"
>
<IconifyIcon icon="ep:arrow-right" />
</div>
</div>
<NodeHandler
v-model:child-node="item.childNode"
:current-node="item"
/>
</div>
</div>
<!-- TODO 条件节点配置 -->
<!-- <ConditionNodeConfig
:node-index="index"
:condition-node="item"
:ref="item.id"
/> -->
<!-- 递归显示子节点 -->
<ProcessNodeTree
v-if="item && item.childNode"
:parent-node="item"
v-model:flow-node="item.childNode"
@recursive-find-parent-node="recursiveFindParentNode"
/>
</div>
</div>
<NodeHandler
v-if="currentNode"
v-model:child-node="currentNode.childNode"
:current-node="currentNode"
/>
</div>
</template>
<style lang="scss" scoped></style>

View File

@ -5,6 +5,7 @@ import { NodeType } from '../consts';
import { useWatchNode } from '../helpers'; import { useWatchNode } from '../helpers';
import CopyTaskNode from './nodes/copy-task-node.vue'; import CopyTaskNode from './nodes/copy-task-node.vue';
import EndEventNode from './nodes/end-event-node.vue'; import EndEventNode from './nodes/end-event-node.vue';
import InclusiveNode from './nodes/inclusive-node.vue';
import StartUserNode from './nodes/start-user-node.vue'; import StartUserNode from './nodes/start-user-node.vue';
import TriggerNode from './nodes/trigger-node.vue'; import TriggerNode from './nodes/trigger-node.vue';
import UserTaskNode from './nodes/user-task-node.vue'; import UserTaskNode from './nodes/user-task-node.vue';
@ -99,12 +100,12 @@ const recursiveFindParentNode = (
@find:parent-node="findFromParentNode" @find:parent-node="findFromParentNode"
/> --> /> -->
<!-- 包容分支节点 --> <!-- 包容分支节点 -->
<!-- <InclusiveNode <InclusiveNode
v-if="currentNode && currentNode.type === NodeType.INCLUSIVE_BRANCH_NODE" v-if="currentNode && currentNode.type === NodeType.INCLUSIVE_BRANCH_NODE"
:flow-node="currentNode" :flow-node="currentNode"
@update:model-value="handleModelValueUpdate" @update:model-value="handleModelValueUpdate"
@find:parent-node="findFromParentNode" @find-parent-node="findParentNode"
/> --> />
<!-- 延迟器节点 --> <!-- 延迟器节点 -->
<!-- <DelayTimerNode <!-- <DelayTimerNode
v-if="currentNode && currentNode.type === NodeType.DELAY_TIMER_NODE" v-if="currentNode && currentNode.type === NodeType.DELAY_TIMER_NODE"

View File

@ -737,3 +737,25 @@ const getOpName = (opCode: string): string | undefined => {
); );
return opName?.label; return opName?.label;
}; };
/** 获取条件节点默认的名称 */
export const getDefaultConditionNodeName = (
index: number,
defaultFlow: boolean | undefined,
): string => {
if (defaultFlow) {
return '其它情况';
}
return `条件${index + 1}`;
};
/** 获取包容分支条件节点默认的名称 */
export const getDefaultInclusiveConditionNodeName = (
index: number,
defaultFlow: boolean | undefined,
): string => {
if (defaultFlow) {
return '其它情况';
}
return `包容条件${index + 1}`;
};

View File

@ -334,12 +334,14 @@
margin-top: 4px; margin-top: 4px;
line-height: 32px; line-height: 32px;
color: #111f2c; color: #111f2c;
background: rgb(0 0 0 / 3%);
border-radius: 4px; border-radius: 4px;
.branch-node-text { .branch-node-text {
display: -webkit-box;
overflow: hidden; overflow: hidden;
text-overflow: ellipsis; text-overflow: ellipsis;
-webkit-line-clamp: 2; /* 这将限制文本显示为两行 */ -webkit-line-clamp: 1; /* 这将限制文本显示为一行 */
font-size: 12px; font-size: 12px;
line-height: 24px; line-height: 24px;
word-break: break-all; word-break: break-all;
@ -486,6 +488,8 @@
border-radius: 18px; border-radius: 18px;
transform: translateX(-50%); transform: translateX(-50%);
transform-origin: center center; transform-origin: center center;
display: flex;
align-items: center;
} }
.branch-node-readonly { .branch-node-readonly {