From 23ff9ba58505814008984ffd4b8125cc67b9cc12 Mon Sep 17 00:00:00 2001
From: yyb <995253665@qq.com>
Date: 星期五, 27 三月 2026 09:10:14 +0800
Subject: [PATCH] 销售台账,其他金额维护,工艺路线维护,客户地区维护

---
 src/views/basicData/customerFile/index.vue                                         |   46 +
 src/views/salesManagement/salesLedger/components/OtherAmountMaintenanceButton.vue  |  278 +++++++++
 src/views/salesManagement/salesLedger/components/ProcessFlowConfigSelectDialog.vue |  234 ++++++++
 src/views/salesManagement/salesLedger/index.vue                                    |  543 ++++++++----------
 src/views/salesManagement/salesLedger/components/ProcessFlowMaintenanceButton.vue  |  525 ++++++++++++++++++
 src/api/salesManagement/salesProcessFlowConfig.js                                  |   89 +++
 6 files changed, 1,394 insertions(+), 321 deletions(-)

diff --git a/src/api/salesManagement/salesProcessFlowConfig.js b/src/api/salesManagement/salesProcessFlowConfig.js
new file mode 100644
index 0000000..3e4fec1
--- /dev/null
+++ b/src/api/salesManagement/salesProcessFlowConfig.js
@@ -0,0 +1,89 @@
+// 宸ヨ壓娴佺▼閰嶇疆锛堟ā鏉匡級涓庣粦瀹氬埌閿�鍞彴璐︿骇鍝佺殑鎺ュ彛
+import request from "@/utils/request";
+
+// 宸ヨ壓璺嚎鍒嗛〉鍒楄〃
+export function salesProcessFlowConfigList(query = {}) {
+  return request({
+    url: "/processRoute/page",
+    method: "get",
+    params: query,
+  });
+}
+
+// 宸ヨ壓璺嚎璇︽儏锛堝吋瀹规棫璋冪敤锛�
+export function salesProcessFlowConfigGetById(configId) {
+  return request({
+    url: `/processRoute/${configId}`,
+    method: "get",
+  });
+}
+
+// 鏂板/缂栬緫宸ヨ壓璺嚎
+export function salesProcessFlowConfigUpsert(data) {
+  return request({
+    url: "/processRoute",
+    method: "post",
+    data: data,
+  });
+}
+
+// 鍒犻櫎宸ヨ壓璺嚎
+export function salesProcessFlowConfigDelete(configId) {
+  return request({
+    url: `/processRoute/${configId}`,
+    method: "delete",
+  });
+}
+
+// 璁剧疆榛樿宸ヨ壓璺嚎
+export function salesProcessFlowConfigSetDefault(configId) {
+  return request({
+    url: `/processRoute/default/${configId}`,
+    method: "put",
+  });
+}
+
+// 鏌ヨ宸ヨ壓璺嚎涓嬬殑宸ュ簭
+export function salesProcessFlowConfigItemList(routeId) {
+  return request({
+    url: "/processRouteItem/list",
+    method: "get",
+    params: { routeId },
+  });
+}
+
+// 鏂板/淇敼宸ュ簭
+export function salesProcessFlowConfigItemUpsert(data) {
+  return request({
+    url: "/processRouteItem/",
+    method: "post",
+    data,
+  });
+}
+
+// 宸ュ簭鎺掑簭鎺ュ彛
+export function salesProcessFlowConfigItemSort(data) {
+  return request({
+    url: "/processRouteItem/sort",
+    method: "post",
+    data,
+  });
+}
+
+// 鍒犻櫎宸ュ簭
+export function salesProcessFlowConfigItemDelete(itemId) {
+  return request({
+    url: `/processRouteItem/batchDelete/${itemId}`,
+    method: "delete",
+  });
+}
+
+// 灏嗘煇濂楅厤缃簲鐢ㄥ埌鍏蜂綋浜у搧锛堝悓姝ユ洿鏂帮級
+export function salesLedgerProductSetProcessFlowConfig(data) {
+  return request({
+    url: "/salesLedgerProduct/setProcessFlowConfig",
+    method: "post",
+    data: data,
+  });
+}
+
diff --git a/src/views/basicData/customerFile/index.vue b/src/views/basicData/customerFile/index.vue
index 79b1616..a800885 100644
--- a/src/views/basicData/customerFile/index.vue
+++ b/src/views/basicData/customerFile/index.vue
@@ -76,7 +76,7 @@
         </el-row>
         <el-row :gutter="30">
           <el-col :span="12">
-            <el-form-item label="鍏徃鍦板潃锛�"
+            <el-form-item label="瀹㈡埛鍦板潃锛�"
                           prop="companyAddress">
               <el-input v-model="form.companyAddress"
                         placeholder="璇疯緭鍏�"
@@ -84,11 +84,35 @@
             </el-form-item>
           </el-col>
           <el-col :span="12">
-            <el-form-item label="鍏徃鐢佃瘽锛�"
+            <el-form-item label="瀹㈡埛鍦板尯锛�"
+                          prop="regions">
+              <el-input v-model="form.regions"
+                        placeholder="璇疯緭鍏�"
+                        clearable />
+            </el-form-item>
+          </el-col>
+
+        </el-row>
+        <el-row :gutter="30">
+          <el-col :span="12">
+            <el-form-item label="瀹㈡埛鐢佃瘽锛�"
                           prop="companyPhone">
               <el-input v-model="form.companyPhone"
                         placeholder="璇疯緭鍏�"
                         clearable />
+            </el-form-item>
+          </el-col>
+          <el-col :span="12">
+            <el-form-item label="瀹㈡埛鍒嗙被锛�"
+                          prop="customerType">
+              <el-select v-model="form.customerType"
+                         placeholder="璇烽�夋嫨"
+                         clearable>
+                <el-option label="闆跺敭瀹㈡埛"
+                           value="闆跺敭瀹㈡埛" />
+                <el-option label="杩涢攢鍟嗗鎴�"
+                           value="杩涢攢鍟嗗鎴�" />
+              </el-select>
             </el-form-item>
           </el-col>
         </el-row>
@@ -117,19 +141,6 @@
               <el-input v-model="form.bankCode"
                         placeholder="璇疯緭鍏�"
                         clearable />
-            </el-form-item>
-          </el-col>
-          <el-col :span="12">
-            <el-form-item label="瀹㈡埛鍒嗙被锛�"
-                          prop="customerType">
-              <el-select v-model="form.customerType"
-                         placeholder="璇烽�夋嫨"
-                         clearable>
-                <el-option label="闆跺敭瀹㈡埛"
-                           value="闆跺敭瀹㈡埛" />
-                <el-option label="杩涢攢鍟嗗鎴�"
-                           value="杩涢攢鍟嗗鎴�" />
-              </el-select>
             </el-form-item>
           </el-col>
         </el-row>
