|
@@ -0,0 +1,781 @@
|
|
|
|
+<template>
|
|
|
|
+ <ContentWrap>
|
|
|
|
+ <!-- 搜索工作栏 -->
|
|
|
|
+ <el-form
|
|
|
|
+ class="-mb-15px"
|
|
|
|
+ :model="queryParams"
|
|
|
|
+ ref="queryFormRef"
|
|
|
|
+ :inline="true"
|
|
|
|
+ label-width="68px"
|
|
|
|
+ >
|
|
|
|
+ <el-form-item label="项目" prop="contractName">
|
|
|
|
+ <el-input
|
|
|
|
+ v-model="queryParams.contractName"
|
|
|
|
+ placeholder="请输入项目"
|
|
|
|
+ clearable
|
|
|
|
+ @keyup.enter="handleQuery"
|
|
|
|
+ class="!w-240px"
|
|
|
|
+ />
|
|
|
|
+ </el-form-item>
|
|
|
|
+ <el-form-item label="任务" prop="taskName">
|
|
|
|
+ <el-input
|
|
|
|
+ v-model="queryParams.taskName"
|
|
|
|
+ placeholder="请输入任务"
|
|
|
|
+ clearable
|
|
|
|
+ @keyup.enter="handleQuery"
|
|
|
|
+ class="!w-240px"
|
|
|
|
+ />
|
|
|
|
+ </el-form-item>
|
|
|
|
+ <el-form-item label="创建时间" prop="createTime">
|
|
|
|
+ <el-date-picker
|
|
|
|
+ v-model="queryParams.createTime"
|
|
|
|
+ value-format="YYYY-MM-DD HH:mm:ss"
|
|
|
|
+ type="daterange"
|
|
|
|
+ start-placeholder="开始日期"
|
|
|
|
+ end-placeholder="结束日期"
|
|
|
|
+ :default-time="[new Date('1 00:00:00'), new Date('1 23:59:59')]"
|
|
|
|
+ class="!w-220px"
|
|
|
|
+ />
|
|
|
|
+ </el-form-item>
|
|
|
|
+ <el-form-item>
|
|
|
|
+ <el-button @click="handleQuery"><Icon icon="ep:search" class="mr-5px" /> 搜索</el-button>
|
|
|
|
+ <el-button @click="resetQuery"><Icon icon="ep:refresh" class="mr-5px" /> 重置</el-button>
|
|
|
|
+ <el-button
|
|
|
|
+ type="primary"
|
|
|
|
+ plain
|
|
|
|
+ @click="openForm('create')"
|
|
|
|
+ v-hasPermi="['pms:iot-rh-daily-report:create']"
|
|
|
|
+ >
|
|
|
|
+ <Icon icon="ep:plus" class="mr-5px" /> 新增
|
|
|
|
+ </el-button>
|
|
|
|
+ <el-button
|
|
|
|
+ type="success"
|
|
|
|
+ plain
|
|
|
|
+ @click="handleExport"
|
|
|
|
+ :loading="exportLoading"
|
|
|
|
+ v-hasPermi="['pms:iot-rh-daily-report:export']"
|
|
|
|
+ >
|
|
|
|
+ <Icon icon="ep:download" class="mr-5px" /> 导出
|
|
|
|
+ </el-button>
|
|
|
|
+ </el-form-item>
|
|
|
|
+ </el-form>
|
|
|
|
+ </ContentWrap>
|
|
|
|
+
|
|
|
|
+ <ContentWrap class="mb-15px">
|
|
|
|
+ <div class="color-legend">
|
|
|
|
+ <div class="legend-item">
|
|
|
|
+ <span class="color-indicator red"></span>
|
|
|
|
+ <span>运行时效=生产时间/额定生产时间 超过100%红色预警</span>
|
|
|
|
+ </div>
|
|
|
|
+ <div class="legend-item">
|
|
|
|
+ <span class="color-indicator orange"></span>
|
|
|
|
+ <span>生产时间+非生产时间=额定生产时间 否则橙色预警</span>
|
|
|
|
+ </div>
|
|
|
|
+ </div>
|
|
|
|
+ </ContentWrap>
|
|
|
|
+
|
|
|
|
+ <!-- 列表 -->
|
|
|
|
+ <ContentWrap ref="tableContainerRef">
|
|
|
|
+ <div class="table-container">
|
|
|
|
+ <el-table ref="tableRef" v-loading="loading" :data="list" :stripe="true"
|
|
|
|
+ style="width: 100%" :cell-style="cellStyle">
|
|
|
|
+ <el-table-column :label="t('iotDevice.serial')" width="70" align="center">
|
|
|
|
+ <template #default="scope">
|
|
|
|
+ {{ scope.$index + 1 }}
|
|
|
|
+ </template>
|
|
|
|
+ </el-table-column>
|
|
|
|
+ <el-table-column
|
|
|
|
+ label="日期"
|
|
|
|
+ align="center"
|
|
|
|
+ prop="createTime"
|
|
|
|
+ :formatter="dateFormatter2"
|
|
|
|
+ :width="columnWidths.createTime"
|
|
|
|
+ />
|
|
|
|
+ <el-table-column label="施工队伍" align="center" prop="deptName" :width="columnWidths.deptName"/>
|
|
|
|
+ <el-table-column label="项目" align="center" prop="contractName" :width="columnWidths.contractName"/>
|
|
|
|
+ <el-table-column label="任务" align="center" prop="taskName" :width="columnWidths.taskName"/>
|
|
|
|
+ <el-table-column :label="t('project.status')" align="center" prop="rigStatus" :width="columnWidths.rigStatus">
|
|
|
|
+ <template #default="scope">
|
|
|
|
+ <dict-tag :type="DICT_TYPE.PMS_PROJECT_TASK_RY_SCHEDULE" :value="scope.row.rigStatus" />
|
|
|
|
+ </template>
|
|
|
|
+ </el-table-column>
|
|
|
|
+
|
|
|
|
+ <!--
|
|
|
|
+ <el-table-column label="上井次完井时间" align="center" prop="latestWellDoneTime" :width="columnWidths.latestWellDoneTime" :formatter="dateFormatter2"/>
|
|
|
|
+ <el-table-column label="设计井深(m)" align="center" prop="designWellDepth" :width="columnWidths.designWellDepth"/>
|
|
|
|
+ <el-table-column label="当前井深(m)" align="center" prop="currentDepth" :width="columnWidths.currentDepth" />
|
|
|
|
+ <el-table-column label="日进尺(m)" align="center" prop="dailyFootage" :width="columnWidths.dailyFootage" />
|
|
|
|
+ <el-table-column label="月进尺(m)" align="center" prop="monthlyFootage" :width="columnWidths.monthlyFootage"/>
|
|
|
|
+ <el-table-column label="年累计进尺(m)" align="center" prop="annualFootage" :width="columnWidths.annualFootage"/> -->
|
|
|
|
+ <el-table-column label="总施工井数" align="center" prop="totalConstructionWells" :width="columnWidths.totalConstructionWells"/>
|
|
|
|
+ <el-table-column label="完工井数" align="center" prop="completedWells" :width="columnWidths.completedWells"/>
|
|
|
|
+ <el-table-column :label="t('project.technology')" align="center" prop="technique" :width="columnWidths.technique">
|
|
|
|
+ <template #default="scope">
|
|
|
|
+ <dict-tag :type="DICT_TYPE.PMS_PROJECT_RY_TECHNOLOGY" :value="scope.row.technique" />
|
|
|
|
+ </template>
|
|
|
|
+ </el-table-column>
|
|
|
|
+ <el-table-column label="井别" align="center" prop="wellCategory" :width="columnWidths.wellCategory"/>
|
|
|
|
+ <el-table-column label="井深(m)" align="center" prop="designWellDepth" :width="columnWidths.designWellDepth"/>
|
|
|
|
+ <el-table-column label="套生段产管尺寸(mm)" align="center" prop="casingPipeSize" :width="columnWidths.casingPipeSize"/>
|
|
|
|
+ <el-table-column label="井控级别" align="center" prop="wellControlLevel" :width="columnWidths.wellControlLevel"/>
|
|
|
|
+ <!--
|
|
|
|
+ <el-table-column label="泥浆性能-密度(g/cm³)" align="center" prop="mudDensity" :width="columnWidths.mudDensity"/>
|
|
|
|
+ <el-table-column label="泥浆性能-粘度(S)" align="center" prop="mudViscosity" :width="columnWidths.mudViscosity"/> -->
|
|
|
|
+ <el-table-column
|
|
|
|
+ label="施工开始日期"
|
|
|
|
+ align="center"
|
|
|
|
+ prop="constructionStartDate"
|
|
|
|
+ :formatter="dateFormatter"
|
|
|
|
+ :width="columnWidths.constructionStartDate"
|
|
|
|
+ />
|
|
|
|
+ <el-table-column
|
|
|
|
+ label="施工结束日期"
|
|
|
|
+ align="center"
|
|
|
|
+ prop="constructionEndDate"
|
|
|
|
+ :formatter="dateFormatter"
|
|
|
|
+ :width="columnWidths.constructionEndDate"
|
|
|
|
+ />
|
|
|
|
+ <!--
|
|
|
|
+ <el-table-column label="水平段长度(m)" align="center" prop="lateralLength" :width="columnWidths.lateralLength" />
|
|
|
|
+ <el-table-column label="井斜(°)" align="center" prop="wellInclination" :width="columnWidths.wellInclination"/>
|
|
|
|
+ <el-table-column label="方位(°)" align="center" prop="azimuth" :width="columnWidths.azimuth"/>
|
|
|
|
+ <el-table-column label="设计井身结构" align="center" :width="columnWidths.designWellStruct" fixed-width>
|
|
|
|
+ <template #default="scope">
|
|
|
|
+ <el-tooltip
|
|
|
|
+ effect="light"
|
|
|
|
+ :content="scope.row.designWellStruct"
|
|
|
|
+ placement="top"
|
|
|
|
+ popper-class="design-well-struct-tooltip"
|
|
|
|
+ :disabled="!scope.row.designWellStruct || scope.row.designWellStruct.length <= 30"
|
|
|
|
+ >
|
|
|
|
+ <span class="design-well-struct-text">{{ formatDesignWellStruct(scope.row.designWellStruct) }}</span>
|
|
|
|
+ </el-tooltip>
|
|
|
|
+ </template>
|
|
|
|
+ </el-table-column> -->
|
|
|
|
+ <el-table-column label="目前工序" align="center" prop="currentOperation" :width="columnWidths.currentOperation"/>
|
|
|
|
+ <el-table-column label="下部工序" align="center" prop="nextPlan" :width="columnWidths.nextPlan"/>
|
|
|
|
+ <el-table-column label="运行时效" align="center" prop="transitTime" :width="columnWidths.transitTime" :formatter="percentageFormatter"/>
|
|
|
|
+ <el-table-column label="额定生产时间(H)" align="center" prop="ratedProductionTime" :width="columnWidths.ratedProductionTime"/>
|
|
|
|
+ <el-table-column label="生产时间(H)" align="center" prop="productionTime" :width="columnWidths.productionTime"/>
|
|
|
|
+ <el-table-column label="非生产时间(H)" align="center" prop="nonProductionTime" :width="columnWidths.nonProductionTime"/>
|
|
|
|
+ <el-table-column :label="t('project.nptReason')" align="center" prop="ryNptReason" :width="columnWidths.ryNptReason">
|
|
|
|
+ <template #default="scope">
|
|
|
|
+ <dict-tag :type="DICT_TYPE.PMS_PROJECT_RY_NPT_REASON" :value="scope.row.ryNptReason" />
|
|
|
|
+ </template>
|
|
|
|
+ </el-table-column>
|
|
|
|
+ <el-table-column label="生产动态" align="center" :width="columnWidths.productionStatus" fixed-width>
|
|
|
|
+ <template #default="scope">
|
|
|
|
+ <el-tooltip
|
|
|
|
+ effect="light"
|
|
|
|
+ :content="scope.row.productionStatus"
|
|
|
|
+ placement="top"
|
|
|
|
+ popper-class="design-well-struct-tooltip"
|
|
|
|
+ :disabled="!scope.row.productionStatus || scope.row.productionStatus.length <= 30"
|
|
|
|
+ >
|
|
|
|
+ <span class="design-well-struct-text">{{ formatDesignWellStruct(scope.row.productionStatus) }}</span>
|
|
|
|
+ </el-tooltip>
|
|
|
|
+ </template>
|
|
|
|
+ </el-table-column>
|
|
|
|
+ <el-table-column label="操作" align="center" :width="columnWidths.operation" fixed="right">
|
|
|
|
+ <template #default="scope">
|
|
|
|
+ <el-button
|
|
|
|
+ link
|
|
|
|
+ type="primary"
|
|
|
|
+ @click="openForm('update', scope.row.id, scope.row)"
|
|
|
|
+ v-hasPermi="['pms:iot-rh-daily-report:update']"
|
|
|
|
+ >
|
|
|
|
+ 编辑
|
|
|
|
+ </el-button>
|
|
|
|
+ <el-button
|
|
|
|
+ link
|
|
|
|
+ type="danger"
|
|
|
|
+ @click="handleDelete(scope.row.id)"
|
|
|
|
+ v-hasPermi="['pms:iot-rh-daily-report:delete']"
|
|
|
|
+ >
|
|
|
|
+ 删除
|
|
|
|
+ </el-button>
|
|
|
|
+ </template>
|
|
|
|
+ </el-table-column>
|
|
|
|
+ </el-table>
|
|
|
|
+ </div>
|
|
|
|
+ <!-- 分页 -->
|
|
|
|
+ <Pagination
|
|
|
|
+ :total="total"
|
|
|
|
+ v-model:page="queryParams.pageNo"
|
|
|
|
+ v-model:limit="queryParams.pageSize"
|
|
|
|
+ @pagination="getList"
|
|
|
|
+ />
|
|
|
|
+ </ContentWrap>
|
|
|
|
+
|
|
|
|
+ <!-- 表单弹窗:添加/修改 -->
|
|
|
|
+ <IotRyDailyReportForm ref="formRef" @success="getList" :row-data="selectedRowData"/>
|
|
|
|
+</template>
|
|
|
|
+
|
|
|
|
+<script setup lang="ts">
|
|
|
|
+import {dateFormatter, dateFormatter2} from '@/utils/formatTime'
|
|
|
|
+import download from '@/utils/download'
|
|
|
|
+import { IotRyDailyReportApi, IotRyDailyReportVO } from '@/api/pms/iotrydailyreport'
|
|
|
|
+import IotRyDailyReportForm from './IotRyDailyReportForm.vue'
|
|
|
|
+import {DICT_TYPE, getDictLabel} from "@/utils/dict";
|
|
|
|
+import { ref, reactive, onMounted, nextTick, watch, onUnmounted } from 'vue'
|
|
|
|
+
|
|
|
|
+/** 瑞鹰日报 列表 */
|
|
|
|
+defineOptions({ name: 'IotRyXjDailyReport' })
|
|
|
|
+
|
|
|
|
+const message = useMessage() // 消息弹窗
|
|
|
|
+const { t } = useI18n() // 国际化
|
|
|
|
+
|
|
|
|
+// 添加 selectedRowData 响应式变量
|
|
|
|
+const selectedRowData = ref<Record<string, any> | null>(null)
|
|
|
|
+
|
|
|
|
+const loading = ref(true) // 列表的加载中
|
|
|
|
+const list = ref<IotRyDailyReportVO[]>([]) // 列表的数据
|
|
|
|
+const total = ref(0) // 列表的总页数
|
|
|
|
+const queryParams = reactive({
|
|
|
|
+ pageNo: 1,
|
|
|
|
+ pageSize: 10,
|
|
|
|
+ deptId: undefined,
|
|
|
|
+ contractName: undefined,
|
|
|
|
+ projectId: undefined,
|
|
|
|
+ taskName: undefined,
|
|
|
|
+ taskId: undefined,
|
|
|
|
+ projectClassification: '2',
|
|
|
|
+ relocationDays: undefined,
|
|
|
|
+ latestWellDoneTime: [],
|
|
|
|
+ currentDepth: undefined,
|
|
|
|
+ dailyFootage: undefined,
|
|
|
|
+ monthlyFootage: undefined,
|
|
|
|
+ annualFootage: undefined,
|
|
|
|
+ dailyPowerUsage: undefined,
|
|
|
|
+ monthlyPowerUsage: undefined,
|
|
|
|
+ dailyFuel: undefined,
|
|
|
|
+ monthlyFuel: undefined,
|
|
|
|
+ nonProductionTime: [],
|
|
|
|
+ nptReason: undefined,
|
|
|
|
+ constructionStartDate: [],
|
|
|
|
+ constructionEndDate: [],
|
|
|
|
+ productionStatus: undefined,
|
|
|
|
+ nextPlan: undefined,
|
|
|
|
+ rigStatus: undefined,
|
|
|
|
+ personnel: undefined,
|
|
|
|
+ mudDensity: undefined,
|
|
|
|
+ mudViscosity: undefined,
|
|
|
|
+ lateralLength: undefined,
|
|
|
|
+ wellInclination: undefined,
|
|
|
|
+ azimuth: undefined,
|
|
|
|
+ extProperty: undefined,
|
|
|
|
+ sort: undefined,
|
|
|
|
+ remark: undefined,
|
|
|
|
+ status: undefined,
|
|
|
|
+ processInstanceId: undefined,
|
|
|
|
+ auditStatus: undefined,
|
|
|
|
+ createTime: [],
|
|
|
|
+})
|
|
|
|
+const queryFormRef = ref() // 搜索的表单
|
|
|
|
+const exportLoading = ref(false) // 导出的加载中
|
|
|
|
+
|
|
|
|
+// 表格引用
|
|
|
|
+const tableRef = ref()
|
|
|
|
+// 表格容器引用
|
|
|
|
+const tableContainerRef = ref()
|
|
|
|
+
|
|
|
|
+// 列宽度配置
|
|
|
|
+const columnWidths = ref({
|
|
|
|
+ deptName: '120px',
|
|
|
|
+ contractName: '150px',
|
|
|
|
+ taskName: '120px',
|
|
|
|
+ technique: '120px',
|
|
|
|
+ rigStatus: '110px',
|
|
|
|
+ latestWellDoneTime: '120px',
|
|
|
|
+ designWellDepth: '120px',
|
|
|
|
+ wellCategory: '120px',
|
|
|
|
+ casingPipeSize: '120px',
|
|
|
|
+ wellControlLevel: '120px',
|
|
|
|
+ currentDepth: '100px',
|
|
|
|
+ dailyFootage: '150px',
|
|
|
|
+ monthlyFootage: '150px',
|
|
|
|
+ annualFootage: '150px',
|
|
|
|
+ totalConstructionWells: '150px',
|
|
|
|
+ completedWells: '150px',
|
|
|
|
+ mudDensity: '150px',
|
|
|
|
+ mudViscosity: '150px',
|
|
|
|
+ constructionStartDate: '180px',
|
|
|
|
+ constructionEndDate: '180px',
|
|
|
|
+ lateralLength: '150px',
|
|
|
|
+ wellInclination: '150px',
|
|
|
|
+ transitTime: '150px',
|
|
|
|
+ productionStatus: '200px',
|
|
|
|
+ currentOperation: '200px',
|
|
|
|
+ nextPlan: '200px',
|
|
|
|
+ ryNptReason: '150px',
|
|
|
|
+ azimuth: '150px',
|
|
|
|
+ designWellStruct: '200px',
|
|
|
|
+ ratedProductionTime: '150px',
|
|
|
|
+ productionTime: '150px',
|
|
|
|
+ nonProductionTime: '150px',
|
|
|
|
+ createTime: '180px',
|
|
|
|
+ operation: '120px'
|
|
|
|
+})
|
|
|
|
+
|
|
|
|
+// 格式化设计井身结构文本
|
|
|
|
+const formatDesignWellStruct = (text: string | null | undefined) => {
|
|
|
|
+ if (!text) return '-';
|
|
|
|
+ // 如果文本长度超过30个字符,显示前30个字符并添加省略号
|
|
|
|
+ return text.length > 30 ? text.substring(0, 30) + '...' : text;
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+// 检查10个时间字段之和是否为24H
|
|
|
|
+const checkTimeSumEquals24 = (row: any) => {
|
|
|
|
+ // 获取三个字段的值,转换为数字,如果为空则视为0
|
|
|
|
+ const drillingWorkingTime = parseFloat(row.drillingWorkingTime) || 0;
|
|
|
|
+ const otherProductionTime = parseFloat(row.otherProductionTime) || 0;
|
|
|
|
+ const accidentTime = parseFloat(row.accidentTime) || 0;
|
|
|
|
+ const repairTime = parseFloat(row.repairTime) || 0;
|
|
|
|
+ const selfStopTime = parseFloat(row.selfStopTime) || 0;
|
|
|
|
+ const complexityTime = parseFloat(row.complexityTime) || 0;
|
|
|
|
+ const relocationTime = parseFloat(row.relocationTime) || 0;
|
|
|
|
+ const rectificationTime = parseFloat(row.rectificationTime) || 0;
|
|
|
|
+ const waitingStopTime = parseFloat(row.waitingStopTime) || 0;
|
|
|
|
+ const winterBreakTime = parseFloat(row.winterBreakTime) || 0;
|
|
|
|
+
|
|
|
|
+ // 计算总和
|
|
|
|
+ const sum = drillingWorkingTime + otherProductionTime + accidentTime + repairTime
|
|
|
|
+ + selfStopTime + complexityTime + relocationTime + rectificationTime + waitingStopTime + winterBreakTime;
|
|
|
|
+
|
|
|
|
+ // 返回是否等于24(允许一定的浮点数误差)
|
|
|
|
+ return Math.abs(sum - 24) < 0.01; // 使用0.01作为误差范围
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+// 单元格样式函数
|
|
|
|
+const cellStyle = ({ row, column, rowIndex, columnIndex }: { row: any; column: any; rowIndex: number; columnIndex: number }) => {
|
|
|
|
+ // 1. 检查三个时间字段:额定生产时间、生产时间、非生产时间
|
|
|
|
+ const timeFields = ['ratedProductionTime', 'productionTime', 'nonProductionTime'];
|
|
|
|
+ if (timeFields.includes(column.property)) {
|
|
|
|
+ const ratedTime = parseFloat(row.ratedProductionTime) || 0
|
|
|
|
+ const prodTime = parseFloat(row.productionTime) || 0
|
|
|
|
+ const nonProdTime = parseFloat(row.nonProductionTime) || 0
|
|
|
|
+
|
|
|
|
+ // 新增:检查三个字段是否有空值
|
|
|
|
+ const hasEmptyField =
|
|
|
|
+ row.ratedProductionTime === null || row.ratedProductionTime === undefined || row.ratedProductionTime === '' ||
|
|
|
|
+ row.productionTime === null || row.productionTime === undefined || row.productionTime === '' ||
|
|
|
|
+ row.nonProductionTime === null || row.nonProductionTime === undefined || row.nonProductionTime === ''
|
|
|
|
+
|
|
|
|
+ // 如果有空字段,应用警告样式
|
|
|
|
+ if (hasEmptyField) {
|
|
|
|
+ return {
|
|
|
|
+ backgroundColor: '#fff2f0', // 浅红色背景,表示数据不完整
|
|
|
|
+ color: '#d46b08', // 深红色文字
|
|
|
|
+ fontWeight: 'bold',
|
|
|
|
+ border: '1px solid #ffa39e' // 可选:添加边框突出显示
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // 如果三个字段都有值,且不满足公式:额定生产时间 = 生产时间 + 非生产时间
|
|
|
|
+ // 使用容差比较,避免浮点数精度问题
|
|
|
|
+ if (Math.abs(ratedTime - (prodTime + nonProdTime)) > 0.01) {
|
|
|
|
+ return {
|
|
|
|
+ backgroundColor: '#fffbe6', // 浅黄色背景
|
|
|
|
+ color: '#d46b08', // 橙色文字
|
|
|
|
+ fontWeight: 'bold'
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // 2. 处理运行时效字段的颜色
|
|
|
|
+ if (column.property === 'transitTime') {
|
|
|
|
+ const transitTime = row.transitTime;
|
|
|
|
+ // 将运行时效转为数字(处理原逻辑中 toFixed(4) 生成的字符串)
|
|
|
|
+ const transitTimeNum = parseFloat(transitTime);
|
|
|
|
+ if (
|
|
|
|
+ transitTime === null ||
|
|
|
|
+ transitTime === undefined ||
|
|
|
|
+ isNaN(transitTimeNum) ||
|
|
|
|
+ transitTimeNum === 0 ||
|
|
|
|
+ transitTimeNum >= 1
|
|
|
|
+ ) {
|
|
|
|
+ return {
|
|
|
|
+ backgroundColor: '#fff2f0', // 浅红色背景(与数据不完整警告样式统一)
|
|
|
|
+ color: '#ff4d4f', // 红色文字
|
|
|
|
+ fontWeight: 'bold'
|
|
|
|
+ };
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ // 默认返回空对象,不应用特殊样式
|
|
|
|
+ return {};
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+// 计算文本宽度
|
|
|
|
+const getTextWidth = (text: string, fontSize = 14) => {
|
|
|
|
+ const span = document.createElement('span');
|
|
|
|
+ span.style.visibility = 'hidden';
|
|
|
|
+ span.style.position = 'absolute';
|
|
|
|
+ span.style.whiteSpace = 'nowrap';
|
|
|
|
+ span.style.fontSize = `${fontSize}px`;
|
|
|
|
+ span.style.fontFamily = 'inherit';
|
|
|
|
+ span.innerText = text;
|
|
|
|
+
|
|
|
|
+ document.body.appendChild(span);
|
|
|
|
+ const width = span.offsetWidth;
|
|
|
|
+ document.body.removeChild(span);
|
|
|
|
+
|
|
|
|
+ return width;
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+// 可伸缩列配置
|
|
|
|
+const FLEXIBLE_COLUMNS = ['deptName', 'contractName', 'taskName', 'technique', 'rigStatus', 'latestWellDoneTime', 'designWellDepth',
|
|
|
|
+ 'currentDepth', 'dailyFootage', 'monthlyFootage', 'annualFootage', 'totalConstructionWells',
|
|
|
|
+ 'completedWells', 'mudDensity', 'mudViscosity', 'constructionStartDate', 'ryNptReason',
|
|
|
|
+ 'constructionEndDate', 'lateralLength', 'wellInclination', 'casingPipeSize',
|
|
|
|
+ 'azimuth', 'ratedProductionTime', 'productionTime', 'nonProductionTime', 'createTime'];
|
|
|
|
+
|
|
|
|
+/** 查询列表 */
|
|
|
|
+const getList = async () => {
|
|
|
|
+ loading.value = true
|
|
|
|
+ try {
|
|
|
|
+ const data = await IotRyDailyReportApi.getIotRyDailyReportPage(queryParams)
|
|
|
|
+ // 计算运行时效
|
|
|
|
+ data.list.forEach((item: any) => {
|
|
|
|
+ const ratedTime = parseFloat(item.ratedProductionTime) || 0
|
|
|
|
+ const productionTime = parseFloat(item.productionTime) || 0
|
|
|
|
+
|
|
|
|
+ if (ratedTime > 0 && !isNaN(productionTime)) {
|
|
|
|
+ // 计算运行时效并保留4位小数用于计算
|
|
|
|
+ item.transitTime = (productionTime / ratedTime).toFixed(4)
|
|
|
|
+ } else {
|
|
|
|
+ item.transitTime = null
|
|
|
|
+ }
|
|
|
|
+ })
|
|
|
|
+ list.value = data.list
|
|
|
|
+ total.value = data.total
|
|
|
|
+ // 获取数据后计算列宽
|
|
|
|
+ nextTick(() => {
|
|
|
|
+ calculateColumnWidths();
|
|
|
|
+ });
|
|
|
|
+ } finally {
|
|
|
|
+ loading.value = false
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+// 计算列宽度
|
|
|
|
+const calculateColumnWidths = () => {
|
|
|
|
+ const MIN_WIDTH = 80; // 最小列宽
|
|
|
|
+ const PADDING = 25; // 列内边距
|
|
|
|
+
|
|
|
|
+ // 确保表格容器存在
|
|
|
|
+ if (!tableContainerRef.value?.$el) return;
|
|
|
|
+
|
|
|
|
+ const container = tableContainerRef.value.$el;
|
|
|
|
+ const containerWidth = container.clientWidth;
|
|
|
|
+
|
|
|
|
+ // 1. 计算所有列的最小宽度
|
|
|
|
+ const minWidths: Record<string, number> = {};
|
|
|
|
+ let totalMinWidth = 0;
|
|
|
|
+
|
|
|
|
+ // 计算列最小宽度的函数
|
|
|
|
+ const calculateColumnMinWidth = (key: string, label: string, getValue: Function) => {
|
|
|
|
+ const headerWidth = getTextWidth(label) * 1.2;
|
|
|
|
+ let contentMaxWidth = 0;
|
|
|
|
+
|
|
|
|
+ // 计算内容最大宽度
|
|
|
|
+ list.value.forEach((row, index) => {
|
|
|
|
+ let text = '';
|
|
|
|
+ if (key === 'rigStatus') {
|
|
|
|
+ // 特殊处理字典列
|
|
|
|
+ const dictValue = row[key];
|
|
|
|
+ // 这里需要根据实际情况获取字典标签,简化处理使用值本身
|
|
|
|
+ text = String(dictValue || '');
|
|
|
|
+ } else if (key.includes('Date') || key === 'createTime') {
|
|
|
|
+ // 日期列使用格式化后的值
|
|
|
|
+ text = dateFormatter(null, null, row[key]) || '';
|
|
|
|
+ } else {
|
|
|
|
+ text = String(getValue ? getValue(row, index) : (row[key] || ''));
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ const textWidth = getTextWidth(text);
|
|
|
|
+ if (textWidth > contentMaxWidth) contentMaxWidth = textWidth;
|
|
|
|
+ });
|
|
|
|
+
|
|
|
|
+ const minWidth = Math.max(headerWidth, contentMaxWidth, MIN_WIDTH) + PADDING;
|
|
|
|
+ minWidths[key] = minWidth;
|
|
|
|
+ totalMinWidth += minWidth;
|
|
|
|
+ return minWidth;
|
|
|
|
+ };
|
|
|
|
+
|
|
|
|
+ // 计算各列最小宽度
|
|
|
|
+ calculateColumnMinWidth('deptName', '施工队伍', (row: any) => row.deptName);
|
|
|
|
+ calculateColumnMinWidth('contractName', '项目', (row: any) => row.contractName);
|
|
|
|
+ calculateColumnMinWidth('taskName', '任务', (row: any) => row.taskName);
|
|
|
|
+ calculateColumnMinWidth('technique', '施工工艺', (row: any) => row.technique);
|
|
|
|
+ calculateColumnMinWidth('rigStatus', t('project.status'), (row: any) => {
|
|
|
|
+ // 这里可以获取字典标签,简化处理使用值本身
|
|
|
|
+ return String(row.rigStatus || '');
|
|
|
|
+ });
|
|
|
|
+ calculateColumnMinWidth('latestWellDoneTime', '上井次完井时间', (row: any) => row.latestWellDoneTime);
|
|
|
|
+ calculateColumnMinWidth('designWellDepth', '设计井深(m)', (row: any) => row.designWellDepth);
|
|
|
|
+ calculateColumnMinWidth('currentDepth', '当前井深(m)', (row: any) => row.currentDepth);
|
|
|
|
+ calculateColumnMinWidth('dailyFootage', '日进尺(m)', (row: any) => row.dailyFootage);
|
|
|
|
+ calculateColumnMinWidth('monthlyFootage', '月进尺(m)', (row: any) => row.monthlyFootage);
|
|
|
|
+ calculateColumnMinWidth('annualFootage', '年累计进尺(m)', (row: any) => row.annualFootage);
|
|
|
|
+ calculateColumnMinWidth('totalConstructionWells', '总施工井数', (row: any) => row.totalConstructionWells);
|
|
|
|
+ calculateColumnMinWidth('completedWells', '完工井数', (row: any) => row.completedWells);
|
|
|
|
+ calculateColumnMinWidth('casingPipeSize', '套生段产管尺寸', (row: any) => row.casingPipeSize);
|
|
|
|
+ calculateColumnMinWidth('mudDensity', '泥浆性能-密度(g/cm³)', (row: any) => row.mudDensity);
|
|
|
|
+ calculateColumnMinWidth('mudViscosity', '泥浆性能-粘度(S)', (row: any) => row.mudViscosity);
|
|
|
|
+ calculateColumnMinWidth('ryNptReason', t('project.nptReason'), (row: any) => row.ryNptReason);
|
|
|
|
+ calculateColumnMinWidth('constructionStartDate', '施工开始日期', (row: any) => dateFormatter(null, null, row.constructionStartDate));
|
|
|
|
+ calculateColumnMinWidth('constructionEndDate', '施工结束日期', (row: any) => dateFormatter(null, null, row.constructionEndDate));
|
|
|
|
+ calculateColumnMinWidth('lateralLength', '水平段长度(m)', (row: any) => row.lateralLength);
|
|
|
|
+ calculateColumnMinWidth('wellInclination', '井斜(°)', (row: any) => row.wellInclination);
|
|
|
|
+ calculateColumnMinWidth('azimuth', '方位(°)', (row: any) => row.azimuth);
|
|
|
|
+ calculateColumnMinWidth('designWellStruct', '设计井身结构', (row: any) => row.designWellStruct);
|
|
|
|
+ calculateColumnMinWidth('productionStatus', '生产动态', (row: any) => row.productionStatus);
|
|
|
|
+ calculateColumnMinWidth('ratedProductionTime', '额定生产时间(H)', (row: any) => row.ratedProductionTime);
|
|
|
|
+ calculateColumnMinWidth('productionTime', '生产时间(H)', (row: any) => row.productionTime);
|
|
|
|
+ calculateColumnMinWidth('nonProductionTime', '非生产时间(H)', (row: any) => row.nonProductionTime);
|
|
|
|
+ calculateColumnMinWidth('createTime', '创建时间', (row: any) => dateFormatter(null, null, row.createTime));
|
|
|
|
+
|
|
|
|
+ // 设计井身结构 生产动态 列使用固定宽度,不参与自动计算
|
|
|
|
+ minWidths.designWellStruct = 200; // 固定宽度200px
|
|
|
|
+ totalMinWidth += 200;
|
|
|
|
+
|
|
|
|
+ minWidths.productionStatus = 200; // 固定宽度200px
|
|
|
|
+ totalMinWidth += 200;
|
|
|
|
+
|
|
|
|
+ // 操作列固定宽度
|
|
|
|
+ minWidths.operation = 120;
|
|
|
|
+ totalMinWidth += 120;
|
|
|
|
+
|
|
|
|
+ // 2. 计算可伸缩列最终宽度
|
|
|
|
+ const newWidths: Record<string, string> = {};
|
|
|
|
+ const availableWidth = containerWidth - 17; // 减去滚动条宽度
|
|
|
|
+
|
|
|
|
+ // 应用最小宽度到所有列
|
|
|
|
+ Object.keys(minWidths).forEach(key => {
|
|
|
|
+ newWidths[key] = `${minWidths[key]}px`;
|
|
|
|
+ });
|
|
|
|
+
|
|
|
|
+ // 计算可伸缩列需要的宽度
|
|
|
|
+ if (totalMinWidth < availableWidth) {
|
|
|
|
+ // 有剩余空间:按比例分配给可伸缩列
|
|
|
|
+ const extraSpace = availableWidth - totalMinWidth;
|
|
|
|
+ const flexibleColumnCount = FLEXIBLE_COLUMNS.length;
|
|
|
|
+ const spacePerColumn = Math.floor(extraSpace / flexibleColumnCount);
|
|
|
|
+
|
|
|
|
+ FLEXIBLE_COLUMNS.forEach(key => {
|
|
|
|
+ newWidths[key] = `${minWidths[key] + spacePerColumn}px`;
|
|
|
|
+ });
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // 3. 更新列宽配置
|
|
|
|
+ columnWidths.value = newWidths;
|
|
|
|
+
|
|
|
|
+ // 4. 触发表格重新布局
|
|
|
|
+ nextTick(() => {
|
|
|
|
+ tableRef.value?.doLayout();
|
|
|
|
+ });
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+// 百分比格式化函数
|
|
|
|
+const percentageFormatter = (row: any, column: any, cellValue: any, index: number) => {
|
|
|
|
+ if (cellValue === null || cellValue === undefined || cellValue === '') return '';
|
|
|
|
+ const value = parseFloat(cellValue)
|
|
|
|
+ if (isNaN(value)) return '-'
|
|
|
|
+ // 将小数转换为百分比,保留两位小数
|
|
|
|
+ return `${(value * 100).toFixed(2)}%`
|
|
|
|
+};
|
|
|
|
+
|
|
|
|
+/** 搜索按钮操作 */
|
|
|
|
+const handleQuery = () => {
|
|
|
|
+ queryParams.pageNo = 1
|
|
|
|
+ getList()
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/** 重置按钮操作 */
|
|
|
|
+const resetQuery = () => {
|
|
|
|
+ queryFormRef.value.resetFields()
|
|
|
|
+ handleQuery()
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/** 添加/修改操作 */
|
|
|
|
+const formRef = ref()
|
|
|
|
+const openForm = (type: string, id?: number, row?: any) => {
|
|
|
|
+ // 保存当前行数据
|
|
|
|
+ if (row) {
|
|
|
|
+ selectedRowData.value = {
|
|
|
|
+ deptName: row.deptName,
|
|
|
|
+ contractName: row.contractName,
|
|
|
|
+ taskName: row.taskName,
|
|
|
|
+ designWellDepth: row.designWellDepth,
|
|
|
|
+ designWellStruct: row.designWellStruct,
|
|
|
|
+ totalConstructionWells: row.totalConstructionWells,
|
|
|
|
+ completedWells: row.completedWells
|
|
|
|
+ }
|
|
|
|
+ } else {
|
|
|
|
+ selectedRowData.value = null
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ formRef.value.open(type, id)
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/** 删除按钮操作 */
|
|
|
|
+const handleDelete = async (id: number) => {
|
|
|
|
+ try {
|
|
|
|
+ // 删除的二次确认
|
|
|
|
+ await message.delConfirm()
|
|
|
|
+ // 发起删除
|
|
|
|
+ await IotRyDailyReportApi.deleteIotRyDailyReport(id)
|
|
|
|
+ message.success(t('common.delSuccess'))
|
|
|
|
+ // 刷新列表
|
|
|
|
+ await getList()
|
|
|
|
+ } catch {}
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/** 导出按钮操作 */
|
|
|
|
+const handleExport = async () => {
|
|
|
|
+ try {
|
|
|
|
+ // 导出的二次确认
|
|
|
|
+ await message.exportConfirm()
|
|
|
|
+ // 发起导出
|
|
|
|
+ exportLoading.value = true
|
|
|
|
+ const data = await IotRyDailyReportApi.exportIotRyDailyReport(queryParams)
|
|
|
|
+ download.excel(data, '瑞鹰日报.xls')
|
|
|
|
+ } catch {
|
|
|
|
+ } finally {
|
|
|
|
+ exportLoading.value = false
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+// 声明 ResizeObserver 实例
|
|
|
|
+let resizeObserver: ResizeObserver | null = null;
|
|
|
|
+
|
|
|
|
+/** 初始化 **/
|
|
|
|
+onMounted(() => {
|
|
|
|
+ getList()
|
|
|
|
+ // 创建 ResizeObserver 监听表格容器尺寸变化
|
|
|
|
+ if (tableContainerRef.value?.$el) {
|
|
|
|
+ resizeObserver = new ResizeObserver(() => {
|
|
|
|
+ // 使用防抖避免频繁触发
|
|
|
|
+ clearTimeout((window as any).resizeTimer);
|
|
|
|
+ (window as any).resizeTimer = setTimeout(() => {
|
|
|
|
+ calculateColumnWidths();
|
|
|
|
+ }, 100);
|
|
|
|
+ });
|
|
|
|
+ resizeObserver.observe(tableContainerRef.value.$el);
|
|
|
|
+ }
|
|
|
|
+})
|
|
|
|
+
|
|
|
|
+onUnmounted(() => {
|
|
|
|
+ // 清除 ResizeObserver
|
|
|
|
+ if (resizeObserver && tableContainerRef.value?.$el) {
|
|
|
|
+ resizeObserver.unobserve(tableContainerRef.value.$el);
|
|
|
|
+ resizeObserver = null;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ // 清除定时器
|
|
|
|
+ if ((window as any).resizeTimer) {
|
|
|
|
+ clearTimeout((window as any).resizeTimer);
|
|
|
|
+ }
|
|
|
|
+})
|
|
|
|
+
|
|
|
|
+// 监听列表数据变化重新计算列宽
|
|
|
|
+watch(list, () => {
|
|
|
|
+ nextTick(calculateColumnWidths)
|
|
|
|
+}, { deep: true })
|
|
|
|
+
|
|
|
|
+</script>
|
|
|
|
+
|
|
|
|
+<style scoped>
|
|
|
|
+/* 表格容器样式,确保水平滚动 */
|
|
|
|
+.table-container {
|
|
|
|
+ width: 100%;
|
|
|
|
+ overflow-x: auto;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* 确保表格单元格内容不换行 */
|
|
|
|
+:deep(.el-table .cell) {
|
|
|
|
+ white-space: nowrap;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* 确保表格列标题不换行 */
|
|
|
|
+:deep(.el-table th > .cell) {
|
|
|
|
+ white-space: nowrap;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* 调整表格最小宽度,确保内容完全显示 */
|
|
|
|
+:deep(.el-table) {
|
|
|
|
+ min-width: 100%;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* 强制显示所有内容,防止省略号 */
|
|
|
|
+:deep(.el-table td.el-table__cell),
|
|
|
|
+:deep(.el-table th.el-table__cell) {
|
|
|
|
+ overflow: visible !important;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+:deep(.el-table .cell) {
|
|
|
|
+ overflow: visible !important;
|
|
|
|
+ text-overflow: clip !important;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* 设计井身结构文本样式 - 多行显示并添加省略号 */
|
|
|
|
+.design-well-struct-text {
|
|
|
|
+ display: -webkit-box;
|
|
|
|
+ -webkit-line-clamp: 2;
|
|
|
|
+ -webkit-box-orient: vertical;
|
|
|
|
+ overflow: hidden;
|
|
|
|
+ text-overflow: ellipsis;
|
|
|
|
+ line-height: 1.5;
|
|
|
|
+ max-height: 3em; /* 两行文本的高度 */
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* 确保设计井身结构列不参与自动调整 */
|
|
|
|
+:deep(.el-table__header-wrapper .el-table__cell.fixed-width),
|
|
|
|
+:deep(.el-table__body-wrapper .el-table__cell.fixed-width) {
|
|
|
|
+ flex-shrink: 0;
|
|
|
|
+ flex-grow: 0;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+/* 颜色说明区域样式 */
|
|
|
|
+.color-legend {
|
|
|
|
+ display: flex;
|
|
|
|
+ flex-direction: column;
|
|
|
|
+ gap: 8px;
|
|
|
|
+ padding: 12px 16px;
|
|
|
|
+ background-color: #f8f9fa;
|
|
|
|
+ border-radius: 4px;
|
|
|
|
+ border-left: 4px solid #e6f7ff;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+.legend-item {
|
|
|
|
+ display: flex;
|
|
|
|
+ align-items: center;
|
|
|
|
+ gap: 8px;
|
|
|
|
+ font-size: 14px;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+.color-indicator {
|
|
|
|
+ display: inline-block;
|
|
|
|
+ width: 12px;
|
|
|
|
+ height: 12px;
|
|
|
|
+ border-radius: 50%;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+.color-indicator.red {
|
|
|
|
+ background-color: red;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+.color-indicator.orange {
|
|
|
|
+ background-color: orange;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+</style>
|
|
|
|
+
|
|
|
|
+<style>
|
|
|
|
+/* 设计井身结构 tooltip 样式 - 保留换行符 */
|
|
|
|
+.design-well-struct-tooltip {
|
|
|
|
+ white-space: pre-line;
|
|
|
|
+ max-width: 500px;
|
|
|
|
+ line-height: 1.5;
|
|
|
|
+}
|
|
|
|
+</style>
|