浏览代码

追溯码

hd 1 周之前
父节点
当前提交
903f9fc195

+ 1 - 0
mini-pro-web/package.json

@ -24,6 +24,7 @@
    "vue": "^2.6.11",
    "vue-i18n": "^8.24.2",
    "vue-router": "^3.2.0",
    "vuedraggable": "^2.24.3",
    "vuex": "^3.4.0"
  },
  "devDependencies": {

+ 7 - 0
mini-pro-web/src/api/api-medicineAbinet.js

@ -751,6 +751,13 @@ let service = {
      method: 'post',
      data
    })
  },
  drugTraceabilityCodeDetail(params){
    return request({
      url: `${BASE}/baseDrug/drugTraceabilityCodeDetail`,
      method: 'get',
      params
    })
  }
}

二进制
mini-pro-web/src/assets/images/traceability/del.png


二进制
mini-pro-web/src/assets/images/traceability/drag.png


二进制
mini-pro-web/src/assets/images/traceability/scan.png


+ 17 - 0
mini-pro-web/src/assets/styles/common.scss

@ -1016,4 +1016,21 @@ input:-moz-placeholder { color:#999999; }
}
@for $fs from 12 through 40 {
    .fs-#{$fs} {font-size: 1px*$fs;}
}
.flex{
    display: flex;
}
.align-center{
    display: flex;
    align-items: center;
}
.justify-center{
    display: flex;
    justify-content: center;
}
.justify-between{
    display: flex;
    justify-content: space-between;
}

+ 0 - 3
mini-pro-web/src/components/layout/NavLayout.vue

@ -55,9 +55,6 @@ export default {
		onClickLeft(){
			this.goBack()
		}
	},
	mounted(){
		console.log(this.$navigation.getRoutes())
	}
};
</script>

+ 0 - 1
mini-pro-web/src/directive/longpress.js

