You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

379 lines
12 KiB

<template>
<div>
<div class="pda-container">
<!-- 头部栏 -->
<div class="header-bar">
<div class="header-left" @click="handleBack">
<i class="el-icon-arrow-left"></i>
<span>Call料</span>
</div>
<div class="header-right" @click="$router.push({ path: '/' })">
首页
</div>
</div>
<!-- 搜索框 -->
<!-- <div class="search-container">-->
<!-- -->
<!-- </div>-->
<div class="table-body" style="max-height: 500px; overflow-y: auto;">
<div class="main-content form-section">
<!-- 表单区域 -->
<div class="input-group">
<label class="input-label">物料编码</label>
<el-input
v-model="formData.partNo"
placeholder="请输入物料编码"
class="form-input"
clearable
@blur="handlePartNoBlur"
/>
</div>
<!-- 栈板编码搜索框 - rqrq -->
<div class="input-group">
<label class="input-label">栈板编码</label>
<el-input
v-model="formData.palletId"
placeholder="请输入栈板编码"
class="form-input"
clearable
/>
</div>
<div class="input-group">
<label class="input-label">物料名称</label>
<el-input
v-model="formData.partDesc"
disabled
class="form-input"
/>
</div>
<div class="input-group">
<label class="input-label">批号</label>
<el-input
v-model="formData.batchNo"
class="form-input"
clearable
/>
</div>
<div class="bottom-actions" style="display: flex; gap: 10px; flex-wrap: nowrap;">
<button class="action-btn secondary" @click="confirmDo" style="flex: 1;">
查询
</button>
<button class="action-btn secondary" @click="cleanData" style="flex: 1;">
清空
</button>
<button
class="action-btn secondary"
@click="handleCall"
:disabled="callLoading||selectedPallets.length === 0"
style="flex: 1;"
>
{{ callLoading ? '下达中...' : `下达 (${selectedPallets.length})` }}
</button>
</div>
</div>
<!-- 栈板列表 -->
<div v-if="palletList.length > 0" class="rma-list">
<!-- Call车按钮 - 当有选中栈板时显示 -->
<div class="list-title" style="flex: 0.75">可用栈板列表</div>
<el-form>
<el-row v-for="(pallet, index) in palletList" :key="index" class="rma-row">
<el-col :span="24">
<div class="rma-item" @click="selectPallet(pallet)" :class="{ 'selected': isSelected(pallet) }">
<div class="item-info">
<span class="part-no">栈板号: {{ pallet.palletId }}</span>
<!-- rqrq - 如果有物料编码显示明细否则只显示栈板 -->
<span class="batch-qty" v-if="pallet.partNo">
物料: {{ pallet.partNo }} | 数量: {{ pallet.qty }}
</span>
<span class="batch-qty" v-if="pallet.partNo && pallet.batchNo">
| 批号: {{ pallet.batchNo }}
</span>
<span class="batch-qty" v-if="pallet.partNo && pallet.wdr">
| WDR: {{ pallet.wdr }}
</span>
<span class="batch-qty" v-if="pallet.partNo && pallet.expiredDate">
| 失效日期: {{ pallet.expiredDate }}
</span>
</div>
<div class="item-status">
<i class="el-icon-check" v-if="isSelected(pallet)"></i>
</div>
</div>
</el-col>
</el-row>
</el-form>
</div>
</div>
</div>
</div>
</template>
<script>
import {
saveTransportTask,
getShopOrderFromIFSWithOrderNo,
getPalletList,
callPalletFromWcs,
callPalletFromWcsNew
} from '../../../api/automatedWarehouse/callOut'
export default {
data() {
return {
scanRma: "",
rmaList: [],
returnList: [], // 退货明细列表
selectedDetail: null, // 当前选择的RMA明细
processType: "inbound", // 固定为入库处理
site:localStorage.getItem('site'),
loading: false, // 查询物料明细的loading状态
lastPartNo: '', // 记录最后查询的物料编码,避免重复查询
formData: {
palletId: '', // 栈板编码 - rqrq
partNo: '',
partDesc: '',
batchNo: '',
rollNo: '',
site:localStorage.getItem('site'),
},
palletList: [], // 栈板列表
selectedPallet: null, // 选中的栈板
selectedPallets: [], // 多选时选中的栈板列表
callLoading: false, // Call车按钮loading状态 - rqrq
};
},
computed: {
},
methods: {
handleBack() {
this.$router.back();
},
// 处理物料编码失去焦点事件
async handlePartNoBlur() {
const partNo = this.formData.partNo;
if (!partNo) {
this.formData.partDesc = '';
return;
}
// 如果物料编码没有变化,不重复查询
// if (this.lastPartNo === partNo) {
// return;
// }
this.loading = true;
getShopOrderFromIFSWithOrderNo({
site: this.formData.site,
partNo: partNo
}).then(({ data }) => {
this.loading = false;
if (data.code === 0) {
let rows=data.rows
if(rows.length>0){
this.formData.partDesc = rows[0].description || '';
}else {
this.formData.partDesc = '';
this.$alert('没有找到该物料编码', '错误', {
confirmButtonText: '确定',
})
}
} else {
// 查询失败或没有数据
this.formData.partDesc = '';
this.$alert(data.msg, '错误', {
confirmButtonText: '确定',
})
}
})
},
// 查询栈板列表 - rqrq 修改校验逻辑:物料编码和栈板码至少填一个
confirmDo() {
// 校验:物料编码和栈板码至少填一个 - rqrq
if((!this.formData.partNo || this.formData.partNo === '') &&
(!this.formData.palletId || this.formData.palletId === '')){
this.$message.error('请输入物料编码或栈板编码');
return;
}
// 构建查询参数 - rqrq
const queryParams = {
site: this.formData.site
};
// 添加栈板码到查询参数 - rqrq
if (this.formData.palletId && this.formData.palletId.trim()) {
queryParams.palletId = this.formData.palletId.trim();
}
// 添加物料编码到查询参数 - rqrq
if (this.formData.partNo && this.formData.partNo.trim()) {
queryParams.partNo = this.formData.partNo.trim();
}
// 如果有批号,添加到查询参数
if (this.formData.batchNo && this.formData.batchNo.trim()) {
queryParams.batchNo = this.formData.batchNo.trim();
}
getPalletList(queryParams).then(({ data }) => {
if (data && data.code === 0) {
this.palletList = data.rows || [];
//如果有些是用batchNo查的 这些字段要一起加进去 呼叫栈板时保存
if (this.palletList.length === 0) {
this.$message.warning('未找到满足条件的栈板');
} else {
this.$message.success(`找到 ${this.palletList.length} 个栈板`);
}
// 清空之前选中的栈板
this.selectedPallet = null;
this.selectedPallets = []; // 清空多选选中的栈板
} else {
this.$message.error(data.msg || '查询失败');
this.palletList = [];
this.selectedPallet = null;
this.selectedPallets = []; // 清空多选选中的栈板
}
}).catch(error => {
console.error('查询栈板列表失败:', error);
this.$message.error('查询失败');
this.palletList = [];
this.selectedPallet = null;
this.selectedPallets = []; // 清空多选选中的栈板
});
},
// 选择栈板 - rqrq 修改逻辑:按栈板ID选择,支持同一栈板多条明细
selectPallet(pallet) {
const palletId = pallet.palletId;
const index = this.selectedPallets.findIndex(item => item.palletId === palletId);
if (index > -1) {
// 如果已选中,取消选择(移除该栈板)- rqrq
this.selectedPallets.splice(index, 1);
this.$message.success('已取消选择栈板');
} else {
// 如果未选中,选择该栈板(只添加一条记录用于Call)- rqrq
this.selectedPallets.push({
site: pallet.site,
palletId: pallet.palletId,
partNo: pallet.partNo || '', // 如果没有物料,传空字符串
qty: pallet.qty || 0,
batchNo: pallet.batchNo || this.formData.batchNo || ''
});
this.$message.success(`已选择栈板: ${palletId}`);
}
},
// 判断栈板是否被选中 - rqrq
isSelected(pallet) {
return this.selectedPallets.some(item => item.palletId === pallet.palletId);
},
// 处理Call按钮点击 - rqrq
handleCall() {
if (this.selectedPallets.length === 0) {
this.$message.error('请至少选择一个栈板');
return;
}
this.$confirm(`确定Call ${this.selectedPallets.length} 个栈板出库?`, '提示', {
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning'
}).then(() => {
// 设置loading状态 - rqrq
this.callLoading = true;
// 调用新的接口 - rqrq
callPalletFromWcsNew(this.selectedPallets).then(({ data }) => {
if (data && data.code === 0) {
const failedCount = data.failedCount || 0;
const successCount = data.successCount || 0;
// 判断是否有失败 - rqrq
if (failedCount === 0) {
// 全部成功 - rqrq
this.$message.success(`成功Call ${successCount} 个栈板出库!`);
this.cleanData();
} else {
// 有失败的栈板 - rqrq
const failedPalletIds = data.failedPalletIds || [];
const failedReasons = data.failedReasons || [];
// 构建失败信息 - rqrq
let errorMsg = `成功:${successCount}个,失败:${failedCount}\n\n失败栈板:\n`;
failedPalletIds.forEach((palletId, index) => {
errorMsg += `${index + 1}. ${palletId}${failedReasons[index] || '未知错误'}\n`;
});
// 显示失败信息弹窗 - rqrq
this.$alert(errorMsg, `Call车结果`, {
confirmButtonText: '确定',
type: 'warning',
callback: () => {
// 点击确定后初始化页面 - rqrq
this.cleanData();
}
});
}
} else {
this.$message.error(data.msg || 'Call车失败');
}
}).catch(error => {
console.error('Call车失败:', error);
this.$message.error('Call车失败');
}).finally(() => {
// 恢复按钮状态 - rqrq
this.callLoading = false;
});
}).catch(() => {
// 用户取消操作 - rqrq
});
},
cleanData(){
// 更安全的清空方式 - rqrq
this.formData = {
palletId: '', // 栈板编码 - rqrq
partNo: '',
partDesc: '',
batchNo: '',
rollNo: '',
site:localStorage.getItem('site'),
};
// 清空栈板列表和选中状态
this.palletList = [];
this.selectedPallet = null;
this.selectedPallets = []; // 清空多选选中的栈板
},
},
mounted() {
}
};
</script>
<style scoped>
/* 按钮禁用状态样式 - rqrq */
.action-btn:disabled {
opacity: 0.6;
cursor: not-allowed;
background-color: #ccc !important;
border-color: #ccc !important;
}
</style>