submitVerify.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529
  1. <template>
  2. <el-dialog v-model="dialog.visible" :title="dialog.title" width="50%" draggable :before-close="cancel" center :close-on-click-modal="false">
  3. <el-form v-loading="loading" :model="form" label-width="120px">
  4. <el-form-item label="消息提醒">
  5. <el-checkbox-group v-model="form.messageType">
  6. <el-checkbox value="1" name="type" disabled>站内信</el-checkbox>
  7. <el-checkbox value="2" name="type">邮件</el-checkbox>
  8. <el-checkbox value="3" name="type">短信</el-checkbox>
  9. </el-checkbox-group>
  10. </el-form-item>
  11. <el-form-item v-if="task.flowStatus === 'waiting'" label="附件">
  12. <fileUpload v-model="form.fileId" :file-type="['png', 'jpg', 'jpeg', 'doc', 'docx', 'xlsx', 'xls', 'ppt', 'txt', 'pdf']" :file-size="20" />
  13. </el-form-item>
  14. <el-form-item label="抄送" v-if="task.flowStatus === 'waiting' && buttonObj.copy">
  15. <el-button type="primary" icon="Plus" circle @click="openUserSelectCopy" />
  16. <el-tag v-for="user in selectCopyUserList" :key="user.userId" closable style="margin: 2px" @close="handleCopyCloseTag(user)">
  17. {{ user.nickName }}
  18. </el-tag>
  19. </el-form-item>
  20. <el-form-item v-if="buttonObj.pop && nestNodeList && nestNodeList.length > 0" label="下一步审批人" prop="assigneeMap">
  21. <div v-for="(item, index) in nestNodeList" :key="index" style="margin-bottom: 5px; width: 500px">
  22. <span>【{{ item.nodeName }}】:</span>
  23. <el-input v-if="false" v-model="form.assigneeMap[item.nodeCode]" />
  24. <el-input placeholder="请选择审批人" readonly v-model="nickName[item.nodeCode]">
  25. <template v-slot:append>
  26. <el-button @click="choosePeople(item)" icon="search">选择</el-button>
  27. </template>
  28. </el-input>
  29. </div>
  30. </el-form-item>
  31. <el-form-item v-if="task.flowStatus === 'waiting'" label="审批意见">
  32. <el-input v-model="form.message" type="textarea" resize="none" />
  33. </el-form-item>
  34. </el-form>
  35. <template #footer>
  36. <span class="dialog-footer">
  37. <el-button :disabled="buttonDisabled" type="primary" @click="handleCompleteTask"> 提交 </el-button>
  38. <el-button v-if="task.flowStatus === 'waiting' && buttonObj.trust" :disabled="buttonDisabled" type="primary" @click="openDelegateTask">
  39. 委托
  40. </el-button>
  41. <el-button v-if="task.flowStatus === 'waiting' && buttonObj.transfer" :disabled="buttonDisabled" type="primary" @click="openTransferTask">
  42. 转办
  43. </el-button>
  44. <el-button
  45. v-if="task.flowStatus === 'waiting' && Number(task.nodeRatio) > 0 && buttonObj.addSign"
  46. :disabled="buttonDisabled"
  47. type="primary"
  48. @click="openMultiInstanceUser"
  49. >
  50. 加签
  51. </el-button>
  52. <el-button
  53. v-if="task.flowStatus === 'waiting' && Number(task.nodeRatio) > 0 && buttonObj.subSign"
  54. :disabled="buttonDisabled"
  55. type="primary"
  56. @click="handleTaskUser"
  57. >
  58. 减签
  59. </el-button>
  60. <el-button
  61. v-if="task.flowStatus === 'waiting' && buttonObj.termination"
  62. :disabled="buttonDisabled"
  63. type="danger"
  64. @click="handleTerminationTask"
  65. >
  66. 终止
  67. </el-button>
  68. <el-button v-if="task.flowStatus === 'waiting' && buttonObj.back" :disabled="buttonDisabled" type="danger" @click="handleBackProcessOpen">
  69. 退回
  70. </el-button>
  71. <el-button :disabled="buttonDisabled" @click="cancel">取消</el-button>
  72. </span>
  73. </template>
  74. <!-- 抄送 -->
  75. <UserSelect ref="userSelectCopyRef" :multiple="true" :data="selectCopyUserIds" @confirm-call-back="userSelectCopyCallBack"></UserSelect>
  76. <!-- 转办 -->
  77. <UserSelect ref="transferTaskRef" :multiple="false" @confirm-call-back="handleTransferTask"></UserSelect>
  78. <!-- 委托 -->
  79. <UserSelect ref="delegateTaskRef" :multiple="false" @confirm-call-back="handleDelegateTask"></UserSelect>
  80. <!-- 加签组件 -->
  81. <UserSelect ref="multiInstanceUserRef" :multiple="true" @confirm-call-back="addMultiInstanceUser"></UserSelect>
  82. <!-- 弹窗选人 -->
  83. <UserSelect ref="porUserRef" :multiple="true" :userIds="popUserIds" @confirm-call-back="handlePopUser"></UserSelect>
  84. <!-- 驳回开始 -->
  85. <el-dialog v-model="backVisible" draggable title="驳回" width="40%" :close-on-click-modal="false">
  86. <el-form v-if="task.flowStatus === 'waiting'" v-loading="backLoading" :model="backForm" label-width="120px">
  87. <el-form-item label="驳回节点">
  88. <el-select v-model="backForm.nodeCode" clearable placeholder="请选择" style="width: 300px">
  89. <el-option v-for="item in taskNodeList" :key="item.nodeCode" :label="item.nodeName" :value="item.nodeCode" />
  90. </el-select>
  91. </el-form-item>
  92. <el-form-item label="消息提醒">
  93. <el-checkbox-group v-model="backForm.messageType">
  94. <el-checkbox label="1" name="type" disabled>站内信</el-checkbox>
  95. <el-checkbox label="2" name="type">邮件</el-checkbox>
  96. <el-checkbox label="3" name="type">短信</el-checkbox>
  97. </el-checkbox-group>
  98. </el-form-item>
  99. <el-form-item v-if="task.flowStatus === 'waiting'" label="附件">
  100. <fileUpload
  101. v-model="backForm.fileId"
  102. :file-type="['png', 'jpg', 'jpeg', 'doc', 'docx', 'xlsx', 'xls', 'ppt', 'txt', 'pdf']"
  103. :file-size="20"
  104. />
  105. </el-form-item>
  106. <el-form-item label="审批意见">
  107. <el-input v-model="backForm.message" type="textarea" resize="none" />
  108. </el-form-item>
  109. </el-form>
  110. <template #footer>
  111. <div class="dialog-footer" style="float: right; padding-bottom: 20px">
  112. <el-button :disabled="backButtonDisabled" type="primary" @click="handleBackProcess">确认</el-button>
  113. <el-button :disabled="backButtonDisabled" @click="backVisible = false">取消</el-button>
  114. </div>
  115. </template>
  116. </el-dialog>
  117. <!-- 驳回结束 -->
  118. <el-dialog v-model="deleteSignatureVisible" draggable title="减签人员" width="700px" height="400px" append-to-body :close-on-click-modal="false">
  119. <div>
  120. <el-table :data="deleteUserList" border>
  121. <el-table-column prop="nodeName" label="任务名称" />
  122. <el-table-column prop="nickName" label="办理人" />
  123. <el-table-column label="操作" align="center" width="160">
  124. <template #default="scope">
  125. <el-button type="danger" size="small" icon="Delete" @click="deleteMultiInstanceUser(scope.row)">删除 </el-button>
  126. </template>
  127. </el-table-column>
  128. </el-table>
  129. </div>
  130. </el-dialog>
  131. </el-dialog>
  132. </template>
  133. <script setup lang="ts">
  134. import { ref } from 'vue';
  135. import { ComponentInternalInstance } from 'vue';
  136. import { ElForm } from 'element-plus';
  137. import {
  138. completeTask,
  139. backProcess,
  140. getTask,
  141. taskOperation,
  142. terminationTask,
  143. getBackTaskNode,
  144. currentTaskAllUser,
  145. getNextNodeList
  146. } from '@/api/workflow/task';
  147. import UserSelect from '@/components/UserSelect';
  148. const { proxy } = getCurrentInstance() as ComponentInternalInstance;
  149. import { UserVO } from '@/api/system/user/types';
  150. import { FlowTaskVO, TaskOperationBo } from '@/api/workflow/task/types';
  151. const userSelectCopyRef = ref<InstanceType<typeof UserSelect>>();
  152. const transferTaskRef = ref<InstanceType<typeof UserSelect>>();
  153. const delegateTaskRef = ref<InstanceType<typeof UserSelect>>();
  154. const multiInstanceUserRef = ref<InstanceType<typeof UserSelect>>();
  155. const porUserRef = ref<InstanceType<typeof UserSelect>>();
  156. const props = defineProps({
  157. taskVariables: {
  158. type: Object as () => Record<string, any>,
  159. default: () => {}
  160. }
  161. });
  162. //遮罩层
  163. const loading = ref(true);
  164. //按钮
  165. const buttonDisabled = ref(true);
  166. //任务id
  167. const taskId = ref<string>('');
  168. //抄送人
  169. const selectCopyUserList = ref<UserVO[]>([]);
  170. //抄送人id
  171. const selectCopyUserIds = ref<string>(undefined);
  172. //可减签的人员
  173. const deleteUserList = ref<any>([]);
  174. //弹窗可选择的人员id
  175. const popUserIds = ref<any>([]);
  176. //驳回是否显示
  177. const backVisible = ref(false);
  178. const backLoading = ref(true);
  179. const backButtonDisabled = ref(true);
  180. // 可驳回得任务节点
  181. const taskNodeList = ref([]);
  182. const nickName = ref({});
  183. //节点编码
  184. const nodeCode = ref<string>('');
  185. const buttonObj = ref<any>({
  186. pop: false,
  187. trust: false,
  188. transfer: false,
  189. addSign: false,
  190. subSign: false,
  191. termination: false,
  192. back: false
  193. });
  194. //下一节点列表
  195. const nestNodeList = ref([]);
  196. //任务
  197. const task = ref<FlowTaskVO>({
  198. id: undefined,
  199. createTime: undefined,
  200. updateTime: undefined,
  201. tenantId: undefined,
  202. definitionId: undefined,
  203. instanceId: undefined,
  204. flowName: undefined,
  205. businessId: undefined,
  206. nodeCode: undefined,
  207. nodeName: undefined,
  208. flowCode: undefined,
  209. flowStatus: undefined,
  210. formCustom: undefined,
  211. formPath: undefined,
  212. nodeType: undefined,
  213. nodeRatio: undefined,
  214. applyNode: false,
  215. buttonList: []
  216. });
  217. const dialog = reactive<DialogOption>({
  218. visible: false,
  219. title: '提示'
  220. });
  221. //减签弹窗
  222. const deleteSignatureVisible = ref(false);
  223. const form = ref<Record<string, any>>({
  224. taskId: undefined,
  225. message: undefined,
  226. assigneeMap: {},
  227. variables: {},
  228. messageType: ['1'],
  229. flowCopyList: []
  230. });
  231. const backForm = ref<Record<string, any>>({
  232. taskId: undefined,
  233. nodeCode: undefined,
  234. message: undefined,
  235. variables: {},
  236. messageType: ['1']
  237. });
  238. //打开弹窗
  239. const openDialog = async (id?: string) => {
  240. selectCopyUserIds.value = undefined;
  241. selectCopyUserList.value = [];
  242. form.value.fileId = undefined;
  243. taskId.value = id;
  244. form.value.message = undefined;
  245. dialog.visible = true;
  246. loading.value = true;
  247. buttonDisabled.value = true;
  248. const response = await getTask(taskId.value);
  249. task.value = response.data;
  250. buttonObj.value = {};
  251. task.value.buttonList.forEach((e) => {
  252. buttonObj.value[e.code] = e.show;
  253. });
  254. buttonDisabled.value = false;
  255. const data = {
  256. taskId: taskId.value,
  257. variables: props.taskVariables
  258. };
  259. const nextData = await getNextNodeList(data);
  260. nestNodeList.value = nextData.data;
  261. loading.value = false;
  262. };
  263. onMounted(() => {});
  264. const emits = defineEmits(['submitCallback', 'cancelCallback']);
  265. /** 办理流程 */
  266. const handleCompleteTask = async () => {
  267. form.value.taskId = taskId.value;
  268. form.value.taskVariables = props.taskVariables;
  269. let verify = false;
  270. if (buttonObj.value.pop && nestNodeList.value && nestNodeList.value.length > 0) {
  271. nestNodeList.value.forEach((e) => {
  272. if (
  273. Object.keys(form.value.assigneeMap).length === 0 ||
  274. form.value.assigneeMap[e.nodeCode] === '' ||
  275. form.value.assigneeMap[e.nodeCode] === null ||
  276. form.value.assigneeMap[e.nodeCode] === undefined
  277. ) {
  278. verify = true;
  279. }
  280. });
  281. if (verify) {
  282. proxy?.$modal.msgWarning('请选择审批人!');
  283. return false;
  284. }
  285. } else {
  286. form.value.assigneeMap = {};
  287. }
  288. if (selectCopyUserList.value && selectCopyUserList.value.length > 0) {
  289. const flowCopyList = [];
  290. selectCopyUserList.value.forEach((e) => {
  291. const copyUser = {
  292. userId: e.userId,
  293. userName: e.nickName
  294. };
  295. flowCopyList.push(copyUser);
  296. });
  297. form.value.flowCopyList = flowCopyList;
  298. }
  299. await proxy?.$modal.confirm('是否确认提交?');
  300. loading.value = true;
  301. buttonDisabled.value = true;
  302. try {
  303. await completeTask(form.value);
  304. dialog.visible = false;
  305. emits('submitCallback');
  306. proxy?.$modal.msgSuccess('操作成功');
  307. } finally {
  308. loading.value = false;
  309. buttonDisabled.value = false;
  310. }
  311. };
  312. /** 驳回弹窗打开 */
  313. const handleBackProcessOpen = async () => {
  314. backForm.value = {};
  315. backForm.value.messageType = ['1'];
  316. backVisible.value = true;
  317. backLoading.value = true;
  318. backButtonDisabled.value = true;
  319. const data = await getBackTaskNode(task.value.definitionId, task.value.nodeCode);
  320. taskNodeList.value = data.data;
  321. backLoading.value = false;
  322. backButtonDisabled.value = false;
  323. backForm.value.nodeCode = taskNodeList.value[0].nodeCode;
  324. };
  325. /** 驳回流程 */
  326. const handleBackProcess = async () => {
  327. backForm.value.taskId = taskId.value;
  328. await proxy?.$modal.confirm('是否确认驳回到申请人?');
  329. loading.value = true;
  330. backLoading.value = true;
  331. backButtonDisabled.value = true;
  332. await backProcess(backForm.value).finally(() => {
  333. loading.value = false;
  334. buttonDisabled.value = false;
  335. });
  336. dialog.visible = false;
  337. backLoading.value = false;
  338. backButtonDisabled.value = false;
  339. emits('submitCallback');
  340. proxy?.$modal.msgSuccess('操作成功');
  341. };
  342. //取消
  343. const cancel = async () => {
  344. dialog.visible = false;
  345. buttonDisabled.value = false;
  346. nickName.value = {};
  347. form.value.assigneeMap = {};
  348. emits('cancelCallback');
  349. };
  350. //打开抄送人员
  351. const openUserSelectCopy = () => {
  352. userSelectCopyRef.value.open();
  353. };
  354. //确认抄送人员
  355. const userSelectCopyCallBack = (data: UserVO[]) => {
  356. if (data && data.length > 0) {
  357. selectCopyUserList.value = data;
  358. selectCopyUserIds.value = selectCopyUserList.value.map((item) => item.userId).join(',');
  359. }
  360. };
  361. //删除抄送人员
  362. const handleCopyCloseTag = (user: UserVO) => {
  363. const userId = user.userId;
  364. // 使用split删除用户
  365. const index = selectCopyUserList.value.findIndex((item) => item.userId === userId);
  366. selectCopyUserList.value.splice(index, 1);
  367. selectCopyUserIds.value = selectCopyUserList.value.map((item) => item.userId).join(',');
  368. };
  369. //加签
  370. const openMultiInstanceUser = async () => {
  371. multiInstanceUserRef.value.open();
  372. };
  373. //加签
  374. const addMultiInstanceUser = async (data) => {
  375. if (data && data.length > 0) {
  376. const taskOperationBo = reactive<TaskOperationBo>({
  377. userIds: data.map((e) => e.userId),
  378. taskId: taskId.value,
  379. message: form.value.message
  380. });
  381. await proxy?.$modal.confirm('是否确认提交?');
  382. loading.value = true;
  383. buttonDisabled.value = true;
  384. await taskOperation(taskOperationBo, 'addSignature').finally(() => {
  385. loading.value = false;
  386. buttonDisabled.value = false;
  387. });
  388. dialog.visible = false;
  389. emits('submitCallback');
  390. proxy?.$modal.msgSuccess('操作成功');
  391. } else {
  392. proxy?.$modal.msgWarning('请选择用户!');
  393. }
  394. };
  395. //减签
  396. const deleteMultiInstanceUser = async (row) => {
  397. await proxy?.$modal.confirm('是否确认提交?');
  398. loading.value = true;
  399. buttonDisabled.value = true;
  400. const taskOperationBo = reactive<TaskOperationBo>({
  401. userIds: [row.userId],
  402. taskId: taskId.value,
  403. message: form.value.message
  404. });
  405. await taskOperation(taskOperationBo, 'reductionSignature').finally(() => {
  406. loading.value = false;
  407. buttonDisabled.value = false;
  408. });
  409. dialog.visible = false;
  410. emits('submitCallback');
  411. proxy?.$modal.msgSuccess('操作成功');
  412. };
  413. //打开转办
  414. const openTransferTask = () => {
  415. transferTaskRef.value.open();
  416. };
  417. //转办
  418. const handleTransferTask = async (data) => {
  419. if (data && data.length > 0) {
  420. const taskOperationBo = reactive<TaskOperationBo>({
  421. userId: data[0].userId,
  422. taskId: taskId.value,
  423. message: form.value.message
  424. });
  425. await proxy?.$modal.confirm('是否确认提交?');
  426. loading.value = true;
  427. buttonDisabled.value = true;
  428. await taskOperation(taskOperationBo, 'transferTask').finally(() => {
  429. loading.value = false;
  430. buttonDisabled.value = false;
  431. });
  432. dialog.visible = false;
  433. emits('submitCallback');
  434. proxy?.$modal.msgSuccess('操作成功');
  435. } else {
  436. proxy?.$modal.msgWarning('请选择用户!');
  437. }
  438. };
  439. //打开委托
  440. const openDelegateTask = () => {
  441. delegateTaskRef.value.open();
  442. };
  443. //委托
  444. const handleDelegateTask = async (data) => {
  445. if (data && data.length > 0) {
  446. const taskOperationBo = reactive<TaskOperationBo>({
  447. userId: data[0].userId,
  448. taskId: taskId.value,
  449. message: form.value.message
  450. });
  451. await proxy?.$modal.confirm('是否确认提交?');
  452. loading.value = true;
  453. buttonDisabled.value = true;
  454. await taskOperation(taskOperationBo, 'delegateTask').finally(() => {
  455. loading.value = false;
  456. buttonDisabled.value = false;
  457. });
  458. dialog.visible = false;
  459. emits('submitCallback');
  460. proxy?.$modal.msgSuccess('操作成功');
  461. } else {
  462. proxy?.$modal.msgWarning('请选择用户!');
  463. }
  464. };
  465. //终止任务
  466. const handleTerminationTask = async () => {
  467. const params = {
  468. taskId: taskId.value,
  469. comment: form.value.message
  470. };
  471. await proxy?.$modal.confirm('是否确认终止?');
  472. loading.value = true;
  473. buttonDisabled.value = true;
  474. await terminationTask(params).finally(() => {
  475. loading.value = false;
  476. buttonDisabled.value = false;
  477. });
  478. dialog.visible = false;
  479. emits('submitCallback');
  480. proxy?.$modal.msgSuccess('操作成功');
  481. };
  482. const handleTaskUser = async () => {
  483. const data = await currentTaskAllUser(taskId.value);
  484. deleteUserList.value = data.data;
  485. if (deleteUserList.value && deleteUserList.value.length > 0) {
  486. deleteUserList.value.forEach((e) => {
  487. e.nodeName = task.value.nodeName;
  488. });
  489. }
  490. deleteSignatureVisible.value = true;
  491. };
  492. // 选择人员
  493. const choosePeople = async (data) => {
  494. if (!data.permissionFlag) {
  495. proxy?.$modal.msgError('没有可选择的人员,请联系管理员!');
  496. }
  497. popUserIds.value = data.permissionFlag;
  498. nodeCode.value = data.nodeCode;
  499. porUserRef.value.open();
  500. };
  501. //确认选择
  502. const handlePopUser = async (userList) => {
  503. const userIds = userList.map((item) => {
  504. return item.userId;
  505. });
  506. const nickNames = userList.map((item) => {
  507. return item.nickName;
  508. });
  509. form.value.assigneeMap[nodeCode.value] = userIds.join(',');
  510. nickName.value[nodeCode.value] = nickNames.join(',');
  511. };
  512. /**
  513. * 对外暴露子组件方法
  514. */
  515. defineExpose({
  516. openDialog
  517. });
  518. </script>