@@ -720,6 +731,11 @@
       width: 220,
     },
     {
+      label: "瀹㈡埛鍦板尯",
+      prop: "regions",
+      width: 120,
+    },
+    {
       label: "绾崇◣浜鸿瘑鍒爜",
       prop: "taxpayerIdentificationNumber",
       width: 220,
diff --git a/src/views/salesManagement/salesLedger/components/OtherAmountMaintenanceButton.vue b/src/views/salesManagement/salesLedger/components/OtherAmountMaintenanceButton.vue
new file mode 100644
index 0000000..62389b5
--- /dev/null
+++ b/src/views/salesManagement/salesLedger/components/OtherAmountMaintenanceButton.vue
@@ -0,0 +1,278 @@
+<template>
+  <div style="display: inline-block;">
+    <el-button type="primary" plain @click="openDialog">
+      鍏朵粬閲戦缁存姢
+    </el-button>
+
+    <el-dialog
+      v-model="visible"
+      title="鍏朵粬閲戦缁存姢"
+      width="80%"
+      :close-on-click-modal="false"
+      @close="closeDialog"
+    >
+      <el-row :gutter="20">
+        <el-col :span="14">
+          <el-table
+            :data="records"
+            border
+            v-loading="loading"
+            height="55vh"
+          >
+            <el-table-column label="缂栫爜" prop="code" min-width="120" show-overflow-tooltip />
+            <el-table-column label="椤圭洰" prop="processName" min-width="180" show-overflow-tooltip />
+            <el-table-column label="鏁伴噺" prop="quantity" min-width="110" :formatter="formattedNumber" />
+            <el-table-column label="鍗曚环(鍏�)" prop="unitPrice" min-width="130" :formatter="formattedNumber" />
+            <el-table-column label="閲戦(鍏�)" prop="amount" min-width="160" :formatter="formattedNumber" />
+            <el-table-column fixed="right" label="鎿嶄綔" width="160" align="center">
+              <template #default="scope">
+                <el-button link type="primary" size="small" @click="handleEdit(scope.row)">缂栬緫</el-button>
+                <el-button link type="danger" size="small" @click="handleDelete(scope.row)">鍒犻櫎</el-button>
+              </template>
+            </el-table-column>
+          </el-table>
+
+          <pagination
+            v-show="total > 0"
+            :total="total"
+            layout="total, sizes, prev, pager, next, jumper"
+            :page="page.current"
+            :limit="page.size"
+            @pagination="paginationChange"
+          />
+        </el-col>
+
+        <el-col :span="10">
+          <div style="padding: 8px 0;">
+            <div style="display:flex; justify-content:space-between; align-items:center; margin-bottom: 10px;">
+              <div style="font-weight:600;">
+                {{ operationType === "add" ? "鏂板鍏朵粬閲戦" : "缂栬緫鍏朵粬閲戦" }}
+              </div>
+              <el-button
+                type="primary"
+                plain
+                size="small"
+                @click="handleAdd"
+                :disabled="operationType === 'add'"
+              >
+                鏂板
+              </el-button>
+            </div>
+
+            <el-form
+              :model="form"
+              label-width="120px"
+              label-position="top"
+              :rules="rules"
+              ref="formRef"
+            >
+              <el-form-item label="缂栫爜">
+                <el-input v-model="form.code" placeholder="璇疯緭鍏ョ紪鐮侊紙鍙�夛級" clearable />
+              </el-form-item>
+
+              <el-form-item label="椤圭洰" prop="processName">
+                <el-input v-model="form.processName" placeholder="璇疯緭鍏ラ」鐩悕绉�" clearable />
+              </el-form-item>
+
+              <el-form-item label="鏁伴噺" prop="quantity">
+                <el-input-number
+                  v-model="form.quantity"
+                  :min="0"
+                  :precision="2"
+                  style="width:100%"
+                  placeholder="璇疯緭鍏ユ暟閲�"
+                  clearable
+                  @change="recalcAmount"
+                />
+              </el-form-item>
+
+              <el-form-item label="鍗曚环(鍏�)" prop="unitPrice">
+                <el-input-number
+                  v-model="form.unitPrice"
+                  :min="0"
+                  :precision="2"
+                  style="width:100%"
+                  placeholder="璇疯緭鍏ュ崟浠�"
+                  clearable
+                  @change="recalcAmount"
+                />
+              </el-form-item>
+
+              <el-form-item label="閲戦(鍏�)">
+                <el-input v-model="form.amount" disabled />
+              </el-form-item>
+
+              <div style="display:flex; justify-content:flex-end; gap: 10px; margin-top: 8px;">
+                <el-button @click="closeDialog">鍙栨秷</el-button>
+                <el-button type="primary" @click="submitForm">淇濆瓨</el-button>
+              </div>
+            </el-form>
+          </div>
+        </el-col>
+      </el-row>
+    </el-dialog>
+  </div>
+</template>
+
+<script setup>
+import { getCurrentInstance, reactive, ref } from "vue";
+import { ElMessageBox } from "element-plus";
+import pagination from "@/components/PIMTable/Pagination.vue";
+import {
+  salesLedgerProductProcessList,
+  salesLedgerProductProcessAdd,
+  salesLedgerProductProcessUpdate,
+  salesLedgerProductProcessDelete,
+} from "@/api/salesManagement/salesLedger.js";
+
+const { proxy } = getCurrentInstance();
+const visible = ref(false);
+const loading = ref(false);
+const records = ref([]);
+const total = ref(0);
+const page = reactive({
+  current: 1,
+  size: 10,
+});
+
+const operationType = ref("add");
+const formRef = ref(null);
+const form = reactive({
+  id: null,
+  code: "",
+  processName: "",
+  quantity: 0,
+  unitPrice: 0,
+  amount: "0.00",
+});
+const rules = reactive({
+  processName: [{ required: true, message: "璇疯緭鍏ラ」鐩悕绉�", trigger: "change" }],
+  quantity: [{ required: true, message: "璇疯緭鍏ユ暟閲�", trigger: "blur" }],
+  unitPrice: [{ required: true, message: "璇疯緭鍏ュ崟浠�", trigger: "blur" }],
+});
+
+const formattedNumber = (row, column, cellValue) => {
+  return Number(cellValue || 0).toFixed(2);
+};
+
+const recalcAmount = () => {
+  const quantity = Number(form.quantity ?? 0) || 0;
+  const unitPrice = Number(form.unitPrice ?? 0) || 0;
+  form.amount = (quantity * unitPrice).toFixed(2);
+};
+
+const resetForm = (type = "add") => {
+  operationType.value = type;
+  form.id = null;
+  form.code = "";
+  form.processName = "";
+  form.quantity = 0;
+  form.unitPrice = 0;
+  form.amount = "0.00";
+};
+
+const fetchList = async () => {
+  loading.value = true;
+  try {
+    const res = await salesLedgerProductProcessList({
+      current: page.current,
+      size: page.size,
+    });
+    const list = res?.records ?? res?.data?.records ?? [];
+    const sum = res?.total ?? res?.data?.total ?? 0;
+    records.value = list.map((item) => {
+      const quantity = Number(item.quantity ?? 0) || 0;
+      const unitPrice = Number(item.unitPrice ?? 0) || 0;
+      const amount = Number(item.amount ?? quantity * unitPrice) || 0;
+      return {
+        id: item.id,
+        code: item.code ?? item.remark ?? "",
+        processName: item.processName ?? "",
+        quantity,
+        unitPrice,
+        amount: amount.toFixed(2),
+      };
+    });
+    total.value = sum;
+  } finally {
+    loading.value = false;
+  }
+};
+
+const openDialog = () => {
+  visible.value = true;
+  resetForm("add");
+  fetchList();
+};
+
+const closeDialog = () => {
+  visible.value = false;
+  resetForm("add");
+};
+
+const paginationChange = (obj) => {
+  page.current = obj.page;
+  page.size = obj.limit;
+  fetchList();
+};
+
+const handleAdd = () => {
+  resetForm("add");
+};
+
+const handleEdit = (row) => {
+  if (!row) return;
+  operationType.value = "edit";
+  form.id = row.id ?? null;
+  form.code = row.code ?? "";
+  form.processName = row.processName ?? "";
+  form.quantity = Number(row.quantity ?? 0) || 0;
+  form.unitPrice = Number(row.unitPrice ?? 0) || 0;
+  recalcAmount();
+};
+
+const submitForm = () => {
+  formRef.value?.validate((valid) => {
+    if (!valid) return;
+    const payload = {
+      processName: form.processName,
+      quantity: Number(form.quantity) || 0,
+      unitPrice: Number(form.unitPrice) || 0,
+      amount: Number(form.amount) || 0,
+      remark: form.code,
+      code: form.code,
+    };
+    const req =
+      operationType.value === "edit"
+        ? salesLedgerProductProcessUpdate({ ...payload, id: form.id })
+        : salesLedgerProductProcessAdd(payload);
+    req.then(() => {
+      proxy.$modal.msgSuccess("淇濆瓨鎴愬姛");
+      fetchList();
+      resetForm("add");
+    });
+  });
+};
+
+const handleDelete = (row) => {
+  if (!row?.id) return;
+  ElMessageBox.confirm("纭鍒犻櫎璇ヨ褰曪紵", "鍒犻櫎", {
+    confirmButtonText: "纭",
+    cancelButtonText: "鍙栨秷",
+    type: "warning",
+  })
+    .then(() => {
+      return salesLedgerProductProcessDelete(row.id).then(() => {
+        proxy.$modal.msgSuccess("鍒犻櫎鎴愬姛");
+        fetchList();
+        if (operationType.value === "edit" && form.id === row.id) {
+          resetForm("add");
+        }
+      });
+    })
+    .catch(() => {
+      proxy.$modal.msg("宸插彇娑�");
+    });
+};
+</script>
+
diff --git a/src/views/salesManagement/salesLedger/components/ProcessFlowConfigSelectDialog.vue b/src/views/salesManagement/salesLedger/components/ProcessFlowConfigSelectDialog.vue
new file mode 100644
index 0000000..1dc7e25
--- /dev/null
+++ b/src/views/salesManagement/salesLedger/components/ProcessFlowConfigSelectDialog.vue
@@ -0,0 +1,234 @@
+<template>
+  <el-dialog
+    v-model="visible"
+    title="閫夋嫨宸ヨ壓璺嚎閰嶇疆"
+    width="1000px"
+    :close-on-click-modal="false"
+    @close="handleClose"
+  >
+    <el-row :gutter="20">
+      <el-col :span="10">
+        <div style="font-weight: 600; margin-bottom: 8px;">閰嶇疆</div>
+        <el-select
+          v-model="selectedConfigId"
+          filterable
+          clearable
+          placeholder="璇烽�夋嫨宸ヨ壓璺嚎閰嶇疆"
+          style="width: 100%;"
+          @change="handleConfigChange"
+        >
+          <el-option
+            v-for="cfg in configList"
+            :key="cfg.configId"
+            :label="cfg.configName"
+            :value="cfg.configId"
+          />
+        </el-select>
+
+        <el-divider style="margin: 16px 0;" />
+
+        <div style="font-weight: 600; margin-bottom: 8px;">姝ラ棰勮</div>
+        <div style="font-size: 12px; color: #909399; margin-bottom: 6px;">
+          鏍规嵁鎵�閫夐厤缃睍绀烘祦绋嬪浘
+        </div>
+      </el-col>
+
+      <el-col :span="14">
+        <div class="process-diagram">
+          <div v-if="steps.length === 0" class="process-diagram-empty">鏆傛棤姝ラ</div>
+          <div
+            v-for="(step, idx) in steps"
+            :key="String(step.processId) + '_' + idx"
+            class="process-diagram-segment"
+          >
+            <div class="process-diagram-node">
+              <div class="process-diagram-index">{{ idx + 1 }}</div>
+              <div class="process-diagram-name">{{ step.processName }}</div>
+            </div>
+            <div v-if="idx < steps.length - 1" class="process-diagram-arrow">鈫�</div>
+          </div>
+        </div>
+        <div v-if="selectedConfigId === null" style="margin-top: 10px; font-size: 12px; color: #909399;">
+          璇峰厛閫夋嫨涓�鏉″凡缁存姢濂界殑宸ヨ壓璺嚎
+        </div>
+      </el-col>
+    </el-row>
+
+    <template #footer>
+      <div class="dialog-footer">
+        <el-button @click="handleClose">鍙栨秷</el-button>
+        <el-button type="primary" :loading="saving" @click="confirmSelect">
+          纭畾
+        </el-button>
+      </div>
+    </template>
+  </el-dialog>
+</template>
+
+<script setup>
+import { computed, getCurrentInstance, ref, watch } from "vue";
+import { salesProcessFlowConfigList, salesProcessFlowConfigGetById } from "@/api/salesManagement/salesProcessFlowConfig.js";
+
+const emit = defineEmits(["update:visible", "confirm"]);
+
+const props = defineProps({
+  visible: { type: Boolean, default: false },
+  defaultConfigId: { type: [Number, String, null], default: null },
+});
+
+const { proxy } = getCurrentInstance();
+
+const visible = computed({
+  get() {
+    return props.visible;
+  },
+  set(v) {
+    emit("update:visible", v);
+  },
+});
+
+const configList = ref([]);
+const selectedConfigId = ref(null);
+const steps = ref([]);
+const saving = ref(false);
+
+const normalizeStepsFromApi = (list) => {
+  if (!Array.isArray(list)) return [];
+  return list.map((s, idx) => ({
+    stepId: s.stepId ?? s.id ?? null,
+    processId: s.processId ?? s.process_id ?? s.id ?? null,
+    processName: s.processName ?? s.process_name ?? s.name ?? "",
+    sortNo: s.sortNo ?? idx + 1,
+  }));
+};
+
+const fetchConfigList = async () => {
+  const res = await salesProcessFlowConfigList();
+  const list = res?.data ?? res?.records ?? res ?? [];
+  configList.value = Array.isArray(list) ? list : [];
+};
+
+const fetchConfigDetail = async (id) => {
+  if (!id) {
+    steps.value = [];
+    return;
+  }
+  const res = await salesProcessFlowConfigGetById(id);
+  const detail = res?.data ?? res ?? {};
+  steps.value = normalizeStepsFromApi(detail?.steps ?? []);
+};
+
+const initDefault = async () => {
+  await fetchConfigList();
+  selectedConfigId.value = props.defaultConfigId ?? null;
+  await fetchConfigDetail(selectedConfigId.value);
+};
+
+watch(
+  () => props.visible,
+  async (v) => {
+    if (v) {
+      try {
+        await initDefault();
+      } catch {
+        proxy?.$modal?.msgError?.("鑾峰彇宸ヨ壓璺嚎閰嶇疆澶辫触");
+      }
+    }
+  }
+);
+
+const handleConfigChange = async () => {
+  await fetchConfigDetail(selectedConfigId.value);
+};
+
+const handleClose = () => {
+  emit("update:visible", false);
+  saving.value = false;
+};
+
+const confirmSelect = async () => {
+  if (saving.value) return;
+  if (selectedConfigId.value === null || selectedConfigId.value === undefined || selectedConfigId.value === "") {
+    proxy?.$modal?.msgWarning?.("璇烽�夋嫨宸ヨ壓璺嚎閰嶇疆");
+    return;
+  }
+  saving.value = true;
+  try {
+    emit("confirm", selectedConfigId.value);
+    handleClose();
+  } catch (e) {
+    proxy?.$modal?.msgError?.("纭澶辫触锛岃绋嶅悗閲嶈瘯");
+  } finally {
+    saving.value = false;
+  }
+};
+</script>
+
+<style scoped>
+.process-diagram {
+  display: flex;
+  align-items: center;
+  gap: 0;
+  flex-wrap: nowrap;
+  overflow-x: auto;
+  padding: 10px 0;
+}
+
+.process-diagram-segment {
+  display: flex;
+  align-items: center;
+}
+
+.process-diagram-node {
+  width: 160px;
+  min-width: 160px;
+  height: 78px;
+  border: 1px solid #ebeef5;
+  border-radius: 10px;
+  background: #fff;
+  display: flex;
+  flex-direction: column;
+  justify-content: center;
+  padding: 10px 12px;
+  margin-right: 10px;
+  box-sizing: border-box;
+}
+
+.process-diagram-index {
+  font-size: 12px;
+  color: #909399;
+  margin-bottom: 4px;
+}
+
+.process-diagram-name {
+  font-size: 14px;
+  font-weight: 600;
+  color: #303133;
+  white-space: nowrap;
+  overflow: hidden;
+  text-overflow: ellipsis;
+}
+
+.process-diagram-arrow {
+  font-size: 18px;
+  color: #909399;
+  margin-right: 14px;
+  margin-left: -6px;
+}
+
+.process-diagram-empty {
+  width: 100%;
+  text-align: center;
+  padding: 40px 0;
+  color: #909399;
+  border: 1px dashed #ebeef5;
+  border-radius: 8px;
+}
+
+.dialog-footer {
+  display: flex;
+  justify-content: flex-end;
+  gap: 10px;
+}
+</style>
+
diff --git a/src/views/salesManagement/salesLedger/components/ProcessFlowMaintenanceButton.vue b/src/views/salesManagement/salesLedger/components/ProcessFlowMaintenanceButton.vue
new file mode 100644
index 0000000..8406af3
--- /dev/null
+++ b/src/views/salesManagement/salesLedger/components/ProcessFlowMaintenanceButton.vue
@@ -0,0 +1,525 @@
+<template>
+  <div style="display: inline-block; margin-left: 8px;">
+    <el-button type="primary" plain @click="openDialog">宸ヨ壓娴佺▼</el-button>
+    <el-dialog
+      v-model="visible"
+      title="宸ヨ壓璺嚎涓庡伐搴忕淮鎶�"
+      width="1280px"
+      class="process-route-dialog"
+      :close-on-click-modal="false"
+      @close="closeDialog"
+    >
+      <el-row :gutter="16" class="dialog-main">
+        <el-col :span="10">
+          <div class="left-panel">
+            <div style="font-weight: 600; margin-bottom: 10px;">宸ヨ壓璺嚎</div>
+            <div class="route-toolbar route-toolbar-left">
+              <el-input
+                v-model="routeKeyword"
+                placeholder="鎸夊伐鑹鸿矾绾垮悕绉版煡璇�"
+                clearable
+                @keyup.enter="handleRouteQuery"
+              />
+              <el-button type="primary" @click="handleRouteQuery">鏌ヨ</el-button>
+              <el-input
+                v-model="routeNameDraft"
+                placeholder="杈撳叆鍚嶇О鍚庢柊澧�"
+                clearable
+              />
+              <el-button type="primary" plain @click="createRoute">鏂板</el-button>
+            </div>
+
+            <div class="left-table-wrap">
+              <el-table
+                :data="routeList"
+                border
+                row-key="routeId"
+                highlight-current-row
+                height="100%"
+                table-layout="fixed"
+                @current-change="handleRouteSelect"
+              >
+                <el-table-column label="搴忓彿" width="56" align="center">
+                  <template #default="scope">{{ scope.$index + 1 }}</template>
+                </el-table-column>
+                <el-table-column label="宸ヨ壓璺嚎鍚嶇О" min-width="150" prop="processRouteName" show-overflow-tooltip />
+                <el-table-column label="榛樿" width="56" align="center">
+                  <template #default="scope">
+                    <el-tag v-if="scope.row.isDefault" type="success" size="small">鏄�</el-tag>
+                    <span v-else>-</span>
+                  </template>
+                </el-table-column>
+                <el-table-column label="鎿嶄綔" width="160" align="center">
+                  <template #default="scope">
+                    <el-button link type="primary" size="small" @click="editRoute(scope.row)">鏀瑰悕</el-button>
+                    <el-button link type="primary" size="small" @click="setDefaultRoute(scope.row)">榛樿</el-button>
+                    <el-button link type="danger" size="small" @click="deleteRoute(scope.row)">鍒犻櫎</el-button>
+                  </template>
+                </el-table-column>
+              </el-table>
+            </div>
+
+            <pagination
+              v-show="routeTotal > 0"
+              :total="routeTotal"
+              layout="total, sizes, prev, pager, next, jumper"
+              :page="routePage.current"
+              :limit="routePage.size"
+              @pagination="handleRoutePaginationChange"
+            />
+          </div>
+        </el-col>
+
+        <el-col :span="14">
+          <div style="display:flex; justify-content:space-between; align-items:center; margin-bottom:10px;">
+            <div style="font-weight: 600;">宸ュ簭缁存姢</div>
+            <el-tag v-if="selectedRouteId" type="success" size="small">褰撳墠璺嚎锛歿{ selectedRouteName }}</el-tag>
+            <el-tag v-else type="info" size="small">璇峰厛閫夋嫨宸ヨ壓璺嚎</el-tag>
+          </div>
+
+          <div class="route-toolbar">
+            <el-input
+              v-model="processNameDraft"
+              placeholder="杈撳叆宸ュ簭鍚嶇О鍚庢柊澧�"
+              style="max-width: 260px;"
+              clearable
+              :disabled="!selectedRouteId"
+            />
+            <el-button type="primary" plain :disabled="!selectedRouteId" @click="createProcessItem">鏂板宸ュ簭</el-button>
+          </div>
+
+          <div class="process-diagram">
+            <div v-if="processItems.length === 0" class="process-diagram-empty">鏆傛棤宸ュ簭</div>
+            <div
+              v-for="(step, idx) in processItems"
+              :key="String(step.itemId) + '_' + idx"
+              class="process-diagram-segment"
+            >
+              <div class="process-diagram-node">
+                <div class="process-diagram-index">{{ idx + 1 }}</div>
+                <div class="process-diagram-name">{{ step.processName }}</div>
+              </div>
+              <div v-if="idx < processItems.length - 1" class="process-diagram-arrow">鈫�</div>
+            </div>
+          </div>
+
+          <el-table ref="processTableRef" :data="processItems" border row-key="itemId" height="420px" size="small">
+            <el-table-column label="搴忓彿" width="80" align="center">
+              <template #default="scope">{{ scope.$index + 1 }}</template>
+            </el-table-column>
+            <el-table-column label="宸ュ簭鍚嶇О" min-width="220" prop="processName" show-overflow-tooltip />
+            <el-table-column label="鎿嶄綔" width="300" align="center">
+              <template #default="scope">
+                <el-button link type="primary" size="small" @click="editProcessItem(scope.row)">缂栬緫</el-button>
+                <el-button link type="danger" size="small" @click="deleteProcessItem(scope.row)">鍒犻櫎</el-button>
+              </template>
+            </el-table-column>
+          </el-table>
+        </el-col>
+      </el-row>
+    </el-dialog>
+  </div>
+</template>
+
+<script setup>
+import { getCurrentInstance, ref, watch, onBeforeUnmount, nextTick } from "vue";
+import { ElMessageBox } from "element-plus";
+import Sortable from "sortablejs";
+import {
+  salesProcessFlowConfigList,
+  salesProcessFlowConfigUpsert,
+  salesProcessFlowConfigDelete,
+  salesProcessFlowConfigSetDefault,
+  salesProcessFlowConfigItemList,
+  salesProcessFlowConfigItemUpsert,
+  salesProcessFlowConfigItemSort,
+  salesProcessFlowConfigItemDelete,
+} from "@/api/salesManagement/salesProcessFlowConfig.js";
+
+const { proxy } = getCurrentInstance();
+const visible = ref(false);
+let prevBodyOverflow = "";
+let prevBodyOverflowY = "";
+
+const lockBodyScroll = () => {
+  // 鍏滃簳澶勭悊锛氭湁浜涘満鏅笅 Element Plus 涓嶄細瀹屽叏绂佹鑳屾櫙椤甸潰婊氬姩
+  prevBodyOverflow = document.body.style.overflow || "";
+  prevBodyOverflowY = document.body.style.overflowY || "";
+  document.body.style.overflow = "hidden";
+  document.body.style.overflowY = "hidden";
+};
+
+const unlockBodyScroll = () => {
+  document.body.style.overflow = prevBodyOverflow;
+  document.body.style.overflowY = prevBodyOverflowY;
+};
+
+watch(visible, (v) => {
+  if (v) lockBodyScroll();
+  else unlockBodyScroll();
+});
+
+onBeforeUnmount(() => {
+  unlockBodyScroll();
+});
+
+const routeKeyword = ref("");
+const routeNameDraft = ref("");
+const processNameDraft = ref("");
+const routeList = ref([]);
+const routeTotal = ref(0);
+const routePage = ref({
+  current: 1,
+  size: 10,
+});
+const selectedRouteId = ref(null);
+const selectedRouteName = ref("");
+const processItems = ref([]);
+const processTableRef = ref(null);
+let processStepsSortable = null;
+let isProcessingDrag = false;
+
+const destroyProcessSortable = () => {
+  if (processStepsSortable) {
+    processStepsSortable.destroy();
+    processStepsSortable = null;
+  }
+};
+
+const initProcessSortable = () => {
+  destroyProcessSortable();
+  if (!processTableRef.value) return;
+  const tbody = processTableRef.value?.$el?.querySelector(".el-table__body tbody")
+    || processTableRef.value?.$el?.querySelector(".el-table__body-wrapper > table > tbody");
+  if (!tbody) return;
+
+  processStepsSortable = new Sortable(tbody, {
+    animation: 150,
+    ghostClass: "sortable-ghost",
+    draggable: ".el-table__row",
+    handle: ".el-table__row",
+    filter: ".el-button, .el-input, .el-select",
+    preventOnFilter: true,
+    onEnd: async (evt) => {
+      if (isProcessingDrag) return;
+      const { oldIndex, newIndex } = evt;
+      if (oldIndex === newIndex) return;
+      if (!selectedRouteId.value) return;
+      if (!processItems.value[oldIndex]) return;
+
+      isProcessingDrag = true;
+      try {
+        const arr = [...processItems.value];
+        const moving = arr.splice(oldIndex, 1)[0];
+        arr.splice(newIndex, 0, moving);
+        processItems.value = arr;
+        ensureSortNo();
+
+        if (!moving?.itemId) {
+          proxy?.$modal?.msgError?.("褰撳墠宸ュ簭缂哄皯ID锛屾棤娉曟帓搴�");
+          await fetchProcessItems(selectedRouteId.value);
+          return;
+        }
+
+          // 浣跨敤涓撶敤鎺掑簭鎺ュ彛锛岄伩鍏� upsert 閫犳垚閲嶅鏂板
+          await salesProcessFlowConfigItemSort({
+            id: moving.itemId,
+            dragSort: newIndex + 1,
+          });
+        proxy?.$modal?.msgSuccess?.("椤哄簭璋冩暣鎴愬姛");
+        await fetchProcessItems(selectedRouteId.value);
+      } finally {
+        isProcessingDrag = false;
+      }
+    },
+  });
+};
+
+const normalizeRouteList = (list) => {
+  if (!Array.isArray(list)) return [];
+  return list.map((r) => ({
+    routeId: r.routeId ?? r.id ?? null,
+    processRouteName: r.processRouteName ?? r.routeName ?? r.name ?? "",
+    isDefault: Boolean(r.isDefault),
+  }));
+};
+
+const normalizeItemList = (list) => {
+  if (!Array.isArray(list)) return [];
+  return list.map((i, idx) => ({
+    itemId: i.itemId ?? i.id ?? null,
+    routeId: i.routeId ?? i.processRouteId ?? selectedRouteId.value,
+    processName: i.processName ?? i.name ?? "",
+    sortNo: i.sortNo ?? idx + 1,
+  }));
+};
+
+const fetchRouteList = async () => {
+  const res = await salesProcessFlowConfigList({
+    current: routePage.value.current,
+    size: routePage.value.size,
+    processRouteName: routeKeyword.value || undefined,
+  });
+  const records = res?.records ?? res?.data?.records ?? [];
+  const total = res?.total ?? res?.data?.total ?? 0;
+  routeTotal.value = Number(total) || 0;
+  routeList.value = normalizeRouteList(records);
+};
+
+const handleRouteQuery = async () => {
+  routePage.value.current = 1;
+  await fetchRouteList();
+};
+
+const handleRoutePaginationChange = async (obj) => {
+  routePage.value.current = obj.page;
+  routePage.value.size = obj.limit;
+  await fetchRouteList();
+};
+
+const fetchProcessItems = async (routeId) => {
+  if (!routeId) {
+    processItems.value = [];
+    destroyProcessSortable();
+    return;
+  }
+  const res = await salesProcessFlowConfigItemList(routeId);
+  const raw = res?.data ?? res ?? [];
+  processItems.value = normalizeItemList(raw);
+  ensureSortNo();
+  await nextTick();
+  initProcessSortable();
+};
+
+const openDialog = async () => {
+  visible.value = true;
+  selectedRouteId.value = null;
+  selectedRouteName.value = "";
+  processItems.value = [];
+  routePage.value.current = 1;
+  await fetchRouteList();
+};
+
+const closeDialog = () => {
+  visible.value = false;
+  selectedRouteId.value = null;
+  selectedRouteName.value = "";
+  processItems.value = [];
+  routeNameDraft.value = "";
+  processNameDraft.value = "";
+  routeTotal.value = 0;
+  destroyProcessSortable();
+};
+
+const handleRouteSelect = async (row) => {
+  if (!row?.routeId) return;
+  selectedRouteId.value = row.routeId;
+  selectedRouteName.value = row.processRouteName;
+  await fetchProcessItems(selectedRouteId.value);
+};
+
+const createRoute = async () => {
+  if (!routeNameDraft.value) {
+    proxy?.$modal?.msgWarning("璇峰厛杈撳叆宸ヨ壓璺嚎鍚嶇О");
+    return;
+  }
+  const payload = { processRouteName: routeNameDraft.value };
+  await salesProcessFlowConfigUpsert(payload);
+  proxy?.$modal?.msgSuccess("宸ヨ壓璺嚎鏂板鎴愬姛");
+  routeNameDraft.value = "";
+  await handleRouteQuery();
+};
+
+const editRoute = async (row) => {
+  const oldName = row?.processRouteName ?? "";
+  const { value } = await ElMessageBox.prompt("璇疯緭鍏ユ柊鐨勫伐鑹鸿矾绾垮悕绉�", "淇敼宸ヨ壓璺嚎", {
+    inputValue: oldName,
+    confirmButtonText: "纭",
+    cancelButtonText: "鍙栨秷",
+  });
+  await salesProcessFlowConfigUpsert({
+    routeId: row.routeId,
+    processRouteName: value,
+  });
+  proxy?.$modal?.msgSuccess("宸ヨ壓璺嚎淇敼鎴愬姛");
+  await fetchRouteList();
+  if (selectedRouteId.value === row.routeId) selectedRouteName.value = value;
+};
+
+const deleteRoute = async (row) => {
+  await ElMessageBox.confirm("纭鍒犻櫎璇ュ伐鑹鸿矾绾匡紵", "鍒犻櫎", {
+    confirmButtonText: "纭",
+    cancelButtonText: "鍙栨秷",
+    type: "warning",
+  });
+  await salesProcessFlowConfigDelete(row.routeId);
+  proxy?.$modal?.msgSuccess("鍒犻櫎鎴愬姛");
+  if (selectedRouteId.value === row.routeId) {
+    selectedRouteId.value = null;
+    selectedRouteName.value = "";
+    processItems.value = [];
+  }
+  // 鍒犻櫎鍚庤嫢褰撳墠椤佃娓呯┖锛屽洖閫�涓�椤�
+  if (routeList.value.length <= 1 && routePage.value.current > 1) {
+    routePage.value.current = routePage.value.current - 1;
+  }
+  await fetchRouteList();
+};
+
+const setDefaultRoute = async (row) => {
+  await salesProcessFlowConfigSetDefault(row.routeId);
+  proxy?.$modal?.msgSuccess("榛樿宸ヨ壓璺嚎璁剧疆鎴愬姛");
+  await fetchRouteList();
+};
+
+const ensureSortNo = () => {
+  processItems.value = processItems.value.map((i, idx) => ({ ...i, sortNo: idx + 1 }));
+};
+
+const createProcessItem = async () => {
+  if (!selectedRouteId.value) {
+    proxy?.$modal?.msgWarning("璇峰厛閫夋嫨宸ヨ壓璺嚎");
+    return;
+  }
+  if (!processNameDraft.value) {
+    proxy?.$modal?.msgWarning("璇峰厛杈撳叆宸ュ簭鍚嶇О");
+    return;
+  }
+  await salesProcessFlowConfigItemUpsert({
+    routeId: selectedRouteId.value,
+    processName: processNameDraft.value,
+    sortNo: processItems.value.length + 1,
+  });
+  proxy?.$modal?.msgSuccess("宸ュ簭鏂板鎴愬姛");
+  processNameDraft.value = "";
+  await fetchProcessItems(selectedRouteId.value);
+};
+
+const editProcessItem = async (row) => {
+  const { value } = await ElMessageBox.prompt("璇疯緭鍏ユ柊鐨勫伐搴忓悕绉�", "淇敼宸ュ簭", {
+    inputValue: row.processName,
+    confirmButtonText: "纭",
+    cancelButtonText: "鍙栨秷",
+  });
+  await salesProcessFlowConfigItemUpsert({
+    id: row.itemId,
+    routeId: selectedRouteId.value,
+    processName: value,
+    sortNo: row.sortNo,
+  });
+  proxy?.$modal?.msgSuccess("宸ュ簭淇敼鎴愬姛");
+  await fetchProcessItems(selectedRouteId.value);
+};
+
+const deleteProcessItem = async (row) => {
+  await ElMessageBox.confirm("纭鍒犻櫎璇ュ伐搴忥紵", "鍒犻櫎", {
+    confirmButtonText: "纭",
+    cancelButtonText: "鍙栨秷",
+    type: "warning",
+  });
+  await salesProcessFlowConfigItemDelete(row.itemId);
+  proxy?.$modal?.msgSuccess("宸ュ簭鍒犻櫎鎴愬姛");
+  await fetchProcessItems(selectedRouteId.value);
+};
+</script>
+
+<style scoped>
+.route-toolbar {
+  display: flex;
+  align-items: center;
+  gap: 10px;
+  margin-bottom: 10px;
+}
+
+.route-toolbar-left {
+  flex-wrap: nowrap;
+}
+
+.route-toolbar-left :deep(.el-input) {
+  width: 190px;
+}
+
+.process-route-dialog :deep(.el-dialog__body) {
+  height: 760px;
+  overflow: hidden;
+  overflow-y: hidden;
+}
+
+.dialog-main {
+  height: 100%;
+}
+
+.left-panel {
+  height: 100%;
+  display: flex;
+  flex-direction: column;
+  min-height: 0;
+}
+
+.left-table-wrap {
+  flex: 1;
+  min-height: 0;
+  overflow: hidden;
+}
+
+.process-diagram {
+  display: flex;
+  align-items: center;
+  gap: 0;
+  flex-wrap: nowrap;
+  overflow-x: auto;
+  padding: 10px 0;
+}
+
+.process-diagram-segment {
+  display: flex;
+  align-items: center;
+}
+
+.process-diagram-node {
+  width: 160px;
+  min-width: 160px;
+  height: 78px;
+  border: 1px solid #ebeef5;
+  border-radius: 10px;
+  background: #fff;
+  display: flex;
+  flex-direction: column;
+  justify-content: center;
+  padding: 10px 12px;
+  margin-right: 10px;
+  box-sizing: border-box;
+}
+
+.process-diagram-index {
+  font-size: 12px;
+  color: #909399;
+  margin-bottom: 4px;
+}
+
+.process-diagram-name {
+  font-size: 14px;
+  font-weight: 600;
+  color: #303133;
+  white-space: nowrap;
+  overflow: hidden;
+  text-overflow: ellipsis;
+}
+
+.process-diagram-arrow {
+  font-size: 18px;
+  color: #909399;
+  margin-right: 14px;
+  margin-left: -6px;
+}
+
+.process-diagram-empty {
+  width: 100%;
+  text-align: center;
+  padding: 24px 0;
+  color: #909399;
+  border: 1px dashed #ebeef5;
+  border-radius: 8px;
+}
+
+</style>
\ No newline at end of file
diff --git a/src/views/salesManagement/salesLedger/index.vue b/src/views/salesManagement/salesLedger/index.vue
index 8b6e024..b9a0e84 100644
--- a/src/views/salesManagement/salesLedger/index.vue
+++ b/src/views/salesManagement/salesLedger/index.vue
@@ -25,13 +25,20 @@
     </div>
     <div class="table_list">
       <div class="actions">
