index.vue 56 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413
  1. <template>
  2. <el-dialog
  3. v-model="state.dialogVisible"
  4. draggable
  5. :title="state.dialogTitle"
  6. ref="dialogRef"
  7. @mouseup="mouseup"
  8. :style="'transform: ' + state.transform + ';'"
  9. destroy-on-close
  10. :close-on-click-modal="false"
  11. @close="close"
  12. :modal-class="isOrderAccept ? 'modal_class' : ''"
  13. :class="isOrderAccept ? 'dialog_class' : ''"
  14. :append-to-body="!isOrderAccept"
  15. :modal="!isOrderAccept"
  16. >
  17. <el-steps :active="activeStep" align-center finish-status="success" class="mb20" v-if="showStepsArr.includes(state.processType)">
  18. <el-step title="业务表单" />
  19. <el-step title="流程表单" />
  20. </el-steps>
  21. <div v-show="activeStep === 0" v-loading="state.loading">
  22. <el-form :model="state.delayForm" label-width="110px" ref="delayFormRef" v-if="state.processType === '延期申请'">
  23. <el-row :gutter="10">
  24. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  25. <el-form-item label="工单编码"> {{ state.orderDetail.no }} </el-form-item>
  26. </el-col>
  27. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  28. <el-form-item label="工单标题"> {{ state.orderDetail.title }} </el-form-item>
  29. </el-col>
  30. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  31. <el-form-item label="申请人"> {{ userInfos.name }} </el-form-item>
  32. </el-col>
  33. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  34. <el-form-item label="申请部门"> {{ userInfos.orgName }} </el-form-item>
  35. </el-col>
  36. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  37. <el-form-item label="申请时间"> {{ formatDate(new Date(), 'YYYY-mm-dd HH:MM:SS') }} </el-form-item>
  38. </el-col>
  39. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  40. <el-form-item label="当前期满时间"> {{ formatDate(state.orderDetail.expiredTime, 'YYYY-mm-dd HH:MM:SS') }} </el-form-item>
  41. </el-col>
  42. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="12" v-loading="state.loading">
  43. <el-form-item prop="timeLimitCount" :rules="[{ required: true, message: '请填写延期申请数量', trigger: 'blur' }]" label-width="130px">
  44. <template #label>
  45. <div style="height: 34px; display: flex; align-items: center">
  46. 延期申请数量
  47. <el-tooltip placement="top-start">
  48. <SvgIcon name="ele-QuestionFilled" size="18px" class="ml3" />
  49. <template #content> 延期申请数量为1-{{ AppConfigInfo.applyDelayTime }} </template>
  50. </el-tooltip>
  51. </div>
  52. </template>
  53. <el-row :gutter="10" class="w100">
  54. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  55. <el-input-number
  56. placeholder="延期申请数量"
  57. v-model="state.delayForm.timeLimitCount"
  58. controls-position="right"
  59. :min="1"
  60. :max="AppConfigInfo.applyDelayTime"
  61. class="w100"
  62. ></el-input-number>
  63. </el-col>
  64. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-loading="state.loading">
  65. <el-form-item
  66. label=""
  67. label-width="0"
  68. prop="timeLimitUnit"
  69. :rules="[{ required: true, message: '请选择延期申请单位', trigger: 'change' }]"
  70. >
  71. <el-select disabled v-model="state.delayForm.timeLimitUnit" placeholder="延期申请单位">
  72. <el-option v-for="item in timeType" :value="item.key" :key="item.key" :label="item.value" />
  73. </el-select>
  74. </el-form-item>
  75. </el-col>
  76. </el-row>
  77. </el-form-item>
  78. </el-col>
  79. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  80. <el-form-item label="申请理由" prop="content" :rules="[{ required: true, message: '请填写延期申请理由', trigger: 'blur' }]">
  81. <common-advice
  82. @chooseAdvice="chooseAdviceDelay"
  83. v-model="state.delayForm.content"
  84. placeholder="请填写延期申请理由"
  85. :loading="state.loading"
  86. :commonEnum="commonOpinionType"
  87. :maxlength="AppConfigInfo.handleOpinionWordLimit"
  88. />
  89. </el-form-item>
  90. </el-col>
  91. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  92. <el-form-item label="附件">
  93. <annex-list name="延期附件" v-model:format="handleFilesDelay" :businessId="state.orderDetail.id" classify="延期上传" />
  94. </el-form-item>
  95. </el-col>
  96. </el-row>
  97. </el-form>
  98. <el-form :model="state.discernForm" label-width="110px" ref="discernFormRef" v-if="state.processType === '甄别申请'">
  99. <el-row :gutter="10">
  100. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  101. <el-form-item label="工单编码"> {{ state.orderDetail.no }} </el-form-item>
  102. </el-col>
  103. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  104. <el-form-item label="工单标题"> {{ state.orderDetail.title }} </el-form-item>
  105. </el-col>
  106. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  107. <el-form-item label="申请人"> {{ userInfos.name }} </el-form-item>
  108. </el-col>
  109. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  110. <el-form-item label="申请部门"> {{ userInfos.orgName }} </el-form-item>
  111. </el-col>
  112. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  113. <el-form-item label="申请时间"> {{ formatDate(Date(),'YYYY-mm-dd HH:MM:SS') }} </el-form-item>
  114. </el-col>
  115. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  116. <el-form-item label="申请类型" prop="type" :rules="[{ required: true, message: '请选择申请类型', trigger: 'change' }]">
  117. <el-select v-model="state.discernForm.type" placeholder="请选择申请类型" class="w100" value-key="dicDataValue">
  118. <el-option v-for="item in screenTypeOptions" :value="item" :key="item.dicDataValue" :label="item.dicDataName" />
  119. </el-select>
  120. </el-form-item>
  121. </el-col>
  122. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  123. <el-form-item label="申请理由" prop="content" :rules="[{ required: true, message: '请填写甄别申请理由', trigger: 'blur' }]">
  124. <common-advice
  125. @chooseAdvice="chooseAdviceDiscern"
  126. v-model="state.discernForm.content"
  127. placeholder="请填写甄别申请理由"
  128. :loading="state.loading"
  129. :commonEnum="commonOpinionType"
  130. :maxlength="AppConfigInfo.handleOpinionWordLimit"
  131. />
  132. </el-form-item>
  133. </el-col>
  134. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  135. <el-form-item label="附件">
  136. <annex-list
  137. name="甄别附件"
  138. ref="discernAnnexListRef"
  139. v-model:format="handleFilesDiscern"
  140. :businessId="state.orderDetail.id"
  141. classify="甄别上传"
  142. />
  143. </el-form-item>
  144. </el-col>
  145. </el-row>
  146. </el-form>
  147. <el-form :model="state.terminateForm" label-width="110px" ref="terminateFormRef" v-if="state.processType === '终止申请'">
  148. <el-row :gutter="10">
  149. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  150. <el-form-item label="工单编码"> {{ state.orderDetail.no }} </el-form-item>
  151. </el-col>
  152. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  153. <el-form-item label="工单标题"> {{ state.orderDetail.title }} </el-form-item>
  154. </el-col>
  155. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  156. <el-form-item label="终止理由" prop="content" :rules="[{ required: true, message: '请填写终止申请理由', trigger: 'blur' }]">
  157. <common-advice
  158. @chooseAdvice="chooseAdviceTerminate"
  159. v-model="state.terminateForm.content"
  160. placeholder="请填写终止申请理由"
  161. :loading="state.loading"
  162. :commonEnum="commonOpinionType"
  163. :maxlength="AppConfigInfo.handleOpinionWordLimit"
  164. />
  165. </el-form-item>
  166. </el-col>
  167. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  168. <el-form-item label="附件">
  169. <annex-list
  170. name="终止附件"
  171. ref="terminateAnnexListRef"
  172. v-model:format="handleFilesTerminate"
  173. :businessId="state.orderDetail.id"
  174. classify="终止上传"
  175. />
  176. </el-form-item>
  177. </el-col>
  178. </el-row>
  179. </el-form>
  180. </div>
  181. <el-form :model="state.ruleForm" label-width="110px" ref="ruleFormRef" v-show="activeStep === 1" v-loading="state.loading">
  182. <slot name="header"></slot>
  183. <el-row :gutter="10">
  184. <!-- 审批流程 -->
  185. <template v-if="auditArr.includes(state.processType) && canReject">
  186. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="!returnArr.includes(state.processType)">
  187. <el-form-item label="审批结果" prop="isPass" :rules="[{ required: true, message: '请选择审批结果', trigger: 'change' }]">
  188. <el-radio-group v-model="state.ruleForm.isPass">
  189. <el-radio :value="true">同意</el-radio>
  190. <el-radio :value="false">不同意</el-radio>
  191. </el-radio-group>
  192. </el-form-item>
  193. </el-col>
  194. <!-- 审批通过 -->
  195. <template v-if="state.ruleForm.isPass">
  196. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  197. <el-form-item label="下一环节" prop="nextStepCode" :rules="[{ required: true, message: '请选择下一环节', trigger: 'change' }]">
  198. <el-select v-model="state.ruleForm.nextStepCode" placeholder="请选择下一环节" class="w100" @change="selectNextStep">
  199. <el-option v-for="item in state.nextStepOptions" :key="item.key" :label="item.value" :value="item.key" />
  200. </el-select>
  201. <p class="flex-center-align color-danger" v-if="showFastSendOrder">
  202. 当前推荐派单办理对象:{{ fastStepName }} <el-button type="primary" link class="ml4" @click="fastSendOrder">快捷派单</el-button>
  203. </p>
  204. </el-form-item>
  205. </el-col>
  206. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="!returnArr.includes(state.processType) && showHandlers">
  207. <el-form-item
  208. label="办理对象"
  209. prop="nextHandlers"
  210. :rules="[{ required: nextHandlersRequired, message: '请选择办理对象', trigger: 'change' }]"
  211. v-if="!returnArr.includes(state.processType) && showHandlers"
  212. >
  213. <el-select-v2
  214. v-model="state.ruleForm.nextHandlers"
  215. :options="state.handlerOptions"
  216. placeholder="请选择办理对象"
  217. class="w100"
  218. multiple
  219. clearable
  220. collapse-tags
  221. collapse-tags-tooltip
  222. filterable
  223. value-key="key"
  224. @change="selectHandlers"
  225. :multiple-limit="multipleLimit"
  226. />
  227. </el-form-item>
  228. </el-col>
  229. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="showMainHandler">
  230. <el-form-item label="主办" prop="nextMainHandler" :rules="[{ required: false, message: '请选择主办', trigger: 'change' }]">
  231. <el-select v-model="state.ruleForm.nextMainHandler" placeholder="请选择主办" class="w100" filterable>
  232. <el-option v-for="item in state.handlerMainOptions" :key="item.key" :label="item.value" :value="item.key" />
  233. </el-select>
  234. </el-form-item>
  235. </el-col>
  236. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="countersignAble">
  237. <el-form-item label="发起会签" prop="isStartCountersign" :rules="[{ required: false, message: '请选择发起会签', trigger: 'change' }]">
  238. <el-switch
  239. v-model="state.ruleForm.isStartCountersign"
  240. inline-prompt
  241. active-text="是"
  242. inactive-text="否"
  243. @change="changeStartCountersign"
  244. :disabled="countersignDisabled"
  245. />
  246. </el-form-item>
  247. </el-col>
  248. </template>
  249. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  250. <el-form-item label="" prop="isSms">
  251. <el-checkbox v-model="state.ruleForm.isSms" label="短信通知" />
  252. </el-form-item>
  253. </el-col>
  254. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  255. <el-form-item
  256. :label="state.inputPlaceholder"
  257. prop="opinion"
  258. :rules="[{ required: true, message: `请填写${state.inputPlaceholder}`, trigger: 'blur' }]"
  259. >
  260. <common-advice
  261. @chooseAdvice="chooseAdvice"
  262. v-model="state.ruleForm.opinion"
  263. :placeholder="'请填写' + state.inputPlaceholder"
  264. :loading="state.loading"
  265. :commonEnum="commonOpinionType"
  266. :maxlength="AppConfigInfo.handleOpinionWordLimit"
  267. />
  268. </el-form-item>
  269. </el-col>
  270. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  271. <el-form-item label="附件" :rules="[{ required: false, message: '请填写诉求内容', trigger: 'change' }]">
  272. <annex-list :name="state.annexName" :businessId="state.orderDetail.id" :classify="state.classify" v-model:format="handleFiles" />
  273. </el-form-item>
  274. </el-col>
  275. </template>
  276. <!-- 办理流程 -->
  277. <template v-else>
  278. <!-- 非退回流程都需要选择 -->
  279. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="!returnArr.includes(state.processType)">
  280. <el-form-item label="下一环节" prop="nextStepCode" :rules="[{ required: true, message: '请选择下一环节', trigger: 'change' }]">
  281. <el-select v-model="state.ruleForm.nextStepCode" placeholder="请选择下一环节" class="w100" @change="selectNextStep">
  282. <el-option v-for="item in state.nextStepOptions" :key="item.key" :label="item.value" :value="item.key" />
  283. </el-select>
  284. <p class="flex-center-align color-danger" v-if="showFastSendOrder">
  285. 当前推荐派单办理对象:{{ fastStepName }} <el-button type="primary" link class="ml4" @click="fastSendOrder">快捷派单</el-button>
  286. </p>
  287. </el-form-item>
  288. </el-col>
  289. <!-- 非退回流程都需要选择并且如果选择了结束节点就不需要选择办理对象 -->
  290. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="!returnArr.includes(state.processType) && showHandlers">
  291. <el-form-item
  292. label="办理对象"
  293. prop="nextHandlers"
  294. :rules="[{ required: nextHandlersRequired, message: '请选择办理对象', trigger: 'change' }]"
  295. >
  296. <el-select-v2
  297. v-model="state.ruleForm.nextHandlers"
  298. :options="state.handlerOptions"
  299. placeholder="请选择办理对象"
  300. class="w100"
  301. multiple
  302. clearable
  303. collapse-tags
  304. collapse-tags-tooltip
  305. filterable
  306. value-key="key"
  307. @change="selectHandlers"
  308. :multiple-limit="multipleLimit"
  309. />
  310. </el-form-item>
  311. </el-col>
  312. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="showMainHandler">
  313. <el-form-item label="主办" prop="nextMainHandler" :rules="[{ required: false, message: '请选择主办', trigger: 'change' }]">
  314. <el-select v-model="state.ruleForm.nextMainHandler" placeholder="请选择主办" class="w100" filterable>
  315. <el-option v-for="item in state.handlerMainOptions" :key="item.key" :label="item.value" :value="item.key" />
  316. </el-select>
  317. </el-form-item>
  318. </el-col>
  319. <!-- 工单办理专有参数 -->
  320. <template v-if="flowDirection">
  321. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  322. <el-form-item label="办理时限" prop="timeLimit" :rules="[{ required: true, message: '请填写办理时限', trigger: 'blur' }]">
  323. <el-row :gutter="10">
  324. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  325. <el-input-number
  326. placeholder="办理时限"
  327. v-model="state.ruleForm.timeLimit"
  328. controls-position="right"
  329. class="w100"
  330. :min="1"
  331. :max="99"
  332. disabled
  333. ></el-input-number>
  334. </el-col>
  335. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-loading="state.loading">
  336. <el-form-item
  337. label=""
  338. label-width="0"
  339. prop="timeLimitUnit"
  340. :rules="[{ required: true, message: '请选择办理时限单位', trigger: 'change' }]"
  341. >
  342. <el-select v-model="state.ruleForm.timeLimitUnit" placeholder="办理时限单位" disabled style="width: 240px">
  343. <el-option v-for="item in timeTypeOptions" :value="item.key" :key="item.key" :label="item.value" />
  344. </el-select>
  345. </el-form-item>
  346. </el-col>
  347. </el-row>
  348. </el-form-item>
  349. </el-col>
  350. </template>
  351. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="countersignAble">
  352. <el-form-item label="发起会签" prop="isStartCountersign" :rules="[{ required: false, message: '请选择发起会签', trigger: 'change' }]">
  353. <el-switch
  354. v-model="state.ruleForm.isStartCountersign"
  355. inline-prompt
  356. active-text="是"
  357. inactive-text="否"
  358. @change="changeStartCountersign"
  359. :disabled="countersignDisabled"
  360. />
  361. </el-form-item>
  362. </el-col>
  363. <!-- 汇总节点需要填写 并且是工单办理 -->
  364. <template v-if="inputRealHandler">
  365. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  366. <el-form-item label="经办人" prop="realHandlerName" :rules="[{ required: false, message: '请填写经办人', trigger: 'blur' }]">
  367. <el-input v-model="state.ruleForm.realHandlerName" placeholder="请填写经办人" clearable> </el-input>
  368. </el-form-item>
  369. </el-col>
  370. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  371. <el-form-item label="经办人电话" prop="realHandlerPhone" :rules="[{ required: false, message: '请填写经办人电话', trigger: 'blur' }]">
  372. <el-input v-model="state.ruleForm.realHandlerPhone" placeholder="请填写办理人电话" 请填写经办人电话> </el-input>
  373. </el-form-item>
  374. </el-col>
  375. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  376. <el-form-item label="是否与市民沟通" prop="realIsContacted">
  377. <el-switch v-model="state.ruleForm.realIsContacted" inline-prompt active-text="是" inactive-text="否" />
  378. </el-form-item>
  379. </el-col>
  380. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  381. <el-form-item label="是否与市民现场沟通" prop="realContactLocale" label-width="150">
  382. <el-switch v-model="state.ruleForm.realContactLocale" inline-prompt active-text="是" inactive-text="否" />
  383. </el-form-item>
  384. </el-col>
  385. </template>
  386. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12" v-if="isSmsSelectShow">
  387. <el-form-item label="" prop="isSms">
  388. <el-checkbox v-model="state.ruleForm.isSms" label="短信通知" />
  389. </el-form-item>
  390. </el-col>
  391. <!-- 选择结束节点时并且是工单办理并且不是中心,显示部门处理结果 -->
  392. <template v-if="showResult">
  393. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  394. <el-form-item label="处理结果" prop="isResolved" :rules="[{ required: true, message: '请选择处理结果', trigger: 'change' }]">
  395. <el-radio-group v-model="state.ruleForm.isResolved">
  396. <el-radio :value="true">已得到解决</el-radio>
  397. <el-radio :value="false">未得到解决</el-radio>
  398. </el-radio-group>
  399. </el-form-item>
  400. </el-col>
  401. </template>
  402. <!-- 选择结束节点时并且是工单办理并且工单来源是110时,显示警情退回 -->
  403. <template v-if="showPoliceReturn">
  404. <el-col :xs="24" :sm="12" :md="12" :lg="12" :xl="12">
  405. <el-form-item label="" prop="isPoliceReturn">
  406. <el-checkbox v-model="state.ruleForm.isPoliceReturn" label="警情退回" />
  407. </el-form-item>
  408. </el-col>
  409. </template>
  410. <template v-if="!['延期申请', '甄别申请', '终止申请'].includes(state.processType)">
  411. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  412. <el-form-item
  413. :label="state.inputPlaceholder"
  414. prop="opinion"
  415. :rules="[{ required: true, message: `请填写${state.inputPlaceholder}`, trigger: 'blur' }]"
  416. >
  417. <common-advice
  418. @chooseAdvice="chooseAdvice"
  419. v-model="state.ruleForm.opinion"
  420. :placeholder="'请填写' + state.inputPlaceholder"
  421. :loading="state.loading"
  422. :commonEnum="commonOpinionType"
  423. :maxlength="AppConfigInfo.handleOpinionWordLimit"
  424. />
  425. </el-form-item>
  426. </el-col>
  427. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24" v-if="showRemarkTextarea">
  428. <el-form-item label="备注信息" prop="remark" :rules="[{ required: false, message: `请填写备注信息`, trigger: 'blur' }]">
  429. <el-input v-model="state.ruleForm.remark" :rows="2" type="textarea" placeholder="请填写备注信息" />
  430. </el-form-item>
  431. </el-col>
  432. <el-col :xs="24" :sm="24" :md="24" :lg="24" :xl="24">
  433. <el-form-item label="附件" :rules="[{ required: false, message: '请填写诉求内容', trigger: 'change' }]">
  434. <annex-list :name="state.annexName" :businessId="state.orderDetail.id" :classify="state.classify" v-model:format="handleFiles" />
  435. </el-form-item>
  436. </el-col>
  437. </template>
  438. </template>
  439. </el-row>
  440. </el-form>
  441. <template #footer>
  442. <span class="dialog-footer">
  443. <el-text type="danger" class="mr15" tag="b" v-if="showTempSave">温馨提示:临时保存只保存{{ state.inputPlaceholder }}</el-text>
  444. <el-button @click="closeDialog" class="default-button">取 消</el-button>
  445. <el-button type="primary" @click="handleTempSave" :loading="state.loading" v-if="showTempSave">临时保存</el-button>
  446. <el-button
  447. class="default-button"
  448. @click="onPrevious"
  449. :loading="state.loading"
  450. v-if="activeStep === 1 && showStepsArr.includes(state.processType)"
  451. >上一步</el-button
  452. >
  453. <el-button class="default-button" @click="onNext" :loading="state.loading" v-if="activeStep === 0 && showStepsArr.includes(state.processType)"
  454. >下一步</el-button
  455. >
  456. <el-popconfirm
  457. title="该工单属于超期状态,若符合延期要求,请延期通过后办结,是否继续办理?"
  458. v-if="isOverdueTips"
  459. @confirm="onSubmit(ruleFormRef)"
  460. width="300"
  461. >
  462. <template #reference>
  463. <el-button type="primary" :loading="state.loading">办理</el-button>
  464. </template>
  465. </el-popconfirm>
  466. <el-button type="primary" @click="onSubmit(ruleFormRef)" :loading="state.loading" v-else-if="!isOverdueTips && activeStep === 1"
  467. >办理</el-button
  468. >
  469. </span>
  470. </template>
  471. </el-dialog>
  472. </template>
  473. <script setup lang="ts" name="processApproval">
  474. import { computed, defineAsyncComponent, nextTick, reactive, ref, watch, watchEffect } from 'vue';
  475. import { ElMessage, FormInstance } from 'element-plus';
  476. import other from '@/utils/other';
  477. import { useUserInfo } from '@/stores/userInfo';
  478. import { storeToRefs } from 'pinia';
  479. import { commonEnum } from '@/utils/constants';
  480. import { orderFlowParams, orderHandle, orderProcessTempSave, orderStartFlow, orderTimeConfig } from '@/api/business/order';
  481. import {
  482. orderPrevious,
  483. workflowNext,
  484. workflowNextSteps,
  485. workflowNextStepsByOrder,
  486. workflowNextStepsByOrderInstead,
  487. workflowPrevious,
  488. workflowRecall,
  489. workflowRecallParams,
  490. workflowReject,
  491. } from '@/api/system/workflow';
  492. import { delayApply, delayApproveParams, delayBaseData, delayCalcEndTime, workflowDelayParams } from '@/api/business/delay';
  493. import { discernApply, discernApproveParams, screenBaseData, workflowDiscernParams } from '@/api/business/discern';
  494. import { debounce } from '@/utils/tools';
  495. import {
  496. KnowledgeAdd,
  497. KnowledgeAddStartFlowParams,
  498. KnowledgeDel,
  499. KnowledgeDeleteStartFlowParams,
  500. KnowledgeOffShelf,
  501. KnowledgeOffShelfStartFlowParams,
  502. KnowledgeUpdate,
  503. KnowledgeUpdateStartFlowParams,
  504. } from '@/api/knowledge';
  505. import { formatDate } from '@/utils/formatTime';
  506. import { useAppConfig } from '@/stores/appConfig';
  507. import { terminateNextFlowParams, terminateStartFlow, terminateStartFlowParams } from '@/api/business/terminate';
  508. // 引入组件
  509. const CommonAdvice = defineAsyncComponent(() => import('@/components/CommonAdvice/index.vue')); // 常用意见
  510. const AnnexList = defineAsyncComponent(() => import('@/components/AnnexList/index.vue')); // 附件列表
  511. // 定义子组件向父组件传值/事件
  512. const emit = defineEmits(['orderProcessSuccess', 'orderProcessFailed']);
  513. // 定义变量内容
  514. const state = reactive<any>({
  515. dialogVisible: false, // 弹窗显示隐藏
  516. ruleForm: {
  517. isPass: true, // 审批结果
  518. //流程表单
  519. opinion: '', // 意见
  520. nextStepCode: '', // 下一节点
  521. nextStepName: '', // 下一节点名称
  522. backToCountersignEnd: false, // 是否回到会签结束节点
  523. nextHandlers: [], // 下一节点办理对象
  524. nextMainHandler: '', // 主办人
  525. isSms: false, // 是否短信通知
  526. isStartCountersign: false, // 是否发起会签
  527. stepId: null, // 步骤id
  528. // isTransferHandle:false, // 是否转办
  529. },
  530. delayForm: {
  531. //延期申请表单
  532. timeLimitCount: null, // 延期申请数量
  533. content: '', // 延期申请理由
  534. timeLimitUnit: 2, // 延期申请单位 默认工作日
  535. },
  536. discernForm: {
  537. // 甄别表单
  538. content: '', // 甄别理由
  539. },
  540. redoForm: {
  541. // 重办表单
  542. content: '', // 重办理由
  543. },
  544. terminateForm: {
  545. //终止表单
  546. content: '', // 终止理由
  547. },
  548. nextStepOptions: [], // 下一节点
  549. handlerOptions: [], // 办理对象
  550. transform: 'translate(0px, 0px)', // 滚动条位置
  551. loading: false, // 提交按钮loading
  552. processType: 'next', // 流程状态
  553. workflowId: '', // 流程id
  554. handlerClassifies: [], //撤回办理对象
  555. handlerMainOptions: [], // 主办人
  556. handleId: '', // 流程处理ID
  557. dialogTitle: '', // 弹窗标题
  558. annexName: '', // 附件标题
  559. inputPlaceholder: '办理意见', // 意见提示
  560. orderDetail: {}, // 工单详情
  561. });
  562. const ruleFormRef = ref<RefType>(); //表单组件
  563. const storesUserInfo = useUserInfo();
  564. const { userInfos } = storeToRefs(storesUserInfo); // 用户信息
  565. const showStepsArr = ['延期申请', '甄别申请', '终止申请']; // 显示步骤条的流程
  566. const handelArr = ['工单办理']; // 处于办理状态的流程 (如果是汇总节点 需要填写办理对象等 办理流程才有期满时间)
  567. const returnArr = ['工单退回', '甄别退回', '延期退回', '终止退回']; // 退回流程 (退回流程不需要展示其他 只需要填写意见和附件即可)
  568. const auditArr = ['甄别审批', '延期审批', '知识审批', '终止审批']; // 审批流程 (审批流程需要选择是否通过和下一环节)
  569. const appConfigStore = useAppConfig();
  570. const { AppConfigInfo } = storeToRefs(appConfigStore); // 系统配置信息
  571. const showTempSave = computed(() => {
  572. // 是否显示临时办理
  573. return ['工单办理', '工单代办', '工单受理'].includes(state.processType);
  574. });
  575. const timeType = ref<EmptyArrayType>([]); // 延期申请单位
  576. const screenTypeOptions = ref<EmptyArrayType>([]); // 甄别类型
  577. const screenType = ref(null); // 甄别类型 自贡甄别特殊需求
  578. // 打开弹窗
  579. const openDialog = async (val: any) => {
  580. console.log(val);
  581. selectNext.value = {};
  582. currentParams.value = {};
  583. state.ruleForm.nextHandlers = [];
  584. state.ruleForm.nextHandler = {};
  585. isSmsSelectShow.value = false;
  586. state.ruleForm.isSms = false;
  587. state.loading = true;
  588. state.dialogVisible = true;
  589. try {
  590. const { id, processType, extra, orderDetail } = val;
  591. state.ruleForm.workflowId = state.workflowId = id ?? ''; // 流程id
  592. state.processType = processType ?? '办理流程'; // 业务类型
  593. state.orderDetail = orderDetail ?? {}; // 工单详情
  594. const { dialogTitle, inputPlaceholder, annexName, classify } = extra ?? {};
  595. state.dialogTitle = dialogTitle ?? '提交流程'; // 流程标题
  596. state.annexName = annexName ?? '办理附件'; // 附件标题
  597. state.classify = classify ?? '办理上传'; // 附件分类
  598. state.inputPlaceholder = inputPlaceholder ?? '办理意见'; // 意见提示
  599. screenType.value = extra.screenType;
  600. switch (state.processType) {
  601. case '工单受理': //开始流程
  602. const [orderStartWorkflowResponse] = await Promise.all([orderFlowParams(orderDetail.id)]); //获取开启流程参数
  603. handleResult(orderStartWorkflowResponse);
  604. break;
  605. case '工单撤回': // 撤回流程
  606. const [workflowRecallResponse] = await Promise.all([workflowRecallParams(state.workflowId)]); //获取开启流程参数
  607. handleResult(workflowRecallResponse);
  608. break;
  609. case '延期申请': // 延期申请
  610. const [workflowDelayResponse, responseDelay] = await Promise.all([workflowDelayParams(), delayBaseData()]); //获取开启流程参数
  611. timeType.value = responseDelay.result?.timeType ?? []; // 延期时间单位
  612. handleResult(workflowDelayResponse);
  613. break;
  614. case '延期审批': // 延期审批
  615. const [nextResponseDelayAudit] = await Promise.all([delayApproveParams(state.workflowId)]); //获取延期审批流程参数
  616. handleResult(nextResponseDelayAudit);
  617. break;
  618. case '甄别申请': // 甄别申请
  619. const [workflowDiscernResponse, responseDiscern] = await Promise.all([workflowDiscernParams(), screenBaseData()]); //获取开启流程参数
  620. screenTypeOptions.value = responseDiscern.result?.screenType ?? []; // 甄别理由
  621. handleResult(workflowDiscernResponse);
  622. break;
  623. case '终止申请': // 终止申请
  624. const [workflowTerminateResponse] = await Promise.all([terminateStartFlowParams()]); //获取开启流程参数
  625. handleResult(workflowTerminateResponse);
  626. break;
  627. case '工单退回': // 退回流程
  628. break;
  629. case '甄别退回': // 退回流程
  630. break;
  631. case '甄别审批': // 甄别审批
  632. const [nextResponseDiscernAudit] = await Promise.all([discernApproveParams(state.workflowId)]); //获取甄别审批流程参数
  633. handleResult(nextResponseDiscernAudit);
  634. break;
  635. case '终止审批': // 终止审批
  636. const [nextResponseTerminateAudit] = await Promise.all([terminateNextFlowParams(state.workflowId)]); //获取终止审批流程参数
  637. handleResult(nextResponseTerminateAudit);
  638. break;
  639. case '延期退回': // 退回流程
  640. break;
  641. case '工单办理': // 工单办理
  642. const [workflowNextStepsResponse] = await Promise.all([workflowNextStepsByOrder(state.orderDetail.id)]); // 获取下一节点和下一节点参数
  643. handleResult(workflowNextStepsResponse);
  644. break;
  645. case '工单代办':
  646. const [orderHandleInstanceResponse] = await Promise.all([workflowNextStepsByOrderInstead(state.orderDetail.id)]); // 工单代办参数获取
  647. handleResult(orderHandleInstanceResponse);
  648. break;
  649. case '新增知识':
  650. const [KnowledgeAddStartFlowResponse] = await Promise.all([KnowledgeAddStartFlowParams()]); // 知识库新增参数
  651. handleResult(KnowledgeAddStartFlowResponse);
  652. break;
  653. case '更新知识':
  654. const [KnowledgeUpdateResponse] = await Promise.all([KnowledgeUpdateStartFlowParams()]); // 知识库更新参数
  655. handleResult(KnowledgeUpdateResponse);
  656. break;
  657. case '更新新增知识':
  658. const [KnowledgeAddUpdateResponse] = await Promise.all([KnowledgeAddStartFlowParams()]); // 知识库更新参数
  659. handleResult(KnowledgeAddUpdateResponse);
  660. break;
  661. case '删除知识':
  662. const [KnowledgeDeleteResponse] = await Promise.all([KnowledgeDeleteStartFlowParams()]); // 知识库删除参数
  663. handleResult(KnowledgeDeleteResponse);
  664. break;
  665. case '知识下架':
  666. const [KnowledgeOffShelfResponse] = await Promise.all([KnowledgeOffShelfStartFlowParams()]); // 知识库下架参数
  667. handleResult(KnowledgeOffShelfResponse);
  668. break;
  669. default: // 默认下一流程 工单办理
  670. // 获取下一节点和下一节点参数
  671. const [nextResponse] = await Promise.all([workflowNextSteps(state.workflowId)]);
  672. handleResult(nextResponse);
  673. break;
  674. }
  675. await nextTick(() => {
  676. restForm(ruleFormRef.value);
  677. });
  678. } finally {
  679. state.loading = false;
  680. }
  681. };
  682. const isOrderAccept = computed(() => {
  683. return state.processType === '工单受理';
  684. });
  685. // 常用意见类型
  686. const commonOpinionType = computed(() => {
  687. switch (state.processType) {
  688. case '工单受理':
  689. return commonEnum.Seat;
  690. case '工单退回':
  691. return commonEnum.Return;
  692. case '工单办理':
  693. return commonEnum.OrderCirculation;
  694. case '甄别审批':
  695. case '甄别申请':
  696. return commonEnum.Discriminate;
  697. case '延期审批':
  698. case '延期申请':
  699. return commonEnum.Delay;
  700. case '终止审批':
  701. case '终止申请':
  702. return commonEnum.OrderCirculation;
  703. case '知识审批':
  704. return commonEnum.KnowledgeL;
  705. default:
  706. return commonEnum.OrderCirculation;
  707. }
  708. });
  709. const canReject = ref<boolean>(false); // 是否可以驳回
  710. const timeTypeOptions = ref<EmptyArrayType>([]); // 办理时限单位
  711. const canStartCountersign = ref<boolean>(false); // 是否可以发起会签
  712. const isMainHandlerShow = ref<boolean>(false); // 是否展示主办人
  713. const currentParams = ref<EmptyObjectType>({}); // 当前获取到的参数(当前节点信息)
  714. const handleResult = (res: any) => {
  715. currentParams.value = res.result; // 当前获取到的参数(当前节点信息)
  716. state.nextStepOptions = res.result.steps; //办理对象选择内容
  717. canReject.value = res.result.canReject ?? false; // 是否可以驳回
  718. timeTypeOptions.value = res.result.timeTypeOptions ?? []; // 办理时限申请单位
  719. canStartCountersign.value = res.result.canStartCountersign ?? false; // 是否可以发起会签
  720. isMainHandlerShow.value = res.result.isMainHandlerShow ?? false; // 是否展示主办人
  721. if (handelArr.includes(state.processType)) {
  722. // 办理才有期满时间
  723. state.ruleForm.expiredTime = res.result.expiredTime ?? null; // 期满时间
  724. }
  725. if (state.nextStepOptions.length === 1) {
  726. // 下一节点是否只有一个 默认选中第一个
  727. setTimeout(() => {
  728. state.ruleForm.nextStepCode = state.nextStepOptions[0].key; // 下一节点code
  729. state.ruleForm.nextStepName = state.nextStepOptions[0].value; // 下一节点name
  730. state.ruleForm.backToCountersignEnd = state.nextStepOptions[0].backToCountersignEnd ?? false; // 是否回到会签结束节点
  731. }, 100);
  732. selectNextStep(state.nextStepOptions[0].key); // 查询流程下一节点参数
  733. } else {
  734. state.ruleForm.nextStepCode = '';
  735. state.ruleForm.nextStepName = '';
  736. }
  737. if (res.result.opinion) {
  738. // 如果有汇总意见没有临时保存 取汇总意见 临时保存会覆盖汇总意见
  739. setTimeout(() => {
  740. state.ruleForm.opinion = res.result.opinion;
  741. }, 100);
  742. }
  743. state.ruleForm.stepId = res.result.stepId;
  744. state.loading = false;
  745. };
  746. // 上一部
  747. const onPrevious = () => {
  748. activeStep.value = 0;
  749. };
  750. const delayFormRef = ref<RefType>(); //延期申请表单组件
  751. const discernFormRef = ref<RefType>(); //甄别申请表单组件
  752. const terminateFormRef = ref<RefType>(); // 终止申请表单组件
  753. const redoFormRef = ref<RefType>(); //重办表单组件
  754. // 下一步
  755. const onNext = () => {
  756. switch (state.processType) {
  757. case '延期申请':
  758. delayFormRef.value?.validate((valid: boolean) => {
  759. if (!valid) return;
  760. activeStep.value = 1;
  761. });
  762. break;
  763. case '甄别申请': // 甄别申请
  764. discernFormRef.value?.validate((valid: boolean) => {
  765. if (!valid) return;
  766. activeStep.value = 1;
  767. });
  768. break;
  769. case '终止申请': // 终止申请
  770. terminateFormRef.value?.validate((valid: boolean) => {
  771. if (!valid) return;
  772. activeStep.value = 1;
  773. });
  774. break;
  775. default: // 默认下一流程
  776. activeStep.value = 1;
  777. break;
  778. }
  779. };
  780. const activeStep = ref(0); //步骤条
  781. watchEffect(() => {
  782. if (state.dialogVisible) {
  783. activeStep.value = showStepsArr.includes(state.processType) ? 0 : 1;
  784. }
  785. });
  786. // 流程选择下一环节
  787. const fastStepCode = ref(''); // 推荐派单处理对象code
  788. const fastStepName = ref(''); // 推荐派单处理对象
  789. // expiredStatus 超期状态(0正常 1即将超期 2已超期) 工单流转选择“结束”节点,点击“办理”时需验证该工单是否处于已超期状态
  790. const isOverdueTips = computed(() => {
  791. return state.orderDetail.expiredStatus === 2 && state.ruleForm.nextStepCode === 'end';
  792. });
  793. // 先确定当前环节是那一步
  794. /*
  795. * currentParams.value.currentStepBusinessType 表示当前节点到哪一步 0坐席 1派单 2部门节点 3部门领导节点
  796. * */
  797. /*selectNext.value.businessType 表示选择节点到哪一步 0 坐席 1派单 2部门节点 3部门领导节点
  798. * selectNext.value.orgLevel 表示部门等级 1 一级部门 2 二级部门 3 三级部门 4 四级部门
  799. * selectNext.value.stepType 3汇总节点
  800. * */
  801. // 话务部到派单组
  802. const seatTopaidan = computed(() => {
  803. return currentParams.value.currentStepBusinessType === 0 && selectNext.value.businessType === 1;
  804. });
  805. // 派单组到部门
  806. const paidanToOrg = computed(() => {
  807. return currentParams.value.currentStepBusinessType === 1 && selectNext.value.businessType === 2;
  808. });
  809. // 是否展示短信通知选择
  810. const isSmsSelectShow = ref(false);
  811. // 办理对象是否必填 0-角色 1-部门类型 2-部门类型 3-指定用户 4-指定部门
  812. // 2024/10/28 新增甄别申请 延期申请 终止申请选择办理对象非必填 !showStepsArr.includes(state.processType)
  813. const nextHandlersRequired = ref<Boolean>(false);
  814. const selectNext = ref<EmptyObjectType>({}); // 选择的下一个节点
  815. const selectNextStep = (val: any) => {
  816. ruleFormRef.value?.resetFields('nextHandlers');
  817. ruleFormRef.value?.resetFields('nextMainHandler');
  818. ruleFormRef.value?.resetFields('isStartCountersign');
  819. const next = state.nextStepOptions.find((item: any) => item.key === val);
  820. selectNext.value = next;
  821. const items = next.items; //获取下一节点
  822. state.ruleForm.nextStepName = next.value; // 下一节点name
  823. state.ruleForm.handlerType = next.handlerType;
  824. state.ruleForm.businessType = next.businessType;
  825. state.ruleForm.flowDirection = next.flowDirection;
  826. state.ruleForm.backToCountersignEnd = next.backToCountersignEnd ?? false; // 是否回到会签结束节点
  827. state.handlerOptions = items ?? [];
  828. state.handlerOptions = state.handlerOptions.map((item: any) => {
  829. return {
  830. value: {
  831. ...item,
  832. },
  833. label: item.value,
  834. };
  835. });
  836. fastStepName.value = next.recommendOrgName; // 推荐派单处理对象
  837. fastStepCode.value = next.recommendOrgId; // 推荐派单处理对象code
  838. if (items.length === 1) {
  839. // 如果办理对象只有一个默认选中
  840. state.ruleForm.nextHandlers = [items[0]];
  841. }
  842. // 办理对象是否必填
  843. // nextHandlersRequired.value = ![0].includes(next.handlerType) && !showStepsArr.includes(state.processType);
  844. nextHandlersRequired.value = ![0].includes(next.handlerType);
  845. // 以下是默认需要吧短信勾上的场景 派单组到部门
  846. state.ruleForm.isSms = paidanToOrg.value;
  847. // 是否展示短信通知 (取消坐席流转至“派单组”时的短信通知选项框 )
  848. isSmsSelectShow.value = !seatTopaidan.value;
  849. };
  850. // 会签是否可用 (配置可以会签并且是非结束节点)
  851. const countersignAble = computed(() => {
  852. return canStartCountersign.value && selectNext.value.stepType !== 2;
  853. });
  854. // 办理对象是否能够选择多个(可以发起会签可以选择多个,不能发起会签只能选择一个)
  855. const multipleLimit = computed(() => {
  856. return canStartCountersign.value ? 0 : 1;
  857. });
  858. watch(
  859. () => state.ruleForm.nextHandlers, // 监听办理对象 多个办理对象自动发起会签
  860. (val) => {
  861. state.ruleForm.isStartCountersign = val.length > 1;
  862. }
  863. );
  864. const countersignDisabled = computed(() => {
  865. // 如果选择的办理对象超过1个就不不能修改发起会签 并且选择下一环节必须是部门节点
  866. return state.ruleForm.nextHandlers.length !== 1 || selectNext.value.businessType !== 2;
  867. });
  868. // 是否发起会签
  869. const changeStartCountersign = (val: boolean) => {
  870. /* if (!val) {
  871. // 如果不能会签清空办理对象
  872. state.ruleForm.nextHandlers = [];
  873. ruleFormRef.value?.resetFields('nextHandlers');
  874. }*/
  875. };
  876. // 是否展示办理对象 (结束节点不展示: next.stepType===2 表示为结束节点,下一环节为派单组时 next.businessType === 1,办理对象下拉框隐藏:AppConfigInfo.value.isAverageSendOrder= true 表示开启了平均派单 )
  877. const showHandlers = computed(() => {
  878. const next = state.nextStepOptions.find((item: any) => item.key === state.ruleForm.nextStepCode);
  879. const isAverageSendOrder = AppConfigInfo.value.isAverageSendOrder && next?.businessType === 1; // 开启平均派单并且下一个环节是派单组 !isAverageSendOrder
  880. if (!next) return true;
  881. return next?.stepType !== 2;
  882. });
  883. // 是否展示部门处理结果 ( 选择结束节点时并且当前操作人不是中心时,显示部门处理结果 )
  884. const showResult = computed(() => {
  885. const next = state.nextStepOptions.find((item: any) => item.key === state.ruleForm.nextStepCode);
  886. return next?.stepType === 2 && !userInfos.value.isCenter;
  887. });
  888. // 是否展示警情退回 (选择结束节点时并且工单来源是110时,显示警情退回)
  889. const showPoliceReturn = computed(() => {
  890. const next = state.nextStepOptions.find((item: any) => item.key === state.ruleForm.nextStepCode);
  891. return next?.stepType === 2 && state.orderDetail?.source === 200;
  892. });
  893. // 是否是汇总节点(汇总需要填入其他参数)并且是工单办理
  894. const inputRealHandler = computed(() => {
  895. const next = state.nextStepOptions.find((item: any) => item.key === state.ruleForm.nextStepCode);
  896. return next?.inputRealHandler && handelArr.includes(state.processType);
  897. });
  898. // 是否显示快捷派单
  899. const showFastSendOrder = computed(() => {
  900. const next = state.nextStepOptions.find((item: any) => item.key === state.ruleForm.nextStepCode);
  901. if (!next) return false;
  902. return next?.recommendOrgName && next?.recommendOrgId;
  903. });
  904. // 快速派单
  905. const fastSendOrder = () => {
  906. if (!fastStepCode.value) return;
  907. // 如果办理对象中没有推荐派单的对象就添加
  908. if (!state.ruleForm.nextHandlers.find((item: any) => item.key === fastStepCode.value)) {
  909. const next = state.handlerOptions.find((item: any) => item.value.key === fastStepCode.value);
  910. if (next) state.ruleForm.nextHandlers = [...state.ruleForm.nextHandlers, next.value];
  911. }
  912. };
  913. // 是否展示办理时限 flowDirection 并且是工单办理
  914. const flowDirection = computed(() => {
  915. const next = state.nextStepOptions.find((item: any) => item.key === state.ruleForm.nextStepCode);
  916. if (!next) return false;
  917. return [0, 1].includes(next.flowDirection) && handelArr.includes(state.processType);
  918. });
  919. // 是否展示备注输入框
  920. const showRemarkTextarea = computed(() => {
  921. return ['工单受理', '工单办理'].includes(state.processType);
  922. });
  923. // 计算工单期满时间
  924. watch(
  925. () => flowDirection.value,
  926. (val) => {
  927. if (val) {
  928. const next = state.nextStepOptions.find((item: any) => item.key === state.ruleForm.nextStepCode);
  929. state.ruleForm.flowDirection = next.flowDirection;
  930. orderTimeConfig({ flowDirection: next.flowDirection, acceptTypeCode: state.orderDetail.acceptTypeCode }).then((res: any) => {
  931. state.ruleForm.timeLimit = res.result.count ?? null; // 办理时限
  932. state.ruleForm.timeLimitUnit = res.result.timeType ?? ''; // 办理时限单位
  933. });
  934. }
  935. }
  936. );
  937. // 选择办理对象
  938. const selectHandlers = () => {
  939. ruleFormRef.value?.resetFields('nextMainHandler');
  940. if (state.ruleForm.nextHandlers.length > 1) {
  941. // 多个办理对象 主办
  942. state.ruleForm.nextMainHandler = state.ruleForm.nextHandlers[0].key;
  943. // AppConfigInfo.value.noSignOrgCode 对应的办理对象不能参与会签
  944. const isProvince12345 = state.ruleForm.nextHandlers.find((item: any) => AppConfigInfo.value.noSignOrgCode.includes(item.key));
  945. if (isProvince12345) {
  946. // 如果选择了省12345平台或者省12345交办就提示不能参与会签 并且从选择中移除
  947. ElMessage({
  948. message: '省12345平台和省12345交办不能参与会签',
  949. grouping: true,
  950. });
  951. state.ruleForm.nextHandlers = state.ruleForm.nextHandlers.filter((item: any) => !AppConfigInfo.value.noSignOrgCode.includes(item.key));
  952. }
  953. }
  954. };
  955. // 是否展示主办
  956. const showMainHandler = computed(() => {
  957. return state.ruleForm.nextHandlers.length > 1 && isMainHandlerShow.value;
  958. });
  959. // 主办从办理对象中选择
  960. state.handlerMainOptions = computed(() => {
  961. return state.ruleForm.nextHandlers;
  962. });
  963. // 设置抽屉
  964. const dialogRef = ref<RefType>();
  965. const mouseup = () => {
  966. state.transform = dialogRef.value.dialogContentRef.$el.style.transform;
  967. };
  968. // 关闭弹窗
  969. const closeDialog = () => {
  970. state.dialogVisible = false;
  971. };
  972. // 重置表单方法
  973. const restForm = (formEl: FormInstance | undefined) => {
  974. if (!formEl) return;
  975. state.ruleForm.opinion = '';
  976. state.delayForm.endTime = '';
  977. formEl.resetFields();
  978. formEl.clearValidate();
  979. };
  980. // 选择常用意见 填入填写框 办理
  981. const chooseAdvice = (item: any) => {
  982. state.ruleForm.opinion += item.content;
  983. };
  984. // 选择常用意见 填入填写框 延期
  985. const chooseAdviceDelay = (item: any) => {
  986. state.delayForm.content += item.content;
  987. };
  988. // 选择常用意见 填入填写框 甄别
  989. const chooseAdviceDiscern = (item: any) => {
  990. state.discernForm.content += item.content;
  991. };
  992. // 选择常用意见 填入填写框 终止
  993. const chooseAdviceTerminate = (item: any) => {
  994. state.terminateForm.content += item.content;
  995. };
  996. const afterSubmit = (emitType?: 'orderProcessSuccess' | 'orderProcessFailed', showMessage?: boolean, message?: string) => {
  997. state.loading = false;
  998. closeDialog();
  999. const msg = message ?? '操作成功';
  1000. if (showMessage) ElMessage.success(msg);
  1001. if (emitType) emit(emitType);
  1002. };
  1003. const close = () => {
  1004. restForm(ruleFormRef.value);
  1005. restForm(delayFormRef.value);
  1006. restForm(discernFormRef.value);
  1007. restForm(terminateFormRef.value);
  1008. restForm(redoFormRef.value);
  1009. };
  1010. // 办理
  1011. const handleFiles = ref<EmptyArrayType>([]); // 流程附件
  1012. const handleFilesDelay = ref<EmptyArrayType>([]); // 延期附件
  1013. const handleFilesDiscern = ref<EmptyArrayType>([]); // 甄别附件
  1014. const handleFilesTerminate = ref<EmptyArrayType>([]); // 终止附件
  1015. const onSubmit = (formEl: FormInstance | undefined) => {
  1016. if (!formEl) return;
  1017. formEl.validate((valid: boolean) => {
  1018. if (!valid) return;
  1019. state.loading = true;
  1020. let submitObj = other.deepClone(state.ruleForm);
  1021. if (submitObj.nextHandlers && submitObj.nextHandlers.length) {
  1022. if (submitObj.nextHandlers.length === 1) {
  1023. submitObj.nextMainHandler = submitObj.nextHandlers[0].key;
  1024. }
  1025. }
  1026. if (!flowDirection.value) {
  1027. // 需要填写办理时限
  1028. Reflect.deleteProperty(submitObj, 'timeLimit');
  1029. Reflect.deleteProperty(submitObj, 'timeLimitUnit');
  1030. } else {
  1031. submitObj.external = {
  1032. timeLimit: state.ruleForm.timeLimit,
  1033. timeLimitUnit: state.ruleForm.timeLimitUnit,
  1034. };
  1035. }
  1036. Reflect.deleteProperty(submitObj, 'isPoliceReturn');
  1037. Reflect.deleteProperty(submitObj, 'isResolved');
  1038. submitObj.external = {
  1039. isPoliceReturn: state.ruleForm.isPoliceReturn,
  1040. isResolved: state.ruleForm.isResolved,
  1041. };
  1042. // submitObj.stepExpiredTime = submitObj.expiredTime; //节点过期时间
  1043. switch (state.processType) {
  1044. case '工单受理':
  1045. const request = {
  1046. data: { orderId: state.orderDetail.id, ...submitObj },
  1047. workflow: { ...submitObj, files: handleFiles.value },
  1048. };
  1049. orderStartFlow(request)
  1050. .then(() => {
  1051. afterSubmit('orderProcessSuccess', true);
  1052. })
  1053. .catch(() => {
  1054. afterSubmit('orderProcessFailed');
  1055. });
  1056. break;
  1057. case '工单撤回':
  1058. workflowRecall({ ...submitObj, files: handleFiles.value })
  1059. .then(() => {
  1060. afterSubmit('orderProcessSuccess', true);
  1061. })
  1062. .catch(() => {
  1063. afterSubmit('orderProcessFailed');
  1064. });
  1065. break;
  1066. case '延期申请':
  1067. const requestDelay = {
  1068. data: {
  1069. orderId: state.orderDetail.id,
  1070. delayNum: state.delayForm.timeLimitCount,
  1071. delayUnit: state.delayForm.timeLimitUnit,
  1072. delayReason: state.delayForm.content,
  1073. files: handleFilesDelay.value,
  1074. },
  1075. workflow: {
  1076. ...submitObj,
  1077. opinion: state.delayForm.content,
  1078. files: handleFilesDelay.value,
  1079. },
  1080. };
  1081. delayApply(requestDelay)
  1082. .then(() => {
  1083. afterSubmit('orderProcessSuccess', true, '延期申请成功');
  1084. })
  1085. .catch(() => {
  1086. afterSubmit('orderProcessFailed');
  1087. });
  1088. break;
  1089. case '延期退回':
  1090. workflowPrevious({ ...submitObj, files: handleFiles.value })
  1091. .then(() => {
  1092. afterSubmit('orderProcessSuccess', true);
  1093. })
  1094. .catch(() => {
  1095. afterSubmit('orderProcessFailed');
  1096. });
  1097. break;
  1098. case '延期审批':
  1099. if (state.ruleForm.isPass) {
  1100. // 审批通过 下一步
  1101. workflowNext({ ...submitObj, reviewResult: 1, files: handleFiles.value })
  1102. .then(() => {
  1103. afterSubmit('orderProcessSuccess', true);
  1104. })
  1105. .catch(() => {
  1106. afterSubmit('orderProcessFailed');
  1107. });
  1108. } else {
  1109. // 审批拒绝
  1110. const requestDelayAudit = {
  1111. opinion: state.ruleForm.opinion,
  1112. workflowId: state.workflowId,
  1113. files: handleFiles.value,
  1114. stepId: state.ruleForm.stepId,
  1115. };
  1116. workflowReject(requestDelayAudit)
  1117. .then(() => {
  1118. afterSubmit('orderProcessSuccess', true);
  1119. })
  1120. .catch(() => {
  1121. afterSubmit('orderProcessFailed');
  1122. });
  1123. }
  1124. break;
  1125. case '甄别申请':
  1126. const requestDiscern = {
  1127. data: {
  1128. no: state.orderDetail.no,
  1129. visitId: state.orderDetail.visitId,
  1130. visitDetailId: state.orderDetail.visitDetailId,
  1131. orderId: state.orderDetail.id,
  1132. typeDicId: state.discernForm.type.dicDataValue,
  1133. typeDicName: state.discernForm.type.dicDataName,
  1134. content: state.discernForm.content,
  1135. screenType: screenType.value,
  1136. files: handleFilesDiscern.value,
  1137. },
  1138. workflow: { ...submitObj, files: handleFilesDiscern.value, opinion: state.discernForm.content },
  1139. };
  1140. discernApply(requestDiscern)
  1141. .then(() => {
  1142. afterSubmit('orderProcessSuccess', true, '甄别申请成功');
  1143. })
  1144. .catch(() => {
  1145. afterSubmit('orderProcessFailed');
  1146. });
  1147. break;
  1148. case '甄别退回':
  1149. workflowPrevious({ ...submitObj, files: handleFiles.value })
  1150. .then(() => {
  1151. afterSubmit('orderProcessSuccess', true);
  1152. })
  1153. .catch(() => {
  1154. afterSubmit('orderProcessFailed');
  1155. });
  1156. break;
  1157. case '甄别审批':
  1158. if (state.ruleForm.isPass) {
  1159. // 审批通过 下一步
  1160. workflowNext({ ...submitObj, reviewResult: 1, files: handleFiles.value })
  1161. .then(() => {
  1162. afterSubmit('orderProcessSuccess', true);
  1163. })
  1164. .catch(() => {
  1165. afterSubmit('orderProcessFailed');
  1166. });
  1167. } else {
  1168. // 审批拒绝
  1169. const requestDiscernAudit = {
  1170. opinion: state.ruleForm.opinion,
  1171. workflowId: state.workflowId,
  1172. files: handleFiles.value,
  1173. stepId: state.ruleForm.stepId,
  1174. };
  1175. workflowReject(requestDiscernAudit)
  1176. .then(() => {
  1177. afterSubmit('orderProcessSuccess', true);
  1178. })
  1179. .catch(() => {
  1180. afterSubmit('orderProcessFailed');
  1181. });
  1182. }
  1183. break;
  1184. case '终止申请':
  1185. const requestTerminate = {
  1186. data: {
  1187. no: state.orderDetail.no,
  1188. orderId: state.orderDetail.id,
  1189. content: state.terminateForm.content,
  1190. files: handleFilesTerminate.value,
  1191. },
  1192. workflow: { ...submitObj, files: handleFilesTerminate.value, opinion: state.terminateForm.content },
  1193. };
  1194. terminateStartFlow(requestTerminate)
  1195. .then(() => {
  1196. afterSubmit('orderProcessSuccess', true, '终止申请成功');
  1197. })
  1198. .catch(() => {
  1199. afterSubmit('orderProcessFailed');
  1200. });
  1201. break;
  1202. case '终止审批':
  1203. if (state.ruleForm.isPass) {
  1204. // 审批通过 下一步
  1205. workflowNext({ ...submitObj, reviewResult: 1, files: handleFiles.value })
  1206. .then(() => {
  1207. afterSubmit('orderProcessSuccess', true);
  1208. })
  1209. .catch(() => {
  1210. afterSubmit('orderProcessFailed');
  1211. });
  1212. } else {
  1213. // 审批拒绝
  1214. const requestTerminateAudit = {
  1215. opinion: state.ruleForm.opinion,
  1216. workflowId: state.workflowId,
  1217. files: handleFiles.value,
  1218. stepId: state.ruleForm.stepId,
  1219. };
  1220. workflowReject(requestTerminateAudit)
  1221. .then(() => {
  1222. afterSubmit('orderProcessSuccess', true);
  1223. })
  1224. .catch(() => {
  1225. afterSubmit('orderProcessFailed');
  1226. });
  1227. }
  1228. break;
  1229. case '终止退回':
  1230. workflowPrevious({ ...submitObj, files: handleFiles.value })
  1231. .then(() => {
  1232. afterSubmit('orderProcessSuccess', true);
  1233. })
  1234. .catch(() => {
  1235. afterSubmit('orderProcessFailed');
  1236. });
  1237. break;
  1238. case '工单退回':
  1239. orderPrevious({ ...submitObj, files: handleFiles.value })
  1240. .then(() => {
  1241. afterSubmit('orderProcessSuccess', true, '退回申请成功');
  1242. })
  1243. .catch(() => {
  1244. afterSubmit('orderProcessFailed');
  1245. });
  1246. break;
  1247. case '工单办理': // 工单办理流程
  1248. case '工单代办': // 工单代办流程
  1249. const requestHandle = {
  1250. data: { orderId: state.orderDetail.id, ...submitObj },
  1251. workflow: { ...submitObj, files: handleFiles.value },
  1252. };
  1253. orderHandle(requestHandle)
  1254. .then(() => {
  1255. afterSubmit('orderProcessSuccess', true);
  1256. })
  1257. .catch(() => {
  1258. afterSubmit('orderProcessFailed');
  1259. });
  1260. break;
  1261. case '新增知识':
  1262. const KnowledgeAddRequest = {
  1263. data: { ...state.orderDetail },
  1264. workflow: { ...submitObj, files: handleFiles.value },
  1265. };
  1266. KnowledgeAdd(KnowledgeAddRequest)
  1267. .then(() => {
  1268. afterSubmit('orderProcessSuccess', true, '新增知识成功');
  1269. })
  1270. .catch(() => {
  1271. afterSubmit('orderProcessFailed');
  1272. });
  1273. break;
  1274. case '知识审批':
  1275. if (state.ruleForm.isPass) {
  1276. // 审批通过 下一步
  1277. workflowNext({ ...submitObj, reviewResult: 1, files: handleFiles.value })
  1278. .then(() => {
  1279. afterSubmit('orderProcessSuccess', true);
  1280. })
  1281. .catch(() => {
  1282. afterSubmit('orderProcessFailed');
  1283. });
  1284. } else {
  1285. // 审批拒绝
  1286. const requestDiscernAudit = {
  1287. opinion: state.ruleForm.opinion,
  1288. workflowId: state.workflowId,
  1289. files: handleFiles.value,
  1290. stepId: state.ruleForm.stepId,
  1291. };
  1292. workflowReject(requestDiscernAudit)
  1293. .then(() => {
  1294. afterSubmit('orderProcessSuccess', true);
  1295. })
  1296. .catch(() => {
  1297. afterSubmit('orderProcessFailed');
  1298. });
  1299. }
  1300. break;
  1301. case '更新新增知识':
  1302. const KnowledgeAddUpdateRequest = {
  1303. data: { ...state.orderDetail },
  1304. workflow: { ...submitObj, files: handleFiles.value },
  1305. };
  1306. KnowledgeUpdate(KnowledgeAddUpdateRequest)
  1307. .then(() => {
  1308. afterSubmit('orderProcessSuccess', true);
  1309. })
  1310. .catch(() => {
  1311. afterSubmit('orderProcessFailed');
  1312. });
  1313. break;
  1314. case '更新知识':
  1315. const KnowledgeUpdateRequest = {
  1316. data: { ...state.orderDetail },
  1317. workflow: { ...submitObj, files: handleFiles.value },
  1318. };
  1319. KnowledgeUpdate(KnowledgeUpdateRequest)
  1320. .then(() => {
  1321. afterSubmit('orderProcessSuccess', true);
  1322. })
  1323. .catch(() => {
  1324. afterSubmit('orderProcessFailed');
  1325. });
  1326. break;
  1327. case '删除知识':
  1328. const KnowledgeRemoveRequest = {
  1329. data: { ...state.orderDetail },
  1330. workflow: { ...submitObj, files: handleFiles.value },
  1331. };
  1332. KnowledgeDel(KnowledgeRemoveRequest)
  1333. .then(() => {
  1334. afterSubmit('orderProcessSuccess', true, '删除知识申请成功');
  1335. })
  1336. .catch(() => {
  1337. afterSubmit('orderProcessFailed');
  1338. });
  1339. break;
  1340. case '知识下架':
  1341. const KnowledgeOffShelfRequest = {
  1342. data: { ...state.orderDetail },
  1343. workflow: { ...submitObj, files: handleFiles.value },
  1344. };
  1345. KnowledgeOffShelf(KnowledgeOffShelfRequest)
  1346. .then(() => {
  1347. afterSubmit('orderProcessSuccess', true, '下架知识申请成功');
  1348. })
  1349. .catch(() => {
  1350. afterSubmit('orderProcessFailed');
  1351. });
  1352. break;
  1353. default: // 默认工单办理
  1354. const requestDefault = {
  1355. data: { orderId: state.orderDetail.id, ...submitObj },
  1356. workflow: { ...submitObj, files: handleFiles.value },
  1357. };
  1358. orderHandle(requestDefault)
  1359. .then(() => {
  1360. afterSubmit('orderProcessSuccess', true);
  1361. })
  1362. .catch(() => {
  1363. afterSubmit('orderProcessFailed');
  1364. });
  1365. break;
  1366. }
  1367. });
  1368. };
  1369. // 临时保存
  1370. const handleTempSave = () => {
  1371. ruleFormRef.value
  1372. .validateField('opinion')
  1373. .then(() => {
  1374. state.loading = true;
  1375. orderProcessTempSave({ orderId: state.orderDetail.id, opinion: state.ruleForm.opinion })
  1376. .then(() => {
  1377. state.loading = false;
  1378. ElMessage.success('临时保存成功');
  1379. })
  1380. .catch((err) => {
  1381. state.loading = false;
  1382. console.log(err);
  1383. });
  1384. })
  1385. .catch(() => {});
  1386. };
  1387. // 暴露变量
  1388. defineExpose({
  1389. openDialog,
  1390. closeDialog,
  1391. });
  1392. </script>
  1393. <style lang="scss">
  1394. .modal_class {
  1395. pointer-events: none;
  1396. height: 100%;
  1397. }
  1398. .dialog_class {
  1399. pointer-events: auto;
  1400. flex-direction: column;
  1401. margin: 0 !important;
  1402. position: absolute;
  1403. top: 20%;
  1404. left: 25%;
  1405. }
  1406. </style>