Преглед на файлове

pms 保养查询功能增强

zhangcl преди 3 седмици
родител
ревизия
d22bbb9395

+ 5 - 0
src/api/pms/iotmainworkorder/index.ts

@@ -41,6 +41,11 @@ export const IotMainWorkOrderApi = {
     return await request.get({ url: `/pms/iot-main-work-order/deviceMainDistances`, params })
   },
 
+  // 以设备为维度统计所有保养明细中最近的保养数据 保养项分组
+  maintenanceSearch: async (params: any) => {
+    return await request.get({ url: `/pms/iot-main-work-order/maintenanceSearch`, params })
+  },
+
   getDeviceIotMainWorkOrderPage: async (params: any) => {
     return await request.get({ url: `/pms/iot-main-work-order/deviceOrderPage`, params })
   },

+ 4 - 1
src/locales/en.ts

@@ -947,7 +947,10 @@ export default {
     mileage:'Mileage',
     runTime:'RunTime',
     date:'Date',
-    currentDate:'Current Date'
+    currentDate:'Current Date',
+    maintenanceMileage:'Maintenance Mileage',
+    maintenanceRuntime:'Maintenance Runtime',
+    maintenanceDate:'Maintenance Date',
   },
   mainPlan:{
     basicMaintenanceRecords:'BasicMaintenanceRecords',

+ 4 - 1
src/locales/ru.ts

@@ -931,7 +931,10 @@ export default {
     mileage:'运行里程',
     runTime:'运行时间',
     date:'自然日期',
-    currentDate:'当前日期'
+    currentDate:'当前日期',
+    maintenanceMileage:'保养里程',
+    maintenanceRuntime:'保养时长',
+    maintenanceDate:'保养日期',
   },
   mainPlan:{
     basicMaintenanceRecords:'基础保养记录',

+ 4 - 1
src/locales/zh-CN.ts

@@ -942,7 +942,10 @@ export default {
     mileage:'运行里程',
     runTime:'运行时间',
     date:'自然日期',
-    currentDate:'当前日期'
+    currentDate:'当前日期',
+    maintenanceMileage:'保养里程',
+    maintenanceRuntime:'保养时长',
+    maintenanceDate:'保养日期',
   },
   mainPlan:{
     basicMaintenanceRecords:'基础保养记录',

+ 15 - 0
src/router/modules/remaining.ts

@@ -72,6 +72,7 @@ const remainingRouter: AppRouteRecordRaw[] = [
   },
   {
     path: '/dingding',
+    name: 'dingtalk',
     component: () => import('@/views/pms/dingding.vue'),
     meta:{
       hidden: true,
@@ -710,6 +711,20 @@ const remainingRouter: AppRouteRecordRaw[] = [
           activeMenu: '/mainworkorder/alarm'
         }
       },
+      {
+        path: 'maintenancesearch',
+        component: () => import('@/views/pms/iotmainworkorder/IotMaintenanceSearch.vue'),
+        name: 'IotMaintenanceSearch',
+        meta: {
+          keepAlive: true,
+          noCache: false,
+          hidden: true,
+          canTo: true,
+          icon: 'ep:menu',
+          title: t('mainPlan.maintenanceQuery'),
+          activeMenu: '/mainworkorder/search'
+        }
+      },
       {
         path: 'mainworkorder/bom/:id(\\d+)',
         component: () => import('@/views/pms/iotmainworkorder/IotMainWorkOrder.vue'),

+ 853 - 0
src/views/pms/iotmainworkorder/IotMaintenanceSearch.vue

@@ -0,0 +1,853 @@
+<template>
+  <el-row :gutter="20">
+    <!-- 左侧部门树 -->
+    <el-col :span="4" :xs="24">
+      <ContentWrap class="h-1/1" v-if="treeShow">
+        <DeptTree @node-click="handleDeptNodeClick" />
+      </ContentWrap>
+    </el-col>
+    <el-col :span="contentSpan" :xs="24">
+      <ContentWrap>
+        <!-- 搜索工作栏 -->
+        <el-form
+          class="-mb-15px"
+          :model="queryParams"
+          ref="queryFormRef"
+          :inline="true"
+          label-width="68px"
+        >
+          <el-form-item :label="t('iotDevice.code')" prop="deviceCode" style="margin-left: 25px">
+            <el-input
+              v-model="queryParams.deviceCode"
+              :placeholder="t('iotDevice.codeHolder')"
+              clearable
+              @keyup.enter="handleQuery"
+              class="!w-200px"
+            />
+          </el-form-item>
+          <el-form-item :label="t('iotDevice.name')" prop="deviceName">
+            <el-input
+              v-model="queryParams.deviceName"
+              :placeholder="t('iotDevice.nameHolder')"
+              clearable
+              @keyup.enter="handleQuery"
+              class="!w-200px"
+            />
+          </el-form-item>
+
+          <el-form-item>
+            <el-button @click="handleQuery"
+              ><Icon icon="ep:search" class="mr-5px" />
+              {{ t('file.search') }}</el-button
+            >
+            <el-button @click="resetQuery"
+              ><Icon icon="ep:refresh" class="mr-5px" />  {{ t('file.reset') }}</el-button
+            >
+            <el-button
+              type="success"
+              plain
+              @click="handleExport"
+              :loading="exportLoading"
+              v-hasPermi="['rq:iot-device:export']"
+            >
+              <Icon icon="ep:download" class="mr-5px" /> 导出
+            </el-button>
+          </el-form-item>
+        </el-form>
+      </ContentWrap>
+
+      <!-- 列表 -->
+      <ContentWrap>
+        <el-table v-loading="loading" :data="list" :stripe="true" style="width: 100%" @header-dragend="handleHeaderDragEnd"
+                  :show-overflow-tooltip="false" :header-cell-style="tableHeaderStyle" ref="tableRef">
+          <el-table-column :label="t('monitor.serial')" width="60" align="center" fixed="left">
+            <template #default="scope">
+              {{ scope.$index + 1 }}
+            </template>
+          </el-table-column>
+          <el-table-column :label="t('iotDevice.code')" align="center" prop="deviceCode" min-width="120"
+                           :show-overflow-tooltip="true" fixed="left"/>
+          <el-table-column :label="t('iotDevice.name')" align="center" prop="deviceName" :min-width="deviceNameMinWidth"
+                           :width="columnWidths['deviceName']" :show-overflow-tooltip="true"
+                           :max-width="deviceNameMaxWidth" fixed="left">
+            <template #default="scope">
+              <el-link :underline="false" type="primary" @click="handleDetail(scope.row.id)">
+                {{ scope.row.deviceName }}
+              </el-link>
+            </template>
+          </el-table-column>
+          <el-table-column :label="t('bomList.serviceDue')" align="center" min-width="120">
+            <template #default="scope">
+              <template v-if="hasMaintenancePlan(scope.row.mainDistance)">
+                <span :class="getDistanceClass(scope.row.mainDistance)">
+                  {{ scope.row.mainDistance }}
+                </span>
+              </template>
+              <span v-else>无保养计划</span>
+            </template>
+          </el-table-column>
+          <el-table-column :label="t('iotDevice.dept')" align="center" prop="deptName" min-width="120"/>
+          <el-table-column :label="t('monitor.status')" align="center" prop="deviceStatus" min-width="100">
+            <template #default="scope">
+              <dict-tag :type="DICT_TYPE.PMS_DEVICE_STATUS" :value="scope.row.deviceStatus" />
+            </template>
+          </el-table-column>
+
+          <!-- 动态分组列 - 修改后结构 -->
+          <template v-for="group in dynamicColumns" :key="group.header">
+            <el-table-column :label="group.header" align="center" :header-cell-class-name="`group-header-${group.groupIndex}`" min-width="auto">
+              <!-- 循环分组内的所有类型 -->
+              <template v-for="type in group.types" :key="type">
+                <!-- 类型标签列 -->
+                <el-table-column :label="getTypeLabel(type)" align="center" min-width="auto">
+                  <!-- 根据类型渲染对应字段 -->
+                  <!-- 1. 时间类型字段 -->
+                  <template v-if="type === 'time'">
+                    <el-table-column
+                      :label="t('mainPlan.lastMaintenanceOperationTime')"
+                      :width="getDynamicColumnWidth(group, type, 'lastRunningTime')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        {{ scope.row.groupBomMap?.[group.header]?.lastRunningTime ?? '-' }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('mainPlan.nextMaintenanceH')"
+                      :width="getDynamicColumnWidth(group, type, 'nextMaintenanceH')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        {{ (scope.row.groupBomMap?.[group.header]?.lastRunningTime || 0) +
+                      (scope.row.groupBomMap?.[group.header]?.nextRunningTime || 0) }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('operationFillForm.sumTime')"
+                      :width="getDynamicColumnWidth(group, type, 'totalRunTime')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        {{ scope.row.groupBomMap?.[group.header]?.totalRunTime ?? '-' }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('mainPlan.remainH')"
+                      :width="getDynamicColumnWidth(group, type, 'remainH')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        <span :class="getNewRemainingClass(scope.row.groupBomMap?.[group.header], type)">
+                          {{ getNewRemainingValue(scope.row.groupBomMap?.[group.header], type) }}
+                        </span>
+                      </template>
+                    </el-table-column>
+                  </template>
+
+                  <!-- 2. 日期类型字段 -->
+                  <template v-else-if="type === 'date'">
+                    <el-table-column
+                      :label="t('mainPlan.lastMaintenanceNaturalDate')"
+                      :width="getDynamicColumnWidth(group, type, 'lastNaturalDate')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        {{ formatDate(scope.row.groupBomMap?.[group.header]?.lastNaturalDate) }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('mainPlan.nextMaintDate')"
+                      :width="getDynamicColumnWidth(group, type, 'nextMaintDate')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        {{
+                          formatDate(
+                            (scope.row.groupBomMap?.[group.header]?.lastNaturalDate || 0) +
+                            (scope.row.groupBomMap?.[group.header]?.nextNaturalDate || 0) * 86400000
+                          )
+                        }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('main.currentDate')"
+                      :width="getDynamicColumnWidth(group, type, 'currentDate')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default>
+                        {{ currentDateFormatted }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('mainPlan.remainDay')"
+                      :width="getDynamicColumnWidth(group, type, 'remainDay')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        <span :class="getNewRemainingClass(scope.row.groupBomMap?.[group.header], type)">
+                          {{ getNewRemainingValue(scope.row.groupBomMap?.[group.header], type) }}
+                        </span>
+                      </template>
+                    </el-table-column>
+                  </template>
+
+                  <!-- 3. 里程类型字段 -->
+                  <template v-else-if="type === 'mileage'">
+                    <el-table-column
+                      :label="t('mainPlan.lastMaintenanceMileage')"
+                      :width="getDynamicColumnWidth(group, type, 'lastRunningKilometers')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        {{ scope.row.groupBomMap?.[group.header]?.lastRunningKilometers ?? '-' }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('mainPlan.nextMaintenanceKm')"
+                      :width="getDynamicColumnWidth(group, type, 'nextMaintenanceKm')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        {{ (scope.row.groupBomMap?.[group.header]?.lastRunningKilometers || 0) +
+                      (scope.row.groupBomMap?.[group.header]?.nextRunningKilometers || 0) }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('operationFillForm.sumKil')"
+                      :width="getDynamicColumnWidth(group, type, 'totalMileage')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        {{ scope.row.groupBomMap?.[group.header]?.totalMileage ?? '-' }}
+                      </template>
+                    </el-table-column>
+                    <el-table-column
+                      :label="t('mainPlan.remainKm')"
+                      :width="getDynamicColumnWidth(group, type, 'remainKm')"
+                      :show-overflow-tooltip="true"
+                      align="center">
+                      <template #default="scope">
+                        <span :class="getNewRemainingClass(scope.row.groupBomMap?.[group.header], type)">
+                          {{ getNewRemainingValue(scope.row.groupBomMap?.[group.header], type) }}
+                        </span>
+                      </template>
+                    </el-table-column>
+                  </template>
+                </el-table-column>
+              </template>
+            </el-table-column>
+          </template>
+
+          <el-table-column :label="t('monitor.operation')" align="center" min-width="120" fixed="right">
+            <template #default="scope">
+              <el-button
+                link
+                type="primary"
+                @click="openBomForm(scope.row)"
+                v-hasPermi="['rq:iot-device:query']"
+                v-if="hasMaintenancePlan(scope.row.mainDistance)"
+              >
+                {{ t('monitor.details') }}
+              </el-button>
+            </template>
+          </el-table-column>
+        </el-table>
+        <!-- 分页 -->
+        <Pagination
+          :total="total"
+          v-model:page="queryParams.pageNo"
+          v-model:limit="queryParams.pageSize"
+          @pagination="getList"
+        />
+      </ContentWrap>
+    </el-col>
+  </el-row>
+  <DeviceAlarmBomList ref="modelFormRef" :flag = "flag" />
+</template>
+
+<script setup lang="ts">
+import download from '@/utils/download'
+import { IotDeviceApi, IotDeviceVO } from '@/api/pms/device'
+import { IotMainWorkOrderApi, IotMainWorkOrderVO } from '@/api/pms/iotmainworkorder'
+import { DICT_TYPE, getStrDictOptions } from '@/utils/dict'
+import DeptTree from '@/views/system/user/DeptTree.vue'
+import { CACHE_KEY, useCache } from '@/hooks/web/useCache'
+import DeviceAlarmBomList from "@/views/pms/iotmainworkorder/DeviceAlarmBomList.vue";
+import dayjs from 'dayjs' // 引入 dayjs 库
+
+/** 保养查询 列表 保养项分组 */
+defineOptions({ name: 'IotMaintenanceSearch' })
+
+const message = useMessage() // 消息弹窗
+const { t } = useI18n() // 国际化
+const { push } = useRouter() // 路由跳转
+const modelFormRef = ref()
+const loading = ref(true) // 列表的加载中
+
+const list = ref<IotDeviceVO[]>([]) // 列表的数据
+const total = ref(0) // 列表的总页数
+const queryParams = reactive({
+  pageNo: 1,
+  pageSize: 10,
+  deviceCode: undefined,
+  deviceName: undefined,
+  brand: undefined,
+  model: undefined,
+  deptId: undefined,
+  deviceStatus: undefined,
+  assetProperty: undefined,
+  picUrl: undefined,
+  remark: undefined,
+  manufacturerId: undefined,
+  supplierId: undefined,
+  manDate: [],
+  nameplate: undefined,
+  expires: undefined,
+  plPrice: undefined,
+  plDate: [],
+  plYear: undefined,
+  plStartDate: [],
+  plMonthed: undefined,
+  plAmounted: undefined,
+  remainAmount: undefined,
+  infoId: undefined,
+  infoType: undefined,
+  infoName: undefined,
+  infoRemark: undefined,
+  infoUrl: undefined,
+  templateJson: undefined,
+  creator: undefined,
+  setFlag: ''
+})
+
+const dynamicColumns = ref<any[]>([]) // 动态列配置
+const queryFormRef = ref() // 搜索的表单
+const flag = ref() // 查询保养计划或保养工单的标识
+const exportLoading = ref(false) // 导出的加载中
+const contentSpan = ref(20)
+const treeShow = ref(true)
+const currentDateFormatted = ref(dayjs().format('YYYY-MM-DD')) // 当前日期格式化
+
+const tableRef = ref(); // 表格引用
+const columnWidths = ref<Record<string, string>>({}); // 列宽存储
+
+// 设备名称列的最小和最大宽度
+const deviceNameMinWidth = 150; // 最小宽度
+const deviceNameMaxWidth = 400; // 最大宽度,防止过长破坏布局
+
+// 表头样式
+const tableHeaderStyle = () => {
+  return {
+    borderBottom: '2px solid #606266',
+    borderRight: '2px solid #606266',
+    backgroundColor: '#f5f7fa',
+    fontWeight: 'bold',
+    whiteSpace: 'nowrap'
+  }
+}
+
+// 处理列宽拖动事件
+const handleHeaderDragEnd = (newWidth: number, oldWidth: number, column: any) => {
+  if (column.property) {
+    columnWidths.value[column.property] = `${newWidth}px`;
+  } else if (column.label) {
+    // 处理没有property的列(如动态列)
+    columnWidths.value[column.label] = `${newWidth}px`;
+  }
+}
+
+/** 计算文本宽度辅助函数 */
+const getTextWidth = (text: string, fontSize = 14): number => {
+  const span = document.createElement('span');
+  span.style.visibility = 'hidden';
+  span.style.position = 'absolute';
+  span.style.whiteSpace = 'nowrap';
+  span.style.fontSize = `${fontSize}px`;
+  span.innerText = text;
+  document.body.appendChild(span);
+  const width = span.offsetWidth;
+  document.body.removeChild(span);
+  return width;
+}
+
+/** 计算列宽主函数 */
+const calculateColumnWidths = () => {
+  const MIN_WIDTH = 100; // 最小列宽
+  const PADDING = 20; // 内边距
+  const MAX_DEVICE_NAME_WIDTH = deviceNameMaxWidth; // 设备名称最大宽度
+
+  // 静态列配置
+  const staticColumns = [
+    { prop: 'serial', label: t('monitor.serial') },
+    { prop: 'deviceCode', label: t('iotDevice.code') },
+    { prop: 'deviceName', label: t('iotDevice.name') },
+    { prop: 'mainDistance', label: t('bomList.serviceDue') },
+    { prop: 'deptName', label: t('iotDevice.dept') },
+    { prop: 'deviceStatus', label: t('monitor.status') },
+    { prop: 'operation', label: t('monitor.operation') }
+  ];
+
+  const newWidths: Record<string, string> = {};
+
+  // 计算静态列宽度
+  staticColumns.forEach(col => {
+    const headerWidth = getTextWidth(col.label);
+    let maxContentWidth = 0;
+
+    list.value.forEach(item => {
+      let value = '';
+      if (col.prop === 'serial') {
+        value = (list.value.indexOf(item) + 1).toString();
+      } else if (col.prop === 'deviceName') {
+        // 设备名称特殊处理,考虑链接内容
+        value = item.deviceName || '';
+      } else {
+        value = item[col.prop]?.toString() || '';
+      }
+
+      const contentWidth = getTextWidth(value);
+      if (contentWidth > maxContentWidth) maxContentWidth = contentWidth;
+    });
+
+    // 对设备名称列应用最大宽度限制
+    if (col.prop === 'deviceName') {
+      maxContentWidth = Math.min(maxContentWidth, MAX_DEVICE_NAME_WIDTH);
+    }
+    const finalWidth = Math.max(headerWidth, maxContentWidth, MIN_WIDTH) + PADDING;
+    newWidths[col.prop] = `${finalWidth}px`;
+  });
+
+  // 计算动态列宽度
+  dynamicColumns.value.forEach((group, groupIndex) => {
+    group.types.forEach(type => {
+      const typeLabel = getTypeLabel(type);
+      const typeHeaderWidth = getTextWidth(typeLabel);
+
+      // 根据类型定义子列
+      const subColumns = [];
+      if (type === 'time') {
+        subColumns.push(
+          { key: 'lastRunningTime', label: t('mainPlan.lastMaintenanceOperationTime') },
+          { key: 'nextMaintenanceH', label: t('mainPlan.nextMaintenanceH') },
+          { key: 'totalRunTime', label: t('operationFillForm.sumTime') },
+          { key: 'remainH', label: t('mainPlan.remainH') }
+        );
+      } else if (type === 'date') {
+        subColumns.push(
+          { key: 'lastNaturalDate', label: t('mainPlan.lastMaintenanceNaturalDate') },
+          { key: 'nextMaintDate', label: t('mainPlan.nextMaintDate') },
+          { key: 'currentDate', label: t('main.currentDate') },
+          { key: 'remainDay', label: t('mainPlan.remainDay') }
+        );
+      } else if (type === 'mileage') {
+        subColumns.push(
+          { key: 'lastRunningKilometers', label: t('mainPlan.lastMaintenanceMileage') },
+          { key: 'nextMaintenanceKm', label: t('mainPlan.nextMaintenanceKm') },
+          { key: 'totalMileage', label: t('operationFillForm.sumKil') },
+          { key: 'remainKm', label: t('mainPlan.remainKm') }
+        );
+      }
+
+      // 计算每个子列的宽度
+      subColumns.forEach(subCol => {
+        const fullKey = `${group.header}-${type}-${subCol.key}`;
+        const headerWidth = getTextWidth(subCol.label);
+        let maxContentWidth = 0;
+
+        list.value.forEach(item => {
+          const groupData = item.groupBomMap?.[group.header];
+          let value = '';
+
+          if (subCol.key === 'currentDate') {
+            value = currentDateFormatted.value;
+          } else if (groupData) {
+            // 特殊处理计算字段
+            if (subCol.key === 'nextMaintenanceH') {
+              value = ((groupData.lastRunningTime || 0) + (groupData.nextRunningTime || 0)).toString();
+            }
+            else if (subCol.key === 'remainH') {
+              value = getNewRemainingValue(groupData, 'time').toString();
+            }
+            else if (subCol.key === 'nextMaintDate') {
+              const nextDate = (groupData.lastNaturalDate || 0) +
+                (groupData.nextNaturalDate || 0) * 86400000;
+              value = formatDate(nextDate);
+            }
+            else if (subCol.key === 'remainDay') {
+              value = getNewRemainingValue(groupData, 'date').toString();
+            }
+            else if (subCol.key === 'nextMaintenanceKm') {
+              value = ((groupData.lastRunningKilometers || 0) +
+                (groupData.nextRunningKilometers || 0)).toString();
+            }
+            else if (subCol.key === 'remainKm') {
+              value = getNewRemainingValue(groupData, 'mileage').toString();
+            }
+            // 其他字段直接取值
+            else {
+              value = groupData[subCol.key]?.toString() || '';
+            }
+          } else {
+            value = '-'; // 无数据时显示短横线
+          }
+
+          const contentWidth = getTextWidth(value);
+          if (contentWidth > maxContentWidth) maxContentWidth = contentWidth;
+        });
+
+        const finalWidth = Math.max(headerWidth, maxContentWidth, MIN_WIDTH) + PADDING;
+        newWidths[fullKey] = `${finalWidth}px`;
+      });
+    });
+  });
+
+  columnWidths.value = newWidths;
+}
+
+/** 获取动态列宽 */
+const getDynamicColumnWidth = (group: any, type: string, key: string) => {
+  const fullKey = `${group.header}-${type}-${key}`;
+  return columnWidths.value[fullKey] || 'auto';
+}
+
+/** 查询列表 */
+const getList = async () => {
+  loading.value = true
+  try {
+    const data = await IotMainWorkOrderApi.maintenanceSearch(queryParams)
+    list.value = data.list.map((item: any) => {
+      // 构建分组映射
+      const groupBomMap: Record<string, any> = {}
+      if (item.groupBomDistances?.length > 0) {
+        item.groupBomDistances.forEach((group: any) => {
+          // 提取分组名称
+          const groupName = group.name?.split('->')[0] || group.name
+          groupBomMap[groupName] = group
+        })
+      }
+      return { ...item, groupBomMap }
+    })
+    total.value = data.total
+
+    // 生成动态列配置
+    generateDynamicColumns(data.list)
+
+    // 计算列宽
+    nextTick(() => {
+      calculateColumnWidths();
+      tableRef.value?.doLayout(); // 重新布局表格
+    });
+  } finally {
+    loading.value = false
+  }
+}
+
+/** 生成动态列配置 */
+const generateDynamicColumns = (dataList: any[]) => {
+  // 使用Map收集分组和类型:Map<分组名称, Set<类型>>
+  const groupTypeMap = new Map<string, Set<string>>()
+
+  dataList.forEach(item => {
+    if (item.groupBomDistances?.length > 0) {
+      item.groupBomDistances.forEach((group: any) => {
+        const groupName = group.name?.split('->')[0] || group.name
+        if (!groupName) return
+
+        // 确定分组类型
+        let type = ''
+        if (group.lastRunningTime != null && group.nextRunningTime != null) {
+          type = 'time'
+        } else if (group.lastNaturalDate != null && group.nextNaturalDate != null) {
+          type = 'date'
+        } else if (group.lastRunningKilometers != null && group.nextRunningKilometers != null) {
+          type = 'mileage'
+        }
+        if (!type) return
+
+        if (!groupTypeMap.has(groupName)) {
+          groupTypeMap.set(groupName, new Set())
+        }
+        groupTypeMap.get(groupName)?.add(type)
+      })
+    }
+  })
+
+  // 转换为需要的结构并按固定顺序排序类型
+  dynamicColumns.value = Array.from(groupTypeMap.entries()).map(([header, typeSet], index) => ({
+    header,
+    types: Array.from(typeSet).sort((a, b) => {
+      // 固定顺序:time -> date -> mileage
+      const order = { 'time': 1, 'date': 2, 'mileage': 3 }
+      return (order[a] || 4) - (order[b] || 5)
+    }),
+    groupIndex: index % 4 // 循环使用4种样式
+  }))
+
+  // 计算动态列宽度
+  nextTick(calculateColumnWidths);
+}
+
+/** 获取类型标签 */
+const getTypeLabel = (type: string) => {
+  const labels = {
+    'time': t('main.maintenanceRuntime'),
+    'date': t('main.maintenanceDate'),
+    'mileage': t('main.maintenanceMileage')
+  }
+  return labels[type] || type
+}
+
+/** 格式化日期 */
+const formatDate = (timestamp: number | null) => {
+  if (!timestamp) return '-'
+  const date = new Date(timestamp)
+  return `${date.getFullYear()}-${(date.getMonth() + 1).toString().padStart(2, '0')}-${date.getDate().toString().padStart(2, '0')}`
+}
+
+/** 新的剩余值计算方法 */
+const getNewRemainingValue = (group: any, type: string) => {
+  if (!group) return '-'
+
+  switch (type) {
+    case 'time':
+      // 剩余时长 = nextRunningTime - (totalRunTime - lastRunningTime)
+      const totalRunTime = group.totalRunTime || 0
+      const lastRunningTime = group.lastRunningTime || 0
+      const nextRunningTime = group.nextRunningTime || 0
+      return (nextRunningTime - (totalRunTime - lastRunningTime)).toFixed(2)
+
+    case 'mileage':
+      // 剩余里程 = nextRunningKilometers - (totalMileage - lastRunningKilometers)
+      const totalMileage = group.totalMileage || 0
+      const lastRunningKilometers = group.lastRunningKilometers || 0
+      const nextRunningKilometers = group.nextRunningKilometers || 0
+      return (nextRunningKilometers - (totalMileage - lastRunningKilometers)).toFixed(2)
+
+    case 'date':
+      // 计算剩余天数
+      if (!group.lastNaturalDate || !group.nextNaturalDate) return '-'
+
+      try {
+        // 上次保养日期:将时间戳转换为 Day.js 对象
+        const lastNaturalDate = dayjs(group.lastNaturalDate)
+        // 计算下次保养日期
+        const nextMaintenanceDate = lastNaturalDate.add(group.nextNaturalDate, 'day')
+        // 计算剩余天数(当前日期到下次保养日期的天数差)
+        return nextMaintenanceDate.diff(dayjs(), 'day')
+      } catch (e) {
+        console.error('日期计算错误:', e)
+        return '-'
+      }
+
+    default:
+      return '-'
+  }
+}
+
+/** 获取剩余值 */
+const getRemainingValue = (group: any, type: string) => {
+  if (!group) return '-'
+
+  switch (type) {
+    case 'time':
+      return group.nextRunningTime ?? '-'
+    case 'date':
+      return group.nextNaturalDate ?? '-'
+    case 'mileage':
+      return group.nextRunningKilometers ?? '-'
+    default:
+      return '-'
+  }
+}
+
+/** 新的剩余值样式 */
+const getNewRemainingClass = (group: any, type: string) => {
+  const value = getNewRemainingValue(group, type)
+  if (value === '-' || value === null) return ''
+
+  const numValue = typeof value === 'string' ? parseFloat(value) : value
+  return numValue < 0 ? 'negative-distance' :
+    numValue > 0 ? 'positive-distance' : ''
+}
+
+/** 获取剩余值样式 */
+const getRemainingClass = (group: any, type: string) => {
+  const value = getRemainingValue(group, type)
+  if (value === '-' || value === null) return ''
+
+  const numValue = typeof value === 'string' ? parseFloat(value) : value
+  return numValue < 0 ? 'negative-distance' :
+    numValue > 0 ? 'positive-distance' : ''
+}
+
+/** 判断是否有分组数据 */
+const hasGroupBomDistances = (row: any) => {
+  return row.groupBomDistances?.length > 0
+}
+
+/** 处理部门被点击 */
+const handleDeptNodeClick = async (row) => {
+  queryParams.deptId = row.id
+  await getList()
+}
+
+/** 搜索按钮操作 */
+const handleQuery = () => {
+  queryParams.pageNo = 1
+  getList()
+}
+
+/** 重置按钮操作 */
+const resetQuery = () => {
+  queryFormRef.value.resetFields()
+  handleQuery()
+}
+
+const getDistanceClass = (distance: number | string | null) => {
+  if (distance === null || distance === undefined) return '';
+
+  // 如果是数字类型,直接处理
+  if (typeof distance === 'number') {
+    return distance < 0 ? 'negative-distance' :
+      distance > 0 ? 'positive-distance' : '';
+  }
+
+  // 如果是字符串,提取数字部分
+  if (typeof distance === 'string') {
+    // 使用正则提取数字部分(包括负号、小数点和科学计数法)
+    const numericPart = distance.match(/[-+]?\d*\.?\d+(?:[eE][-+]?\d+)?/)?.[0];
+
+    // 如果提取到数字部分,转换为数值
+    if (numericPart) {
+      const num = parseFloat(numericPart);
+      return num < 0 ? 'negative-distance' :
+        num > 0 ? 'positive-distance' : '';
+    }
+  }
+
+  return '';
+};
+
+// 判断是否有保养计划
+const hasMaintenancePlan = (mainDistance: any) => {
+  // 检查:非null、非undefined、非空字符串
+  return mainDistance != null && mainDistance !== '';
+};
+
+const handleDetail = (id: number) => {
+  push({ name: 'DeviceDetailInfo', params: { id } })
+}
+
+const drawerVisible = ref<boolean>(false)
+const showDrawer = ref()
+
+const openBomForm = async (row) => {
+  if (row.workOrderId) {
+    flag.value = 'workOrder';
+    modelFormRef.value.open(row.workOrderId, flag.value, row.id)
+  } else if (row.planId) {
+    flag.value = 'plan';
+    modelFormRef.value.open(row.planId, flag.value, row.id)
+  }
+}
+
+/** 导出按钮操作 */
+const handleExport = async () => {
+  try {
+    // 导出的二次确认
+    await message.exportConfirm()
+    // 发起导出
+    exportLoading.value = true
+    const data = await IotDeviceApi.exportIotDevice(queryParams)
+    download.excel(data, '设备台账.xls')
+  } catch {
+  } finally {
+    exportLoading.value = false
+  }
+}
+const { wsCache } = useCache()
+/** 初始化 **/
+onMounted(() => {
+  getList()
+})
+</script>
+<style scoped>
+/* 正数样式 - 淡绿色 */
+.positive-distance {
+  color: #67c23a;  /* element-plus 成功色 */
+  background-color: rgba(103, 194, 58, 0.1); /* 10% 透明度的淡绿色背景 */
+  padding: 2px 8px;
+  border-radius: 4px;
+  display: inline-block;
+}
+
+/* 负数样式 - 淡红色 */
+.negative-distance {
+  color: #f56c6c;  /* element-plus 危险色 */
+  background-color: rgba(245, 108, 108, 0.1); /* 10% 透明度的淡红色背景 */
+  padding: 2px 8px;
+  border-radius: 4px;
+  display: inline-block;
+}
+
+/* 分组表头样式 - 使用::v-deep穿透 */
+:deep(.group-header-0) {
+  background-color: #f0f9ff !important; /* 淡蓝色 */
+  border-right: 2px solid #606266 !important;
+}
+
+:deep(.group-header-1) {
+  background-color: #fdf6ec !important; /* 淡橙色 */
+  border-right: 2px solid #606266 !important;
+}
+
+:deep(.group-header-2) {
+  background-color: #f0f4ff !important; /* 淡紫色 */
+  border-right: 2px solid #606266 !important;
+}
+
+:deep(.group-header-3) {
+  background-color: #e6f7ff !important; /* 淡青色 */
+  border-right: 2px solid #606266 !important;
+}
+
+/* 表头边框加深 */
+:deep(.el-table th) {
+  border-bottom: 2px solid #606266 !important;
+  border-right: 2px solid #606266 !important;
+}
+
+/* 内容不换行 */
+:deep(.el-table .cell) {
+  white-space: nowrap !important;
+}
+
+/* 表头文字不换行 */
+:deep(.el-table__header .cell) {
+  white-space: nowrap !important;
+  overflow: visible !important;
+  text-overflow: unset !important;
+}
+
+/* 列宽自适应调整 */
+:deep(.el-table) {
+  table-layout: auto !important;
+}
+
+/* 分组列之间的分隔线 */
+:deep(.el-table__header .el-table__cell) {
+  overflow: visible !important;
+  border-right: 2px solid #606266 !important;
+}
+
+/* 最后列去除右边框 */
+:deep(.el-table__header .el-table__cell:last-child) {
+  border-right: none !important;
+}
+
+/* 设备名称列特殊处理 */
+:deep(.el-table__body .device-name-cell .cell) {
+  overflow: hidden;
+  text-overflow: ellipsis;
+}
+</style>