-        <div></div>
+        <div>
+          <OtherAmountMaintenanceButton />
+          <ProcessFlowMaintenanceButton />
+        </div>
+        <ProcessFlowConfigSelectDialog
+          v-model:visible="processFlowSelectDialogVisible"
+          @confirm="handleProcessFlowSelectConfirm"
+        />
         <div>
           <el-button type="primary" @click="openForm('add')">
             鏂板鍙拌处
           </el-button>
-          <el-button type="primary" plain @click="openOtherAmountDialog">
-            鍏朵粬閲戦缁存姢
+		  <el-button type="primary"  @click="handleBulkDelivery">
+            鍙戣揣
           </el-button>
           <el-button type="primary" plain @click="handleImport">瀵煎叆</el-button>
           <el-button @click="handleOut">瀵煎嚭</el-button>
@@ -49,7 +56,11 @@
               <el-table-column align="center" label="搴忓彿" type="index"/>
               <el-table-column label="浜у搧澶х被" prop="productCategory" />
               <el-table-column label="瑙勬牸鍨嬪彿" prop="specificationModel" />
-              <el-table-column label="鍗曚綅" prop="unit" />
+              <el-table-column label="鍘氬害" prop="thickness" min-width="90">
+                <template #default="scope">
+                  {{ scope.row.thickness ?? "" }}
+                </template>
+              </el-table-column>
 							<el-table-column label="浜у搧鐘舵��"
 															 width="100px"
 															 align="center">
