|
|
<template>
|
|
|
<div ref="detailPageRef" class="device-ledger-detail-page">
|
|
|
<ContentWrap>
|
|
|
<div v-loading="detailLoading" class="device-ledger-detail-body">
|
|
|
<div class="device-ledger-detail-title">设备基本信息</div>
|
|
|
<div class="device-ledger-detail-main" >
|
|
|
<el-form
|
|
|
ref="formRef"
|
|
|
:model="formData"
|
|
|
:rules="formRules"
|
|
|
label-width="110px"
|
|
|
v-loading="formLoading || fileUploading"
|
|
|
:element-loading-text="t('common.loading')"
|
|
|
>
|
|
|
<el-row :gutter="20" class="device-basic-layout">
|
|
|
<el-col :xs="24" :lg="18" class="device-basic-fields">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.deviceCode')" prop="deviceCode">
|
|
|
<el-row :gutter="12" class="device-code-row">
|
|
|
<el-col :xs="18" :sm="18" :md="18" :lg="19" :xl="20">
|
|
|
<el-input v-model="formData.deviceCode"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderDeviceCode')"
|
|
|
:disabled="Boolean(formData.isCode) || formType === 'update'" />
|
|
|
</el-col>
|
|
|
<el-col :xs="6" :sm="6" :md="6" :lg="5" :xl="4">
|
|
|
<div class="device-code-switch">
|
|
|
<el-switch v-model="formData.isCode" :disabled="formType === 'update'"
|
|
|
@change="handleCodeAutoChange" />
|
|
|
</div>
|
|
|
</el-col>
|
|
|
|
|
|
</el-row>
|
|
|
</el-form-item>
|
|
|
|
|
|
<el-row :gutter="20">
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.deviceName')" prop="deviceName" required>
|
|
|
<el-input v-model="formData.deviceName"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderDeviceName')" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.deviceType')" prop="deviceType" required>
|
|
|
<el-tree-select v-model="formData.deviceType" :data="deviceTypeTree" :props="treeSelectProps"
|
|
|
check-strictly default-expand-all value-key="id"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderDeviceType')" class="!w-full" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
</el-row>
|
|
|
|
|
|
<el-row :gutter="20">
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.deviceSpec')" prop="deviceSpec">
|
|
|
<el-input v-model="formData.deviceSpec"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderDeviceSpec')" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
<el-col v-if="isScheduledEnabled" :span="12">
|
|
|
<el-form-item
|
|
|
:label="t('EquipmentManagement.EquipmentLedger.ratedCapacity')"
|
|
|
prop="ratedCapacity"
|
|
|
:required="isScheduledEnabled"
|
|
|
>
|
|
|
<el-input-number v-model="formData.ratedCapacity" :min="0" :precision="0" controls-position="right"
|
|
|
class="!w-full" :placeholder="t('EquipmentManagement.EquipmentLedger.placeholderRatedCapacity')" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
|
|
|
</el-row>
|
|
|
|
|
|
<el-row :gutter="20">
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.isSchedueld')" prop="isScheduled">
|
|
|
<el-switch v-model="formData.isScheduled" :active-value="1" :inactive-value="0"
|
|
|
:active-text="t('EquipmentManagement.EquipmentLedger.yes')"
|
|
|
:inactive-text="t('EquipmentManagement.EquipmentLedger.no')" inline-prompt />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.productionDate')" prop="productionDate">
|
|
|
<el-date-picker v-model="formData.productionDate" type="date" value-format="YYYY-MM-DD"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderProductionDate')" class="!w-full" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
</el-row>
|
|
|
|
|
|
<el-row :gutter="20">
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.factoryEntryDate')" prop="factoryEntryDate">
|
|
|
<el-date-picker v-model="formData.factoryEntryDate" type="date" value-format="YYYY-MM-DD"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderFactoryEntryDate')" class="!w-full" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.deviceLocation')" prop="deviceLocation">
|
|
|
<el-input v-model="formData.deviceLocation"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderDeviceLocation')" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
</el-row>
|
|
|
|
|
|
<el-row :gutter="20">
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.deviceManagerName')" prop="deviceManagerIds">
|
|
|
<el-select v-model="formData.deviceManagerIds" multiple filterable clearable
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderDeviceManagerIds')" class="!w-full">
|
|
|
<el-option v-for="item in users" :key="item.id" :label="item.nickname" :value="item.id" />
|
|
|
</el-select>
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.criticalComponent')" prop="componentIds">
|
|
|
<el-input :model-value="criticalComponentDisplay" readonly clearable class="device-ledger-selection-input"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderComponentIds')"
|
|
|
@clear="clearCriticalComponent" @click="openCriticalComponentDialog" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
</el-row>
|
|
|
|
|
|
<el-row :gutter="20">
|
|
|
<el-col :span="12">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.sparePart')" prop="beijianIds">
|
|
|
<el-input :model-value="beijianDisplay" readonly clearable class="device-ledger-selection-input"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderBeijianIds')" @clear="clearBeijian" @click="openBeijianDialog"/>
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
</el-row>
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :xs="24" :lg="6" class="device-basic-media">
|
|
|
<div class="device-media-panel">
|
|
|
<el-form-item label-width="0" prop="images" class="device-media-item">
|
|
|
<div class="device-media-box device-media-upload-box">
|
|
|
<span class="device-media-label">{{ t('EquipmentManagement.EquipmentLedger.images') }}</span>
|
|
|
<UploadImg class="device-media-upload" v-model="formData.images" />
|
|
|
</div>
|
|
|
</el-form-item>
|
|
|
<el-form-item v-if="formType === 'update'" label-width="0" prop="qrcodeUrl" class="device-media-item">
|
|
|
<div class="device-media-box device-media-code-box">
|
|
|
<span class="device-media-label">{{ t('EquipmentManagement.EquipmentLedger.qrcode') }}</span>
|
|
|
<div class="device-media-code">
|
|
|
<QrcodeActionCard :image-url="formData.qrcodeUrl" :print-id="formData.id"
|
|
|
:print-title="`${formData.deviceName || 'Device'} QR Code`" :print-paper-width="80" :print-paper-height="80"
|
|
|
:print-max-width="220" :empty-text="t('EquipmentManagement.EquipmentLedger.qrcodeEmpty')"
|
|
|
:error-text="t('EquipmentManagement.EquipmentLedger.qrcodeLoadError')"
|
|
|
:refresh-url="getQrcodeRefreshUrl()" :refresh-disabled="!formData.id || !formData.deviceCode"
|
|
|
refresh-confirm-text="Refresh QR code?"
|
|
|
:template-json="formData.templateJson"
|
|
|
:print-data="buildPrintData()"
|
|
|
@refresh-success="handleQrcodeRefreshSuccess" />
|
|
|
</div>
|
|
|
</div>
|
|
|
</el-form-item>
|
|
|
</div>
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :span="24">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.fileUrl')" prop="fileUrl">
|
|
|
<UploadFile
|
|
|
:is-show-tip="false"
|
|
|
v-model="formData.fileUrl"
|
|
|
:limit="9"
|
|
|
@uploading-change="handleFileUploadingChange"
|
|
|
/>
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
|
|
|
<el-col :span="24">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentLedger.remark')" prop="remark">
|
|
|
<el-input v-model="formData.remark"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentLedger.placeholderRemark')" type="textarea" />
|
|
|
</el-form-item>
|
|
|
</el-col>
|
|
|
</el-row>
|
|
|
</el-form>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="device-ledger-detail-tabs">
|
|
|
<el-tabs v-model="detailActiveTab" class="mt-12px">
|
|
|
|
|
|
<el-tab-pane :label="t('EquipmentManagement.EquipmentLedger.criticalComponent')" name="criticalComponent">
|
|
|
<div class="device-ledger-tab-toolbar">
|
|
|
<el-button
|
|
|
type="success"
|
|
|
plain
|
|
|
:loading="criticalExportLoading"
|
|
|
@click="handleExportCriticalComponent"
|
|
|
v-hasPermi="['mes:device-ledger:export']"
|
|
|
>
|
|
|
<Icon icon="ep:download" class="mr-5px" /> {{ t('action.export') }}
|
|
|
</el-button>
|
|
|
</div>
|
|
|
<el-table v-loading="tableLoading" :data="editableCriticalComponentRows" :stripe="true" :show-overflow-tooltip="true">
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentLedger.componentCode')" align="center" prop="code" min-width="140" sortable />
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentLedger.componentName')" align="center" prop="name" min-width="140" sortable />
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentKeyItems.count')" align="center" prop="count" min-width="180">
|
|
|
<template #default="scope">
|
|
|
<el-input-number
|
|
|
v-model="scope.row.count"
|
|
|
:min="0"
|
|
|
:precision="0"
|
|
|
controls-position="right"
|
|
|
class="!w-full"
|
|
|
/>
|
|
|
</template>
|
|
|
</el-table-column>
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentLedger.componentDesc')" align="center" prop="description" min-width="180" />
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentLedger.remark')" align="center" prop="remark" min-width="180" />
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentLedger.createTime')" align="center" prop="createTime" :formatter="dateFormatter" width="180" sortable />
|
|
|
</el-table>
|
|
|
</el-tab-pane>
|
|
|
|
|
|
|
|
|
</el-tabs>
|
|
|
</div>
|
|
|
<div class="device-ledger-action-bar" :style="actionBarStyle">
|
|
|
<el-button @click="handleCancelEdit">{{ t('common.cancel') }}</el-button>
|
|
|
<el-button type="primary" :loading="formLoading" :disabled="fileUploading" @click="submitForm">{{ t('common.save') }}</el-button>
|
|
|
</div>
|
|
|
</div>
|
|
|
<el-dialog v-model="criticalComponentDialogVisible" :title="t('EquipmentManagement.EquipmentLedger.gjTitle')" width="1200px" class="device-ledger-transfer-dialog"
|
|
|
append-to-body>
|
|
|
<el-form class="-mb-15px" :model="queryParams" ref="queryFormRef" :inline="true" min-label-width="68px"
|
|
|
style="margin-bottom: 10px">
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentKeyItems.code')" prop="code">
|
|
|
<el-input v-model="queryParams.code" :placeholder="t('EquipmentManagement.EquipmentKeyItems.placeholderCode')"
|
|
|
clearable @keyup.enter="handleQuery" class="!w-240px" />
|
|
|
</el-form-item>
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentKeyItems.name')" prop="name">
|
|
|
<el-input v-model="queryParams.name" :placeholder="t('EquipmentManagement.EquipmentKeyItems.placeholderName')"
|
|
|
clearable @keyup.enter="handleQuery" class="!w-240px" />
|
|
|
</el-form-item>
|
|
|
<el-form-item :label="t('EquipmentManagement.EquipmentKeyItems.description')" prop="description">
|
|
|
<el-input v-model="queryParams.description"
|
|
|
:placeholder="t('EquipmentManagement.EquipmentKeyItems.placeholderDescription')" clearable
|
|
|
@keyup.enter="handleQuery" class="!w-240px" />
|
|
|
</el-form-item>
|
|
|
<el-form-item>
|
|
|
<el-button @click="handleQuery">
|
|
|
<Icon icon="ep:search" class="mr-5px" />
|
|
|
{{ t('common.query') }}
|
|
|
</el-button>
|
|
|
<el-button @click="resetQuery">
|
|
|
<Icon icon="ep:refresh" class="mr-5px" />
|
|
|
{{ t('common.reset') }}
|
|
|
</el-button>
|
|
|
</el-form-item>
|
|
|
</el-form>
|
|
|
<ContentWrap>
|
|
|
<el-table ref="multipleTableRef" v-loading="loading" :data="list" :stripe="true" :show-overflow-tooltip="true" class="no-select-all"
|
|
|
@selection-change="handleSelectionChange" @select="handleSelect" @select-all="handleSelectAll" row-key="id">
|
|
|
<el-table-column type="selection" width="55" :reserve-selection="true" />
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentKeyItems.code')" align="center" prop="code"
|
|
|
min-width="140" sortable />
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentKeyItems.name')" align="center" prop="name"
|
|
|
min-width="140" sortable />
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentKeyItems.description')" align="center"
|
|
|
prop="description" min-width="180" />
|
|
|
<!-- <el-table-column :label="t('EquipmentManagement.EquipmentKeyItems.count')" align="center" prop="count" min-width="180">
|
|
|
<template #default="scope">
|
|
|
<el-input-number
|
|
|
v-model="scope.row.count"
|
|
|
:min="0"
|
|
|
:precision="0"
|
|
|
controls-position="right"
|
|
|
class="!w-full"
|
|
|
/>
|
|
|
</template>
|
|
|
</el-table-column>-->
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentKeyItems.remark')" align="center" prop="remark"
|
|
|
min-width="180" />
|
|
|
<el-table-column :label="t('EquipmentManagement.EquipmentKeyItems.createTime')" align="center" prop="createTime"
|
|
|
:formatter="dateFormatter" width="180" sortable />
|
|
|
</el-table>
|
|
|
|
|
|
<Pagination :total="total" v-model:page="queryParams.pageNo" v-model:limit="queryParams.pageSize"
|
|
|
@pagination="getList" />
|
|
|
|
|
|
</ContentWrap>
|
|
|
<template #footer>
|
|
|
<el-button @click="criticalComponentDialogVisible = false">{{ t('common.cancel') }}</el-button>
|
|
|
<el-button type="primary" @click="confirmCriticalComponentDialog">{{ t('common.ok') }}</el-button>
|
|
|
</template>
|
|
|
</el-dialog>
|
|
|
|
|
|
<el-dialog v-model="beijianDialogVisible" :title="t('EquipmentManagement.EquipmentLedger.bjTitle')" width="860px" class="device-ledger-transfer-dialog"
|
|
|
append-to-body>
|
|
|
<!-- <div class="device-ledger-transfer">
|
|
|
<el-transfer
|
|
|
v-model="beijianDraft"
|
|
|
:data="beijianTransferData"
|
|
|
filterable
|
|
|
:filter-placeholder="t('EquipmentManagement.EquipmentLedger.placeholderBeijianIds')"
|
|
|
/>
|
|
|
</div>-->
|
|
|
<!-- 列表 -->
|
|
|
<el-form class="-mb-15px" :model="bjQueryParams" ref="bjQueryFormRef" :inline="true" min-label-width="68px"
|
|
|
style="margin-bottom: 10px">
|
|
|
<el-form-item :label="t('SparePartsManagement.SpareInfo.code')" prop="barCode">
|
|
|
<el-input v-model="bjQueryParams.barCode" :placeholder="t('SparePartsManagement.SpareInfo.placeholderCode')"
|
|
|
clearable @keyup.enter="handleQuery" class="!w-240px" />
|
|
|
</el-form-item>
|
|
|
<el-form-item :label="t('SparePartsManagement.SpareInfo.name')" prop="name">
|
|
|
<el-input v-model="bjQueryParams.name" :placeholder="t('SparePartsManagement.SpareInfo.placeholderName')"
|
|
|
clearable @keyup.enter="bjHandleQuery" class="!w-240px" />
|
|
|
</el-form-item>
|
|
|
<el-form-item>
|
|
|
<el-button @click="bjHandleQuery">
|
|
|
<Icon icon="ep:search" class="mr-5px" />
|
|
|
{{ t('common.query') }}
|
|
|
</el-button>
|
|
|
<el-button @click="bjResetQuery">
|
|
|
<Icon icon="ep:refresh" class="mr-5px" />
|
|
|
{{ t('common.reset') }}
|
|
|
</el-button>
|
|
|
</el-form-item>
|
|
|
</el-form>
|
|
|
<ContentWrap>
|
|
|
<el-table ref="bjMultipleTableRef" v-loading="loading" :data="bjList" :stripe="true" class="no-select-all"
|
|
|
@selection-change="bjHandleSelectionChange" @select="bjHandleSelect" @select-all="bjHandleSelectAll"
|
|
|
:show-overflow-tooltip="true" row-key="id">
|
|
|
<el-table-column type="selection" width="55" :reserve-selection="true" />
|
|
|
<el-table-column :label="t('SparePartsManagement.SpareInfo.code')" align="center" prop="barCode" width="240px"
|
|
|
sortable />
|
|
|
<el-table-column :label="t('SparePartsManagement.SpareInfo.name')" align="left" prop="name" width="220px"
|
|
|
sortable />
|
|
|
<el-table-column :label="t('SparePartsManagement.SpareInfo.unit')" align="center" prop="unitName" sortable />
|
|
|
<el-table-column :label="t('SparePartsManagement.SpareInfo.status')" align="center" prop="status" sortable>
|
|
|
<template #default="scope">
|
|
|
<dict-tag :type="DICT_TYPE.COMMON_STATUS" :value="scope.row.status" />
|
|
|
</template>
|
|
|
</el-table-column>
|
|
|
<el-table-column :label="t('SparePartsManagement.SpareInfo.createTime')" align="center" prop="createTime"
|
|
|
:formatter="dateFormatter" width="180px" sortable />
|
|
|
</el-table>
|
|
|
<!-- 分页 -->
|
|
|
<Pagination :total="bjTotal" v-model:page="bjQueryParams.pageNo" v-model:limit="bjQueryParams.pageSize"
|
|
|
@pagination="bjGetList" />
|
|
|
</ContentWrap>
|
|
|
<template #footer>
|
|
|
<el-button @click="beijianDialogVisible = false">{{ t('common.cancel') }}</el-button>
|
|
|
<el-button type="primary" @click="confirmBeijianDialog">{{ t('common.ok') }}</el-button>
|
|
|
</template>
|
|
|
</el-dialog>
|
|
|
|
|
|
</ContentWrap>
|
|
|
</div>
|
|
|
</template>
|
|
|
|
|
|
<script setup lang="ts">
|
|
|
import { dateFormatter, formatDate } from '@/utils/formatTime'
|
|
|
import download from '@/utils/download'
|
|
|
import { DeviceLedgerApi, DeviceLedgerVO } from '@/api/mes/deviceledger'
|
|
|
import { DeviceTypeApi, DeviceTypeTreeVO } from '@/api/mes/devicetype'
|
|
|
import {CriticalComponentApi, CriticalComponentVO} from '@/api/mes/criticalComponent'
|
|
|
import { getSimpleUserList, UserVO } from '@/api/system/user'
|
|
|
import {ProductApi, ProductVO} from '@/api/erp/product/product'
|
|
|
import QrcodeActionCard from '@/components/QrcodeActionCard/index.vue'
|
|
|
import { useTagsViewStore } from '@/store/modules/tagsView'
|
|
|
import { useAppStore } from '@/store/modules/app'
|
|
|
import type { FormRules } from 'element-plus'
|
|
|
import { ref } from 'vue'
|
|
|
import {ElTable} from "element-plus";
|
|
|
import { DICT_TYPE } from '@/utils/dict'
|
|
|
defineOptions({ name: 'MesDeviceLedgerEditDetail' })
|
|
|
|
|
|
const { t } = useI18n()
|
|
|
const message = useMessage()
|
|
|
const route = useRoute()
|
|
|
const { delView } = useTagsViewStore()
|
|
|
const { currentRoute } = useRouter()
|
|
|
const appStore = useAppStore()
|
|
|
const deviceId = computed(() => Number(route.params.id))
|
|
|
const detailLoading = ref(false)
|
|
|
const tableLoading = ref(false)
|
|
|
const detailData = ref<DeviceLedgerVO | undefined>()
|
|
|
const detailActiveTab = ref('criticalComponent')
|
|
|
const detailPageRef = ref<HTMLElement>()
|
|
|
const actionBarStyle = reactive({
|
|
|
left: '0px',
|
|
|
width: '100%'
|
|
|
})
|
|
|
let actionBarObserver: ResizeObserver | undefined
|
|
|
|
|
|
const updateActionBarRect = () => {
|
|
|
const rect = detailPageRef.value?.getBoundingClientRect()
|
|
|
if (!rect) return
|
|
|
actionBarStyle.left = rect.left + 'px'
|
|
|
actionBarStyle.width = rect.width + 'px'
|
|
|
}
|
|
|
|
|
|
const updateActionBarRectAfterLayout = () => {
|
|
|
nextTick(updateActionBarRect)
|
|
|
window.setTimeout(updateActionBarRect, 320)
|
|
|
}
|
|
|
const criticalComponentDialogVisible = ref(false)
|
|
|
const beijianDialogVisible = ref(false)
|
|
|
const criticalComponentDraft = ref<number[]>([])
|
|
|
const beijianDraft = ref<number[]>([])
|
|
|
const list = ref<CriticalComponentVO[]>([])
|
|
|
const bjList = ref<ProductVO[]>([])
|
|
|
const loading = ref(true)
|
|
|
const total = ref(0)
|
|
|
const bjTotal = ref(0)
|
|
|
const selectedIds = ref<number[]>([])
|
|
|
const bjSelectedIds = ref<number[]>([])
|
|
|
// 表格引用
|
|
|
const multipleTableRef = ref<InstanceType<typeof ElTable>>()
|
|
|
const bjMultipleTableRef = ref<InstanceType<typeof ElTable>>()
|
|
|
const queryFormRef = ref()
|
|
|
const bjQueryFormRef = ref()
|
|
|
const parseIdsValue = (value: any): number[] => {
|
|
|
if (!value) return []
|
|
|
if (Array.isArray(value)) return value.map((v) => Number(v)).filter((v) => !Number.isNaN(v))
|
|
|
return String(value)
|
|
|
.split(',')
|
|
|
.map((v) => Number(v.trim()))
|
|
|
.filter((v) => !Number.isNaN(v))
|
|
|
}
|
|
|
const selectedRows = ref<any[]>([]) // 存储所有选中的行
|
|
|
const bjSelectedRows = ref<any[]>([]) // 存储所有选中的行
|
|
|
const queryParams = reactive({
|
|
|
pageNo: 1,
|
|
|
pageSize: 10,
|
|
|
code: undefined as string | undefined,
|
|
|
name: undefined as string | undefined,
|
|
|
description: undefined as string | undefined,
|
|
|
remark: undefined as string | undefined,
|
|
|
createTime: [] as string[]
|
|
|
})
|
|
|
|
|
|
const bjQueryParams = reactive({
|
|
|
pageNo: 1,
|
|
|
pageSize: 10,
|
|
|
barCode: undefined as string | undefined,
|
|
|
name: undefined as string | undefined,
|
|
|
categoryId: undefined as number | undefined
|
|
|
})
|
|
|
|
|
|
const handleQuery = () => {
|
|
|
queryParams.pageNo = 1
|
|
|
getList()
|
|
|
}
|
|
|
const resetQuery = () => {
|
|
|
queryFormRef.value.resetFields()
|
|
|
handleQuery()
|
|
|
}
|
|
|
|
|
|
/** 配件搜索按钮操作 */
|
|
|
const bjHandleQuery = () => {
|
|
|
bjQueryParams.pageNo = 1
|
|
|
bjGetList()
|
|
|
}
|
|
|
|
|
|
/** 配件重置按钮操作 */
|
|
|
const bjResetQuery = () => {
|
|
|
bjQueryFormRef.value.resetFields()
|
|
|
bjHandleQuery()
|
|
|
}
|
|
|
|
|
|
|
|
|
const handleSelectionChange = (rows: CriticalComponentVO[]) => {
|
|
|
selectedIds.value = rows.map((r) => r.id).filter((id): id is number => typeof id === 'number')
|
|
|
// 获取当前页所有行<E69C89>?id
|
|
|
const currentPageIds = rows.map(item => item.id)
|
|
|
|
|
|
// 从已选中的数组中移除当前页的数据
|
|
|
selectedRows.value = selectedRows.value.filter(
|
|
|
item => !currentPageIds.includes(item.id)
|
|
|
)
|
|
|
// 添加当前页新选中的数<E79A84>? selectedRows.value.push(...rows)
|
|
|
}
|
|
|
|
|
|
// 存储当前已选中的行
|
|
|
|
|
|
const handleSelect = (selection: any[], row: any) => {
|
|
|
// 判断是选中还是取消选中
|
|
|
const isSelected = selection.includes(row)
|
|
|
if (isSelected) {
|
|
|
// console.log(`<60>?行被选中: ID=${row.id}, Name=${row.name}`)
|
|
|
ids.value.push(row.id)
|
|
|
} else {
|
|
|
|
|
|
ids.value = ids.value.filter(
|
|
|
item => item !== row.id
|
|
|
)
|
|
|
// console.log(`<60>?行被取消选中: ID=${row.id}, Name=${row.name}`)
|
|
|
}
|
|
|
// 更新当前选中状<E4B8AD>? currentSelectedRows.value = selection
|
|
|
}
|
|
|
|
|
|
|
|
|
const handleSelectAll = (selection) => {
|
|
|
ids.value = selection?.map((row) => row.id).filter((id) => id !== undefined) ?? []
|
|
|
|
|
|
/* let newVar = selection?.map((row) => row.id).filter((id) => id !== undefined) ?? [];
|
|
|
newVar.forEach(row => {
|
|
|
ids.value.push(row)
|
|
|
})*/
|
|
|
}
|
|
|
const confirmCriticalComponentDialog = () => {
|
|
|
//let ids = selectedRows.value.map(item => item.id);
|
|
|
//const validMap = new Set(criticalComponentOptions.value.map((item) => item.value))
|
|
|
//const selected = Array.from(new Set(criticalComponentDraft.value.map((v) => Number(v)).filter((v) => validMap.has(v))))
|
|
|
formData.value.componentIds = filterValidSelectedIds(ids.value, criticalComponentOptions.value)
|
|
|
criticalComponentDialogVisible.value = false
|
|
|
syncCriticalComponentRows()
|
|
|
//multipleTableRef.value.clearSelection()
|
|
|
}
|
|
|
const filterValidSelectedIds = (selectedIds: any[], options: SelectionOption[]) => {
|
|
|
const validIds = new Set(options.map((item) => item.value))
|
|
|
return Array.from(
|
|
|
new Set(
|
|
|
(selectedIds ?? [])
|
|
|
.map((id) => normalizeNumberish(id))
|
|
|
.filter((id): id is number => id !== undefined && validIds.has(id))
|
|
|
)
|
|
|
)
|
|
|
}
|
|
|
const confirmBeijianDialog = () => {
|
|
|
// let ids = bjSelectedRows.value.map(item => item.id);
|
|
|
/* const validMap = new Set(beijianOptions.value.map((item) => item.value))
|
|
|
const selected = Array.from(new Set(beijianDraft.value.map((v) => Number(v)).filter((v) => validMap.has(v))))*/
|
|
|
formData.value.beijianIds = filterValidSelectedIds(bjIds.value, beijianOptions.value)
|
|
|
beijianDialogVisible.value = false
|
|
|
//multipleTableRef.value.clearSelection()
|
|
|
}
|
|
|
|
|
|
const bjHandleSelectionChange = (rows: CriticalComponentVO[]) => {
|
|
|
bjSelectedIds.value = rows.map((r) => r.id).filter((id): id is number => typeof id === 'number')
|
|
|
// 获取当前页所有行<E69C89>?id
|
|
|
const currentPageIds = rows.map(item => item.id)
|
|
|
|
|
|
// 从已选中的数组中移除当前页的数据
|
|
|
bjSelectedRows.value = bjSelectedRows.value.filter(
|
|
|
item => !currentPageIds.includes(item.id)
|
|
|
)
|
|
|
// 添加当前页新选中的数<E79A84>? bjSelectedRows.value.push(...rows)
|
|
|
}
|
|
|
|
|
|
// 存储当前已选中的行
|
|
|
|
|
|
const bjHandleSelect = (selection: any[], row: any) => {
|
|
|
// 判断是选中还是取消选中
|
|
|
const isSelected = selection.includes(row)
|
|
|
if (isSelected) {
|
|
|
// console.log(`<60>?行被选中: ID=${row.id}, Name=${row.name}`)
|
|
|
bjIds.value.push(row.id)
|
|
|
} else {
|
|
|
bjIds.value = bjIds.value.filter(
|
|
|
item => item !== row.id
|
|
|
)
|
|
|
// console.log(`<60>?行被取消选中: ID=${row.id}, Name=${row.name}`)
|
|
|
}
|
|
|
// 更新当前选中状<E4B8AD>? bjCurrentSelectedRows.value = selection
|
|
|
}
|
|
|
|
|
|
|
|
|
const bjHandleSelectAll = (selection) => {
|
|
|
bjIds.value = selection?.map((row) => row.id).filter((id) => id !== undefined) ?? []
|
|
|
}
|
|
|
|
|
|
|
|
|
const getList = async () => {
|
|
|
loading.value = true
|
|
|
try {
|
|
|
const data = await CriticalComponentApi.getCriticalComponentPage(queryParams)
|
|
|
list.value = data.list
|
|
|
total.value = data.total
|
|
|
nextTick(() => {
|
|
|
toggleSelection()
|
|
|
})
|
|
|
} finally {
|
|
|
loading.value = false
|
|
|
}
|
|
|
}
|
|
|
|
|
|
/** 配件查询列表 */
|
|
|
const bjGetList = async () => {
|
|
|
loading.value = true
|
|
|
try {
|
|
|
bjQueryParams.categoryId = 5
|
|
|
const data = await ProductApi.getProductPage(bjQueryParams)
|
|
|
bjList.value = data.list
|
|
|
bjTotal.value = data.total
|
|
|
nextTick(() => {
|
|
|
bjToggleSelection()
|
|
|
})
|
|
|
} finally {
|
|
|
loading.value = false
|
|
|
}
|
|
|
}
|
|
|
const toggleSelection = () => {
|
|
|
if (!multipleTableRef.value || !selectedRows.value.length) return
|
|
|
|
|
|
// 遍历当前页的数据
|
|
|
list.value.forEach(row => {
|
|
|
const isSelected = selectedRows.value.some(item => item.id === row.id)
|
|
|
|
|
|
if (isSelected) {
|
|
|
// 如果应该选中,就选中
|
|
|
multipleTableRef.value!.toggleRowSelection(row, true)
|
|
|
} else {
|
|
|
// 否则取消选中
|
|
|
multipleTableRef.value!.toggleRowSelection(row, false)
|
|
|
}
|
|
|
})
|
|
|
}
|
|
|
|
|
|
const bjToggleSelection = () => {
|
|
|
if (!bjMultipleTableRef.value || !bjSelectedRows.value.length) return
|
|
|
|
|
|
// 遍历当前页的数据
|
|
|
bjList.value.forEach(row => {
|
|
|
const isSelected = bjSelectedRows.value.some(item => item.id === row.id)
|
|
|
|
|
|
if (isSelected) {
|
|
|
// 如果应该选中,就选中
|
|
|
bjMultipleTableRef.value!.toggleRowSelection(row, true)
|
|
|
} else {
|
|
|
// 否则取消选中
|
|
|
bjMultipleTableRef.value!.toggleRowSelection(row, false)
|
|
|
}
|
|
|
})
|
|
|
}
|
|
|
|
|
|
const ids = ref<number[]>([])
|
|
|
const bjIds = ref<number[]>([])
|
|
|
const openCriticalComponentDialog = () => {
|
|
|
criticalComponentDraft.value = [...(formData.value.componentIds ?? [])]
|
|
|
criticalComponentDialogVisible.value = true
|
|
|
ids.value = [...(formData.value.componentIds ?? [])]
|
|
|
setDefaultSelections()
|
|
|
}
|
|
|
const openBeijianDialog = () => {
|
|
|
beijianDraft.value = [...(formData.value.beijianIds ?? [])]
|
|
|
beijianDialogVisible.value = true
|
|
|
bjIds.value = [...(formData.value.beijianIds ?? [])]
|
|
|
setBJDefaultSelections()
|
|
|
}
|
|
|
|
|
|
// 设置默认选中的行
|
|
|
const setDefaultSelections = () => {
|
|
|
// 等待DOM更新完成
|
|
|
nextTick(() => {
|
|
|
if (!multipleTableRef.value) return
|
|
|
multipleTableRef.value.clearSelection()
|
|
|
const rawSubjectIds = toRaw(formData.value.componentIds)
|
|
|
if (rawSubjectIds.length != 0) {
|
|
|
let row = {
|
|
|
id: undefined
|
|
|
}
|
|
|
multipleTableRef.value!.toggleRowSelection(row, true)
|
|
|
}
|
|
|
// 遍历数据,找到需要选中的行
|
|
|
list.value.forEach(row => {
|
|
|
if (rawSubjectIds.includes(row.id)) {
|
|
|
multipleTableRef.value!.toggleRowSelection(row, true)
|
|
|
}
|
|
|
})
|
|
|
})
|
|
|
}
|
|
|
|
|
|
const normalizeNumberish = (value: any): number | undefined => {
|
|
|
if (value === null || value === undefined || value === '') return undefined
|
|
|
if (typeof value === 'number') return Number.isFinite(value) ? value : undefined
|
|
|
if (typeof value === 'string') {
|
|
|
const trimmed = value.trim()
|
|
|
if (!trimmed) return undefined
|
|
|
const n = Number(trimmed)
|
|
|
return Number.isFinite(n) ? n : undefined
|
|
|
}
|
|
|
return undefined
|
|
|
}
|
|
|
|
|
|
const normalizeYmd = (value: any): string | undefined => {
|
|
|
if (value === null || value === undefined || value === '') return undefined
|
|
|
if (typeof value === 'string') {
|
|
|
const trimmed = value.trim()
|
|
|
const matched = trimmed.match(/^(\d{4}-\d{2}-\d{2})/)
|
|
|
if (matched?.[1]) return matched[1]
|
|
|
const parsed = Date.parse(trimmed)
|
|
|
if (!Number.isNaN(parsed)) return formatDate(new Date(parsed), 'YYYY-MM-DD')
|
|
|
return trimmed
|
|
|
}
|
|
|
if (typeof value === 'number') return formatDate(new Date(value), 'YYYY-MM-DD')
|
|
|
if (value instanceof Date) return formatDate(value, 'YYYY-MM-DD')
|
|
|
return formatDate(new Date(value), 'YYYY-MM-DD')
|
|
|
}
|
|
|
|
|
|
const initFormData = () => ({
|
|
|
id: undefined,
|
|
|
images: undefined,
|
|
|
deviceCode: undefined,
|
|
|
isCode: true,
|
|
|
deviceName: undefined,
|
|
|
deviceStatus: undefined,
|
|
|
deviceBrand: undefined,
|
|
|
deviceModel: undefined,
|
|
|
deviceSpec: undefined,
|
|
|
isScheduled: 0,
|
|
|
ratedCapacity: undefined,
|
|
|
deviceType: undefined as number | undefined,
|
|
|
deviceLine: undefined as number | undefined,
|
|
|
supplier: undefined,
|
|
|
workshop: undefined,
|
|
|
deviceLocation: undefined,
|
|
|
systemOrg: undefined,
|
|
|
deviceManagerIds: [] as number[],
|
|
|
productionDate: undefined,
|
|
|
factoryEntryDate: undefined,
|
|
|
remark: undefined,
|
|
|
componentIds: [] as number[],
|
|
|
beijianIds: [] as number[],
|
|
|
fileUrl: '',
|
|
|
qrcodeUrl: undefined,
|
|
|
templateJson: undefined,
|
|
|
sort: undefined,
|
|
|
dvId: undefined
|
|
|
})
|
|
|
|
|
|
const formLoading = ref(false)
|
|
|
const fileUploading = ref(false)
|
|
|
const formType = ref('update')
|
|
|
const formRef = ref()
|
|
|
const formData = ref<any>({
|
|
|
...initFormData()
|
|
|
})
|
|
|
const isScheduledEnabled = computed(() => Number(formData.value.isScheduled) === 1)
|
|
|
const validateDeviceCode = (_rule, value, callback) => {
|
|
|
if (Boolean(formData.value.isCode)) {
|
|
|
callback()
|
|
|
return
|
|
|
}
|
|
|
if (value === undefined || value === null || String(value).trim() === '') {
|
|
|
callback(new Error(t('EquipmentManagement.EquipmentLedger.validatorDeviceCodeRequired')))
|
|
|
return
|
|
|
}
|
|
|
callback()
|
|
|
}
|
|
|
const validateScheduledRequired = (label: string) => (_rule, value, callback) => {
|
|
|
if (!isScheduledEnabled.value) {
|
|
|
callback()
|
|
|
return
|
|
|
}
|
|
|
const normalized = normalizeNumberish(value)
|
|
|
if (normalized === undefined) {
|
|
|
callback(new Error(String(label)))
|
|
|
return
|
|
|
}
|
|
|
callback()
|
|
|
}
|
|
|
const formRules = reactive<FormRules>({
|
|
|
deviceCode: [{ validator: validateDeviceCode, trigger: ['blur', 'change'] }],
|
|
|
deviceName: [{ required: true, message: t('EquipmentManagement.EquipmentLedger.placeholderDeviceName'), trigger: 'blur' }],
|
|
|
deviceType: [{ required: true, message: t('EquipmentManagement.EquipmentLedger.placeholderDeviceType'), trigger: 'change' }],
|
|
|
ratedCapacity: [{ validator: validateScheduledRequired('ratedCapacity'), trigger: ['blur', 'change'] }]
|
|
|
})
|
|
|
const treeSelectProps = { label: 'name', children: 'children' }
|
|
|
const deviceTypeTree = ref<DeviceTypeTreeVO[]>([])
|
|
|
const users = ref<UserVO[]>([])
|
|
|
type SelectionOption = { label: string; value: number }
|
|
|
const criticalComponentOptions = ref<SelectionOption[]>([])
|
|
|
const beijianOptions = ref<SelectionOption[]>([])
|
|
|
const savedCriticalComponentOptions = ref<SelectionOption[]>([])
|
|
|
const savedBeijianOptions = ref<SelectionOption[]>([])
|
|
|
const editableCriticalComponentRows = ref<any[]>([])
|
|
|
|
|
|
const buildCriticalComponentOptions = (items: any[] = []): SelectionOption[] =>
|
|
|
(items ?? [])
|
|
|
.map((item: any) => {
|
|
|
const id = normalizeNumberish(item?.id)
|
|
|
if (id === undefined) return undefined
|
|
|
const code = item.code ? String(item.code) : ''
|
|
|
const name = item.name ? String(item.name) : ''
|
|
|
const label = code && name ? `${code}-${name}` : name || code || String(id)
|
|
|
return { label, value: id }
|
|
|
})
|
|
|
.filter((item): item is SelectionOption => Boolean(item))
|
|
|
|
|
|
const buildBeijianOptions = (items: any[] = []): SelectionOption[] =>
|
|
|
(items ?? [])
|
|
|
.map((item: any) => {
|
|
|
const id = normalizeNumberish(item?.id)
|
|
|
if (id === undefined) return undefined
|
|
|
const code = item.barCode ? String(item.barCode) : ''
|
|
|
const name = item.name ? String(item.name) : ''
|
|
|
const label = code && name ? `${code}-${name}` : name || code || String(id)
|
|
|
return { label, value: id }
|
|
|
})
|
|
|
.filter((item): item is SelectionOption => Boolean(item))
|
|
|
|
|
|
const mergeSelectionOptions = (...groups: SelectionOption[][]): SelectionOption[] => {
|
|
|
const optionMap = new Map<number, SelectionOption>()
|
|
|
groups.flat().forEach((item) => optionMap.set(item.value, item))
|
|
|
return Array.from(optionMap.values())
|
|
|
}
|
|
|
|
|
|
const formatSelectedSummary = (ids: number[], options: SelectionOption[]) => {
|
|
|
const optionMap = new Map<number, string>(options.map((item) => [item.value, item.label]))
|
|
|
const labels = ids.map((id) => optionMap.get(id)).filter((v): v is string => Boolean(v))
|
|
|
if (!labels.length) return ''
|
|
|
if (labels.length <= 3) return labels.join(', ')
|
|
|
return `${labels.slice(0, 3).join(', ')}...${labels.length}`
|
|
|
}
|
|
|
|
|
|
const criticalComponentDisplay = computed(() =>
|
|
|
formatSelectedSummary(
|
|
|
formData.value.componentIds ?? [],
|
|
|
mergeSelectionOptions(criticalComponentOptions.value, savedCriticalComponentOptions.value)
|
|
|
)
|
|
|
)
|
|
|
const beijianDisplay = computed(() =>
|
|
|
formatSelectedSummary(formData.value.beijianIds ?? [], mergeSelectionOptions(beijianOptions.value, savedBeijianOptions.value))
|
|
|
)
|
|
|
const buildCriticalComponentTableRows = () => {
|
|
|
const detailList = (detailData.value as any)?.componentList
|
|
|
if (Array.isArray(detailList) && detailList.length) {
|
|
|
/* detailList.forEach(item => {
|
|
|
item.id = null
|
|
|
})*/
|
|
|
return detailList.map((item: any) => ({ ...item, count: normalizeNumberish(item?.count) ?? 0 }))
|
|
|
}
|
|
|
|
|
|
const selectedIds = new Set<number>((formData.value.componentIds ?? []).map((id: any) => Number(id)).filter((id: number) => !Number.isNaN(id)))
|
|
|
if (!selectedIds.size) return []
|
|
|
|
|
|
const optionMap = new Map<number, any>()
|
|
|
;[...criticalComponentOptions.value, ...savedCriticalComponentOptions.value].forEach((item: any) => {
|
|
|
optionMap.set(Number(item.value), item)
|
|
|
})
|
|
|
|
|
|
return Array.from(selectedIds).map((id) => {
|
|
|
const option = optionMap.get(id)
|
|
|
const label = option?.label ?? String(id)
|
|
|
const [code, ...nameParts] = label.split('-')
|
|
|
return {
|
|
|
id,
|
|
|
code: nameParts.length ? code : '',
|
|
|
name: nameParts.length ? nameParts.join('-') : label,
|
|
|
count: 0,
|
|
|
description: '',
|
|
|
remark: '',
|
|
|
createTime: undefined
|
|
|
}
|
|
|
})
|
|
|
}
|
|
|
|
|
|
const syncCriticalComponentRows = () => {
|
|
|
editableCriticalComponentRows.value = buildCriticalComponentTableRows()
|
|
|
}
|
|
|
|
|
|
const clearCriticalComponent = () => {
|
|
|
formData.value.componentIds = []
|
|
|
}
|
|
|
const clearBeijian = () => {
|
|
|
formData.value.beijianIds = []
|
|
|
}
|
|
|
|
|
|
const handleFileUploadingChange = (uploading: boolean) => {
|
|
|
fileUploading.value = uploading
|
|
|
}
|
|
|
|
|
|
watch(
|
|
|
() => formData.value.isScheduled,
|
|
|
() => {
|
|
|
formRef.value?.clearValidate?.(['ratedCapacity'])
|
|
|
}
|
|
|
)
|
|
|
|
|
|
const handleCodeAutoChange = (value: boolean) => {
|
|
|
if (value) {
|
|
|
formData.value.deviceCode = undefined
|
|
|
}
|
|
|
formRef.value?.clearValidate('deviceCode')
|
|
|
}
|
|
|
|
|
|
const getQrcodeRefreshUrl = () => {
|
|
|
if (!formData.value.id || !formData.value.deviceCode) return ''
|
|
|
return `/mes/device-ledger/regenerate-code?id=${formData.value.id}&code=${encodeURIComponent(String(formData.value.deviceCode))}`
|
|
|
}
|
|
|
|
|
|
const buildPrintData = () => {
|
|
|
return {
|
|
|
id: formData.value.id,
|
|
|
deviceCode: formData.value.deviceCode,
|
|
|
deviceName: formData.value.deviceName,
|
|
|
deviceSpec: formData.value.deviceSpec,
|
|
|
deviceBrand: formData.value.deviceBrand,
|
|
|
deviceModel: formData.value.deviceModel,
|
|
|
deviceLocation: formData.value.deviceLocation,
|
|
|
remark: formData.value.remark,
|
|
|
qrcodeUrl: formData.value.qrcodeUrl
|
|
|
}
|
|
|
}
|
|
|
|
|
|
const handleQrcodeRefreshSuccess = async (data: any) => {
|
|
|
if (!formData.value.id) return
|
|
|
if (data?.qrcodeUrl) {
|
|
|
formData.value.qrcodeUrl = data.qrcodeUrl
|
|
|
return
|
|
|
}
|
|
|
const detail = await DeviceLedgerApi.getDeviceLedger(formData.value.id)
|
|
|
formData.value.qrcodeUrl = detail?.qrcodeUrl
|
|
|
formData.value.deviceCode = detail?.deviceCode ?? formData.value.deviceCode
|
|
|
}
|
|
|
|
|
|
const ensureOptionsLoaded = async () => {
|
|
|
const [deviceTypeRes, userRes, criticalRes, beijianRes] = await Promise.all([
|
|
|
DeviceTypeApi.getDeviceTypeTree({ pageNo: 1, pageSize: 10 }),
|
|
|
getSimpleUserList(),
|
|
|
CriticalComponentApi.getCriticalComponentList(),
|
|
|
ProductApi.getComponentSimpleList()
|
|
|
])
|
|
|
deviceTypeTree.value = deviceTypeRes ?? []
|
|
|
users.value = userRes ?? []
|
|
|
criticalComponentOptions.value = buildCriticalComponentOptions(criticalRes ?? [])
|
|
|
beijianOptions.value = buildBeijianOptions(beijianRes ?? [])
|
|
|
}
|
|
|
|
|
|
const bindFormData = (detail: DeviceLedgerVO) => {
|
|
|
const templateJson = (detail as any)?.templateJson
|
|
|
const parsedTemplateJson =
|
|
|
typeof templateJson === 'string'
|
|
|
? (() => {
|
|
|
try {
|
|
|
return JSON.parse(templateJson)
|
|
|
} catch {
|
|
|
return undefined
|
|
|
}
|
|
|
})()
|
|
|
: templateJson
|
|
|
formData.value = {
|
|
|
...initFormData(),
|
|
|
...(detail as any),
|
|
|
templateJson: parsedTemplateJson,
|
|
|
isCode: (detail as any)?.isCode ?? false,
|
|
|
isScheduled: normalizeNumberish((detail as any)?.isScheduled) ?? 0,
|
|
|
ratedCapacity: normalizeNumberish((detail as any)?.ratedCapacity),
|
|
|
deviceType: normalizeNumberish((detail as any)?.deviceType),
|
|
|
deviceLine: normalizeNumberish((detail as any)?.deviceLine),
|
|
|
deviceManagerIds: parseIdsValue((detail as any)?.deviceManager),
|
|
|
productionDate: normalizeYmd((detail as any)?.productionDate),
|
|
|
factoryEntryDate: normalizeYmd((detail as any)?.factoryEntryDate),
|
|
|
componentIds: parseIdsValue((detail as any)?.componentId),
|
|
|
beijianIds: parseIdsValue((detail as any)?.beijianId),
|
|
|
qrcodeUrl: (detail as any)?.qrcodeUrl
|
|
|
}
|
|
|
savedCriticalComponentOptions.value = buildCriticalComponentOptions((detail as any)?.componentList ?? [])
|
|
|
savedBeijianOptions.value = buildBeijianOptions((detail as any)?.beijianList ?? [])
|
|
|
}
|
|
|
syncCriticalComponentRows()
|
|
|
|
|
|
const criticalExportLoading = ref(false)
|
|
|
|
|
|
const normalizeFileUrlAsJsonArrayString = (value: any): string | undefined => {
|
|
|
if (value === null || value === undefined || value === '') return undefined
|
|
|
if (typeof value === 'string') return value.trim() || undefined
|
|
|
if (Array.isArray(value)) return JSON.stringify(value)
|
|
|
return JSON.stringify(value)
|
|
|
}
|
|
|
|
|
|
const buildSubmitData = () => {
|
|
|
const componentRows = editableCriticalComponentRows.value.map((row: any) => ({
|
|
|
...row,
|
|
|
id: normalizeNumberish(row?.id),
|
|
|
count: normalizeNumberish(row?.count) ?? 0
|
|
|
})).filter((row: any) => row.id !== undefined)
|
|
|
formData.value.componentIds = componentRows.map((row: any) => row.id)
|
|
|
|
|
|
const data = {
|
|
|
...(formData.value as any),
|
|
|
isScheduled: normalizeNumberish((formData.value as any).isScheduled) ?? 0,
|
|
|
ratedCapacity: normalizeNumberish((formData.value as any).ratedCapacity),
|
|
|
deviceType: normalizeNumberish(formData.value.deviceType),
|
|
|
deviceLine: normalizeNumberish((formData.value as any).deviceLine),
|
|
|
productionDate: normalizeYmd(formData.value.productionDate),
|
|
|
factoryEntryDate: normalizeYmd(formData.value.factoryEntryDate),
|
|
|
deviceManager: formData.value.deviceManagerIds?.length ? formData.value.deviceManagerIds.join(',') : undefined,
|
|
|
componentId: formData.value.componentIds?.length ? formData.value.componentIds.join(',') : undefined,
|
|
|
componentList: componentRows,
|
|
|
beijianId: formData.value.beijianIds?.length ? formData.value.beijianIds.join(',') : undefined,
|
|
|
fileUrl: normalizeFileUrlAsJsonArrayString((formData.value as any).fileUrl)
|
|
|
} as unknown as DeviceLedgerVO
|
|
|
|
|
|
delete (data as any).deviceManagerIds
|
|
|
delete (data as any).componentIds
|
|
|
delete (data as any).beijianIds
|
|
|
return data
|
|
|
}
|
|
|
|
|
|
const submitForm = async () => {
|
|
|
if (fileUploading.value) {
|
|
|
message.warning(t('common.loading'))
|
|
|
return
|
|
|
}
|
|
|
await formRef.value?.validate?.()
|
|
|
formLoading.value = true
|
|
|
try {
|
|
|
await DeviceLedgerApi.updateDeviceLedger(buildSubmitData())
|
|
|
message.success(t('common.updateSuccess'))
|
|
|
await getDetail()
|
|
|
} finally {
|
|
|
formLoading.value = false
|
|
|
}
|
|
|
}
|
|
|
|
|
|
const handleCancelEdit = async () => {
|
|
|
await getDetail()
|
|
|
}
|
|
|
const handleExportCriticalComponent = async () => {
|
|
|
if (!deviceId.value) return
|
|
|
try {
|
|
|
await message.exportConfirm()
|
|
|
criticalExportLoading.value = true
|
|
|
const data = await DeviceLedgerApi.exportDeviceComponent({ deviceId: deviceId.value })
|
|
|
download.excel(data, 'critical-component.xls')
|
|
|
} catch {
|
|
|
} finally {
|
|
|
criticalExportLoading.value = false
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
const getDetail = async () => {
|
|
|
if (!deviceId.value) {
|
|
|
message.warning(t('EquipmentManagement.EquipmentLedger.Detail.invalidId'))
|
|
|
delView(unref(currentRoute))
|
|
|
return
|
|
|
}
|
|
|
detailLoading.value = true
|
|
|
try {
|
|
|
const detail = await DeviceLedgerApi.getDeviceLedger(deviceId.value)
|
|
|
detailData.value = detail
|
|
|
bindFormData(detail)
|
|
|
} finally {
|
|
|
detailLoading.value = false
|
|
|
}
|
|
|
}
|
|
|
|
|
|
|
|
|
watch(bjList, (newData) => {
|
|
|
if (newData.length > 0) {
|
|
|
setBJDefaultSelections()
|
|
|
}
|
|
|
}, { deep: true })
|
|
|
|
|
|
|
|
|
//const defaultSelectedIds = [144, 143,141]
|
|
|
|
|
|
// 设置默认选中的行
|
|
|
const setBJDefaultSelections = () => {
|
|
|
// 等待DOM更新完成
|
|
|
nextTick(() => {
|
|
|
if (!bjMultipleTableRef.value) return
|
|
|
bjMultipleTableRef.value.clearSelection()
|
|
|
const rawSubjectIds = toRaw(formData.value.beijianIds)
|
|
|
if (rawSubjectIds.length != 0) {
|
|
|
let row = {
|
|
|
id: undefined
|
|
|
}
|
|
|
bjMultipleTableRef.value!.toggleRowSelection(row, true)
|
|
|
}
|
|
|
// 遍历数据,找到需要选中的行
|
|
|
bjList.value.forEach(row => {
|
|
|
if (rawSubjectIds.includes(row.id)) {
|
|
|
bjMultipleTableRef.value!.toggleRowSelection(row, true)
|
|
|
}
|
|
|
})
|
|
|
})
|
|
|
}
|
|
|
async function initForm() {
|
|
|
// 修改时,设置数据
|
|
|
if (deviceId.value) {
|
|
|
formLoading.value = true
|
|
|
try {
|
|
|
const detail = await DeviceLedgerApi.getDeviceLedger(deviceId.value)
|
|
|
detailData.value = detail
|
|
|
const templateJson = (detail as any)?.templateJson
|
|
|
const parsedTemplateJson = typeof templateJson === 'string'
|
|
|
? JSON.parse(templateJson)
|
|
|
: templateJson
|
|
|
formData.value = {
|
|
|
...initFormData(),
|
|
|
...(detail as any),
|
|
|
templateJson: parsedTemplateJson,
|
|
|
isCode: (detail as any)?.isCode ?? false,
|
|
|
isScheduled: normalizeNumberish((detail as any)?.isScheduled ?? (detail as any)?.isScheduled) ?? 0,
|
|
|
ratedCapacity: normalizeNumberish((detail as any)?.ratedCapacity),
|
|
|
deviceType: normalizeNumberish((detail as any)?.deviceType),
|
|
|
deviceLine: normalizeNumberish((detail as any)?.deviceLine),
|
|
|
deviceManagerIds: parseIdsValue((detail as any)?.deviceManager),
|
|
|
productionDate: normalizeYmd((detail as any)?.productionDate),
|
|
|
factoryEntryDate: normalizeYmd((detail as any)?.factoryEntryDate),
|
|
|
componentIds: parseIdsValue((detail as any)?.componentId),
|
|
|
beijianIds: parseIdsValue((detail as any)?.beijianId),
|
|
|
qrcodeUrl: (detail as any)?.qrcodeUrl,
|
|
|
}
|
|
|
savedCriticalComponentOptions.value = buildCriticalComponentOptions((detail as any)?.componentList ?? [])
|
|
|
savedBeijianOptions.value = buildBeijianOptions((detail as any)?.beijianList ?? [])
|
|
|
// syncCriticalComponentRows()
|
|
|
} finally {
|
|
|
formLoading.value = false
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
async function getDetailList() {
|
|
|
if (!deviceId.value) return
|
|
|
detailLoading.value = true
|
|
|
try {
|
|
|
let detail:any = await DeviceLedgerApi.getDeviceCriticalComponent(deviceId.value)
|
|
|
editableCriticalComponentRows.value = detail
|
|
|
} finally {
|
|
|
detailLoading.value = false
|
|
|
}
|
|
|
}
|
|
|
onMounted(async () => {
|
|
|
await ensureOptionsLoaded()
|
|
|
initForm()
|
|
|
//初始化设备关键件
|
|
|
getList()
|
|
|
//初始化备件
|
|
|
bjGetList()
|
|
|
//初始化设备关键件明细
|
|
|
getDetailList()
|
|
|
await nextTick()
|
|
|
updateActionBarRect()
|
|
|
window.addEventListener('resize', updateActionBarRect)
|
|
|
if (detailPageRef.value) {
|
|
|
actionBarObserver = new ResizeObserver(updateActionBarRect)
|
|
|
actionBarObserver.observe(detailPageRef.value)
|
|
|
}
|
|
|
})
|
|
|
|
|
|
watch(() => appStore.getCollapse, updateActionBarRectAfterLayout)
|
|
|
|
|
|
onBeforeUnmount(() => {
|
|
|
window.removeEventListener('resize', updateActionBarRect)
|
|
|
actionBarObserver?.disconnect()
|
|
|
})
|
|
|
</script>
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
.device-ledger-detail-page {
|
|
|
width: 100%;
|
|
|
}
|
|
|
|
|
|
.device-ledger-detail-body {
|
|
|
position: relative;
|
|
|
max-height: 100vh;
|
|
|
padding: 2px 0 72px;
|
|
|
}
|
|
|
|
|
|
.device-ledger-detail-title {
|
|
|
position: relative;
|
|
|
display: flex;
|
|
|
align-items: center;
|
|
|
height: 28px;
|
|
|
color: var(--el-text-color-primary);
|
|
|
font-size: 16px;
|
|
|
font-weight: 600;
|
|
|
line-height: 22px;
|
|
|
}
|
|
|
|
|
|
.device-ledger-detail-title::before {
|
|
|
width: 3px;
|
|
|
height: 16px;
|
|
|
margin-right: 8px;
|
|
|
content: '';
|
|
|
border-radius: 2px;
|
|
|
background: var(--el-color-primary);
|
|
|
}
|
|
|
|
|
|
.device-ledger-detail-main {
|
|
|
min-height: 224px;
|
|
|
padding-top: 12px;
|
|
|
margin-bottom: 10px;
|
|
|
gap: 28px;
|
|
|
box-sizing: border-box;
|
|
|
}
|
|
|
|
|
|
.device-ledger-detail-main :deep(.el-form) {
|
|
|
width: 100%;
|
|
|
}
|
|
|
|
|
|
.device-basic-layout {
|
|
|
align-items: flex-start;
|
|
|
}
|
|
|
|
|
|
.device-basic-fields {
|
|
|
min-width: 0;
|
|
|
}
|
|
|
|
|
|
.device-code-row {
|
|
|
width: 100%;
|
|
|
}
|
|
|
|
|
|
.device-code-switch {
|
|
|
display: flex;
|
|
|
align-items: center;
|
|
|
height: 32px;
|
|
|
}
|
|
|
|
|
|
.device-basic-media {
|
|
|
display: flex;
|
|
|
justify-content: flex-end;
|
|
|
}
|
|
|
|
|
|
.device-media-panel {
|
|
|
width: 100%;
|
|
|
max-width: 260px;
|
|
|
padding: 12px 12px 4px;
|
|
|
box-sizing: border-box;
|
|
|
border: 1px solid var(--el-border-color-lighter);
|
|
|
border-radius: 6px;
|
|
|
background: var(--el-fill-color-extra-light);
|
|
|
}
|
|
|
|
|
|
.device-media-item {
|
|
|
margin-bottom: 16px;
|
|
|
}
|
|
|
|
|
|
.device-media-item :deep(.el-form-item__label) {
|
|
|
justify-content: flex-start;
|
|
|
}
|
|
|
|
|
|
.device-media-item :deep(.el-form-item__content) {
|
|
|
justify-content: center;
|
|
|
}
|
|
|
|
|
|
.device-media-upload {
|
|
|
width: 100%;
|
|
|
min-height: 120px;
|
|
|
}
|
|
|
|
|
|
.device-media-code {
|
|
|
width: 148px;
|
|
|
height: 148px;
|
|
|
}
|
|
|
|
|
|
.device-media-code :deep(.qrcode-action-card__img),
|
|
|
.device-media-code :deep(.qrcode-action-card__error) {
|
|
|
width: 148px;
|
|
|
height: 148px;
|
|
|
}
|
|
|
|
|
|
.device-ledger-detail-tabs {
|
|
|
padding-right: 4px;
|
|
|
}
|
|
|
|
|
|
:deep(.el-tabs__content) {
|
|
|
max-height: 65vh;
|
|
|
overflow-y: auto;
|
|
|
}
|
|
|
|
|
|
.device-ledger-tab-toolbar {
|
|
|
margin-bottom: 8px;
|
|
|
text-align: right;
|
|
|
}
|
|
|
|
|
|
.device-ledger-selection-input :deep(.el-input__inner) {
|
|
|
overflow: hidden;
|
|
|
text-overflow: ellipsis;
|
|
|
white-space: nowrap;
|
|
|
}
|
|
|
.device-ledger-action-bar {
|
|
|
position: fixed;
|
|
|
bottom: 0;
|
|
|
z-index: 10;
|
|
|
display: flex;
|
|
|
justify-content: center;
|
|
|
gap: 12px;
|
|
|
padding: 12px 16px;
|
|
|
box-sizing: border-box;
|
|
|
border-top: 1px solid var(--el-border-color-lighter);
|
|
|
background: var(--el-bg-color);
|
|
|
box-shadow: 0 -4px 12px rgb(0 0 0 / 6%);
|
|
|
}
|
|
|
|
|
|
@media (max-width: 1199px) {
|
|
|
.device-basic-media {
|
|
|
justify-content: flex-start;
|
|
|
}
|
|
|
|
|
|
.device-media-panel {
|
|
|
max-width: none;
|
|
|
}
|
|
|
}
|
|
|
|
|
|
@media (max-width: 767px) {
|
|
|
.device-basic-fields :deep(.el-col) {
|
|
|
max-width: 100%;
|
|
|
flex: 0 0 100%;
|
|
|
}
|
|
|
|
|
|
.device-code-row :deep(.el-col:first-child) {
|
|
|
max-width: 75%;
|
|
|
flex: 0 0 75%;
|
|
|
}
|
|
|
|
|
|
.device-code-row :deep(.el-col:last-child) {
|
|
|
max-width: 25%;
|
|
|
flex: 0 0 25%;
|
|
|
}
|
|
|
}
|
|
|
</style>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|