index.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368
  1. <template>
  2. <div>
  3. <el-dialog
  4. v-model="state.dialogVisible"
  5. draggable
  6. :title="state.title + '流程'"
  7. ref="dialogRef"
  8. @mouseup="mouseup"
  9. :style="'transform: ' + state.transform + ';'"
  10. append-to-body
  11. destroy-on-close
  12. @opened="opened"
  13. >
  14. <el-form :model="state.ruleForm" label-width="100px" ref="ruleFormRef">
  15. <el-row :gutter="35">
  16. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="!['return'].includes(state.processType)">
  17. <el-form-item label="下一环节" prop="nextStepCode" :rules="[{ required: true, message: '请选择下一环节', trigger: 'change' }]">
  18. <el-select v-model="state.ruleForm.nextStepCode" placeholder="请选择下一环节" class="w100" @change="selectNextStep">
  19. <el-option v-for="item in state.nextStepOptions" :key="item.key" :label="item.value" :value="item.key" />
  20. </el-select>
  21. </el-form-item>
  22. </el-col>
  23. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="!['return'].includes(state.processType)">
  24. <el-form-item label="处理人" prop="nextHandlers" :rules="[{ required: false, message: '请选择处理人', trigger: 'change' }]">
  25. <el-select
  26. v-model="state.ruleForm.nextHandlers"
  27. multiple
  28. filterable
  29. placeholder="请选择处理人"
  30. class="w100"
  31. @change="selectHandlers"
  32. value-key="key"
  33. clearable
  34. >
  35. <el-option v-for="item in state.handlerOptions" :key="item.key" :label="item.value" :value="item" />
  36. </el-select>
  37. </el-form-item>
  38. </el-col>
  39. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="!['return'].includes(state.processType) && showMainHandler">
  40. <el-form-item label="主办" prop="nextMainHandler" :rules="[{ required: false, message: '请选择主办', trigger: 'change' }]">
  41. <el-select v-model="state.ruleForm.nextMainHandler" placeholder="请选择主办" class="w100" filterable>
  42. <el-option v-for="item in state.handlerMainOptions" :key="item.key" :label="item.value" :value="item.key" />
  43. </el-select>
  44. </el-form-item>
  45. </el-col>
  46. <!-- <el-col :xs="24" :sm="12" :md="8" :lg="12" :xl="12">
  47. <el-form-item label="" prop="acceptSms">
  48. <el-checkbox v-model="state.ruleForm.acceptSms" label="短信通知" />
  49. </el-form-item>
  50. </el-col> -->
  51. <!-- 办理流程展示期满时间 -->
  52. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="['next'].includes(state.processType)">
  53. <el-form-item label="期满时间" prop="expiredTime" :rules="[{ required: true, message: '请选择期满时间', trigger: 'change' }]">
  54. <el-date-picker
  55. v-model="state.ruleForm.expiredTime"
  56. type="datetime"
  57. placeholder="请选择期满时间"
  58. value-format="YYYY-MM-DD[T]HH:mm:ss"
  59. class="w100"
  60. />
  61. </el-form-item>
  62. </el-col>
  63. <!-- 办理流程和开始流程是否发起会签 -->
  64. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="['next', 'start'].includes(state.processType)">
  65. <el-form-item
  66. label="是否发起会签"
  67. prop="isStartCountersign"
  68. :rules="[{ required: false, message: '请选择是否发起会签', trigger: 'change' }]"
  69. >
  70. <el-switch v-model="state.ruleForm.isStartCountersign" inline-prompt active-text="是" inactive-text="否" />
  71. </el-form-item>
  72. </el-col>
  73. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  74. <el-form-item :label="state.title + '意见'" prop="opinion" :rules="[{ required: true, message: '请填写常用意见', trigger: 'blur' }]">
  75. <common-advice
  76. @chooseAdvice="chooseAdvice"
  77. v-model="state.ruleForm.opinion"
  78. placeholder="请填写常用意见"
  79. :loading="state.loading"
  80. :commonEnum="state.commonEnum"
  81. />
  82. </el-form-item>
  83. </el-col>
  84. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  85. <el-form-item label="附件" prop="additions" :rules="[{ required: false, message: '请填写诉求内容', trigger: 'change' }]">
  86. <annex-list></annex-list>
  87. </el-form-item>
  88. </el-col>
  89. </el-row>
  90. </el-form>
  91. <template #footer>
  92. <span class="dialog-footer">
  93. <el-button @click="onCancel" class="default-button">取 消</el-button>
  94. <el-button type="primary" @click="onSubmit(ruleFormRef)" :loading="state.loading">提 交</el-button>
  95. </span>
  96. </template>
  97. </el-dialog>
  98. </div>
  99. </template>
  100. <script setup lang="ts" name="processApproval">
  101. import { defineAsyncComponent, reactive, ref, computed } from 'vue';
  102. import { ElMessage, ElMessageBox, FormInstance } from 'element-plus';
  103. import other from '/@/utils/other';
  104. import { OrderFlowParams, orderStartFlow } from '/@/api/business/order';
  105. import { workflowNextSteps, workflowNext, workflowRecall, workflowPrevious, workflowRecallParams, workflowStepOptions } from '/@/api/system/workflow';
  106. // 引入组件
  107. const CommonAdvice = defineAsyncComponent(() => import('/@/components/CommonAdvice/index.vue')); // 常用意见
  108. const AnnexList = defineAsyncComponent(() => import('/@/components/AnnexList/index.vue')); // 附件列表
  109. // 定义子组件向父组件传值/事件
  110. const emit = defineEmits(['orderProcessSuccess', 'orderProcessFailed']);
  111. // 定义变量内容
  112. const state = reactive<any>({
  113. dialogVisible: false, // 弹窗显示隐藏
  114. title: '提交', // 弹窗标题
  115. ruleForm: {
  116. acceptSms: false, // 是否接收短信
  117. opinion: '', // 意见
  118. nextHandlers: [], // 下一节点处理人
  119. nextMainHandler: '', // 主办人
  120. expiredTime: '', //期满时间
  121. isStartCountersign: false, // 是否发起会签
  122. },
  123. nextStepOptions: [], // 下一节点
  124. handlerOptions: [], // 处理人
  125. transform: 'translate(0px, 0px)', // 滚动条位置
  126. fileList: [], // 附件列表
  127. loading: false, // 提交按钮loading
  128. processType: 'next', // 流程状态
  129. workflowId: '', // 流程id
  130. commonEnum: '', // 常用意见类型
  131. handlerClassifies: [], //撤回处理人
  132. handlerMainOptions: [], // 主办人
  133. handleId: '', // 流程处理ID
  134. });
  135. const ruleFormRef = ref<RefType>(); //表单组件
  136. // 打开弹窗
  137. const openDialog = async (val: any) => {
  138. state.processType = val.processType ?? 'next'; // 流程状态
  139. state.ruleForm.workflowId = state.workflowId = val.id ?? ''; // 流程id
  140. state.commonEnum = val.commonEnum ?? ''; // 常用意见类型
  141. state.title = val.title ?? '提交流程'; // 流程标题
  142. let res: any = {};
  143. switch (state.processType) {
  144. case 'start': //开始流程
  145. res = await OrderFlowParams(); //获取开启流程参数
  146. state.nextStepOptions = res.result.steps;
  147. state.handleId = res.result.id;
  148. if (state.nextStepOptions.length === 1) {
  149. state.ruleForm.nextStepCode = state.nextStepOptions[0].key;
  150. getNextStepOption(res.result.id, state.nextStepOptions[0].key);
  151. }
  152. break;
  153. case 'recall': // 撤回流程
  154. res = await workflowRecallParams(state.workflowId); //撤回参数
  155. state.nextStepOptions = res.result.steps;
  156. state.handleId = res.result.id;
  157. if (state.nextStepOptions.length === 1) {
  158. state.ruleForm.nextStepCode = state.nextStepOptions[0].key;
  159. getNextStepOption(res.result.id, state.nextStepOptions[0].key);
  160. }
  161. break;
  162. case 'return': // 退回流程
  163. break;
  164. case 'next': // 默认下一流程
  165. res = await workflowNextSteps(state.workflowId);
  166. state.nextStepOptions = res.result.steps;
  167. state.ruleForm.expiredTime = res.result?.expiredTime ?? '';
  168. state.handleId = res.result.id;
  169. if (state.nextStepOptions.length === 1) {
  170. state.ruleForm.nextStepCode = state.nextStepOptions[0].key;
  171. getNextStepOption(res.result.id, state.nextStepOptions[0].key);
  172. }
  173. break;
  174. case 'delay': // 延期
  175. res = await workflowNextSteps(state.workflowId);
  176. state.nextStepOptions = res.result.steps;
  177. state.ruleForm.expiredTime = res.result?.expiredTime ?? '';
  178. state.handleId = res.result.id;
  179. if (state.nextStepOptions.length === 1) {
  180. state.ruleForm.nextStepCode = state.nextStepOptions[0].key;
  181. getNextStepOption(res.result.id, state.nextStepOptions[0].key);
  182. }
  183. break;
  184. case 'supervise': // 督办
  185. res = await workflowNextSteps(state.workflowId);
  186. state.nextStepOptions = res.result.steps;
  187. state.ruleForm.expiredTime = res.result?.expiredTime ?? '';
  188. state.handleId = res.result.id;
  189. if (state.nextStepOptions.length === 1) {
  190. state.ruleForm.nextStepCode = state.nextStepOptions[0].key;
  191. getNextStepOption(res.result.id, state.nextStepOptions[0].key);
  192. }
  193. break;
  194. default: // 默认下一流程
  195. res = await workflowNextSteps(state.workflowId);
  196. state.nextStepOptions = res.result.steps;
  197. state.ruleForm.expiredTime = res.result?.expiredTime ?? '';
  198. state.handleId = res.result.id;
  199. if (state.nextStepOptions.length === 1) {
  200. state.ruleForm.nextStepCode = state.nextStepOptions[0].key;
  201. getNextStepOption(res.result.id, state.nextStepOptions[0].key);
  202. }
  203. break;
  204. }
  205. state.dialogVisible = true;
  206. };
  207. // 打开弹窗清空表单
  208. const opened = () => {
  209. ruleFormRef.value?.clearValidate();
  210. ruleFormRef.value?.resetFields();
  211. };
  212. // 流程选择下一环节
  213. const selectNextStep = (val: any) => {
  214. ruleFormRef.value?.resetFields('nextHandlers');
  215. ruleFormRef.value?.resetFields('nextMainHandler');
  216. let next: any;
  217. next = state.nextStepOptions.find((item: any) => item.key === val);
  218. getNextStepOption(state.handleId, next.key);
  219. };
  220. // 查询流程下一节点参数
  221. const getNextStepOption = async (DefineId: string, Code: string) => {
  222. try {
  223. const res: any = await workflowStepOptions({ DefineId, Code });
  224. state.handlerOptions = res.result ?? [];
  225. } catch (error) {
  226. console.log(error);
  227. }
  228. };
  229. // 选择处理人
  230. const selectHandlers = () => {
  231. ruleFormRef.value?.resetFields('nextMainHandler');
  232. };
  233. const showMainHandler = computed(() => {
  234. return state.ruleForm.nextHandlers.length > 1;
  235. });
  236. // 主办从处理人中选择
  237. state.handlerMainOptions = computed(() => {
  238. return state.ruleForm.nextHandlers;
  239. });
  240. // 设置抽屉
  241. const dialogRef = ref<RefType>();
  242. const mouseup = () => {
  243. state.transform = dialogRef.value.dialogContentRef.$el.style.transform;
  244. };
  245. // 关闭弹窗
  246. const closeDialog = () => {
  247. state.dialogVisible = false;
  248. };
  249. // 取消
  250. const onCancel = () => {
  251. closeDialog();
  252. };
  253. // 选择常用意见 填入填写框
  254. const chooseAdvice = (item: any) => {
  255. state.ruleForm.opinion += item.content;
  256. };
  257. // 提交
  258. const onSubmit = (formEl: FormInstance | undefined) => {
  259. if (!formEl) return;
  260. formEl.validate((valid: boolean) => {
  261. if (!valid) return;
  262. ElMessageBox.confirm(`确认提交?`, '提示', {
  263. confirmButtonText: '确认',
  264. cancelButtonText: '取消',
  265. type: 'warning',
  266. draggable: true,
  267. cancelButtonClass: 'default-button',
  268. autofocus: false,
  269. })
  270. .then(() => {
  271. state.loading = true;
  272. state.ruleForm.additions = state.fileList;
  273. let submitObj = other.deepClone(state.ruleForm);
  274. if (submitObj.nextHandlers && submitObj.nextHandlers.length) {
  275. submitObj.nextHandlers = submitObj.nextHandlers.map((item: any) => {
  276. return {
  277. id: item.key,
  278. name: item.value,
  279. };
  280. });
  281. if (submitObj.nextHandlers.length === 1) {
  282. submitObj.nextMainHandler = submitObj.nextHandlers[0].id;
  283. }
  284. }
  285. switch (state.processType) {
  286. case 'start': //开始流程
  287. orderStartFlow(state.workflowId, submitObj)
  288. .then(() => {
  289. ElMessage.success('操作成功');
  290. state.loading = false;
  291. state.dialogVisible = false;
  292. emit('orderProcessSuccess');
  293. })
  294. .catch(() => {
  295. state.loading = false;
  296. state.dialogVisible = false;
  297. });
  298. break;
  299. case 'recall': // 撤回流程
  300. workflowRecall(submitObj)
  301. .then(() => {
  302. ElMessage.success('操作成功');
  303. state.loading = false;
  304. state.dialogVisible = false;
  305. emit('orderProcessSuccess');
  306. })
  307. .catch(() => {
  308. state.loading = false;
  309. state.dialogVisible = false;
  310. emit('orderProcessFailed');
  311. });
  312. break;
  313. case 'return': // 退回流程
  314. workflowPrevious(submitObj)
  315. .then(() => {
  316. ElMessage.success('操作成功');
  317. state.loading = false;
  318. state.dialogVisible = false;
  319. emit('orderProcessSuccess');
  320. })
  321. .catch(() => {
  322. state.loading = false;
  323. state.dialogVisible = false;
  324. emit('orderProcessFailed');
  325. });
  326. break;
  327. case 'next': // 默认下一流程
  328. workflowNext(submitObj)
  329. .then(() => {
  330. ElMessage.success('操作成功');
  331. state.loading = false;
  332. state.dialogVisible = false;
  333. emit('orderProcessSuccess');
  334. })
  335. .catch(() => {
  336. state.loading = false;
  337. state.dialogVisible = false;
  338. emit('orderProcessFailed');
  339. });
  340. break;
  341. default: // 默认下一流程
  342. workflowNext(submitObj)
  343. .then(() => {
  344. ElMessage.success('操作成功');
  345. state.loading = false;
  346. state.dialogVisible = false;
  347. emit('orderProcessSuccess');
  348. })
  349. .catch(() => {
  350. state.loading = false;
  351. state.dialogVisible = false;
  352. emit('orderProcessFailed');
  353. });
  354. break;
  355. }
  356. })
  357. .catch(() => {});
  358. });
  359. };
  360. // 暴露变量
  361. defineExpose({
  362. openDialog,
  363. closeDialog,
  364. });
  365. </script>