@@ -96,7 +107,7 @@
               <el-table-column label="鍚◣鎬讳环(鍏�)" prop="taxInclusiveTotalPrice" :formatter="formattedNumber" />
               <el-table-column label="涓嶅惈绋庢�讳环(鍏�)" prop="taxExclusiveTotalPrice" :formatter="formattedNumber" />
             <!--鎿嶄綔-->
-              <el-table-column Width="60px" label="鎿嶄綔" align="center">
+              <!-- <el-table-column Width="60px" label="鎿嶄綔" align="center">
                 <template #default="scope">
                   <el-button 
                     link 
@@ -106,7 +117,7 @@
                     鍙戣揣
                   </el-button>
                 </template>
-              </el-table-column>
+              </el-table-column> -->
             </el-table>
           </template>
         </el-table-column>
@@ -123,9 +134,10 @@
         <el-table-column label="绛捐鏃ユ湡" prop="executionDate" width="120" show-overflow-tooltip />
         <el-table-column label="浜や粯鏃ユ湡" prop="deliveryDate" width="120" show-overflow-tooltip />
         <el-table-column label="澶囨敞" prop="remarks" width="200" show-overflow-tooltip />
-        <el-table-column fixed="right" label="鎿嶄綔" width="130" align="center">
+        <el-table-column fixed="right" label="鎿嶄綔" width="200" align="center">
           <template #default="scope">
             <el-button link type="primary" @click="openForm('edit', scope.row)" :disabled="!scope.row.isEdit">缂栬緫</el-button>
