|
|
@@ -8,7 +8,7 @@
|
|
|
<audio src="{{asset('sound/ding.mp3')}}" controls="controls" preload id="soundDing" hidden>
|
|
|
</audio>
|
|
|
<div class="row">
|
|
|
- <div class="col-4" id="countGoods">
|
|
|
+ <div class="col-4 p-0 m-0" id="countGoods">
|
|
|
<div class="col-12 text-center bg-white mt-2">
|
|
|
<h4 class="font-weight-bold text-dark">清点模式</h4>
|
|
|
</div>
|
|
|
@@ -50,18 +50,18 @@
|
|
|
</tr>
|
|
|
</table>
|
|
|
<hr>
|
|
|
- <span class="btn btn-outline-dark btn form-control" style="cursor: pointer" @click="submitExcelData">一键收货</span>
|
|
|
+ <span class="btn btn-outline-dark btn form-control" style="cursor: pointer" @click="submitExcelData">结束清点</span>
|
|
|
</div>
|
|
|
</div>
|
|
|
- <div class="col-8" id="receive">
|
|
|
+ <div class="col-8 p-0 m-0" id="receive">
|
|
|
<div class="col-12 text-center bg-white mt-2">
|
|
|
<h4 class="font-weight-bold text-dark">收货模式</h4>
|
|
|
</div>
|
|
|
- <div class="card-body">
|
|
|
+ <div class="card-body m-0">
|
|
|
<div class="col-8 offset-2 row">
|
|
|
<label for="asnno" class="font-weight-bold text-muted ">输入ASN</label>
|
|
|
- <input type="text" id="asnno" class="form-control" @focusin="focusOutDocument"
|
|
|
- @change="getReceiveTaskByAsnNoAndBarcodes()" placeholder="ASN单号" v-model="asnno">
|
|
|
+ <input type="text" id="asnno" class="form-control" @focusin="focusOutDocument" autocomplete="off"
|
|
|
+ @change="getReceiveTaskByAsnNoAndBarcodes()" disabled="true" placeholder="ASN单号" v-model="asnno">
|
|
|
</div>
|
|
|
<table class="table table-sm table-striped" v-if="regroupGoods.length>0">
|
|
|
<hr>
|
|
|
@@ -82,66 +82,89 @@
|
|
|
</tr>
|
|
|
<tr v-for="(goods,i) in regroupGoods">
|
|
|
<td class="text-center">@{{ goods.customerid }}</td>
|
|
|
- <td class="text-center">@{{ goods.barcode }}</td>
|
|
|
+ <td class="text-center">@{{ goods.sku }}</td>
|
|
|
<td class="text-center">@{{ goods.amount }}</td>
|
|
|
<td class="text-center">@{{ goods.asn_amount }}</td>
|
|
|
<td class="text-center">@{{ goods.diff_val }}</td>
|
|
|
|
|
|
<td class="text-center">
|
|
|
<span v-if="goods.receiveStatus">
|
|
|
- <input type="text" class="form-control form-control-sm" id="trackNumber" v-model="goods.trackNumber">
|
|
|
+ <input type="text" autocomplete="off" class="form-control form-control-sm" :class="errors.trackNumber ? 'is-invalid' : ''"
|
|
|
+ id="trackNumber" v-model="goods.trackNumber">
|
|
|
+ <span class="invalid-feedback offset-3" role="alert" v-if="errors.trackNumber">
|
|
|
+ <strong>@{{ errors.trackNumber[0] }}</strong>
|
|
|
+ </span>
|
|
|
</span>
|
|
|
</td>
|
|
|
|
|
|
<td class="text-center">
|
|
|
- <span v-if="basSku.lot_id">
|
|
|
- <input type="date" class="form-control form-control-sm input-sm"
|
|
|
- :disabled="goods.basSku.lot_id.lotkey01==='Y' ? '' : 'true'" id="lotatt01" v-model="goods.lotatt01">
|
|
|
+ <span v-if="goods.basSku.lot_id">
|
|
|
+ <input type="date" autocomplete="off" class="form-control form-control-sm input-sm" :class="errors.lotatt01 ? 'is-invalid' : ''"
|
|
|
+ :disabled="goods.basSku.lot_id.lotkey01==='N'" id="lotatt01" v-model="goods.lotatt01">
|
|
|
+ <span class="invalid-feedback offset-3" role="alert" v-if="errors.lotatt01">
|
|
|
+ <strong>@{{ errors.lotatt01[0] }}</strong>
|
|
|
+ </span>
|
|
|
</span>
|
|
|
</td>
|
|
|
<td class="text-center">
|
|
|
- <span v-if="basSku.lot_id">
|
|
|
- <input type="date" class="form-control form-control-sm input-sm"
|
|
|
- :disabled="goods.basSku.lot_id.lotkey02==='Y' ? '' : 'true'" id="lotatt02" v-model="goods.lotatt02">
|
|
|
+ <span v-if="goods.basSku.lot_id">
|
|
|
+ <input type="date" autocomplete="off" class="form-control form-control-sm input-sm" :class="errors.lotatt02 ? 'is-invalid' : ''"
|
|
|
+ :disabled="goods.basSku.lot_id.lotkey02==='N'" id="lotatt02" v-model="goods.lotatt02">
|
|
|
+ <span class="invalid-feedback offset-3" role="alert" v-if="errors.lotatt02">
|
|
|
+ <strong>@{{ errors.lotatt02[0] }}</strong>
|
|
|
+ </span>
|
|
|
</span>
|
|
|
</td>
|
|
|
<td class="text-center">
|
|
|
- <span v-if="basSku.lot_id">
|
|
|
- <input type="text" class="form-control form-control-sm input-sm"
|
|
|
- :disabled="goods.basSku.lot_id.lotkey04==='Y' ? '' : 'true'" id="lotatt04" v-model="goods.lotatt04">
|
|
|
+ <span v-if="goods.basSku.lot_id">
|
|
|
+ <input type="text" autocomplete="off" class="form-control form-control-sm input-sm" :class="errors.lotatt04 ? 'is-invalid' : ''"
|
|
|
+ :disabled="goods.basSku.lot_id.lotkey04==='N'" id="lotatt04" v-model="goods.lotatt04">
|
|
|
+ <span class="invalid-feedback offset-3" role="alert" v-if="errors.lotatt04">
|
|
|
+ <strong>@{{ errors.lotatt04[0] }}</strong>
|
|
|
+ </span>
|
|
|
</span>
|
|
|
</td>
|
|
|
<td class="text-center">
|
|
|
- <span v-if="basSku.lot_id">
|
|
|
- <select class="form-control form-control-sm" :disabled="goods.basSku.lot_id.lotkey05==='Y' ? '' : 'true'"
|
|
|
- id="lotatt05" v-model="goods.lotatt05">
|
|
|
+ <span v-if="goods.basSku.lot_id">
|
|
|
+ <select class="form-control form-control-sm" :disabled="goods.basSku.lot_id.lotkey05==='N'"
|
|
|
+ id="lotatt05" v-model="goods.lotatt05" :class="errors.lotatt05 ? 'is-invalid' : ''">
|
|
|
<option v-for="(attributeLocation,i) in attributeLocations"
|
|
|
:value="attributeLocation.code">@{{ attributeLocation.codename_c }}</option>
|
|
|
</select>
|
|
|
+ <span class="invalid-feedback offset-3" role="alert" v-if="errors.lotatt05">
|
|
|
+ <strong>@{{ errors.lotatt05[0] }}</strong>
|
|
|
+ </span>
|
|
|
</span>
|
|
|
</td>
|
|
|
<td class="text-center">
|
|
|
- <span v-if="basSku.lot_id">
|
|
|
- <select class="form-control form-control-sm"
|
|
|
- :disabled="goods.basSku.lot_id.lotkey08==='Y' ? '' : 'true'" id="lotatt08" v-model="goods.lotatt08">
|
|
|
+ <span v-if="goods.basSku.lot_id">
|
|
|
+ <select class="form-control form-control-sm" :class="errors.lotatt08 ? 'is-invalid' : ''"
|
|
|
+ :disabled="goods.basSku.lot_id.lotkey08==='N'" id="lotatt08" v-model="goods.lotatt08">
|
|
|
<option v-for="(quality,i) in qualityStatus" :value="quality.code">@{{ quality.codename_c }}</option>
|
|
|
</select>
|
|
|
+ <span class="invalid-feedback offset-3" role="alert" v-if="errors.lotatt08">
|
|
|
+ <strong>@{{ errors.lotatt08[0] }}</strong>
|
|
|
+ </span>
|
|
|
</span>
|
|
|
</td>
|
|
|
<td class="text-center">
|
|
|
- <span v-if="basSku.lot_id">
|
|
|
- <input type="date" class="form-control form-control-sm input-sm"
|
|
|
- :disabled="goods.basSku.lot_id.lotkey03==='Y' ? '' : 'true'" id="lotatt03" v-model="goods.lotatt03">
|
|
|
+ <span v-if="goods.basSku.lot_id">
|
|
|
+ <input type="date" autocomplete="off" class="form-control form-control-sm input-sm" :class="errors.lotatt03 ? 'is-invalid' : ''"
|
|
|
+ :disabled="goods.basSku.lot_id.lotkey03==='N'" id="lotatt03" v-model="goods.lotatt03">
|
|
|
+ <span class="invalid-feedback offset-3" role="alert" v-if="errors.lotatt03">
|
|
|
+ <strong>@{{ errors.lotatt03[0] }}</strong>
|
|
|
+ </span>
|
|
|
</span>
|
|
|
</td>
|
|
|
-
|
|
|
<td class="text-center">
|
|
|
- <span v-if="goods.receiveStatus">
|
|
|
- <button class="btn btn-outline-info btn-sm" @click="receiveGoods(i,goods.barcode)">收货</button>
|
|
|
+ <span v-if="goods.receiveStatus&&goods.diff_val>=0">
|
|
|
+ <button class="btn btn-outline-info btn-sm" @click="receiveGoods(goods)">收</button>
|
|
|
</span>
|
|
|
</td>
|
|
|
</tr>
|
|
|
</table>
|
|
|
+ <span class="btn btn-outline-dark btn form-control mt-5" v-if="status.finishReceiveButton"
|
|
|
+ style="cursor: pointer" @click="finishReceive()">结束收货</span>
|
|
|
</div>
|
|
|
</div>
|
|
|
</div>
|
|
|
@@ -162,12 +185,16 @@
|
|
|
barcode:'',amount:'',fromIncreasing:true,cast_number:'',countGoodStatus:true
|
|
|
},
|
|
|
status:{
|
|
|
- scanEndInputted:false,barcodeDisable:true,amountDisable:true,
|
|
|
+ scanEndInputted:false,barcodeDisable:true,amountDisable:true,finishReceiveButton:false,
|
|
|
},
|
|
|
asnno:'',
|
|
|
- basSku:{},
|
|
|
+ basSku:{
|
|
|
+ lot_id:'',
|
|
|
+ },
|
|
|
goodses:[],
|
|
|
+ initGoods:[],
|
|
|
regroupGoods:[],
|
|
|
+ errors:{},
|
|
|
},
|
|
|
mounted() {
|
|
|
if (navigator.userAgent.indexOf("Android")!==-1)this.isAndroid = true;
|
|
|
@@ -327,50 +354,95 @@
|
|
|
},
|
|
|
assignRegroupGoods(){
|
|
|
let data=this;
|
|
|
- data.regroupGoods=data.regroupArr(data.goodses);
|
|
|
- data.regroupGoods.forEach(function (obj){
|
|
|
+ data.initGoods=data.regroupArr(data.goodses);
|
|
|
+ data.initGoods.forEach(function (obj){
|
|
|
delete obj.fromIncreasing;
|
|
|
delete obj.cast_number;
|
|
|
delete obj.countGoodStatus;
|
|
|
- obj.customerid='';
|
|
|
- obj.asn_amount='';
|
|
|
- obj.diff_val='';
|
|
|
- obj.receiveStatus=false;
|
|
|
- obj.basSku='';
|
|
|
+ obj.asnno='';obj.customerid='';
|
|
|
+ obj.location='';obj.asn_amount='';
|
|
|
+ obj.diff_val='';obj.receiveStatus=false;
|
|
|
+ obj.basSku=data.basSku;obj.asnlineno='';
|
|
|
obj.lotatt01='';obj.lotatt02='';
|
|
|
obj.lotatt03='';obj.lotatt04='';
|
|
|
obj.lotatt05='';obj.lotatt08='';
|
|
|
})
|
|
|
+ data.initGoods=JSON.parse(JSON.stringify(data.initGoods).replace(/barcode/g,"sku"));
|
|
|
+ data.regroupGoods=data.initGoods;
|
|
|
},
|
|
|
- receiveGoods(){
|
|
|
+ receiveGoods(goods){
|
|
|
let data=this;
|
|
|
- if(data.asnno===''||data.asnno===null||data.asnno===undefined) window.tempTip.show('请先输入ASN单号!');
|
|
|
- if (data.asnno && data.asnno.indexOf('ASN')===-1)window.tempTip.show('无效ASN号!');
|
|
|
-
|
|
|
-
|
|
|
+ this.errors={};
|
|
|
+ this.checkGood(goods);
|
|
|
+ if (JSON.stringify(this.errors)!=='{}')return;
|
|
|
+ goods.asnno=data.asnno;
|
|
|
+ tempTip.setDuration(99999);
|
|
|
+ tempTip.waitingTip('提交中');
|
|
|
+ let url='{{url("store/countGoodsAndReceive/fluxReceive")}}';
|
|
|
+ axios.post(url,{good:goods})
|
|
|
+ .then(function(res){
|
|
|
+ if(res.data.success){
|
|
|
+ data.regroupGoods.forEach(function (good,i){
|
|
|
+ if (goods.sku==good.sku){
|
|
|
+ data.regroupGoods.splice(i,1)
|
|
|
+ }
|
|
|
+ })
|
|
|
+ tempTip.setDuration(2000);
|
|
|
+ tempTip.cancelWaitingTip();
|
|
|
+ tempTip.showSuccess(res.data.data);
|
|
|
+ }else{
|
|
|
+ tempTip.setDuration(3000);
|
|
|
+ tempTip.cancelWaitingTip();
|
|
|
+ tempTip.show(res.data.data);
|
|
|
+ data.alertVibrate()
|
|
|
+ }
|
|
|
+ })
|
|
|
+ .catch(function (err) {
|
|
|
+ tempTip.setDuration(2000);
|
|
|
+ tempTip.cancelWaitingTip();
|
|
|
+ tempTip.show("网络错误:"+err);
|
|
|
+ data.alertVibrate()
|
|
|
+ });
|
|
|
+ },
|
|
|
+ checkGood(good){
|
|
|
+ let error = {};
|
|
|
+ if (!good.trackNumber)error.trackNumber = ["容器号必填"];
|
|
|
+ if (good.basSku.lot_id && good.basSku.lot_id.lotkey01==='Y' && !good.lotatt01) error.lotatt01=["生产日期为选"];
|
|
|
+ if (good.basSku.lot_id && good.basSku.lot_id.lotkey02==='Y' && !good.lotatt02) error.lotatt02=["失效日期为选"];
|
|
|
+ if (good.basSku.lot_id && good.basSku.lot_id.lotkey03==='Y' && !good.lotatt03) error.lotatt03=["入库日期为选"];
|
|
|
+ if (good.basSku.lot_id && good.basSku.lot_id.lotkey04==='Y' && !good.lotatt04) error.lotatt04=["批号未填"];
|
|
|
+ if (good.basSku.lot_id && good.basSku.lot_id.lotkey05==='Y' && !good.lotatt05) error.lotatt05=["属性仓未选"];
|
|
|
+ if (good.basSku.lot_id && good.basSku.lot_id.lotkey08==='Y' && !good.lotatt08) error.lotatt08=["质量状态未选"];
|
|
|
+ if (JSON.stringify(error)!=='{}'){this.errors = error;}
|
|
|
},
|
|
|
getReceiveTaskByAsnNoAndBarcodes(){
|
|
|
let data=this;
|
|
|
+ data.regroupGoods=data.initGoods;//验证asn下商品时 初始商品数据
|
|
|
if(data.asnno===''||data.asnno===null||data.asnno===undefined) window.tempTip.show('请先输入ASN单号!');
|
|
|
if (data.asnno && data.asnno.indexOf('ASN')===-1)window.tempTip.show('无效ASN号!');
|
|
|
let url='{{url("store/countGoodsAndReceive/getReceiveTaskByAsnNoAndBarcodes")}}';
|
|
|
- window.axios.post(url,{asnno:data.asnno,goods:data.regroupGoods})
|
|
|
+ window.axios.post(url,{asnno:data.asnno,goods:data.initGoods})
|
|
|
.then(res=>{
|
|
|
if (res.data.success){
|
|
|
data.regroupGoods=res.data.data;
|
|
|
this.$forceUpdate()
|
|
|
- console.log(data.regroupGoods)
|
|
|
}else {
|
|
|
-
|
|
|
+ tempTip.setDuration(3000);
|
|
|
+ tempTip.show(res.data.data);
|
|
|
}
|
|
|
}).catch(err=>{
|
|
|
window.tempTip.setDuration(2000);
|
|
|
window.tempTip.show("网络错误:"+err);
|
|
|
})
|
|
|
},
|
|
|
+ removeDisable(){
|
|
|
+ let asnNo=$('#asnno');
|
|
|
+ asnNo.removeAttr('disabled');
|
|
|
+ this.status.finishReceiveButton=true;
|
|
|
+ asnNo.focus();
|
|
|
+ },
|
|
|
submitExcelData: function () {
|
|
|
let data = this;
|
|
|
- data.assignRegroupGoods();
|
|
|
data.focusOutDocument();
|
|
|
if(data.goodses.length===0){
|
|
|
window.tempTip.show('请先录入数据再提交收货');return;
|
|
|
@@ -396,17 +468,16 @@
|
|
|
axios.post(url,{'goodses':data.goodses,'filename':val})
|
|
|
.then(function(response){
|
|
|
if(response.data.result==='success'){
|
|
|
- data.assignRegroupGoods();//统计清点信息,并生成收货所需数据
|
|
|
+ data.removeDisable();//收货解锁asnno input标签
|
|
|
+ data.assignRegroupGoods();//结束清点,重组收货所需数据
|
|
|
data.goodses=[];
|
|
|
data.cleanInputs();
|
|
|
window.tempTip.cancelWaitingTip();
|
|
|
window.tempTip.setDuration(1500);
|
|
|
window.tempTip.showSuccess('成功生成EXCEL,可在列表页查看');
|
|
|
- data.focusDocument();
|
|
|
}else{
|
|
|
window.tempTip.setDuration(1500);
|
|
|
- window.tempTip.show('生成EXCEL失败:'+response.data.fail_info);
|
|
|
- console.log(response);
|
|
|
+ window.tempTip.show('生成EXCEL失败');
|
|
|
data.focusDocument();
|
|
|
data.alertVibrate()
|
|
|
}
|
|
|
@@ -422,6 +493,10 @@
|
|
|
},function () {
|
|
|
data.focusDocument();
|
|
|
})
|
|
|
+ },
|
|
|
+ finishReceive(){
|
|
|
+ if(!confirm('确定要结束收货吗?'))return;
|
|
|
+ setTimeout(function (){window.location.reload()},10)
|
|
|
}
|
|
|
},
|
|
|
});
|