ProcessInstanceTimeline.vue 8.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228
  1. <!-- 审批详情的右侧:审批流 -->
  2. <template>
  3. <el-timeline class="pt-20px">
  4. <!-- 遍历每个审批节点 -->
  5. <el-timeline-item
  6. v-for="(activity, index) in approveNodes"
  7. :key="index"
  8. size="large"
  9. :icon="getApprovalNodeIcon(activity.status, activity.nodeType)"
  10. :color="getApprovalNodeColor(activity.status)"
  11. >
  12. <template #dot>
  13. <div
  14. class="position-absolute left--10px top--6px rounded-full border border-solid border-#dedede w-30px h-30px flex justify-center items-center bg-#3f73f7 p-5px"
  15. >
  16. <img class="w-full h-full" :src="getApprovalNodeImg(activity.nodeType)" alt="" />
  17. <div
  18. class="position-absolute top-17px left-17px bg-#fff rounded-full flex items-center p-2px"
  19. >
  20. <el-icon :size="12" :color="getApprovalNodeColor(activity.status)">
  21. <component :is="getApprovalNodeIcon(activity.status, activity.nodeType)" />
  22. </el-icon>
  23. </div>
  24. </div>
  25. </template>
  26. <div class="flex flex-col items-start">
  27. <!-- 第一行:节点名称、时间 -->
  28. <div class="flex w-full">
  29. <div class="font-bold"> {{ activity.name }}</div>
  30. <!-- 信息:时间 -->
  31. <div
  32. v-if="activity.status !== TaskStatusEnum.NOT_START"
  33. class="text-#a5a5a5 text-13px mt-1 ml-auto"
  34. >
  35. {{ getApprovalNodeTime(activity) }}
  36. </div>
  37. </div>
  38. <div class="flex items-center flex-wrap mt-1">
  39. <!-- 情况一:遍历每个审批节点下的【进行中】task 任务 -->
  40. <div v-for="(task, idx) in activity.tasks" :key="idx" class="flex items-center">
  41. <div class="flex flex-col pr-2 gap2">
  42. <div
  43. class="position-relative flex flex-wrap gap2"
  44. v-if="task.assigneeUser || task.ownerUser"
  45. >
  46. <!-- 信息:头像昵称 -->
  47. <div
  48. class="bg-gray-100 h-35px rounded-3xl flex items-center p-8px gap-2 dark:color-gray-600 position-relative"
  49. >
  50. <template v-if="task.assigneeUser?.avatar || task.assigneeUser?.nickname">
  51. <el-avatar
  52. :size="28"
  53. v-if="task.assigneeUser?.avatar"
  54. :src="task.assigneeUser?.avatar"
  55. />
  56. <el-avatar :size="28" v-else>
  57. {{ task.assigneeUser?.nickname.substring(0, 1) }}
  58. </el-avatar>
  59. {{ task.assigneeUser?.nickname }}
  60. </template>
  61. <template v-else-if="task.ownerUser?.avatar || task.ownerUser?.nickname">
  62. <el-avatar
  63. :size="28"
  64. v-if="task.ownerUser?.avatar"
  65. :src="task.ownerUser?.avatar"
  66. />
  67. <el-avatar :size="28" v-else>
  68. {{ task.ownerUser?.nickname.substring(0, 1) }}
  69. </el-avatar>
  70. {{ task.ownerUser?.nickname }}
  71. </template>
  72. <!-- 信息:任务 ICON -->
  73. <div
  74. v-if="onlyStatusIconShow.includes(task.status)"
  75. class="position-absolute top-22px left-26px bg-#fff rounded-full flex items-center p-2px"
  76. >
  77. <Icon
  78. :size="12"
  79. :icon="statusIconMap2[task.status]?.icon"
  80. :color="statusIconMap2[task.status]?.color"
  81. />
  82. </div>
  83. </div>
  84. <!-- 情况二:遍历每个审批节点下的【候选的】task 任务。例如说,1)依次审批,2)未来的审批任务等 -->
  85. <div
  86. v-for="(user, idx1) in activity.candidateUsers"
  87. :key="idx1"
  88. class="bg-gray-100 h-35px rounded-3xl flex items-center p-8px gap-2 dark:color-gray-600 position-relative"
  89. >
  90. <el-avatar :size="28" v-if="user.avatar" :src="user.avatar" />
  91. <el-avatar :size="28" v-else>
  92. {{ user.nickname.substring(0, 1) }}
  93. </el-avatar>
  94. {{ user.nickname }}
  95. <!-- 信息:任务 ICON -->
  96. <div
  97. v-if="onlyStatusIconShow.includes(task.status)"
  98. class="position-absolute top-22px left-26px bg-#fff rounded-full flex items-center p-2px"
  99. >
  100. <Icon
  101. :size="12"
  102. :icon="statusIconMap2['-1']?.icon"
  103. :color="statusIconMap2['-1']?.color"
  104. />
  105. </div>
  106. </div>
  107. </div>
  108. <div
  109. v-if="task.reason && activity.nodeType === NodeType.USER_TASK_NODE"
  110. class="text-#a5a5a5 text-13px mt-1 w-full bg-#f8f8fa p2 rounded-md"
  111. >
  112. 审批意见:{{ task.reason }}
  113. </div>
  114. </div>
  115. </div>
  116. </div>
  117. </div>
  118. </el-timeline-item>
  119. </el-timeline>
  120. </template>
  121. <script lang="ts" setup>
  122. import { formatDate } from '@/utils/formatTime'
  123. import * as ProcessInstanceApi from '@/api/bpm/processInstance'
  124. import { TaskStatusEnum } from '@/api/bpm/task'
  125. import { NodeType } from '@/components/SimpleProcessDesignerV2/src/consts'
  126. import { Check, Close, Loading, Clock, Minus, Delete } from '@element-plus/icons-vue'
  127. import starterSvg from '@/assets/svgs/bpm/starter.svg'
  128. import auditorSvg from '@/assets/svgs/bpm/auditor.svg'
  129. import copySvg from '@/assets/svgs/bpm/copy.svg'
  130. import conditionSvg from '@/assets/svgs/bpm/condition.svg'
  131. import parallelSvg from '@/assets/svgs/bpm/parallel.svg'
  132. defineOptions({ name: 'BpmProcessInstanceTimeline' })
  133. defineProps<{
  134. approveNodes: ProcessInstanceApi.ApprovalNodeInfo[] // 审批节点信息
  135. }>()
  136. // 审批节点
  137. const statusIconMap2 = {
  138. // 未开始
  139. '-1': { color: '#909398', icon: 'ep-clock' },
  140. // 待审批
  141. '0': { color: '#e5e7ec', icon: 'ep:loading' },
  142. // 审批中
  143. '1': { color: '#448ef7', icon: 'ep:loading' },
  144. // 审批通过
  145. '2': { color: '#00b32a', icon: 'ep:circle-check-filled' },
  146. // 审批不通过
  147. '3': { color: '#f46b6c', icon: 'fa-solid:times-circle' },
  148. // 取消
  149. '4': { color: '#cccccc', icon: 'ep:delete-filled' },
  150. // 退回
  151. '5': { color: '#f46b6c', icon: 'ep:remove-filled' },
  152. // 委派中
  153. '6': { color: '#448ef7', icon: 'ep:loading' },
  154. // 审批通过中
  155. '7': { color: '#00b32a', icon: 'ep:circle-check-filled' }
  156. }
  157. const statusIconMap = {
  158. // 审批未开始
  159. '-1': { color: '#909398', icon: Clock },
  160. '0': { color: '#e5e7ec', icon: Clock },
  161. // 审批中
  162. '1': { color: '#448ef7', icon: Loading },
  163. // 审批通过
  164. '2': { color: '#00b32a', icon: Check },
  165. // 审批不通过
  166. '3': { color: '#f46b6c', icon: Close },
  167. // 已取消
  168. '4': { color: '#cccccc', icon: Delete },
  169. // 退回
  170. '5': { color: '#f46b6c', icon: Minus },
  171. // 委派中
  172. '6': { color: '#448ef7', icon: Loading },
  173. // 审批通过中
  174. '7': { color: '#00b32a', icon: Check }
  175. }
  176. const nodeTypeSvgMap = {
  177. // 发起人节点
  178. [NodeType.START_USER_NODE]: { color: '#ffffff', svg: starterSvg },
  179. // 审批人节点
  180. [NodeType.USER_TASK_NODE]: { color: '#ff943e', svg: auditorSvg },
  181. // 抄送人节点
  182. [NodeType.COPY_TASK_NODE]: { color: '#3296fb', svg: copySvg },
  183. // 条件分支节点
  184. [NodeType.CONDITION_NODE]: { color: '#14bb83', svg: conditionSvg },
  185. // 并行分支节点
  186. [NodeType.PARALLEL_BRANCH_NODE]: { color: '#14bb83', svg: parallelSvg }
  187. }
  188. // 只有只有状态是 -1、0、1 才展示头像右小角状态小icon
  189. const onlyStatusIconShow = [-1, 0, 1]
  190. // timeline时间线上icon图标
  191. const getApprovalNodeImg = (nodeType: NodeType) => {
  192. return nodeTypeSvgMap[nodeType]?.svg
  193. }
  194. const getApprovalNodeIcon = (taskStatus: number, nodeType: NodeType) => {
  195. if (taskStatus == TaskStatusEnum.NOT_START) {
  196. return statusIconMap[taskStatus]?.icon
  197. }
  198. if (nodeType === NodeType.START_USER_NODE || nodeType === NodeType.USER_TASK_NODE) {
  199. return statusIconMap[taskStatus]?.icon
  200. }
  201. }
  202. const getApprovalNodeColor = (taskStatus: number) => {
  203. return statusIconMap[taskStatus]?.color
  204. }
  205. const getApprovalNodeTime = (node: ProcessInstanceApi.ApprovalNodeInfo) => {
  206. if (node.nodeType === NodeType.START_USER_NODE && node.startTime) {
  207. return `${formatDate(node.startTime)}`
  208. }
  209. if (node.endTime) {
  210. return `${formatDate(node.endTime)}`
  211. }
  212. if (node.startTime) {
  213. return `${formatDate(node.startTime)}`
  214. }
  215. }
  216. </script>