+            <el-button link type="primary" @click="openProcessFlowSelect(scope.row)" :disabled="!scope.row.isEdit">宸ヨ壓璺嚎</el-button>
             <el-button link type="primary" @click="downLoadFile(scope.row)">闄勪欢</el-button>
           </template>
         </el-table-column>
@@ -229,7 +241,11 @@
 					<el-table-column align="center" label="搴忓彿" type="index" width="60" />
 					<el-table-column label="浜у搧澶х被" prop="productCategory" />
 					<el-table-column label="瑙勬牸鍨嬪彿" prop="specificationModel" />
-					<el-table-column label="鍗曚綅" prop="unit" />
+					<el-table-column label="鍘氬害" prop="thickness" min-width="90">
+						<template #default="scope">
+							{{ scope.row.thickness ?? "" }}
+						</template>
+					</el-table-column>
 					<el-table-column label="鏁伴噺" prop="quantity" />
 					<el-table-column label="绋庣巼(%)" prop="taxRate" />
 					<el-table-column label="鍚◣鍗曚环(鍏�)" prop="taxInclusiveUnitPrice" :formatter="formattedNumber" />
@@ -247,6 +263,20 @@
 					<el-col :span="24">
 						<el-form-item label="澶囨敞锛�" prop="remarks">
 							<el-input v-model="form.remarks" placeholder="璇疯緭鍏�" clearable type="textarea" :rows="2" :disabled="operationType === 'view'" />
+						</el-form-item>
+					</el-col>
+				</el-row>
+				<el-row :gutter="30">
+					<el-col :span="24">
+						<el-form-item label="瀹㈡埛澶囨敞锛�" prop="customerRemarks">
+							<el-input
+								v-model="form.customerRemarks"
+								placeholder="璇疯緭鍏�"
+								clearable
+								type="textarea"
+								:rows="2"
+								:disabled="operationType === 'view'"
+							/>
 						</el-form-item>
 					</el-col>
 				</el-row>
@@ -374,8 +404,16 @@
 						</el-form-item>
 					</el-col>
 					<el-col :span="8">
-						<el-form-item label="鍗曚綅锛�" prop="unit">
-							<el-input v-model="productForm.unit" placeholder="璇疯緭鍏�" clearable />
+						<el-form-item label="鍘氬害锛�" prop="thickness">
+							<el-input-number
+								v-model="productForm.thickness"
+								:min="0"
+								:step="0.000000000000001"
+								:precision="15"
+								style="width: 100%;"
+								placeholder="璇疯緭鍏�"
+								clearable
+							/>
 						</el-form-item>
 					</el-col>
 				</el-row>
@@ -564,6 +602,13 @@
 						</el-form-item>
 					</el-col>
 				</el-row>
+				<el-row :gutter="30">
+					<el-col :span="24">
+						<el-form-item label="妤煎眰缂栧彿锛�" prop="floorCode">
+							<el-input v-model="productForm.floorCode" placeholder="璇疯緭鍏ユゼ灞傜紪鍙�" clearable type="textarea" :rows="2" :disabled="operationType === 'view'" />
+						</el-form-item>
+					</el-col>
+				</el-row>
 				<!-- 鍏朵粬閲戦锛堝崰婊′竴琛岋細绛夊悓浜� 3 鍒楃綉鏍肩殑鏁磋锛� -->
 				<el-row :gutter="30">
 					<el-col :span="12">
@@ -684,6 +729,7 @@
 				</el-button>
 			</template>
 		</el-dialog>
