ProcessInstanceOperationButton.vue 33 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048
  1. <template>
  2. <div
  3. class="h-50px bottom-10 text-14px flex items-center color-#32373c dark:color-#fff font-bold btn-container"
  4. >
  5. <!-- 【通过】按钮 -->
  6. <el-popover
  7. :visible="popOverVisible.approve"
  8. placement="top-end"
  9. :width="420"
  10. trigger="click"
  11. v-if="runningTask && isHandleTaskStatus() && isShowButton(OperationButtonType.APPROVE)"
  12. >
  13. <template #reference>
  14. <el-button plain type="success" @click="openPopover('approve')">
  15. <Icon icon="ep:select" />&nbsp; {{ getButtonDisplayName(OperationButtonType.APPROVE) }}
  16. </el-button>
  17. </template>
  18. <!-- 审批表单 -->
  19. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  20. <el-form
  21. label-position="top"
  22. class="mb-auto"
  23. ref="approveFormRef"
  24. :model="approveReasonForm"
  25. :rules="approveReasonRule"
  26. label-width="100px"
  27. >
  28. <el-card v-if="runningTask?.formId > 0" class="mb-15px !-mt-10px">
  29. <template #header>
  30. <span class="el-icon-picture-outline"> 填写表单【{{ runningTask?.formName }}】 </span>
  31. </template>
  32. <form-create
  33. :key="'approve-task-' + (runningTask?.id ?? '') + '-' + (runningTask?.formId ?? '')"
  34. v-model="approveForm.value"
  35. v-model:api="approveFormFApi"
  36. :option="approveForm.option"
  37. :rule="approveForm.rule"
  38. />
  39. </el-card>
  40. <el-form-item label="审批意见" prop="reason">
  41. <el-input
  42. v-model="approveReasonForm.reason"
  43. placeholder="请输入审批意见"
  44. type="textarea"
  45. :rows="4"
  46. />
  47. </el-form-item>
  48. <el-form-item
  49. v-if="runningTask.signEnable"
  50. label="签名"
  51. prop="signPicUrl"
  52. ref="approveSignFormRef"
  53. >
  54. <el-button @click="signRef.open()">点击签名</el-button>
  55. <el-image
  56. class="w-90px h-40px ml-5px"
  57. v-if="approveReasonForm.signPicUrl"
  58. :src="approveReasonForm.signPicUrl"
  59. :preview-src-list="[approveReasonForm.signPicUrl]"
  60. />
  61. </el-form-item>
  62. <el-form-item>
  63. <el-button
  64. :disabled="formLoading"
  65. type="success"
  66. @click="handleAudit(true, approveFormRef)"
  67. >
  68. {{ getButtonDisplayName(OperationButtonType.APPROVE) }}
  69. </el-button>
  70. <el-button @click="closePropover('approve', approveFormRef)"> 取消 </el-button>
  71. </el-form-item>
  72. </el-form>
  73. </div>
  74. </el-popover>
  75. <!-- 【拒绝】按钮 -->
  76. <el-popover
  77. :visible="popOverVisible.reject"
  78. placement="top-end"
  79. :width="420"
  80. trigger="click"
  81. v-if="runningTask && isHandleTaskStatus() && isShowButton(OperationButtonType.REJECT)"
  82. >
  83. <template #reference>
  84. <el-button class="mr-20px" plain type="danger" @click="openPopover('reject')">
  85. <Icon icon="ep:close" />&nbsp; {{ getButtonDisplayName(OperationButtonType.REJECT) }}
  86. </el-button>
  87. </template>
  88. <!-- 审批表单 -->
  89. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  90. <el-form
  91. label-position="top"
  92. class="mb-auto"
  93. ref="rejectFormRef"
  94. :model="rejectReasonForm"
  95. :rules="rejectReasonRule"
  96. label-width="100px"
  97. >
  98. <el-form-item label="审批意见" prop="reason">
  99. <el-input
  100. v-model="rejectReasonForm.reason"
  101. placeholder="请输入审批意见"
  102. type="textarea"
  103. :rows="4"
  104. />
  105. </el-form-item>
  106. <el-form-item>
  107. <el-button
  108. :disabled="formLoading"
  109. type="danger"
  110. @click="handleAudit(false, rejectFormRef)"
  111. >
  112. {{ getButtonDisplayName(OperationButtonType.REJECT) }}
  113. </el-button>
  114. <el-button @click="closePropover('reject', rejectFormRef)"> 取消 </el-button>
  115. </el-form-item>
  116. </el-form>
  117. </div>
  118. </el-popover>
  119. <!-- 【抄送】按钮 -->
  120. <el-popover
  121. :visible="popOverVisible.copy"
  122. placement="top-start"
  123. :width="420"
  124. trigger="click"
  125. v-if="runningTask && isHandleTaskStatus() && isShowButton(OperationButtonType.COPY)"
  126. >
  127. <template #reference>
  128. <div @click="openPopover('copy')" class="hover-bg-gray-100 rounded-xl p-6px">
  129. <Icon :size="14" icon="svg-icon:send" />&nbsp;
  130. {{ getButtonDisplayName(OperationButtonType.COPY) }}
  131. </div>
  132. </template>
  133. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  134. <el-form
  135. label-position="top"
  136. class="mb-auto"
  137. ref="copyFormRef"
  138. :model="copyForm"
  139. :rules="copyFormRule"
  140. label-width="100px"
  141. >
  142. <el-form-item label="抄送人" prop="copyUserIds">
  143. <el-select
  144. v-model="copyForm.copyUserIds"
  145. clearable
  146. style="width: 100%"
  147. multiple
  148. placeholder="请选择抄送人"
  149. >
  150. <el-option
  151. v-for="item in userOptions"
  152. :key="item.id"
  153. :label="item.nickname"
  154. :value="item.id"
  155. />
  156. </el-select>
  157. </el-form-item>
  158. <el-form-item label="抄送意见" prop="copyReason">
  159. <el-input
  160. v-model="copyForm.copyReason"
  161. clearable
  162. placeholder="请输入抄送意见"
  163. type="textarea"
  164. :rows="3"
  165. />
  166. </el-form-item>
  167. <el-form-item>
  168. <el-button :disabled="formLoading" type="primary" @click="handleCopy">
  169. {{ getButtonDisplayName(OperationButtonType.COPY) }}
  170. </el-button>
  171. <el-button @click="closePropover('copy', copyFormRef)"> 取消 </el-button>
  172. </el-form-item>
  173. </el-form>
  174. </div>
  175. </el-popover>
  176. <!-- 【转办】按钮 -->
  177. <el-popover
  178. :visible="popOverVisible.transfer"
  179. placement="top-start"
  180. :width="420"
  181. trigger="click"
  182. v-if="runningTask && isHandleTaskStatus() && isShowButton(OperationButtonType.TRANSFER)"
  183. >
  184. <template #reference>
  185. <div @click="openPopover('transfer')" class="hover-bg-gray-100 rounded-xl p-6px">
  186. <Icon :size="14" icon="fa:share-square-o" />&nbsp;
  187. {{ getButtonDisplayName(OperationButtonType.TRANSFER) }}
  188. </div>
  189. </template>
  190. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  191. <el-form
  192. label-position="top"
  193. class="mb-auto"
  194. ref="transferFormRef"
  195. :model="transferForm"
  196. :rules="transferFormRule"
  197. label-width="100px"
  198. >
  199. <el-form-item label="新审批人" prop="assigneeUserId">
  200. <el-select v-model="transferForm.assigneeUserId" clearable style="width: 100%">
  201. <el-option
  202. v-for="item in userOptions"
  203. :key="item.id"
  204. :label="item.nickname"
  205. :value="item.id"
  206. />
  207. </el-select>
  208. </el-form-item>
  209. <el-form-item label="审批意见" prop="reason">
  210. <el-input
  211. v-model="transferForm.reason"
  212. clearable
  213. placeholder="请输入审批意见"
  214. type="textarea"
  215. :rows="3"
  216. />
  217. </el-form-item>
  218. <el-form-item>
  219. <el-button :disabled="formLoading" type="primary" @click="handleTransfer()">
  220. {{ getButtonDisplayName(OperationButtonType.TRANSFER) }}
  221. </el-button>
  222. <el-button @click="closePropover('transfer', transferFormRef)"> 取消 </el-button>
  223. </el-form-item>
  224. </el-form>
  225. </div>
  226. </el-popover>
  227. <!-- 【委派】按钮 -->
  228. <el-popover
  229. :visible="popOverVisible.delegate"
  230. placement="top-start"
  231. :width="420"
  232. trigger="click"
  233. v-if="runningTask && isHandleTaskStatus() && isShowButton(OperationButtonType.DELEGATE)"
  234. >
  235. <template #reference>
  236. <div @click="openPopover('delegate')" class="hover-bg-gray-100 rounded-xl p-6px">
  237. <Icon :size="14" icon="ep:position" />&nbsp;
  238. {{ getButtonDisplayName(OperationButtonType.DELEGATE) }}
  239. </div>
  240. </template>
  241. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  242. <el-form
  243. label-position="top"
  244. class="mb-auto"
  245. ref="delegateFormRef"
  246. :model="delegateForm"
  247. :rules="delegateFormRule"
  248. label-width="100px"
  249. >
  250. <el-form-item label="接收人" prop="delegateUserId">
  251. <el-select v-model="delegateForm.delegateUserId" clearable style="width: 100%">
  252. <el-option
  253. v-for="item in userOptions"
  254. :key="item.id"
  255. :label="item.nickname"
  256. :value="item.id"
  257. />
  258. </el-select>
  259. </el-form-item>
  260. <el-form-item label="审批意见" prop="reason">
  261. <el-input
  262. v-model="delegateForm.reason"
  263. clearable
  264. placeholder="请输入审批意见"
  265. type="textarea"
  266. :rows="3"
  267. />
  268. </el-form-item>
  269. <el-form-item>
  270. <el-button :disabled="formLoading" type="primary" @click="handleDelegate()">
  271. {{ getButtonDisplayName(OperationButtonType.DELEGATE) }}
  272. </el-button>
  273. <el-button @click="closePropover('delegate', delegateFormRef)"> 取消 </el-button>
  274. </el-form-item>
  275. </el-form>
  276. </div>
  277. </el-popover>
  278. <!-- 【加签】按钮 当前任务审批人为A,向前加签选了一个C,则需要C先审批,然后再是A审批,向后加签B,A审批完,需要B再审批完,才算完成这个任务节点 -->
  279. <el-popover
  280. :visible="popOverVisible.addSign"
  281. placement="top-start"
  282. :width="420"
  283. trigger="click"
  284. v-if="runningTask && isHandleTaskStatus() && isShowButton(OperationButtonType.ADD_SIGN)"
  285. >
  286. <template #reference>
  287. <div @click="openPopover('addSign')" class="hover-bg-gray-100 rounded-xl p-6px">
  288. <Icon :size="14" icon="ep:plus" />&nbsp;
  289. {{ getButtonDisplayName(OperationButtonType.ADD_SIGN) }}
  290. </div>
  291. </template>
  292. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  293. <el-form
  294. label-position="top"
  295. class="mb-auto"
  296. ref="addSignFormRef"
  297. :model="addSignForm"
  298. :rules="addSignFormRule"
  299. label-width="100px"
  300. >
  301. <el-form-item label="加签处理人" prop="addSignUserIds">
  302. <el-select v-model="addSignForm.addSignUserIds" multiple clearable style="width: 100%">
  303. <el-option
  304. v-for="item in userOptions"
  305. :key="item.id"
  306. :label="item.nickname"
  307. :value="item.id"
  308. />
  309. </el-select>
  310. </el-form-item>
  311. <el-form-item label="审批意见" prop="reason">
  312. <el-input
  313. v-model="addSignForm.reason"
  314. clearable
  315. placeholder="请输入审批意见"
  316. type="textarea"
  317. :rows="3"
  318. />
  319. </el-form-item>
  320. <el-form-item>
  321. <el-button :disabled="formLoading" type="primary" @click="handlerAddSign('before')">
  322. 向前{{ getButtonDisplayName(OperationButtonType.ADD_SIGN) }}
  323. </el-button>
  324. <el-button :disabled="formLoading" type="primary" @click="handlerAddSign('after')">
  325. 向后{{ getButtonDisplayName(OperationButtonType.ADD_SIGN) }}
  326. </el-button>
  327. <el-button @click="closePropover('addSign', addSignFormRef)"> 取消 </el-button>
  328. </el-form-item>
  329. </el-form>
  330. </div>
  331. </el-popover>
  332. <!-- 【减签】按钮 -->
  333. <el-popover
  334. :visible="popOverVisible.deleteSign"
  335. placement="top-start"
  336. :width="420"
  337. trigger="click"
  338. v-if="runningTask?.children.length > 0"
  339. >
  340. <template #reference>
  341. <div @click="openPopover('deleteSign')" class="hover-bg-gray-100 rounded-xl p-6px">
  342. <Icon :size="14" icon="ep:semi-select" />&nbsp; 减签
  343. </div>
  344. </template>
  345. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  346. <el-form
  347. label-position="top"
  348. class="mb-auto"
  349. ref="deleteSignFormRef"
  350. :model="deleteSignForm"
  351. :rules="deleteSignFormRule"
  352. label-width="100px"
  353. >
  354. <el-form-item label="减签人员" prop="deleteSignTaskId">
  355. <el-select v-model="deleteSignForm.deleteSignTaskId" clearable style="width: 100%">
  356. <el-option
  357. v-for="item in runningTask.children"
  358. :key="item.id"
  359. :label="getDeleteSignUserLabel(item)"
  360. :value="item.id"
  361. />
  362. </el-select>
  363. </el-form-item>
  364. <el-form-item label="审批意见" prop="reason">
  365. <el-input
  366. v-model="deleteSignForm.reason"
  367. clearable
  368. placeholder="请输入审批意见"
  369. type="textarea"
  370. :rows="3"
  371. />
  372. </el-form-item>
  373. <el-form-item>
  374. <el-button :disabled="formLoading" type="primary" @click="handlerDeleteSign()">
  375. 减签
  376. </el-button>
  377. <el-button @click="closePropover('deleteSign', deleteSignFormRef)"> 取消 </el-button>
  378. </el-form-item>
  379. </el-form>
  380. </div>
  381. </el-popover>
  382. <!-- 【退回】按钮 -->
  383. <el-popover
  384. :visible="popOverVisible.return"
  385. placement="top-start"
  386. :width="420"
  387. trigger="click"
  388. v-if="runningTask && isHandleTaskStatus() && isShowButton(OperationButtonType.RETURN)"
  389. >
  390. <template #reference>
  391. <div @click="openPopover('return')" class="hover-bg-gray-100 rounded-xl p-6px">
  392. <Icon :size="14" icon="ep:back" />&nbsp;
  393. {{ getButtonDisplayName(OperationButtonType.RETURN) }}
  394. </div>
  395. </template>
  396. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  397. <el-form
  398. label-position="top"
  399. class="mb-auto"
  400. ref="returnFormRef"
  401. :model="returnForm"
  402. :rules="returnFormRule"
  403. label-width="100px"
  404. >
  405. <el-form-item label="退回节点" prop="targetTaskDefinitionKey">
  406. <el-select v-model="returnForm.targetTaskDefinitionKey" clearable style="width: 100%">
  407. <el-option
  408. v-for="item in returnList"
  409. :key="item.taskDefinitionKey"
  410. :label="item.name"
  411. :value="item.taskDefinitionKey"
  412. />
  413. </el-select>
  414. </el-form-item>
  415. <el-form-item label="退回理由" prop="returnReason">
  416. <el-input
  417. v-model="returnForm.returnReason"
  418. clearable
  419. placeholder="请输入退回理由"
  420. type="textarea"
  421. :rows="3"
  422. />
  423. </el-form-item>
  424. <el-form-item>
  425. <el-button :disabled="formLoading" type="primary" @click="handleReturn()">
  426. {{ getButtonDisplayName(OperationButtonType.RETURN) }}
  427. </el-button>
  428. <el-button @click="closePropover('return', returnFormRef)"> 取消 </el-button>
  429. </el-form-item>
  430. </el-form>
  431. </div>
  432. </el-popover>
  433. <!--【取消】按钮 这个对应发起人的取消, 只有发起人可以取消 -->
  434. <el-popover
  435. :visible="popOverVisible.cancel"
  436. placement="top-start"
  437. :width="420"
  438. trigger="click"
  439. v-if="
  440. userId === processInstance?.startUser?.id && !isEndProcessStatus(processInstance?.status)
  441. "
  442. >
  443. <template #reference>
  444. <div @click="openPopover('cancel')" class="hover-bg-gray-100 rounded-xl p-6px">
  445. <Icon :size="14" icon="fa:mail-reply" />&nbsp; 取消
  446. </div>
  447. </template>
  448. <div class="flex flex-col flex-1 pt-20px px-20px" v-loading="formLoading">
  449. <el-form
  450. label-position="top"
  451. class="mb-auto"
  452. ref="cancelFormRef"
  453. :model="cancelForm"
  454. :rules="cancelFormRule"
  455. label-width="100px"
  456. >
  457. <el-form-item label="取消理由" prop="cancelReason">
  458. <span class="text-#878c93 text-12px">&nbsp; 取消后,该审批流程将自动结束</span>
  459. <el-input
  460. v-model="cancelForm.cancelReason"
  461. clearable
  462. placeholder="请输入取消理由"
  463. type="textarea"
  464. :rows="3"
  465. />
  466. </el-form-item>
  467. <el-form-item>
  468. <el-button :disabled="formLoading" type="primary" @click="handleCancel()">
  469. 确认
  470. </el-button>
  471. <el-button @click="closePropover('cancel', cancelFormRef)"> 取消 </el-button>
  472. </el-form-item>
  473. </el-form>
  474. </div>
  475. </el-popover>
  476. <!-- 【再次提交】 按钮-->
  477. <div
  478. @click="handleReCreate()"
  479. class="hover-bg-gray-100 rounded-xl p-6px"
  480. v-if="
  481. userId === processInstance?.startUser?.id &&
  482. isEndProcessStatus(processInstance?.status) &&
  483. processDefinition?.formType === 10
  484. "
  485. >
  486. <Icon :size="14" icon="ep:refresh" />&nbsp; 再次提交
  487. </div>
  488. </div>
  489. <!-- 签名弹窗 -->
  490. <SignDialog ref="signRef" @success="handleSignFinish" />
  491. </template>
  492. <script lang="ts" setup>
  493. import { useUserStoreWithOut } from '@/store/modules/user'
  494. import { setConfAndFields2 } from '@/utils/formCreate'
  495. import * as TaskApi from '@/api/bpm/task'
  496. import * as ProcessInstanceApi from '@/api/bpm/processInstance'
  497. import * as UserApi from '@/api/system/user'
  498. import {
  499. OPERATION_BUTTON_NAME,
  500. OperationButtonType
  501. } from '@/components/SimpleProcessDesignerV2/src/consts'
  502. import { BpmModelFormType, BpmProcessInstanceStatus } from '@/utils/constants'
  503. import type { FormInstance, FormRules } from 'element-plus'
  504. import SignDialog from './SignDialog.vue'
  505. defineOptions({ name: 'ProcessInstanceBtnContainer' })
  506. const router = useRouter() // 路由
  507. const message = useMessage() // 消息弹窗
  508. const userId = useUserStoreWithOut().getUser.id // 当前登录的编号
  509. const emit = defineEmits(['success']) // 定义 success 事件,用于操作成功后的回调
  510. const props = defineProps<{
  511. processInstance: any // 流程实例信息
  512. processDefinition: any // 流程定义信息
  513. userOptions: UserApi.UserVO[]
  514. normalForm: any // 流程表单 formCreate
  515. normalFormApi: any // 流程表单 formCreate Api
  516. writableFields: string[] // 流程表单可以编辑的字段
  517. }>()
  518. const formLoading = ref(false) // 表单加载中
  519. const popOverVisible = ref({
  520. approve: false,
  521. reject: false,
  522. transfer: false,
  523. delegate: false,
  524. addSign: false,
  525. return: false,
  526. copy: false,
  527. cancel: false,
  528. deleteSign: false
  529. }) // 气泡卡是否展示
  530. const returnList = ref([] as any) // 退回节点
  531. // ========== 审批信息 ==========
  532. const runningTask = ref<any>() // 运行中的任务
  533. const approveForm = ref<any>({}) // 审批通过时,额外的补充信息
  534. const approveFormFApi = ref<any>({}) // approveForms 的 fAPi
  535. // 审批通过意见表单
  536. const reasonRequire = ref()
  537. const approveFormRef = ref<FormInstance>()
  538. const signRef = ref()
  539. const approveSignFormRef = ref()
  540. const approveReasonForm = reactive({
  541. reason: '',
  542. signPicUrl: ''
  543. })
  544. const approveReasonRule = computed(() => {
  545. return {
  546. reason: [{ required: reasonRequire.value, message: '审批意见不能为空', trigger: 'blur' }],
  547. signPicUrl: [{ required: true, message: '签名不能为空', trigger: 'change' }]
  548. }
  549. })
  550. // 拒绝表单
  551. const rejectFormRef = ref<FormInstance>()
  552. const rejectReasonForm = reactive({
  553. reason: ''
  554. })
  555. const rejectReasonRule = computed(() => {
  556. return {
  557. reason: [{ required: reasonRequire.value, message: '审批意见不能为空', trigger: 'blur' }]
  558. }
  559. })
  560. // 抄送表单
  561. const copyFormRef = ref<FormInstance>()
  562. const copyForm = reactive({
  563. copyUserIds: [],
  564. copyReason: ''
  565. })
  566. const copyFormRule = reactive<FormRules<typeof copyForm>>({
  567. copyUserIds: [{ required: true, message: '抄送人不能为空', trigger: 'change' }]
  568. })
  569. // 转办表单
  570. const transferFormRef = ref<FormInstance>()
  571. const transferForm = reactive({
  572. assigneeUserId: undefined,
  573. reason: ''
  574. })
  575. const transferFormRule = reactive<FormRules<typeof transferForm>>({
  576. assigneeUserId: [{ required: true, message: '新审批人不能为空', trigger: 'change' }],
  577. reason: [{ required: true, message: '审批意见不能为空', trigger: 'blur' }]
  578. })
  579. // 委派表单
  580. const delegateFormRef = ref<FormInstance>()
  581. const delegateForm = reactive({
  582. delegateUserId: undefined,
  583. reason: ''
  584. })
  585. const delegateFormRule = reactive<FormRules<typeof delegateForm>>({
  586. delegateUserId: [{ required: true, message: '接收人不能为空', trigger: 'change' }],
  587. reason: [{ required: true, message: '审批意见不能为空', trigger: 'blur' }]
  588. })
  589. // 加签表单
  590. const addSignFormRef = ref<FormInstance>()
  591. const addSignForm = reactive({
  592. addSignUserIds: undefined,
  593. reason: ''
  594. })
  595. const addSignFormRule = reactive<FormRules<typeof addSignForm>>({
  596. addSignUserIds: [{ required: true, message: '加签处理人不能为空', trigger: 'change' }],
  597. reason: [{ required: true, message: '审批意见不能为空', trigger: 'blur' }]
  598. })
  599. // 减签表单
  600. const deleteSignFormRef = ref<FormInstance>()
  601. const deleteSignForm = reactive({
  602. deleteSignTaskId: undefined,
  603. reason: ''
  604. })
  605. const deleteSignFormRule = reactive<FormRules<typeof deleteSignForm>>({
  606. deleteSignTaskId: [{ required: true, message: '减签人员不能为空', trigger: 'change' }],
  607. reason: [{ required: true, message: '审批意见不能为空', trigger: 'blur' }]
  608. })
  609. // 退回表单
  610. const returnFormRef = ref<FormInstance>()
  611. const returnForm = reactive({
  612. targetTaskDefinitionKey: undefined,
  613. returnReason: ''
  614. })
  615. const returnFormRule = reactive<FormRules<typeof returnForm>>({
  616. targetTaskDefinitionKey: [{ required: true, message: '退回节点不能为空', trigger: 'change' }],
  617. returnReason: [{ required: true, message: '退回理由不能为空', trigger: 'blur' }]
  618. })
  619. // 取消表单
  620. const cancelFormRef = ref<FormInstance>()
  621. const cancelForm = reactive({
  622. cancelReason: ''
  623. })
  624. const cancelFormRule = reactive<FormRules<typeof cancelForm>>({
  625. cancelReason: [{ required: true, message: '取消理由不能为空', trigger: 'blur' }]
  626. })
  627. /** 监听 approveFormFApis,实现它对应的 form-create 初始化后,隐藏掉对应的表单提交按钮 */
  628. watch(
  629. () => approveFormFApi.value,
  630. (val) => {
  631. val?.btn?.show(false)
  632. val?.resetBtn?.show(false)
  633. },
  634. {
  635. deep: true
  636. }
  637. )
  638. /** 弹出气泡卡 */
  639. const openPopover = async (type: string) => {
  640. if (type === 'approve') {
  641. // 校验流程表单
  642. const valid = await validateNormalForm()
  643. if (!valid) {
  644. message.warning('表单校验不通过,请先完善表单!!')
  645. return
  646. }
  647. }
  648. if (type === 'return') {
  649. // 获取退回节点
  650. returnList.value = await TaskApi.getTaskListByReturn(runningTask.value.id)
  651. if (returnList.value.length === 0) {
  652. message.warning('当前没有可退回的节点')
  653. return
  654. }
  655. }
  656. Object.keys(popOverVisible.value).forEach((item) => {
  657. popOverVisible.value[item] = item === type
  658. })
  659. // await nextTick()
  660. // formRef.value.resetFields()
  661. }
  662. /** 关闭气泡卡 */
  663. const closePropover = (type: string, formRef: FormInstance | undefined) => {
  664. if (formRef) {
  665. formRef.resetFields()
  666. }
  667. popOverVisible.value[type] = false
  668. }
  669. /** 处理审批通过和不通过的操作 */
  670. const handleAudit = async (pass: boolean, formRef: FormInstance | undefined) => {
  671. if (formLoading.value) {
  672. return
  673. }
  674. formLoading.value = true
  675. try {
  676. // 校验表单
  677. if (!formRef) return
  678. await formRef.validate()
  679. if (pass) {
  680. // 获取修改的流程变量, 暂时只支持流程表单
  681. const variables = getUpdatedProcessInstanceVariables()
  682. // 审批通过数据
  683. const data = {
  684. id: runningTask.value.id,
  685. reason: approveReasonForm.reason,
  686. variables // 审批通过, 把修改的字段值赋于流程实例变量
  687. }
  688. // 签名
  689. if (runningTask.value.signEnable) {
  690. data.signPicUrl = approveReasonForm.signPicUrl
  691. }
  692. // 多表单处理:任务级自定义表单(form-create);须校验并将填写值写入 variables
  693. // 注意:不可用 Object.keys(fApi) 判断 —— loadTodoTask 会先把 approveFormFApi 置为 {},
  694. // 若在 API 尚未重新注入前判断会得到 0 键,从而跳过校验且 variables 未覆盖为审批表单的值。
  695. const hasTaskApproveForm = runningTask.value?.formId > 0
  696. if (hasTaskApproveForm) {
  697. await nextTick()
  698. const formCreateApi = approveFormFApi.value
  699. if (typeof formCreateApi?.validate !== 'function') {
  700. message.warning('审批自定义表单未完成初始化,请稍后再试')
  701. return
  702. }
  703. await formCreateApi.validate()
  704. // @ts-ignore
  705. data.variables = approveForm.value.value
  706. }
  707. await TaskApi.approveTask(data)
  708. popOverVisible.value.approve = false
  709. message.success('审批通过成功')
  710. } else {
  711. // 审批不通过数据
  712. const data = {
  713. id: runningTask.value.id,
  714. reason: rejectReasonForm.reason
  715. }
  716. await TaskApi.rejectTask(data)
  717. popOverVisible.value.reject = false
  718. message.success('审批不通过成功')
  719. }
  720. // 重置表单
  721. formRef.resetFields()
  722. // 加载最新数据
  723. reload()
  724. } finally {
  725. setTimeout(() => {
  726. formLoading.value = false
  727. }, 400)
  728. }
  729. }
  730. /** 处理抄送 */
  731. const handleCopy = async () => {
  732. formLoading.value = true
  733. try {
  734. // 1. 校验表单
  735. if (!copyFormRef.value) return
  736. await copyFormRef.value.validate()
  737. // 2. 提交抄送
  738. const data = {
  739. id: runningTask.value.id,
  740. reason: copyForm.copyReason,
  741. copyUserIds: copyForm.copyUserIds
  742. }
  743. await TaskApi.copyTask(data)
  744. copyFormRef.value.resetFields()
  745. popOverVisible.value.copy = false
  746. message.success('操作成功')
  747. } finally {
  748. formLoading.value = false
  749. }
  750. }
  751. /** 处理转交 */
  752. const handleTransfer = async () => {
  753. formLoading.value = true
  754. try {
  755. // 1.1 校验表单
  756. if (!transferFormRef.value) return
  757. await transferFormRef.value.validate()
  758. // 1.2 提交转交
  759. const data = {
  760. id: runningTask.value.id,
  761. reason: transferForm.reason,
  762. assigneeUserId: transferForm.assigneeUserId
  763. }
  764. await TaskApi.transferTask(data)
  765. transferFormRef.value.resetFields()
  766. popOverVisible.value.transfer = false
  767. message.success('操作成功')
  768. // 2. 加载最新数据
  769. reload()
  770. } finally {
  771. formLoading.value = false
  772. }
  773. }
  774. /** 处理委派 */
  775. const handleDelegate = async () => {
  776. formLoading.value = true
  777. try {
  778. // 1.1 校验表单
  779. if (!delegateFormRef.value) return
  780. await delegateFormRef.value.validate()
  781. // 1.2 处理委派
  782. const data = {
  783. id: runningTask.value.id,
  784. reason: delegateForm.reason,
  785. delegateUserId: delegateForm.delegateUserId
  786. }
  787. await TaskApi.delegateTask(data)
  788. popOverVisible.value.delegate = false
  789. delegateFormRef.value.resetFields()
  790. message.success('操作成功')
  791. // 2. 加载最新数据
  792. reload()
  793. } finally {
  794. formLoading.value = false
  795. }
  796. }
  797. /** 处理加签 */
  798. const handlerAddSign = async (type: string) => {
  799. formLoading.value = true
  800. try {
  801. // 1.1 校验表单
  802. if (!addSignFormRef.value) return
  803. await addSignFormRef.value.validate()
  804. // 1.2 提交加签
  805. const data = {
  806. id: runningTask.value.id,
  807. type,
  808. reason: addSignForm.reason,
  809. userIds: addSignForm.addSignUserIds
  810. }
  811. await TaskApi.signCreateTask(data)
  812. message.success('操作成功')
  813. addSignFormRef.value.resetFields()
  814. popOverVisible.value.addSign = false
  815. // 2 加载最新数据
  816. reload()
  817. } finally {
  818. formLoading.value = false
  819. }
  820. }
  821. /** 处理退回 */
  822. const handleReturn = async () => {
  823. formLoading.value = true
  824. try {
  825. // 1.1 校验表单
  826. if (!returnFormRef.value) return
  827. await returnFormRef.value.validate()
  828. // 1.2 提交退回
  829. const data = {
  830. id: runningTask.value.id,
  831. reason: returnForm.returnReason,
  832. targetTaskDefinitionKey: returnForm.targetTaskDefinitionKey
  833. }
  834. await TaskApi.returnTask(data)
  835. popOverVisible.value.return = false
  836. returnFormRef.value.resetFields()
  837. message.success('操作成功')
  838. // 2 重新加载数据
  839. reload()
  840. } finally {
  841. formLoading.value = false
  842. }
  843. }
  844. /** 处理取消 */
  845. const handleCancel = async () => {
  846. formLoading.value = true
  847. try {
  848. // 1.1 校验表单
  849. if (!cancelFormRef.value) return
  850. await cancelFormRef.value.validate()
  851. // 1.2 提交取消
  852. await ProcessInstanceApi.cancelProcessInstanceByStartUser(
  853. props.processInstance.id,
  854. cancelForm.cancelReason
  855. )
  856. popOverVisible.value.return = false
  857. message.success('操作成功')
  858. cancelFormRef.value.resetFields()
  859. // 2 重新加载数据
  860. reload()
  861. } finally {
  862. formLoading.value = false
  863. }
  864. }
  865. /** 处理再次提交 */
  866. const handleReCreate = async () => {
  867. // 跳转发起流程界面
  868. await router.push({
  869. name: 'BpmProcessInstanceCreate',
  870. query: { processInstanceId: props.processInstance?.id }
  871. })
  872. }
  873. /** 获取减签人员标签 */
  874. const getDeleteSignUserLabel = (task: any): string => {
  875. const deptName = task?.assigneeUser?.deptName || task?.ownerUser?.deptName
  876. const nickname = task?.assigneeUser?.nickname || task?.ownerUser?.nickname
  877. return `${nickname} ( 所属部门:${deptName} )`
  878. }
  879. /** 处理减签 */
  880. const handlerDeleteSign = async () => {
  881. formLoading.value = true
  882. try {
  883. // 1.1 校验表单
  884. if (!deleteSignFormRef.value) return
  885. await deleteSignFormRef.value.validate()
  886. // 1.2 提交减签
  887. const data = {
  888. id: deleteSignForm.deleteSignTaskId,
  889. reason: deleteSignForm.reason
  890. }
  891. await TaskApi.signDeleteTask(data)
  892. message.success('减签成功')
  893. deleteSignFormRef.value.resetFields()
  894. popOverVisible.value.deleteSign = false
  895. // 2 加载最新数据
  896. reload()
  897. } finally {
  898. formLoading.value = false
  899. }
  900. }
  901. /** 重新加载数据 */
  902. const reload = () => {
  903. emit('success')
  904. }
  905. /** 任务是否为处理中状态 */
  906. const isHandleTaskStatus = () => {
  907. let canHandle = false
  908. if (TaskApi.TaskStatusEnum.RUNNING === runningTask.value?.status) {
  909. canHandle = true
  910. }
  911. return canHandle
  912. }
  913. /** 流程状态是否为结束状态 */
  914. const isEndProcessStatus = (status: number) => {
  915. let isEndStatus = false
  916. if (
  917. BpmProcessInstanceStatus.APPROVE === status ||
  918. BpmProcessInstanceStatus.REJECT === status ||
  919. BpmProcessInstanceStatus.CANCEL === status
  920. ) {
  921. isEndStatus = true
  922. }
  923. return isEndStatus
  924. }
  925. /** 是否显示按钮 */
  926. const isShowButton = (btnType: OperationButtonType): boolean => {
  927. let isShow = true
  928. if (runningTask.value?.buttonsSetting && runningTask.value?.buttonsSetting[btnType]) {
  929. isShow = runningTask.value.buttonsSetting[btnType].enable
  930. }
  931. return isShow
  932. }
  933. /** 获取按钮的显示名称 */
  934. const getButtonDisplayName = (btnType: OperationButtonType) => {
  935. let displayName = OPERATION_BUTTON_NAME.get(btnType)
  936. if (runningTask.value?.buttonsSetting && runningTask.value?.buttonsSetting[btnType]) {
  937. displayName = runningTask.value.buttonsSetting[btnType].displayName
  938. }
  939. return displayName
  940. }
  941. const loadTodoTask = (task: any) => {
  942. approveForm.value = {}
  943. approveFormFApi.value = {}
  944. runningTask.value = task
  945. reasonRequire.value = task?.reasonRequire ?? false
  946. // 处理 approve 表单.
  947. if (task && task.formId && task.formConf) {
  948. const tempApproveForm = {}
  949. setConfAndFields2(tempApproveForm, task.formConf, task.formFields, task.formVariables)
  950. approveForm.value = tempApproveForm
  951. } else {
  952. approveForm.value = {} // 占位,避免为空
  953. }
  954. }
  955. /** 校验流程表单 */
  956. const validateNormalForm = async () => {
  957. if (props.processDefinition?.formType === BpmModelFormType.NORMAL) {
  958. let valid = true
  959. try {
  960. await props.normalFormApi?.validate()
  961. } catch {
  962. valid = false
  963. }
  964. return valid
  965. } else {
  966. return true
  967. }
  968. }
  969. /** 从可以编辑的流程表单字段,获取需要修改的流程实例的变量 */
  970. const getUpdatedProcessInstanceVariables = () => {
  971. const variables = {}
  972. props.writableFields.forEach((field) => {
  973. variables[field] = props.normalFormApi.getValue(field)
  974. })
  975. return variables
  976. }
  977. /** 处理签名完成 */
  978. const handleSignFinish = (url: string) => {
  979. approveReasonForm.signPicUrl = url
  980. approveSignFormRef.value.validate('change')
  981. }
  982. defineExpose({ loadTodoTask })
  983. </script>
  984. <style lang="scss" scoped>
  985. :deep(.el-affix--fixed) {
  986. background-color: var(--el-bg-color);
  987. }
  988. .btn-container {
  989. > div {
  990. display: flex;
  991. margin: 0 8px;
  992. cursor: pointer;
  993. align-items: center;
  994. &:hover {
  995. color: #6db5ff;
  996. }
  997. }
  998. }
  999. </style>