880 lines
27 KiB
Vue
880 lines
27 KiB
Vue
<template>
|
||
<div class="typing-coil-container">
|
||
<!-- 顶部操作栏 -->
|
||
<div class="header-bar">
|
||
<div class="header-title">
|
||
<i class="el-icon-edit"></i>
|
||
<span>钢卷信息更新</span>
|
||
</div>
|
||
<div class="header-actions">
|
||
<el-button v-if="!readonly" type="primary" size="small" @click="handleSave" :loading="loading">保存更新</el-button>
|
||
<el-button size="small" @click="handleCancel" :disabled="loading">{{ readonly ? '返回' : '取消' }}</el-button>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 主内容区 - 左右布局 -->
|
||
<div class="content-wrapper">
|
||
<!-- 左侧:当前信息 -->
|
||
<div class="left-panel">
|
||
<el-card class="info-card">
|
||
<div slot="header" class="card-header">
|
||
<span><i class="el-icon-info"></i> 当前信息</span>
|
||
</div>
|
||
|
||
<div class="info-section">
|
||
<div class="info-row">
|
||
<span class="info-label">入场钢卷号:</span>
|
||
<span class="info-value">{{ currentInfo.enterCoilNo || '—' }}</span>
|
||
</div>
|
||
<div class="info-row">
|
||
<span class="info-label">当前钢卷号:</span>
|
||
<span class="info-value">{{ currentInfo.currentCoilNo || '—' }}</span>
|
||
</div>
|
||
<div class="info-row">
|
||
<span class="info-label">厂家原料卷号:</span>
|
||
<span class="info-value">{{ currentInfo.supplierCoilNo || '—' }}</span>
|
||
</div>
|
||
<div class="info-row">
|
||
<span class="info-label">班组:</span>
|
||
<span class="info-value">{{ currentInfo.team || '—' }}</span>
|
||
</div>
|
||
<div class="info-row">
|
||
<span class="info-label">物品类型:</span>
|
||
<span class="info-value">{{ getItemTypeText(currentInfo.itemType) }}</span>
|
||
</div>
|
||
<div class="info-row">
|
||
<span class="info-label">物料名称:</span>
|
||
<span class="info-value">{{ currentInfo.itemName || '—' }}</span>
|
||
</div>
|
||
<div class="info-row">
|
||
<span class="info-label">毛重:</span>
|
||
<span class="info-value">{{ currentInfo.grossWeight ? currentInfo.grossWeight + ' t' : '—' }}</span>
|
||
</div>
|
||
<div class="info-row">
|
||
<span class="info-label">净重:</span>
|
||
<span class="info-value">{{ currentInfo.netWeight ? currentInfo.netWeight + ' t' : '—' }}</span>
|
||
</div>
|
||
<div class="info-row">
|
||
<span class="info-label">逻辑库区:</span>
|
||
<span class="info-value">{{ currentInfo.nextWarehouseName || '—' }}</span>
|
||
</div>
|
||
<div class="info-row" v-if="currentInfo.remark">
|
||
<span class="info-label">备注:</span>
|
||
<span class="info-value">{{ currentInfo.remark }}</span>
|
||
</div>
|
||
</div>
|
||
</el-card>
|
||
</div>
|
||
|
||
<!-- 右侧:更新表单 -->
|
||
<div class="right-panel">
|
||
<el-card class="form-card">
|
||
<div slot="header" class="card-header">
|
||
<span><i class="el-icon-edit-outline"></i> {{ readonly ? '查看信息' : '更新信息' }}</span>
|
||
<el-button v-if="!readonly" type="text" size="mini" @click="copyFromCurrent" icon="el-icon-document-copy">
|
||
复制当前信息
|
||
</el-button>
|
||
</div>
|
||
|
||
<el-form ref="updateForm" :model="updateForm" :rules="rules" label-width="120px" size="small">
|
||
<el-form-item label="当前钢卷号" prop="currentCoilNo">
|
||
<el-input v-model="updateForm.currentCoilNo" placeholder="请输入当前钢卷号" :disabled="readonly">
|
||
<template slot="prepend">
|
||
<i class="el-icon-document"></i>
|
||
</template>
|
||
</el-input>
|
||
</el-form-item>
|
||
|
||
<el-form-item label="班组" prop="team">
|
||
<el-input v-model="updateForm.team" placeholder="请输入班组名称" :disabled="readonly">
|
||
<template slot="prepend">
|
||
<i class="el-icon-user-solid"></i>
|
||
</template>
|
||
</el-input>
|
||
</el-form-item>
|
||
|
||
<el-form-item label="材料类型" prop="materialType">
|
||
<el-select v-model="updateForm.materialType" placeholder="请选择材料类型" style="width: 100%"
|
||
:disabled="readonly" @change="handleMaterialTypeChange">
|
||
<el-option label="原料" value="原料" />
|
||
<el-option label="成品" value="成品" />
|
||
<el-option label="废品" value="废品" />
|
||
</el-select>
|
||
</el-form-item>
|
||
|
||
<!-- 物品类型由材料类型自动决定,不显示选择框 -->
|
||
|
||
<el-form-item v-if="updateForm.materialType === '成品'" label="质量状态" prop="qualityStatus">
|
||
<el-input v-model="updateForm.qualityStatus" placeholder="请输入质量状态"
|
||
:disabled="readonly">
|
||
</el-input>
|
||
</el-form-item>
|
||
|
||
<el-form-item v-if="updateForm.materialType === '成品'" label="切边要求" prop="qualityStatus">
|
||
<el-input v-model="updateForm.trimmingRequirement" placeholder="请输入切边要求"
|
||
:disabled="readonly">
|
||
</el-input>
|
||
</el-form-item>
|
||
|
||
<el-form-item v-if="updateForm.materialType === '成品'" label="打包状态" prop="qualityStatus">
|
||
<el-input v-model="updateForm.packingStatus" placeholder="请输入打包状态"
|
||
:disabled="readonly">
|
||
</el-input>
|
||
</el-form-item>
|
||
|
||
<el-form-item v-if="updateForm.materialType === '成品'" label="包装要求" prop="qualityStatus">
|
||
<el-input v-model="updateForm.packagingRequirement" placeholder="请输入包装要求"
|
||
:disabled="readonly">
|
||
</el-input>
|
||
</el-form-item>
|
||
|
||
<el-form-item :label="getItemLabel" :prop="updateForm.materialType === '废品' ? '' : 'itemId'"
|
||
:rules="updateForm.materialType === '废品' ? [] : rules.itemId">
|
||
<!-- <el-select v-model="updateForm.itemId" :placeholder="getItemPlaceholder" filterable remote
|
||
:remote-method="searchItems" :loading="itemSearchLoading" style="width: 100%"
|
||
:disabled="readonly || !updateForm.materialType">
|
||
<el-option v-for="item in currentItemList" :key="item.id" :label="item.name" :value="item.id" />
|
||
</el-select> -->
|
||
<RawMaterialSelect v-if="updateForm.materialType === '原料'" v-model="updateForm.itemId"
|
||
placeholder="请选择原料" style="width: 100%" clearable
|
||
:disabled="readonly || !updateForm.materialType" />
|
||
<ProductSelect v-else-if="updateForm.materialType === '成品'" v-model="updateForm.itemId"
|
||
placeholder="请选择成品" style="width: 100%" clearable
|
||
:disabled="readonly || !updateForm.materialType" />
|
||
<div v-else>请先选择物料类型</div>
|
||
</el-form-item>
|
||
|
||
<el-form-item label="毛重(t)" prop="grossWeight">
|
||
<el-input v-model.number="updateForm.grossWeight" placeholder="请输入毛重" type="number" step="0.01"
|
||
:disabled="readonly">
|
||
<template slot="append">吨</template>
|
||
</el-input>
|
||
</el-form-item>
|
||
|
||
<el-form-item label="净重(t)" prop="netWeight">
|
||
<el-input v-model.number="updateForm.netWeight" placeholder="请输入净重" type="number" step="0.01"
|
||
:disabled="readonly">
|
||
<template slot="append">吨</template>
|
||
</el-input>
|
||
</el-form-item>
|
||
|
||
<el-form-item label="逻辑库区" prop="warehouseId">
|
||
<el-select v-model="updateForm.warehouseId" placeholder="请选择逻辑库区" style="width: 100%" filterable
|
||
:disabled="readonly">
|
||
<el-option v-for="warehouse in warehouseList" :key="warehouse.warehouseId"
|
||
:label="warehouse.warehouseName" :value="warehouse.warehouseId" />
|
||
</el-select>
|
||
</el-form-item>
|
||
|
||
<el-form-item label="真实库区" prop="warehouseId">
|
||
<ActualWarehouseSelect
|
||
v-model="updateForm.actualWarehouseId"
|
||
placeholder="请选择真实库区"
|
||
block
|
||
:disabled="readonly"
|
||
/>
|
||
</el-form-item>
|
||
|
||
<el-form-item label="备注" prop="remark">
|
||
<el-input v-model="updateForm.remark" type="textarea" :rows="4" placeholder="请输入备注信息(非必填)" maxlength="500"
|
||
show-word-limit :disabled="readonly" />
|
||
</el-form-item>
|
||
</el-form>
|
||
</el-card>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 变更历史(占满整行) -->
|
||
<div class="history-section">
|
||
<el-card class="history-card">
|
||
<div slot="header" class="card-header">
|
||
<span><i class="el-icon-time"></i> 变更历史</span>
|
||
<el-button type="text" size="mini" @click="loadHistory" :loading="historyLoading">
|
||
<i class="el-icon-refresh"></i> 刷新
|
||
</el-button>
|
||
</div>
|
||
|
||
<el-timeline v-if="historySteps.length > 0">
|
||
<el-timeline-item v-for="(step, index) in historySteps" :key="index"
|
||
:timestamp="`步骤 ${step.display_step || step.step}`" placement="top"
|
||
:type="step.operation === '新增' ? 'success' : 'primary'">
|
||
<div class="history-item">
|
||
<div class="history-title">{{ step.operation || step.action }}</div>
|
||
<div class="history-detail" v-if="step.operator">
|
||
<span class="detail-label">操作人:</span>
|
||
<span>{{ step.operator }}</span>
|
||
</div>
|
||
<div class="history-detail" v-if="step.old_current_coil_no">
|
||
<span class="detail-label">原钢卷号:</span>
|
||
<span>{{ step.old_current_coil_no }}</span>
|
||
</div>
|
||
<div class="history-detail" v-if="step.new_current_coil_no">
|
||
<span class="detail-label">新钢卷号:</span>
|
||
<span>{{ step.new_current_coil_no }}</span>
|
||
</div>
|
||
</div>
|
||
</el-timeline-item>
|
||
</el-timeline>
|
||
|
||
<div v-else class="empty-history">
|
||
<i class="el-icon-document"></i>
|
||
<p>暂无变更历史</p>
|
||
</div>
|
||
</el-card>
|
||
</div>
|
||
</div>
|
||
</template>
|
||
|
||
<script>
|
||
import { getMaterialCoil, updateMaterialCoil, getMaterialCoilTrace } from '@/api/wms/coil';
|
||
import { completeAction } from '@/api/wms/pendingAction';
|
||
import { listWarehouse } from '@/api/wms/warehouse';
|
||
import { listRawMaterialWithBom } from '@/api/wms/rawMaterial';
|
||
import { listProductWithBom } from '@/api/wms/product';
|
||
import ActualWarehouseSelect from "@/components/KLPService/ActualWarehouseSelect";
|
||
import RawMaterialSelect from "@/components/KLPService/RawMaterialSelect";
|
||
import ProductSelect from "@/components/KLPService/ProductSelect";
|
||
|
||
export default {
|
||
name: 'TypingCoil',
|
||
components: {
|
||
ActualWarehouseSelect,
|
||
RawMaterialSelect,
|
||
ProductSelect,
|
||
},
|
||
data() {
|
||
return {
|
||
loading: false,
|
||
historyLoading: false,
|
||
// 当前信息(只读)
|
||
currentInfo: {
|
||
coilId: null,
|
||
enterCoilNo: '',
|
||
currentCoilNo: '',
|
||
supplierCoilNo: '',
|
||
team: '',
|
||
itemType: null,
|
||
itemId: null,
|
||
itemName: '',
|
||
grossWeight: null,
|
||
netWeight: null,
|
||
warehouseId: null,
|
||
warehouseId: null,
|
||
nextWarehouseName: '',
|
||
status: 0,
|
||
remark: ''
|
||
},
|
||
// 更新表单
|
||
updateForm: {
|
||
currentCoilNo: '',
|
||
team: '',
|
||
materialType: null,
|
||
itemType: null,
|
||
itemId: null,
|
||
grossWeight: null,
|
||
netWeight: null,
|
||
warehouseId: null,
|
||
actualWarehouseId: null,
|
||
remark: '',
|
||
qualityStatus: '',
|
||
packagingRequirement: '',
|
||
packingStatus: '',
|
||
trimmingRequirement: ''
|
||
},
|
||
rules: {
|
||
currentCoilNo: [
|
||
{ required: true, message: '请输入当前钢卷号', trigger: 'blur' }
|
||
],
|
||
team: [
|
||
{ required: true, message: '请输入班组', trigger: 'blur' }
|
||
],
|
||
materialType: [
|
||
{ required: true, message: '请选择材料类型', trigger: 'change' }
|
||
],
|
||
itemType: [
|
||
{ required: true, message: '请选择物品类型', trigger: 'change' }
|
||
],
|
||
itemId: [
|
||
{ required: true, message: '请选择物品', trigger: 'change' }
|
||
],
|
||
grossWeight: [
|
||
{ required: true, message: '请输入毛重', trigger: 'blur' },
|
||
{ type: 'number', message: '毛重必须为数字', trigger: 'blur' }
|
||
],
|
||
netWeight: [
|
||
{ required: true, message: '请输入净重', trigger: 'blur' },
|
||
{ type: 'number', message: '净重必须为数字', trigger: 'blur' }
|
||
],
|
||
warehouseId: [
|
||
{ required: true, message: '请选择逻辑库区', trigger: 'change' }
|
||
],
|
||
},
|
||
warehouseList: [],
|
||
historySteps: [],
|
||
actionId: null,
|
||
// 原材料和产品列表(实时搜索,不再保存完整备份)
|
||
rawMaterialList: [],
|
||
productList: [],
|
||
itemSearchLoading: false,
|
||
// 只读模式
|
||
readonly: false
|
||
};
|
||
},
|
||
computed: {
|
||
// 动态显示标签
|
||
getItemLabel() {
|
||
if (this.updateForm.materialType === '成品') {
|
||
return '产品类型';
|
||
} else if (this.updateForm.materialType === '原料' || this.updateForm.materialType === '废品') {
|
||
return '原料类型';
|
||
}
|
||
return '物品';
|
||
},
|
||
// 动态显示占位符
|
||
getItemPlaceholder() {
|
||
if (this.updateForm.materialType === '成品') {
|
||
return '请选择产品类型';
|
||
} else if (this.updateForm.materialType === '原料') {
|
||
return '请选择原料类型';
|
||
} else if (this.updateForm.materialType === '废品') {
|
||
return '请选择原料类型(非必填)';
|
||
}
|
||
return '请先选择材料类型';
|
||
},
|
||
// 当前物品列表(根据物品类型动态切换)
|
||
currentItemList() {
|
||
if (this.updateForm.itemType === 'raw_material') {
|
||
return this.rawMaterialList.map(item => ({
|
||
id: item.rawMaterialId,
|
||
name: this.formatItemName(item)
|
||
}));
|
||
} else if (this.updateForm.itemType === 'product') {
|
||
return this.productList.map(item => ({
|
||
id: item.productId,
|
||
name: this.formatItemName(item)
|
||
}));
|
||
}
|
||
return [];
|
||
}
|
||
},
|
||
async created() {
|
||
// 先加载库区列表
|
||
await this.loadWarehouses();
|
||
|
||
// 不再一次性加载所有数据,改为实时搜索
|
||
// await this.loadAllItems();
|
||
|
||
// 从路由参数获取coilId和actionId
|
||
const coilId = this.$route.query.coilId;
|
||
const actionId = this.$route.query.actionId;
|
||
const readonly = this.$route.query.readonly;
|
||
|
||
if (coilId) {
|
||
await this.loadCoilInfo(coilId);
|
||
}
|
||
|
||
if (actionId) {
|
||
this.actionId = actionId;
|
||
}
|
||
|
||
// 设置只读模式
|
||
if (readonly === 'true' || readonly === true) {
|
||
this.readonly = true;
|
||
}
|
||
},
|
||
methods: {
|
||
// 处理材料类型变化
|
||
handleMaterialTypeChange(value) {
|
||
// 清空物品选择
|
||
this.$set(this.updateForm, 'itemId', null);
|
||
|
||
// 根据材料类型设置物品类型
|
||
if (value === '成品') {
|
||
this.$set(this.updateForm, 'itemType', 'product');
|
||
// 清空列表,等待用户搜索
|
||
this.productList = [];
|
||
} else if (value === '原料' || value === '废品') {
|
||
this.$set(this.updateForm, 'itemType', 'raw_material');
|
||
// 清空列表,等待用户搜索
|
||
this.rawMaterialList = [];
|
||
}
|
||
},
|
||
|
||
// 加载钢卷信息
|
||
async loadCoilInfo(coilId) {
|
||
try {
|
||
this.loading = true;
|
||
const response = await getMaterialCoil(coilId);
|
||
if (response.code === 200 && response.data) {
|
||
const data = response.data;
|
||
|
||
// 填充当前信息(左侧)
|
||
this.currentInfo = {
|
||
...data,
|
||
itemName: this.getItemName(data),
|
||
nextWarehouseName: this.getWarehouseName(data.warehouseId),
|
||
};
|
||
|
||
// 不再预加载物品列表,改为实时搜索
|
||
|
||
// 加载变更历史
|
||
this.loadHistory();
|
||
}
|
||
} catch (error) {
|
||
this.$message.error('加载钢卷信息失败');
|
||
} finally {
|
||
this.loading = false;
|
||
}
|
||
},
|
||
|
||
// 获取物料名称
|
||
getItemName(data) {
|
||
if (data.rawMaterial) {
|
||
return data.rawMaterial.rawMaterialName;
|
||
} else if (data.product) {
|
||
return data.product.productName;
|
||
}
|
||
return '';
|
||
},
|
||
|
||
// 获取物品类型文本
|
||
getItemTypeText(itemType) {
|
||
if (itemType === 'raw_material') return '原材料';
|
||
if (itemType === 'product') return '产品';
|
||
return '—';
|
||
},
|
||
|
||
// 获取库区名称
|
||
getWarehouseName(warehouseId) {
|
||
if (!warehouseId) return '';
|
||
const warehouse = this.warehouseList.find(w => w.warehouseId === warehouseId);
|
||
return warehouse ? warehouse.warehouseName : '';
|
||
},
|
||
|
||
// 格式化物品名称(添加规格和参数信息)
|
||
formatItemName(item) {
|
||
if (!item) return '';
|
||
|
||
// 获取名称(原材料或产品)
|
||
const name = item.rawMaterialName || item.productName || '';
|
||
if (!name) return '';
|
||
|
||
let displayName = name;
|
||
const specs = [];
|
||
|
||
// 1. 优先显示规格(从对象的specification字段)
|
||
if (item.specification) {
|
||
specs.push(item.specification);
|
||
}
|
||
|
||
// 2. 添加参数参数(最多2个)
|
||
if (item.bomItems && item.bomItems.length > 0) {
|
||
const bomParams = item.bomItems
|
||
.filter(bomItem => bomItem.attrKey && bomItem.attrValue)
|
||
.slice(0, 2); // 最多2个参数参数
|
||
|
||
bomParams.forEach(param => {
|
||
specs.push(`${param.attrKey}:${param.attrValue}`);
|
||
});
|
||
}
|
||
|
||
// 3. 拼接成最终格式
|
||
if (specs.length > 0) {
|
||
displayName += `(${specs.join(' ')})`;
|
||
}
|
||
|
||
return displayName;
|
||
},
|
||
|
||
// 加载库区列表
|
||
async loadWarehouses() {
|
||
try {
|
||
const response = await listWarehouse({ pageNum: 1, pageSize: 1000 });
|
||
if (response.code === 200) {
|
||
this.warehouseList = response.rows || response.data || [];
|
||
}
|
||
} catch (error) {
|
||
console.error('加载库区列表失败', error);
|
||
}
|
||
},
|
||
|
||
// 已移除 loadAllItems,改为实时搜索
|
||
|
||
// 已移除 loadItemList,改为实时搜索
|
||
|
||
// 实时搜索物品(后端搜索)
|
||
async searchItems(query) {
|
||
if (!this.updateForm.itemType) {
|
||
this.$message.warning('请先选择材料类型');
|
||
return;
|
||
}
|
||
|
||
// 如果没有输入,清空列表
|
||
if (!query || query.trim() === '') {
|
||
if (this.updateForm.itemType === 'raw_material') {
|
||
this.rawMaterialList = [];
|
||
} else if (this.updateForm.itemType === 'product') {
|
||
this.productList = [];
|
||
}
|
||
return;
|
||
}
|
||
|
||
try {
|
||
this.itemSearchLoading = true;
|
||
const searchQuery = query.trim();
|
||
|
||
if (this.updateForm.itemType === 'raw_material') {
|
||
// 后端搜索原材料(支持名称或规格模糊搜索)
|
||
const response = await listRawMaterialWithBom({
|
||
pageNum: 1,
|
||
pageSize: 20,
|
||
rawMaterialName: searchQuery,
|
||
specification: searchQuery // 同时传入,后端会使用 OR 条件
|
||
});
|
||
if (response.code === 200) {
|
||
this.rawMaterialList = response.rows || [];
|
||
}
|
||
} else if (this.updateForm.itemType === 'product') {
|
||
// 后端搜索产品(支持名称或规格模糊搜索)
|
||
const response = await listProductWithBom({
|
||
pageNum: 1,
|
||
pageSize: 20,
|
||
productName: searchQuery,
|
||
specification: searchQuery // 同时传入,后端会使用 OR 条件
|
||
});
|
||
if (response.code === 200) {
|
||
this.productList = response.rows || [];
|
||
}
|
||
}
|
||
} catch (error) {
|
||
console.error('搜索物品失败', error);
|
||
this.$message.error('搜索失败,请重试');
|
||
} finally {
|
||
this.itemSearchLoading = false;
|
||
}
|
||
},
|
||
|
||
// 物品选择变化
|
||
handleItemChange(itemId) {
|
||
// 物品选择变化处理
|
||
},
|
||
|
||
// 加载变更历史
|
||
async loadHistory() {
|
||
if (!this.currentInfo.enterCoilNo) {
|
||
return;
|
||
}
|
||
|
||
try {
|
||
this.historyLoading = true;
|
||
const response = await getMaterialCoilTrace({
|
||
enterCoilNo: this.currentInfo.enterCoilNo,
|
||
currentCoilNo: this.currentInfo.currentCoilNo || undefined
|
||
});
|
||
|
||
if (response.code === 200 && response.data) {
|
||
this.historySteps = response.data.steps || [];
|
||
}
|
||
} catch (error) {
|
||
console.error('加载变更历史失败', error);
|
||
} finally {
|
||
this.historyLoading = false;
|
||
}
|
||
},
|
||
|
||
// 复制当前信息到更新表单
|
||
copyFromCurrent() {
|
||
const itemType = this.currentInfo.materialType === '原料' ? 'raw_material' : 'product';
|
||
this.updateForm = {
|
||
currentCoilNo: this.currentInfo.currentCoilNo,
|
||
team: this.currentInfo.team,
|
||
materialType: this.currentInfo.materialType,
|
||
// 不复制 itemType 和 itemId,让它们由 materialType 自动决定
|
||
itemType,
|
||
itemId: null,
|
||
grossWeight: parseFloat(this.currentInfo.grossWeight) || null,
|
||
netWeight: parseFloat(this.currentInfo.netWeight) || null,
|
||
warehouseId: this.currentInfo.warehouseId,
|
||
remark: this.currentInfo.remark
|
||
};
|
||
|
||
// materialType 会触发 watch,自动设置 itemType 并加载物品列表
|
||
this.$message.success('已复制当前信息');
|
||
},
|
||
|
||
// 保存更新
|
||
async handleSave() {
|
||
this.$refs.updateForm.validate(async (valid) => {
|
||
if (!valid) {
|
||
return false;
|
||
}
|
||
|
||
const loadingInstance = this.$loading({
|
||
lock: true,
|
||
text: '正在更新钢卷信息,请稍后...',
|
||
background: 'rgba(0, 0, 0, 0.7)'
|
||
})
|
||
|
||
try {
|
||
this.loading = true;
|
||
|
||
// 构造更新数据(使用标准update接口,直接更新原记录)
|
||
const updateData = {
|
||
...this.updateForm,
|
||
coilId: this.currentInfo.coilId,
|
||
enterCoilNo: this.currentInfo.enterCoilNo,
|
||
supplierCoilNo: this.currentInfo.supplierCoilNo,
|
||
// 注意:不要传newCoils,否则会走批量更新逻辑
|
||
};
|
||
|
||
const response = await updateMaterialCoil(updateData);
|
||
|
||
if (response.code === 200) {
|
||
this.$message.success('钢卷信息更新成功');
|
||
|
||
// 如果是从待操作列表进来的,标记操作为完成
|
||
if (this.actionId) {
|
||
await completeAction(this.actionId);
|
||
}
|
||
|
||
// 延迟返回
|
||
setTimeout(() => {
|
||
this.$router.back();
|
||
}, 100);
|
||
} else {
|
||
this.$message.error(response.msg || '更新失败');
|
||
}
|
||
} catch (error) {
|
||
this.$message.error('更新失败');
|
||
console.error(error);
|
||
} finally {
|
||
this.loading = false;
|
||
loadingInstance.close();
|
||
}
|
||
});
|
||
},
|
||
|
||
// 取消操作
|
||
handleCancel() {
|
||
this.$router.back();
|
||
}
|
||
}
|
||
};
|
||
</script>
|
||
|
||
<style scoped lang="scss">
|
||
.typing-coil-container {
|
||
padding: 20px;
|
||
background: #f5f7fa;
|
||
min-height: calc(100vh - 84px);
|
||
}
|
||
|
||
/* 顶部操作栏 */
|
||
.header-bar {
|
||
display: flex;
|
||
justify-content: space-between;
|
||
align-items: center;
|
||
background: #fff;
|
||
padding: 16px 20px;
|
||
margin-bottom: 20px;
|
||
border-radius: 4px;
|
||
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
|
||
}
|
||
|
||
.header-title {
|
||
font-size: 18px;
|
||
font-weight: 500;
|
||
color: #303133;
|
||
display: flex;
|
||
align-items: center;
|
||
gap: 8px;
|
||
|
||
i {
|
||
color: #0066cc;
|
||
font-size: 20px;
|
||
}
|
||
}
|
||
|
||
/* 主内容区 */
|
||
.content-wrapper {
|
||
display: grid;
|
||
grid-template-columns: 400px 1fr;
|
||
gap: 20px;
|
||
align-items: stretch; // 改为stretch,让子元素高度一致
|
||
}
|
||
|
||
/* 左侧面板 */
|
||
.left-panel {
|
||
min-width: 0;
|
||
display: flex;
|
||
flex-direction: column;
|
||
}
|
||
|
||
/* 右侧面板 */
|
||
.right-panel {
|
||
min-width: 0;
|
||
display: flex;
|
||
flex-direction: column;
|
||
}
|
||
|
||
/* 确保两侧卡片高度一致 */
|
||
.info-card,
|
||
.form-card {
|
||
flex: 1;
|
||
display: flex;
|
||
flex-direction: column;
|
||
|
||
::v-deep .el-card__body {
|
||
flex: 1;
|
||
display: flex;
|
||
flex-direction: column;
|
||
}
|
||
}
|
||
|
||
/* 变更历史区域(占满整行) */
|
||
.history-section {
|
||
margin-top: 20px;
|
||
}
|
||
|
||
/* 卡片头部 */
|
||
.card-header {
|
||
display: flex;
|
||
align-items: center;
|
||
justify-content: space-between;
|
||
font-weight: 500;
|
||
|
||
i {
|
||
color: #0066cc;
|
||
margin-right: 5px;
|
||
}
|
||
}
|
||
|
||
/* 当前信息展示 */
|
||
.info-section {
|
||
.info-row {
|
||
display: flex;
|
||
align-items: flex-start;
|
||
margin-bottom: 12px;
|
||
|
||
&:last-child {
|
||
margin-bottom: 0;
|
||
}
|
||
|
||
.info-label {
|
||
color: #909399;
|
||
font-size: 14px;
|
||
min-width: 110px;
|
||
flex-shrink: 0;
|
||
}
|
||
|
||
.info-value {
|
||
color: #303133;
|
||
font-size: 14px;
|
||
font-weight: 500;
|
||
flex: 1;
|
||
word-break: break-all;
|
||
}
|
||
}
|
||
}
|
||
|
||
/* 变更历史 */
|
||
.history-card {
|
||
::v-deep .el-card__body {
|
||
max-height: 400px;
|
||
overflow-y: auto;
|
||
|
||
&::-webkit-scrollbar {
|
||
width: 6px;
|
||
}
|
||
|
||
&::-webkit-scrollbar-thumb {
|
||
background: #dcdfe6;
|
||
border-radius: 3px;
|
||
}
|
||
}
|
||
|
||
::v-deep .el-timeline {
|
||
padding-left: 10px;
|
||
}
|
||
}
|
||
|
||
.history-item {
|
||
.history-title {
|
||
font-weight: 500;
|
||
color: #303133;
|
||
margin-bottom: 8px;
|
||
}
|
||
|
||
.history-detail {
|
||
font-size: 13px;
|
||
color: #606266;
|
||
margin-bottom: 4px;
|
||
|
||
.detail-label {
|
||
color: #909399;
|
||
margin-right: 5px;
|
||
}
|
||
}
|
||
}
|
||
|
||
.empty-history {
|
||
text-align: center;
|
||
padding: 40px 0;
|
||
color: #909399;
|
||
|
||
i {
|
||
font-size: 48px;
|
||
margin-bottom: 10px;
|
||
display: block;
|
||
}
|
||
|
||
p {
|
||
margin: 0;
|
||
}
|
||
}
|
||
|
||
/* 表单样式优化 */
|
||
.form-card {
|
||
::v-deep .el-input-number {
|
||
width: 100%;
|
||
|
||
.el-input__inner {
|
||
text-align: left;
|
||
}
|
||
}
|
||
|
||
::v-deep .el-form-item {
|
||
margin-bottom: 20px;
|
||
}
|
||
|
||
// 修复数字输入框的样式
|
||
::v-deep input[type="number"] {
|
||
appearance: textfield;
|
||
-moz-appearance: textfield;
|
||
|
||
&::-webkit-outer-spin-button,
|
||
&::-webkit-inner-spin-button {
|
||
-webkit-appearance: none;
|
||
appearance: none;
|
||
margin: 0;
|
||
}
|
||
}
|
||
}
|
||
|
||
// 优化按钮文字颜色
|
||
// 实心primary按钮:白色文字
|
||
::v-deep .el-button--primary.el-button--small:not(.is-plain) {
|
||
color: #fff;
|
||
}
|
||
|
||
// plain按钮和text按钮:蓝色文字
|
||
::v-deep .el-button--primary.el-button--small.is-plain,
|
||
::v-deep .el-button--text {
|
||
color: #409eff;
|
||
|
||
&:hover {
|
||
color: #66b1ff;
|
||
}
|
||
}
|
||
</style>
|