+
 		<!-- 瀵煎叆寮圭獥 -->
 		<FormDialog
 			v-model="importUpload.open"
@@ -778,7 +824,7 @@
 									<tr>
 										<th>浜у搧鍚嶇О</th>
 										<th>瑙勬牸鍨嬪彿</th>
-										<th>鍗曚綅</th>
+										<th>鍘氬害</th>
 										<th>鍗曚环</th>
 										<th>闆跺敭鏁伴噺</th>
 										<th>闆跺敭閲戦</th>
@@ -788,7 +834,7 @@
 									<tr v-for="product in item.products" :key="product.id">
 										<td>{{ product.productCategory || '' }}</td>
 										<td>{{ product.specificationModel || '' }}</td>
-										<td>{{ product.unit || '' }}</td>
+										<td>{{ product.thickness ?? '' }}</td>
 										<td>{{ product.taxInclusiveUnitPrice || '0' }}</td>
 										<td>{{ product.quantity || '0' }}</td>
 										<td>{{ product.taxInclusiveTotalPrice || '0' }}</td>
@@ -916,114 +962,6 @@
 			</template>
 		</el-dialog>
 
-		<!-- 鍏朵粬閲戦缁存姢锛堟柊澧�/缂栬緫/鍒犻櫎锛� -->
-		<el-dialog
-			v-model="otherAmountDialogVisible"
-			title="鍏朵粬閲戦缁存姢"
-			width="80%"
-			:close-on-click-modal="false"
-			@close="closeOtherAmountDialog"
-		>
-			<el-row :gutter="20">
-				<el-col :span="14">
-					<el-table
-						:data="otherAmountRecords"
-						border
-						v-loading="otherAmountLoading"
-						height="55vh"
-					>
-						<el-table-column label="缂栫爜" prop="code" min-width="120" show-overflow-tooltip />
-						<el-table-column label="椤圭洰" prop="processName" min-width="180" show-overflow-tooltip />
-						<el-table-column label="鏁伴噺" prop="quantity" min-width="110" :formatter="formattedNumber" />
-						<el-table-column label="鍗曚环(鍏�)" prop="unitPrice" min-width="130" :formatter="formattedNumber" />
-						<el-table-column label="閲戦(鍏�)" prop="amount" min-width="160" :formatter="formattedNumber" />
-						<el-table-column fixed="right" label="鎿嶄綔" width="160" align="center">
-							<template #default="scope">
-								<el-button link type="primary" size="small" @click="handleOtherEdit(scope.row)">缂栬緫</el-button>
-								<el-button link type="danger" size="small" @click="handleOtherDelete(scope.row)">鍒犻櫎</el-button>
-							</template>
-						</el-table-column>
-					</el-table>
-
-					<pagination
-						v-show="otherAmountTotal > 0"
-						:total="otherAmountTotal"
-						layout="total, sizes, prev, pager, next, jumper"
-						:page="otherAmountPage.current"
-						:limit="otherAmountPage.size"
-						@pagination="otherAmountPaginationChange"
-					/>
-				</el-col>
-
-				<el-col :span="10">
-					<div style="padding: 8px 0;">
-						<div style="display:flex; justify-content:space-between; align-items:center; margin-bottom: 10px;">
-							<div style="font-weight:600;">
-								{{ otherAmountOperationType === 'add' ? '鏂板鍏朵粬閲戦' : '缂栬緫鍏朵粬閲戦' }}
-							</div>
-							<el-button
-								type="primary"
-								plain
-								size="small"
-								@click="handleOtherAdd"
-								:disabled="otherAmountOperationType === 'add'"
-							>
-								鏂板
-							</el-button>
-						</div>
-
-						<el-form
-							:model="otherAmountForm"
-							label-width="120px"
-							label-position="top"
-							:rules="otherAmountRules"
-							ref="otherAmountFormRef"
-						>
-							<el-form-item label="缂栫爜">
-								<el-input v-model="otherAmountForm.code" placeholder="璇疯緭鍏ョ紪鐮侊紙鍙�夛級" clearable />
-							</el-form-item>
-
-							<el-form-item label="椤圭洰" prop="processName">
-								<el-input v-model="otherAmountForm.processName" placeholder="璇疯緭鍏ラ」鐩悕绉�" clearable />
-							</el-form-item>
-
-							<el-form-item label="鏁伴噺" prop="quantity">
-								<el-input-number
-									v-model="otherAmountForm.quantity"
-									:min="0"
-									:precision="2"
-									style="width:100%"
-									placeholder="璇疯緭鍏ユ暟閲�"
-									clearable
-									@change="recalcOtherAmount"
-								/>
-							</el-form-item>
-
-							<el-form-item label="鍗曚环(鍏�)" prop="unitPrice">
-								<el-input-number
-									v-model="otherAmountForm.unitPrice"
-									:min="0"
-									:precision="2"
-									style="width:100%"
-									placeholder="璇疯緭鍏ュ崟浠�"
-									clearable
-									@change="recalcOtherAmount"
-								/>
-							</el-form-item>
-
-							<el-form-item label="閲戦(鍏�)">
-								<el-input v-model="otherAmountForm.amount" disabled />
-							</el-form-item>
-
-							<div style="display:flex; justify-content:flex-end; gap: 10px; margin-top: 8px;">
-								<el-button @click="closeOtherAmountDialog">鍙栨秷</el-button>
-								<el-button type="primary" @click="submitOtherAmountForm">淇濆瓨</el-button>
-							</div>
-						</el-form>
-					</div>
-				</el-col>
-			</el-row>
-		</el-dialog>
 	</div>
 </template>
 
@@ -1038,6 +976,9 @@
 import { userListNoPage } from "@/api/system/user.js";
 import FileListDialog from '@/components/Dialog/FileListDialog.vue';
 import FormDialog from '@/components/Dialog/FormDialog.vue';
+import OtherAmountMaintenanceButton from "./components/OtherAmountMaintenanceButton.vue";
+import ProcessFlowMaintenanceButton from "./components/ProcessFlowMaintenanceButton.vue";
+import ProcessFlowConfigSelectDialog from "./components/ProcessFlowConfigSelectDialog.vue";
 import { getQuotationList } from "@/api/salesManagement/salesQuotation.js";
 import {
 	ledgerListPage,
@@ -1051,14 +992,12 @@
 	delLedgerFile,
 	getProductInventory,
 	salesLedgerProductProcessList,
-	salesLedgerProductProcessAdd,
-	salesLedgerProductProcessUpdate,
-	salesLedgerProductProcessDelete,
 } from "@/api/salesManagement/salesLedger.js";
 import { modelList, productTreeList } from "@/api/basicData/product.js";
 import useFormData from "@/hooks/useFormData.js";
 import dayjs from "dayjs";
 import { getCurrentDate } from "@/utils/index.js";
+import { salesLedgerProductSetProcessFlowConfig } from "@/api/salesManagement/salesProcessFlowConfig.js";
 
 const userStore = useUserStore();
 const { proxy } = getCurrentInstance();
@@ -1077,6 +1016,10 @@
 });
 const total = ref(0);
 const fileList = ref([]);
+
+// 宸ヨ壓璺嚎閰嶇疆閫夋嫨寮圭獥锛堢粦瀹氬埌鍙拌处浜у搧锛�
+const processFlowSelectDialogVisible = ref(false);
+const processFlowSelectLedgerRow = ref(null);
 
 // 鐢ㄦ埛淇℃伅琛ㄥ崟寮规鏁版嵁
 const operationType = ref("");
