| <template> | 
|   <div class="scan-container"> | 
|     <el-card class="scan-card"> | 
|       <div slot="header" class="clearfix"> | 
|         <span>余料退库</span> | 
|       </div> | 
|       <el-form ref="form" :model="formData" label-width="120px" class="scan-form"> | 
|         <el-form-item label="仓库名称"> | 
|           <el-select v-model="formData.storage_Id" placeholder="请选择仓库" class="input-300" @change="getPositionName()"> | 
|             <el-option v-for="item in storageNames" :key="item.storage_Id" :label="item.storageName" :value="item.storage_Id"></el-option> | 
|           </el-select> | 
|         </el-form-item> | 
|         <el-form-item label="退货类型"> | 
|   | 
|           <el-select v-model="orderType" placeholder="请选择" class="input-300"> | 
|             <el-option | 
|               v-for="item in orderTypes" | 
|               :key="item.orderType" | 
|               :label="item.label" | 
|               :value="item.orderType"> | 
|             </el-option> | 
|           </el-select> | 
|   | 
|         </el-form-item> | 
|         <!-- <el-form-item label="收货区"> | 
|           <el-select ref="positionName" v-model="formData.positionName" placeholder="请选择收货位" class="input-300"> | 
|             <el-option v-for="(item, index) in positionList" :key="index" :label="item.positionName" :value="item.positionName"></el-option> | 
|           </el-select> | 
|         </el-form-item> --> | 
|         <el-form-item label="母托盘号"> | 
|           <el-input ref="plateCode" v-model="formData.plateCode" placeholder="请输入母托盘号" class="input-300" autofocus @keyup.native.enter.stop="getData"></el-input> | 
|   | 
|           <span class="sub-item"> | 
|             <el-button type="primary">呼叫库存托盘出库</el-button> | 
|           </span> | 
|         </el-form-item> | 
|         <el-form-item label="物料编号"> | 
|           <el-input ref="productCode" v-model="productCode" class="input-300" @keyup.native.enter.stop="checkPackingPlateCode"></el-input> | 
|           <span class="sub-item"> | 
|             <span class="sub-label">扫描数量:</span> | 
|             <el-input-number v-model="formData.scanQty" :min="1" class="input-100" controls-position="right" @change="setScanQty"></el-input-number> | 
|           </span> | 
|         </el-form-item> | 
|         <el-form-item> | 
|           <el-button type="primary" @click="saveCheck">确认码盘</el-button> | 
|           <el-button @click="onReset">重置</el-button> | 
|         </el-form-item> | 
|       </el-form> | 
|     </el-card> | 
|     <el-card class="scan-card body-no-padding"> | 
|       <div slot="header" class="clearfix"> | 
|         <span class="padding-top-10">码盘明细</span> | 
|         <el-button type="text" class="floatRight" @click="setting.visible=true">字段设置</el-button> | 
|       </div> | 
|       <el-table ref="scan-table" :data="tableData" :row-class-name="rowClass" stripe style="width: 100%" class="scan-table" @row-dblclick="setCurrent"> | 
|         <template v-for="(item, index) in setting.fields"> | 
|           <template v-if="'positionName,scanWeight,purchasePrice,batchNumber,quantity,finishedQuantity'.indexOf(item.prop)>=0"> | 
|             <el-table-column v-if="item.visible" :key="index" :prop="item.prop" :label="item.label" :width="item.width"> | 
|               <template slot-scope="scope"> | 
|                 <template> | 
|                   <el-input v-model="scope.row[item.prop]" size="mini" class="w-100pc"></el-input> | 
|                 </template> | 
|               </template> | 
|             </el-table-column> | 
|           </template> | 
|           <template v-else-if="'produceDate,limitDate'.indexOf(item.prop)>=0"> | 
|             <el-table-column v-if="item.visible" :key="index" :prop="item.prop" :label="item.label" :width="item.width"> | 
|               <template slot-scope="scope"> | 
|                 <template> | 
|                   <el-date-picker v-model="scope.row[item.prop]" size="mini" type="date" placeholder="选择日期" class="w-110" value-format="yyyy-MM-dd"> | 
|                   </el-date-picker> | 
|                 </template> | 
|               </template> | 
|             </el-table-column> | 
|           </template> | 
|           <template v-else-if="'isBoosterArm'.indexOf(item.prop)>=0"> | 
|             <el-table-column v-if="item.visible" :key="index" :prop="item.prop" :label="item.label" :width="item.width"> | 
|               <template slot-scope="scope"> | 
|                 <template> | 
|                   <el-switch v-model="scope.row[item.prop]" :active-value="1" :inactive-value="0"> | 
|                   </el-switch> | 
|                 </template> | 
|               </template> | 
|             </el-table-column> | 
|           </template> | 
|           <template v-else> | 
|             <el-table-column v-if="item.visible" :key="index" :prop="item.prop" :label="item.label" :width="item.width"> | 
|             </el-table-column> | 
|           </template> | 
|         </template> | 
|       </el-table> | 
|     </el-card> | 
|   | 
|     <el-dialog :visible.sync="dialogVisible" title="选择SKU" width="700px"> | 
|       <template> | 
|         <el-table :data="findProductList" class="scan-table" style="width: 100%" @row-click="rowClick" @row-dblclick="(row, column, event)=>{changeProduct(row)}"> | 
|           <el-table-column prop="productCode" label="物料编号" width="150"> | 
|           </el-table-column> | 
|           <!-- <el-table-column prop="productModel" label="条码" width="180"> | 
|           </el-table-column> --> | 
|           <el-table-column prop="productName" label="物料名称" width="150"> | 
|           </el-table-column> | 
|           <el-table-column prop="productSpec" label="物料规格"> | 
|           </el-table-column> | 
|           <el-table-column prop="smallUnit" label="库存单位"> | 
|           </el-table-column> | 
|           <el-table-column prop="validQty" label="有效库存"> | 
|           </el-table-column> | 
|           <el-table-column prop="inStorageDate" label="入库日期" width="130"> | 
|           </el-table-column> | 
|         </el-table> | 
|       </template> | 
|       <span slot="footer" class="dialog-footer"> | 
|         <el-button @click="dialogVisible = false">取 消</el-button> | 
|         <el-button type="primary" @click="changeProduct(multiProductSelected)">确 定</el-button> | 
|       </span> | 
|     </el-dialog> | 
|     <scan-setting-dialog ref="setting-dialog" :visible.sync="setting.visible" :fields="setting.fields" :name="setting.name"> | 
|     </scan-setting-dialog> | 
|   | 
|     <!--声音文件--> | 
|     <audio ref="sound_error" controls="controls" style="display:none;"> | 
|       <source src="@/assets/sounds/error2.mp3" type="audio/mpeg" /> | 
|     </audio> | 
|     <audio ref="sound_correct" controls="controls" style="display:none;"> | 
|       <source src="@/assets/sounds/feixin.mp3" type="audio/mpeg" /> | 
|     </audio> | 
|     <audio controls="controls" style="display:none;"> | 
|       <source ref="sound_scan" src="@/assets/sounds/saomiao.wav" type="audio/mpeg" /> | 
|     </audio> | 
|   | 
|   </div> | 
| </template> | 
|   | 
| <script> | 
| import YrtScanMixins from "@/components/common/yrtScanMixins.vue"; | 
| import ScanSettingDialog from "@/components/common/components/scanSettingDialog.vue"; | 
|   | 
| // import { jsonp } from "restify"; | 
|   | 
| export default { | 
|   name: "scan-purchase-noBillscan", | 
|   components: { | 
|     ScanSettingDialog | 
|   }, | 
|   mixins: [YrtScanMixins], | 
|   data() { | 
|     return { | 
|       formData: { | 
|         // 仓库ID | 
|         storage_Id: 87, | 
|         storageName: "立体库", | 
|         positionName: null, // 货位id | 
|         consignor_Id: null, | 
|         onsignorCode: null, | 
|         consignorName: null, | 
|         productModel: null, | 
|         scanQty: null, | 
|         provider_Id: "", | 
|         providerCode: "", | 
|         providerShortName: "", | 
|         plateCode: null, | 
|         isBoosterArm: 0, // 是否需要助力臂,0=否,1=是 | 
|         quantity: null, | 
|         productCode: null | 
|       }, | 
|       productCode: null, | 
|       tableDataSKU: [{}], | 
|       positionName: null, | 
|       // 供应商 | 
|       providerShortNames: null, | 
|       // 货主 | 
|       consignorNames: null, | 
|       // 仓库 | 
|       storageNames: null, | 
|       // 收货位候选项 | 
|       positionList: [], | 
|       // 是否展示SKU | 
|       showProduct: false, | 
|       dialogVisible: false, | 
|       // SKU列表 | 
|       findProductList: [], | 
|       // 明细数据 | 
|       tableData: [], | 
|   | 
|       // 仓库信息 | 
|       storageInfo: {}, | 
|   | 
|       // 当前正在扫描的数据  {} | 
|       currentRow: null, | 
|       // 已经找到的数据 | 
|       existRows: [], | 
|       // 装箱方式:0:常规扫描,1:一品一箱,2:多品一箱 | 
|       caseMode: 0, | 
|       // 是否弹出一码多拍 | 
|       isCount: false, | 
|       // 一次扫描的数量 | 
|       scanCount: 1, | 
|       // 装箱新增行 | 
|       caseNewRows: [], | 
|       // 配置参数 | 
|       config: { | 
|         // 自动生成上架单 | 
|         in_generateShelve: true, | 
|         // 是否启用装箱操作 | 
|         in_caseNumber: false, | 
|         // 支持一品多码 | 
|         sku_productToMultiBarcode: true, | 
|         caseMode: 0 | 
|       }, | 
|       // 多个物料时需要选择一个物料 | 
|       multiProductSelected: {}, | 
|       productPosition_Id: null, | 
|       // 退货类型 | 
|       orderTypes: [{ | 
|         orderType: "生产订单退料", | 
|         label: "生产订单退料" | 
|       }, { | 
|         orderType: "发料到成本中心", | 
|         label: "发料到成本中心" | 
|       }, { | 
|         orderType: "成本中心退料", | 
|         label: "成本中心退料" | 
|       }, { | 
|         orderType: "销售订单退库", | 
|         label: "销售订单退库" | 
|       }], | 
|       orderType: "生产订单退料", | 
|   | 
|       // 扫描列设置对话框参数 | 
|       setting: { | 
|         visible: false, | 
|         name: "scan-purchase-noBillscan", | 
|         fields: [ | 
|           { | 
|             prop: "productCode", | 
|             label: "物料编码", | 
|             visible: true, | 
|             width: 150, | 
|             order: 1 | 
|           }, | 
|           { | 
|             prop: "productName", | 
|             label: "物料名称", | 
|             visible: true, | 
|             width: 160, | 
|             order: 4 | 
|           }, | 
|           { | 
|             prop: "productStorage", | 
|             label: "原托盘存量", | 
|             visible: true, | 
|             width: 130, | 
|             order: 9 | 
|           }, | 
|           { | 
|             prop: "finishedQuantity", | 
|             label: "本次码盘数量", | 
|             visible: true, | 
|             width: 120, | 
|             order: 4 | 
|           }, | 
|           { | 
|             prop: "smallUnit", | 
|             label: "库存单位", | 
|             visible: true, | 
|             width: 130, | 
|             order: 10 | 
|           }, | 
|           { | 
|             prop: "isBoosterArm", | 
|             label: "是否需要助力臂", | 
|             visible: true, | 
|             width: 150, | 
|             order: 6 | 
|           }, | 
|           { | 
|             prop: "poCode", | 
|             label: "PO单号", | 
|             visible: true, | 
|             width: 130, | 
|             order: 6 | 
|           }, | 
|           { | 
|             prop: "trackingNumber", | 
|             label: "跟踪号", | 
|             visible: true, | 
|             width: 120, | 
|             order: 7 | 
|           }, | 
|           { | 
|             prop: "saleCode", | 
|             label: "销售单号", | 
|             visible: true, | 
|             width: 120, | 
|             order: 7 | 
|           } | 
|         ] | 
|       } | 
|     }; | 
|   }, | 
|   // 监听数据 | 
|   watch: { | 
|     // 当前行扫描数据改变后,将扫描数量也改变 | 
|     currentRow: { | 
|       handler: function(rowData) { | 
|         this.formData.scanQty = rowData.finishedQuantity; | 
|       }, | 
|       deep: true | 
|     } | 
|   }, | 
|   mounted() { | 
|     this.getStorageList(); | 
|     // 字段设置 | 
|     const setting = localStorage[this.setting.name + "-setting"]; | 
|     if (setting) { | 
|       this.setting.fields = JSON.parse(setting); | 
|     } | 
|     // 加载默认仓库的收货位 | 
|     this.getPositionName(); | 
|   }, | 
|   methods: { | 
|     // 获得明细数据 | 
|     getData() { | 
|       var plateCode = this.formData.plateCode; | 
|       if (!plateCode) { | 
|         this.$message.error("母托盘号不能为空!"); | 
|         return; | 
|       } | 
|       this.formData.plateCode = plateCode; | 
|       this.tableData = []; | 
|   | 
|       const url = "/api/outbound/shelveTk/getScanData"; | 
|       const params = { | 
|         plateCode: plateCode | 
|       }; | 
|       var callback = res => { | 
|         if (res.result) { | 
|           // 构建数据 | 
|           this.currentRow = {}; | 
|           this.existRows = []; | 
|           debugger; | 
|           this.tableData = res.data.map(row => { | 
|             debugger; | 
|             let unFinishedQuantity = row.validQty; | 
|             const finishedQuantity = 0; | 
|             // 不验证条码 | 
|             if (!this.formData.isValidateProductCode) { | 
|               unFinishedQuantity = 0; | 
|               // finishedQuantity = row.validQty; | 
|               row.scanWeight = row.totalWeight; | 
|             } | 
|             this.formData.storage_Id = row.storage_Id; | 
|             this.formData.storageName = row.storageName; | 
|             row.unFinishedQuantity = unFinishedQuantity; | 
|             row.finishedQuantity = finishedQuantity; | 
|             row.sortIndex = 0; | 
|             return row; | 
|           }); | 
|         } else { | 
|           this.$message.error(res.msg); | 
|         } | 
|       }; | 
|       this.common.ajax(url, params, callback, this.$refs["form"]); | 
|     }, | 
|     // 确认入库 | 
|     saveCheck() { | 
|       if (!window.confirm("当前扫描要确认码盘吗?")) return; | 
|   | 
|       var dataList = this.tableData.filter(rowData => rowData.quantityOrder > 0).map(rowData => { | 
|         return { | 
|           scanCount: rowData.finishedQuantity, | 
|           productModel: rowData.productModel, | 
|           productCode: rowData.productCode, | 
|           product_Id: rowData.product_Id, | 
|           plateCode: rowData.plateCode, | 
|           productName: rowData.productName, | 
|           quantityOrder: rowData.quantityOrder | 
|         }; | 
|       }); | 
|       if (!dataList.length) { | 
|         this.$message.error("请先扫描条码!"); | 
|         return; | 
|       } | 
|       var url = "/api/outbound/shelveTk/save"; | 
|       var params = { | 
|         masterData: this.formData, | 
|         detailList: dataList, | 
|         orderType: this.orderType | 
|       }; | 
|       var callBack = res => { | 
|         this.common.showMsg(res); | 
|         if (res.result) { | 
|           this.onReset(); | 
|         } | 
|       }; | 
|       this.common.ajax(url, params, callBack, true); | 
|     }, | 
|     // 获取仓库 | 
|     getStorageList() { | 
|       const url = "/api/basicInfo/base/storage/getList"; | 
|       const params = {}; | 
|       var callback = res => { | 
|         if (res.result) { | 
|           this.storageNames = res.data; | 
|         } | 
|       }; | 
|       this.common.ajax(url, params, callback, true); | 
|     }, | 
|     // 根据仓库获取货位 | 
|     getPositionName() { | 
|       var url = "/api/inbound/inScan/getPosition"; | 
|       // 仓库 | 
|       for (var index3 in this.storageNames) { | 
|         if (this.storageNames[index3].storage_Id === this.formData.storage_Id) { | 
|           this.formData.storage_Id = this.storageNames[index3].storage_Id; | 
|           this.formData.storageName = this.storageNames[index3].storageName; | 
|         } | 
|       } | 
|       var params = { | 
|         openNodeApi: true, | 
|         storage_Id: this.formData.storage_Id | 
|       }; | 
|       this.common.ajax( | 
|         url, | 
|         params, | 
|         res => { | 
|           if (res.result) { | 
|             this.storageInfo = res.data; | 
|             this.positionList = res.dynamic; | 
|             if (this.positionList.length) { | 
|               this.formData.positionName = this.positionList[0].positionName; | 
|             } | 
|           } else { | 
|             this.storageInfo = {}; | 
|             this.positionList = []; | 
|           } | 
|           // 收货位框获得焦点 | 
|           this.focus("plateCode"); | 
|         }, | 
|         true | 
|       ); | 
|     }, | 
|     // 获取所属物料编号的数据 | 
|     checkPackingPlateCode() { | 
|       debugger; | 
|       this.existRows = this.tableData.filter(item => { | 
|         return item.productCode === this.productCode; | 
|       }); | 
|       if (this.existRows.length <= 0) { | 
|         this.$message.error(this.productCode + "不存在!"); | 
|         this.$refs.sound_error.play(); // 播放声音 | 
|         return; | 
|       } | 
|       this.onPlateCodeChange(); | 
|     }, | 
|     // 当拍号存在改变所选数据数量 | 
|     onPlateCodeChange() { | 
|       this.tableData = this.tableData.map(row => { | 
|         if (row.finishedQuantity >= row.productStorage) { | 
|           this.$message({ | 
|             message: "扫描数量不可大于原托盘存量", | 
|             type: "warning" | 
|           }); | 
|         } else { | 
|           row.scanWeight = row.totalWeight; | 
|           row.finishedQuantity += 1; | 
|           row.unFinishedQuantity = row.quantityOrder - row.finishedQuantity; | 
|           row.sortIndex = 0; | 
|           this.formData.scanQty = row.finishedQuantity; | 
|         } | 
|         return row; | 
|       }); | 
|     }, | 
|     // 拍号回车 | 
|     plateCodeKeyup() { | 
|       // 条码框获得焦点 | 
|       this.$refs.productModel.focus(); | 
|       this.$refs.productModel.select(); | 
|     }, | 
|     // 选中行 | 
|     rowClick(row, column, event) { | 
|       this.multiProductSelected = row; | 
|     }, | 
|     // 选择物料 | 
|     changeProduct(row, column, event) { | 
|       // this.$message.error("选择物料"); | 
|       this.dialogVisible = false; | 
|       var positionName = this.formData.positionName; | 
|       var productModel = this.formData.productModel; | 
|       this.productPosition_Id = row.productPosition_Id; | 
|       var count = 1; | 
|       debugger; | 
|       if (this.isCount) { | 
|         for (var index in this.findProductList) { | 
|           if (this.findProductList[index].product_Id === row.product_Id) { | 
|             this.currentRow = this.findProductList[index]; | 
|             break; | 
|           } | 
|         } | 
|       } else { | 
|         for (var index1 in this.findProductList) { | 
|           if (this.findProductList[index1].product_Id === row) { | 
|             this.currentRow = this.findProductList[index1]; | 
|           } | 
|         } | 
|       } | 
|   | 
|       if (this.currentRow.middleBarcode === productModel) { | 
|         // 中包装条码 | 
|         count = this.currentRow.middleUnitConvert; | 
|       } else if (this.currentRow.bigBarcode === productModel) { | 
|         // 大包装条码 | 
|         count = this.currentRow.unitConvert; | 
|       } | 
|       this.$set(this.currentRow, "plateCode", this.formData.plateCode); | 
|       this.$set(this.currentRow, "positionName", positionName); | 
|       this.$set(this.currentRow, "finishedQuantity", count); | 
|       this.$set(this.currentRow, "quantity", count); | 
|       this.$set(this.currentRow, "scanWeight", count * this.currentRow.weight); | 
|       this.$set(this.tableData, this.tableData.length, this.currentRow); | 
|       this.existRows.push(this.currentRow); | 
|       this.sortRow(); | 
|       this.showProduct = false; | 
|     }, | 
|     // 排序高亮行靠前 | 
|     sortRow() { | 
|       this.tableData.forEach(element => { | 
|         element.sortIndex = 0; | 
|       }); | 
|       this.currentRow.sortIndex = 1; | 
|       this.tableData = this.tableData.sort(function(a, b) { | 
|         return b.sortIndex - a.sortIndex; | 
|       }); | 
|       this.tableData.forEach(element => { | 
|         element.sortIndex = 0; | 
|       }); | 
|     }, | 
|     onReset() { | 
|       this.formData = { | 
|         productModel: null, | 
|         scanQty: null, | 
|         provider_Id: "", | 
|         providerCode: "", | 
|         providerShortName: "", | 
|         storage_Id: 87, | 
|         storageName: "立体库" | 
|       }; | 
|       // 加载默认仓库的收货位 | 
|       this.getPositionName(); | 
|       this.tableData = []; | 
|     } | 
|   } | 
| }; | 
| </script> | 
|   | 
| <style lang="scss" scoped> | 
| @import "../../../styles/scan.scss"; | 
| </style> | 
| <style lang="css"> | 
| .row-active { | 
|   background-color: skyblue !important; | 
| } | 
| </style> |