ParallelNode.vue 5.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184
  1. <template>
  2. <div class="branch-node-wrapper">
  3. <div class="branch-node-container">
  4. <div
  5. v-if="readonly"
  6. class="branch-node-readonly"
  7. :class="`${useTaskStatusClass(currentNode?.activityStatus)}`"
  8. >
  9. <span class="iconfont icon-parallel icon-size parallel"></span>
  10. </div>
  11. <el-button v-else class="branch-node-add" color="#626aef" @click="addCondition" plain
  12. >添加分支</el-button
  13. >
  14. <div
  15. class="branch-node-item"
  16. v-for="(item, index) in currentNode.conditionNodes"
  17. :key="index"
  18. >
  19. <template v-if="index == 0">
  20. <div class="branch-line-first-top"></div>
  21. <div class="branch-line-first-bottom"></div>
  22. </template>
  23. <template v-if="index + 1 == currentNode.conditionNodes?.length">
  24. <div class="branch-line-last-top"></div>
  25. <div class="branch-line-last-bottom"></div>
  26. </template>
  27. <div class="node-wrapper">
  28. <div class="node-container">
  29. <div class="node-box" :class="`${useTaskStatusClass(item.activityStatus)}`">
  30. <div class="branch-node-title-container">
  31. <div v-if="showInputs[index]">
  32. <input
  33. type="text"
  34. class="input-max-width editable-title-input"
  35. @blur="blurEvent(index)"
  36. v-mountedFocus
  37. v-model="item.name"
  38. />
  39. </div>
  40. <div v-else class="branch-title" @click="clickEvent(index)"> {{ item.name }} </div>
  41. <div class="branch-priority">无优先级</div>
  42. </div>
  43. <div class="branch-node-content" @click="conditionNodeConfig(item.id)">
  44. <div class="branch-node-text" :title="item.showText" v-if="item.showText">
  45. {{ item.showText }}
  46. </div>
  47. <div class="branch-node-text" v-else>
  48. {{ NODE_DEFAULT_TEXT.get(NodeType.CONDITION_NODE) }}
  49. </div>
  50. </div>
  51. <div v-if="!readonly" class="node-toolbar">
  52. <div class="toolbar-icon">
  53. <Icon
  54. color="#0089ff"
  55. icon="ep:circle-close-filled"
  56. :size="18"
  57. @click="deleteCondition(index)"
  58. />
  59. </div>
  60. </div>
  61. </div>
  62. <NodeHandler v-model:child-node="item.childNode" :current-node="item" />
  63. </div>
  64. </div>
  65. <!-- 递归显示子节点 -->
  66. <ProcessNodeTree
  67. v-if="item && item.childNode"
  68. :parent-node="item"
  69. v-model:flow-node="item.childNode"
  70. @find:recursive-find-parent-node="recursiveFindParentNode"
  71. />
  72. </div>
  73. </div>
  74. <NodeHandler
  75. v-if="currentNode"
  76. v-model:child-node="currentNode.childNode"
  77. :current-node="currentNode"
  78. />
  79. </div>
  80. </template>
  81. <script setup lang="ts">
  82. import NodeHandler from '../NodeHandler.vue'
  83. import ProcessNodeTree from '../ProcessNodeTree.vue'
  84. import { SimpleFlowNode, NodeType, NODE_DEFAULT_TEXT } from '../consts'
  85. import { useTaskStatusClass } from '../node'
  86. import { generateUUID } from '@/utils'
  87. const { proxy } = getCurrentInstance() as any
  88. defineOptions({
  89. name: 'ParallelNode'
  90. })
  91. const props = defineProps({
  92. flowNode: {
  93. type: Object as () => SimpleFlowNode,
  94. required: true
  95. }
  96. })
  97. // 定义事件,更新父组件
  98. const emits = defineEmits<{
  99. 'update:modelValue': [node: SimpleFlowNode | undefined]
  100. 'find:parentNode': [nodeList: SimpleFlowNode[], nodeType: number]
  101. 'find:recursiveFindParentNode': [
  102. nodeList: SimpleFlowNode[],
  103. curentNode: SimpleFlowNode,
  104. nodeType: number
  105. ]
  106. }>()
  107. const currentNode = ref<SimpleFlowNode>(props.flowNode)
  108. // 是否只读
  109. const readonly = inject<Boolean>('readonly')
  110. watch(
  111. () => props.flowNode,
  112. (newValue) => {
  113. currentNode.value = newValue
  114. }
  115. )
  116. const showInputs = ref<boolean[]>([])
  117. // 失去焦点
  118. const blurEvent = (index: number) => {
  119. showInputs.value[index] = false
  120. const conditionNode = currentNode.value.conditionNodes?.at(index) as SimpleFlowNode
  121. conditionNode.name = conditionNode.name || `并行${index + 1}`
  122. }
  123. // 点击条件名称
  124. const clickEvent = (index: number) => {
  125. showInputs.value[index] = true
  126. }
  127. const conditionNodeConfig = (nodeId: string) => {
  128. const conditionNode = proxy.$refs[nodeId][0]
  129. conditionNode.open()
  130. }
  131. // 新增条件
  132. const addCondition = () => {
  133. const conditionNodes = currentNode.value.conditionNodes
  134. if (conditionNodes) {
  135. const len = conditionNodes.length
  136. let lastIndex = len - 1
  137. const conditionData: SimpleFlowNode = {
  138. id: 'Flow_' + generateUUID(),
  139. name: '并行' + len,
  140. showText: '无需配置条件同时执行',
  141. type: NodeType.CONDITION_NODE,
  142. childNode: undefined,
  143. conditionNodes: []
  144. }
  145. conditionNodes.splice(lastIndex, 0, conditionData)
  146. }
  147. }
  148. // 删除条件
  149. const deleteCondition = (index: number) => {
  150. const conditionNodes = currentNode.value.conditionNodes
  151. if (conditionNodes) {
  152. conditionNodes.splice(index, 1)
  153. if (conditionNodes.length == 1) {
  154. const childNode = currentNode.value.childNode
  155. // 更新此节点为后续孩子节点
  156. emits('update:modelValue', childNode)
  157. }
  158. }
  159. }
  160. // 递归从父节点中查询匹配的节点
  161. const recursiveFindParentNode = (
  162. nodeList: SimpleFlowNode[],
  163. node: SimpleFlowNode,
  164. nodeType: number
  165. ) => {
  166. if (!node || node.type === NodeType.START_USER_NODE) {
  167. return
  168. }
  169. if (node.type === nodeType) {
  170. nodeList.push(node)
  171. }
  172. // 条件节点 (NodeType.CONDITION_NODE) 比较特殊。需要调用其父节点并行节点(NodeType.PARALLEL_NODE) 继续查找
  173. emits('find:parentNode', nodeList, nodeType)
  174. }
  175. </script>