| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577 |
- <!-- eslint-disable vue/multi-word-component-names -->
- <template>
- <!-- Tab标签页 -->
- <Tabs v-model:activeKey="activeKey" class="common-page-tabs" type="line" @change="handleTabChange">
- <TabPane key="unresolved" tab="未解决">
- <BasicTable style="padding: 0" @register="registerUnresolvedTable">
- <template #resetBefore>
- <a-button
- v-if="mineStore.getRoot?.isLeaf"
- type="primary"
- class="ml-8px"
- preIcon="mdi:page-next-outline"
- @click="handleOpenModal({}, 'add')"
- >
- 新增问题
- </a-button>
- <a-button type="default" class="ml-8px" preIcon="mdi:download" @click="handleOpenExportModal('unresolved')"> 导出 </a-button>
- </template>
- <template #queJson="{ record }">
- <div style="display: flex; align-items: center; gap: 8px; width: 100%; justify-content: space-between">
- <span style="white-space: pre-line; word-wrap: break-word; line-height: 1.6; display: block; text-align: left; padding: 2px 4px">
- {{ record?.queJson ? formatQueJson(record.queJson) : '' }}
- </span>
- <button @click="record && handleOpenModal(record, 'view')" class="action-btn" title="查看问题详情">
- <SvgIcon name="view" />
- </button>
- </div>
- </template>
- <template #action="{ record }">
- <button @click="handleOpenModal(record, 'edit')" class="action-btn" title="编辑该问题">
- <SvgIcon name="edit" />
- </button>
- <!-- 删除按钮 -->
- <Popconfirm
- title="删除确认"
- description="是否确认删除?"
- okText="确认"
- cancelText="取消"
- @confirm="handleDeleteRecord(record)"
- @cancel="handleCancel"
- placement="top"
- >
- <button class="action-btn" title="删除该问题">
- <SvgIcon name="delete" />
- </button>
- </Popconfirm>
- <Popconfirm
- title="标记已解决确认"
- :description="getResolveDesc(record)"
- okText="确认"
- cancelText="取消"
- @confirm="handleOKRecord(record)"
- @cancel="handleCancel"
- placement="top"
- >
- <button class="action-btn" title="标记该问题为已解决">
- <SvgIcon name="resolved" />
- </button>
- </Popconfirm>
- <button @click="handleGoToPage(record)" class="action-btn" title="监测详情">
- <SvgIcon name="details" />
- </button>
- </template>
- <template #form-mine-cascader>
- <MineCascader
- v-model:value="innerValue"
- :sync-to-store="false"
- :init-from-store="false"
- :change-on-select="false"
- :allow-clear="false"
- @change="changeCascader"
- />
- </template>
- <template #form-goaf-select>
- <a-select v-model:value="goafId" :options="goafOptions" placeholder="请选择" />
- </template>
- </BasicTable>
- </TabPane>
- <TabPane key="resolved" tab="已解决">
- <BasicTable style="padding: 0" @register="registerResolvedTable">
- <template #resetBefore>
- <a-button type="default" class="ml-8px" preIcon="mdi:download" @click="handleOpenExportModal('resolved')"> 导出 </a-button>
- </template>
- <template #queJson="{ record }">
- <div style="display: flex; align-items: center; gap: 8px; width: 100%">
- <span style="flex: 1; text-align: center">
- {{ record?.queJson ? formatQueJson(record.queJson) : '' }}
- </span>
- <button @click="record && handleOpenModal(record, 'view')" class="action-btn" title="查看问题详情">
- <SvgIcon name="view" />
- </button>
- </div>
- </template>
- <template #action="{ record }">
- <button @click="handleOpenModal(record, 'view')" class="action-btn" title="问题详情">
- <SvgIcon name="details" />
- </button>
- </template>
- <template #form-mine-cascader>
- <MineCascader
- v-model:value="innerValue"
- style="width: 330px"
- :sync-to-store="false"
- :init-from-store="false"
- :change-on-select="false"
- :allow-clear="false"
- @change="changeCascader"
- />
- </template>
- <template #form-goaf-select>
- <a-select v-model:value="goafId" :options="goafOptions" placeholder="请选择" />
- </template>
- </BasicTable>
- </TabPane>
- </Tabs>
- <!-- 处理弹框 -->
- <ProblemReportModal @register="registerModal" @success="handleModalSuccess" />
- <!-- 导出时间选择Modal-->
- <Modal
- v-model:open="exportModalVisible"
- title="选择导出时间范围"
- :width="600"
- @ok="handleExportConfirm"
- @cancel="handleExportCancel"
- okText="确认导出"
- cancelText="取消"
- centered
- :confirm-loading="exportConfirmLoading"
- destroyOnClose
- :bodyStyle="{ padding: '20px' }"
- >
- <a-form
- ref="exportFormRef"
- :model="exportFormData"
- :rules="exportFormRules"
- layout="horizontal"
- :label-col="{ span: 6 }"
- :wrapper-col="{ span: 18 }"
- >
- <a-form-item v-for="schema in exportFormSchema" :key="schema.field" :name="schema.field" :label="schema.label" :rules="schema.rules">
- <component
- :is="getExportFormComponent(schema.component)"
- v-model:value="exportFormData[schema.field]"
- v-bind="schema.componentProps"
- :placeholder="`请选择${schema.label}`"
- style="width: 100%"
- />
- </a-form-item>
- </a-form>
- </Modal>
- </template>
- <script setup lang="ts">
- import { ref, nextTick, computed, onMounted, reactive } from 'vue';
- import { useRouter } from 'vue-router';
- import { BasicTable } from '/@/components/Table';
- import { useModal } from '/@/components/Modal';
- import { Tabs, TabPane, Popconfirm, message, Modal, DatePicker } from 'ant-design-vue';
- import type { FormInstance } from 'ant-design-vue/es/form';
- import ProblemReportModal from './components/ProblemReportModal.vue';
- import { SvgIcon } from '/@/components/Icon';
- import { columns, searchFormSchema, exportFormSchema } from './problemReport.data';
- import { getGoafQuestionReportList, addGoafQuestionReport, deleteGoafQuestionReport, editGoafQuestionReport } from '../basicInfo.api';
- import { findNode } from '/@/utils/helper/treeHelper';
- import { useMineDepartmentStore } from '/@/store/modules/mine';
- // import { getDictItemsByCode } from '/@/utils/dict';
- import dayjs, { Dayjs } from 'dayjs';
- import { useListPage } from '/@/hooks/system/useListPage';
- import { useInitForm } from '/@/views/analysis/warningAnalysis/connectAnalysis/hooks/form';
- import MineCascader from '@/components/Form/src/jeecg/components/MineCascader/MineCascader.vue';
- // 路由实例
- const router = useRouter();
- // 实例化矿井Store
- const mineStore = useMineDepartmentStore();
- const { goafOptions, goafId, innerValue, initGoafOptions } = useInitForm();
- // 响应式数据
- const activeKey = ref('unresolved'); // 激活的Tab键
- const pageMode = ref('add');
- // 导出Modal相关状态
- const exportModalVisible = ref(false);
- const exportFormRef = ref<FormInstance>();
- const exportConfirmLoading = ref(false);
- const exportType = ref<'resolved' | 'unresolved'>('unresolved'); // 记录当前导出类型
- const exportConfig = ref({
- url: '/province/goafQuestionReport/exportDataQuaQueList',
- name: '密闭墙问题反馈',
- params: {
- isOk: 0,
- startTime: '',
- endTime: '',
- goafId: '',
- deptId: '',
- },
- });
- // 导出表单数据和规则(参考ProblemReportModal写法)
- const exportFormData = reactive({
- startTime: null as Dayjs | null,
- endTime: null as Dayjs | null,
- });
- // 提取导出表单规则(从exportFormSchema)
- const exportFormRules = reactive({
- startTime: exportFormSchema.find((item) => item.field === 'startTime')?.rules || [],
- endTime: exportFormSchema.find((item) => item.field === 'endTime')?.rules || [],
- });
- // 导出表单组件映射
- const exportFormComponentMap = {
- DatePicker,
- };
- // 获取导出表单组件
- const getExportFormComponent = (componentName: string) => {
- return exportFormComponentMap[componentName as keyof typeof exportFormComponentMap];
- };
- // 未解决表格注册
- const { tableContext: tableContextA, onExportXls: onExportXlsA } = useListPage({
- tableProps: {
- api: async (params: any) => {
- return await getGoafQuestionReportList({
- ...params,
- isOk: false,
- goafId: goafId.value,
- deptId: innerValue.value,
- });
- },
- columns: columns, // 绑定动态列
- formConfig: {
- labelWidth: 120,
- schemas: searchFormSchema,
- showAdvancedButton: false,
- schemaGroupNames: ['常规查询'],
- actionColOptions: { span: 6 },
- },
- useSearchForm: true,
- pagination: true,
- showIndexColumn: false,
- indexColumnProps: {
- title: '序号',
- },
- actionColumn: {
- width: 200,
- title: '操作',
- dataIndex: 'action',
- slots: { customRender: 'action' },
- },
- immediate: false, // 先不立即加载,等状态数据获取后再加载
- },
- exportConfig: exportConfig.value,
- });
- const [registerUnresolvedTable, { reload: reloadUnresolved }] = tableContextA;
- // 已解决表格注册
- const { tableContext: tableContextB, onExportXls: onExportXlsB } = useListPage({
- tableProps: {
- api: async (params: any) => {
- return await getGoafQuestionReportList({
- ...params,
- isOk: true,
- goafId: goafId.value,
- deptId: innerValue.value,
- });
- },
- columns: columns,
- formConfig: {
- labelWidth: 120,
- schemas: searchFormSchema,
- showAdvancedButton: false,
- schemaGroupNames: ['常规查询'],
- },
- useSearchForm: true,
- pagination: true,
- striped: true,
- showIndexColumn: false,
- indexColumnProps: {
- title: '序号',
- },
- showActionColumn: false,
- // actionColumn: {
- // width: 60,
- // title: '操作',
- // dataIndex: 'action',
- // slots: { customRender: 'action' },
- // },
- immediate: false, // 先不立即加载
- },
- exportConfig: exportConfig.value,
- });
- const [registerResolvedTable, { reload: reloadResolved }] = tableContextB;
- // 弹框注册
- const [registerModal, { openModal }] = useModal();
- /**
- * 打开导出时间选择Modal
- * @param type 导出类型:resolved 或 unresolved
- */
- function handleOpenExportModal(type: 'resolved' | 'unresolved') {
- exportType.value = type;
- exportModalVisible.value = true;
- // 打开Modal时重置表单
- nextTick(() => {
- exportFormRef.value?.resetFields();
- exportFormData.startTime = null;
- exportFormData.endTime = null;
- });
- }
- /**
- * 导出Modal确认处理
- */
- async function handleExportConfirm() {
- try {
- exportConfirmLoading.value = true;
- // 表单校验(参考ProblemReportModal的校验方式)
- if (!exportFormRef.value) return;
- const validateResult = await exportFormRef.value.validate();
- if (!validateResult) return;
- // 格式化时间
- const startTime = exportFormData.startTime?.format('YYYY-MM-DD HH:mm:ss') || '';
- const endTime = exportFormData.endTime?.format('YYYY-MM-DD HH:mm:ss') || '';
- // 校验时间逻辑(结束时间不能早于开始时间)
- if (exportFormData.startTime && exportFormData.endTime && dayjs(endTime).isBefore(dayjs(startTime))) {
- message.error('结束时间不能早于开始时间');
- return;
- }
- // 根据类型选择对应的exportConfig和导出方法
- exportConfig.value.params = {
- isOk: exportType.value === 'resolved' ? 1 : 0,
- startTime,
- endTime,
- goafId: goafId.value,
- deptId: innerValue.value,
- };
- if (exportType.value === 'unresolved') {
- await onExportXlsA();
- } else {
- await onExportXlsB();
- }
- // 关闭Modal
- exportModalVisible.value = false;
- // message.success('导出请求已发送,请注意查收文件');
- } catch (error: any) {
- console.error('导出失败:', error);
- message.error(error.message || '导出失败,请检查表单填写是否正确');
- } finally {
- exportConfirmLoading.value = false;
- }
- }
- /**
- * 导出Modal取消处理
- */
- function handleExportCancel() {
- exportModalVisible.value = false;
- // 取消时重置表单
- exportFormRef.value?.resetFields();
- exportFormData.startTime = null;
- exportFormData.endTime = null;
- }
- // 解析queJson并拼接orderNum+queCon的辅助函数
- function formatQueJson(queJsonStr: string) {
- // 空值处理
- if (!queJsonStr) return '无密闭墙问题';
- try {
- const queList = JSON.parse(queJsonStr);
- // 非数组格式处理
- if (!Array.isArray(queList)) return '问题格式异常';
- // 空数组处理
- if (queList.length === 0) return '无密闭墙问题';
- return queList
- .map((item) => {
- const goafName = item.goafName;
- const queCon = item.queCon || '无描述';
- return `<${goafName}工作面老空区永久密闭监测>存在的问题:${queCon}`;
- })
- .join('\n'); // 多个问题分行显示
- } catch (error) {
- console.error('解析密闭墙问题JSON失败:', error);
- return '问题数据解析失败';
- }
- }
- // 安全重载当前激活的表格
- async function safeReloadActiveTable() {
- await nextTick();
- if (activeKey.value === 'unresolved') {
- try {
- await reloadUnresolved();
- } catch (e) {
- console.warn('未解决表格重载失败:', e);
- }
- } else {
- try {
- await reloadResolved();
- } catch (e) {
- console.warn('已解决表格重载失败:', e);
- }
- }
- }
- // 按需重载双表格(先注释掉)
- // async function reloadBothTableSafely() {
- // await nextTick();
- // // 未解决表格:await + try/catch 捕获异步错误
- // try {
- // await reloadUnresolved();
- // } catch (e) {
- // console.warn('未解决表格暂未就绪,跳过重载:', e);
- // }
- // // 已解决表格:同理,一个报错不影响另一个
- // try {
- // await reloadResolved();
- // } catch (e) {
- // console.warn('已解决表格暂未就绪,跳过重载:', e);
- // }
- // }
- // tabs切换事件
- async function handleTabChange(key: string) {
- activeKey.value = key;
- await safeReloadActiveTable();
- }
- /**
- * 打开弹框函数
- * @param result 弹框数据
- */
- function handleOpenModal(record: any, mode: 'view' | 'edit' | 'add' = 'view') {
- pageMode.value = mode;
- openModal(true, {
- record: {
- deptId: innerValue.value,
- goafId: goafId.value,
- ...record,
- },
- mode,
- });
- }
- /**
- * 弹框结果处理函数
- * @param result 弹框数据
- */
- async function handleModalSuccess(result: any) {
- try {
- if (pageMode.value === 'add') {
- await addGoafQuestionReport(result);
- } else if (pageMode.value === 'edit') {
- await editGoafQuestionReport(result);
- }
- await safeReloadActiveTable();
- } catch (error) {
- console.error('操作失败:', error);
- }
- }
- /**
- * 通用页面跳转方法
- * @param record 当前行数据
- * @param path 目标路径(树形结构所在页面的路由地址)
- */
- async function handleGoToPage(record: any) {
- try {
- const mineCode = record.mineCode;
- const targetNode = findNode(mineStore.getDepartTree, (item) => item.fax === mineCode, { id: 'id', pid: 'parentId', children: 'childDepart' });
- let minePath = '';
- if (targetNode) {
- minePath = targetNode.parentId;
- } else {
- message.warning(`未找到矿码【${mineCode}】对应的矿井节点`);
- return;
- }
- // 跳转页面(可携带拼接后的矿名/路径等参数)
- router.push({ path: `/sealed/${minePath}`, query: { id: targetNode.id } });
- } catch (error) {
- console.error('矿节点定位失败:', error);
- message.error('矿节点定位失败,请稍后重试');
- }
- }
- /**
- * 气泡取消按钮通用回调
- */
- function handleCancel() {
- // 取消操作,无逻辑(仅关闭气泡)
- }
- /**
- * 生成已解决气泡提示文案:XX矿井XX问题是否解决了吗?
- */
- const getResolveDesc = computed(() => (record: any) => {
- const mineName = record.mineName || '该';
- return `是否确认${mineName}矿井问题已解决?`;
- });
- /**
- * 删除记录方法
- * @param record 当前行数据
- */
- async function handleDeleteRecord(record: any) {
- try {
- await deleteGoafQuestionReport({ id: record.id });
- await nextTick();
- await safeReloadActiveTable();
- } catch (error) {
- console.error('删除失败:', error);
- }
- }
- /**
- * 将记录改为已处理
- * @param record 当前行数据
- */
- async function handleOKRecord(record: any) {
- const copyRecord = {
- ...record,
- isOk: true,
- updateTime: dayjs().format('YYYY-MM-DD HH:mm:ss'),
- };
- try {
- await editGoafQuestionReport(copyRecord);
- await safeReloadActiveTable();
- message.success('标记为已解决成功');
- } catch (error) {
- console.error('操作失败:', error);
- message.error('操作失败,请稍后重试');
- }
- }
- function changeCascader(val) {
- innerValue.value = val;
- initGoafOptions(val);
- }
- // ========== 初始化:先获取状态数据,再加载表格 ==========
- onMounted(async () => {
- await safeReloadActiveTable();
- });
- </script>
- <style scoped lang="less">
- .form-part {
- padding: 12px 10px 6px 10px;
- margin-bottom: 8px;
- background-color: @white;
- border-radius: 2px;
- }
- .add-button {
- margin-bottom: 10px;
- }
- .action-btn {
- height: 30px;
- cursor: pointer;
- margin-right: 10px;
- &:last-child {
- margin-right: 0;
- }
- }
- </style>
|