index.vue 22 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727
  1. <template>
  2. <div class="supervision-result-manage">
  3. <!-- 页面标题 -->
  4. <!-- <h2>成本监审督办结果管理</h2> -->
  5. <!-- 查询区域 -->
  6. <div class="search-container">
  7. <el-form :inline="true" :model="searchForm" class="demo-form-inline">
  8. <el-form-item label="年度:">
  9. <el-date-picker
  10. v-model="searchForm.year"
  11. type="year"
  12. placeholder="请选择年度"
  13. format="yyyy"
  14. value-format="yyyy"
  15. clearable
  16. ></el-date-picker>
  17. </el-form-item>
  18. <el-form-item label="状态:">
  19. <el-select v-model="searchForm.status" placeholder="请选择状态">
  20. <el-option label="在办" value="在办"></el-option>
  21. <el-option label="办结" value="办结"></el-option>
  22. </el-select>
  23. </el-form-item>
  24. <el-form-item label="督办人:">
  25. <el-input
  26. v-model="searchForm.supervisor"
  27. placeholder="请输入督办人"
  28. clearable
  29. ></el-input>
  30. </el-form-item>
  31. <el-form-item label="督办项目名称:">
  32. <el-input
  33. v-model="searchForm.project"
  34. placeholder="请输入督办项目名称"
  35. clearable
  36. ></el-input>
  37. </el-form-item>
  38. <el-form-item>
  39. <el-button icon="el-icon-search" type="primary" @click="handleSearch">
  40. 搜索
  41. </el-button>
  42. <el-button
  43. plain
  44. type="primary"
  45. icon="el-icon-refresh"
  46. @click="handleReset"
  47. >
  48. 重置
  49. </el-button>
  50. </el-form-item>
  51. </el-form>
  52. </div>
  53. <!-- 数据表格 -->
  54. <el-table
  55. v-loading="loading"
  56. :data="tableData"
  57. border
  58. style="width: 100%"
  59. :row-class-name="getRowClassName"
  60. >
  61. <el-table-column
  62. prop="serialNumber"
  63. label="序号"
  64. width="60"
  65. header-align="center"
  66. align="center"
  67. ></el-table-column>
  68. <el-table-column
  69. prop="planYear"
  70. label="立项年度"
  71. width="100"
  72. header-align="center"
  73. align="center"
  74. ></el-table-column>
  75. <el-table-column
  76. prop="projectName"
  77. label="成本监审项目名称"
  78. header-align="center"
  79. align="left"
  80. show-overflow-tooltip
  81. >
  82. <template slot-scope="scope">
  83. <span :style="scope.row.style">{{ scope.row.projectName }}</span>
  84. </template>
  85. </el-table-column>
  86. <el-table-column
  87. prop="auditedUnit"
  88. label="被监审单位"
  89. header-align="center"
  90. align="left"
  91. show-overflow-tooltip
  92. >
  93. <template slot-scope="scope">
  94. <span :style="scope.row.style">{{ scope.row.auditedUnit }}</span>
  95. </template>
  96. </el-table-column>
  97. <el-table-column
  98. prop="auditPeriod"
  99. label="监审期间"
  100. width="150"
  101. header-align="center"
  102. align="center"
  103. >
  104. <template slot-scope="scope">
  105. <span :style="scope.row.style">{{ scope.row.auditPeriod }}</span>
  106. </template>
  107. </el-table-column>
  108. <el-table-column
  109. prop="expectedDeadline"
  110. label="预计截止时间"
  111. width="150"
  112. header-align="center"
  113. align="center"
  114. ></el-table-column>
  115. <el-table-column
  116. prop="taskStatus"
  117. label="任务状态"
  118. width="100"
  119. header-align="center"
  120. align="center"
  121. >
  122. <template slot-scope="scope">
  123. <span>{{ scope.row.taskStatus }}</span>
  124. </template>
  125. </el-table-column>
  126. <el-table-column
  127. prop="supervisor"
  128. label="督办人"
  129. width="100"
  130. header-align="center"
  131. align="center"
  132. ></el-table-column>
  133. <el-table-column
  134. prop="supervisionStatus"
  135. label="督办状态"
  136. width="100"
  137. header-align="center"
  138. align="center"
  139. ></el-table-column>
  140. <el-table-column
  141. prop="supervisionMessage"
  142. label="督办消息"
  143. width="150"
  144. header-align="center"
  145. align="left"
  146. >
  147. <template slot-scope="scope">
  148. <span
  149. v-if="scope.row.supervisionMessage.includes('新报告')"
  150. style="color: red"
  151. >
  152. {{ scope.row.supervisionMessage }}
  153. </span>
  154. <span v-else>{{ scope.row.supervisionMessage }}</span>
  155. </template>
  156. </el-table-column>
  157. <el-table-column
  158. label="操作"
  159. width="260"
  160. header-align="center"
  161. align="center"
  162. >
  163. <template slot-scope="scope">
  164. <template v-if="scope.row.isSubTask">
  165. <el-button size="mini" type="text" @click="handleView(scope.row)">
  166. 查看
  167. </el-button>
  168. </template>
  169. <template v-else>
  170. <el-button
  171. size="mini"
  172. type="text"
  173. @click="handleTaskDetail(scope.row)"
  174. >
  175. 任务详情
  176. </el-button>
  177. <el-button
  178. size="mini"
  179. type="text"
  180. @click="handleSupervisionDetail(scope.row)"
  181. >
  182. 督办情况
  183. </el-button>
  184. <el-button
  185. v-if="scope.row.supervisionStatus === '0'"
  186. size="mini"
  187. type="text"
  188. @click="handleEndSupervision(scope.row)"
  189. >
  190. 结束督办
  191. </el-button>
  192. </template>
  193. </template>
  194. </el-table-column>
  195. </el-table>
  196. <!-- 督办情况弹窗 -->
  197. <el-dialog
  198. title="督办情况"
  199. :visible.sync="isSupervisionDetail"
  200. width="80%"
  201. @close="handleBack"
  202. >
  203. <div class="detail-content">
  204. <el-table :data="supervisionRecords" border style="width: 100%">
  205. <el-table-column
  206. prop="serialNumber"
  207. label="序号"
  208. width="60"
  209. header-align="center"
  210. align="center"
  211. ></el-table-column>
  212. <el-table-column
  213. prop="supervisor"
  214. label="督办人"
  215. width="100"
  216. align="center"
  217. ></el-table-column>
  218. <el-table-column
  219. prop="reportTime"
  220. label="报告时间"
  221. width="150"
  222. align="center"
  223. ></el-table-column>
  224. <el-table-column
  225. prop="reportContent"
  226. label="报告内容"
  227. align="center"
  228. show-overflow-tooltip
  229. ></el-table-column>
  230. <el-table-column
  231. prop="relatedFiles"
  232. label="相关附件"
  233. width="120"
  234. align="center"
  235. >
  236. <template slot-scope="scope">
  237. <span
  238. v-if="scope.row.relatedFiles"
  239. style="color: #409eff; cursor: pointer"
  240. @click="handleDownload(scope.row.relatedFiles)"
  241. >
  242. {{ scope.row.relatedFiles }}
  243. </span>
  244. </template>
  245. </el-table-column>
  246. </el-table>
  247. <div class="button-group">
  248. <el-button @click="handleBack">返回</el-button>
  249. </div>
  250. </div>
  251. </el-dialog>
  252. <!-- 任务详情弹窗(子项“查看”) -->
  253. <task-detail
  254. ref="taskDetail"
  255. :visible.sync="taskDetailVisible"
  256. :is-view="isView"
  257. />
  258. <!-- 成本监审任务制定弹窗(母项“任务详情”) -->
  259. <task-customized-release-dialog
  260. :visible.sync="taskReleaseDialogVisible"
  261. :project="project"
  262. :is-view="true"
  263. @backToList="taskReleaseDialogVisible = false"
  264. @close="taskReleaseDialogVisible = false"
  265. />
  266. </div>
  267. </template>
  268. <script>
  269. import {
  270. deleteSuperviseTask,
  271. getSuperviseReportList,
  272. getSuperviseReportDetail,
  273. } from '@/api/audit/supervise'
  274. import TaskDetail from '@/components/task/taskDetail.vue'
  275. import TaskCustomizedReleaseDialog from '@/components/task/TaskCustomizedReleaseDialog.vue'
  276. import { getCostProjectDetail } from '@/api/taskCustomizedRelease.js'
  277. export default {
  278. data() {
  279. return {
  280. loading: false,
  281. searchForm: {
  282. year: '2025',
  283. status: '在办',
  284. supervisor: '',
  285. project: '',
  286. },
  287. projectList: [
  288. // 示例数据
  289. {
  290. id: 1,
  291. serialNumber: 1,
  292. planYear: '2025',
  293. projectName: '省内油气管道运输成本监审',
  294. auditedUnit: '山西省油气管道公司、太原市油气公司',
  295. auditPeriod: '2023年;2024年',
  296. expectedDeadline: '2025年7月31日',
  297. taskStatus: '资料初审',
  298. supervisor: '张**',
  299. supervisionStatus: '在办',
  300. supervisionMessage: '新报告',
  301. isSubTask: false,
  302. subTasks: [
  303. {
  304. id: 101,
  305. serialNumber: '',
  306. planYear: '',
  307. projectName: '子任务',
  308. auditedUnit: '山西省油气管道公司',
  309. auditPeriod: '2023年;2024年',
  310. expectedDeadline: '',
  311. taskStatus: '通过',
  312. supervisor: '',
  313. supervisionStatus: '',
  314. supervisionMessage: '',
  315. isSubTask: true,
  316. },
  317. {
  318. id: 102,
  319. serialNumber: '',
  320. planYear: '',
  321. projectName: '子任务',
  322. auditedUnit: '太原市油气公司',
  323. auditPeriod: '2023年;2024年',
  324. expectedDeadline: '',
  325. taskStatus: '待补充材料',
  326. supervisor: '',
  327. supervisionStatus: '',
  328. supervisionMessage: '',
  329. isSubTask: true,
  330. },
  331. ],
  332. },
  333. {
  334. id: 2,
  335. serialNumber: 2,
  336. planYear: '2025',
  337. projectName: '太原市电网输配电成本监审',
  338. auditedUnit: '太原市热电厂、****热电厂',
  339. auditPeriod: '2023年',
  340. expectedDeadline: '2025年7月31日',
  341. taskStatus: '集体审议',
  342. supervisor: '李**',
  343. supervisionStatus: '办结',
  344. supervisionMessage: '',
  345. isSubTask: false,
  346. subTasks: [
  347. {
  348. id: 201,
  349. serialNumber: '',
  350. planYear: '',
  351. projectName: '子任务',
  352. auditedUnit: '太原市热电厂',
  353. auditPeriod: '2023年',
  354. expectedDeadline: '',
  355. taskStatus: '通过',
  356. supervisor: '',
  357. supervisionStatus: '',
  358. supervisionMessage: '',
  359. isSubTask: true,
  360. },
  361. {
  362. id: 202,
  363. serialNumber: '',
  364. planYear: '',
  365. projectName: '子任务',
  366. auditedUnit: '****热电厂',
  367. auditPeriod: '2023年',
  368. expectedDeadline: '',
  369. taskStatus: '通过',
  370. supervisor: '',
  371. supervisionStatus: '',
  372. supervisionMessage: '',
  373. isSubTask: true,
  374. },
  375. ],
  376. },
  377. ],
  378. tableData: [],
  379. currentTask: null,
  380. supervisionRecords: [],
  381. isSupervisionDetail: false,
  382. // 统一与其它页面一致的弹窗控制
  383. isView: true,
  384. taskDetailVisible: false,
  385. taskReleaseDialogVisible: false,
  386. project: {},
  387. }
  388. },
  389. mounted() {
  390. this.generateTableData()
  391. },
  392. methods: {
  393. handleReset() {
  394. this.searchForm = {
  395. year: '',
  396. status: '',
  397. supervisor: '',
  398. project: '',
  399. }
  400. this.generateTableData()
  401. },
  402. generateTableData() {
  403. this.loading = true
  404. const statusLabel = (this.searchForm.status || '').toString().trim()
  405. const status =
  406. statusLabel === '在办'
  407. ? '0'
  408. : statusLabel === '办结'
  409. ? '1'
  410. : statusLabel
  411. const params = {
  412. year: (this.searchForm.year || '').toString().trim(),
  413. status,
  414. supervisor: (this.searchForm.supervisor || '').toString().trim(),
  415. project: (this.searchForm.project || '').toString().trim(),
  416. pageNum: 1,
  417. pageSize: 9999,
  418. }
  419. getSuperviseReportList(params)
  420. .then((res) => {
  421. // 兼容不同返回结构
  422. const raw =
  423. (res &&
  424. res.value &&
  425. (res.value.records || res.value.list || res.value)) ||
  426. (res &&
  427. res.data &&
  428. (res.data.records || res.data.list || res.data)) ||
  429. []
  430. const arr = Array.isArray(raw) ? raw : []
  431. const mapStatus = (s) => {
  432. const v = s != null ? String(s) : ''
  433. if (v === '100') return '待提交'
  434. if (v === '200') return '审核中'
  435. if (v === '300') return '中止'
  436. if (v === '400') return '办结'
  437. return v
  438. }
  439. const mapSupervise = (s) => {
  440. const v = s != null ? String(s) : ''
  441. if (v === '0') return '在办'
  442. if (v === '1') return '办结'
  443. return v || ''
  444. }
  445. const flat = []
  446. arr.forEach((item, idx) => {
  447. const parent = {
  448. ...item,
  449. serialNumber: idx + 1,
  450. isSubTask: false,
  451. // 映射列字段,避免后端字段名不一致
  452. planYear: item.planYear || item.year || '',
  453. auditedUnit:
  454. item.auditedUnit ||
  455. item.auditedUnitName ||
  456. item.auditedUnitNames ||
  457. '',
  458. expectedDeadline:
  459. item.expectedDeadline ||
  460. item.processDeadline ||
  461. item.deadline ||
  462. '',
  463. taskStatus:
  464. item.taskStatus || item.statusName || mapStatus(item.status),
  465. supervisionStatus:
  466. item.supervisionStatus || mapSupervise(item.superviseStatus),
  467. supervisionMessage:
  468. item.supervisionMessage || item.message || item.remark || '',
  469. }
  470. flat.push(parent)
  471. const children =
  472. (item && (item.children || item.subTasks || item.childTasks)) ||
  473. []
  474. if (Array.isArray(children) && children.length > 0) {
  475. children.forEach((c) => {
  476. flat.push({
  477. ...c,
  478. isSubTask: true,
  479. serialNumber: '',
  480. style: { paddingLeft: '40px' },
  481. planYear: c.planYear || c.year || '',
  482. auditedUnit:
  483. c.auditedUnit ||
  484. c.auditedUnitName ||
  485. c.auditedUnitNames ||
  486. '',
  487. expectedDeadline:
  488. c.expectedDeadline ||
  489. c.processDeadline ||
  490. c.deadline ||
  491. '',
  492. taskStatus:
  493. c.taskStatus || c.statusName || mapStatus(c.status),
  494. supervisionStatus:
  495. c.supervisionStatus || mapSupervise(c.superviseStatus),
  496. supervisionMessage:
  497. c.supervisionMessage || c.message || c.remark || '',
  498. })
  499. })
  500. }
  501. })
  502. this.tableData = flat
  503. })
  504. .catch(() => {
  505. this.$message.error('获取数据失败')
  506. this.tableData = []
  507. })
  508. .finally(() => {
  509. this.loading = false
  510. })
  511. },
  512. getRowClassName({ row }) {
  513. if (row.isSubTask) {
  514. return 'sub-task-row'
  515. }
  516. return ''
  517. },
  518. handleSearch() {
  519. this.generateTableData()
  520. },
  521. handleView(row) {
  522. // 子项“查看”:打开任务详情弹窗
  523. this.isView = true
  524. this.taskDetailVisible = true
  525. this.$nextTick(() => {
  526. this.$refs.taskDetail && this.$refs.taskDetail.open(row, 'chengben')
  527. })
  528. },
  529. handleTaskDetail(row) {
  530. // 母项“任务详情”:打开成本监审任务制定弹窗(只读)
  531. this.openTaskReleaseDialog(row)
  532. },
  533. openTaskReleaseDialog(row) {
  534. if (!row) return
  535. const projectId =
  536. row.projectId || row.projectID || row.id || row.taskId || ''
  537. if (!projectId) {
  538. this.$message &&
  539. this.$message.warning &&
  540. this.$message.warning('缺少项目ID,无法查看详情')
  541. return
  542. }
  543. this.isView = true
  544. getCostProjectDetail({ id: projectId })
  545. .then((res) => {
  546. this.project = (res && res.value) || {}
  547. this.taskReleaseDialogVisible = true
  548. })
  549. .catch(() => {
  550. this.project = row || {}
  551. this.taskReleaseDialogVisible = true
  552. })
  553. },
  554. handleSupervisionDetail(row) {
  555. this.isSupervisionDetail = true
  556. this.currentTask = row
  557. const id = row && (row.superviseId || row.id || row.ID)
  558. if (!id) {
  559. this.supervisionRecords = []
  560. return
  561. }
  562. const loading = this.$loading({
  563. lock: true,
  564. text: '加载中...',
  565. spinner: 'el-icon-loading',
  566. background: 'rgba(0, 0, 0, 0.3)',
  567. })
  568. getSuperviseReportDetail({ id })
  569. .then((res) => {
  570. const raw = (res && (res.value || res.data)) || []
  571. const arr = Array.isArray(raw)
  572. ? raw
  573. : Array.isArray(raw?.records)
  574. ? raw.records
  575. : []
  576. this.supervisionRecords = arr.map((it, idx) => ({
  577. serialNumber: idx + 1,
  578. supervisor: it.supervisor || it.supervisorName || '',
  579. reportTime: it.reportTime || it.createTime || it.reportDate || '',
  580. reportContent: it.reportContent || it.content || '',
  581. relatedFiles:
  582. it.attachmentUrls || it.attachmentIds || it.relatedFiles || '',
  583. }))
  584. })
  585. .catch(() => {
  586. this.$message.error('获取督办情况失败')
  587. this.supervisionRecords = []
  588. })
  589. .finally(() => {
  590. loading && loading.close && loading.close()
  591. })
  592. },
  593. handleDownload(paths) {
  594. if (!paths) return
  595. const base =
  596. window.context && window.context.form ? window.context.form : ''
  597. const list = String(paths)
  598. .split(',')
  599. .map((s) => s.trim())
  600. .filter(Boolean)
  601. list.forEach((p) => {
  602. const url = p.startsWith('http') ? p : base + p
  603. window.open(url)
  604. })
  605. },
  606. handleEndSupervision(row) {
  607. console.log('结束督办:', row)
  608. this.$confirm('确定要结束督办吗?', '结束督办', {
  609. confirmButtonText: '确定',
  610. cancelButtonText: '取消',
  611. type: 'warning',
  612. })
  613. .then(() => {
  614. const loading = this.$loading({
  615. lock: true,
  616. text: '处理中...',
  617. spinner: 'el-icon-loading',
  618. background: 'rgba(0, 0, 0, 0.3)',
  619. })
  620. const id = row && (row.superviseId || row.ID)
  621. if (!id) {
  622. loading && loading.close && loading.close()
  623. this.$message.error('缺少ID,无法结束督办')
  624. return
  625. }
  626. deleteSuperviseTask({ id })
  627. .then((res) => {
  628. if (res && Number(res.code) === 200) {
  629. this.$message.success('已结束督办')
  630. this.generateTableData()
  631. } else {
  632. this.$message.error((res && res.message) || '操作失败')
  633. }
  634. })
  635. .catch(() => {
  636. this.$message.error('操作失败')
  637. })
  638. .finally(() => {
  639. loading && loading.close && loading.close()
  640. })
  641. })
  642. .catch(() => {
  643. this.$message.info('已取消')
  644. })
  645. },
  646. handleBack() {
  647. this.isSupervisionDetail = false
  648. this.currentTask = null
  649. },
  650. },
  651. }
  652. </script>
  653. <style scoped>
  654. .supervision-result-manage {
  655. padding: 20px;
  656. }
  657. h2 {
  658. margin-bottom: 20px;
  659. font-size: 18px;
  660. color: #303133;
  661. }
  662. h3 {
  663. margin-bottom: 15px;
  664. font-size: 16px;
  665. color: #303133;
  666. }
  667. .demo-form-inline {
  668. display: flex;
  669. align-items: center;
  670. flex-wrap: wrap;
  671. }
  672. .description {
  673. margin-top: 15px;
  674. margin-bottom: 20px;
  675. padding: 10px;
  676. background-color: #fff7e6;
  677. border: 1px solid #ffe7ba;
  678. border-radius: 4px;
  679. }
  680. .supervision-detail {
  681. margin-top: 20px;
  682. padding: 15px;
  683. background-color: #fafafa;
  684. border: 1px solid #ebeef5;
  685. border-radius: 4px;
  686. }
  687. .detail-content {
  688. background-color: #fff;
  689. padding: 15px;
  690. border-radius: 4px;
  691. }
  692. .button-group {
  693. margin-top: 15px;
  694. text-align: center;
  695. }
  696. /* 子任务样式 */
  697. .el-table .sub-task-row {
  698. background-color: #fafafa !important;
  699. }
  700. /* 新报告标记样式 */
  701. .new-report-tag {
  702. display: inline-block;
  703. padding: 0 5px;
  704. margin-left: 5px;
  705. font-size: 12px;
  706. color: #fff;
  707. background-color: #ff4949;
  708. border-radius: 4px;
  709. }
  710. </style>