<template>
|
<el-dialog v-model="dialogVisible"
|
:title="dialogTitle"
|
width="1000px"
|
:close-on-click-modal="false"
|
custom-class="custom-dialog">
|
<div class="detail-container">
|
<!-- 基础信息 -->
|
<div class="detail-section">
|
<h3 class="section-title">基础信息</h3>
|
<el-descriptions :column="3"
|
border>
|
<el-descriptions-item label="生产订单号">{{ detailData.npsNo || '-' }}</el-descriptions-item>
|
<el-descriptions-item label="班组">{{ detailData.schedule || '-' }}</el-descriptions-item>
|
<el-descriptions-item label="岗位人员">{{ detailData.postName || '-' }}</el-descriptions-item>
|
<el-descriptions-item label="产品编码">{{ detailData.materialCode || '-' }}</el-descriptions-item>
|
<el-descriptions-item label="产品名称">{{ detailData.productName || '-' }}</el-descriptions-item>
|
<el-descriptions-item label="规格">{{ detailData.model || '-' }}</el-descriptions-item>
|
<el-descriptions-item label="合格数量">{{ detailData.qualifiedQuantity || 0 }} 方</el-descriptions-item>
|
<el-descriptions-item label="不合格数量">{{ detailData.unqualifiedQuantity || 0 }} 方</el-descriptions-item>
|
<el-descriptions-item label="总数量">{{ detailData.quantity || 0 }} 方</el-descriptions-item>
|
<el-descriptions-item label="报工时间">{{ formatTime(detailData.reportingTime) }}</el-descriptions-item>
|
<el-descriptions-item label="创建时间">{{ formatTime(detailData.createTime) }}</el-descriptions-item>
|
<el-descriptions-item label="更新时间">{{ formatTime(detailData.updateTime) }}</el-descriptions-item>
|
</el-descriptions>
|
</div>
|
<!-- 工序信息 -->
|
<div class="detail-section"
|
v-if="detailData.productionProductRouteItemDtoList && detailData.productionProductRouteItemDtoList.length > 0">
|
<h3 class="section-title">工序信息</h3>
|
<div v-for="(process, index) in detailData.productionProductRouteItemDtoList"
|
:key="process.id"
|
class="process-item">
|
<div class="process-header">
|
<h4 class="process-title">工序 {{ index + 1 }}</h4>
|
<div class="process-info">
|
<span class="process-label">岗位人员:{{ process.postName || '-' }}</span>
|
<span class="process-label">工序ID:{{ process.processId || '-' }}</span>
|
</div>
|
</div>
|
<!-- 工序基本信息 -->
|
<div class="process-details">
|
<el-descriptions :column="2"
|
border>
|
<el-descriptions-item label="设备异常情况">{{ process.equipmentMalfunction || '-' }}</el-descriptions-item>
|
<el-descriptions-item label="当班设备处置">{{ process.equipmentDisposal || '-' }}</el-descriptions-item>
|
<el-descriptions-item label="工艺人员交待"
|
:span="2">{{ process.processExplained || '-' }}</el-descriptions-item>
|
</el-descriptions>
|
</div>
|
<!-- 工序参数 -->
|
<div v-if="process.productionProductRouteItemParamDtoList && process.productionProductRouteItemParamDtoList.length > 0">
|
<!-- BOM信息 -->
|
<div class="param-section"
|
v-if="getBomList(process.productionProductRouteItemParamDtoList).length > 0">
|
<h5 class="param-title">BOM信息</h5>
|
<el-table :data="getBomList(process.productionProductRouteItemParamDtoList)"
|
style="width: 100%"
|
size="small">
|
<el-table-column prop="paramName"
|
label="参数名称"
|
min-width="120"></el-table-column>
|
<el-table-column prop="productValue"
|
label="产品值"
|
min-width="100"></el-table-column>
|
<el-table-column prop="unit"
|
label="单位"
|
width="80"></el-table-column>
|
</el-table>
|
</div>
|
<!-- 参数信息 -->
|
<div class="param-section"
|
v-if="getParamList(process.productionProductRouteItemParamDtoList).length > 0">
|
<h5 class="param-title">参数信息</h5>
|
<div v-for="(group, sort) in getParamGroups(process.productionProductRouteItemParamDtoList)"
|
:key="sort"
|
class="param-group">
|
<div class="group-header">
|
<span class="group-title">参数组 {{ sort }}</span>
|
</div>
|
<div class="param-grid">
|
<div v-for="param in group"
|
:key="param.id"
|
class="param-item">
|
<span class="param-label">{{ param.paramName || '-' }}:</span>
|
<span class="param-value">{{ param.paramValue || '-' }}</span>
|
<span v-if="param.unit && param.unit !== '/'"
|
class="param-unit">{{ param.unit }}</span>
|
</div>
|
</div>
|
</div>
|
</div>
|
</div>
|
<!-- 上传文件 -->
|
<div class="file-section"
|
v-if="process.fileList && process.fileList.length > 0">
|
<h5 class="file-title">上传文件</h5>
|
<div class="file-grid">
|
<div v-for="file in process.fileList"
|
:key="file.id"
|
class="file-item">
|
<img v-if="file.fileUrl"
|
:src="baseUrl + file.fileUrl"
|
:alt="file.fileName"
|
class="file-image"
|
@click="handleFilePreview(file)" />
|
<div class="file-info">
|
<span class="file-name">{{ file.fileName || '-' }}</span>
|
</div>
|
</div>
|
</div>
|
</div>
|
</div>
|
</div>
|
</div>
|
<template #footer>
|
<div class="dialog-footer">
|
<el-button @click="dialogVisible = false">关闭</el-button>
|
</div>
|
</template>
|
</el-dialog>
|
</template>
|
|
<script setup>
|
import { ref, computed, watch } from "vue";
|
import dayjs from "dayjs";
|
|
const baseUrl = import.meta.env.VITE_APP_BASE_API;
|
const props = defineProps({
|
visible: {
|
type: Boolean,
|
default: false,
|
},
|
data: {
|
type: Object,
|
default: () => ({}),
|
},
|
});
|
|
const emit = defineEmits(["update:visible"]);
|
|
const dialogVisible = computed({
|
get: () => props.visible,
|
set: value => emit("update:visible", value),
|
});
|
|
const dialogTitle = computed(() => "生产报工详情");
|
const detailData = ref(props.data);
|
|
// 格式化时间
|
const formatTime = time => {
|
return time ? dayjs(time).format("YYYY-MM-DD HH:mm:ss") : "-";
|
};
|
|
// 格式化文件列表
|
const formatFileList = fileList => {
|
return fileList.map(file => ({
|
name: file.fileName,
|
url: file.fileUrl,
|
size: file.fileSize,
|
}));
|
};
|
|
// 处理文件预览
|
const handleFilePreview = file => {
|
// 这里可以实现文件预览逻辑
|
console.log("预览文件:", file);
|
};
|
|
// 获取BOM列表
|
const getBomList = paramList => {
|
return paramList.filter(item => item.bomId);
|
};
|
|
// 获取参数列表
|
const getParamList = paramList => {
|
return paramList.filter(item => !item.bomId);
|
};
|
|
// 按sourceSort分组参数
|
const getParamGroups = paramList => {
|
const params = getParamList(paramList);
|
const groups = {};
|
|
params.forEach(param => {
|
const sort = param.sourceSort || 1;
|
if (!groups[sort]) {
|
groups[sort] = [];
|
}
|
groups[sort].push(param);
|
});
|
|
return groups;
|
};
|
|
// 监听数据变化
|
watch(
|
() => props.data,
|
newData => {
|
detailData.value = newData;
|
},
|
{ deep: true }
|
);
|
</script>
|
|
<style scoped>
|
.detail-container {
|
max-height: 600px;
|
overflow-y: auto;
|
padding: 0 16px;
|
}
|
|
.detail-section {
|
margin-bottom: 28px;
|
background-color: #ffffff;
|
border-radius: 8px;
|
padding: 20px;
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.05);
|
}
|
|
.section-title {
|
font-size: 16px;
|
font-weight: 600;
|
margin-bottom: 16px;
|
color: #1a1a1a;
|
border-bottom: 2px solid #409eff;
|
padding-bottom: 10px;
|
}
|
|
.process-item {
|
margin-bottom: 24px;
|
padding: 20px;
|
background-color: #ffffff;
|
border-radius: 8px;
|
border: 1px solid #ebeef5;
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
|
}
|
|
.process-header {
|
margin-bottom: 20px;
|
padding-bottom: 12px;
|
border-bottom: 1px solid #f0f2f5;
|
}
|
|
.process-title {
|
font-size: 15px;
|
font-weight: 600;
|
margin-bottom: 12px;
|
color: #1a1a1a;
|
display: flex;
|
align-items: center;
|
}
|
|
.process-title::before {
|
content: "";
|
display: inline-block;
|
width: 4px;
|
height: 16px;
|
background-color: #409eff;
|
margin-right: 8px;
|
border-radius: 2px;
|
}
|
|
.process-info {
|
display: flex;
|
gap: 20px;
|
font-size: 13px;
|
color: #606266;
|
}
|
|
.process-label {
|
padding: 4px 12px;
|
background-color: #ecf5ff;
|
border-radius: 4px;
|
color: #409eff;
|
font-weight: 500;
|
}
|
|
.process-details {
|
margin-bottom: 20px;
|
}
|
|
.param-section {
|
margin-bottom: 20px;
|
background-color: #f9f9f9;
|
border-radius: 6px;
|
padding: 16px;
|
border: 1px solid #f0f2f5;
|
}
|
|
.param-title {
|
font-size: 14px;
|
font-weight: 600;
|
margin-bottom: 14px;
|
color: #1a1a1a;
|
padding-bottom: 8px;
|
border-bottom: 1px solid #e8e8e8;
|
}
|
|
.file-section {
|
margin-top: 20px;
|
background-color: #f9f9f9;
|
border-radius: 6px;
|
padding: 16px;
|
border: 1px solid #f0f2f5;
|
}
|
|
.file-title {
|
font-size: 14px;
|
font-weight: 600;
|
margin-bottom: 14px;
|
color: #1a1a1a;
|
padding-bottom: 8px;
|
border-bottom: 1px solid #e8e8e8;
|
}
|
|
.file-grid {
|
display: grid;
|
grid-template-columns: repeat(auto-fill, minmax(120px, 1fr));
|
gap: 16px;
|
}
|
|
.file-item {
|
display: flex;
|
flex-direction: column;
|
align-items: center;
|
background-color: #ffffff;
|
border: 1px solid #e8e8e8;
|
border-radius: 6px;
|
padding: 10px;
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.05);
|
transition: all 0.3s ease;
|
}
|
|
.file-item:hover {
|
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
|
border-color: #409eff;
|
transform: translateY(-2px);
|
}
|
|
.file-image {
|
width: 80px;
|
height: 80px;
|
object-fit: cover;
|
border-radius: 4px;
|
cursor: pointer;
|
margin-bottom: 8px;
|
}
|
|
.file-info {
|
width: 100%;
|
text-align: center;
|
}
|
|
.file-name {
|
font-size: 12px;
|
color: #606266;
|
word-break: break-all;
|
line-height: 1.4;
|
}
|
|
.param-group {
|
margin-bottom: 16px;
|
padding: 14px;
|
background-color: #ffffff;
|
border-radius: 6px;
|
border: 1px solid #e8e8e8;
|
}
|
|
.group-header {
|
margin-bottom: 12px;
|
padding-bottom: 8px;
|
border-bottom: 1px solid #f0f2f5;
|
}
|
|
.group-title {
|
font-size: 14px;
|
font-weight: 600;
|
color: #303133;
|
}
|
|
.param-grid {
|
display: grid;
|
grid-template-columns: repeat(auto-fill, minmax(280px, 1fr));
|
gap: 16px;
|
}
|
|
.param-item {
|
display: flex;
|
align-items: center;
|
gap: 12px;
|
padding: 8px 0;
|
border-bottom: 1px solid #f5f7fa;
|
}
|
|
.param-item:last-child {
|
border-bottom: none;
|
}
|
|
.param-label {
|
font-size: 13px;
|
color: #606266;
|
min-width: 100px;
|
font-weight: 500;
|
}
|
|
.param-value {
|
font-size: 13px;
|
color: #1a1a1a;
|
font-weight: 600;
|
flex: 1;
|
}
|
|
.param-unit {
|
font-size: 12px;
|
color: #909399;
|
background-color: #f0f2f5;
|
padding: 2px 6px;
|
border-radius: 3px;
|
}
|
|
.dialog-footer {
|
text-align: center;
|
padding: 20px;
|
border-top: 1px solid #ebeef5;
|
}
|
|
.dialog-footer .el-button {
|
min-width: 100px;
|
padding: 8px 20px;
|
}
|
|
/* 自定义对话框样式 */
|
:deep(.custom-dialog) {
|
border-radius: 12px;
|
overflow: hidden;
|
}
|
|
:deep(.custom-dialog .el-dialog__header) {
|
background-color: #f5f7fa;
|
padding: 20px;
|
border-bottom: 1px solid #ebeef5;
|
}
|
|
:deep(.custom-dialog .el-dialog__title) {
|
font-size: 18px;
|
font-weight: 600;
|
color: #1a1a1a;
|
}
|
|
:deep(.custom-dialog .el-dialog__body) {
|
padding: 20px;
|
}
|
|
/* 表格样式优化 */
|
:deep(.el-table) {
|
border-radius: 6px;
|
overflow: hidden;
|
}
|
|
:deep(.el-table th) {
|
background-color: #f5f7fa;
|
font-weight: 600;
|
color: #303133;
|
}
|
|
:deep(.el-table tr:hover > td) {
|
background-color: #ecf5ff !important;
|
}
|
|
/* 描述列表样式优化 */
|
:deep(.el-descriptions) {
|
border-radius: 6px;
|
overflow: hidden;
|
}
|
|
:deep(.el-descriptions__label) {
|
font-weight: 500;
|
color: #606266;
|
}
|
|
:deep(.el-descriptions__content) {
|
color: #1a1a1a;
|
font-weight: 500;
|
}
|
</style>
|