| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431 |
- <template>
- <view class="container" style="padding-top: 44px; padding-bottom: 10px">
- <u-navbar
- title="快速盘点"
- :border="false"
- fixed
- safe-area-inset-top
- bgColor="#22ac38"
- titleStyle="font-size:36rpx;color:#fff"
- >
- <template #left>
- <u-icon name="arrow-left" color="#fff" size="20" @click="goBack"></u-icon>
- </template>
- <template #right>
- <text style="color: #ffffff" @click="onSubmit" v-permission="'app:wms:speedyCheck:confirm'">提交</text>
- </template>
- </u-navbar>
- <!-- 盘点信息选择区域 -->
- <view class="select-area" style="margin-top: 44px">
- <u-cell-group :border="true">
- <u-cell
- title="盘点方式"
- :value="checkMethod"
- @click="showCheckMethodPicker = true"
- :isLink="true"
- :border="true"
- />
- <u-cell
- title="目标库位"
- :value="location"
- @click="handleLocationSelect"
- :isLink="true"
- :border="true"
- />
- </u-cell-group>
- </view>
- <!-- 盘点方式选择器 -->
- <u-picker
- :show="showCheckMethodPicker"
- :columns="[checkMethodOptions]"
- @confirm="onCheckMethodConfirm"
- @cancel="showCheckMethodPicker = false"
- :showToolbar="true"
- title="选择盘点方式"
- :closeOnClickOverlay="true"
- cancelText="取消"
- confirmText="确定"
- :itemHeight="44"
- :visibleItemCount="5"
- />
- <!-- 库位选择弹出层 -->
- <u-popup
- :show="showLocationPopup"
- mode="bottom"
- @close="close"
- @open="open"
- :closeOnClickOverlay="true"
- :round="10"
- :safeAreaInsetBottom="true"
- >
- <view class="location-popup">
- <view class="location-popup-header">
- <text style="flex: 1">选择库位</text>
- <u-text
- style="width: 40px; flex: none"
- type="primary"
- text="确定"
- @click="onLocationConfirm"
- ></u-text>
- </view>
- <view class="location-popup-content">
- <u-input
- v-model="selectedLocation"
- placeholder="请输入库位"
- customStyle="background-color: #f6f6f6; border-radius: 4px;"
- ></u-input>
- <view class="location-list">
- <view
- class="location-item"
- v-for="(item, index) in locationList"
- :key="index"
- @click="selectLocation(item)"
- :class="{ 'location-item--selected': selectedLocation === item }"
- >
- <text>{{ item }}</text>
- <u-icon
- v-if="selectedLocation === item"
- name="checkmark"
- color="#19be6b"
- size="16"
- ></u-icon>
- </view>
- </view>
- </view>
- </view>
- </u-popup>
- <!-- 订单列表 -->
- <view class="product-details">
- <LocationOrderItem v-for="(item, index) in products" :isLink :key="index" :item="item" />
- </view>
- <view class="add-btn" @click="handleAdd">
- <u-icon name="plus-circle" size="40" color="#19be6b"></u-icon>
- </view>
- <!-- 底部扫码输入框 -->
- <view class="fixed-bottom pad-20" style="background: #ffffff">
- <u-search
- placeholder="请输入快递单号/订单编号"
- v-model="searchValue"
- @search="onSearch"
- :show-action="false"
- custom-style="margin-right:10px"
- ></u-search>
- <u-icon name="scan" size="28" color="#19be6b" @click="openScan"></u-icon>
- </view>
- </view>
- </template>
- <script setup>
- import { reactive, ref } from "vue";
- import { onLoad, onUnload, onShow } from "@dcloudio/uni-app";
- import LocationOrderItem from "./components/LocationOrderItem.vue";
- const goBack = () => {
- uni.navigateBack();
- };
- // 盘点方式相关
- const showCheckMethodPicker = ref(false);
- const checkMethod = ref("增加数量");
- const resetType = ref(0);
- const checkMethodOptions = ["增加数量", "减少数量", "实际数量"];
- //获取用户的默认仓库
- const getUserDefaultWarehouse = () => {
- uni.$u.http.get("/app/appUser/getUserBindGodown").then((res) => {
- if (res.code == 200) {
- getWarehouseLocationList(res.data.id);
- }
- });
- };
- // 库位相关
- const location = ref("");
- const showLocationPopup = ref(false);
- const locationList = ref([]);
- const selectedLocation = ref("");
- //根据仓库获取库位列表
- const godownId = ref();
- const getWarehouseLocationList = (id) => {
- godownId.value = id;
- uni.$u.http.get("/app/stock/getGodownPosition?godownId=" + id).then((res) => {
- if (res.code == 200) {
- locationList.value = res.data;
- }
- });
- };
- // 其他数据
- const searchValue = ref("");
- const products = ref([]);
- // 方法
- const onCheckMethodConfirm = (e) => {
- checkMethod.value = e.value[0];
- showCheckMethodPicker.value = false;
- resetType.value = e.indexs[0] + 1;
- };
- const handleLocationSelect = () => {
- showLocationPopup.value = true;
- selectedLocation.value = location.value;
- };
- const onLocationConfirm = () => {
- if (selectedLocation.value) {
- location.value = selectedLocation.value;
- showLocationPopup.value = false;
- selectedLocation.value = "";
- } else {
- uni.$u.toast("请选择库位");
- }
- };
- const selectLocation = (item) => {
- selectedLocation.value = item;
- };
- //盘点提交
- const onSubmit = () => {
- if (products.value.length === 0) {
- uni.$u.toast("请添加盘点订单");
- return;
- }
- // 构建请求参数
- const params = {
- godownId: godownId.value,
- positionCode: location.value,
- orderInfo: products.value.map((item) => ({
- orderId: item.orderId,
- waybillCode: item.waybillCode,
- bookNum: item.badNum,
- remark: item.remark || "",
- })),
- resetType: resetType.value,
- };
- // 调用重置库存API
- uni.$u.http
- .post("/app/stock/resetStock", params)
- .then((res) => {
- if (res.code === 200) {
- uni.$u.toast("盘点成功");
- location.value = "";
- products.value = [];
- searchValue.value = "";
- } else {
- uni.$u.toast(res.msg || "提交失败");
- }
- })
- .catch((err) => {
- uni.$u.toast("提交失败");
- console.error(err);
- });
- };
- const onSearch = () => {
- if (!searchValue.value) {
- uni.$u.toast("请输入快递单号或订单编号");
- return;
- }
- // 判断搜索类型:纯数字为订单号(searchType=1),字母+数字组合为物流单号(searchType=2)
- const searchType = /^[0-9]+$/.test(searchValue.value) ? 1 : 2;
- // 调用接口获取订单数据
- uni.$u.http
- .get("/app/stock/searchBadOrderForReset", {
- params: {
- searchType,
- search: searchValue.value,
- },
- })
- .then((res) => {
- if (res.code === 200) {
- // 检查是否已存在相同订单
- const exists = products.value.some(
- (item) => item.orderId === res.data.orderId || item.waybillCode === res.data.waybillCode
- );
- if (exists) {
- uni.$u.toast("该订单已添加");
- return;
- }
- // 添加新订单到列表
- products.value.push(res.data);
- searchValue.value = ""; // 清空搜索框
- } else {
- uni.$u.toast(res.msg || "查询失败");
- }
- })
- .catch((err) => {
- uni.$u.toast("查询失败");
- console.error(err);
- });
- };
- //扫码后判断是否是库位还是 订单号
- const dealScanText = (scanText) => {
- let isLocation = scanText.indexOf("-") > -1 && scanText.length == 9;
- if (isLocation) {
- if (locationList.value.includes(scanText)) {
- location.value = scanText;
- } else {
- uni.$u.toast("库位不存在");
- }
- } else {
- searchValue.value = scanText;
- onSearch();
- }
- };
- const openScan = () => {
- // #ifdef APP-PLUS || MP-WEIXIN
- uni.scanCode({
- success: (res) => {
- dealScanText(res.result);
- },
- fail: (err) => {
- uni.showToast({
- title: "扫码失败",
- icon: "error",
- });
- },
- });
- // #endif
- };
- function handleAdd() {
- if (!location.value) {
- uni.$u.toast("请选择库位");
- return;
- }
- uni.navigateTo({
- url: "/pages/index/wms/speedy-check-add?location=" + location.value,
- });
- }
- const close = () => {
- showLocationPopup.value = false;
- };
- const open = () => {
- showLocationPopup.value = true;
- };
- onLoad(() => {
- getUserDefaultWarehouse();
- uni.$on("selectedProducts", (list) => {
- products.value = list;
- });
- // #ifdef APP-PLUS
- uni.$u.useGlobalEvent((e) => {
- if (e.barcode) {
- dealScanText(e.barcode);
- }
- });
- // #endif
- });
- onShow(() => {
- uni.$u.updateActivePageOnShow()
- })
- onUnload(() => {
- uni.$off("selectedProducts");
- });
- </script>
- <style lang="scss" scoped>
- .select-area {
- background-color: #fff;
- }
- .product-details {
- margin-bottom: 120rpx; // 为底部搜索框留出空间
- }
- .fixed-bottom {
- position: fixed;
- bottom: 0;
- left: 0;
- right: 0;
- display: flex;
- align-items: center;
- padding: 20rpx;
- background: #ffffff;
- box-shadow: 0 -2rpx 6rpx rgba(0, 0, 0, 0.1);
- }
- .add-btn {
- position: fixed;
- right: 0;
- bottom: 30%;
- z-index: 99;
- cursor: pointer;
- }
- .location-input-wrapper {
- display: flex;
- align-items: center;
- justify-content: flex-end;
- }
- .location-popup {
- background-color: #fff;
- padding: 20rpx;
- &-header {
- display: flex;
- justify-content: space-between;
- align-items: center;
- padding-bottom: 24rpx;
- border-bottom: 1px solid #eee;
- position: relative;
- text {
- font-size: 32rpx;
- font-weight: 500;
- flex: 1;
- }
- .u-button {
- position: absolute;
- right: 20rpx;
- }
- }
- &-content {
- padding: 20rpx 0;
- max-height: 600rpx;
- overflow-y: auto;
- }
- }
- .location-list {
- margin-top: 20rpx;
- max-height: 400rpx;
- overflow-y: auto;
- }
- .location-item {
- padding: 20rpx;
- border-bottom: 1px solid #eee;
- display: flex;
- align-items: center;
- justify-content: space-between;
- &--selected {
- color: #19be6b;
- }
- }
- </style>
|