You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
besure_web/src/views/mes/deviceledger/detail/editIndex.vue

1324 lines
49 KiB
Vue

<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>