index.vue 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504
  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 type="primary" class="ml-8px" preIcon="mdi:page-next-outline" @click="handleOpenModal({}, 'add')"> 新增问题 </a-button>
  9. <a-button type="default" class="ml-8px" preIcon="mdi:download" @click="handleExportExcel"> 导出 </a-button>
  10. </template>
  11. <!-- <template #submitBefore>
  12. <a-button type="primary" preIcon="mdi:page-next-outline" @click="handleOpenModal({}, 'add')"> 新增问题 </a-button>
  13. </template> -->
  14. <template #queJson="{ record }">
  15. <div style="display: flex; align-items: center; gap: 8px; width: 100%; justify-content: space-between;">
  16. <span style=" white-space: pre-line; word-wrap: break-word; line-height: 1.6; display: block; text-align: left; padding: 2px 4px;">
  17. {{ record?.queJson ? formatQueJson(record.queJson) : '' }}
  18. </span>
  19. <button @click="record && handleOpenModal(record, 'view')" class="action-btn" title="查看问题详情">
  20. <SvgIcon name="view" />
  21. </button>
  22. </div>
  23. </template>
  24. <template #action="{ record }">
  25. <button @click="handleOpenModal(record, 'edit')" class="action-btn" title="编辑该问题">
  26. <SvgIcon name="edit" />
  27. </button>
  28. <!-- 删除按钮 -->
  29. <Popconfirm
  30. title="删除确认"
  31. description="是否确认删除?"
  32. okText="确认"
  33. cancelText="取消"
  34. @confirm="handleDeleteRecord(record)"
  35. @cancel="handleCancel"
  36. placement="top"
  37. >
  38. <button class="action-btn" title="删除该问题">
  39. <SvgIcon name="delete" />
  40. </button>
  41. </Popconfirm>
  42. <Popconfirm
  43. title="标记已解决确认"
  44. :description="getResolveDesc(record)"
  45. okText="确认"
  46. cancelText="取消"
  47. @confirm="handleOKRecord(record)"
  48. @cancel="handleCancel"
  49. placement="top"
  50. >
  51. <button class="action-btn" title="标记该问题为已解决">
  52. <SvgIcon name="resolved" />
  53. </button>
  54. </Popconfirm>
  55. <button @click="handleGoToPage(record)" class="action-btn" title="监测详情">
  56. <SvgIcon name="details" />
  57. </button>
  58. </template>
  59. </BasicTable>
  60. </TabPane>
  61. <TabPane key="resolved" tab="已解决">
  62. <div class="add-button">
  63. <a-button type="default" preIcon="mdi:download" @click="handleExportExcel" style="margin-right: 8px"> 导出 </a-button>
  64. </div>
  65. <BasicTable style="padding: 0" @register="registerResolvedTable">
  66. <template #queJson="{ record }">
  67. <div style="display: flex; align-items: center; gap: 8px; width: 100%">
  68. <span style="flex: 1; text-align: center">
  69. {{ record?.queJson ? formatQueJson(record.queJson) : '' }}
  70. </span>
  71. <!-- 按钮:无需额外样式,自然靠最右侧 -->
  72. <button @click="record && handleOpenModal(record, 'view')" class="action-btn" title="查看问题详情">
  73. <SvgIcon name="view" />
  74. </button>
  75. </div>
  76. </template>
  77. <template #action="{ record }">
  78. <button @click="handleOpenModal(record, 'view')" class="action-btn" title="问题详情">
  79. <SvgIcon name="details" />
  80. </button>
  81. </template>
  82. </BasicTable>
  83. </TabPane>
  84. </Tabs>
  85. <!-- 处理弹框 -->
  86. <DataQualityModal @register="registerModal" @success="handleModalSuccess" />
  87. </template>
  88. <script setup lang="ts">
  89. import { ref, nextTick, computed, onMounted } from 'vue';
  90. import { useRouter } from 'vue-router';
  91. import { BasicTable, useTable } from '/@/components/Table';
  92. import { useModal } from '/@/components/Modal';
  93. import { Tabs, TabPane, Popconfirm, message } from 'ant-design-vue';
  94. import DataQualityModal from './components/DataQualityModal.vue';
  95. import { SvgIcon } from '/@/components/Icon';
  96. import { getColumns, getSearchFormSchema, type ProductionStatusMap } from './dataQuality.data';
  97. import { getDataQuaQueList, addDataQuaQue, deleteDataQuaQue, editDataQuaQue } from '../basicInfo.api';
  98. import { findNode } from '/@/utils/helper/treeHelper';
  99. import { useMineDepartmentStore } from '/@/store/modules/mine';
  100. import { getDictItemsByCode } from '/@/utils/dict';
  101. import dayjs from 'dayjs';
  102. import * as XLSX from 'xlsx';
  103. // 路由实例
  104. const router = useRouter();
  105. // 实例化矿井Store
  106. const mineStore = useMineDepartmentStore();
  107. // 响应式数据
  108. const activeKey = ref('unresolved'); // 激活的Tab键
  109. const pageMode = ref('add');
  110. // ========== 定义动态状态映射/下拉选项 ==========
  111. // 1. 动态生产状态映射(key: 状态value,value: 包含label/color的配置)
  112. const dynamicProductionStatusMap = ref<ProductionStatusMap>({});
  113. // 2. 动态下拉选项(供搜索表单使用)
  114. const dynamicProductionStatusOptions = ref<{ label: string; value: string | number }[]>([]);
  115. // 3. 颜色分配规则(可根据业务灵活调整)
  116. const getStatusColor = (statusText: string) => {
  117. if (statusText.includes('正常生产')) return 'green'; // 正常生产 → 绿色
  118. if (statusText.includes('拟建矿井'))
  119. return 'blue'; // 拟建矿井 → 蓝色
  120. else return 'red'; // 停产/停建/关闭/整改/责令 → 红色
  121. };
  122. // 4. 从接口获取生产状态列表并生成动态映射/下拉选项
  123. const fetchProductionStatus = async () => {
  124. try {
  125. // 调用接口获取状态列表
  126. const statusList = await getDictItemsByCode('mineProStatus');
  127. if (!Array.isArray(statusList)) return;
  128. // 生成动态映射和下拉选项
  129. const statusMap: ProductionStatusMap = {};
  130. const statusOptions: { label: string; value: string | number }[] = [];
  131. statusList.forEach((item) => {
  132. const value = item.value; // 接口返回的value(数字/字符串)
  133. const label = item.text || item.label; // 接口返回的文本
  134. const color = getStatusColor(label); // 按规则分配颜色
  135. // 填充映射表
  136. statusMap[value] = { label, value, color };
  137. // 填充下拉选项
  138. statusOptions.push({ label, value });
  139. });
  140. // 赋值到响应式变量
  141. dynamicProductionStatusMap.value = statusMap;
  142. dynamicProductionStatusOptions.value = statusOptions;
  143. // 刷新表格(确保表格使用最新的映射)
  144. await safeReloadActiveTable();
  145. } catch (error) {
  146. console.error('获取生产状态列表失败:', error);
  147. message.error('生产状态数据加载失败');
  148. }
  149. };
  150. // 生成动态列和搜索表单配置
  151. const columns = computed(() => getColumns(dynamicProductionStatusMap));
  152. const searchFormSchema = computed(() => getSearchFormSchema(dynamicProductionStatusOptions));
  153. // 未解决表格注册
  154. const [registerUnresolvedTable, { reload: reloadUnresolved }] = useTable({
  155. api: async (params: any) => {
  156. return await getDataQuaQueList({ ...params, isOk: false });
  157. },
  158. columns: columns, // 绑定动态列
  159. formConfig: {
  160. labelWidth: 120,
  161. schemas: searchFormSchema.value,
  162. showAdvancedButton: false,
  163. schemaGroupNames: ['常规查询'],
  164. actionColOptions: { span: 6 },
  165. },
  166. useSearchForm: true,
  167. pagination: true,
  168. showIndexColumn: false,
  169. indexColumnProps: {
  170. title: '序号',
  171. },
  172. // canResize: false,
  173. actionColumn: {
  174. width: 200,
  175. title: '操作',
  176. dataIndex: 'action',
  177. slots: { customRender: 'action' },
  178. },
  179. immediate: false, // 先不立即加载,等状态数据获取后再加载
  180. });
  181. // 已解决表格注册
  182. const [registerResolvedTable, { reload: reloadResolved }] = useTable({
  183. api: async (params: any) => {
  184. return await getDataQuaQueList({ ...params, isOk: true });
  185. },
  186. columns: columns,
  187. formConfig: {
  188. labelWidth: 120,
  189. schemas: searchFormSchema.value,
  190. showAdvancedButton: false,
  191. schemaGroupNames: ['常规查询'],
  192. },
  193. useSearchForm: true,
  194. pagination: true,
  195. striped: true,
  196. // bordered: true,
  197. showIndexColumn: false,
  198. indexColumnProps: {
  199. title: '序号',
  200. },
  201. actionColumn: {
  202. width: 60,
  203. title: '操作',
  204. dataIndex: 'action',
  205. slots: { customRender: 'action' },
  206. },
  207. immediate: false, // 先不立即加载
  208. });
  209. // 弹框注册
  210. const [registerModal, { openModal }] = useModal();
  211. // 解析queJson并拼接orderNum+queCon的辅助函数
  212. function formatQueJson(queJsonStr: string) {
  213. // 空值处理
  214. if (!queJsonStr) return '无质量问题';
  215. try {
  216. const queList = JSON.parse(queJsonStr);
  217. // 非数组格式处理
  218. if (!Array.isArray(queList)) return '问题格式异常';
  219. // 空数组处理
  220. if (queList.length === 0) return '无质量问题';
  221. return queList.map((item) => {
  222. const goafName = item.goafName;
  223. const queCon = item.queCon || '无描述';
  224. return `<${goafName}工作面采空区密闭监测>存在的问题:${queCon}`;
  225. }).join('\n'); // 多个问题分行显示
  226. } catch (error) {
  227. console.error('解析质量问题JSON失败:', error);
  228. return '问题数据解析失败';
  229. }
  230. }
  231. // 安全重载当前激活的表格
  232. async function safeReloadActiveTable() {
  233. await nextTick();
  234. if (activeKey.value === 'unresolved') {
  235. try {
  236. await reloadUnresolved();
  237. } catch (e) {
  238. console.warn('未解决表格重载失败:', e);
  239. }
  240. } else {
  241. try {
  242. await reloadResolved();
  243. } catch (e) {
  244. console.warn('已解决表格重载失败:', e);
  245. }
  246. }
  247. }
  248. // 按需重载双表格(先注释掉)
  249. // async function reloadBothTableSafely() {
  250. // await nextTick();
  251. // // 未解决表格:await + try/catch 捕获异步错误
  252. // try {
  253. // await reloadUnresolved();
  254. // } catch (e) {
  255. // console.warn('未解决表格暂未就绪,跳过重载:', e);
  256. // }
  257. // // 已解决表格:同理,一个报错不影响另一个
  258. // try {
  259. // await reloadResolved();
  260. // } catch (e) {
  261. // console.warn('已解决表格暂未就绪,跳过重载:', e);
  262. // }
  263. // }
  264. // tabs切换事件
  265. async function handleTabChange(key: string) {
  266. activeKey.value = key;
  267. await safeReloadActiveTable();
  268. }
  269. /**
  270. * 根据标签获取表格数据(已解决/未解决)
  271. * @param result 弹框数据
  272. */
  273. // function getQuaQueListByTab() {
  274. // return async (params: any) => {
  275. // const isOk = activeKey.value === 'resolved' ? true : false;
  276. // return await getDataQuaQueList({ ...params, isOk: isOk });
  277. // };
  278. // }
  279. /**
  280. * 打开弹框函数
  281. * @param result 弹框数据
  282. */
  283. function handleOpenModal(record: any, mode: 'view' | 'edit' | 'add' = 'view') {
  284. pageMode.value = mode;
  285. openModal(true, { record, mode });
  286. }
  287. /**
  288. * 弹框结果处理函数
  289. * @param result 弹框数据
  290. */
  291. async function handleModalSuccess(result: any) {
  292. try {
  293. if (pageMode.value === 'add') {
  294. await addDataQuaQue(result);
  295. } else if (pageMode.value === 'edit') {
  296. await editDataQuaQue(result);
  297. }
  298. await safeReloadActiveTable();
  299. } catch (error) {
  300. console.error('操作失败:', error);
  301. }
  302. }
  303. /**
  304. * 通用页面跳转方法
  305. * @param record 当前行数据
  306. * @param path 目标路径(树形结构所在页面的路由地址)
  307. */
  308. async function handleGoToPage(record: any) {
  309. try {
  310. const mineCode = record.mineCode;
  311. const targetNode = findNode(mineStore.getDepartTree, (item) => item.id === mineCode, { id: 'id', pid: 'parentId', children: 'childDepart' });
  312. let minePath = '';
  313. if (targetNode) {
  314. minePath = targetNode.parentId;
  315. } else {
  316. message.warning(`未找到矿码【${mineCode}】对应的矿井节点`);
  317. return;
  318. }
  319. // 跳转页面(可携带拼接后的矿名/路径等参数)
  320. router.push({ path: `/sealed/${minePath}`, query: { mineCode } });
  321. } catch (error) {
  322. console.error('矿节点定位失败:', error);
  323. message.error('矿节点定位失败,请稍后重试');
  324. }
  325. }
  326. /**
  327. * 气泡取消按钮通用回调
  328. */
  329. function handleCancel() {
  330. // 取消操作,无逻辑(仅关闭气泡)
  331. }
  332. /**
  333. * 生成已解决气泡提示文案:XX矿井XX问题是否解决了吗?
  334. */
  335. const getResolveDesc = computed(() => (record: any) => {
  336. const mineName = record.mineName || '该';
  337. return `是否确认${mineName}矿井问题已解决?`;
  338. });
  339. /**
  340. * 删除记录方法
  341. * @param record 当前行数据
  342. */
  343. async function handleDeleteRecord(record: any) {
  344. try {
  345. await deleteDataQuaQue({ id: record.id });
  346. await nextTick();
  347. await safeReloadActiveTable();
  348. } catch (error) {
  349. console.error('删除失败:', error);
  350. }
  351. }
  352. /**
  353. * 将记录改为已处理
  354. * @param record 当前行数据
  355. */
  356. async function handleOKRecord(record: any) {
  357. const copyRecord = {
  358. ...record,
  359. isOk: true,
  360. updateTime: dayjs().format('YYYY-MM-DD HH:mm:ss'),
  361. };
  362. try {
  363. await editDataQuaQue(copyRecord);
  364. await safeReloadActiveTable();
  365. } catch (error) {
  366. console.error('操作失败:', error);
  367. }
  368. }
  369. /**
  370. * 获取全部未解决数据(不分页)
  371. */
  372. async function getAllUnresolvedData() {
  373. try {
  374. const res = await getDataQuaQueList({ pageNum: 1, pageSize: 9999, isOk: false });
  375. return res.records || [];
  376. } catch (error) {
  377. console.error('获取未解决数据失败:', error);
  378. return [];
  379. }
  380. }
  381. /**
  382. * 获取全部已解决数据(不分页)
  383. */
  384. async function getAllResolvedData() {
  385. try {
  386. const res = await getDataQuaQueList({ pageNum: 1, pageSize: 9999, isOk: true });
  387. return res.records || [];
  388. } catch (error) {
  389. console.error('获取已解决数据失败:', error);
  390. return [];
  391. }
  392. }
  393. /**
  394. * 格式化导出数据(统一字段名 + 状态文本)
  395. */
  396. function formatExportData(dataList: any[]) {
  397. return dataList.map((item) => ({
  398. 煤矿名称: item.mineName || '',
  399. 煤矿简称: item.mineNameAbbr || '',
  400. 生产状态: dynamicProductionStatusMap.value[item.mineProStatus]?.label || '-',
  401. 在线状态: item.mineLinkStatus === 1 ? '在线' : item.mineLinkStatus === 0 ? '离线' : '/',
  402. 质量问题详情: formatQueJson(item.queJson),
  403. 当前状态: item.isOk ? '已解决' : '未解决',
  404. 处理时间: item.updateTime || '',
  405. }));
  406. }
  407. /**
  408. * 导出Excel核心方法
  409. */
  410. async function handleExportExcel() {
  411. let hideLoading: () => void = () => {};
  412. try {
  413. hideLoading = message.loading('正在导出数据,请稍候...');
  414. const unresolvedList = await getAllUnresolvedData();
  415. const resolvedList = await getAllResolvedData();
  416. const formattedUnresolved = formatExportData(unresolvedList);
  417. const formattedResolved = formatExportData(resolvedList);
  418. const workbook = XLSX.utils.book_new();
  419. const unresolvedSheet = XLSX.utils.json_to_sheet(formattedUnresolved);
  420. const resolvedSheet = XLSX.utils.json_to_sheet(formattedResolved);
  421. XLSX.utils.book_append_sheet(workbook, unresolvedSheet, '未解决');
  422. XLSX.utils.book_append_sheet(workbook, resolvedSheet, '已解决');
  423. const fileName = `数据质量问题_${dayjs().format('YYYYMMDDHHmmss')}.xlsx`;
  424. XLSX.writeFile(workbook, fileName);
  425. hideLoading();
  426. message.success('导出成功!');
  427. } catch (error) {
  428. if (hideLoading) hideLoading();
  429. console.error('Excel导出失败:', error);
  430. message.error('导出失败,请稍后重试');
  431. }
  432. }
  433. // ========== 初始化:先获取状态数据,再加载表格 ==========
  434. onMounted(async () => {
  435. await fetchProductionStatus(); // 先获取动态状态数据
  436. await safeReloadActiveTable();
  437. });
  438. </script>
  439. <style scoped lang="less">
  440. // .data-quality-page {
  441. // padding: 0 12px;
  442. // margin: 16px;
  443. // border: 1px solid @border-color-base;
  444. // .ant-form {
  445. // background-color: #fff;
  446. // }
  447. // }
  448. .form-part {
  449. padding: 12px 10px 6px 10px;
  450. margin-bottom: 8px;
  451. background-color: @white;
  452. border-radius: 2px;
  453. }
  454. .add-button {
  455. margin-bottom: 10px;
  456. }
  457. .action-btn {
  458. height: 30px;
  459. cursor: pointer;
  460. margin-right: 10px;
  461. &:last-child {
  462. margin-right: 0;
  463. }
  464. }
  465. </style>