@ -5,7 +5,6 @@ export default {
    }) {
      Vue.directive('longpress', {
        bind: function(el, binding, vNode) {
            console.log("longpress")
          // 确保提供的表达式是函数
          if (typeof binding.value !== 'function') {
            // 获取组件名称

+ 0 - 2
mini-pro-web/src/router/index.js

@ -189,7 +189,6 @@ router.beforeEach(async (to, from, next) => {
                                routerLoaded = true
                                var routeList = formatAuthMenu(res.detailModelList)
                                store.dispatch('SetAuthMenu', routeList)
                                console.log('routeList', routeList)
                                next()
                            })
                            .catch(err=>{
@ -307,7 +306,6 @@ function authLogin(code, to){
const noSetTitleList = []
router.afterEach((to, from) => {
    // 更新页面标题
    console.log(to,from)
    if (to.meta && to.meta.title && noSetTitleList.indexOf(to.path)==-1) {
        setTitle(to.meta.title)
        store.dispatch("SetPageTitle", to.meta.title)

+ 9 - 0
mini-pro-web/src/router/replenishment.js

@ -178,6 +178,15 @@ const routeMap = [
                },
				component: () => import("@/views/replenishment/start/selProduct.vue")
            },
            {
                path: '/replenishment/start/traceability',
                name: 'replenishmentStartTraceability',
                meta: {
                    title: '药品追溯码',
                    hasBot: false
                },
				component: () => import("@/views/replenishment/start/traceability.vue")
            }
		]
    }
];

+ 0 - 2
mini-pro-web/src/views/device/detail.vue

@ -208,7 +208,6 @@ export default {
        return v.url == 'operate'
      }
    ).children
    console.log('this.menuAuth', this.menuAuth)
  },
  methods: {
    openLock() {
@ -244,7 +243,6 @@ export default {
      medicineAbinetApi
        .findDeviceById(p)
        .then(res => {
          console.log(' res.obj.device', res.obj.device)
          this.$toast.clear()
          this.info = res.obj.device
        })

+ 146 - 109
mini-pro-web/src/views/drug/list.vue

@ -1,131 +1,168 @@
<template>
    <div class="drug-list">
        <van-sticky :offset-top="offsetTop">
            <div class="search">
                <form action="/">
                    <van-search
                        v-model="value"
                        placeholder="至少输入两个关键词进行搜索"
                        show-action
                        @search="getDrugDictionary"
                        @cancel="onCancel"
                        autofocus
                    />
                </form>
            </div>
        </van-sticky>
        <div class="list" v-if="list.length">
            <div 
                v-for="(item, i) in list" :key="i"
                @click="addDrug(item)"
                class="item mb10 bgc-fff btb-e1e1e1 pt10 pb10 plr15">
                <van-row class="lh25 c-333">
                    <span class="fs-17">{{item.drugName}}</span>
                    <span class="fs-14 ml20">药品编码:{{item.drugCode}}</span>
                </van-row>
                <van-row class="lh25">
                    <van-col span="12" class="c-999">规格:<span class="c-333">{{item.specif}}</span></van-col>
                    <van-col span="12" class="c-999">分类:<span class="c-333">{{item.drugClass}}</span></van-col>
                </van-row>
                <van-row class="lh25">
                    <van-col span="24" class="c-999">剂型:<span class="c-333">{{item.dosForm}}</span></van-col>
                    <!-- <van-col span="12" class="c-999">医保支付比例:<span class="c-333"></span></van-col> -->
                </van-row>
                <!-- <van-row class="lh25">
  <div class="drug-list">
    <van-sticky :offset-top="offsetTop">
      <div class="search">
        <form action="/">
          <van-search v-model="value" placeholder="至少输入两个关键词进行搜索" show-action @search="getDrugDictionary" @cancel="onCancel" autofocus />
        </form>
      </div>
    </van-sticky>
    <div class="list" v-if="list.length">
      <div v-for="(item, i) in list" :key="i" @click="addDrug(item)" class="item mb10 bgc-fff btb-e1e1e1 pt10 pb10 plr15">
        <van-row class="lh25 c-333">
          <span class="fs-17">{{ item.drugName }}</span>
          <span class="fs-14 ml20">药品编码:{{ item.drugCode }}</span>
        </van-row>
        <van-row class="lh25">
          <van-col span="12" class="c-999">
            规格:
            <span class="c-333">{{ item.specif }}</span>
          </van-col>
          <van-col span="12" class="c-999">
            分类:
            <span class="c-333">{{ item.drugClass }}</span>
          </van-col>
        </van-row>
        <van-row class="lh25">
          <van-col span="24" class="c-999">
            剂型:
            <span class="c-333">{{ item.dosForm }}</span>
          </van-col>
          <!-- <van-col span="12" class="c-999">医保支付比例:<span class="c-333"></span></van-col> -->
        </van-row>
        <!-- <van-row class="lh25">
                    <van-col span="12" class="c-999">基药:<span class="c-333"></span></van-col>
                    <van-col span="12" class="c-999">药品类别:<span class="c-333"></span></van-col>
                </van-row> -->
                <!-- <div class="c-ff5e6c fs-14 lh25">高危药品:</div> -->
            </div>
        </div>
        <div v-else class="c-999 fs-14 tc pt20">
            <div v-if="value">搜索不到数据~~</div>
            <div v-else>请输入关键词进行搜索</div>
        </div>
        <!-- <div class="c-ff5e6c fs-14 lh25">高危药品:</div> -->
      </div>
    </div>
    <div v-else class="c-999 fs-14 tc pt20">
      <div v-if="value">搜索不到数据~~</div>
      <div v-else>请输入关键词进行搜索</div>
    </div>
    <div class="replenishment-start-set-stock-dialog  ">
      <van-popup v-model="setStockShow" close-on-popstate :close-on-click-overlay="false">
        <div class="plr15 ptb10 c-pr" style="width: 80vw">
          <div class="close fs-20"><van-icon @click="setStockShow = false" name="close" /></div>
          <div class="tc fs-14 c-333">
            <span>设置库存</span>
          </div>
          <div class="tc pt30 pb20"><van-stepper v-model="stock" min="1" max="99" button-size="36px" input-width="60px" /></div>
        <SetStockDialog 
            v-model="setStockShow" 
            :trackId="trackId"
            :bussiness="bussiness"
            :drugId="drug.id"
            @onConfirm="onConfirm"/>
            
          <div class="ptb10">
            <van-button type="info" @click="onConfirm" size="small" block>确认</van-button>
          </div>
        </div>
      </van-popup>
    </div>
    <!-- <SetStockDialog v-model="setStockShow" :trackId="trackId" :bussiness="bussiness" :drugId="drug.id" @onConfirm="onConfirm" /> -->
  </div>
</template>
<script>
import medicineAbinetApi from '@/api/api-medicineAbinet'
import SetStockDialog from '@/views/replenishment/start/components/SetStockDialog'
export default {
    name: "DrugSel",
    components: {
        SetStockDialog
    },
    watch: {
        value(n, o){
            this.getDrugDictionary()
  name: 'DrugSel',
  components: {
    SetStockDialog
  },
  watch: {
    value(n, o) {
      this.getDrugDictionary()
    }
  },
  data() {
    return {
      deviceId: this.$route.query.deviceId, //轨道id
      trackId: this.$route.query.trackId, //轨道id
      value: '',
      list: [],
      setStockShow: false,
      bussiness: this.$route.query.bussiness || '',
      drug: '',
      stock: 0
    }
  },
  async created() {
    this.bussiness = this.trackId ? 5 : this.bussiness
    this.getDrugDictionary()
  },
  methods: {
    getDrugDictionary() {
      var val = this.value
      if (val && val.length >= 2) {
        var p = {
          content: this.value,
          userId: this.user.id,
          deviceId: this.deviceId,
          page: 1,
          size: 999
        }
        medicineAbinetApi
          .getDrugListByDevice(p)
          .then(res => {
            if (res.status == 200) {
              this.list = res.detailModelList
            }
          })
          .catch(err => {
            console.error(err)
          })
      }
    },
    data(){
        return {
            deviceId: this.$route.query.deviceId,//轨道id
            trackId: this.$route.query.trackId,//轨道id
            value: "",
            list: [],
            setStockShow: false,
            bussiness: this.$route.query.bussiness || '',
            drug: ''
        }
    onCancel() {
      this.$router.back()
    },
    async created(){
        this.bussiness = this.trackId? 5 : this.bussiness
        this.getDrugDictionary()
    addDrug(item) {
      if (this.trackId || this.bussiness == 6) {
        this.drug = item
        this.setStockShow = true
        return
      }
      this.$EventBus.$emit('drug-sel', item)
      this.$router.back()
    },
    methods: {
        getDrugDictionary(){
            var val = this.value
			if(val&&val.length>=2){
                var p = {
                    content: this.value,
                    userId: this.user.id,
                    deviceId: this.deviceId,
                    page: 1,
                    size: 999
                }
                console.log('params', p)
                medicineAbinetApi
                    .getDrugListByDevice(p)
                    .then(res => {
                        if(res.status == 200){
                            console.log('drugList', res)
                            this.list = res.detailModelList
                        }
                    })
                    .catch(err=>{
                        console.error(err)
                    })
			}
        },
        onCancel(){
            this.$router.back()
        },
        addDrug(item){
            if(this.trackId || this.bussiness==6){
                this.drug = item
                this.setStockShow = true
                return
            }
            this.$EventBus.$emit('drug-sel', item)
            this.$router.back()
        },
        onConfirm(stock){
            this.drug.upInventory = stock
            this.$EventBus.$emit('drug-sel', this.drug)
            this.$router.back()
    onConfirm() {
      this.$router.push({
        path: '/replenishment/start/traceability',
        query: {
          deviceId: this.deviceId,
          trackId: this.trackId,
          drugId: this.drug.id,
          stock: this.stock,
          type: 2
        }
      })
      //   this.drug.upInventory = stock
      //   this.$EventBus.$emit('drug-sel', this.drug)
      //   this.$router.back()
    }
};
  }
}
</script>
<style lang="scss" scoped>
@import './css/drugSel.scss';
.replenishment-start-set-stock-dialog {
  .close {
    position: absolute;
    right: 10px;
    top: 10px;
  }
  ::v-deep .van-field {
    padding: 0;
    .van-cell__value {
      border: 1px solid #e1e1e1;
      input {
        text-align: center;
      }
    }
  }
  .van-image {
    width: 200px;
    height: 150px;
  }
}
</style>

+ 168 - 123
mini-pro-web/src/views/replenishment/start/components/SetStockDialog.vue

@ -1,136 +1,181 @@
<template>
    <div class='replenishment-start-set-stock-dialog  '>
        <van-popup v-model="value" close-on-popstate :close-on-click-overlay="false">
            <div class="plr15 ptb10 c-pr" style="width: 80vw">
                <div class="close fs-20"><van-icon @click="emitValue(false)" name="close" /></div>
                <div class="tc fs-14 c-333">
                    <span>{{bussiness==1? '修改容量' : bussiness==2? '矫正库存' : bussiness==4||bussiness==5? '设置库存' : bussiness==6? '新增' : ''}}</span>
                </div>
                <template v-if="bussiness==1||bussiness==4||bussiness==5||bussiness==6">
                    <div class="tc pt30 pb20"><van-stepper v-model="stock" min="1" max="99" button-size="36px" input-width="60px"/></div>
                </template>
                <div class="tc" v-else>
                    <div class="pt10">
                        <van-image
                            radius="5px"
                            :src="require('@/assets/images/drug_default.png')"
                        />
                    </div>
                    <div class="pt10 c-333 fs-16">{{data.drugName}}</div>
                    <div class="pt10 c-999">软件显示库存:{{data.qty}}</div>
                    <div class="pt10 pb20 ipt plr50">
                        <div class="tc  ">
                            <van-stepper 
                                placeholder="实际库存"
                                v-model="stock" 
                                min="1" 
                                max="99" 
                                button-size="36px" 
                                input-width="60px"/>
                            </div>
                    </div>
                </div>
                <div class="ptb10">
                    <van-button type="info" @click="confirm" size="small" block>确认</van-button>
                </div>
  <div class="replenishment-start-set-stock-dialog  ">
    <van-popup v-model="value" close-on-popstate :close-on-click-overlay="false">
      <div class="plr15 ptb10 c-pr" style="width: 80vw">
        <div class="close fs-20"><van-icon @click="emitValue(false)" name="close" /></div>
        <div class="tc fs-14 c-333">
          <span>{{ bussiness == 1 ? '修改容量' : bussiness == 2 ? '矫正库存' : bussiness == 4 || bussiness == 5 ? '设置库存' : bussiness == 6 ? '新增' : '' }}</span>
        </div>
        <template v-if="bussiness == 1 || bussiness == 4 || bussiness == 5 || bussiness == 6">
          <div class="tc pt30 pb20"><van-stepper v-model="stock" min="1" max="99" button-size="36px" input-width="60px" /></div>
        </template>
        <div class="tc" v-else>
          <div class="pt10">
            <van-image radius="5px" :src="require('@/assets/images/drug_default.png')" />
          </div>
          <div class="pt10 c-333 fs-16">{{ data.drugName }}</div>
          <div class="pt10 c-999">软件显示库存:{{ data.qty }}</div>
          <div class="pt10 pb20 ipt plr50">
            <div class="tc  ">
              <van-stepper placeholder="实际库存" v-model="stock" min="1" max="99" button-size="36px" input-width="60px" />
            </div>
        </van-popup>
    </div>
          </div>
        </div>
        <div>
          <div class="title">
            <span>药品追溯码(</span>
            <span class="c-17b3ec">{{ 0 }}</span>
            <span>/{{ stock }})</span>
          </div>
          <div class="code-box">
            <draggable v-model="items" :group="{ name: 'items' }" :animation="200" handle=".drag">
              <div class="flex justify-between align-center mb10" style="position: relative;">
                <van-field v-model="keyword" label="第5位次" placeholder="请扫描或输入追溯码" class="mr8" />
                <img src="@/assets/images/traceability/scan.png" class="scan" alt="" v-if="data.qty <= stock" />
                <img src="@/assets/images/traceability/drag.png" class="drag" alt="" v-if="data.qty <= stock" />
                <img src="@/assets/images/traceability/del.png" alt="" v-if="data.qty > stock" @click="delDrug" />
              </div>
              <div class="flex justify-between align-center" style="position: relative;">
                <van-field v-model="keyword" label="第5位次" placeholder="请扫描或输入追溯码" class="mr8" />
                <img src="@/assets/images/traceability/scan.png" class="scan" alt="" v-if="data.qty <= stock" />
                <img src="@/assets/images/traceability/drag.png" class="drag" alt="" v-if="data.qty <= stock" />
                <img src="@/assets/images/traceability/del.png" alt="" v-if="data.qty > stock" @click="delDrug()" />
              </div>
            </draggable>
          </div>
        </div>
        <div class="ptb10">
          <van-button type="info" @click="confirm" size="small" block>确认</van-button>
        </div>
      </div>
    </van-popup>
  </div>
</template>
<script>
import draggable from 'vuedraggable'
import medicineAbinetApi from '@/api/api-medicineAbinet'
export default{
    props:{
        value: {},
        bussiness: {
            default: 1 //1修改容量 2矫正库存 4设置库存 5选择药品后 设置库存  6备药时选择药品
        },
        data: {
            default: function(){
                return {}
            }
        },
        trackId: {default: ""},
        drugId: {default: ""}
export default {
  components: {
    draggable
  },
  props: {
    value: {},
    bussiness: {
      default: 1 //1修改容量 2矫正库存 4设置库存 5选择药品后 设置库存  6备药时选择药品
    },
    data(){
        return {
            stock: 1
        }
    data: {
      default: function() {
        return {}
      }
    },
    watch:{
        value:{
            handler(){
                this.stock = this.bussiness==1? this.data.cargoCapacity : this.data.qty 
            },
            immediate: true
        }
    trackId: { default: '' },
    drugId: { default: '' }
  },
  data() {
    return {
      stock: 1,
      keyword: '',
      items: []
    }
  },
  watch: {
    value: {
      handler() {
        this.stock = this.bussiness == 1 ? this.data.cargoCapacity : this.data.qty
      },
      immediate: true
    }
  },
  created() {},
  methods: {
    confirm() {
      if (this.trackId) {
        this.updateMediicinecabineInventory()
        return
      }
      this.$emit('onConfirm', this.stock)
    },
    created() {
        
    emitValue(show) {
      this.$emit('input', show)
    },
    methods:{
        confirm(){
            if(this.trackId){
                this.updateMediicinecabineInventory()
                return
            }
            this.$emit('onConfirm', this.stock)
        },
        emitValue(show){
            this.$emit('input', show)
        },
        updateMediicinecabineInventory(data){
            this.$loading('保存中..')
            let p = {
                id: this.trackId,
                drugId: this.drugId,
                qty: this.stock,
                userId: this.user.id,
            }
            
            console.log('params', p)
            medicineAbinetApi
                .updateMediicinecabineInventory(p)
                .then(res=>{
                    console.log('updateMediicinecabineInventory', res)
                    this.setStockShow = false
                    this.$toast.clear()
                    this.$emitRefreshPage('replenishmentStartDeviceDetail')
                    history.back()
                    // this.$router.replace({
                    //     path: '/replenishment/start/productDetail',
                    //     query: {
                    //         id: this.trackId
                    //     }
                    // })
                })
                .catch(err=>{
                    console.error(err)
                })
        },
    updateMediicinecabineInventory(data) {
      this.$loading('保存中..')
      let p = {
        id: this.trackId,
        drugId: this.drugId,
        qty: this.stock,
        userId: this.user.id
      }
      medicineAbinetApi
        .updateMediicinecabineInventory(p)
        .then(res => {
          this.setStockShow = false
          this.$toast.clear()
          this.$emitRefreshPage('replenishmentStartDeviceDetail')
          history.back()
          // this.$router.replace({
          //     path: '/replenishment/start/productDetail',
          //     query: {
          //         id: this.trackId
          //     }
          // })
        })
        .catch(err => {
          console.error(err)
        })
    },
    delDrug() {
      this.$dialog
        .confirm({
          message: '是否确定删除该药品'
        })
        .then(() => {
          // on confirm
        })
        .catch(() => {
          // on cancel
        })
    }
  }
}
</script>
<style lang='scss' scoped>
.replenishment-start-set-stock-dialog{
    .close{
        position: absolute;
        right: 10px;
        top: 10px;
    }
    ::v-deep .van-field{
        padding: 0;
        .van-cell__value{
            border: 1px solid #e1e1e1;
            input{
                text-align: center;
            }
        }
    }
    .van-image{
        width: 200px;
        height: 150px;
    }
<style lang="scss" scoped>
.replenishment-start-set-stock-dialog {
  .close {
    position: absolute;
    right: 10px;
    top: 10px;
  }
  //   ::v-deep .van-field {
  //     padding: 0;
  //     .van-cell__value {
  //       border: 1px solid #e1e1e1;
  //       input {
  //         text-align: center;
  //       }
  //     }
  //   }
  .van-image {
    width: 200px;
    height: 150px;
  }
  .scan {
    position: absolute;
    right: 36px;
    top: 50%;
    transform: translateY(-50%);
  }
  .drag {
    width: 20px;
    height: 20px;
  }
}
::v-deep {
  .van-field__label {
    color: #999;
    width: 60px;
    margin-right: 0;
  }
}
</style>
</style>

+ 1 - 23
mini-pro-web/src/views/replenishment/start/deviceDetail.vue

@ -220,12 +220,10 @@ export default{
                    id: item.outId || item.id,
                    status
                }
                console.log('updateStockUpStatus', p)
                medicineAbinetApi
                    .updateStockUpStatus(p)
                    .then(res=>{
                        this.$toast.clear()
                        console.log('updateStockUpStatus', res)
                        if(res && res.status == 200){
                            if(status == 1){
                                this.showOrderPicker = false
@ -254,11 +252,9 @@ export default{
                replenishEr: this.user.id,
                deviceId: this.deviceId
            }
            console.log('queryUpListByStatus', p)
            await medicineAbinetApi
                .queryUpListByStatus(p)
                .then(res=>{
                    console.log('queryUpListByStatus', res)
                    this.orderList = res.detailModelList
                })
                .catch(err=>{
@ -271,11 +267,9 @@ export default{
                replenishEr: this.user.id,
                deviceId: this.deviceId
            }
            console.log('queryUpListByDeviceIdAndStatus', p)
            await medicineAbinetApi
                .queryUpListByDeviceIdAndStatus(p)
                .then(res=>{
                    console.log('queryUpListByDeviceIdAndStatus', res)
                    if(res && res.detailModelList && res.detailModelList.length){
                        var mediicinestockout = res.detailModelList[0].mediicinestockout
                        mediicinestockout.outId = res.detailModelList[0].id
@ -284,7 +278,6 @@ export default{
                            v.isShow = true
                            return this.accessLeftNum(v)>0
                        })
                        console.log(this.list[0])
                    }
                })
                .catch(err=>{
@ -295,11 +288,9 @@ export default{
            let p = {
                deviceId: this.deviceId
            }
            console.log('getDrugInventoryCount', p)
            await medicineAbinetApi
                .getDrugInventoryCount(p)
                .then(res=>{
                    console.log('getDrugInventoryCount', res)
                    _.each(res.detailModelList, v=>{
                        v.isShow = true
                    })
@ -317,11 +308,9 @@ export default{
                deviceId: this.deviceId,
                userId: this.user.id
            }
            console.log('getOutDetailByUserIdAndDeviceId', p)
            await medicineAbinetApi
                .getOutDetailByUserIdAndDeviceId(p)
                .then(res=>{
                    console.log('getOutDetailByUserIdAndDeviceId', res)
                    res.detailModelList.forEach(v=>{
                        var exist = _.find(this.deviceDrugList, m=>{
                            return v.drugCode == m.drugCode
@ -340,13 +329,12 @@ export default{
            let p = {
                deviceId: this.deviceId
            }
            console.log('params', p)
            medicineAbinetApi
                .findMediicinecabinetInventoryByDeviceId(p)
                .then(res=>{
                    console.log('findMediicinecabinetInventoryByDeviceId', res)
                    this.$toast.clear()
                    if(res.detailModelList && res.detailModelList.length){
                        
                        var openCount = 0, closeCount = 0, faultCount = 0
                        var list = _.map(res.detailModelList, v=>{
                            var key = _.keys(v)[0]
@ -370,7 +358,6 @@ export default{
                        list = _.sortBy(list, 'index')
                        this.list = this.list.concat(list)
                        this.curItem = this.list[this.activeKey]
                        console.log(list, this.curItem)
                    }
                })
                .catch(err=>{
@ -381,11 +368,9 @@ export default{
            let p = {
                deviceId: this.deviceId
            }
            console.log('params', p)
            medicineAbinetApi
                .findDeviceById(p)
                .then(res=>{
                    console.log('findDeviceById', res)
                    this.$toast.clear()
                    this.deviceInfo = res.obj
                })
@ -401,7 +386,6 @@ export default{
            if(this.isView){
                return
            }
            console.log(item)
            this.gotoUrl('/replenishment/start/quick', {
                deviceId: this.deviceId,
                drugName: item.drugName,
@ -479,16 +463,13 @@ export default{
                            ids.push(v.id)
                        }
                    })
                    console.log('ids', ids)
                    let p = {
                        cargoIds: ids.join(','),//必穿		货道id,多个id,英文逗号隔开,例如   1,2 		4,5,6
                        cargoState 		//必传		1开启,0关闭
                    }
                    console.log('params', p)
                    medicineAbinetApi
                        .openAndCloseCargo(p)
                        .then(res=>{
                            console.log('openAndCloseCargo', res)
                            this.selItemList.forEach(v=>{
                                v.isActive = false
                                v.cargoState = cargoState
@ -579,7 +560,6 @@ export default{
                    }
                    return false
                })
                console.log(this.selItemList, 'this.selectList')
                if(exist){
                    this.$toast('只能选择相邻的货道进行合并')
@ -609,11 +589,9 @@ export default{
                cargoIds: ids.join(','),
                isMerge
            }
            console.log(p)
            medicineAbinetApi
                .mergeAndSplitCargo(p)
                .then(res=>{
                    console.log('mergeAndSplitCargo', res)
                    if(res.status == 200){
                        this.selItemList = []
                        this.$toast('操作成功')

+ 294 - 226
mini-pro-web/src/views/replenishment/start/productDetail.vue

@ -1,252 +1,320 @@
<template>
    <div class='replenishment-start-product-detail fs-14' v-if="info">
        <div class="tc pt50">
            <van-image
                radius="5px"
                :src="require('@/assets/images/drug_default.png')"
            />
        </div>
        <div class="tc c-333 pt20 pb10 plr15 break-all">{{info.drugName}}</div>
        <div class="tc c-333 pt20 pb10 plr15 break-all">{{info.specif}}</div>
  <div class="replenishment-start-product-detail fs-14" v-if="info">
    <div class="tc pt50">
      <van-image radius="5px" :src="require('@/assets/images/drug_default.png')" />
    </div>
    <div class="tc c-333 pt20 pb10 plr15 break-all">{{ info.drugName }}</div>
    <div class="tc c-333 pt20 pb10 plr15 break-all">{{ info.specif }}</div>
        <div class="c-17b3ec tc"><span @click="gotoUrl('/drugList', {deviceId: info.idDevice})">更换药品</span></div>
        <div class=" tc pt20">
            <span class="mr10">现有数量:{{info.qty}} </span>
            <span class="ml10">货道容量:{{info.cargoCapacity}} </span>
        </div>
        <div class="pt20">
            <div class="progress">
                <div class="bar" :style="'width:'+ (rate) +'%'"></div>
                <div class="text">{{info.qty}}/{{info.cargoCapacity}}</div>
            </div>
        </div>
        <div class="bot-banner  ptb10">
            <div><van-button type="info" @click="bussiness=2;setStockShow=true" round block >矫正库存</van-button></div>
            <div class="kitbox pt10">
                <div class="box-flex-1 pr10"><van-button @click="bussiness=1;setStockShow=true" type="info" round block plain >修改容量</van-button></div>
                <div class="box-flex-1 pl10"><van-button @click="checkRemoveDrug()" type="info" round block plain class="red" >下架药品</van-button></div>
            </div>
    <div class="c-17b3ec tc"><span @click="gotoUrl('/drugList', { deviceId: info.idDevice })">更换药品</span></div>
    <div class=" tc pt20">
      <span class="mr10">现有数量:{{ info.qty }}</span>
      <span class="ml10">货道容量:{{ info.cargoCapacity }}</span>
    </div>
    <div class="pt20">
      <div class="progress">
        <div class="bar" :style="'width:' + rate + '%'"></div>
        <div class="text">{{ info.qty }}/{{ info.cargoCapacity }}</div>
      </div>
    </div>
    <div class="box">
      <div class="mb10">
        <span>药品追溯码(</span>
        <span class="c-17b3ec">{{ info.qty }}</span>
        <span>/{{ info.cargoCapacity }})</span>
      </div>
      <div class="drug-item">
        <span class="c-999">1位最外:</span>
        <span>9431924004563435393</span>
      </div>
      <div class="drug-item">
        <span class="c-999">2位最外:</span>
        <span>9431924004563435393</span>
      </div>
    </div>
    <div class="bot-banner  ptb10">
      <div>
        <van-button
          type="info"
          @click="
            bussiness = 2
            setStockShow = true
          "
          round
          block
        >
          矫正库存
        </van-button>
      </div>
      <div class="kitbox pt10">
        <div class="box-flex-1 pr10">
          <van-button
            @click="
              bussiness = 1
              setStockShow = true
            "
            type="info"
            round
            block
            plain
          >
            修改容量
          </van-button>
        </div>
        <!-- 修改容量 -->
        <SetStockDialog 
            v-model="setStockShow" 
            :data="info"
            @onConfirm="updateMediicinecabineInventoryInfoById" 
            :bussiness="bussiness"/>
        <div class="box-flex-1 pl10"><van-button @click="checkRemoveDrug()" type="info" round block plain class="red">下架药品</van-button></div>
      </div>
    </div>
    <!-- 修改容量 -->
    <SetStockDialog v-model="setStockShow" :data="info" @onConfirm="updateMediicinecabineInventoryInfoById" :bussiness="bussiness" />
  </div>
</template>
<script>
import medicineAbinetApi from '@/api/api-medicineAbinet'
import SetStockDialog from './components/SetStockDialog'
export default{
    name: 'replenishmentStartProductDetail',
    components:{
        SetStockDialog
    },
    data(){
        return {
            id: this.$route.query.id,
            shelfStatus: this.$route.query.shelfStatus,
            setStockShow: false,
            info: '',
            bussiness: 1, //1修改容量 2矫正库存 3上下架
            drugId: '',
            drugName: '',
            drugPrice: '',
            from: this.$route.query.from,
        }
export default {
  name: 'replenishmentStartProductDetail',
  components: {
    SetStockDialog
  },
  data() {
    return {
      id: this.$route.query.id,
      shelfStatus: this.$route.query.shelfStatus,
      setStockShow: false,
      info: '',
      bussiness: 1, //1修改容量 2矫正库存 3上下架
      drugId: '',
      drugName: '',
      drugPrice: '',
      from: this.$route.query.from
    }
  },
  computed: {
    rate() {
      var { info } = this
      return (info && (Number(info.qty) / Number(info.cargoCapacity)) * 100) || 0
    }
  },
  created() {
    this.$EventBus.$on('drug-sel', data => {
      this.drugId = data.id
      this.drugName = data.drugName
      this.drugPrice = data.price
      this.bussiness = 4
      this.setStockShow = true
    })
    this.selectMediicinecabineInventoryById()
  },
  methods: {
    drugTraceabilityCodeDetail(deviceId, drugId, trackId) {
      const params = {
        deviceId: deviceId,
        drugId: drugId,
        inventoryIds: trackId
      }
      medicineAbinetApi.drugTraceabilityCodeDetail(params).then(res => {
        // const data = res.obj
        // this.drug = data.drug
        // this.inventoryList = data.inventoryList.map(el => {
        //   let obj = {
        //     ...el,
        //     arr: []
        //   }
        //   var maxNum = el.cargoCapacity * 1 > this.stock * 1 ? this.stock * 1 : el.cargoCapacity * 1
        //   var len = obj.codeList.length
        //   if (obj.codeList.length < maxNum) {
        //     for (let i = 0; i < maxNum - len; i++) {
        //       obj.codeList.push({ code: '' })
        //     }
        //   }
        //   return obj
        // })
      })
    },
    computed:{
        rate(){
            var {info} = this
            return (info&&(Number(info.qty) / Number(info.cargoCapacity) * 100))|| 0
        }
    selectMediicinecabineInventoryById() {
      this.$loading('加载中..')
      let p = {
        id: this.id
      }
      medicineAbinetApi
        .selectMediicinecabineInventoryById(p)
        .then(res => {
          console.log(res,"REs");
          const obj = res.obj
          this.$toast.clear()
          this.info = obj
          this.drugTraceabilityCodeDetail(obj.idDevice, obj.drugId, obj.id)
        })
        .catch(err => {
          console.error(err)
        })
    },
    created() {
        this.$EventBus.$on('drug-sel', data=>{
            this.drugId = data.id
            this.drugName = data.drugName
            this.drugPrice = data.price
            this.bussiness = 4
            this.setStockShow = true
    checkRemoveDrug() {
      this.$dialog
        .confirm({
          title: '提示',
          message: '确认下架该药品?'
        })
        .then(() => {
          this.updateMediicinecabineInventoryInfoById(0, 3)
        })
        this.selectMediicinecabineInventoryById()
    },
    methods:{
        selectMediicinecabineInventoryById(){
            this.$loading('加载中..')
            let p = {
                id: this.id
            }
            console.log('params', p)
            medicineAbinetApi
                .selectMediicinecabineInventoryById(p)
                .then(res=>{
                    console.log('selectMediicinecabineInventoryById', res)
                    this.$toast.clear()
                    this.info = res.obj
                })
                .catch(err=>{
                    console.error(err)
                })
        },
        checkRemoveDrug(){
            this.$dialog.confirm({
                title: '提示',
                message: '确认下架该药品?',
            })
            .then(() => {
                this.updateMediicinecabineInventoryInfoById(0, 3)
    updateMediicinecabineInventoryInfoById(data, bussiness) {
      if (this.bussiness == 4) {
        this.updateMediicinecabineInventory(data)
        return
      }
      this.$loading('保存中..')
      let p = {
        id: this.id,
        userId: this.user.id
      }
      if (bussiness == 3) {
        p.status = data //下架状态0,上架状态1
      } else if (this.bussiness == 1) {
        p.cargoCapacity = data
      } else if (this.bussiness == 2) {
        p.qty = data
      }
      medicineAbinetApi
        .updateMediicinecabineInventoryInfoById(p)
        .then(res => {
          this.setStockShow = false
          this.$toast.clear()
          this.$emitRefreshPage('replenishmentStartDeviceDetail')
          this.$emitRefreshPage('replenishmentDeviceDetail')
          if (this.from == 'quick') {
            this.$emitRefreshPage('replenishmentStartQuick', {
              data,
              bussiness: bussiness || this.bussiness
            })
        },
        updateMediicinecabineInventoryInfoById(data, bussiness){
            if(this.bussiness == 4){
                this.updateMediicinecabineInventory(data)
                return
            }
            this.$loading('保存中..')
            let p = {
                id: this.id,
                userId: this.user.id,
            }
            if(bussiness==3){
                p.status = data //下架状态0,上架状态1
            } else if(this.bussiness == 1){
                p.cargoCapacity = data
            } else if(this.bussiness == 2){
                p.qty = data
            } 
            console.log('params', p)
            medicineAbinetApi
                .updateMediicinecabineInventoryInfoById(p)
                .then(res=>{
                    console.log('updateMediicinecabineInventoryInfoById', res)
                    this.setStockShow = false
                    this.$toast.clear()
                    this.$emitRefreshPage('replenishmentStartDeviceDetail')
                    this.$emitRefreshPage('replenishmentDeviceDetail')
          }
          if (bussiness == 3) {
            this.$router.back()
          } else {
            this.selectMediicinecabineInventoryById()
          }
        })
        .catch(err => {
          console.error(err)
        })
    },
                    if(this.from == 'quick'){
                        this.$emitRefreshPage('replenishmentStartQuick', {
                            data,
                            bussiness: bussiness || this.bussiness
                        })
                    }
                    if(bussiness==3){
                        this.$router.back()
                    } else {
                        this.selectMediicinecabineInventoryById()
                    }
                })
                .catch(err=>{
                    console.error(err)
                })
        },
    updateMediicinecabineInventory(data) {
      this.$loading('保存中..')
      let p = {
        id: this.id,
        drugId: this.drugId,
        qty: data,
        userId: this.user.id
      }
        updateMediicinecabineInventory(data){
            this.$loading('保存中..')
            let p = {
                id: this.id,
                drugId: this.drugId,
                qty:data,
                userId: this.user.id,
            }
            
            console.log('params', p)
            medicineAbinetApi
                .updateMediicinecabineInventory(p)
                .then(res=>{
                    console.log('updateMediicinecabineInventory', res)
                    this.setStockShow = false
                    this.$toast.clear()
                    this.$emitRefreshPage('replenishmentStartDeviceDetail')
                    this.$emitRefreshPage('replenishmentDeviceDetail')
                    this.selectMediicinecabineInventoryById()
                    if(this.from == 'quick'){
                        this.$emitRefreshPage('replenishmentStartQuick', {
                            drugId: this.drugId,
                            drugName: this.drugName,
                            drugPrice: this.drugPrice,
                            data,
                            bussiness: 4
                        })
                        history.go(-1)
                    }
                })
                .catch(err=>{
                    console.error(err)
                })
        },
    },
      medicineAbinetApi
        .updateMediicinecabineInventory(p)
        .then(res => {
          this.setStockShow = false
          this.$toast.clear()
          this.$emitRefreshPage('replenishmentStartDeviceDetail')
          this.$emitRefreshPage('replenishmentDeviceDetail')
          this.selectMediicinecabineInventoryById()
          if (this.from == 'quick') {
            this.$emitRefreshPage('replenishmentStartQuick', {
              drugId: this.drugId,
              drugName: this.drugName,
              drugPrice: this.drugPrice,
              data,
              bussiness: 4
            })
            history.go(-1)
          }
        })
        .catch(err => {
          console.error(err)
        })
    }
  }
}
</script>
<style lang='scss' scoped>
.replenishment-start-product-detail{
    .van-image{
        width: 100px;
        height: 100px;
<style lang="scss" scoped>
.replenishment-start-product-detail {
  .van-image {
    width: 100px;
    height: 100px;
  }
  .progress {
    width: 80vw;
    background: #bbb;
    height: 20px;
    border-radius: 20px;
    margin: 0 auto;
    position: relative;
    overflow: hidden;
    .bar {
      background: #17b3ec;
      height: 100%;
      display: inline-block;
      color: #fff;
      text-align: center;
      border-top-left-radius: 20px;
      border-bottom-left-radius: 20px;
      line-height: 20px;
    }
    .progress{
        width: 80vw;
        background: #bbb;
        height: 20px;
    .text {
      position: absolute;
      left: 0;
      top: 0;
      width: 100%;
      background: transparent;
      text-align: center;
      color: #fff;
      line-height: 20px;
    }
    &.full {
      .bar {
        border-radius: 20px;
        margin: 0 auto  ;
        position: relative;
        overflow: hidden;
        .bar{
            background: #17b3ec;
            height: 100%;
            display: inline-block;
            color: #fff;
            text-align: center;
            border-top-left-radius: 20px;
            border-bottom-left-radius: 20px;
            line-height: 20px;
        }
        .text{
            position: absolute;
            left: 0;
            top: 0;
            width: 100%;
            background: transparent;
            text-align: center;
            color: #fff;
            line-height: 20px;
        }
        &.full{
            .bar{
                border-radius: 20px;
            }
        }
      }
    }
  }
  .bot-banner {
    position: fixed;
    bottom: 0;
    width: 80%;
    left: 10%;
    // background: #fff;
    .van-button--plain {
      &.red {
        color: #ff5e6c;
        border-color: #ff5e6c;
      }
    }
    .bot-banner{
        position: fixed;
        bottom: 0;
        width: 80%;
        left: 10%;
        // background: #fff;
        .van-button--plain{
            &.red{
                color: #ff5e6c;
                border-color: #ff5e6c;
            }
        }
        .van-button{
            padding: 0;
            font-size: 13px;
            height: 36px;
        }
    .van-button {
      padding: 0;
      font-size: 13px;
      height: 36px;
    }
  }
}
.box {
  margin: 15px;
  padding: 12px;
  border-radius: 8px;
}
.drug-item {
  background: #fff;
  padding: 8px;
  border-radius: 4px;
  margin-bottom: 10px;
}
</style>
<style lang='scss'>
@supports(bottom: env(safe-area-inset-bottom)){
    .replenishment-start-product-detail{
        .bot-banner{
            bottom: env(safe-area-inset-bottom) !important;
        }
<style lang="scss">
@supports (bottom: env(safe-area-inset-bottom)) {
  .replenishment-start-product-detail {
    .bot-banner {
      bottom: env(safe-area-inset-bottom) !important;
    }
  }
}
</style>
</style>

+ 608 - 581
mini-pro-web/src/views/replenishment/start/quick.vue

@ -1,640 +1,667 @@
<template>
    <div class='replenishment-start-quick fs-14'>
        <div class="plr10 bgc-fff ptb10">
            <div class="drug-item kitbox" :key="k">
                <div class="photo">
                    <img src="@/assets/images/drug_default.png" alt="">
                </div>
                <div class="box-flex-1 pl10">
                    <div class="fs-15 ellipsis_1 c-333">{{query.drugName}}</div>
                    <div class="fs-14 c-666 kitbox">
                        <div class="box-flex-1">
                            <div>规格:{{decodeStr(query.specif)}}</div>
                            <div>当前库存量:{{query.qty}}</div>
                            <div v-if="outDetailId">未放置数量:{{query.quantity}}</div>
                        </div>
                        <div class="stepper kitbox">
                            <van-stepper v-model="num" integer min="0" max="99" />
                            <div v-if="repAll" class="text">补满</div>
                        </div>
                    </div>
                </div>
            </div>
  <div class="replenishment-start-quick fs-14">
    <div class="plr10 bgc-fff ptb10">
      <div class="drug-item kitbox" :key="k">
        <div class="photo">
          <img src="@/assets/images/drug_default.png" alt="" />
        </div>
        <div class="fs-14 pl15 ptb10 bgc-fff kitbox pr15">
            <div class="box-flex-1 lh24">
                <span>从上往下数</span>
                <div class="color-block"></div>
                <span class="ml5 fs-12">已关闭</span>
                <div class="color-block ml10 red"></div>
                <span class="ml5 fs-12">故障</span>
        <div class="box-flex-1 pl10">
          <div class="fs-15 ellipsis_1 c-333">{{ query.drugName }}</div>
          <div class="fs-14 c-666 kitbox">
            <div class="box-flex-1">
              <div>规格:{{ decodeStr(query.specif) }}</div>
              <div>当前库存量:{{ query.qty }}</div>
              <div v-if="outDetailId">未放置数量:{{ query.quantity }}</div>
            </div>
            <div class="c-17B3EC" @click="selectAll" v-if="!isSelectAll">全选</div>
            <div class="c-17B3EC" @click="cancelAll" v-else>取消</div>
            <div class="stepper kitbox">
              <van-stepper v-model="num" integer min="0" max="99" />
              <div v-if="repAll" class="text">补满</div>
            </div>
          </div>
        </div>
      </div>
    </div>
        <div class="list-panel">
            <div class="kitbox ">
                <div class="left-list">
                    <van-sidebar v-model="activeKey">
                        <van-sidebar-item v-for="(item, i) in list" :key="i" :title="(i+1)+'层'"  />
                    </van-sidebar>
                </div>
                <div class="list box-flex-1 bgc-fff">
                    <template v-if="curItem">
                        <template v-for="(item, k) in curItem.list">
                            <div class="item" v-if="item.state!=20 && (item.drugCode == query.drugCode || !item.drugCode)" :key="k">
                                <ProductItem v-longpress="()=>{gotoDetail(item)}" bussuness="quick" @onChange="onChange($event, item)" :data="item" :index="k"/>
                            </div>
                        </template>
    <div class="fs-14 pl15 ptb10 bgc-fff kitbox pr15">
      <div class="box-flex-1 lh24">
        <span>从上往下数</span>
        <div class="color-block"></div>
        <span class="ml5 fs-12">已关闭</span>
        <div class="color-block ml10 red"></div>
        <span class="ml5 fs-12">故障</span>
      </div>
      <div class="c-17B3EC" @click="selectAll" v-if="!isSelectAll">全选</div>
      <div class="c-17B3EC" @click="cancelAll" v-else>取消</div>
    </div>
    <div class="list-panel">
      <div class="kitbox ">
        <div class="left-list">
          <van-sidebar v-model="activeKey">
            <van-sidebar-item v-for="(item, i) in list" :key="i" :title="i + 1 + '层'" />
          </van-sidebar>
        </div>
        <div class="list box-flex-1 bgc-fff">
          <template v-if="curItem">
            <template v-for="(item, k) in curItem.list">
              <div class="item" v-if="item.state != 20 && (item.drugCode == query.drugCode || !item.drugCode)" :key="k">
                <ProductItem
                  v-longpress="
                    () => {
                      gotoDetail(item)
                    }
                  "
                  bussuness="quick"
                  @onChange="onChange($event, item)"
                  :data="item"
                  :index="k"
                />
              </div>
            </template>
                        <!-- <div class="item" v-for="(item, k) in curItem.list" :key="k">
            <!-- <div class="item" v-for="(item, k) in curItem.list" :key="k">
                            <ProductItem v-if="item.state!=20" bussuness="quick" @onChange="onChange($event, item)" :data="item" :index="k"/>
                        </div> -->
                    </template>
                </div>
            </div>
          </template>
        </div>
      </div>
    </div>
        <div class="bot-banner">
            <div class="kitbox">
                <div class="box-flex-1 plr5"><van-button @click="openAndCloseCargo" block plain color="#17b3ec"  type="primary" round>关闭/开启货道</van-button></div>
                <div class="box-flex-1 plr5"><van-button @click="mergeAndSplitCargo" block plain color="#17b3ec"  type="primary" round>合并/拆分货道</van-button></div>
                <div class="box-flex-1 plr5"><van-button @click="sumit" block color="#17b3ec"  type="primary"  round>完成补货({{allNum}})</van-button></div>
            </div>
    <div class="bot-banner">
      <div class="kitbox">
        <div class="box-flex-1 plr5"><van-button @click="openAndCloseCargo" block plain color="#17b3ec" type="primary" round>关闭/开启货道</van-button></div>
        <div class="box-flex-1 plr5"><van-button @click="mergeAndSplitCargo" block plain color="#17b3ec" type="primary" round>合并/拆分货道</van-button></div>
        <!-- <div class="box-flex-1 plr5"><van-button @click="sumit" block color="#17b3ec"  type="primary"  round>完成补货({{allNum}})</van-button></div> -->
        <div class="box-flex-1 plr5">
          <van-button @click="enterTraceability" block color="#17b3ec" type="primary" round>录入追溯码({{ allNum }})</van-button>
        </div>
      </div>
    </div>
  </div>
</template>
<script>
import medicineAbinetApi from '@/api/api-medicineAbinet'
import ProductItem from '@/components/ProductItem'
export default{
    name: 'replenishmentStartQuick',
    components: {
        ProductItem
export default {
  name: 'replenishmentStartQuick',
  components: {
    ProductItem
  },
  data() {
    return {
      num: 0,
      query: this.$route.query,
      selItemList: [],
      allNum: 0,
      curCargo: '',
      deviceId: this.$route.query.deviceId,
      list: [],
      activeKey: 0,
      curItem: [],
      repAll: true,
      outDetailId: this.$route.query.outDetailId || '',
      isSelectAll: false
    }
  },
  watch: {
    activeKey(n) {
      this.curItem = ''
      this.$nextTick(() => {
        this.curItem = this.list[n]
        this.calIsSelectAll()
      })
    },
    num(n, o) {
      this.repAll = false
      this.accessNum()
    }
  },
  created() {
    this.initPage()
  },
  methods: {
    decodeStr(str) {
      return (str && decodeURIComponent(str)) || ''
    },
    data(){
        return {
            num: 0,
            query: this.$route.query,
            selItemList: [],
            allNum: 0,
            curCargo: '',
    $refreshData(p) {
      if (p) {
        var oQty = this.curCargo.cargoCapacity - this.curCargo.qty
        //1修改容量 2矫正库存 3下架药品  4更换药品
        switch (Number(p.bussiness)) {
          case 1:
          case 2:
            if (p.bussiness == 1) {
              // 1修改容量
              this.curCargo.cargoCapacity = p.data
              if (this.curCargo.qty > this.curCargo.cargoCapacity) {
                this.curCargo.qty = this.curCargo.cargoCapacity
              }
            } else {
              // 2矫正库存
              this.curCargo.qty = p.data
              if (this.curCargo.qty > this.curCargo.cargoCapacity) {
                this.curCargo.cargoCapacity = this.curCargo.qty
              }
            }
            deviceId: this.$route.query.deviceId,
            list: [],
            activeKey: 0,
            curItem: [],
            repAll: true,
            outDetailId: this.$route.query.outDetailId || '',
            isSelectAll: false
            if (this.curCargo.isActive) {
              this.allNum = this.allNum - oQty + (this.curCargo.cargoCapacity - this.curCargo.qty)
            }
            break
          case 3:
            // 3下架药品
            this.curCargo.shelfStatus = 0
            this.curCargo.qty = 0
            if (this.curCargo.isActive) {
              this.allNum = this.allNum - oQty + this.curCargo.cargoCapacity
            }
            break
          case 4:
            // 4更换药品
            this.curItem.list.splice(this.curItem.list.indexOf(this.curCargo), 1)
            if (this.curCargo.isActive) {
              this.allNum = this.allNum - oQty
            }
            break
        }
      } else {
        this.findMediicinecabinetInventoryByDeviceId()
      }
    },
    initPage() {
      this.findMediicinecabinetInventoryByDeviceId()
    },
    watch:{
        activeKey(n){
            this.curItem = ''
            this.$nextTick(()=>{
                this.curItem = this.list[n]
                this.calIsSelectAll()
    findMediicinecabinetInventoryByDeviceId() {
      this.$loading('加载中..')
      let p = {
        deviceId: this.deviceId
      }
      medicineAbinetApi
        .findMediicinecabinetInventoryByDeviceId(p)
        .then(res => {
          this.$toast.clear()
          if (res.detailModelList && res.detailModelList.length) {
            var list = _.map(res.detailModelList, v => {
              var key = _.keys(v)[0]
              return {
                index: key,
                list: _.filter(v[key], m => {
                  m.layerNo = Number(m.layerNo)
                  m.wayerNo = Number(m.wayerNo)
                  m.isActive = false
                  return true
                  return m.drugCode == this.query.drugCode || !m.drugCode
                })
              }
            })
        },
        num(n, o){
            this.repAll = false
            this.accessNum()
            list = _.sortBy(list, 'index')
            this.list = list
            this.curItem = this.list[this.activeKey]
          }
        })
        .catch(err => {
          console.error(err)
        })
    },
    onChange(isActive, item) {
      var num = this.accessItemNum(item)
      item.isActive = isActive
      if (isActive) {
        this.selItemList.push(item)
        this.allNum += num
      } else {
        this.selItemList.splice(this.selItemList.indexOf(item), 1)
        this.allNum -= num
      }
      this.calIsSelectAll()
    },
    calIsSelectAll() {
      let flag = true
      this.list[this.activeKey].list.forEach(el => {
        if (el.cargoState == 1 && !el.isActive) {
          flag = false
        }
      })
      this.isSelectAll = flag
    },
    created() {
        this.initPage()
    accessItemNum(item) {
      var num = 0
      if (this.repAll) {
        num += item.cargoCapacity - item.qty
      } else {
        if (this.num - item.qty > 0) {
          num += this.num - item.qty
        }
      }
      return num
    },
    methods:{
        decodeStr(str){
            return (str&&decodeURIComponent(str)) || ""
        },
        $refreshData(p){
            if(p){
                var oQty = this.curCargo.cargoCapacity - this.curCargo.qty
                //1修改容量 2矫正库存 3下架药品  4更换药品
                switch(Number(p.bussiness)){
                    case 1: 
                    case 2: 
                        if(p.bussiness==1){
                            // 1修改容量
                            this.curCargo.cargoCapacity = p.data
                            if(this.curCargo.qty > this.curCargo.cargoCapacity){
                                this.curCargo.qty = this.curCargo.cargoCapacity
                            }
                        } else {
                            // 2矫正库存
                            this.curCargo.qty = p.data
                            if(this.curCargo.qty > this.curCargo.cargoCapacity){
                                this.curCargo.cargoCapacity = this.curCargo.qty
                            }
                        }
                        
                        if(this.curCargo.isActive){
                            this.allNum = this.allNum - oQty + (this.curCargo.cargoCapacity - this.curCargo.qty)
                        }
                        break;
                    case 3: 
                        // 3下架药品 
                        this.curCargo.shelfStatus = 0
                        this.curCargo.qty = 0
                        if(this.curCargo.isActive){
                            this.allNum = this.allNum - oQty + this.curCargo.cargoCapacity
                        }
                        break;
                    case 4: 
                        // 4更换药品
                        this.curItem.list.splice(this.curItem.list.indexOf(this.curCargo), 1)
                        if(this.curCargo.isActive){
                            this.allNum = this.allNum - oQty
                        }
                        break;
                }
            } else {
                this.findMediicinecabinetInventoryByDeviceId()
    accessNum() {
      var num = 0
      this.selItemList.forEach(v => {
        num += this.accessItemNum(v)
      })
      this.allNum = num
    },
    sumit() {
      if (!this.selItemList || !this.selItemList.length) {
        this.$toast('请选择操作货道')
        return
      }
      if (!this.repAll) {
        var exist = _.find(this.selItemList, v => {
          return v.qty > this.num
        })
        if (exist) {
          this.$dialog.alert({
            title: '补货设置',
            message: `补货数量不能小于当前货道库存,请重新选择数量进行补货`,
            confirmButtonText: '我已知悉'
          })
          return
        }
      }
      if (this.outDetailId && this.query.quantity < this.allNum) {
        this.$toast('当前补货数量超过了未放置数量,请重新设置')
        return
      }
      this.$dialog
        .confirm({
          title: '补货设置',
          message: `确定进行补货?`,
          confirmButtonText: '确定'
        })
        .then(() => {
          var list = _.map(this.selItemList, v => {
            return {
              cargoId: v.id,
              drugId: this.query.drugId,
              qty: this.repAll ? Number(v.cargoCapacity) : this.num
            }
        },
        initPage(){
            this.findMediicinecabinetInventoryByDeviceId()
        },
        findMediicinecabinetInventoryByDeviceId(){
            this.$loading('加载中..')
          })
          this.$nextTick(() => {
            this.$loading('保存中..')
            let p = {
                deviceId: this.deviceId
              list: JSON.stringify({ list: list }),
              userId: this.user.id,
              outDetailId: this.outDetailId
            }
            console.log('params', p)
            medicineAbinetApi
                .findMediicinecabinetInventoryByDeviceId(p)
                .then(res=>{
                    console.log('findMediicinecabinetInventoryByDeviceId', res)
                    this.$toast.clear()
                    if(res.detailModelList && res.detailModelList.length){
                        var list = _.map(res.detailModelList, v=>{
                            var key = _.keys(v)[0]
                            return {
                                index: key,
                                list: _.filter(v[key], m=>{
                                    m.layerNo = Number(m.layerNo)
                                    m.wayerNo = Number(m.wayerNo)
                                    m.isActive = false
                                    return true
                                    return m.drugCode == this.query.drugCode || !m.drugCode
                                }),
                            }
                        })
                        list = _.sortBy(list, 'index')
                        this.list = list
                        this.curItem = this.list[this.activeKey]
                        console.log(list, this.curItem)
                    }
                })
                .catch(err=>{
                    console.error(err)
                })
        },
        onChange(isActive, item){
            var num = this.accessItemNum(item)
            item.isActive = isActive
            if(isActive){
                this.selItemList.push(item)
                this.allNum += num
            } else {
                this.selItemList.splice(this.selItemList.indexOf(item), 1)
                this.allNum -= num
            }
            this.calIsSelectAll()
        },
        calIsSelectAll() {
            let flag = true
            this.list[this.activeKey].list.forEach(el=>{
                if(el.cargoState==1&&!el.isActive){
                    flag = false
                }
            })
            this.isSelectAll = flag
        },
        accessItemNum(item){
            var num = 0
            if(this.repAll){
                num += item.cargoCapacity - item.qty
            } else {
                if(this.num - item.qty > 0){
                    num += (this.num - item.qty)
              .batchAddInventory(p)
              .then(res => {
                this.query.qty += this.allNum
                this.query.quantity -= this.allNum
                if (this.query.quantity < 0) {
                  this.query.quantity = 0
                }
            }
            return num
        },
        accessNum(){
            var num = 0
            this.selItemList.forEach(v=>{
                num += this.accessItemNum(v)
            })
            this.allNum = num
        },
        sumit(){
            if(!this.selItemList || !this.selItemList.length){
                this.$toast("请选择操作货道")
                return
            }
            if(!this.repAll){
                var exist = _.find(this.selItemList, v=>{
                    return v.qty > this.num
                this.allNum = 0
                this.selItemList.forEach(v => {
                  v.isActive = false
                })
                if(exist){
                    this.$dialog.alert({
                        title: '补货设置',
                        message: `补货数量不能小于当前货道库存,请重新选择数量进行补货`,
                        confirmButtonText: '我已知悉',
                    })
                    return
                }
            }
            if(this.outDetailId && this.query.quantity<this.allNum){
                this.$toast('当前补货数量超过了未放置数量,请重新设置')
                return
            }
            this.$dialog.confirm({
                title: '补货设置',
                message: `确定进行补货?`,
                confirmButtonText: '确定',
            })
            .then(() => {
                var list = _.map(this.selItemList, v=>{
                    return {
                        cargoId: v.id,
                        drugId: this.query.drugId,
                        qty: this.repAll? Number(v.cargoCapacity) : this.num
                    }
                }) 
                this.$nextTick(()=>{
                    this.$loading('保存中..')
                    let p = {
                        list: JSON.stringify({list: list}),
                        userId: this.user.id,
                        outDetailId: this.outDetailId
                    }
                    console.log('params', p)
                    medicineAbinetApi
                        .batchAddInventory(p)
                        .then(res=>{
                            this.query.qty += this.allNum
                            this.query.quantity -= this.allNum
                            if(this.query.quantity<0){
                                this.query.quantity = 0
                            }
                            this.allNum = 0
                            console.log('batchAddInventory', res)
                            this.selItemList.forEach(v=>{
                                v.isActive = false
                            })
                            this.selItemList = []
                            this.$toast({
                                message: '操作成功',
                                forbidClick: true,
                                onClose: ()=>{
                                    
                                }
                            })
                            this.setBackRefresh(true)
                            this.$refreshData()
                        })
                        .catch(err=>{
                            console.error(err)
                        })
                this.selItemList = []
                this.$toast({
                  message: '操作成功',
                  forbidClick: true,
                  onClose: () => {}
                })
                this.setBackRefresh(true)
                this.$refreshData()
              })
              .catch(err => {
                console.error(err)
              })
          })
        })
      var exist = _.find(this.selItemList, v => {
        return v.faultState == 1
      })
      // if(exist){
      //     this.$toast('所选货道存在故障货道')
      //     return
      // }
      // batchAddInventory
    },
    enterTraceability() {
      if (this.selItemList.length == 0) {
        this.$toast('请选择要补货的货道')
        return
      }
      const trackIds = this.selItemList.map(el => el.id).join(',')
      this.$router.push({
        path: '/replenishment/start/traceability',
        query: {
          deviceId: this.deviceId,
          trackId: trackIds,
          drugId: this.query.drugId,
          stock: this.repAll ? this.query.qty : this.num,
          outDetailId: this.$route.query.outDetailId,
          type: 1
        }
      })
    },
    checkOpenOrClose() {
      return new Promise((resolve, reject) => {
        var g = _.groupBy(this.selItemList, 'cargoState')
        if (g[1] && g[0]) {
          this.$dialog
            .confirm({
              title: '货道设置',
              message: `选中货道已包含"关闭"和"开启"货道,请选择`,
              confirmButtonText: '全部开启',
              cancelButtonText: '全部关闭'
            })
            var exist = _.find(this.selItemList, v=>{
                return v.faultState == 1
            .then(() => {
              resolve(1)
            })
            // if(exist){
            //     this.$toast('所选货道存在故障货道')
            //     return
            // }
            // batchAddInventory
        },
        checkOpenOrClose(){
            return new Promise((resolve, reject)=>{
                var g = _.groupBy(this.selItemList, 'cargoState')
                if(g[1] && g[0]){
                    this.$dialog.confirm({
                        title: '货道设置',
                        message: `选中货道已包含"关闭"和"开启"货道,请选择`,
                        confirmButtonText: '全部开启',
                        cancelButtonText: '全部关闭'
                    })
                    .then(() => {
                        resolve(1)
                    })
                    .catch(()=>{
                        resolve(0)
                    })
                } else {
                    resolve(g[1]? 0 : 1)
                }
            .catch(() => {
              resolve(0)
            })
        },
        openAndCloseCargo(){
            if(!this.selItemList || !this.selItemList.length){
                this.$toast('请选择要操作的货道')
                return
        } else {
          resolve(g[1] ? 0 : 1)
        }
      })
    },
    openAndCloseCargo() {
      if (!this.selItemList || !this.selItemList.length) {
        this.$toast('请选择要操作的货道')
        return
      }
      this.checkOpenOrClose().then(cargoState => {
        this.$nextTick(() => {
          this.$loading('保存中..')
          var ids = []
          this.selItemList.forEach(v => {
            if (v.state == 21) {
              var cargoIds = v.merge.split(',')
              cargoIds.forEach(m => {
                ids.push(this.list[Number(v.layerNo) - 1].list[Number(m) - 1].id)
              })
            } else {
              ids.push(v.id)
            }
            this.checkOpenOrClose().then(cargoState=>{
                this.$nextTick(()=>{
                    this.$loading('保存中..')
                    var ids = []
                    this.selItemList.forEach(v=>{
                        if(v.state == 21){
                            var cargoIds = v.merge.split(',')
                            cargoIds.forEach(m => {
                                ids.push(this.list[Number(v.layerNo)-1].list[Number(m)-1].id)
                            })
                        } else {
                            ids.push(v.id)
                        }
                    })
                    console.log('ids', ids)
                    let p = {
                        cargoIds: ids.join(','),//必穿		货道id,多个id,英文逗号隔开,例如   1,2 		4,5,6
                        cargoState 		//必传		1开启,0关闭
                    }
                    console.log('params', p)
                    medicineAbinetApi
                        .openAndCloseCargo(p)
                        .then(res=>{
                            console.log('openAndCloseCargo', res)
                            this.selItemList.forEach(v=>{
                                v.isActive = false
                                v.cargoState = cargoState
                            })
                            this.selItemList = []
                            this.$toast({
                                message: '操作成功',
                                forbidClick: true,
                                onClose: ()=>{
                                }
                            })
                            this.setBackRefresh(true)
                        })
                        .catch(err=>{
                            console.error(err)
                        })
                })
          })
          let p = {
            cargoIds: ids.join(','), //必穿		货道id,多个id,英文逗号隔开,例如   1,2 		4,5,6
            cargoState //必传		1开启,0关闭
          }
          medicineAbinetApi
            .openAndCloseCargo(p)
            .then(res => {
              this.selItemList.forEach(v => {
                v.isActive = false
                v.cargoState = cargoState
              })
              this.selItemList = []
              this.$toast({
                message: '操作成功',
                forbidClick: true,
                onClose: () => {}
              })
              this.setBackRefresh(true)
            })
        },
        checkMerge(){
            return new Promise((resolve, reject)=>{
                var exist = _.find(this.selItemList, v=>{
                    return !!v.drugCode
                })
                if(exist){
                    this.$toast.clear()
                    this.$dialog.confirm({
                        message: '当前合并货道存在药品,需把药品下架并取出,如您继续合并默认下架全部药品',
                        confirmButtonText: '继续合并'
                    })
                    .then(() => {
                        this.$loading('保存中..')
                        resolve(true)
                    })
                    .catch(() => {
                        resolve(false)
                    });
                    return 
                }
                resolve(true)
            .catch(err => {
              console.error(err)
            })
        },
        checkSplit(){
            return new Promise((resolve, reject)=>{
                this.$toast.clear()
                this.$dialog.confirm({
                    message: '货道拆分后请取下货道上的药品',
                    confirmButtonText: '继续拆分'
                })
                .then(() => {
                    resolve(true)
                })
                .catch(() => {
                    resolve(false)
                });
        })
      })
    },
    checkMerge() {
      return new Promise((resolve, reject) => {
        var exist = _.find(this.selItemList, v => {
          return !!v.drugCode
        })
        if (exist) {
          this.$toast.clear()
          this.$dialog
            .confirm({
              message: '当前合并货道存在药品,需把药品下架并取出,如您继续合并默认下架全部药品',
              confirmButtonText: '继续合并'
            })
        },
        async mergeAndSplitCargo(){
            this.$loading('保存中..')
            var ids = []
            var g = _.groupBy(this.selItemList, 'state')
            if((g['21'] || g['20']) && g['1']){
                this.$toast.clear()
                this.$dialog.alert({
                    title: '货道设置',
                    message: `当前选中货道包含合并,您需要先手动拆分合并的货道`,
                    confirmButtonText: '我已知悉',
                })
                return
            }
            var isMerge = g['21']? false : true
            if(isMerge){
                console.log(this.selItemList, 'this.selectList')
                if(!this.selItemList.length || this.selItemList.length<2){
                    this.$toast('至少选择两个相邻的货道')
                    return
                }
                var list = _.sortBy(this.selItemList, ['layerNo', 'wayerNo'])
                var exist = _.find(list, (v, i)=>{
                    var next = list[i+1]
                    if(next){
                        if(next && v.layerNo!=next.layerNo){
                            return true
                        }
                        if(Number(v.wayerNo)+1 != Number(next.wayerNo)){
                            return true
                        }
                    }
                    return false
                })
                if(exist){
                    this.$toast('只能选择相邻的货道进行合并')
                    return 
                }
                if(!await this.checkMerge()){
                    return
                }
                ids = _.map(list, v=>{
                    return v.id
                })
            } else {
                if(!this.selItemList.length || this.selItemList.length>1 || this.selItemList[0].state!=21){
                    this.$toast('请选择一个主货道')
                    return
                }
                if(!await this.checkSplit()){
                    return
                }
                this.$loading('保存中..')
                var item = this.selItemList[0]
                ids.push(item.id)
            }
            
            var p = {
                userId: this.user.id,
                cargoIds: ids.join(','),
                isMerge
            }
            console.log(p)
            medicineAbinetApi
                .mergeAndSplitCargo(p)
                .then(res=>{
                    console.log('mergeAndSplitCargo', res)
                    if(res.status == 200){
                        this.selItemList = []
                        this.$toast('操作成功')
                        this.$refreshData()
                        this.setBackRefresh(true)
                    } 
                })
                .catch(err=>{
                    console.error(err)
                })
        },
        gotoDetail(item){
            if(item.shelfStatus==1){
                this.curCargo = item; 
                this.gotoUrl('./productDetail', {
                    id: item.id,
                    from: 'quick'
                })
            }
        },
        selectAll(){
            this.isSelectAll = true
            this.list[this.activeKey].list.forEach(el=>{
                if(this.selItemList.findIndex(item=>{return item.id == el.id})<0&&el.cargoState==1){
                    el.isActive = true
                    this.selItemList.push(el)
                    const num = this.accessItemNum(el)
                    this.allNum += num
                }
            .then(() => {
              this.$loading('保存中..')
              resolve(true)
            })
        },
        cancelAll(){
            this.isSelectAll = false
            this.list[this.activeKey].list.forEach(el=>{
                const index = this.selItemList.findIndex(item=>{return item.id == el.id})
                if(index > -1){
                    el.isActive = false
                    this.selItemList.splice(index, 1)
                    const num = this.accessItemNum(el)
                    this.allNum -= num
                }
            .catch(() => {
              resolve(false)
            })
          return
        }
        resolve(true)
      })
    },
}
</script>
<style lang='scss' scoped>
.replenishment-start-quick{
    display: -webkit-box;
    -webkit-box-orient: vertical;
    height: 100vh;
    .list-panel{
        -webkit-box-flex: 1;
        position: relative;
        >.kitbox{
            height: 100%;
            width: 100%;
            position: absolute;
        }
    }
    .drug-item{
        img{
            width: 60px;
            border-radius: 5px;
    checkSplit() {
      return new Promise((resolve, reject) => {
        this.$toast.clear()
        this.$dialog
          .confirm({
            message: '货道拆分后请取下货道上的药品',
            confirmButtonText: '继续拆分'
          })
          .then(() => {
            resolve(true)
          })
          .catch(() => {
            resolve(false)
          })
      })
    },
    async mergeAndSplitCargo() {
      this.$loading('保存中..')
      var ids = []
      var g = _.groupBy(this.selItemList, 'state')
      if ((g['21'] || g['20']) && g['1']) {
        this.$toast.clear()
        this.$dialog.alert({
          title: '货道设置',
          message: `当前选中货道包含合并,您需要先手动拆分合并的货道`,
          confirmButtonText: '我已知悉'
        })
        return
      }
      var isMerge = g['21'] ? false : true
      if (isMerge) {
        if (!this.selItemList.length || this.selItemList.length < 2) {
          this.$toast('至少选择两个相邻的货道')
          return
        }
        .stepper{
            -webkit-box-align: end;
            position: relative;
            .text{
                position: absolute;
                background: #f2f3f5;
                width: 34px;
                text-align: center;
                left: 50%;
                transform: translateX(-50%);
                bottom: 0;
                line-height: 27px;
        var list = _.sortBy(this.selItemList, ['layerNo', 'wayerNo'])
        var exist = _.find(list, (v, i) => {
          var next = list[i + 1]
          if (next) {
            if (next && v.layerNo != next.layerNo) {
              return true
            }
        }
    }
    .list{
        display: flex;
        flex-wrap: wrap;
        align-content: flex-start;
        .item{
            margin-left: 5px;
            margin-bottom: 5px;
            height: 80px;
            .components-product-item{
                width: 90px;
            if (Number(v.wayerNo) + 1 != Number(next.wayerNo)) {
              return true
            }
          }
          return false
        })
        if (exist) {
          this.$toast('只能选择相邻的货道进行合并')
          return
        }
        if (!(await this.checkMerge())) {
          return
        }
        ids = _.map(list, v => {
          return v.id
        })
      } else {
        if (!this.selItemList.length || this.selItemList.length > 1 || this.selItemList[0].state != 21) {
          this.$toast('请选择一个主货道')
          return
        }
        if (!(await this.checkSplit())) {
          return
        }
        
        this.$loading('保存中..')
        var item = this.selItemList[0]
        ids.push(item.id)
      }
      var p = {
        userId: this.user.id,
        cargoIds: ids.join(','),
        isMerge
      }
      medicineAbinetApi
        .mergeAndSplitCargo(p)
        .then(res => {
          if (res.status == 200) {
            this.selItemList = []
            this.$toast('操作成功')
            this.$refreshData()
            this.setBackRefresh(true)
          }
        })
        .catch(err => {
          console.error(err)
        })
    },
    gotoDetail(item) {
      if (item.shelfStatus == 1) {
        this.curCargo = item
        this.gotoUrl('./productDetail', {
          id: item.id,
          from: 'quick'
        })
      }
    },
    selectAll() {
      this.isSelectAll = true
      this.list[this.activeKey].list.forEach(el => {
        if (
          this.selItemList.findIndex(item => {
            return item.id == el.id
          }) < 0 &&
          el.cargoState == 1
        ) {
          el.isActive = true
          this.selItemList.push(el)
          const num = this.accessItemNum(el)
          this.allNum += num
        }
      })
    },
    cancelAll() {
      this.isSelectAll = false
      this.list[this.activeKey].list.forEach(el => {
        const index = this.selItemList.findIndex(item => {
          return item.id == el.id
        })
        if (index > -1) {
          el.isActive = false
          this.selItemList.splice(index, 1)
          const num = this.accessItemNum(el)
          this.allNum -= num
        }
      })
    }
    .left-list, .list{
        height: 100%;
        overflow-y: auto;
  }
}
</script>
<style lang="scss" scoped>
.replenishment-start-quick {
  display: -webkit-box;
  -webkit-box-orient: vertical;
  height: 100vh;
  .list-panel {
    -webkit-box-flex: 1;
    position: relative;
    > .kitbox {
      height: 100%;
      width: 100%;
      position: absolute;
    }
    .color-block{
        background: #ff9526;
        width: 10px;
        height: 10px;
        display: inline-block;
        margin-left: 20px;
        border-radius: 2px;
        &.red{
            background: #ff5e6c;
        }
  }
  .drug-item {
    img {
      width: 60px;
      border-radius: 5px;
    }
    .bot-banner{
        padding: 10px 5px;
        position: relative;
    .stepper {
      -webkit-box-align: end;
      position: relative;
      .text {
        position: absolute;
        background: #f2f3f5;
        width: 34px;
        text-align: center;
        left: 50%;
        transform: translateX(-50%);
        bottom: 0;
        width: 100%;
        left: 0;
        background: #fff;
        z-index: 3;
        >div{
            -webkit-box-flex: 1;
            .van-button{
                padding: 0;
                font-size: 13px;
                height: 36px;
            }
        }
        line-height: 27px;
      }
    }
  }
  .list {
    display: flex;
    flex-wrap: wrap;
    align-content: flex-start;
    .item {
      margin-left: 5px;
      margin-bottom: 5px;
      height: 80px;
      .components-product-item {
        width: 90px;
      }
    }
  }
  .left-list,
  .list {
    height: 100%;
    overflow-y: auto;
  }
  .color-block {
    background: #ff9526;
    width: 10px;
    height: 10px;
    display: inline-block;
    margin-left: 20px;
    border-radius: 2px;
    &.red {
      background: #ff5e6c;
    }
  }
  .bot-banner {
    padding: 10px 5px;
    position: relative;
    bottom: 0;
    width: 100%;
    left: 0;
    background: #fff;
    z-index: 3;
    > div {
      -webkit-box-flex: 1;
      .van-button {
        padding: 0;
        font-size: 13px;
        height: 36px;
      }
    }
  }
}
</style>
<style lang="scss">
.mainNobotHasTop{
    .replenishment-start-quick{
        height: calc(100vh - 46px);
    }
.mainNobotHasTop {
  .replenishment-start-quick {
    height: calc(100vh - 46px);
  }
}
@supports(bottom: env(safe-area-inset-bottom)){
    .replenishment-start-quick{
        .bot-banner{
            &::after{
                content: "";
                height: env(safe-area-inset-bottom);
                width: 100%;
                display: block;
            }
        }
@supports (bottom: env(safe-area-inset-bottom)) {
  .replenishment-start-quick {
    .bot-banner {
      &::after {
        content: '';
        height: env(safe-area-inset-bottom);
        width: 100%;
        display: block;
      }
    }
  }
}
</style>
</style>

+ 301 - 0
mini-pro-web/src/views/replenishment/start/traceability.vue

@ -0,0 +1,301 @@
<template>
  <div>
    <div style="height: calc(100vh - 106px);overflow: auto;">
      <div class="box" style="display: flex;">
        <img :src="drug.pic" class="img" alt="" />
        <div class="info">
          <div class="fs-14 mb10">{{ drug.drugName }}</div>
          <div class="fs-13">
            <span class="c-666">编码:</span>
            {{ drug.drugCode }}
          </div>
          <div class="fs-13">
            <span class="c-666">规格:</span>
            <span class="pr10">{{ drug.specif }}</span>
          </div>
        </div>
      </div>
      <div class="box">
        <div class="title">
          药品追溯码(
          <span class="c-17b3ec">{{ fillNum }}</span>
          /{{ allNum }})
        </div>
        <div class="fs-12 c-999">请按照追溯码录入的位次进行顺序摆放,序号为最里的位次摆放在该号位的最里层,以此类推</div>
        <div class="line"></div>
        <div v-for="item in inventoryList" :key="item.id">
          <div class="title">{{ item.layerNo }}层{{ item.wayerNo }}号位</div>
          <div class="code-box">
            <draggable v-model="item.arr" :group="{ name: item.id }" :animation="200" handle=".drag" @end="onEnd" @sort="onSort($event, item.id)">
              <div class="flex justify-between align-center mb10" style="position: relative;" v-for="(el, index) in item.codeList" :key="index">
                <van-field v-model="el.code" :label="getLabel(index, item.codeList.length)" placeholder="请扫描或输入追溯码" class="mr8" />
                <img src="@/assets/images/traceability/scan.png" class="scan" alt="" />
                <img src="@/assets/images/traceability/drag.png" class="drag" alt="" />
              </div>
            </draggable>
          </div>
        </div>
      </div>
    </div>
    <div class="footer">
      <div class="plain-btn" @click="backFn">
        返回上一步
      </div>
      <div class="primary-btn" @click="finish">完成{{ type == 1 ? '补货' : '入库' }}</div>
    </div>
  </div>
</template>
<script>
import medicineAbinetApi from '@/api/api-medicineAbinet'
import draggable from 'vuedraggable'
export default {
  components: { draggable },
  data() {
    return {
      stock: this.$route.query.stock,
      deviceId: this.$route.query.deviceId,
      trackId: this.$route.query.trackId,
      drugId: this.$route.query.drugId,
      outDetailId: this.$route.query.outDetailId || '',
      type: this.$route.query.type, // 1: '补货' 2:'入库
      drug: {
        'drugCode': '41187',
        'drugName': '复方酮康唑乳膏',
        'drugId': '808080eb7d84c42a017d84c801ee0000',
        'pic': null,
        'specif': '7g/支'
      },
      inventoryList: []
    }
  },
  computed: {
    allNum() {
      let all = 0
      this.inventoryList.forEach(el => {
        if (el.cargoCapacity * 1 > this.stock * 1) {
          all += this.stock
        } else {
          all += el.cargoCapacity * 1
        }
      })
      return all
    },
    fillNum() {
      let fill = 0
      this.inventoryList.forEach(el => {
        el.codeList.forEach(el1 => {
          if (el1.code) {
            fill++
          }
        })
      })
      return fill
    }
  },
  mounted() {
    this.drugTraceabilityCodeDetail()
  },
  methods: {
    getLabel(index, len) {
      if (index == 0) {
        return `${len - index}位最里:`
      } else if (index == len - 1) {
        return '1位最外:'
      } else {
        return `第${len - index}位次:`
      }
    },
    drugTraceabilityCodeDetail() {
      const params = {
        deviceId: this.deviceId,
        drugId: this.drugId,
        inventoryIds: this.trackId
      }
      medicineAbinetApi.drugTraceabilityCodeDetail(params).then(res => {
        const data = res.obj
        this.drug = data.drug
        this.inventoryList = data.inventoryList.map(el => {
          let obj = {
            ...el,
            arr: []
          }
          var maxNum = el.cargoCapacity * 1 > this.stock * 1 ? this.stock * 1 : el.cargoCapacity * 1
          var len = obj.codeList.length
          if (obj.codeList.length < maxNum) {
            for (let i = 0; i < maxNum - len; i++) {
              obj.codeList.push({ code: '' })
            }
          }
          return obj
        })
      })
    },
    finish() {
      if (this.allNum == this.fillNum) {
        if (type == 1) {
          this.batchAddInventory()
        } else {
          this.updateMediicinecabineInventory()
        }
      } else {
        this.$dialog
          .confirm({
            message: '未录入所有追溯码,是否完成补货?请确保顺序靠后的位次为空,药品放在前置位。'
          })
          .then(() => {
            if (type == 1) {
              this.batchAddInventory()
            } else {
              this.updateMediicinecabineInventory
            }
          })
          .catch(() => {})
      }
    },
    batchAddInventory() {
      // 补货
      const list = []
      this.inventoryList.forEach(el => {
        const codeList = []
        el.codeList.forEach((item, index) => {
          codeList.push({
            code: item.code,
            sort: el.codeList.length - index
          })
        })
        list.push({
          cargoId: el.id,
          drugId: this.drugId,
          qty: el.qty,
          codeList
        })
      })
      let p = {
        list: JSON.stringify({ list: list }),
        userId: this.user.id,
        outDetailId: this.outDetailId
      }
      medicineAbinetApi.batchAddInventory(p).then(res => {
        if (res.status == 200) {
          this.$toast('操作成功!')
          this.backFn()
        }
      })
    },
    updateMediicinecabineInventory() {
      // 入库
      const codeList = []
      this.inventoryList[0].codeList.forEach((el, index) => {
        codeList.push({
          code: el.code,
          sort: el.codeList.length - index
        })
      })
      let p = {
        id: this.trackId,
        drugId: this.drugId,
        qty: this.stock,
        userId: this.user.id,
        codeStr: JSON.stringify(codeList)
      }
      medicineAbinetApi.updateMediicinecabineInventory(p).then(res => {
        if (res.status == 200) {
          this.$toast('入库成功!')
          this.backFn()
        }
      })
    },
    backFn() {
      this.$router.go(-1)
    },
    onSort(val, id) {
      const oldIndex = val.oldIndex
      const newIndex = val.newIndex
      const obj = this.inventoryList.find(el => el.id == id)
      const obj1 = obj.codeList[oldIndex]
      obj.codeList.splice(oldIndex, 1)
      obj.codeList.splice(newIndex, 0, obj1)
    }
  }
}
</script>
<style lang="scss" scoped>
.box {
  background: #fff;
  margin: 15px;
  padding: 12px;
  border-radius: 8px;
}
.img {
  width: 96px;
  height: 96px;
  border-radius: 8px;
  margin-right: 10px;
}
.info {
  width: 200px;
}
.title {
  margin-bottom: 6px;
}
.line {
  border: 1px solid #eaeaea;
  margin: 10px 0;
}
.code-box {
  border-radius: 8px;
  padding: 10px;
  border: 1px solid rgba($color: #000000, $alpha: 0.1);
}
.van-cell {
  background: #f1f3f4;
  padding: 6px 16px;
}
.scan {
  position: absolute;
  right: 36px;
  top: 50%;
  transform: translateY(-50%);
}
.drag {
  width: 20px;
  height: 20px;
}
.footer {
  background: #fff;
  height: 60px;
  display: flex;
  align-items: center;
  justify-content: space-around;
  font-size: 14px;
  .plain-btn {
    height: 40px;
    border-radius: 20px;
    padding: 0 24px;
    border: 1px solid #17b3ec;
    color: #17b3ec;
    line-height: 40px;
  }
  .primary-btn {
    background: #17b3ec;
    height: 40px;
    border-radius: 20px;
    padding: 0 24px;
    color: #fff;
    line-height: 40px;
  }
}
::v-deep {
  .van-field__label {
    color: #999;
    width: 60px;
    margin-right: 0;
  }
}
</style>