@@ -1119,7 +1062,7 @@
 	productForm: {
 		productCategory: "",
 		specificationModel: "",
-		unit: "",
+		thickness:null,
 		quantity: "",
 		taxInclusiveUnitPrice: "",
 		taxRate: "",
@@ -1138,6 +1081,8 @@
 		processRequirement: "", // 鍔犲伐瑕佹眰
 		remark: "", // 澶囨敞
 		salesProductProcessList: [], // 鍏朵粬閲戦锛歔{id, processName, quantity}]
+		processFlowConfigId: null, // 宸ヨ壓娴佺▼閰嶇疆缁戝畾
+		floorCode: "", // 妤煎眰缂栧彿
 	},
 	productRules: {
 		productCategory: [{ required: true, message: "璇烽�夋嫨", trigger: "change" }],
@@ -1145,7 +1090,7 @@
 		specificationModel: [
 			{ required: true, message: "璇烽�夋嫨", trigger: "change" },
 		],
-		unit: [{ required: true, message: "璇疯緭鍏�", trigger: "blur" }],
+		thickness: [{ required: true, message: "璇疯緭鍏�", trigger: "blur" }],
 		quantity: [{ required: true, message: "璇疯緭鍏�", trigger: "blur" }],
 		taxInclusiveUnitPrice: [
 			{ required: true, message: "璇疯緭鍏�", trigger: "blur" },
@@ -1191,7 +1136,7 @@
 
 // 鍙戣揣鐩稿叧
 const deliveryFormVisible = ref(false);
-const currentDeliveryRow = ref(null);
+const currentDeliveryRows = ref([]);
 const deliveryFormData = reactive({
   deliveryForm: {
     type: "璐ц溅", // 璐ц溅, 蹇��
@@ -1203,169 +1148,6 @@
   },
 });
 const { deliveryForm, deliveryRules } = toRefs(deliveryFormData);
-
-// 鍏朵粬閲戦缁存姢锛堝伐搴�/娴佺▼閲戦缁存姢锛�
-const otherAmountDialogVisible = ref(false);
-const otherAmountLoading = ref(false);
-const otherAmountRecords = ref([]);
-const otherAmountTotal = ref(0);
-const otherAmountPage = reactive({
-	current: 1,
-	size: 10,
-});
-
-const otherAmountOperationType = ref("add"); // add/edit
-const otherAmountFormRef = ref(null);
-const otherAmountForm = reactive({
-	id: null,
-	code: "", // 鍓嶇瀛楁鍚嶏細code锛涘悗绔帴鍙e垪琛ㄨ繑鍥� remark锛屾澶勮繘琛屾槧灏�
-	processName: "",
-	quantity: 0,
-	unitPrice: 0,
-	amount: "0.00",
-});
-const otherAmountRules = reactive({
-	processName: [{ required: true, message: "璇疯緭鍏ラ」鐩悕绉�", trigger: "change" }],
-	quantity: [{ required: true, message: "璇疯緭鍏ユ暟閲�", trigger: "blur" }],
-	unitPrice: [{ required: true, message: "璇疯緭鍏ュ崟浠�", trigger: "blur" }],
-});
-
-const recalcOtherAmount = () => {
-	const quantity = Number(otherAmountForm.quantity ?? 0) || 0;
-	const unitPrice = Number(otherAmountForm.unitPrice ?? 0) || 0;
-	otherAmountForm.amount = (quantity * unitPrice).toFixed(2);
-};
-
-const resetOtherAmountForm = (type = "add") => {
-	otherAmountOperationType.value = type;
-	otherAmountForm.id = null;
-	otherAmountForm.code = "";
-	otherAmountForm.processName = "";
-	otherAmountForm.quantity = 0;
-	otherAmountForm.unitPrice = 0;
-	otherAmountForm.amount = "0.00";
-};
-
-const openOtherAmountDialog = () => {
-	otherAmountDialogVisible.value = true;
-	resetOtherAmountForm("add");
-	// 鎵撳紑寮规鏃跺埛鏂版暟鎹紝閬垮厤闀挎椂闂村仠鐣欏鑷存暟鎹繃鏈�
-	otherAmountPage.current = otherAmountPage.current || 1;
-	fetchOtherAmountList();
-};
-
-const closeOtherAmountDialog = () => {
-	otherAmountDialogVisible.value = false;
-	resetOtherAmountForm("add");
-};
-
-const fetchOtherAmountList = async () => {
-	otherAmountLoading.value = true;
-	try {
-		const params = {
-			current: otherAmountPage.current,
-			size: otherAmountPage.size,
-		};
-		const res = await salesLedgerProductProcessList(params);
-
-		// 鍏煎涓嶅悓鎺ュ彛鍝嶅簲缁撴瀯锛氬彲鑳芥槸 res.records / res.total 鎴� res.data.records / res.data.total
-		const records = res?.records ?? res?.data?.records ?? [];
-		const total = res?.total ?? res?.data?.total ?? 0;
-
-		otherAmountRecords.value = records.map((item) => {
-			const quantity = Number(item.quantity ?? 0) || 0;
-			const unitPrice = Number(item.unitPrice ?? 0) || 0;
-			const amount = Number(item.amount ?? quantity * unitPrice) || 0;
-			return {
-				id: item.id,
-				code: item.code ?? item.remark ?? "",
-				processName: item.processName ?? "",
-				quantity,
-				unitPrice,
-				amount: amount.toFixed(2),
-			};
-		});
-
-		otherAmountTotal.value = total;
-	} finally {
-		otherAmountLoading.value = false;
-	}
-};
-
-const otherAmountPaginationChange = (obj) => {
-	otherAmountPage.current = obj.page;
-	otherAmountPage.size = obj.limit;
-	fetchOtherAmountList();
-};
-
-const handleOtherAdd = () => {
-	resetOtherAmountForm("add");
-};
-
-const handleOtherEdit = (row) => {
-	if (!row) return;
-	otherAmountOperationType.value = "edit";
-	otherAmountForm.id = row.id ?? null;
-	otherAmountForm.code = row.code ?? "";
-	otherAmountForm.processName = row.processName ?? "";
-	otherAmountForm.quantity = Number(row.quantity ?? 0) || 0;
-	otherAmountForm.unitPrice = Number(row.unitPrice ?? 0) || 0;
-	recalcOtherAmount();
-};
-
-const submitOtherAmountForm = () => {
-	otherAmountFormRef.value?.validate((valid) => {
-		if (!valid) return;
-
-		const payload = {
-			processName: otherAmountForm.processName,
-			quantity: Number(otherAmountForm.quantity) || 0,
-			unitPrice: Number(otherAmountForm.unitPrice) || 0,
-			amount: Number(otherAmountForm.amount) || 0,
-			// 鍒楄〃杩斿洖瀛楁鏄� remark锛岃繖閲屾寜鈥渃ode=remark鈥濆仛鏄犲皠
-			remark: otherAmountForm.code,
-			// 鍏煎鍚庣鍙兘鐩存帴浣跨敤 code 瀛楁
-			code: otherAmountForm.code,
-		};
-
-		if (otherAmountOperationType.value === "edit") {
-			payload.id = otherAmountForm.id;
-			salesLedgerProductProcessUpdate(payload).then(() => {
-				proxy.$modal.msgSuccess("淇濆瓨鎴愬姛");
-				fetchOtherAmountList();
-				resetOtherAmountForm("add");
-			});
-		} else {
-			salesLedgerProductProcessAdd(payload).then(() => {
-				proxy.$modal.msgSuccess("淇濆瓨鎴愬姛");
-				fetchOtherAmountList();
-				resetOtherAmountForm("add");
-			});
-		}
-	});
-};
-
-const handleOtherDelete = (row) => {
-	if (!row?.id) return;
-	ElMessageBox.confirm("纭鍒犻櫎璇ヨ褰曪紵", "鍒犻櫎", {
-		confirmButtonText: "纭",
-		cancelButtonText: "鍙栨秷",
-		type: "warning",
-	})
-		.then(() => {
-			return salesLedgerProductProcessDelete(row.id).then(() => {
-				proxy.$modal.msgSuccess("鍒犻櫎鎴愬姛");
-				fetchOtherAmountList();
-
-				if (otherAmountOperationType.value === "edit" && otherAmountForm.id === row.id) {
-					resetOtherAmountForm("add");
-				}
-			});
-		})
-		.catch(() => {
-			proxy.$modal.msg("宸插彇娑�");
-		});
-};
 
 // 浜у搧寮规锛氬叾浠栭噾棰濆閫変笅鎷夛紙鍩轰簬鈥滃叾浠栭噾棰濈淮鎶も�濇煡璇㈡帴鍙o級
 const otherAmountSelectOptions = ref([]); // [{id, processName}]
@@ -1636,6 +1418,59 @@
 			tableLoading.value = false;
 		});
 };
+
+// 鎵撳紑鈥滃伐鑹鸿矾绾块厤缃�濋�夋嫨寮圭獥锛堝繀椤绘樉寮忛�夋嫨锛�
+const openProcessFlowSelect = (ledgerRow) => {
+	if (!ledgerRow) return;
+	if (!ledgerRow.isEdit) return;
+	processFlowSelectLedgerRow.value = ledgerRow;
+	processFlowSelectDialogVisible.value = true;
+};
+
+// 灏嗛厤缃簲鐢ㄥ埌鍙拌处涓嬬殑鎵�鏈夋湭鍙戣揣浜у搧
+const handleProcessFlowSelectConfirm = async (configId) => {
+	const ledgerRow = processFlowSelectLedgerRow.value;
+	if (!ledgerRow?.id) return;
+
+	const finalConfigId = configId ?? null;
+	if (!finalConfigId) return;
+
+	proxy?.$modal?.loading?.("姝e湪搴旂敤宸ヨ壓璺嚎閰嶇疆锛岃绋嶅��...");
+	try {
+		const res = await productList({ salesLedgerId: ledgerRow.id, type: 1 });
+		const products = res?.data ?? res?.records ?? res ?? [];
+		if (!Array.isArray(products) || products.length === 0) {
+			proxy?.$modal?.msgWarning?.("璇ュ彴璐︿笅娌℃湁浜у搧鏁版嵁");
+			return;
+		}
+
+		for (const product of products) {
+			// 宸插彂璐�/瀹℃牳閫氳繃涓嶅彲缂栬緫锛岃烦杩�
+			if (isProductShipped(product)) continue;
+			await salesLedgerProductSetProcessFlowConfig({
+				salesLedgerProductId: product.id,
+				processFlowConfigId: finalConfigId,
+			});
+		}
+
+		proxy?.$modal?.msgSuccess?.("宸ヨ壓璺嚎閰嶇疆搴旂敤鎴愬姛");
+
+		// 鑻ュ綋鍓嶈宸插睍寮�锛屽埛鏂板叾瀛愪骇鍝佸垪琛�
+		if (expandedRowKeys.value.includes(ledgerRow.id)) {
+			const childRes = await productList({ salesLedgerId: ledgerRow.id, type: 1 });
+			const children = childRes?.data ?? childRes?.records ?? childRes ?? [];
+			const index = tableData.value.findIndex((item) => item.id === ledgerRow.id);
+			if (index > -1) {
+				tableData.value[index].children = children;
+			}
+		}
+	} catch (e) {
+		proxy?.$modal?.msgError?.("搴旂敤澶辫触锛岃绋嶅悗閲嶈瘯");
+	} finally {
+		proxy?.$modal?.closeLoading?.();
+	}
+};
+
 // 鑾峰彇浜у搧澶х被tree鏁版嵁
 const getProductOptions = () => {
 	// 杩斿洖 Promise锛屼究浜庡湪缂栬緫浜у搧鏃剁瓑寰呭姞杞藉畬鎴�
@@ -1658,10 +1493,8 @@
 	const index = modelOptions.value.findIndex((item) => item.id === value);
 	if (index !== -1) {
 		productForm.value.specificationModel = modelOptions.value[index].model;
-		productForm.value.unit = modelOptions.value[index].unit;
 	} else {
 		productForm.value.specificationModel = null;
-		productForm.value.unit = null;
 	}
 };
 const findNodeById = (nodes, productId) => {
@@ -1784,11 +1617,14 @@
 		form.value.entryDate = getCurrentDate();
 		// 绛捐鏃ユ湡榛樿涓哄綋澶�
 		form.value.executionDate = getCurrentDate();
+		form.value.customerRemarks = "";
 	} else {
 		currentId.value = row.id;
 		getSalesLedgerWithProducts({ id: row.id, type: 1 }).then((res) => {
 			form.value = { ...res };
 			form.value.entryPerson = Number(res.entryPerson);
+			// 瀛楁鍚嶅吋瀹癸細鍚庣鍙兘杩斿洖 customer_remarks
+			form.value.customerRemarks = res?.customerRemarks ?? res?.customer_remarks ?? "";
 			productData.value = form.value.productData;
 			fileList.value = form.value.salesLedgerFiles;
 		});
@@ -1886,7 +1722,7 @@
 			// 鍙拌处瀛楁
 			productCategory: p.product || p.productName || "",
 			specificationModel: p.specification || "",
-			unit: p.unit || "",
+			thickness: p.thickness,
 			quantity: quantity,
 			taxRate: taxRate,
 			taxInclusiveUnitPrice: unitPrice.toFixed(2),
@@ -1901,6 +1737,7 @@
 			settlePieceArea: 0,
 			settleTotalArea: 0,
 			processRequirement: "",
+			floorCode: "",
 			remark: "",
 			salesProductProcessList: [],
 		};
