647556386
11 小时以前 b5d01891fbbd69d8d50d2b4fb562fac3130fc2d6
ÏîÄ¿´úÂë/WIDESEA_WMSClient/src/extension/outbound/extend/NoStockOut.vue
@@ -8,71 +8,71 @@
      title="虚拟出入库"
      class="custom-vol-box"
    >
      <!-- æäº¤é®ç½©å±‚:覆盖整个弹窗内容区域 -->
      <div class="submit-mask" v-show="submitLoading">
        <div class="mask-content">
          <el-loading-spinner class="loading-icon" />
          <span class="loading-text">正在提交出库,请稍候...</span>
        </div>
      </div>
      <div>
        <!-- å•据选择区域(可输入搜索) -->
        <el-form :inline="true" :model="orderForm" style="margin-bottom: 20px; align-items: flex-end;">
          <el-form-item label="出库单据:" name="outboundOrderId">
            <el-select
              v-model="orderForm.outboundOrderId"
              placeholder="请选择或扫描出库单据号"
        <!-- å•据输入区域(支持扫码) -->
        <el-form
          :inline="true"
          :model="orderForm"
          style="margin-bottom: 20px; align-items: flex-end;"
          @submit.prevent
        >
          <el-form-item label="出库单据:" name="outboundOrderNo">
            <el-input
              v-model="orderForm.outboundOrderNo"
              placeholder="请输入或扫描出库单据号"
              clearable
              style="width: 220px; margin-right: 10px;"
              @change="handleOrderChange"
              @visible-change="handleOutboundVisibleChange"
              :loading="loading"
              filterable
              :filter-method="filterOutboundOrders"
              @input="handleOutboundScanInput"
              ref="outboundSelectRef"
            >
              <el-option
                v-for="order in filteredOutboundOrders"
                :key="order.id"
                :label="order.orderNo"
                :value="order.id"
              ></el-option>
            </el-select>
              @input="handleOutboundInput"
              @keyup.enter="validateOutboundOrder"
              ref="outboundInputRef"
              :disabled="loading || submitLoading || isOutboundVerified"
            ></el-input>
            <span v-if="isOutboundVerified" class="verified-tag">✓ å·²éªŒè¯</span>
            <span v-else-if="loading" class="loading-tag">✦ éªŒè¯ä¸­...</span>
          </el-form-item>
          <el-form-item label="采购单据:" name="purchaseOrderId">
            <el-select
              v-model="orderForm.purchaseOrderId"
              placeholder="请选择或扫描采购单据号"
          <el-form-item label="采购单据:" name="purchaseOrderNo">
            <el-input
              v-model="orderForm.purchaseOrderNo"
              placeholder="扫码条码后自动填充"
              clearable
              style="width: 220px; margin-right: 10px;"
              @change="handleOrderChange"
              @visible-change="handlePurchaseVisibleChange"
              :loading="loading"
              filterable
              :filter-method="filterPurchaseOrders"
              @input="handlePurchaseScanInput"
              ref="purchaseSelectRef"
            >
              <el-option
                v-for="order in filteredPurchaseOrders"
                :key="order.id"
                :label="order.orderNo"
                :value="order.id"
              ></el-option>
            </el-select>
              @input="handlePurchaseInput"
              readonly
              ref="purchaseInputRef"
              :disabled="submitLoading"
            ></el-input>
          </el-form-item>
        </el-form>
        <!-- ä¸Šæ–¹è¾“入框 -->
        <el-form :inline="true" :model="formData" ref="formRef" style="margin-bottom: 20px; align-items: flex-end;">
        <el-form
          :inline="true"
          :model="formData"
          ref="formRef"
          style="margin-bottom: 20px; align-items: flex-end;"
          @submit.prevent
        >
          <el-form-item
            label="扫描条码:"
            style="width: 80%"
            name="barcode"
            :rules="[{ required: true, message: '请扫描或输入条码', trigger: 'blur' }]"
          >
            <el-input
              ref="barcodeInputRef"
              v-model="formData.barcode"
              placeholder="请使用扫码枪扫描条码,或手动输入"
              @keyup.enter="handleScan"
              @keydown.enter="debouncedHandleScan"
              autofocus
              class="custom-input"
              :disabled="!orderForm.outboundOrderId || !orderForm.purchaseOrderId"
              :disabled="!isOutboundVerified || loading || submitLoading"
            ></el-input>
          </el-form-item>
          <el-form-item>
