Delay-audit.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390
  1. <template>
  2. <el-dialog v-model="state.dialogVisible" draggable title="延期审批" ref="dialogRef" width="50%" append-to-body destroy-on-close @close="close">
  3. <el-form label-width="110px" :model="state.infoForm" class="show-info-form" v-loading="state.loading">
  4. <p class="border-title">申请信息</p>
  5. <el-row :gutter="10">
  6. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  7. <el-form-item label="工单编号"> {{ state.infoForm?.order?.no }} </el-form-item>
  8. </el-col>
  9. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  10. <el-form-item label="工单标题"> {{ state.infoForm?.order?.title }} </el-form-item>
  11. </el-col>
  12. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  13. <el-form-item label="申请人">{{ state.infoForm?.creatorName }} </el-form-item>
  14. </el-col>
  15. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  16. <el-form-item label="申请部门">{{ state.infoForm?.creatorOrgName }} </el-form-item>
  17. </el-col>
  18. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  19. <el-form-item label="申请时间">{{ formatDate(state.infoForm?.creationTime, 'YYYY-mm-dd HH:MM:SS') }} </el-form-item>
  20. </el-col>
  21. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  22. <el-form-item label="当前期满时间"> {{ formatDate(state.infoForm?.beforeDelay, 'YYYY-mm-dd HH:MM:SS') }} </el-form-item>
  23. </el-col>
  24. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  25. <el-form-item label="延期申请天数">
  26. <span v-if="!editMode"> {{ state.infoForm?.delayNum }} {{ state.infoForm?.delayUnitText }}</span>
  27. <template v-else>
  28. <el-input-number
  29. :min="1"
  30. :max="99"
  31. :precision="0"
  32. placeholder="延期申请天数"
  33. size="small"
  34. v-model="state.infoForm.delayNum"
  35. ></el-input-number>
  36. <el-button size="small" type="primary" class="ml5" @click="editConfirm">确定</el-button>
  37. <el-button size="small" class="decrease" @click="editMode = false">取消</el-button>
  38. </template>
  39. <el-button type="primary" link class="ml10" @click="editMode = true" v-if="!editMode && isAlterDay" v-auth="'business:delay:audit:edit'"
  40. >修改</el-button
  41. >
  42. </el-form-item>
  43. </el-col>
  44. <!-- <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  45. <el-form-item label="延期后期满时间">{{ formatDate(state.ruleForm?.afterDelay, 'YYYY-mm-dd HH:MM:SS') }} </el-form-item>
  46. </el-col>-->
  47. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  48. <el-form-item label="是否省延期">{{ state.infoForm?.isProDelay ? '是' : '否' }} </el-form-item>
  49. </el-col>
  50. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  51. <el-form-item label="申请理由" class="formatted-text">{{ state.infoForm?.delayReason }} </el-form-item>
  52. </el-col>
  53. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  54. <el-form-item label="附件"
  55. ><annex-list name="延期申请附件" v-model="state.infoForm.files" readonly classify="延期申请附件" />
  56. </el-form-item>
  57. </el-col>
  58. </el-row>
  59. </el-form>
  60. <p class="border-title mt10 mb10">延期审核明细</p>
  61. <vxe-table :expand-config="{ trigger: 'row', padding: true }" border :data="tableData" :loading="state.loading">
  62. <vxe-column type="seq" width="70"></vxe-column>
  63. <vxe-column type="expand" width="60">
  64. <template #content="{ row }">
  65. <div class="mb5 formatted-text">{{ row.handleModeTypeText }}:{{ row.opinion }}</div>
  66. <annex-list name="附件" readonly businessId="" classify="查看附件" v-model="row.files" />
  67. </template>
  68. </vxe-column>
  69. <vxe-column field="name" title="节点"></vxe-column>
  70. <vxe-column field="handlerName" title="审批人"></vxe-column>
  71. <vxe-column field="handleTime" title="审批时间" width="160">
  72. <template #default="{ row }">{{ formatDate(row.handleTime, 'YYYY-mm-dd HH:MM:SS') }}</template>
  73. </vxe-column>
  74. <vxe-column field="traceTypeText" title="状态"></vxe-column>
  75. </vxe-table>
  76. <p class="border-title mt20 mb10">审核意见</p>
  77. <el-form :model="state.ruleForm" label-width="110px" ref="ruleFormRef" v-loading="state.loading">
  78. <el-row :gutter="10">
  79. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  80. <el-form-item label="审批结果" prop="isPass" :rules="[{ required: true, message: '请选择审批结果', trigger: 'change' }]">
  81. <el-radio-group v-model="state.ruleForm.isPass">
  82. <el-radio :value="true">同意</el-radio>
  83. <el-radio :value="false">不同意</el-radio>
  84. </el-radio-group>
  85. </el-form-item>
  86. </el-col>
  87. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  88. <el-form-item label="" prop="isSms">
  89. <el-checkbox v-model="state.ruleForm.isSms" label="短信通知" />
  90. </el-form-item>
  91. </el-col>
  92. <template v-if="state.ruleForm.isPass">
  93. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  94. <el-form-item label="下一环节" prop="nextStepCode" :rules="[{ required: true, message: '请选择下一环节', trigger: 'change' }]">
  95. <el-select v-model="state.ruleForm.nextStepCode" placeholder="请选择下一环节" class="w100" @change="selectNextStep">
  96. <el-option v-for="item in state.nextStepOptions" :key="item.key" :label="item.value" :value="item.key" />
  97. </el-select>
  98. </el-form-item>
  99. </el-col>
  100. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="showHandlers">
  101. <el-form-item
  102. label="办理对象"
  103. prop="nextHandlers"
  104. :rules="[{ required: nextHandlersRequired, message: '请选择办理对象', trigger: 'change' }]"
  105. >
  106. <el-select-v2
  107. v-model="state.ruleForm.nextHandlers"
  108. :options="state.handlerOptions"
  109. placeholder="请选择办理对象"
  110. class="w100"
  111. multiple
  112. clearable
  113. collapse-tags
  114. collapse-tags-tooltip
  115. filterable
  116. value-key="key"
  117. @change="selectHandlers"
  118. :multiple-limit="multipleLimit"
  119. />
  120. </el-form-item>
  121. </el-col>
  122. </template>
  123. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  124. <el-form-item label="审批意见" prop="opinion" :rules="[{ required: true, message: '请填写审批意见', trigger: 'blur' }]">
  125. <common-advice
  126. @chooseAdvice="chooseAdvice"
  127. v-model="state.ruleForm.opinion"
  128. placeholder="请填写审批意见"
  129. :loading="state.loading"
  130. :commonEnum="commonEnum.Delay"
  131. :maxlength="AppConfigInfo.handleOpinionWordLimit"
  132. />
  133. </el-form-item>
  134. </el-col>
  135. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  136. <el-form-item label="附件" :rules="[{ required: false, message: '请上传附件', trigger: 'change' }]">
  137. <annex-list name="延期附件" classify="延期上传" v-model:format="handleFiles" />
  138. </el-form-item>
  139. </el-col>
  140. </el-row>
  141. </el-form>
  142. <template #footer>
  143. <span class="dialog-footer">
  144. <el-button @click="closeDialog" class="default-button">取 消</el-button>
  145. <el-button type="primary" @click="onSubmit(ruleFormRef)" :loading="state.loading">确定</el-button>
  146. </span>
  147. </template>
  148. </el-dialog>
  149. </template>
  150. <script setup lang="ts" name="discernDetail">
  151. import { computed, defineAsyncComponent, reactive, ref } from 'vue';
  152. import { formatDate } from '@/utils/formatTime';
  153. import { transformFile } from '@/utils/tools';
  154. import { ElMessage, FormInstance } from 'element-plus';
  155. import { workflowTraces } from '@/api/system/workflow';
  156. import { commonEnum } from '@/utils/constants';
  157. import { useAppConfig } from '@/stores/appConfig';
  158. import { storeToRefs } from 'pinia';
  159. import other from '@/utils/other';
  160. import { delayApproveModify, delayApproveParams, delayBatchApprove, delayDetail } from '@/api/business/delay';
  161. const AnnexList = defineAsyncComponent(() => import('@/components/AnnexList/index.vue')); // 附件列表
  162. const CommonAdvice = defineAsyncComponent(() => import('@/components/CommonAdvice/index.vue')); // 常用意见
  163. const appConfigStore = useAppConfig();
  164. const { AppConfigInfo } = storeToRefs(appConfigStore); // 系统配置信息
  165. const emit = defineEmits(['updateList']); // 定义事件
  166. // 定义变量内容
  167. const state = reactive<any>({
  168. dialogVisible: false, // 是否显示弹窗
  169. loading: false, // 是否显示加载
  170. infoForm: {},
  171. ruleForm: {
  172. isPass: true,
  173. nextHandlers: [],
  174. isSms: false,
  175. opinion: '', // 意见
  176. nextStepCode: '', // 下一节点
  177. nextStepName: '', // 下一节点名称
  178. backToCountersignEnd: false,
  179. stepId: null,
  180. workflowId: null,
  181. },
  182. handlerOptions: [],
  183. nextStepOptions: [],
  184. });
  185. /*
  186. * @params row 工单详情
  187. * @description 打开弹窗
  188. * */
  189. const currentRow = ref<EmptyObjectType>({});
  190. const openDialog = (row: any) => {
  191. currentRow.value = row;
  192. state.dialogVisible = true;
  193. getDetail();
  194. };
  195. // 获取延期和流程详情
  196. const getDetail = async () => {
  197. state.loading = true;
  198. try {
  199. const { result } = await delayDetail(currentRow.value.id);
  200. state.infoForm = result ?? {};
  201. state.infoForm.files = transformFile(state.infoForm.files);
  202. state.ruleForm.workflowId = result.workflowId;
  203. await getWorkflow(result.workflowId);
  204. await selectNextStepOptions(result.workflowId);
  205. state.loading = false;
  206. } catch (e) {
  207. console.log(e);
  208. state.loading = false;
  209. }
  210. };
  211. // 修改延期数量
  212. const editMode = ref<boolean>(false);
  213. // 确定修改延期数量
  214. const editConfirm = () => {
  215. delayApproveModify({ id: currentRow.value.id, delayNum: state.infoForm.delayNum })
  216. .then(() => {
  217. editMode.value = false;
  218. ElMessage.success('修改成功');
  219. getDetail();
  220. })
  221. .catch(() => {
  222. getDetail();
  223. });
  224. };
  225. const formatTraces = (val: any) => {
  226. if (!val || !val.length) return [];
  227. val.forEach((item: any) => {
  228. item.files = transformFile(item.files);
  229. // 需要给意见做一个映射关系 正常就是审核意见 其他状态就是其他意见 比如重办就是重办意见
  230. switch (item.handleMode) {
  231. case 0:
  232. item.handleModeTypeText = '审核意见';
  233. break;
  234. case 100:
  235. item.handleModeTypeText = '退回意见';
  236. break;
  237. case 200:
  238. item.handleModeTypeText = '特提意见';
  239. break;
  240. case 201:
  241. item.handleModeTypeText = '重办意见';
  242. break;
  243. case 203:
  244. item.handleModeTypeText = '发布退回意见';
  245. break;
  246. default:
  247. break;
  248. }
  249. });
  250. return val;
  251. };
  252. // 获取流程
  253. const tableData = ref<EmptyArrayType>([]);
  254. const getWorkflow = async (workflowId: string) => {
  255. try {
  256. // 查询审核记录
  257. const { result } = await workflowTraces(workflowId);
  258. tableData.value = formatTraces(result?.traces);
  259. } catch (e) {
  260. console.log(e);
  261. }
  262. };
  263. const nextHandlersRequired = ref<Boolean>(false); // 办理对象是否必填
  264. const canStartCountersign = ref<Boolean>(false); // 是否可以会签
  265. const selectNext = ref<any>({});
  266. const isAlterDay = ref<boolean>(false); // 是否可以修改申请天数
  267. // 查询流程办理节点
  268. const selectNextStepOptions = async (workflowId: string) => {
  269. try {
  270. const { result } = await delayApproveParams(workflowId); //获取延期审批流程参数
  271. isAlterDay.value = result.isAlterDay ?? false;
  272. state.nextStepOptions = result.steps; //办理对象选择内容
  273. canStartCountersign.value = result.canStartCountersign ?? false; // 是否可以发起会签
  274. if (state.nextStepOptions.length === 1) {
  275. // 下一节点是否只有一个 默认选中第一个
  276. setTimeout(() => {
  277. state.ruleForm.nextStepCode = state.nextStepOptions[0].key; // 下一节点code
  278. state.ruleForm.nextStepName = state.nextStepOptions[0].value; // 下一节点name
  279. state.ruleForm.backToCountersignEnd = state.nextStepOptions[0].backToCountersignEnd ?? false; // 是否回到会签结束节点
  280. }, 100);
  281. selectNextStep(state.nextStepOptions[0].key); // 查询流程下一节点参数
  282. } else {
  283. state.ruleForm.nextStepCode = '';
  284. state.ruleForm.nextStepName = '';
  285. }
  286. state.ruleForm.stepId = result.stepId;
  287. state.loading = false;
  288. } catch (e) {
  289. console.log(e);
  290. }
  291. };
  292. const selectNextStep = (val: any) => {
  293. ruleFormRef.value?.resetFields('nextHandlers');
  294. const next = state.nextStepOptions.find((item: any) => item.key === val);
  295. selectNext.value = next;
  296. const items = next.items; //获取下一节点
  297. state.ruleForm.nextStepName = next.value; // 下一节点name
  298. state.ruleForm.handlerType = next.handlerType;
  299. state.ruleForm.businessType = next.businessType;
  300. state.ruleForm.flowDirection = next.flowDirection;
  301. state.ruleForm.backToCountersignEnd = next.backToCountersignEnd ?? false; // 是否回到会签结束节点
  302. state.handlerOptions = items ?? [];
  303. state.handlerOptions = state.handlerOptions.map((item: any) => {
  304. return {
  305. value: {
  306. ...item,
  307. },
  308. label: item.value,
  309. };
  310. });
  311. // 办理对象是否必填
  312. nextHandlersRequired.value = ![0].includes(next.handlerType);
  313. if (items.length === 1) {
  314. // 如果办理对象只有一个默认选中
  315. state.ruleForm.nextHandlers = [items[0]];
  316. }
  317. };
  318. // 选择办理对象
  319. const selectHandlers = () => {
  320. if (state.ruleForm.nextHandlers.length <= 1) {
  321. // 如果只有一个办理对象就不需要发起会签
  322. state.ruleForm.isStartCountersign = false;
  323. }
  324. };
  325. // 是否展示办理对象 (结束节点不展示: next.stepType===2 表示为结束节点)
  326. const showHandlers = computed(() => {
  327. /*const next = state.nextStepOptions.find((item: any) => item.key === state.ruleForm.nextStepCode);
  328. if (!next) return true;
  329. // 话务部到派单 派单组到一级部门 也不需要展示办理对象
  330. return next?.stepType !== 2;*/
  331. return false;
  332. });
  333. // 办理对象是否能够选择多个(可以发起会签可以选择多个,不能发起会签只能选择一个) 加个判断 选择的下一环节必须是部门(会签必须是选择的部门)
  334. const multipleLimit = computed(() => {
  335. return canStartCountersign.value && selectNext.value.businessType === 2 ? 0 : 1;
  336. });
  337. // 选择常用意见 填入填写框 办理
  338. const chooseAdvice = (item: any) => {
  339. state.ruleForm.opinion += item.content;
  340. };
  341. // 关闭弹窗
  342. const closeDialog = () => {
  343. state.dialogVisible = false;
  344. };
  345. const close = () => {
  346. ruleFormRef.value?.resetFields();
  347. ruleFormRef.value?.resetFields();
  348. };
  349. // 提交
  350. const afterSubmit = (emitType?: 'updateList', showMessage?: boolean, message?: string) => {
  351. state.loading = false;
  352. closeDialog();
  353. const msg = message ?? '延期审批成功';
  354. if (showMessage) ElMessage.success(msg);
  355. if (emitType) emit(emitType);
  356. };
  357. const ruleFormRef = ref<RefType>();
  358. const handleFiles = ref<EmptyArrayType>([]); // 附件
  359. const onSubmit = (formEl: FormInstance | undefined) => {
  360. if (!formEl) return;
  361. formEl.validate((valid: boolean) => {
  362. if (!valid) return;
  363. state.loading = true;
  364. let submitObj = other.deepClone(state.ruleForm);
  365. const request = {
  366. nextWorkflow: submitObj,
  367. files: handleFiles.value,
  368. delayId: [currentRow.value.id],
  369. isPass: state.ruleForm.isPass,
  370. };
  371. delayBatchApprove(request)
  372. .then(() => {
  373. afterSubmit('updateList', true);
  374. })
  375. .catch((err: any) => {
  376. console.log(err, 'err');
  377. afterSubmit('updateList');
  378. });
  379. });
  380. };
  381. defineExpose({
  382. openDialog,
  383. closeDialog,
  384. });
  385. </script>