@@ -2007,10 +1844,17 @@
 		productForm.value.processRequirement =
 			row?.processRequirement ?? row?.process_requirement ?? "";
 		productForm.value.remark = row?.remark ?? row?.remarks ?? "";
+		productForm.value.floorCode = row?.floorCode ?? row?.floor_code ?? "";
+		// 宸ヨ壓娴佺▼閰嶇疆缁戝畾瀛楁锛堝悗缁敱鍚庣纭瀛楁鍚嶏級
+		productForm.value.processFlowConfigId =
+			row?.processFlowConfigId ?? row?.process_flow_config_id ?? null;
 
 		// 鍛ㄩ暱鍥炴樉锛堝鍚庣杩斿洖锛涙渶缁堜粛浠ュ叕寮忚绠椾负鍑嗭級
 		productForm.value.perimeter =
 			row?.perimeter ?? row?.heavyBoxPerimeter ?? row?.heavyboxPerimeter ?? 0;
+
+		// 鍚庣鐩存帴杩斿洖 thickness
+		productForm.value.thickness = row?.thickness;
 
 		productForm.value.salesProductProcessList = normalizeOtherAmountsFromRow(row);
 		productIndex.value = index;
@@ -2055,6 +1899,11 @@
 const submitProduct = () => {
 	proxy.$refs["productFormRef"].validate((valid) => {
 		if (valid) {
+			// 鍘氬害淇濈暀 15 浣嶅皬鏁帮紝閬垮厤鐢变簬娴偣璁$畻/杈撳叆瀵艰嚧绮惧害鍋忓樊
+			if (productForm.value.thickness !== null && productForm.value.thickness !== undefined) {
+				productForm.value.thickness = Number(Number(productForm.value.thickness).toFixed(15));
+			}
+
 			// 闈㈢Н/鎬昏瀛楁鍦ㄦ彁浜ゅ墠鍏滃簳璁$畻涓�娆�
 			recalcAreaTotals();
 			// 鍏朵粬閲戦鍙彁浜� {id, processName, quantity}锛堝悗绔瓧娈碉細salesProductProcessList锛�
@@ -2471,7 +2320,7 @@
                 <tr>
                   <th>浜у搧鍚嶇О</th>
                   <th>瑙勬牸鍨嬪彿</th>
-                  <th>鍗曚綅</th>
+                  <th>鍘氬害</th>
                   <th>鍗曚环</th>
                   <th>闆跺敭鏁伴噺</th>
                   <th>闆跺敭閲戦</th>
@@ -2483,7 +2332,7 @@
                     <tr>
                       <td>${product.productCategory || ''}</td>
                       <td>${product.specificationModel || ''}</td>
-                      <td>${product.unit || ''}</td>
+                      <td>${product.thickness ?? ''}</td>
                       <td>${product.taxInclusiveUnitPrice || '0'}</td>
                       <td>${product.quantity || '0'}</td>
                       <td>${product.taxInclusiveTotalPrice || '0'}</td>
@@ -2926,6 +2775,72 @@
 	return statusStr === '寰呭彂璐�' || statusStr === '瀹℃牳鎷掔粷';
 };
 
+const handleBulkDelivery = async () => {
+	if (selectedRows.value.length === 0) {
+		proxy.$modal.msgWarning("璇烽�夋嫨鏁版嵁");
+		return;
+	}
+
+	const customerNames = selectedRows.value.map((r) => String(r.customerName || "").trim());
+	const uniqueCustomers = Array.from(new Set(customerNames));
+
+	// 瀹㈡埛鍚嶇О涓嶄竴鑷翠笉鍏佽鍙戣揣
+	if (uniqueCustomers.length > 1) {
+		proxy.$modal.msgWarning("瀹㈡埛鍚嶇О涓嶄竴鑷达紝涓嶅厑璁稿彂璐�");
+		return;
+	}
+
+	// 澶氭潯涓斿鎴蜂竴鑷达細浜屾纭
+	if (selectedRows.value.length > 1) {
+		try {
+			await ElMessageBox.confirm("鏄惁纭鍚堝苟鍙戣揣锛�", "鍚堝苟鍙戣揣", {
+				confirmButtonText: "纭",
+				cancelButtonText: "鍙栨秷",
+				type: "warning",
+			});
+		} catch (e) {
+			proxy.$modal.msg("宸插彇娑�");
+			return;
+		}
+	}
+
+	proxy.$modal.loading("姝e湪鑾峰彇浜у搧鏁版嵁锛岃绋嶅��...");
+	try {
+		const targets = [];
+		for (const ledger of selectedRows.value) {
+			let products = [];
+			try {
+				const res = await productList({ salesLedgerId: ledger.id, type: 1 });
+				products = res?.data || [];
+			} catch {
+				products = [];
+			}
+
+			for (const product of products) {
+				if (!canShip(product)) continue;
+				targets.push({
+					...product,
+					salesLedgerId: product.salesLedgerId || ledger.id,
+				});
+			}
+		}
+
+		if (targets.length === 0) {
+			proxy.$modal.msgWarning("娌℃湁鍙彂璐х殑鏁版嵁");
+			return;
+		}
+
+		currentDeliveryRows.value = targets;
+		deliveryForm.value = { type: "璐ц溅" };
+		// 閲嶇疆瀹℃壒浜鸿妭鐐癸紙榛樿涓�涓┖鑺傜偣锛�
+		approverNodes.value = [{ id: 1, userId: null }];
+		nextApproverId = 2;
+		deliveryFormVisible.value = true;
+	} finally {
+		proxy.$modal.closeLoading();
+	}
+};
+
 /**
  * 涓嬭浇鏂囦欢
  *
@@ -2949,7 +2864,7 @@
 		return;
 	}
 	
-	currentDeliveryRow.value = row;
+	currentDeliveryRows.value = [row];
   deliveryForm.value = {
     type: "璐ц溅",
   };
@@ -2972,13 +2887,28 @@
       const approveUserIds = approverNodes.value.map(node => node.userId).join(",");
       // 淇濆瓨褰撳墠灞曞紑鐨勮ID锛屼互渚垮彂璐у悗閲嶆柊鍔犺浇瀛愯〃鏍兼暟鎹�
       const currentExpandedKeys = [...expandedRowKeys.value];
-      const salesLedgerId = currentDeliveryRow.value.salesLedgerId;
-      addShippingInfo({
-        salesLedgerId: salesLedgerId,
-        salesLedgerProductId: currentDeliveryRow.value.id,
-        type: deliveryForm.value.type,
-				approveUserIds,
-      })
+
+      const targets = currentDeliveryRows.value || [];
+      if (targets.length === 0) {
+        proxy.$modal.msgWarning("鏈�夋嫨鍙彂璐х殑鏁版嵁");
+        return;
+      }
+
+      // 渚濇鍙戣揣锛堥伩鍏嶅苟鍙戜笅搴撳瓨鎵e噺/鐘舵�佹洿鏂颁簰鐩稿奖鍝嶏級
+      const run = async () => {
+        for (const item of targets) {
+          const salesLedgerId = item.salesLedgerId;
+          if (!salesLedgerId) continue;
+          await addShippingInfo({
+            salesLedgerId,
+            salesLedgerProductId: item.id,
+            type: deliveryForm.value.type,
+            approveUserIds,
+          });
+        }
+      };
+
+      run()
         .then(() => {
           proxy.$modal.msgSuccess("鍙戣揣鎴愬姛");
           closeDeliveryDia();
@@ -2986,8 +2916,7 @@
           getList().then(() => {
             // 濡傛灉涔嬪墠鏈夊睍寮�鐨勮锛岄噸鏂板姞杞借繖浜涜鐨勫瓙琛ㄦ牸鏁版嵁
             if (currentExpandedKeys.length > 0) {
-              // 浣跨敤 Promise.all 骞惰鍔犺浇鎵�鏈夊睍寮�琛岀殑瀛愯〃鏍兼暟鎹�
-              const loadPromises = currentExpandedKeys.map(ledgerId => {
+              const loadPromises = currentExpandedKeys.map((ledgerId) => {
                 return productList({ salesLedgerId: ledgerId, type: 1 }).then((res) => {
                   const index = tableData.value.findIndex((item) => item.id === ledgerId);
                   if (index > -1) {
@@ -2996,12 +2925,14 @@
                 });
               });
               Promise.all(loadPromises).then(() => {
-                // 鎭㈠灞曞紑鐘舵��
                 expandedRowKeys.value = currentExpandedKeys;
               });
             }
           });
         })
+        .catch(() => {
+          proxy.$modal.msgError("鍙戣揣澶辫触锛岃绋嶅悗閲嶈瘯");
+        });
     }
   });
 };
@@ -3010,7 +2941,7 @@
 const closeDeliveryDia = () => {
   proxy.resetForm("deliveryFormRef");
   deliveryFormVisible.value = false;
-  currentDeliveryRow.value = null;
+  currentDeliveryRows.value = [];
 };
 const currentFactoryName = ref("");
 const getCurrentFactoryName = async () => {

--
Gitblit v1.9.3