index.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577
  1. <!-- eslint-disable vue/multi-word-component-names -->
  2. <template>
  3. <!-- Tab标签页 -->
  4. <Tabs v-model:activeKey="activeKey" class="common-page-tabs" type="line" @change="handleTabChange">
  5. <TabPane key="unresolved" tab="未解决">
  6. <BasicTable style="padding: 0" @register="registerUnresolvedTable">
  7. <template #resetBefore>
  8. <a-button
  9. v-if="mineStore.getRoot?.isLeaf"
  10. type="primary"
  11. class="ml-8px"
  12. preIcon="mdi:page-next-outline"
  13. @click="handleOpenModal({}, 'add')"
  14. >
  15. 新增问题
  16. </a-button>
  17. <a-button type="default" class="ml-8px" preIcon="mdi:download" @click="handleOpenExportModal('unresolved')"> 导出 </a-button>
  18. </template>
  19. <template #queJson="{ record }">
  20. <div style="display: flex; align-items: center; gap: 8px; width: 100%; justify-content: space-between">
  21. <span style="white-space: pre-line; word-wrap: break-word; line-height: 1.6; display: block; text-align: left; padding: 2px 4px">
  22. {{ record?.queJson ? formatQueJson(record.queJson) : '' }}
  23. </span>
  24. <button @click="record && handleOpenModal(record, 'view')" class="action-btn" title="查看问题详情">
  25. <SvgIcon name="view" />
  26. </button>
  27. </div>
  28. </template>
  29. <template #action="{ record }">
  30. <button @click="handleOpenModal(record, 'edit')" class="action-btn" title="编辑该问题">
  31. <SvgIcon name="edit" />
  32. </button>
  33. <!-- 删除按钮 -->
  34. <Popconfirm
  35. title="删除确认"
  36. description="是否确认删除?"
  37. okText="确认"
  38. cancelText="取消"
  39. @confirm="handleDeleteRecord(record)"
  40. @cancel="handleCancel"
  41. placement="top"
  42. >
  43. <button class="action-btn" title="删除该问题">
  44. <SvgIcon name="delete" />
  45. </button>
  46. </Popconfirm>
  47. <Popconfirm
  48. title="标记已解决确认"
  49. :description="getResolveDesc(record)"
  50. okText="确认"
  51. cancelText="取消"
  52. @confirm="handleOKRecord(record)"
  53. @cancel="handleCancel"
  54. placement="top"
  55. >
  56. <button class="action-btn" title="标记该问题为已解决">
  57. <SvgIcon name="resolved" />
  58. </button>
  59. </Popconfirm>
  60. <button @click="handleGoToPage(record)" class="action-btn" title="监测详情">
  61. <SvgIcon name="details" />
  62. </button>
  63. </template>
  64. <template #form-mine-cascader>
  65. <MineCascader
  66. v-model:value="innerValue"
  67. :sync-to-store="false"
  68. :init-from-store="false"
  69. :change-on-select="false"
  70. :allow-clear="false"
  71. @change="changeCascader"
  72. />
  73. </template>
  74. <template #form-goaf-select>
  75. <a-select v-model:value="goafId" :options="goafOptions" placeholder="请选择" />
  76. </template>
  77. </BasicTable>
  78. </TabPane>
  79. <TabPane key="resolved" tab="已解决">
  80. <BasicTable style="padding: 0" @register="registerResolvedTable">
  81. <template #resetBefore>
  82. <a-button type="default" class="ml-8px" preIcon="mdi:download" @click="handleOpenExportModal('resolved')"> 导出 </a-button>
  83. </template>
  84. <template #queJson="{ record }">
  85. <div style="display: flex; align-items: center; gap: 8px; width: 100%">
  86. <span style="flex: 1; text-align: center">
  87. {{ record?.queJson ? formatQueJson(record.queJson) : '' }}
  88. </span>
  89. <button @click="record && handleOpenModal(record, 'view')" class="action-btn" title="查看问题详情">
  90. <SvgIcon name="view" />
  91. </button>
  92. </div>
  93. </template>
  94. <template #action="{ record }">
  95. <button @click="handleOpenModal(record, 'view')" class="action-btn" title="问题详情">
  96. <SvgIcon name="details" />
  97. </button>
  98. </template>
  99. <template #form-mine-cascader>
  100. <MineCascader
  101. v-model:value="innerValue"
  102. style="width: 330px"
  103. :sync-to-store="false"
  104. :init-from-store="false"
  105. :change-on-select="false"
  106. :allow-clear="false"
  107. @change="changeCascader"
  108. />
  109. </template>
  110. <template #form-goaf-select>
  111. <a-select v-model:value="goafId" :options="goafOptions" placeholder="请选择" />
  112. </template>
  113. </BasicTable>
  114. </TabPane>
  115. </Tabs>
  116. <!-- 处理弹框 -->
  117. <ProblemReportModal @register="registerModal" @success="handleModalSuccess" />
  118. <!-- 导出时间选择Modal-->
  119. <Modal
  120. v-model:open="exportModalVisible"
  121. title="选择导出时间范围"
  122. :width="600"
  123. @ok="handleExportConfirm"
  124. @cancel="handleExportCancel"
  125. okText="确认导出"
  126. cancelText="取消"
  127. centered
  128. :confirm-loading="exportConfirmLoading"
  129. destroyOnClose
  130. :bodyStyle="{ padding: '20px' }"
  131. >
  132. <a-form
  133. ref="exportFormRef"
  134. :model="exportFormData"
  135. :rules="exportFormRules"
  136. layout="horizontal"
  137. :label-col="{ span: 6 }"
  138. :wrapper-col="{ span: 18 }"
  139. >
  140. <a-form-item v-for="schema in exportFormSchema" :key="schema.field" :name="schema.field" :label="schema.label" :rules="schema.rules">
  141. <component
  142. :is="getExportFormComponent(schema.component)"
  143. v-model:value="exportFormData[schema.field]"
  144. v-bind="schema.componentProps"
  145. :placeholder="`请选择${schema.label}`"
  146. style="width: 100%"
  147. />
  148. </a-form-item>
  149. </a-form>
  150. </Modal>
  151. </template>
  152. <script setup lang="ts">
  153. import { ref, nextTick, computed, onMounted, reactive } from 'vue';
  154. import { useRouter } from 'vue-router';
  155. import { BasicTable } from '/@/components/Table';
  156. import { useModal } from '/@/components/Modal';
  157. import { Tabs, TabPane, Popconfirm, message, Modal, DatePicker } from 'ant-design-vue';
  158. import type { FormInstance } from 'ant-design-vue/es/form';
  159. import ProblemReportModal from './components/ProblemReportModal.vue';
  160. import { SvgIcon } from '/@/components/Icon';
  161. import { columns, searchFormSchema, exportFormSchema } from './problemReport.data';
  162. import { getGoafQuestionReportList, addGoafQuestionReport, deleteGoafQuestionReport, editGoafQuestionReport } from '../basicInfo.api';
  163. import { findNode } from '/@/utils/helper/treeHelper';
  164. import { useMineDepartmentStore } from '/@/store/modules/mine';
  165. // import { getDictItemsByCode } from '/@/utils/dict';
  166. import dayjs, { Dayjs } from 'dayjs';
  167. import { useListPage } from '/@/hooks/system/useListPage';
  168. import { useInitForm } from '/@/views/analysis/warningAnalysis/connectAnalysis/hooks/form';
  169. import MineCascader from '@/components/Form/src/jeecg/components/MineCascader/MineCascader.vue';
  170. // 路由实例
  171. const router = useRouter();
  172. // 实例化矿井Store
  173. const mineStore = useMineDepartmentStore();
  174. const { goafOptions, goafId, innerValue, initGoafOptions } = useInitForm();
  175. // 响应式数据
  176. const activeKey = ref('unresolved'); // 激活的Tab键
  177. const pageMode = ref('add');
  178. // 导出Modal相关状态
  179. const exportModalVisible = ref(false);
  180. const exportFormRef = ref<FormInstance>();
  181. const exportConfirmLoading = ref(false);
  182. const exportType = ref<'resolved' | 'unresolved'>('unresolved'); // 记录当前导出类型
  183. const exportConfig = ref({
  184. url: '/province/goafQuestionReport/exportDataQuaQueList',
  185. name: '密闭墙问题反馈',
  186. params: {
  187. isOk: 0,
  188. startTime: '',
  189. endTime: '',
  190. goafId: '',
  191. deptId: '',
  192. },
  193. });
  194. // 导出表单数据和规则(参考ProblemReportModal写法)
  195. const exportFormData = reactive({
  196. startTime: null as Dayjs | null,
  197. endTime: null as Dayjs | null,
  198. });
  199. // 提取导出表单规则(从exportFormSchema)
  200. const exportFormRules = reactive({
  201. startTime: exportFormSchema.find((item) => item.field === 'startTime')?.rules || [],
  202. endTime: exportFormSchema.find((item) => item.field === 'endTime')?.rules || [],
  203. });
  204. // 导出表单组件映射
  205. const exportFormComponentMap = {
  206. DatePicker,
  207. };
  208. // 获取导出表单组件
  209. const getExportFormComponent = (componentName: string) => {
  210. return exportFormComponentMap[componentName as keyof typeof exportFormComponentMap];
  211. };
  212. // 未解决表格注册
  213. const { tableContext: tableContextA, onExportXls: onExportXlsA } = useListPage({
  214. tableProps: {
  215. api: async (params: any) => {
  216. return await getGoafQuestionReportList({
  217. ...params,
  218. isOk: false,
  219. goafId: goafId.value,
  220. deptId: innerValue.value,
  221. });
  222. },
  223. columns: columns, // 绑定动态列
  224. formConfig: {
  225. labelWidth: 120,
  226. schemas: searchFormSchema,
  227. showAdvancedButton: false,
  228. schemaGroupNames: ['常规查询'],
  229. actionColOptions: { span: 6 },
  230. },
  231. useSearchForm: true,
  232. pagination: true,
  233. showIndexColumn: false,
  234. indexColumnProps: {
  235. title: '序号',
  236. },
  237. actionColumn: {
  238. width: 200,
  239. title: '操作',
  240. dataIndex: 'action',
  241. slots: { customRender: 'action' },
  242. },
  243. immediate: false, // 先不立即加载,等状态数据获取后再加载
  244. },
  245. exportConfig: exportConfig.value,
  246. });
  247. const [registerUnresolvedTable, { reload: reloadUnresolved }] = tableContextA;
  248. // 已解决表格注册
  249. const { tableContext: tableContextB, onExportXls: onExportXlsB } = useListPage({
  250. tableProps: {
  251. api: async (params: any) => {
  252. return await getGoafQuestionReportList({
  253. ...params,
  254. isOk: true,
  255. goafId: goafId.value,
  256. deptId: innerValue.value,
  257. });
  258. },
  259. columns: columns,
  260. formConfig: {
  261. labelWidth: 120,
  262. schemas: searchFormSchema,
  263. showAdvancedButton: false,
  264. schemaGroupNames: ['常规查询'],
  265. },
  266. useSearchForm: true,
  267. pagination: true,
  268. striped: true,
  269. showIndexColumn: false,
  270. indexColumnProps: {
  271. title: '序号',
  272. },
  273. showActionColumn: false,
  274. // actionColumn: {
  275. // width: 60,
  276. // title: '操作',
  277. // dataIndex: 'action',
  278. // slots: { customRender: 'action' },
  279. // },
  280. immediate: false, // 先不立即加载
  281. },
  282. exportConfig: exportConfig.value,
  283. });
  284. const [registerResolvedTable, { reload: reloadResolved }] = tableContextB;
  285. // 弹框注册
  286. const [registerModal, { openModal }] = useModal();
  287. /**
  288. * 打开导出时间选择Modal
  289. * @param type 导出类型:resolved 或 unresolved
  290. */
  291. function handleOpenExportModal(type: 'resolved' | 'unresolved') {
  292. exportType.value = type;
  293. exportModalVisible.value = true;
  294. // 打开Modal时重置表单
  295. nextTick(() => {
  296. exportFormRef.value?.resetFields();
  297. exportFormData.startTime = null;
  298. exportFormData.endTime = null;
  299. });
  300. }
  301. /**
  302. * 导出Modal确认处理
  303. */
  304. async function handleExportConfirm() {
  305. try {
  306. exportConfirmLoading.value = true;
  307. // 表单校验(参考ProblemReportModal的校验方式)
  308. if (!exportFormRef.value) return;
  309. const validateResult = await exportFormRef.value.validate();
  310. if (!validateResult) return;
  311. // 格式化时间
  312. const startTime = exportFormData.startTime?.format('YYYY-MM-DD HH:mm:ss') || '';
  313. const endTime = exportFormData.endTime?.format('YYYY-MM-DD HH:mm:ss') || '';
  314. // 校验时间逻辑(结束时间不能早于开始时间)
  315. if (exportFormData.startTime && exportFormData.endTime && dayjs(endTime).isBefore(dayjs(startTime))) {
  316. message.error('结束时间不能早于开始时间');
  317. return;
  318. }
  319. // 根据类型选择对应的exportConfig和导出方法
  320. exportConfig.value.params = {
  321. isOk: exportType.value === 'resolved' ? 1 : 0,
  322. startTime,
  323. endTime,
  324. goafId: goafId.value,
  325. deptId: innerValue.value,
  326. };
  327. if (exportType.value === 'unresolved') {
  328. await onExportXlsA();
  329. } else {
  330. await onExportXlsB();
  331. }
  332. // 关闭Modal
  333. exportModalVisible.value = false;
  334. // message.success('导出请求已发送,请注意查收文件');
  335. } catch (error: any) {
  336. console.error('导出失败:', error);
  337. message.error(error.message || '导出失败,请检查表单填写是否正确');
  338. } finally {
  339. exportConfirmLoading.value = false;
  340. }
  341. }
  342. /**
  343. * 导出Modal取消处理
  344. */
  345. function handleExportCancel() {
  346. exportModalVisible.value = false;
  347. // 取消时重置表单
  348. exportFormRef.value?.resetFields();
  349. exportFormData.startTime = null;
  350. exportFormData.endTime = null;
  351. }
  352. // 解析queJson并拼接orderNum+queCon的辅助函数
  353. function formatQueJson(queJsonStr: string) {
  354. // 空值处理
  355. if (!queJsonStr) return '无密闭墙问题';
  356. try {
  357. const queList = JSON.parse(queJsonStr);
  358. // 非数组格式处理
  359. if (!Array.isArray(queList)) return '问题格式异常';
  360. // 空数组处理
  361. if (queList.length === 0) return '无密闭墙问题';
  362. return queList
  363. .map((item) => {
  364. const goafName = item.goafName;
  365. const queCon = item.queCon || '无描述';
  366. return `<${goafName}工作面老空区永久密闭监测>存在的问题:${queCon}`;
  367. })
  368. .join('\n'); // 多个问题分行显示
  369. } catch (error) {
  370. console.error('解析密闭墙问题JSON失败:', error);
  371. return '问题数据解析失败';
  372. }
  373. }
  374. // 安全重载当前激活的表格
  375. async function safeReloadActiveTable() {
  376. await nextTick();
  377. if (activeKey.value === 'unresolved') {
  378. try {
  379. await reloadUnresolved();
  380. } catch (e) {
  381. console.warn('未解决表格重载失败:', e);
  382. }
  383. } else {
  384. try {
  385. await reloadResolved();
  386. } catch (e) {
  387. console.warn('已解决表格重载失败:', e);
  388. }
  389. }
  390. }
  391. // 按需重载双表格(先注释掉)
  392. // async function reloadBothTableSafely() {
  393. // await nextTick();
  394. // // 未解决表格:await + try/catch 捕获异步错误
  395. // try {
  396. // await reloadUnresolved();
  397. // } catch (e) {
  398. // console.warn('未解决表格暂未就绪,跳过重载:', e);
  399. // }
  400. // // 已解决表格:同理,一个报错不影响另一个
  401. // try {
  402. // await reloadResolved();
  403. // } catch (e) {
  404. // console.warn('已解决表格暂未就绪,跳过重载:', e);
  405. // }
  406. // }
  407. // tabs切换事件
  408. async function handleTabChange(key: string) {
  409. activeKey.value = key;
  410. await safeReloadActiveTable();
  411. }
  412. /**
  413. * 打开弹框函数
  414. * @param result 弹框数据
  415. */
  416. function handleOpenModal(record: any, mode: 'view' | 'edit' | 'add' = 'view') {
  417. pageMode.value = mode;
  418. openModal(true, {
  419. record: {
  420. deptId: innerValue.value,
  421. goafId: goafId.value,
  422. ...record,
  423. },
  424. mode,
  425. });
  426. }
  427. /**
  428. * 弹框结果处理函数
  429. * @param result 弹框数据
  430. */
  431. async function handleModalSuccess(result: any) {
  432. try {
  433. if (pageMode.value === 'add') {
  434. await addGoafQuestionReport(result);
  435. } else if (pageMode.value === 'edit') {
  436. await editGoafQuestionReport(result);
  437. }
  438. await safeReloadActiveTable();
  439. } catch (error) {
  440. console.error('操作失败:', error);
  441. }
  442. }
  443. /**
  444. * 通用页面跳转方法
  445. * @param record 当前行数据
  446. * @param path 目标路径(树形结构所在页面的路由地址)
  447. */
  448. async function handleGoToPage(record: any) {
  449. try {
  450. const mineCode = record.mineCode;
  451. const targetNode = findNode(mineStore.getDepartTree, (item) => item.fax === mineCode, { id: 'id', pid: 'parentId', children: 'childDepart' });
  452. let minePath = '';
  453. if (targetNode) {
  454. minePath = targetNode.parentId;
  455. } else {
  456. message.warning(`未找到矿码【${mineCode}】对应的矿井节点`);
  457. return;
  458. }
  459. // 跳转页面(可携带拼接后的矿名/路径等参数)
  460. router.push({ path: `/sealed/${minePath}`, query: { id: targetNode.id } });
  461. } catch (error) {
  462. console.error('矿节点定位失败:', error);
  463. message.error('矿节点定位失败,请稍后重试');
  464. }
  465. }
  466. /**
  467. * 气泡取消按钮通用回调
  468. */
  469. function handleCancel() {
  470. // 取消操作,无逻辑(仅关闭气泡)
  471. }
  472. /**
  473. * 生成已解决气泡提示文案:XX矿井XX问题是否解决了吗?
  474. */
  475. const getResolveDesc = computed(() => (record: any) => {
  476. const mineName = record.mineName || '该';
  477. return `是否确认${mineName}矿井问题已解决?`;
  478. });
  479. /**
  480. * 删除记录方法
  481. * @param record 当前行数据
  482. */
  483. async function handleDeleteRecord(record: any) {
  484. try {
  485. await deleteGoafQuestionReport({ id: record.id });
  486. await nextTick();
  487. await safeReloadActiveTable();
  488. } catch (error) {
  489. console.error('删除失败:', error);
  490. }
  491. }
  492. /**
  493. * 将记录改为已处理
  494. * @param record 当前行数据
  495. */
  496. async function handleOKRecord(record: any) {
  497. const copyRecord = {
  498. ...record,
  499. isOk: true,
  500. updateTime: dayjs().format('YYYY-MM-DD HH:mm:ss'),
  501. };
  502. try {
  503. await editGoafQuestionReport(copyRecord);
  504. await safeReloadActiveTable();
  505. message.success('标记为已解决成功');
  506. } catch (error) {
  507. console.error('操作失败:', error);
  508. message.error('操作失败,请稍后重试');
  509. }
  510. }
  511. function changeCascader(val) {
  512. innerValue.value = val;
  513. initGoafOptions(val);
  514. }
  515. // ========== 初始化:先获取状态数据,再加载表格 ==========
  516. onMounted(async () => {
  517. await safeReloadActiveTable();
  518. });
  519. </script>
  520. <style scoped lang="less">
  521. .form-part {
  522. padding: 12px 10px 6px 10px;
  523. margin-bottom: 8px;
  524. background-color: @white;
  525. border-radius: 2px;
  526. }
  527. .add-button {
  528. margin-bottom: 10px;
  529. }
  530. .action-btn {
  531. height: 30px;
  532. cursor: pointer;
  533. margin-right: 10px;
  534. &:last-child {
  535. margin-right: 0;
  536. }
  537. }
  538. </style>