BoilerTaskOrderDetailViewDialog.vue 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440
  1. <template>
  2. <el-dialog
  3. v-model="dialogVisible"
  4. title="修改后任务单详情"
  5. width="80%"
  6. draggable
  7. :close-on-click-modal="false"
  8. :before-close="handleClose"
  9. top="5vh"
  10. >
  11. <div v-if="!taskOrderData" class="loading-text">无法加载任务单详情。</div>
  12. <div v-else class="dialog-body">
  13. <!-- 受检信息 -->
  14. <ContentWrap title="受检信息">
  15. <el-form :model="formData" label-width="90px" disabled class="p-3">
  16. <div class="check-info-label-title">申请单位</div>
  17. <el-row class="form-group" :gutter="24">
  18. <el-col :span="8">
  19. <el-form-item label="申请单位名称" prop="unitName">
  20. <el-input v-model="formData.unitName" placeholder="请输入单位名称" />
  21. </el-form-item>
  22. </el-col>
  23. <el-col :span="8">
  24. <el-form-item label="统一社会信用代码" prop="socialCreditCode" label-width="120px">
  25. <el-input v-model="formData.socialCreditCode" placeholder="请输入统一社会信用代码" />
  26. </el-form-item>
  27. </el-col>
  28. <el-col :span="8">
  29. <el-form-item label="申请单位地址" prop="unitAddress" label-width="120px">
  30. <el-input v-model="formData.unitAddress" placeholder="请输入申请单位地址" />
  31. </el-form-item>
  32. </el-col>
  33. <el-col :span="8">
  34. <el-form-item label="邮政编码" prop="zipCode">
  35. <el-input v-model="formData.zipCode" placeholder="请输入邮政编码" />
  36. </el-form-item>
  37. </el-col>
  38. <el-col :span="8">
  39. <el-form-item label="约检联系人" prop="unitContact">
  40. <el-input v-model="formData.unitContact" placeholder="请输入约检联系人" />
  41. </el-form-item>
  42. </el-col>
  43. <el-col :span="8">
  44. <el-form-item label="约检联系人电话" prop="unitPhone" label-width="120px">
  45. <el-input v-model="formData.unitPhone" placeholder="请输入约检联系人电话" />
  46. </el-form-item>
  47. </el-col>
  48. <el-col :span="8">
  49. <el-form-item label="电子报告接收人" prop="recipient" label-width="120px">
  50. <el-input v-model="formData.recipient" placeholder="请输入电子报告接收人" />
  51. </el-form-item>
  52. </el-col>
  53. <el-col :span="8">
  54. <el-form-item label="接收人电话" prop="recipientPhone">
  55. <el-input v-model="formData.recipientPhone" placeholder="请输入接收人电话" />
  56. </el-form-item>
  57. </el-col>
  58. <el-col :span="8">
  59. <el-form-item label="接收人邮箱" prop="recipientEmail">
  60. <el-input v-model="formData.recipientEmail" placeholder="请输入接收人邮箱" />
  61. </el-form-item>
  62. </el-col>
  63. </el-row>
  64. <div class="check-info-label-title">使用单位</div>
  65. <el-row class="form-group" :gutter="24">
  66. <el-col :span="8">
  67. <el-form-item label="使用单位名称" prop="useUnitName">
  68. <el-input v-model="formData.useUnitName" placeholder="请输入使用单位名称" />
  69. </el-form-item>
  70. </el-col>
  71. <el-col :span="8">
  72. <el-form-item label="统一社会信用代码" prop="useUnitSocialCreditCode" label-width="120px">
  73. <el-input v-model="formData.useUnitSocialCreditCode" placeholder="请输入统一社会信用代码" />
  74. </el-form-item>
  75. </el-col>
  76. <el-col :span="8">
  77. <el-form-item label="使用单位地址" prop="useUnitAddress" label-width="120px">
  78. <el-input v-model="formData.useUnitAddress" placeholder="请输入使用单位地址" />
  79. </el-form-item>
  80. </el-col>
  81. <el-col :span="8">
  82. <el-form-item label="邮政编码" prop="useUnitZipcode">
  83. <el-input v-model="formData.useUnitZipcode" placeholder="请输入邮政编码" />
  84. </el-form-item>
  85. </el-col>
  86. <el-col :span="8">
  87. <el-form-item label="单位联系人" prop="useUnitContact" label-width="120px">
  88. <el-input v-model="formData.useUnitContact" placeholder="请输入单位联系人" />
  89. </el-form-item>
  90. </el-col>
  91. <el-col :span="8">
  92. <el-form-item label="联系人电话" prop="useUnitPhone" label-width="120px">
  93. <el-input v-model="formData.useUnitPhone" placeholder="请输入联系人电话" />
  94. </el-form-item>
  95. </el-col>
  96. </el-row>
  97. <div class="check-info-label-title">受理信息</div>
  98. <el-row class="form-group" :gutter="24">
  99. <el-col :span="8">
  100. <el-form-item label="检验部门" prop="deptId">
  101. <el-input v-model="formData.deptName" placeholder="检验部门" />
  102. </el-form-item>
  103. </el-col>
  104. <el-col :span="8">
  105. <el-form-item label="联系电话" prop="deptPhone" label-width="120px">
  106. <el-input v-model="formData.deptPhone" placeholder="请输入联系电话" />
  107. </el-form-item>
  108. </el-col>
  109. <el-col :span="8">
  110. <el-form-item label="约检时间" prop="checkDate" label-width="120px">
  111. <el-input v-model="formData.checkDate" placeholder="请选择约检时间" />
  112. </el-form-item>
  113. </el-col>
  114. <el-col :span="24">
  115. <el-form-item label="检验员" prop="teamList">
  116. <div v-if="formattedCheckers.length > 0" class="selected-checkers">
  117. <el-tag
  118. v-for="checker in formattedCheckers"
  119. :key="checker.id"
  120. effect="plain"
  121. >
  122. {{ checker.label }}
  123. </el-tag>
  124. </div>
  125. <span v-else>-</span>
  126. </el-form-item>
  127. </el-col>
  128. </el-row>
  129. <div class="check-info-label-title">检验收费</div>
  130. <el-row class="form-group" :gutter="24">
  131. <el-col :span="24">
  132. <el-form-item label="检验性质" prop="checkType" label-width="120px">
  133. <el-input :value="checkTypeLabel" disabled class="!w-120px" />
  134. </el-form-item>
  135. <el-form-item prop="feeNature" label-width="120px">
  136. <template #label>
  137. <div class="flex flex-col text-right">
  138. <span>收费性质</span>
  139. <span>(行政事业收费)</span>
  140. </div>
  141. </template>
  142. <el-radio-group v-model="formData.feeNature" disabled>
  143. <el-radio v-for="(item, key) in PressureFeeNatureMap" :key="key" :value="key" :label="item" />
  144. </el-radio-group>
  145. </el-form-item>
  146. </el-col>
  147. <el-col :span="24">
  148. <el-form-item label="收费形式" prop="feeType" label-width="120px">
  149. <el-radio-group v-model="formData.feeType" disabled>
  150. <el-radio v-for="(item, key) in PressureFeeTypeMap" :key="key" :value="key" :label="item" />
  151. </el-radio-group>
  152. </el-form-item>
  153. </el-col>
  154. <el-col :span="24">
  155. <div class="flex items-center gap-32 ml-40px text-14px">
  156. <span>应收法定金额: {{ formData.shouldAmount }}</span>
  157. <span>服务收费金额: {{ formData.serviceAmount }}</span>
  158. <span>免征费用: {{ formData.reduceFee }}</span>
  159. </div>
  160. </el-col>
  161. </el-row>
  162. </el-form>
  163. </ContentWrap>
  164. <!-- 设备清单 -->
  165. <div class="plan-section mt-4">
  166. <div class="section-title">
  167. <span>设备信息</span>
  168. </div>
  169. <ContentWrap>
  170. <el-table
  171. :data="taskOrderData.orderItems || []"
  172. border
  173. stripe
  174. empty-text="暂无设备信息"
  175. >
  176. <el-table-column label="区域" align="center" min-width="120">
  177. <template #default="{ row }">
  178. <div>{{ row.equipDistrictName || '-' }}</div>
  179. <div class="text-gray-400 text-sm">{{ row.equipStreetName || '' }}</div>
  180. </template>
  181. </el-table-column>
  182. <el-table-column label="设备注册代码" align="center" min-width="190">
  183. <template #default="{ row }">
  184. <el-tag type="primary" size="small">{{ row.equipCode || '-' }}</el-tag>
  185. </template>
  186. </el-table-column>
  187. <el-table-column label="是否租借" align="center" prop="isRent" min-width="80">
  188. <template #default="{ row }">
  189. {{ row.isRent ? '是' : '否' }}
  190. </template>
  191. </el-table-column>
  192. <el-table-column label="使用证编号" align="center" prop="useRegisterNo" min-width="120" />
  193. <el-table-column label="出厂编号" align="center" prop="factoryCode" min-width="120" />
  194. <el-table-column label="约检联系人" align="center" prop="contact" min-width="120" />
  195. <el-table-column label="约检联系人电话" align="center" prop="contactPhone" min-width="140" />
  196. <el-table-column label="蒸发量" align="center" prop="maxContinueEvapor" min-width="80" />
  197. <el-table-column label="型号" align="center" prop="boilerModel" min-width="120" />
  198. </el-table>
  199. </ContentWrap>
  200. </div>
  201. <!-- 备注 -->
  202. <ContentWrap title="备注" class="mt-4">
  203. <el-row>
  204. <el-col :span="12">
  205. <el-form-item class="!mb-0" label="上次退回原因" prop="reason">
  206. <el-input type="textarea" :rows="4" v-model="formData.reason" readonly disabled />
  207. </el-form-item>
  208. </el-col>
  209. </el-row>
  210. </ContentWrap>
  211. </div>
  212. <template #footer>
  213. <div class="flex justify-center">
  214. <el-button @click="handleClose">关闭</el-button>
  215. </div>
  216. </template>
  217. </el-dialog>
  218. </template>
  219. <script setup lang="ts">
  220. import { ref, computed, watch } from 'vue'
  221. import { formatArrayDate } from '@/utils/formatTime'
  222. import {
  223. PressureFeeNatureMap,
  224. PressureFeeTypeMap,
  225. PressureBoilerCheckTypeMap
  226. } from '@/utils/constants'
  227. defineOptions({ name: 'BoilerTaskOrderDetailViewDialog' })
  228. const props = defineProps({
  229. visible: {
  230. type: Boolean,
  231. default: false
  232. },
  233. taskOrder: {
  234. type: Object as () => Record<string, any>,
  235. default: () => ({})
  236. }
  237. })
  238. const emit = defineEmits<{
  239. 'update:visible': [value: boolean]
  240. }>()
  241. const dialogVisible = ref(props.visible)
  242. watch(() => props.visible, (val) => {
  243. dialogVisible.value = val
  244. })
  245. watch(dialogVisible, (val) => {
  246. emit('update:visible', val)
  247. })
  248. const taskOrderData = computed(() => props.taskOrder)
  249. const checkStatusMap: Record<number, string> = {
  250. 100: '内检',
  251. 200: '外检',
  252. 300: '耐压检'
  253. }
  254. const checkTypeLabel = computed(() => {
  255. const ct = formData.value.checkType
  256. if (!ct) return '-'
  257. return PressureBoilerCheckTypeMap[ct] || checkStatusMap[ct] || ct
  258. })
  259. const formData = ref({
  260. unitName: '',
  261. socialCreditCode: '',
  262. unitAddress: '',
  263. zipCode: '',
  264. unitContact: '',
  265. unitPhone: '',
  266. recipient: '',
  267. recipientPhone: '',
  268. recipientEmail: '',
  269. useUnitName: '',
  270. useUnitSocialCreditCode: '',
  271. useUnitAddress: '',
  272. useUnitZipcode: '',
  273. useUnitContact: '',
  274. useUnitPhone: '',
  275. deptName: '',
  276. deptPhone: '',
  277. checkDate: '',
  278. feeNature: '',
  279. feeType: '',
  280. shouldAmount: 0,
  281. serviceAmount: 0,
  282. reduceFee: 0,
  283. checkType: undefined as number | undefined,
  284. reason: ''
  285. })
  286. const formattedCheckers = computed(() => {
  287. const teamList = taskOrderData.value?.teamInfoList
  288. console.log("teamList",teamList)
  289. if (!teamList || !Array.isArray(teamList) || teamList.length === 0) return []
  290. const result: { id: string; label: string }[] = []
  291. teamList.forEach((team: any, teamIndex: number) => {
  292. if (team.isLeader) {
  293. result.push({
  294. id: `leader-${team.member.id || teamIndex}`,
  295. label: `组长: ${team.member.nickname || ''}`
  296. })
  297. }else{
  298. result.push({
  299. id: `member-${team.member.id || teamIndex}`,
  300. label: `${team.member.nickname || ''}`
  301. })
  302. }
  303. })
  304. return result
  305. })
  306. const initFormData = () => {
  307. const data = taskOrderData.value
  308. if (!data) return
  309. const dateVal = data.checkDate
  310. ? Array.isArray(data.checkDate)
  311. ? formatArrayDate(data.checkDate)
  312. : data.checkDate
  313. : ''
  314. formData.value = {
  315. unitName: data.unitName || '',
  316. socialCreditCode: data.socialCreditCode || '',
  317. unitAddress: data.unitAddress || '',
  318. zipCode: data.zipCode || '',
  319. unitContact: data.unitContact || '',
  320. unitPhone: data.unitPhone || '',
  321. recipient: data.recipient || '',
  322. recipientPhone: data.recipientPhone || '',
  323. recipientEmail: data.recipientEmail || '',
  324. useUnitName: typeof data.useUnitName === 'object' ? data.useUnitName?.name || '' : data.useUnitName || '',
  325. useUnitSocialCreditCode: data.useUnitSocialCreditCode || '',
  326. useUnitAddress: data.useUnitAddress || '',
  327. useUnitZipcode: data.useUnitZipcode || '',
  328. useUnitContact: data.useUnitContact || '',
  329. useUnitPhone: data.useUnitPhone || '',
  330. deptName: data.deptName || '',
  331. deptPhone: data.deptPhone || '',
  332. checkDate: dateVal,
  333. feeNature: String(data.feeNature || ''),
  334. feeType: String(data.feeType || ''),
  335. shouldAmount: data.shouldAmount ?? data.payAmount ?? 0,
  336. serviceAmount: data.serviceAmount ?? 0,
  337. reduceFee: data.reduceFee ?? 0,
  338. checkType: data.checkType,
  339. reason: data.reason || ''
  340. }
  341. }
  342. watch(() => props.taskOrder, () => {
  343. initFormData()
  344. }, { immediate: true, deep: true })
  345. const handleClose = () => {
  346. dialogVisible.value = false
  347. }
  348. </script>
  349. <style lang="scss" scoped>
  350. .loading-text {
  351. text-align: center;
  352. padding: 40px;
  353. color: #909399;
  354. }
  355. .dialog-body {
  356. max-height: 70vh;
  357. overflow-y: auto;
  358. }
  359. .check-info-label-title {
  360. width: 100%;
  361. line-height: 32px;
  362. display: flex;
  363. align-items: center;
  364. margin-bottom: 12px;
  365. &::before {
  366. content: '';
  367. display: inline-block;
  368. width: 4px;
  369. height: 16px;
  370. background: var(--el-color-primary);
  371. margin-right: 8px;
  372. }
  373. }
  374. .section-title {
  375. font-size: 16px;
  376. font-weight: 600;
  377. color: #303133;
  378. padding: 16px 0 8px;
  379. display: flex;
  380. align-items: center;
  381. &::before {
  382. content: '';
  383. display: inline-block;
  384. width: 4px;
  385. height: 16px;
  386. background: var(--el-color-primary);
  387. margin-right: 8px;
  388. }
  389. }
  390. :deep(.el-form) {
  391. .el-form-item {
  392. margin-bottom: 12px;
  393. align-items: center;
  394. &__label {
  395. height: auto;
  396. font-size: 13px;
  397. line-height: 20px;
  398. color: #606266;
  399. padding-right: 8px;
  400. }
  401. }
  402. }
  403. .selected-checkers {
  404. display: inline-flex;
  405. flex-wrap: wrap;
  406. gap: 8px;
  407. min-height: 32px;
  408. padding: 4px 8px;
  409. }
  410. </style>