@@ -81,7 +81,7 @@
              size="small" 
              @click="handleScan" 
              class="custom-button"
              :disabled="!orderForm.outboundOrderId || !orderForm.purchaseOrderId || loading"
              :disabled="!isOutboundVerified || loading || submitLoading"
            >
              <Search /> ç¡®è®¤æ‰«æ
            </el-button>
@@ -97,19 +97,28 @@
            <div class="card-body">
              <el-scrollbar height="400px" class="custom-scrollbar">
                <transition-group name="barcode-item-transition">
                  <div class="barcode-item" v-for="(item, index) in scannedBarcodes" :key="item.barcode" :data-index="index">
                    <span class="barcode-text">{{ index + 1 }}. {{ item.barcode }}</span>
                  <div class="barcode-item" v-for="(item, index) in scannedBarcodes" :key="`${item.barcode}-${index}`">
                    <div class="barcode-detail">
                      <div class="detail-row"><span class="label">条码:</span><span class="value">{{ item.barcode || '-' }}</span></div>
                      <div class="detail-row"><span class="label">物料编码:</span><span class="value">{{ item.materielCode || '-' }}</span></div>
                      <div class="detail-row"><span class="label">物料名称:</span><span class="value">{{ item.materielName || '-' }}</span></div>
                      <div class="detail-row"><span class="label">批次号:</span><span class="value">{{ item.batchNo || '-' }}</span></div>
                      <div class="detail-row"><span class="label">条码数量:</span><span class="value">{{ item.orderQuantity || item.quantity || 0 }}</span></div>
                      <div class="detail-row"><span class="label">供应商编码:</span><span class="value">{{ item.supplyCode || '-' }}</span></div>
                      <div class="detail-row"><span class="label">采购单号:</span><span class="value">{{ item.purchaseOrderNo || '-' }}</span></div>
                    </div>
                    <el-button
                      class="delete-btn"
                      @click="removeItem(index, item.barcode)"
                      icon="Delete"
                      circle
                      :disabled="loading"
                      :disabled="loading || submitLoading"
                    ></el-button>
                  </div>
                </transition-group>
                <div class="empty-tip" v-if="scannedBarcodes.length === 0">
                  <span>暂无扫描记录,请先选择单据后扫描条码</span>
                  <span v-if="isOutboundVerified">暂无扫描记录,请扫描条码</span>
                  <span v-else>请先输入并验证有效的出库单据号</span>
                </div>
              </el-scrollbar>
            </div>
@@ -123,12 +132,18 @@
            type="primary" 
            size="small" 
            @click="submit" 
            :disabled="scannedBarcodes.length === 0 || !orderForm.outboundOrderId || !orderForm.purchaseOrderId || loading"
            :disabled="scannedBarcodes.length === 0 || !isOutboundVerified || loading || submitLoading"
            class="submit-btn"
          >
            <Check /> æäº¤å‡ºåº“
          </el-button>
          <el-button type="text" size="small" @click="showDetailBox = false" class="cancel-btn" :disabled="loading">
          <el-button
            type="text"
            size="small"
            @click="handleCancel"
            class="cancel-btn"
            :disabled="loading || submitLoading"
          >
            å–消
          </el-button>
        </div>
@@ -138,310 +153,279 @@
</template>
<script setup>
import { ref, reactive, onMounted, nextTick } from 'vue';
import { ref, reactive, onMounted, onBeforeUnmount, nextTick, watch } from 'vue';
import { ElMessage } from 'element-plus';
import { Search, Check } from '@element-plus/icons-vue';
import VolBox from "@/components/basic/VolBox.vue";
import http from '@/api/http';
// å“åº”式数据
const showDetailBox = ref(false);
const orderForm = reactive({
  outboundOrderId: "",
  purchaseOrderId: ""
});
const formData = reactive({
  barcode: "",
});
const orderForm = reactive({ outboundOrderNo: "", purchaseOrderNo: "" });
const formData = reactive({ barcode: "" });
const scannedBarcodes = ref([]);
const outboundOrders = ref([]);
const purchaseOrders = ref([]);
const filteredOutboundOrders = ref([]);
const filteredPurchaseOrders = ref([]);
const loading = ref(false);
const submitLoading = ref(false);
const isOutboundVerified = ref(false);
let unlockCalled = false; // é˜²æ­¢é‡å¤è§£é”
// æ¨¡æ¿å¼•用
const formRef = ref(null);
const barcodeInputRef = ref(null);
const outboundSelectRef = ref(null); // æ–°å¢žï¼šå‡ºåº“单select的ref
const purchaseSelectRef = ref(null); // æ–°å¢žï¼šé‡‡è´­å•select的ref
const outboundInputRef = ref(null);
const purchaseInputRef = ref(null);
// ç”¨äºŽé˜²æ­¢è¾“入事件和change事件冲突的锁
const isProcessingScan = ref(false);
// éŸ³é¢‘资源
const successAudioSrc = require('@/assets/audio/success.mp3');
const errorAudioSrc = require('@/assets/audio/error.mp3');
const playAudio = (src, volume = 0.8) => {
  try { const audio = new Audio(src); audio.volume = volume; audio.play().catch(() => {}); } catch(e) {}
};
const playSuccess = () => playAudio(successAudioSrc);
const playError = () => playAudio(errorAudioSrc);
// ç»„件挂载时初始化过滤后的列表
onMounted(() => {
  filteredOutboundOrders.value = [...outboundOrders.value];
  filteredPurchaseOrders.value = [...purchaseOrders.value];
// ========== è§£é”æ ¸å¿ƒé€»è¾‘(使用新接口 MovePickingOrders) ==========
const unlockOutboundOrder = async () => {
  if (!isOutboundVerified.value || !orderForm.outboundOrderNo?.trim()) return;
  if (unlockCalled) return;
  unlockCalled = true;
  const outboundOrderNo = orderForm.outboundOrderNo.trim();
  try {
    // æ›¿æ¢ä¸º MovePickingOrders æŽ¥å£
    await http.post(`/api/OutboundPicking/MovePickingOrders?outOrder=${outboundOrderNo}`, null, "解锁出库单中...").catch(() => {});
  } catch (error) {
    // é™é»˜å¤±è´¥
  }
};
// é¡µé¢å…³é—­/刷新时的同步解锁(使用 sendBeacon æˆ– fetch keepalive)
const unlockOnPageUnload = () => {
  if (!isOutboundVerified.value || !orderForm.outboundOrderNo?.trim()) return;
  const outboundOrderNo = orderForm.outboundOrderNo.trim();
  const url = `/api/OutboundPicking/MovePickingOrders?outOrder=${encodeURIComponent(outboundOrderNo)}`;
  // ä½¿ç”¨ fetch keepalive å‘送
  fetch(url, { method: 'POST', keepalive: true, headers: { 'Content-Type': 'application/json' } }).catch(() => {});
};
// ç›‘听弹窗关闭(任何方式:叉号、ESC、遮罩、取消按钮等)
watch(showDetailBox, (newVal, oldVal) => {
  if (oldVal === true && newVal === false) {
    // å¼¹çª—关闭时调用解锁
    unlockOutboundOrder();
  }
});
// æ‰“开弹窗
// å–消按钮
const handleCancel = (e) => {
  e?.stopPropagation();
  e?.preventDefault();
  showDetailBox.value = false;
};
// æ‰“开弹窗(重置所有状态)
const open = () => {
  unlockCalled = false;
  showDetailBox.value = true;
  scannedBarcodes.value = [];
  formData.barcode = "";
  orderForm.outboundOrderId = "";
  orderForm.purchaseOrderId = "";
  nextTick(() => {
    barcodeInputRef.value?.focus();
  });
  orderForm.outboundOrderNo = "";
  orderForm.purchaseOrderNo = "";
  submitLoading.value = false;
  isOutboundVerified.value = false;
  nextTick(() => outboundInputRef.value?.focus());
};
// åŠ è½½å‡ºåº“å•æ®åˆ—è¡¨
const loadOutboundOrders = async () => {
  if (outboundOrders.value.length > 0) return;
// éªŒè¯å‡ºåº“单号(仍使用原接口,此接口用于验证单据有效性)
const validateOutboundOrder = async () => {
  const outboundOrderNo = orderForm.outboundOrderNo.trim();
  if (!outboundOrderNo) { ElMessage.warning("请输入出库单据号"); return; }
  try {
    loading.value = true;
    const res = await http.get("/api/OutboundPicking/GetAvailablePickingOrders");
    if (res.code === 0) {
      outboundOrders.value = res.data.map(orderNo => ({
        id: orderNo,
        orderNo: orderNo
      }));
      filteredOutboundOrders.value = [...outboundOrders.value];
    } else {
      ElMessage.error("加载出库单据失败:" + (res.message || '未知错误'));
    const res = await http.post(`/api/OutboundPicking/GetAvailablePickingOrders?outOrder=${outboundOrderNo}`, "验证出库单据号中...");
    if (res.status !== true) {
      orderForm.outboundOrderNo = "";
      isOutboundVerified.value = false;
      ElMessage.error(res.message || "出库单据号验证失败");
      nextTick(() => outboundInputRef.value?.focus());
      return;
    }
    isOutboundVerified.value = true;
    ElMessage.success("出库单据号验证通过");
    nextTick(() => barcodeInputRef.value?.focus());
  } catch (error) {
    ElMessage.error("加载出库单据异常:" + error.message);
    orderForm.outboundOrderNo = "";
    isOutboundVerified.value = false;
    ElMessage.error(`验证异常:${error.message || "网络错误"}`);
    nextTick(() => outboundInputRef.value?.focus());
  } finally {
    loading.value = false;
  }
};
// åŠ è½½é‡‡è´­å•æ®åˆ—è¡¨
const loadPurchaseOrders = async () => {
  if (purchaseOrders.value.length > 0) return;
  try {
    loading.value = true;
    const res = await http.get("/api/OutboundPicking/GetAvailablePurchaseOrders");
    if (res.status === true) {
      purchaseOrders.value = res.data.map(orderNo => ({
        id: orderNo,
        orderNo: orderNo
      }));
      filteredPurchaseOrders.value = [...purchaseOrders.value];
    } else {
      ElMessage.error("加载采购单据失败:" + (res.message || '未知错误'));
    }
  } catch (error) {
    ElMessage.error("加载采购单据异常:" + error.message);
  } finally {
    loading.value = false;
  }
const handleOutboundInput = (val) => { if (val?.trim()) isOutboundVerified.value = false; };
const handlePurchaseInput = (val) => {};
// èŽ·å–é‡‡è´­å•å·
const getPurchaseOrderByBarcode = async (barcode) => {
  const res = await http.post(`/api/OutboundPicking/GetPurchaseOrderByBarcode?barCode=${encodeURIComponent(barcode)}`, "查询采购单号中...");
  if (res.status !== true) throw new Error(res.message || "查询失败");
  let purchaseOrderNo = '';
  if (Array.isArray(res.data) && res.data.length > 0) purchaseOrderNo = res.data[0].purchaseOrderNo || res.data[0].orderId;
  else purchaseOrderNo = res.data?.purchaseOrderNo || res.data?.orderId;
  return purchaseOrderNo;
};
// å‡ºåº“单据过滤方法
const filterOutboundOrders = (value) => {
  if (!value) {
    filteredOutboundOrders.value = [...outboundOrders.value];
  } else {
    const lowerValue = value.toLowerCase();
    filteredOutboundOrders.value = outboundOrders.value.filter(order =>
      order.orderNo.toLowerCase().includes(lowerValue)
    );
  }
};
// é‡‡è´­å•据过滤方法
const filterPurchaseOrders = (value) => {
  if (!value) {
    filteredPurchaseOrders.value = [...purchaseOrders.value];
  } else {
    const lowerValue = value.toLowerCase();
    filteredPurchaseOrders.value = purchaseOrders.value.filter(order =>
      order.orderNo.toLowerCase().includes(lowerValue)
    );
  }
};
// å‡ºåº“单据下拉框显示/隐藏时触发
const handleOutboundVisibleChange = (visible) => {
  if (visible) {
    loadOutboundOrders();
  } else {
    // å½“下拉框关闭时,如果是扫描操作导致的,清空输入框
    if (isProcessingScan.value) {
        nextTick(() => {
            outboundSelectRef.value?.clearInput();
            isProcessingScan.value = false;
        });
    }
  }
};
// é‡‡è´­å•据下拉框显示/隐藏时触发
const handlePurchaseVisibleChange = (visible) => {
  if (visible) {
    loadPurchaseOrders();
  } else {
    // å½“下拉框关闭时,如果是扫描操作导致的,清空输入框
    if (isProcessingScan.value) {
        nextTick(() => {
            purchaseSelectRef.value?.clearInput();
            isProcessingScan.value = false;
        });
    }
  }
};
// å•据选择变化时触发
const handleOrderChange = () => {
  // å¦‚果是手动选择,则不清空输入框
  isProcessingScan.value = false;
  scannedBarcodes.value = [];
  nextTick(() => {
    barcodeInputRef.value?.focus();
  });
};
/**
 * å¤„理出库单扫描输入
 * @param {string} scanText - æ‰«ææžªè¾“入的文本
 */
const handleOutboundScanInput = async (scanText) => {
  // é¿å…å¤„理空字符串或由change事件触发的内部操作
  if (!scanText || isProcessingScan.value) return;
  // æŸ¥æ‰¾å®Œå…¨åŒ¹é…çš„订单
  const matchedOrder = outboundOrders.value.find(order => order.orderNo === scanText);
  if (matchedOrder) {
    isProcessingScan.value = true; // æ ‡è®°ä¸ºæ­£åœ¨å¤„理扫描
    // æ‰‹åŠ¨èµ‹å€¼
    orderForm.outboundOrderId = matchedOrder.id;
    ElMessage.success(`成功选择出库单:${matchedOrder.orderNo}`);
    // å»¶è¿Ÿèšç„¦åˆ°ä¸‹ä¸€ä¸ªè¾“入框,确保赋值完成
    setTimeout(() => {
        purchaseSelectRef.value?.focus();
    }, 100);
  }
  // å¦‚果没有匹配项,不做任何事,让el-select保持过滤状态,用户可以手动选择
};
/**
 * å¤„理采购单扫描输入
 * @param {string} scanText - æ‰«ææžªè¾“入的文本
 */
const handlePurchaseScanInput = async (scanText) => {
  if (!scanText || isProcessingScan.value) return;
  const matchedOrder = purchaseOrders.value.find(order => order.orderNo === scanText);
  if (matchedOrder) {
    isProcessingScan.value = true; // æ ‡è®°ä¸ºæ­£åœ¨å¤„理扫描
    orderForm.purchaseOrderId = matchedOrder.id;
    ElMessage.success(`成功选择采购单:${matchedOrder.orderNo}`);
    setTimeout(() => {
        barcodeInputRef.value?.focus();
    }, 100);
  }
};
// æ‰«ææ¡ç å¤„理
// æ‰«ææ¡ç 
const handleScan = async () => {
  if (!formRef.value) return;
  await formRef.value.validateField('barcode');
  if (!isOutboundVerified.value) {
    ElMessage.warning("请先验证出库单据号");
    playError();
    nextTick(() => outboundInputRef.value?.focus());
    return;
  }
  const barcode = formData.barcode.trim();
  if (scannedBarcodes.value.some(item => item.barcode === barcode)) {
    ElMessage.warning(`条码 ${barcode} å·²æ‰«æè¿‡ï¼Œè¯·å‹¿é‡å¤æ‰«æ`);
  if (!barcode) return;
  const isDuplicate = scannedBarcodes.value.some(item => item.barcode === barcode);
  if (isDuplicate) {
    ElMessage.warning(`条码【${barcode}】已存在`);
    playError();
    formData.barcode = "";
    nextTick(() => barcodeInputRef.value?.focus());
    return;
  }
  try {
    loading.value = true;
    const res = await http.post("/api/OutboundPicking/BarcodeValidate", {
      outOder: orderForm.outboundOrderId,
      inOder: orderForm.purchaseOrderId,
    let purchaseOrderNo = '';
    try {
      purchaseOrderNo = await getPurchaseOrderByBarcode(barcode);
      if (purchaseOrderNo) orderForm.purchaseOrderNo = purchaseOrderNo;
    } catch (e) { ElMessage.info("未查询到采购单号,继续验证条码"); playError(); }
    const validateRes = await http.post("/api/OutboundPicking/BarcodeValidate", {
      outOder: orderForm.outboundOrderNo,
      inOder: purchaseOrderNo || orderForm.purchaseOrderNo,
      barCode: barcode
    });
    if (res.status === true) {
      scannedBarcodes.value.push({ barcode });
      ElMessage.success("扫描成功");
    if (validateRes.status === true && Array.isArray(validateRes.data) && validateRes.data.length) {
      const newItems = validateRes.data.map(item => ({
        barcode: item.barcode || '',
        materielCode: item.materielCode || '',
        materielName: item.materielName || '',
        batchNo: item.batchNo || '',
        orderQuantity: item.orderQuantity || item.quantity || 0,
        supplyCode: item.supplyCode || '',
        purchaseOrderNo: purchaseOrderNo || ''
      }));
      scannedBarcodes.value.push(...newItems);
      ElMessage.success(`扫描成功,新增 ${newItems.length} æ¡`);
      playSuccess();
      formData.barcode = "";
    } else {
      ElMessage.error("扫描失败:" + (res.message || '验证失败'));
      ElMessage.error(validateRes.message || '条码验证失败');
      playError();
      formData.barcode = "";
    }
  } catch (error) {
    ElMessage.error("扫描验证异常:" + error.message);
    ElMessage.error('条码验证异常');
    playError();
    formData.barcode = "";
  } finally {
    loading.value = false;
    nextTick(() => barcodeInputRef.value?.focus());
  }
};
// ç§»é™¤å•条扫描记录
const debounce = (fn, delay = 100) => {
  let timer = null;
  return (...args) => { if (timer) clearTimeout(timer); timer = setTimeout(() => fn.apply(this, args), delay); };
};
const debouncedHandleScan = debounce(async (e) => { e.stopPropagation(); e.preventDefault(); await handleScan(); }, 100);
// åˆ é™¤æ¡ç 
const removeItem = async (index, barcode) => {
  try {
    loading.value = true;
    const currentItem = scannedBarcodes.value[index];
    const res = await http.post("/api/OutboundPicking/DeleteBarcode", {
      outOder: orderForm.outboundOrderId,
      inOder: orderForm.purchaseOrderId,
      outOder: orderForm.outboundOrderNo,
      inOder: currentItem?.purchaseOrderNo || orderForm.purchaseOrderNo,
      barCode: barcode
    });
    if (res.status === true) {
      scannedBarcodes.value.splice(index, 1);
      ElMessage.success("删除成功");
      if (scannedBarcodes.value.length === 0) orderForm.purchaseOrderNo = "";
    } else {
      ElMessage.error("删除失败:" + (res.message || '删除失败'));
      ElMessage.error("删除失败:" + (res.message || ''));
    }
  } catch (error) {
    ElMessage.error("删除条码异常:" + error.message);
    ElMessage.error("删除异常");
  } finally {
    loading.value = false;
    nextTick(() => barcodeInputRef.value?.focus());
  }
};
// æäº¤å‡ºåº“
const submit = async () => {
  if (scannedBarcodes.value.length === 0) {
    ElMessage.warning("请先扫描至少一条条码");
  if (!isOutboundVerified.value) {
    ElMessage.warning("出库单据号未验证");
    nextTick(() => outboundInputRef.value?.focus());
    return;
  }
  if (scannedBarcodes.value.length === 0) {
    ElMessage.warning("请先扫描条码");
    nextTick(() => barcodeInputRef.value?.focus());
    return;
  }
  const barcodes = scannedBarcodes.value.map(item => item.barcode);
  const purchaseOrderNos = [...new Set(scannedBarcodes.value.map(item => item.purchaseOrderNo).filter(Boolean))];
  try {
    loading.value = true;
    submitLoading.value = true;
    const res = await http.post("/api/OutboundPicking/NoStockOutSubmit", {
      OutOderSubmit: orderForm.outboundOrderId,
      InOderSubmit: orderForm.purchaseOrderId,
      OutOderSubmit: orderForm.outboundOrderNo,
      InOderSubmit: purchaseOrderNos.join(',') || '',
      BarCodeSubmit: barcodes
    });
    if (res.status === true) {
      ElMessage.success("出库提交成功");
      showDetailBox.value = false;
      unlockOutboundOrder();
      showDetailBox.value = false; // è§¦å‘ watch è§£é”ï¼ˆä½†æäº¤æˆåŠŸåŽé€šå¸¸ä¸éœ€è¦è§£é”ï¼Œå› ä¸ºå·²ç»å‡ºåº“äº†ï¼Œä¸è¿‡åŽç«¯åº”è‡ªè¡Œå¤„ç†çŠ¶æ€ï¼‰
      // æ³¨æ„ï¼šæäº¤æˆåŠŸåŽä¸åº”è¯¥å†è°ƒç”¨è§£é”æŽ¥å£ï¼Œå› ä¸ºå•æ®å·²å¤„ç†å®Œæ¯•ã€‚è¿™é‡Œé€šè¿‡æ ‡å¿—é¿å…ï¼šsubmit æˆåŠŸåŽé‡ç½®éªŒè¯çŠ¶æ€
      isOutboundVerified.value = false;
      unlockCalled = true; // é˜²æ­¢ watch å†æ¬¡è°ƒç”¨è§£é”
    } else {
      ElMessage.error("出库提交失败:" + (res.message || '提交失败'));
      unlockOutboundOrder();
      ElMessage.error("出库提交失败:" + (res.message || ''));
      nextTick(() => barcodeInputRef.value?.focus());
    }
  } catch (error) {
    ElMessage.error("出库提交异常:" + error.message);
    unlockOutboundOrder();
    ElMessage.error("提交异常:" + error.message);
    nextTick(() => barcodeInputRef.value?.focus());
  } finally {
    submitLoading.value = false;
    loading.value = false;
  }
};
// æš´éœ²ç»™çˆ¶ç»„件的方法
defineExpose({
  open
// ç”Ÿå‘½å‘¨æœŸï¼šæ·»åŠ é¡µé¢å…³é—­æ—¶çš„è§£é”ç›‘å¬
onMounted(() => {
  window.addEventListener('beforeunload', unlockOnPageUnload);
});
onBeforeUnmount(() => {
  window.removeEventListener('beforeunload', unlockOnPageUnload);
  // ç»„件卸载时如果弹窗还开着,也尝试解锁(但通常页面关闭会触发 beforeunload)
  if (showDetailBox.value) {
    unlockOutboundOrder();
  }
});
// æš´éœ²æ–¹æ³•
defineExpose({ open });
</script>
<style scoped>
/* å…³é”®ï¼šå®šä¹‰åˆ—表项的过渡动画 */
/* è¿‡æ¸¡åŠ¨ç”» */
.barcode-item-transition-enter-active,
.barcode-item-transition-leave-active {
  transition: all 0.3s ease;
@@ -454,15 +438,59 @@
  opacity: 0;
  transform: translateX(30px);
}
/* ç¡®ä¿åˆ é™¤æ—¶å…¶ä»–元素平滑上移 */
.barcode-item-transition-move {
  transition: transform 1s ease;
}
/* æäº¤é®ç½©å±‚样式 */
.submit-mask {
  position: absolute;
  top: 0;
  left: 0;
  width: 100%;
  height: 100%;
  background-color: rgba(255, 255, 255, 0.85);
  display: flex;
  justify-content: center;
  align-items: center;
  z-index: 100;
  border-radius: inherit;
}
.mask-content {
  display: flex;
  flex-direction: column;
  align-items: center;
  gap: 12px;
  color: #409eff;
  font-size: 15px;
}
.loading-icon {
  font-size: 24px;
  animation: el-loading-circle 1.5s linear infinite;
}
/* éªŒè¯çŠ¶æ€æ ‡ç­¾æ ·å¼ */
.verified-tag {
  color: #67c23a;
  font-size: 12px;
  margin-left: 8px;
  font-weight: 500;
}
.loading-tag {
  color: #409eff;
  font-size: 12px;
  margin-left: 8px;
  font-weight: 500;
  animation: spin 1s linear infinite;
}
@keyframes spin {
  0% { transform: rotate(0deg); }
  100% { transform: rotate(360deg); }
}
.scan-list {
  width: 100%;
}
.custom-card {
  border-radius: 12px;
  box-shadow: 0 4px 20px rgba(0, 0, 0, 0.08) !important;
@@ -471,7 +499,6 @@
.custom-card:hover {
  box-shadow: 0 10px 30px rgba(0, 0, 0, 0.12) !important;
}
.card-header {
  display: flex;
  justify-content: space-between;
@@ -485,12 +512,9 @@
  font-size: 15px;
  color: #333;
}
.card-body {
  padding: 0;
}
/* è‡ªå®šä¹‰æ»šåŠ¨æ¡ */
.custom-scrollbar :deep(.el-scrollbar__thumb) {
  background: rgba(0, 0, 0, 0.2);
  border-radius: 4px;
@@ -503,37 +527,58 @@
.custom-scrollbar :deep(.el-scrollbar__wrap) {
  overflow-x: hidden;
}
.barcode-item {
  display: flex;
  justify-content: space-between;
  align-items: center;
  padding: 10px 15px;
  align-items: flex-start;
  padding: 15px;
  border-bottom: 1px solid #f7f7f7;
  transition: background-color 0.2s ease;
}
.barcode-item:hover {
  background-color: #fafafa;
}
/* ä¸ºå¥‡æ•°è¡Œæ·»åŠ è½»å¾®çš„èƒŒæ™¯è‰²ï¼Œå¢žå¼ºå¯è¯»æ€§ */
.barcode-item:nth-child(odd) {
  background-color: #f9f9f9;
}
.barcode-text {
.barcode-detail {
  flex: 1;
  display: grid;
  grid-template-columns: repeat(4, 1fr);
  gap: 8px 15px;
  font-size: 14px;
}
@media (max-width: 1200px) {
  .barcode-detail {
    grid-template-columns: repeat(3, 1fr);
  }
}
@media (max-width: 992px) {
  .barcode-detail {
    grid-template-columns: repeat(2, 1fr);
  }
}
.detail-row {
  display: flex;
  align-items: center;
}
.label {
  color: #999;
  margin-right: 5px;
  white-space: nowrap;
}
.value {
  color: #666;
  transition: color 0.2s;
  flex: 1;
  word-break: break-all;
}
.barcode-item:hover .barcode-text {
  color: #409eff;
}
.delete-btn {
  color: #ea1919;
  font-size: 16px;
  transition: all 0.2s;
  opacity: 0.7;
  margin-left: 10px;
  flex-shrink: 0;
}
.barcode-item:hover .delete-btn {
  opacity: 1;
@@ -542,7 +587,6 @@
  color: #f56c6c !important;
  background: rgba(245, 108, 108, 0.1);
}
.empty-tip {
  text-align: center;
  padding: 80px 0;
@@ -558,8 +602,6 @@
  font-size: 40px;
  color: #dcdfe6;
}
/* è‡ªå®šä¹‰è¾“入框 */
.custom-input :deep(.el-input__inner) {
  border-radius: 6px;
  border-color: #e4e7ed;
@@ -571,8 +613,6 @@
  border-color: #409eff;
  box-shadow: 0 0 0 3px rgba(64, 158, 255, 0.1);
}
/* è‡ªå®šä¹‰æŒ‰é’® */
.custom-button {
  border-radius: 6px;
  height: 36px;
@@ -585,7 +625,6 @@
  transform: translateY(-1px);
  box-shadow: 0 4px 12px rgba(64, 158, 255, 0.2);
}
.footer-actions {
  text-align: right;
}
@@ -611,7 +650,6 @@
</style>
<style>
/* å…¨å±€æ ·å¼éƒ¨åˆ†ä¿æŒä¸å˜ */
.text-button:hover {
  background-color: #f0f9eb !important;
}