From 2cba069ff6f6a3f2dd13c9e7708eaf54497a6d7e Mon Sep 17 00:00:00 2001
From: yyb <995253665@qq.com>
Date: 星期三, 15 四月 2026 11:38:55 +0800
Subject: [PATCH] 优化销售台账页面:为产品信息的内联编辑功能添加禁用状态,调整字段宽度,更新结算单片面积的标签,提升用户交互体验和界面一致性

---
 src/views/salesManagement/salesLedger/index.vue | 2670 +++++++++++++++++++++++++++++++++++-----------------------
 1 files changed, 1,594 insertions(+), 1,076 deletions(-)

diff --git a/src/views/salesManagement/salesLedger/index.vue b/src/views/salesManagement/salesLedger/index.vue
index 8b6e024..a2bf7d7 100644
--- a/src/views/salesManagement/salesLedger/index.vue
+++ b/src/views/salesManagement/salesLedger/index.vue
@@ -3,8 +3,23 @@
     <div class="search_form">
       <el-form :model="searchForm" :inline="true">
         <el-form-item label="瀹㈡埛鍚嶇О锛�">
-          <el-input v-model="searchForm.customerName" placeholder="璇疯緭鍏�" clearable prefix-icon="Search"
-            @change="handleQuery" />
+          <el-select
+            v-model="searchForm.customerId"
+            filterable
+            placeholder="璇烽�夋嫨瀹㈡埛鍚嶇О"
+            clearable
+            style="width: 220px"
+            @change="handleQuery"
+          >
+            <el-option
+              v-for="item in customerOption"
+              :key="item.id"
+              :label="item.customerName"
+              :value="item.id"
+            >
+              {{ item.customerName + "鈥斺��" + item.taxpayerIdentificationNumber }}
+            </el-option>
+          </el-select>
         </el-form-item>
         <el-form-item label="閿�鍞悎鍚屽彿锛�">
           <el-input v-model="searchForm.salesContractNo" placeholder="璇疯緭鍏�" clearable prefix-icon="Search"
@@ -18,6 +33,14 @@
           <el-date-picker v-model="searchForm.entryDate" value-format="YYYY-MM-DD" format="YYYY-MM-DD" type="daterange"
             placeholder="璇烽�夋嫨" clearable @change="changeDaterange" />
         </el-form-item>
+        <el-form-item label="鍙戣揣鐘舵�侊細">
+          <el-select v-model="searchForm.deliveryStatus" placeholder="璇烽�夋嫨" clearable style="width: 140px">
+            <el-option label="鏈彂璐�" :value="1" />
+            <el-option label="瀹℃壒涓�" :value="2" />
+            <el-option label="瀹℃壒澶辫触" :value="3" />
+            <el-option label="宸插彂璐�" :value="4" />
+          </el-select>
+        </el-form-item>
         <el-form-item>
           <el-button type="primary" @click="handleQuery"> 鎼滅储 </el-button>
         </el-form-item>
@@ -25,19 +48,40 @@
     </div>
     <div class="table_list">
       <div class="actions">
-        <div></div>
         <div>
-          <el-button type="primary" @click="openForm('add')">
-            鏂板鍙拌处
-          </el-button>
-          <el-button type="primary" plain @click="openOtherAmountDialog">
-            鍏朵粬閲戦缁存姢
-          </el-button>
-          <el-button type="primary" plain @click="handleImport">瀵煎叆</el-button>
-          <el-button @click="handleOut">瀵煎嚭</el-button>
-          <el-button type="danger" plain @click="handleDelete">鍒犻櫎</el-button>
-          <el-button type="primary" plain @click="handlePrint">鎵撳嵃</el-button>
+          <OtherAmountMaintenanceButton />
+          <ProcessFlowMaintenanceButton />
         </div>
+        <ProcessFlowConfigSelectDialog
+          v-model:visible="processFlowSelectDialogVisible"
+          :default-route-id="processFlowSelectDefaultRouteId"
+          :bound-route-name="processFlowSelectBoundRouteName"
+          @confirm="handleProcessFlowSelectConfirm"
+        />
+			<el-space wrap>
+					<el-button type="primary" @click="openForm('add')">鏂板鍙拌处</el-button>
+					<el-button type="primary" @click="handleBulkDelivery">鍙戣揣</el-button>
+					<el-button type="primary" plain @click="handleImport">瀵煎叆</el-button>
+					<el-button @click="handleOut">瀵煎嚭</el-button>
+
+					<el-button type="danger" plain @click="handleDelete">鍒犻櫎</el-button>
+
+					<el-dropdown @command="handlePrintCommand">
+						<el-button type="primary" plain>
+							鎵撳嵃鍗曟嵁<el-icon class="el-icon--right">
+								<ArrowDown />
+							</el-icon>
+						</el-button>
+						<template #dropdown>
+							<el-dropdown-menu>
+								<el-dropdown-item command="finishedProcessCard">鐢熶骇娴佺▼鍗★紙鎴愬搧锛�</el-dropdown-item>
+								<el-dropdown-item command="salesOrder">閿�鍞鍗�</el-dropdown-item>
+								<el-dropdown-item command="salesDeliveryNote">閿�鍞彂璐у崟</el-dropdown-item>
+							</el-dropdown-menu>
+						</template>
+					</el-dropdown>
+					<el-button type="primary" plain @click="handlePrintLabel">鎵撳嵃鏍囩</el-button>
+				</el-space>
       </div>
       <el-table :data="tableData" border v-loading="tableLoading" @selection-change="handleSelectionChange"
         :expand-row-keys="expandedRowKeys" :row-key="(row) => row.id" :row-class-name="tableRowClassName" show-summary style="width: 100%"
@@ -47,9 +91,42 @@
           <template #default="props">
             <el-table :data="props.row.children" border show-summary :summary-method="summarizeChildrenTable">
               <el-table-column align="center" label="搴忓彿" type="index"/>
+			<el-table-column label="妤煎眰缂栧彿" prop="floorCode" min-width="100" show-overflow-tooltip />
               <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="瀹�(mm)" prop="width" min-width="80">
+								<template #default="scope">
+									{{ scope.row.width ?? "" }}
+								</template>
+							</el-table-column>
+							<el-table-column label="楂�(mm)" prop="height" min-width="80">
+								<template #default="scope">
+									{{ scope.row.height ?? "" }}
+								</template>
+							</el-table-column>
+							<el-table-column label="鍛ㄩ暱(cm)" prop="perimeter" min-width="90">
+								<template #default="scope">
+									{{ scope.row.perimeter ?? "" }}
+								</template>
+							</el-table-column>
+							<el-table-column label="鎬婚潰绉�(cm虏)" prop="actualTotalArea" min-width="100">
+								<template #default="scope">
+									{{ scope.row.actualTotalArea ?? "" }}
+								</template>
+							</el-table-column>
+							<el-table-column label="鍔犲伐瑕佹眰" prop="processRequirement" min-width="120"
+								show-overflow-tooltip />
+							<el-table-column label="澶囨敞" prop="remark" min-width="120" show-overflow-tooltip />
+							<el-table-column label="閲嶇" prop="heavyBox" min-width="80">
+								<template #default="scope">
+									{{ scope.row.heavyBox ?? "" }}
+								</template>
+							</el-table-column>
 							<el-table-column label="浜у搧鐘舵��"
 															 width="100px"
 															 align="center">
@@ -62,13 +139,13 @@
 									<el-tag v-else type="danger">涓嶈冻</el-tag>
                 </template>
               </el-table-column>
-							<el-table-column label="鍙戣揣鐘舵��" width="140" align="center">
+							<!-- <el-table-column label="鍙戣揣鐘舵��" width="140" align="center">
 								<template #default="scope">
 									<el-tag :type="getShippingStatusType(scope.row)" size="small">
 										{{ getShippingStatusText(scope.row) }}
 									</el-tag>
 								</template>
-							</el-table-column>
+							</el-table-column> -->
 							<el-table-column label="蹇�掑叕鍙�" prop="expressCompany" show-overflow-tooltip />
 							<el-table-column label="蹇�掑崟鍙�" prop="expressNumber" show-overflow-tooltip />
               <el-table-column label="鍙戣揣杞︾墝" minWidth="100px" align="center">
@@ -96,7 +173,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 +183,7 @@
                     鍙戣揣
                   </el-button>
                 </template>
-              </el-table-column>
+              </el-table-column> -->
             </el-table>
           </template>
         </el-table-column>
@@ -115,17 +192,27 @@
         <el-table-column label="瀹㈡埛鍚嶇О" prop="customerName" width="300" show-overflow-tooltip />
         <el-table-column label="涓氬姟鍛�" prop="salesman" width="100" show-overflow-tooltip />
         <el-table-column label="椤圭洰鍚嶇О" prop="projectName" width="180" show-overflow-tooltip />
-        <el-table-column label="浠樻鏂瑰紡" prop="paymentMethod" show-overflow-tooltip />
         <el-table-column label="鍚堝悓閲戦(鍏�)" prop="contractAmount" width="220" show-overflow-tooltip
           :formatter="formattedNumber" />
+		  <el-table-column label="鍙戣揣鐘舵��" width="140" align="center">
+				<template #default="scope">
+						<el-tag v-if="Number(scope.row.deliveryStatus) === 1" type="info">鏈彂璐�</el-tag>
+						<el-tag v-else-if="Number(scope.row.deliveryStatus) === 2" type="warning">瀹℃壒涓�</el-tag>
+						<el-tag v-else-if="Number(scope.row.deliveryStatus) === 3" type="danger">瀹℃壒涓嶉�氳繃</el-tag>
+						<el-tag v-else-if="Number(scope.row.deliveryStatus) === 4" type="primary">瀹℃壒閫氳繃</el-tag>
+						<el-tag v-else-if="Number(scope.row.deliveryStatus) === 5" type="success">宸插彂璐�</el-tag>
+						<el-tag v-else type="info">-</el-tag>
+					</template>
+		  </el-table-column>
         <el-table-column label="褰曞叆浜�" prop="entryPersonName" width="100" show-overflow-tooltip />
         <el-table-column label="褰曞叆鏃ユ湡" prop="entryDate" width="120" show-overflow-tooltip />
         <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>
@@ -162,7 +249,7 @@
         <el-row :gutter="30">
           <el-col :span="12">
             <el-form-item label="瀹㈡埛鍚嶇О锛�" prop="customerId">
-              <el-select v-model="form.customerId" placeholder="璇烽�夋嫨" clearable :disabled="operationType === 'view'">
+              <el-select v-model="form.customerId" filterable placeholder="璇烽�夋嫨" clearable :disabled="operationType === 'view'">
                 <el-option v-for="item in customerOption" :key="item.id" :label="item.customerName" :value="item.id">
                   {{
                     item.customerName + "鈥斺��" + item.taxpayerIdentificationNumber
@@ -184,11 +271,12 @@
 															format="YYYY-MM-DD" type="date" placeholder="璇烽�夋嫨" clearable :disabled="operationType === 'view'" />
 						</el-form-item>
 					</el-col>
-					<el-col :span="12">
-						<el-form-item label="浠樻鏂瑰紡">
-							<el-input v-model="form.paymentMethod" placeholder="璇疯緭鍏�" clearable :disabled="operationType === 'view'" />
-						</el-form-item>
-					</el-col>
+          <el-col :span="12">
+            <el-form-item label="浜よ揣鏃ユ湡锛�" prop="deliveryDate">
+              <el-date-picker style="width: 100%" v-model="form.deliveryDate" value-format="YYYY-MM-DD" format="YYYY-MM-DD"
+                              type="date" placeholder="璇烽�夋嫨" clearable />
+            </el-form-item>
+          </el-col>
 				</el-row>
 				<el-row :gutter="30">
 					<el-col :span="12">
@@ -208,17 +296,16 @@
 						</el-form-item>
 					</el-col>
 				</el-row>
-        <el-row :gutter="30">
-          <el-col :span="12">
-            <el-form-item label="浜よ揣鏃ユ湡锛�" prop="entryDate">
-              <el-date-picker style="width: 100%" v-model="form.deliveryDate" value-format="YYYY-MM-DD" format="YYYY-MM-DD"
-                              type="date" placeholder="璇烽�夋嫨" clearable />
-            </el-form-item>
-          </el-col>
-        </el-row>
 				<el-row>
 					<el-form-item label="浜у搧淇℃伅锛�" prop="entryDate">
-						<el-button v-if="operationType !== 'view'" type="primary" @click="openProductForm('add')">娣诲姞</el-button>
+						<el-button
+							v-if="operationType !== 'view'"
+							type="primary"
+							:disabled="hasEditingProductRow()"
+							@click="addProductInline"
+						>
+							娣诲姞
+						</el-button>
 						<el-button v-if="operationType !== 'view'" plain type="danger" @click="deleteProduct" >鍒犻櫎</el-button>
 					</el-form-item>
 				</el-row>
@@ -227,19 +314,456 @@
 					<el-table-column align="center" type="selection" width="55" v-if="operationType !== 'view'"
 						:selectable="(row) => !isProductShipped(row)" />
 					<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="quantity" />
-					<el-table-column label="绋庣巼(%)" prop="taxRate" />
-					<el-table-column label="鍚◣鍗曚环(鍏�)" prop="taxInclusiveUnitPrice" :formatter="formattedNumber" />
-					<el-table-column label="鍚◣鎬讳环(鍏�)" prop="taxInclusiveTotalPrice" :formatter="formattedNumber" />
-					<el-table-column label="涓嶅惈绋庢�讳环(鍏�)" prop="taxExclusiveTotalPrice" :formatter="formattedNumber" />
-					<el-table-column fixed="right" label="鎿嶄綔" min-width="60" align="center" v-if="operationType !== 'view'">
+					<el-table-column label="浜у搧澶х被" prop="productCategory" min-width="160">
 						<template #default="scope">
-							<el-button link type="primary" size="small" 
-								:disabled="isProductShipped(scope.row)"
-								@click="openProductForm('edit', scope.row,scope.$index)">缂栬緫</el-button>
+							<el-tree-select
+								v-if="scope.row.__editing"
+								v-model="scope.row.__productCategoryId"
+								placeholder="璇烽�夋嫨"
+								clearable
+								filterable
+								check-strictly
+								:data="productOptions"
+								:render-after-expand="false"
+								style="width: 100%"
+								:filter-node-method="filterProductCategoryNode"
+								@change="(val) => handleInlineProductCategoryChange(scope.row, val)"
+							/>
+							<span v-else>{{ scope.row.productCategory ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="瑙勬牸鍨嬪彿" prop="specificationModel" min-width="160">
+						<template #default="scope">
+							<el-select
+								v-if="scope.row.__editing"
+								v-model="scope.row.productModelId"
+								placeholder="璇烽�夋嫨"
+								clearable
+								filterable
+								style="width: 100%"
+								@change="(val) => handleInlineProductModelChange(scope.row, val)"
+							>
+								<el-option v-for="item in modelOptions" :key="item.id" :label="item.model" :value="item.id" />
+							</el-select>
+							<span v-else>{{ scope.row.specificationModel ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="鍘氬害(mm)" prop="thickness" min-width="160">
+						<template #default="scope">
+							<el-input-number
+								v-if="scope.row.__editing"
+								v-model="scope.row.thickness"
+								:min="0"
+								:step="0.000000000000001"
+								:precision="15"
+								style="width: 100%"
+								placeholder="璇疯緭鍏�"
+								clearable
+							/>
+							<span v-else>{{ scope.row.thickness ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="瀹�(mm)" prop="width" min-width="160">
+						<template #default="scope">
+							<el-input-number
+								v-if="scope.row.__editing"
+								v-model="scope.row.width"
+								:min="0"
+								:step="1"
+								:precision="2"
+								style="width:100%"
+								placeholder="璇疯緭鍏�"
+								clearable
+								@change="() => handleInlineSizeChange(scope.row)"
+								@input="() => handleInlineSizeChange(scope.row)"
+							/>
+							<span v-else>{{ scope.row.width ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="楂�(mm)" prop="height" min-width="160">
+						<template #default="scope">
+							<el-input-number
+								v-if="scope.row.__editing"
+								v-model="scope.row.height"
+								:min="0"
+								:step="1"
+								:precision="2"
+								style="width: 100%"
+								placeholder="璇疯緭鍏�"
+								clearable
+								@change="() => handleInlineSizeChange(scope.row)"
+								@input="() => handleInlineSizeChange(scope.row)"
+							/>
+							<span v-else>{{ scope.row.height ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="缁撶畻鍗曠墖闈㈢Н(銕�)" prop="settlePieceArea" min-width="160">
+						<template #default="scope">
+							<el-input-number
+								v-if="scope.row.__editing"
+								v-model="scope.row.settlePieceArea"
+								:min="0"
+								:step="0.00001"
+								:precision="5"
+								style="width: 100%"
+								placeholder="璇疯緭鍏�"
+								clearable
+								@change="() => handleInlineSettleAreaChange(scope.row)"
+							/>
+							<span v-else>{{ scope.row.settlePieceArea ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="鏁伴噺" prop="quantity" min-width="150">
+						<template #default="scope">
+							<el-input-number
+								v-if="scope.row.__editing"
+								v-model="scope.row.quantity"
+								:step="0.1"
+								:min="0"
+								:precision="2"
+								style="width: 100%"
+								placeholder="璇疯緭鍏�"
+								clearable
+								@change="() => handleInlineQuantityChange(scope.row)"
+								@input="() => handleInlineQuantityChange(scope.row)"
+							/>
+							<span v-else>{{ scope.row.quantity ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="闈㈢Н(m虏)" prop="actualTotalArea" min-width="160">
+						<template #default="scope">
+							<el-input-number
+								v-if="scope.row.__editing"
+								v-model="scope.row.actualTotalArea"
+								:min="0"
+								:step="0.00001"
+								:precision="5"
+								style="width: 100%"
+								placeholder="鑷姩璁$畻"
+							/>
+							<span v-else>{{ scope.row.actualTotalArea ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="鍚◣鍗曚环(鍏�)" prop="taxInclusiveUnitPrice" min-width="140">
+						<template #default="scope">
+							<el-input-number
+								v-if="scope.row.__editing"
+								:step="0.01"
+								:min="0"
+								:precision="2"
+								style="width: 100%"
+								v-model="scope.row.taxInclusiveUnitPrice"
+								placeholder="璇疯緭鍏�"
+								clearable
+								@change="() => handleInlineUnitPriceChange(scope.row)"
+								@input="() => handleInlineUnitPriceChange(scope.row)"
+							/>
+							<span v-else>{{ formattedNumber(null, null, scope.row.taxInclusiveUnitPrice ?? 0) }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="绋庣巼(%)" prop="taxRate" min-width="120">
+						<template #default="scope">
+							<el-select
+								v-if="scope.row.__editing"
+								v-model="scope.row.taxRate"
+								placeholder="璇烽�夋嫨"
+								clearable
+								style="width: 100%"
+								@change="() => handleInlineTaxRateChange(scope.row)"
+							>
+								<el-option label="1" value="1" />
+								<el-option label="3" value="3" />
+								<el-option label="6" value="6" />
+								<el-option label="9" value="9" />
+								<el-option label="13" value="13" />
+							</el-select>
+							<span v-else>{{ scope.row.taxRate ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="鍚◣鎬讳环(鍏�)" prop="taxInclusiveTotalPrice" :formatter="formattedNumber"  min-width="120"/>
+					<el-table-column label="涓嶅惈绋庢�讳环(鍏�)" prop="taxExclusiveTotalPrice" :formatter="formattedNumber"  min-width="120"/>
+					<el-table-column label="鍙戠エ绫诲瀷" prop="invoiceType" min-width="120">
+						<template #default="scope">
+							<el-select
+								v-if="scope.row.__editing"
+								v-model="scope.row.invoiceType"
+								placeholder="璇烽�夋嫨"
+								clearable
+								style="width: 100%"
+							>
+								<el-option label="澧炴櫘绁�" value="澧炴櫘绁�" />
+								<el-option label="澧炰笓绁�" value="澧炰笓绁�" />
+							</el-select>
+							<span v-else>{{ scope.row.invoiceType ?? "" }}</span>
+						</template>
+					</el-table-column>
+
+					<el-table-column label="鍔犲伐瑕佹眰" prop="processRequirement" min-width="160" show-overflow-tooltip>
+						<template #default="scope">
+							<el-input
+								v-if="scope.row.__editing"
+								v-model="scope.row.processRequirement"
+								placeholder="璇疯緭鍏�"
+								clearable
+								style="width: 100%"
+							/>
+							<span v-else>{{ scope.row.processRequirement ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="澶囨敞" prop="remark" min-width="140" show-overflow-tooltip>
+						<template #default="scope">
+							<el-input
+								v-if="scope.row.__editing"
+								v-model="scope.row.remark"
+								placeholder="璇疯緭鍏�"
+								clearable
+								style="width: 100%"
+							/>
+							<span v-else>{{ scope.row.remark ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="妤煎眰缂栧彿" prop="floorCode" min-width="140" show-overflow-tooltip>
+						<template #default="scope">
+							<el-input
+								v-if="scope.row.__editing"
+								v-model="scope.row.floorCode"
+								placeholder="璇疯緭鍏�"
+								clearable
+								style="width: 100%"
+							/>
+							<span v-else>{{ scope.row.floorCode ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column label="閲嶇" prop="heavyBox" min-width="100">
+						<template #default="scope">
+							<el-input
+								v-if="scope.row.__editing"
+								v-model="scope.row.heavyBox"
+								placeholder="璇疯緭鍏�"
+								clearable
+								style="width: 100%"
+							/>
+							<span v-else>{{ scope.row.heavyBox ?? "" }}</span>
+						</template>
+					</el-table-column>
+					<el-table-column fixed="right" label="鎿嶄綔" min-width="220" align="center" v-if="operationType !== 'view'">
+						<template #default="scope">
+							<template v-if="scope.row.__editing">
+								<el-button link type="primary" size="small" @click="saveProductInline(scope.row, scope.$index)">淇濆瓨</el-button>
+								<el-button link type="danger" size="small" @click="cancelProductInline(scope.row, scope.$index)">鍙栨秷</el-button>
+								<el-popover
+									:width="560"
+									trigger="click"
+									:hide-after="0"
+									:visible="scope.row.__otherAmountPopoverVisible"
+									@update:visible="(val) => handleOtherAmountPopoverVisibleChange(scope.row, val)"
+								>
+									<template #reference>
+										<el-button
+											link
+											type="primary"
+											size="small"
+											@click="openOtherAmountInline(scope.row)"
+										>
+											鍏朵粬閲戦({{ (scope.row.salesProductProcessList || []).length || 0 }})
+										</el-button>
+									</template>
+
+									<div style="display:flex; align-items:center; justify-content:space-between; gap: 10px; margin-bottom: 8px;">
+										<div style="font-weight: 600; color:#303133;">
+											鍏朵粬閲戦
+										</div>
+										<el-button type="primary" plain size="small" @click="startAddOtherAmountForRow(scope.row)">
+											鏂板
+										</el-button>
+									</div>
+									<div
+										v-if="scope.row.__inlineOtherAmountAdding"
+										style="display:flex; flex-direction:column; gap: 8px; margin-bottom: 10px;"
+										@click.stop
+									>
+										<el-select
+											v-model="scope.row.__inlineOtherAmountAddId"
+											filterable
+											clearable
+											placeholder="璇烽�夋嫨鍏朵粬閲戦椤圭洰"
+											style="width: 100%;"
+										>
+											<el-option
+												v-for="item in otherAmountSelectOptions"
+												:key="item.id"
+												:label="item.processName"
+												:value="item.id"
+											/>
+										</el-select>
+										<div style="display:flex; justify-content:flex-end; gap: 8px;">
+											<el-button
+												size="small"
+												@click="scope.row.__inlineOtherAmountAdding = false; scope.row.__inlineOtherAmountAddId = null"
+											>
+												鍙栨秷
+											</el-button>
+											<el-button
+												type="primary"
+												size="small"
+												:disabled="scope.row.__inlineOtherAmountAddId === null || scope.row.__inlineOtherAmountAddId === undefined || scope.row.__inlineOtherAmountAddId === ''"
+												@click="confirmAddOtherAmountForRow(scope.row)"
+											>
+												纭娣诲姞
+											</el-button>
+										</div>
+									</div>
+
+									<div v-if="Array.isArray(scope.row.salesProductProcessList) && scope.row.salesProductProcessList.length > 0"
+										style="display:flex; flex-wrap:wrap; gap: 8px;"
+									>
+										<div
+											v-for="(item, idx) in scope.row.salesProductProcessList"
+											:key="String(item.id) + '_' + idx"
+											style="display:flex; align-items:center; gap: 8px; flex: 0 0 calc(50% - 4px); max-width: calc(50% - 4px);"
+										>
+											<el-tag type="info" style="max-width: 170px; overflow:hidden; text-overflow:ellipsis; white-space:nowrap;">
+												{{ item.processName }}
+											</el-tag>
+											<el-input-number
+												v-model="item.quantity"
+												:min="0"
+												:step="1"
+												:precision="0"
+												style="width: 120px;"
+												placeholder="鏁伴噺"
+												:disabled="operationType === 'view'"
+												@change="handleOtherAmountQuantityChange(scope.row)"
+											/>
+											<el-button type="danger" link size="small" @click="removeOtherAmountAtForRow(scope.row, idx)">
+												鍒犻櫎
+											</el-button>
+										</div>
+									</div>
+									<div v-else style="color:#909399; font-size: 13px;">
+										鏆傛棤鍏朵粬閲戦
+									</div>
+								</el-popover>
+							</template>
+							<template v-else>
+								<el-button
+									link
+									type="primary"
+									size="small"
+									:disabled="isProductShipped(scope.row)"
+									@click="editProductInline(scope.row, scope.$index)"
+								>
+									缂栬緫
+								</el-button>
+								<el-popover
+									:width="560"
+									trigger="click"
+									:hide-after="0"
+									:visible="scope.row.__otherAmountPopoverVisible"
+									@update:visible="(val) => handleOtherAmountPopoverVisibleChange(scope.row, val)"
+								>
+									<template #reference>
+										<el-button
+											link
+											type="primary"
+											size="small"
+											:disabled="isProductShipped(scope.row)"
+											@click="openOtherAmountInline(scope.row)"
+										>
+											鍏朵粬閲戦({{ (scope.row.salesProductProcessList || []).length || 0 }})
+										</el-button>
+									</template>
+
+									<div style="display:flex; align-items:center; justify-content:space-between; gap: 10px; margin-bottom: 8px;">
+										<div style="font-weight: 600; color:#303133;">
+											鍏朵粬閲戦
+										</div>
+										<el-button
+											type="primary"
+											plain
+											size="small"
+											:disabled="isProductShipped(scope.row)"
+											@click="startAddOtherAmountForRow(scope.row)"
+										>
+											鏂板
+										</el-button>
+									</div>
+									<div
+										v-if="scope.row.__inlineOtherAmountAdding"
+										style="display:flex; flex-direction:column; gap: 8px; margin-bottom: 10px;"
+										@click.stop
+									>
+										<el-select
+											v-model="scope.row.__inlineOtherAmountAddId"
+											filterable
+											clearable
+											placeholder="璇烽�夋嫨鍏朵粬閲戦椤圭洰"
+											style="width: 100%;"
+											:disabled="isProductShipped(scope.row)"
+										>
+											<el-option
+												v-for="item in otherAmountSelectOptions"
+												:key="item.id"
+												:label="item.processName"
+												:value="item.id"
+											/>
+										</el-select>
+										<div style="display:flex; justify-content:flex-end; gap: 8px;">
+											<el-button
+												size="small"
+												:disabled="isProductShipped(scope.row)"
+												@click="scope.row.__inlineOtherAmountAdding = false; scope.row.__inlineOtherAmountAddId = null"
+											>
+												鍙栨秷
+											</el-button>
+											<el-button
+												type="primary"
+												size="small"
+												:disabled="isProductShipped(scope.row) || scope.row.__inlineOtherAmountAddId === null || scope.row.__inlineOtherAmountAddId === undefined || scope.row.__inlineOtherAmountAddId === ''"
+												@click="confirmAddOtherAmountForRow(scope.row)"
+											>
+												纭娣诲姞
+											</el-button>
+										</div>
+									</div>
+
+									<div v-if="Array.isArray(scope.row.salesProductProcessList) && scope.row.salesProductProcessList.length > 0"
+										style="display:flex; flex-wrap:wrap; gap: 8px;"
+									>
+										<div
+											v-for="(item, idx) in scope.row.salesProductProcessList"
+											:key="String(item.id) + '_' + idx"
+											style="display:flex; align-items:center; gap: 8px; flex: 0 0 calc(50% - 4px); max-width: calc(50% - 4px);"
+										>
+											<el-tag type="info" style="max-width: 170px; overflow:hidden; text-overflow:ellipsis; white-space:nowrap;">
+												{{ item.processName }}
+											</el-tag>
+											<el-input-number
+												v-model="item.quantity"
+												:min="0"
+												:step="1"
+												:precision="0"
+												style="width: 120px;"
+												placeholder="鏁伴噺"
+												:disabled="operationType === 'view' || isProductShipped(scope.row)"
+												@change="handleOtherAmountQuantityChange(scope.row)"
+											/>
+											<el-button
+												type="danger"
+												link
+												size="small"
+												:disabled="isProductShipped(scope.row)"
+												@click="removeOtherAmountAtForRow(scope.row, idx)"
+											>
+												鍒犻櫎
+											</el-button>
+										</div>
+									</div>
+									<div v-else style="color:#909399; font-size: 13px;">
+										鏆傛棤鍏朵粬閲戦
+									</div>
+								</el-popover>
+							</template>
 						</template>
 					</el-table-column>
 				</el-table>
@@ -247,6 +771,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>
@@ -351,6 +889,8 @@
 								v-model="productForm.productCategory"
 								placeholder="璇烽�夋嫨"
 								clearable
+								filterable
+								:filter-node-method="filterProductCategoryNode"
 								check-strictly
 								@change="getModels"
 								:data="productOptions"
@@ -374,23 +914,22 @@
 						</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>
 
 				<!-- 姣忚涓変釜锛氱◣鐜�/鍚◣鍗曚环/鏁伴噺 -->
 				<el-row :gutter="30">
-					<el-col :span="8">
-						<el-form-item label="绋庣巼(%)锛�" prop="taxRate">
-							<el-select v-model="productForm.taxRate" placeholder="璇烽�夋嫨" clearable @change="calculateFromTaxRate" style="width: 100%">
-								<el-option label="1" value="1" />
-								<el-option label="6" value="6" />
-								<el-option label="13" value="13" />
-							</el-select>
-						</el-form-item>
-					</el-col>
 					<el-col :span="8">
 						<el-form-item label="鍚◣鍗曚环(鍏�)锛�" prop="taxInclusiveUnitPrice">
 							<el-input-number
@@ -403,6 +942,17 @@
 								clearable
 								@change="calculateFromUnitPrice"
 							/>
+						</el-form-item>
+					</el-col>
+					<el-col :span="8">
+						<el-form-item label="绋庣巼(%)锛�" prop="taxRate">
+							<el-select v-model="productForm.taxRate" placeholder="璇烽�夋嫨" clearable @change="calculateFromTaxRate" style="width: 100%">
+								<el-option label="1" value="1" />
+								<el-option label="3" value="3" />
+								<el-option label="6" value="6" />
+								<el-option label="9" value="9" />
+								<el-option label="13" value="13" />
+							</el-select>
 						</el-form-item>
 					</el-col>
 					<el-col :span="8">
@@ -528,7 +1078,7 @@
 								style="width: 100%"
 								placeholder="璇疯緭鍏�"
 								clearable
-								@change="recalcAreaTotals"
+								@change="() => { recalcAreaTotals(); calculateFromUnitPrice(true); }"
 							/>
 						</el-form-item>
 					</el-col>
@@ -561,6 +1111,13 @@
 					<el-col :span="8">
 						<el-form-item label="閲嶇锛�" prop="heavyBox">
 							<el-input v-model="productForm.heavyBox" placeholder="璇疯緭鍏�" clearable @change="calculateFromExclusiveTotalPrice" />
+						</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>
@@ -618,11 +1175,12 @@
 											<el-input-number
 												v-model="item.quantity"
 												:min="0"
-												:step="0.1"
-												:precision="2"
+												:step="1"
+												:precision="0"
 												style="width: 100%;"
 												placeholder="璇疯緭鍏ユ暟閲�"
 												:disabled="operationType === 'view'"
+												@change="calculateFromUnitPrice(true)"
 											/>
 										</div>
 										<el-button
@@ -684,6 +1242,7 @@
 				</el-button>
 			</template>
 		</el-dialog>
+
 		<!-- 瀵煎叆寮圭獥 -->
 		<FormDialog
 			v-model="importUpload.open"
@@ -725,122 +1284,6 @@
 			v-model="fileListDialogVisible"
 			title="闄勪欢鍒楄〃"
 		/>
-		<!-- 鎵撳嵃棰勮寮圭獥 -->
-		<el-dialog
-			v-model="printPreviewVisible"
-			title="鎵撳嵃棰勮"
-			width="90%"
-			:close-on-click-modal="false"
-			class="print-preview-dialog"
-		>
-			<div class="print-preview-container">
-				<div class="print-preview-header">
-					<el-button type="primary" @click="executePrint">鎵ц鎵撳嵃</el-button>
-					<el-button @click="printPreviewVisible = false">鍏抽棴棰勮</el-button>
-				</div>
-				<div class="print-preview-content">
-					<div v-if="printData.length === 0" style="text-align: center; padding: 50px; color: #999;">
-						鏆傛棤鎵撳嵃鏁版嵁
-					</div>
-					<div v-else style="text-align: center; padding: 10px; color: #666; font-size: 14px; background: #e8f4fd; margin-bottom: 10px;">
-						鍏� {{ printData.length }} 鏉℃暟鎹緟鎵撳嵃
-					</div>
-					<div v-for="(item, index) in printData" :key="index" class="print-page">
-						<div class="delivery-note">
-							<div class="header">
-								<div class="document-title">闆跺敭鍙戣揣鍗�</div>
-							</div>
-							
-							<div class="info-section">
-								<div class="info-row">
-									<div>
-										<span class="label">鍙戣揣鏃ユ湡锛�</span>
-										<span class="value">{{ formatDate(item.createTime) }}</span>
-									</div>
-									<div>
-										<span class="label">鍙戣揣杞︾墝鍙凤細</span>
-										<span class="value">{{ item.shippingCarNumber }}</span>
-									</div>
-								</div>
-								<div class="info-row">
-									<div>
-										<span class="label">瀹㈡埛鍚嶇О锛�</span>
-										<span class="value">{{ item.customerName }}</span>
-									</div>
-									<span class="label">鍗曞彿锛�</span>
-									<span class="value">{{ item.salesContractNo }}</span>
-								</div>
-							</div>
-							
-							<div class="table-section">
-								<table class="product-table">
-									<thead>
-									<tr>
-										<th>浜у搧鍚嶇О</th>
-										<th>瑙勬牸鍨嬪彿</th>
-										<th>鍗曚綅</th>
-										<th>鍗曚环</th>
-										<th>闆跺敭鏁伴噺</th>
-										<th>闆跺敭閲戦</th>
-									</tr>
-									</thead>
-									<tbody>
-									<tr v-for="product in item.products" :key="product.id">
-										<td>{{ product.productCategory || '' }}</td>
-										<td>{{ product.specificationModel || '' }}</td>
-										<td>{{ product.unit || '' }}</td>
-										<td>{{ product.taxInclusiveUnitPrice || '0' }}</td>
-										<td>{{ product.quantity || '0' }}</td>
-										<td>{{ product.taxInclusiveTotalPrice || '0' }}</td>
-									</tr>
-									<tr v-if="!item.products || item.products.length === 0">
-										<td colspan="6" style="text-align: center; color: #999;">鏆傛棤浜у搧鏁版嵁</td>
-									</tr>
-									</tbody>
-									<tfoot>
-									<tr>
-										<td class="label">鍚堣</td>
-										<td class="total-value"></td>
-										<td class="total-value"></td>
-										<td class="total-value"></td>
-										<td class="total-value">{{ getTotalQuantity(item.products) }}</td>
-										<td class="total-value">{{ getTotalAmount(item.products) }}</td>
-									</tr>
-									</tfoot>
-								</table>
-							</div>
-							
-							<div class="footer-section">
-								<div class="footer-row">
-									<div class="footer-item">
-										<span class="label">鏀惰揣鐢佃瘽锛�</span>
-										<span class="value"></span>
-									</div>
-									<div class="footer-item">
-										<span class="label">鏀惰揣浜猴細</span>
-										<span class="value"></span>
-									</div>
-									<div class="footer-item address-item">
-										<span class="label">鏀惰揣鍦板潃锛�</span>
-										<span class="value address-value"></span>
-									</div>
-								</div>
-								<div class="footer-row">
-									<div class="footer-item">
-										<span class="label">鎿嶄綔鍛橈細</span>
-										<span class="value">{{ userStore.nickName || '鎾曞紑鍓�' }}</span>
-									</div>
-									<div class="footer-item">
-										<span class="label">鎵撳嵃鏃ユ湡锛�</span>
-										<span class="value">{{ formatDateTime(new Date()) }}</span>
-									</div>
-								</div>
-							</div>
-						</div>
-					</div>
-				</div>
-			</div>
-		</el-dialog>
 		<!-- 鍙戣揣寮规 -->
 		<el-dialog
 			v-model="deliveryFormVisible"
@@ -916,128 +1359,23 @@
 			</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>
 
 <script setup>
 import { getToken } from "@/utils/auth";
 import pagination from "@/components/PIMTable/Pagination.vue";
-import {onMounted, ref, getCurrentInstance} from "vue";
+import {onMounted, ref, getCurrentInstance, watch, nextTick} from "vue";
 import { addShippingInfo } from "@/api/salesManagement/deliveryLedger.js";
 import { ElMessageBox, ElMessage } from "element-plus";
-import { UploadFilled, Download } from "@element-plus/icons-vue";
+import { ArrowDown } from "@element-plus/icons-vue";
 import useUserStore from "@/store/modules/user";
 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 +1389,22 @@
 	delLedgerFile,
 	getProductInventory,
 	salesLedgerProductProcessList,
-	salesLedgerProductProcessAdd,
-	salesLedgerProductProcessUpdate,
-	salesLedgerProductProcessDelete,
+	saleProcessBind,
+	getSaleProcessBindInfo,
+	getProcessCard,
+	getSalesOrder,
+	getSalesInvoices,
+	getSalesLabel,
 } 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 { printFinishedProcessCard } from "./components/processCardPrint.js";
+import { printSalesOrder } from "./components/salesOrderPrint.js";
+import { printSalesDeliveryNote } from "./components/salesDeliveryPrint.js";
+import { printSalesLabel } from "./components/salesLabelPrint.js";
+// import { salesLedgerProductSetProcessFlowConfig } from "@/api/salesManagement/salesProcessFlowConfig.js";
 
 const userStore = useUserStore();
 const { proxy } = getCurrentInstance();
@@ -1078,16 +1424,25 @@
 const total = ref(0);
 const fileList = ref([]);
 
+// 宸ヨ壓璺嚎閰嶇疆閫夋嫨寮圭獥锛堢粦瀹氬埌鍙拌处浜у搧锛�
+const processFlowSelectDialogVisible = ref(false);
+const processFlowSelectLedgerRow = ref(null);
+const processFlowSelectDefaultRouteId = ref(null);
+const processFlowSelectBoundRouteId = ref(null);
+const processFlowSelectBoundRouteName = ref("");
+
 // 鐢ㄦ埛淇℃伅琛ㄥ崟寮规鏁版嵁
 const operationType = ref("");
 const dialogFormVisible = ref(false);
 const data = reactive({
 	searchForm: {
 		customerName: "", // 瀹㈡埛鍚嶇О
+		customerId: "", // 瀹㈡埛ID锛堟煡璇笅鎷夛級
 		salesContractNo: "", // 閿�鍞悎鍚岀紪鍙�
 		entryDate: null, // 褰曞叆鏃ユ湡
 		entryDateStart: undefined,
 		entryDateEnd: undefined,
+		deliveryStatus: undefined, // 鍙戣揣鐘舵�侊細1鏈彂璐� 2瀹℃壒涓� 3瀹℃壒澶辫触 4宸插彂璐�
 	},
 	form: {
 		salesContractNo: "",
@@ -1111,6 +1466,17 @@
 });
 const { form, rules } = toRefs(data);
 const { form: searchForm } = useFormData(data.searchForm);
+
+// 鏂板鍙拌处锛氬綍鍏ユ棩鏈熷彉鏇存椂锛屼氦璐ф棩鏈熼粯璁や繚鎸佷负褰曞叆鏃ユ湡鍚庣 7 澶�
+watch(
+	() => [operationType.value, form.value?.entryDate],
+	() => {
+		if (operationType.value !== "add") return;
+		const ed = form.value?.entryDate;
+		if (!ed) return;
+		form.value.deliveryDate = dayjs(ed).add(7, "day").format("YYYY-MM-DD");
+	}
+);
 // 浜у搧琛ㄥ崟寮规鏁版嵁
 const productFormVisible = ref(false);
 const productOperationType = ref("");
@@ -1119,7 +1485,7 @@
 	productForm: {
 		productCategory: "",
 		specificationModel: "",
-		unit: "",
+		thickness:null,
 		quantity: "",
 		taxInclusiveUnitPrice: "",
 		taxRate: "",
@@ -1138,41 +1504,408 @@
 		processRequirement: "", // 鍔犲伐瑕佹眰
 		remark: "", // 澶囨敞
 		salesProductProcessList: [], // 鍏朵粬閲戦锛歔{id, processName, quantity}]
+		processFlowConfigId: null, // 宸ヨ壓娴佺▼閰嶇疆缁戝畾
+		floorCode: "", // 妤煎眰缂栧彿
 	},
 	productRules: {
 		productCategory: [{ required: true, message: "璇烽�夋嫨", trigger: "change" }],
 		productModelId: [{ required: true, message: "璇烽�夋嫨", trigger: "change" }],
-		specificationModel: [
-			{ required: true, message: "璇烽�夋嫨", trigger: "change" },
-		],
-		unit: [{ required: true, message: "璇疯緭鍏�", trigger: "blur" }],
-		quantity: [{ required: true, message: "璇疯緭鍏�", trigger: "blur" }],
-		taxInclusiveUnitPrice: [
-			{ required: true, message: "璇疯緭鍏�", trigger: "blur" },
-		],
-		taxRate: [{ required: true, message: "璇烽�夋嫨", trigger: "change" }],
-		taxInclusiveTotalPrice: [
-			{ required: true, message: "璇疯緭鍏�", trigger: "blur" },
-		],
-		taxExclusiveTotalPrice: [
-			{ required: true, message: "璇疯緭鍏�", trigger: "blur" },
-		],
-		invoiceType: [{ required: true, message: "璇烽�夋嫨", trigger: "change" }],
 	},
 });
 const { productForm, productRules } = toRefs(productFormData);
 // 闃叉寰幆璁$畻鐨勬爣蹇�
 const isCalculating = ref(false);
+
+// 浜у搧琛屽唴缂栬緫锛氬彧鍏佽鍚屾椂缂栬緫涓�琛�
+const editingProductRow = ref(null);
+
+const ensureProductRowDefaults = (row) => {
+	if (!row || typeof row !== "object") return;
+	if (!Array.isArray(row.salesProductProcessList)) row.salesProductProcessList = [];
+	if (row.__otherAmountPopoverVisible === undefined || row.__otherAmountPopoverVisible === null) row.__otherAmountPopoverVisible = false;
+	if (row.__inlineOtherAmountAdding === undefined || row.__inlineOtherAmountAdding === null) row.__inlineOtherAmountAdding = false;
+	if (row.__inlineOtherAmountAddId === undefined) row.__inlineOtherAmountAddId = null;
+	if (row.width === undefined || row.width === null) row.width = 0;
+	if (row.height === undefined || row.height === null) row.height = 0;
+	if (row.perimeter === undefined || row.perimeter === null) row.perimeter = 0;
+	if (row.actualPieceArea === undefined || row.actualPieceArea === null) row.actualPieceArea = 0;
+	if (row.actualTotalArea === undefined || row.actualTotalArea === null) row.actualTotalArea = 0;
+	if (row.settlePieceArea === undefined || row.settlePieceArea === null) row.settlePieceArea = 0;
+	if (row.settleTotalArea === undefined || row.settleTotalArea === null) row.settleTotalArea = 0;
+	if (row.processRequirement === undefined || row.processRequirement === null) row.processRequirement = "";
+	if (row.remark === undefined || row.remark === null) row.remark = "";
+	if (row.floorCode === undefined || row.floorCode === null) row.floorCode = "";
+	if (row.invoiceType === undefined || row.invoiceType === null) row.invoiceType = "";
+	if (row.taxRate === undefined || row.taxRate === null) row.taxRate = "";
+	if (row.quantity === undefined || row.quantity === null) row.quantity = 0;
+	if (row.taxInclusiveUnitPrice === undefined || row.taxInclusiveUnitPrice === null) row.taxInclusiveUnitPrice = 0;
+	if (row.taxInclusiveTotalPrice === undefined || row.taxInclusiveTotalPrice === null) row.taxInclusiveTotalPrice = 0;
+	if (row.taxExclusiveTotalPrice === undefined || row.taxExclusiveTotalPrice === null) row.taxExclusiveTotalPrice = 0;
+};
+
+const stopOtherEditingRows = () => {
+	(productData.value || []).forEach((r) => {
+		if (r && r.__editing) r.__editing = false;
+	});
+	editingProductRow.value = null;
+};
+
+const hasEditingProductRow = () => {
+	return (productData.value || []).some((r) => r && r.__editing);
+};
+
+const addProductInline = async () => {
+	if (operationType.value === "view") return;
+	if (hasEditingProductRow()) {
+		proxy.$modal.msgWarning("璇峰厛淇濆瓨鎴栧彇娑堝綋鍓嶇紪杈戣");
+		return;
+	}
+	await getProductOptions();
+	await fetchOtherAmountSelectOptions(true);
+	const row = {
+		id: null,
+		__tempKey: `__temp_${Date.now()}_${Math.random().toString(16).slice(2)}`,
+		__editing: true,
+		__isNew: true,
+		__productCategoryId: null,
+		productCategory: "",
+		productModelId: null,
+		specificationModel: "",
+		thickness: null,
+		quantity: 0,
+		taxInclusiveUnitPrice: 0,
+		taxRate: "",
+		taxInclusiveTotalPrice: 0,
+		taxExclusiveTotalPrice: 0,
+		invoiceType: "",
+		width: 0,
+		height: 0,
+		perimeter: 0,
+		actualPieceArea: 0,
+		actualTotalArea: 0,
+		settlePieceArea: 0,
+		settleTotalArea: 0,
+		processRequirement: "",
+		remark: "",
+		salesProductProcessList: [],
+		processFlowConfigId: null,
+		floorCode: "",
+		heavyBox: "",
+	};
+	productData.value.push(row);
+	editingProductRow.value = row;
+	// 璁╃幇鏈夌殑璁$畻/鍏朵粬閲戦閫昏緫澶嶇敤褰撳墠琛�
+	productForm.value = row;
+};
+
+const editProductInline = async (row, index) => {
+	if (operationType.value === "view") return;
+	if (!row) return;
+	if (isProductShipped(row)) {
+		proxy.$modal.msgWarning("宸插彂璐ф垨瀹℃牳閫氳繃鐨勪骇鍝佷笉鑳界紪杈�");
+		return;
+	}
+	stopOtherEditingRows();
+	await getProductOptions();
+	await fetchOtherAmountSelectOptions(true);
+	ensureProductRowDefaults(row);
+	// 浜у搧澶х被 tree-select 鍥炴樉锛氬悕绉� -> id
+	row.__productCategoryId = findNodeIdByLabel(productOptions.value, row.productCategory);
+
+	// 鍏煎鍚庣瀛楁鍛藉悕锛堜繚鎸佸師閫昏緫锛�
+	row.actualPieceArea = row?.actualPieceArea ?? row?.actual_piece_area ?? 0;
+	row.actualTotalArea = row?.actualTotalArea ?? row?.actual_total_area ?? 0;
+	row.settlePieceArea = row?.settlePieceArea ?? row?.settle_piece_area ?? 0;
+	row.settleTotalArea = row?.settleTotalArea ?? row?.settle_total_area ?? 0;
+	row.processRequirement = row?.processRequirement ?? row?.process_requirement ?? "";
+	row.remark = row?.remark ?? row?.remarks ?? "";
+	row.floorCode = row?.floorCode ?? row?.floor_code ?? "";
+	row.processFlowConfigId = row?.processFlowConfigId ?? row?.process_flow_config_id ?? null;
+	row.perimeter = row?.perimeter ?? row?.heavyBoxPerimeter ?? row?.heavyboxPerimeter ?? 0;
+	row.thickness = row?.thickness;
+
+	row.salesProductProcessList = normalizeOtherAmountsFromRow(row);
+	mergeOtherAmountOptionsBySelection(row.salesProductProcessList);
+	row.salesProductProcessList = fillOtherAmountProcessName(row.salesProductProcessList);
+
+	// 澶囦唤鐢ㄤ簬鍙栨秷
+	row.__backup = JSON.parse(JSON.stringify(row));
+	row.__editing = true;
+	editingProductRow.value = row;
+	productForm.value = row;
+
+	// 鏍规嵁浜у搧澶х被鍚嶇О鍙嶆煡 tree 鑺傜偣 id锛屽苟鍔犺浇瑙勬牸鍨嬪彿鍒楄〃
+	try {
+		const options = productOptions.value && productOptions.value.length > 0 ? productOptions.value : await getProductOptions();
+		const categoryId = findNodeIdByLabel(options, row.productCategory);
+		if (categoryId) {
+			const models = await modelList({ id: categoryId });
+			modelOptions.value = models || [];
+			const currentModel = (modelOptions.value || []).find((m) => m.model === row.specificationModel);
+			if (currentModel) row.productModelId = currentModel.id;
+		}
+	} catch (e) {
+		console.error("鍔犺浇浜у搧瑙勬牸鍨嬪彿澶辫触", e);
+	}
+
+	// 鍚屾璁$畻涓�娆�
+	recalcPerimeterFromWidthHeight();
+	recalcAreaFromWidthHeight();
+};
+
+const validateInlineProductRow = (row) => {
+	if (!row) return false;
+	if (!row.productCategory) {
+		proxy.$modal.msgWarning("璇烽�夋嫨浜у搧澶х被");
+		return false;
+	}
+	if (!row.productModelId) {
+		proxy.$modal.msgWarning("璇烽�夋嫨瑙勬牸鍨嬪彿");
+		return false;
+	}
+	return true;
+};
+
+const saveProductInline = async (row, index) => {
+	if (operationType.value === "view") return;
+	if (!row) return;
+	if (isProductShipped(row)) {
+		proxy.$modal.msgWarning("宸插彂璐ф垨瀹℃牳閫氳繃鐨勪骇鍝佷笉鑳界紪杈�");
+		return;
+	}
+	// 纭繚 productForm 鎸囧悜褰撳墠琛岋紝浠ュ鐢ㄨ绠楅�昏緫
+	productForm.value = row;
+	ensureProductRowDefaults(row);
+
+	if (!validateInlineProductRow(row)) return;
+
+	// 鍘氬害绮惧害澶勭悊
+	if (row.thickness !== null && row.thickness !== undefined && row.thickness !== "") {
+		row.thickness = Number(Number(row.thickness).toFixed(15));
+	}
+
+	// 鎻愪氦鍓嶅厹搴曡绠椾竴娆★紙娌跨敤鍘熼�昏緫锛�
+	recalcAreaTotals();
+	// 鎻愪氦鍏滃簳锛氱◣鐜�/鏁伴噺鏈~鏃舵寜鏁板瓧 0 浼犻��
+	row.taxRate = Number(row.taxRate ?? 0) || 0;
+	row.quantity = Number(row.quantity ?? 0) || 0;
+
+	// 瑙勮寖鍖栧叾浠栭噾棰濇彁浜ょ粨鏋�
+	row.salesProductProcessList = (Array.isArray(row.salesProductProcessList) ? row.salesProductProcessList : [])
+		.map((it) => ({
+			id: it?.id,
+			processName: it?.processName ?? "",
+			unitPrice: Number(it?.unitPrice ?? 0) || 0,
+			quantity: Number(it?.quantity ?? 0) || 0,
+		}))
+		.filter((it) => it.id !== null && it.id !== undefined && it.id !== "");
+
+	// 瑙勬牸鍨嬪彿锛氭牴鎹� productModelId 鍥炲~鍚嶇О
+	const model = (modelOptions.value || []).find((m) => String(m.id) === String(row.productModelId));
+	if (model?.model) row.specificationModel = model.model;
+
+	if (operationType.value === "edit") {
+		// 鍙拌处宸插瓨鍦細璧板師鎺ュ彛淇濆瓨鍒板悗绔紝鍐嶅洖鎷夊埛鏂�
+		const payload = { ...row, salesLedgerId: currentId.value, type: 1 };
+		delete payload.__backup;
+		delete payload.__editing;
+		delete payload.__isNew;
+		delete payload.__productCategoryId;
+		delete payload.__tempKey;
+		await addOrUpdateSalesLedgerProduct(payload);
+		proxy.$modal.msgSuccess("鎻愪氦鎴愬姛");
+		await getSalesLedgerWithProducts({ id: currentId.value, type: 1 }).then((res) => {
+			productData.value = res.productData;
+		});
+	} else {
+		// 鏂板鍙拌处锛氫粎鍦ㄦ湰鍦� productData 鐢熸晥锛屾渶缁堥殢鍙拌处涓�璧锋彁浜�
+		row.__isNew = false;
+		row.__editing = false;
+		delete row.__backup;
+	}
+
+	stopOtherEditingRows();
+};
+
+const cancelProductInline = (row, index) => {
+	if (!row) return;
+	if (row.__isNew) {
+		productData.value.splice(index, 1);
+	} else if (row.__backup) {
+		const restored = JSON.parse(JSON.stringify(row.__backup));
+		// 淇濈暀 id 涓庣姸鎬佸瓧娈�
+		const keepId = row.id;
+		Object.keys(row).forEach((k) => delete row[k]);
+		Object.assign(row, restored);
+		row.id = keepId;
+		row.__editing = false;
+		delete row.__backup;
+	}
+	stopOtherEditingRows();
+};
+
+const openOtherAmountInline = async (row) => {
+	if (!row) return;
+	if (operationType.value === "view") return;
+	if (isProductShipped(row)) {
+		proxy.$modal.msgWarning("宸插彂璐ф垨瀹℃牳閫氳繃鐨勪骇鍝佷笉鑳界紪杈�");
+		return;
+	}
+	ensureProductRowDefaults(row);
+	productForm.value = row;
+	otherAmountAddTargetRow.value = row;
+	await fetchOtherAmountSelectOptions(true);
+	mergeOtherAmountOptionsBySelection(row.salesProductProcessList);
+	row.salesProductProcessList = fillOtherAmountProcessName(row.salesProductProcessList);
+	// 鍙仛鏁版嵁鍑嗗涓庢墦寮�娴眰锛堟柊澧炵敱娴眰鍐呮寜閽Е鍙戯級
+	row.__otherAmountPopoverVisible = true;
+};
+
+const keepOtherAmountPopoverOpenKey = ref(null);
+const keepOtherAmountPopoverOpenUntil = ref(0);
+
+const getOtherAmountRowKey = (row) => String(row?.__tempKey ?? row?.id ?? "");
+
+const lockOtherAmountPopoverOpen = (row, durationMs = 1200) => {
+	const key = getOtherAmountRowKey(row);
+	if (!key) return;
+	keepOtherAmountPopoverOpenKey.value = key;
+	keepOtherAmountPopoverOpenUntil.value = Date.now() + durationMs;
+};
+
+const handleOtherAmountPopoverVisibleChange = (row, visible) => {
+	if (!row) return;
+	if (visible) {
+		row.__otherAmountPopoverVisible = true;
+		return;
+	}
+	if (row.__inlineOtherAmountAdding) {
+		row.__otherAmountPopoverVisible = true;
+		return;
+	}
+	const key = getOtherAmountRowKey(row);
+	const shouldKeepOpen = Boolean(
+		key &&
+		keepOtherAmountPopoverOpenKey.value === key &&
+		Date.now() < keepOtherAmountPopoverOpenUntil.value
+	);
+	row.__otherAmountPopoverVisible = shouldKeepOpen;
+};
+
+const startAddOtherAmountForRow = async (row) => {
+	if (!row) return;
+	if (operationType.value === "view") return;
+	if (isProductShipped(row)) {
+		proxy.$modal.msgWarning("宸插彂璐ф垨瀹℃牳閫氳繃鐨勪骇鍝佷笉鑳界紪杈�");
+		return;
+	}
+	ensureProductRowDefaults(row);
+	productForm.value = row;
+	await fetchOtherAmountSelectOptions(true);
+	mergeOtherAmountOptionsBySelection(row.salesProductProcessList);
+	row.salesProductProcessList = fillOtherAmountProcessName(row.salesProductProcessList);
+	row.__inlineOtherAmountAddId = null;
+	row.__inlineOtherAmountAdding = true;
+	row.__otherAmountPopoverVisible = true;
+};
+
+const confirmAddOtherAmountForRow = (row) => {
+	if (!row) return;
+	ensureProductRowDefaults(row);
+	productForm.value = row;
+	const selectedId = row.__inlineOtherAmountAddId;
+	if (selectedId === null || selectedId === undefined || selectedId === "") return;
+	const opt = otherAmountSelectOptions.value.find((o) => String(o.id) === String(selectedId));
+	if (!opt) return;
+	const exists = (row.salesProductProcessList ?? []).some(
+		(it) => String(it?.id) === String(opt.id)
+	);
+	if (exists) {
+		proxy.$modal.msgWarning("璇ュ叾浠栭噾棰濋」鐩凡娣诲姞");
+		return;
+	}
+	row.salesProductProcessList.push({
+		id: opt.id,
+		processName: opt.processName,
+		unitPrice: opt.unitPrice ?? 0,
+		quantity: 0,
+	});
+	row.__inlineOtherAmountAddId = null;
+	row.__inlineOtherAmountAdding = false;
+	row.__otherAmountPopoverVisible = true;
+	calculateFromUnitPrice(true);
+};
+
+const removeOtherAmountAtForRow = (row, index) => {
+	if (!row) return;
+	if (operationType.value === "view") return;
+	if (isProductShipped(row)) return;
+	productForm.value = row;
+	removeOtherAmountAt(index);
+};
+
+const handleOtherAmountQuantityChange = (row) => {
+	if (!row) return;
+	productForm.value = row;
+	calculateFromUnitPrice(true);
+};
+
+const handleInlineProductCategoryChange = async (row, val) => {
+	if (!row) return;
+	productForm.value = row;
+	// 澶嶇敤鍘熸湁閫昏緫锛氫細鍐欏叆 productCategory(鍚嶇О)銆侀噸缃鏍�/鍘氬害骞舵媺鍙栧瀷鍙�
+	await getModels(val);
+	// 琛屽唴缂栬緫鏃舵妸閫変腑鐨� id 璁板綍涓嬫潵锛屼究浜庡洖鏄�
+	row.__productCategoryId = val;
+};
+
+const handleInlineProductModelChange = (row, val) => {
+	if (!row) return;
+	productForm.value = row;
+	// 澶嶇敤鍘熸湁閫昏緫锛氫細鍐欏叆 specificationModel銆佸帤搴�
+	getProductModel(val);
+};
+
+const handleInlineSizeChange = (row) => {
+	if (!row) return;
+	productForm.value = row;
+	recalcPerimeterFromWidthHeight();
+	recalcAreaFromWidthHeight();
+	recalcAreaTotals();
+};
+
+const handleInlineUnitPriceChange = (row) => {
+	if (!row) return;
+	productForm.value = row;
+	calculateFromUnitPrice();
+	recalcAreaTotals();
+};
+
+const handleInlineQuantityChange = (row) => {
+	if (!row) return;
+	productForm.value = row;
+	calculateFromQuantity();
+	recalcAreaTotals();
+};
+
+const handleInlineTaxRateChange = (row) => {
+	if (!row) return;
+	productForm.value = row;
+	calculateFromTaxRate();
+};
+
+const handleInlineSettleAreaChange = (row) => {
+	if (!row) return;
+	productForm.value = row;
+	recalcAreaTotals();
+	calculateFromUnitPrice(true);
+};
 const upload = reactive({
 	// 涓婁紶鐨勫湴鍧�
 	url: import.meta.env.VITE_APP_BASE_API + "/file/upload",
 	// 璁剧疆涓婁紶鐨勮姹傚ご閮�
 	headers: { Authorization: "Bearer " + getToken() },
 });
-// 鎵撳嵃鐩稿叧
-const printPreviewVisible = ref(false);
-const printData = ref([]);
-
 // 鎶ヤ环鍗曞鍏ョ浉鍏�
 const quotationDialogVisible = ref(false);
 const quotationLoading = ref(false);
@@ -1191,181 +1924,18 @@
 
 // 鍙戣揣鐩稿叧
 const deliveryFormVisible = ref(false);
-const currentDeliveryRow = ref(null);
+const currentDeliveryRows = ref([]);
 const deliveryFormData = reactive({
   deliveryForm: {
     type: "璐ц溅", // 璐ц溅, 蹇��
   },
   deliveryRules: {
-    type: [
+    type: [	
       { required: true, message: "璇烽�夋嫨鍙戣揣绫诲瀷", trigger: "change" }
     ]
   },
 });
 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}]
@@ -1386,6 +1956,7 @@
 		otherAmountSelectOptions.value = records.map((item) => ({
 			id: item.id,
 			processName: item.processName ?? "",
+			unitPrice: item.unitPrice ?? 0,
 		}));
 	} finally {
 		otherAmountSelectOptionsLoading.value = false;
@@ -1457,6 +2028,7 @@
 		return {
 			id: s.id,
 			processName: opt?.processName ?? s.processName ?? "",
+			unitPrice: opt?.unitPrice ?? s.unitPrice ?? 0,
 			quantity: Number(s.quantity ?? 0) || 0,
 		};
 	});
@@ -1479,6 +2051,8 @@
 // 鍏朵粬閲戦锛氱偣鍑烩�滄柊澧炩�濆悗鍦ㄥ脊绐楅噷閫夋嫨涓�涓」鐩�
 const otherAmountAddDialogVisible = ref(false);
 const otherAmountAddId = ref(null);
+const otherAmountAddTargetRow = ref(null);
+const otherAmountAddTargetRowKey = ref(null);
 
 const startAddOtherAmount = () => {
 	if (operationType.value === "view") return;
@@ -1493,6 +2067,10 @@
 const cancelAddOtherAmount = () => {
 	otherAmountAddDialogVisible.value = false;
 	otherAmountAddId.value = null;
+	otherAmountAddTargetRow.value = null;
+	otherAmountAddTargetRowKey.value = null;
+	keepOtherAmountPopoverOpenKey.value = null;
+	keepOtherAmountPopoverOpenUntil.value = 0;
 };
 
 const handleOtherAmountSelected = (id) => {
@@ -1512,12 +2090,35 @@
 	productForm.value.salesProductProcessList.push({
 		id: opt.id,
 		processName: opt.processName,
+		unitPrice: opt.unitPrice ?? 0,
 		quantity: 0,
 	});
+	calculateFromUnitPrice(true);
 
-	// 閫夋嫨瀹屾垚鍚庡叧闂脊绐楋紝涓嬩竴娆″彲鍐嶆鐐瑰嚮鈥滄柊澧炩�濈户缁坊鍔�
+	// 閫夋嫨瀹屾垚鍚庡叧闂�滄柊澧炲叾浠栭噾棰濃�濆脊绐楋紝骞朵繚鎸佽鍐呪�滃叾浠栭噾棰濃�濆脊灞傚紑鍚紝渚夸簬鐩存帴濉啓鏁伴噺
 	otherAmountAddDialogVisible.value = false;
 	otherAmountAddId.value = null;
+	const reopenOtherAmountPopover = () => {
+		let targetRow = otherAmountAddTargetRow.value;
+		const rowKey = otherAmountAddTargetRowKey.value;
+		if (rowKey) {
+			const matchedRow = (productData.value || []).find(
+				(it) => String(it?.__tempKey ?? it?.id ?? "") === rowKey
+			);
+			if (matchedRow) targetRow = matchedRow;
+		}
+		if (targetRow && typeof targetRow === "object") {
+			lockOtherAmountPopoverOpen(targetRow, 1500);
+			targetRow.__otherAmountPopoverVisible = true;
+		}
+	};
+	nextTick(() => {
+		reopenOtherAmountPopover();
+		setTimeout(reopenOtherAmountPopover, 0);
+		setTimeout(reopenOtherAmountPopover, 80);
+	});
+	otherAmountAddTargetRow.value = null;
+	otherAmountAddTargetRowKey.value = null;
 };
 
 const confirmAddOtherAmount = () => {
@@ -1528,6 +2129,7 @@
 	if (operationType.value === "view") return;
 	if (!Array.isArray(productForm.value?.salesProductProcessList)) return;
 	productForm.value.salesProductProcessList.splice(index, 1);
+	calculateFromUnitPrice(true);
 };
 
 // 鍙戣揣瀹℃壒浜鸿妭鐐癸紙浠垮崗鍚屽鎵� infoFormDia.vue锛�
@@ -1622,6 +2224,21 @@
 	const params = { ...rest, ...page };
 	// 绉婚櫎褰曞叆鏃ユ湡鐨勯粯璁ゅ�艰缃紝鍙繚鐣欒寖鍥存棩鏈熷瓧娈�
 	delete params.entryDate;
+	// 鏌ヨ瀹㈡埛鍚嶇О涓庢柊澧炰繚鎸佷竴鑷达細鍏堥�� customerId锛屽啀鏄犲皠涓� customerName 鏌ヨ
+	const selectedCustomer = (customerOption.value || []).find(
+		(item) => String(item?.id ?? "") === String(params.customerId ?? "")
+	);
+	if (selectedCustomer?.customerName) {
+		params.customerName = String(selectedCustomer.customerName).trim();
+	} else {
+		const cn = params.customerName != null ? String(params.customerName).trim() : "";
+		if (cn) {
+			params.customerName = cn;
+		} else {
+			delete params.customerName;
+		}
+	}
+	delete params.customerId;
 	return ledgerListPage(params)
 		.then((res) => {
 			tableLoading.value = false;
@@ -1636,6 +2253,86 @@
 			tableLoading.value = false;
 		});
 };
+
+// 鎵撳紑鈥滃伐鑹鸿矾绾块厤缃�濋�夋嫨寮圭獥锛堝繀椤绘樉寮忛�夋嫨锛�
+const openProcessFlowSelect = async (ledgerRow) => {
+	if (!ledgerRow) return;
+	if (!ledgerRow.isEdit) return;
+
+	processFlowSelectLedgerRow.value = ledgerRow;
+	processFlowSelectDefaultRouteId.value = null;
+	processFlowSelectBoundRouteId.value = null;
+	processFlowSelectBoundRouteName.value = "";
+
+	try {
+		const res = await getSaleProcessBindInfo(ledgerRow.id);
+		const info = res?.data ?? res ?? {};
+		const boundId =
+			info?.processRouteId ??
+			info?.routeId ??
+			info?.id ??
+			null;
+		const boundName =
+			info?.processRouteName ??
+			info?.routeName ??
+			info?.name ??
+			"";
+		processFlowSelectBoundRouteId.value = boundId;
+		processFlowSelectBoundRouteName.value = boundName;
+		processFlowSelectDefaultRouteId.value = boundId;
+	} catch (e) {
+		// 鏌ヨ澶辫触鏃舵寜鏈粦瀹氬鐞嗭紝涓嶉樆濉炲脊绐�
+		processFlowSelectBoundRouteId.value = null;
+		processFlowSelectBoundRouteName.value = "";
+		processFlowSelectDefaultRouteId.value = null;
+	}
+
+	processFlowSelectDialogVisible.value = true;
+};
+
+// 缁戝畾宸ヨ壓璺嚎鍒板綋鍓嶅彴璐︽暟鎹�
+const handleProcessFlowSelectConfirm = async (routeId) => {
+	const ledgerRow = processFlowSelectLedgerRow.value;
+	if (!ledgerRow?.id) return;
+
+	const finalRouteId = routeId ?? null;
+	if (!finalRouteId) return;
+
+	const oldRouteId = processFlowSelectBoundRouteId.value;
+	if (oldRouteId !== null && oldRouteId !== undefined && oldRouteId !== "" && String(oldRouteId) !== String(finalRouteId)) {
+		try {
+			await ElMessageBox.confirm(
+				"璇ヨ鍗曞凡缁戝畾宸ヨ壓璺嚎锛屾槸鍚︾‘瀹氭洿鎹紵",
+				"鎻愮ず",
+				{
+					confirmButtonText: "纭畾",
+					cancelButtonText: "鍙栨秷",
+					type: "warning",
+				}
+			);
+		} catch {
+			return;
+		}
+	}
+
+	proxy?.$modal?.loading?.("姝e湪缁戝畾宸ヨ壓璺嚎锛岃绋嶅��...");
+	try {
+		await saleProcessBind({
+			salesLedgerId: ledgerRow.id,
+			processRouteId: finalRouteId,
+		});
+
+		proxy?.$modal?.msgSuccess?.("宸ヨ壓璺嚎缁戝畾鎴愬姛");
+		processFlowSelectDialogVisible.value = false;
+		// 缁戝畾鍚庡埛鏂板垪琛紝纭繚鎿嶄綔鍒楀啀娆$偣鍑昏兘鍥炴樉缁戝畾
+		await getList();
+	} catch (e) {
+		proxy?.$modal?.msgError?.("缁戝畾澶辫触锛岃绋嶅悗閲嶈瘯");
+	} finally {
+		proxy?.$modal?.closeLoading?.();
+	}
+};
+
 // 鑾峰彇浜у搧澶х被tree鏁版嵁
 const getProductOptions = () => {
 	// 杩斿洖 Promise锛屼究浜庡湪缂栬緫浜у搧鏃剁瓑寰呭姞杞藉畬鎴�
@@ -1649,20 +2346,44 @@
 };
 // 鑾峰彇tree瀛愭暟鎹�
 const getModels = (value) => {
+	// 浜у搧澶х被鍙樺寲鏃讹紝閲嶇疆瑙勬牸鍨嬪彿涓庡帤搴︼紝閬垮厤鏃у�兼畫鐣�
+	productForm.value.productModelId = null;
+	productForm.value.specificationModel = "";
+	productForm.value.thickness = null;
+
+	if (!value) {
+		productForm.value.productCategory = "";
+		modelOptions.value = [];
+		return;
+	}
+
 	productForm.value.productCategory = findNodeById(productOptions.value, value);
 	modelList({ id: value }).then((res) => {
-		modelOptions.value = res;
+		modelOptions.value = res || [];
 	});
 };
 const getProductModel = (value) => {
 	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;
+		const selectedModel = modelOptions.value[index];
+		const modelThickness =
+			selectedModel?.thickness ??
+			selectedModel?.modelThickness ??
+			selectedModel?.thick ??
+			null;
+		productForm.value.thickness =
+			modelThickness === null || modelThickness === undefined || modelThickness === ""
+				? null
+				: Number(modelThickness);
 	} else {
 		productForm.value.specificationModel = null;
-		productForm.value.unit = null;
+		productForm.value.thickness = null;
 	}
+};
+const filterProductCategoryNode = (value, data) => {
+	if (!value) return true;
+	return String(data?.label || "").toLowerCase().includes(String(value).toLowerCase());
 };
 const findNodeById = (nodes, productId) => {
 	for (let i = 0; i < nodes.length; i++) {
@@ -1784,11 +2505,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;
 		});
@@ -1800,6 +2524,9 @@
 	//   }
 	// });
 	form.value.entryDate = getCurrentDate(); // 璁剧疆榛樿褰曞叆鏃ユ湡涓哄綋鍓嶆棩鏈�
+	if (type === "add") {
+		form.value.deliveryDate = dayjs(form.value.entryDate).add(7, "day").format("YYYY-MM-DD");
+	}
 	dialogFormVisible.value = true;
 };
 
@@ -1879,14 +2606,15 @@
 	productData.value = products.map((p) => {
 		const quantity = Number(p.quantity ?? 0) || 0;
 		const unitPrice = Number(p.unitPrice ?? 0) || 0;
+		const settlePieceArea = Number(p.settlePieceArea ?? 0) || 1;
 		const taxRate = "13"; // 榛樿 13%锛屼究浜庣洿鎺ユ彁浜わ紙濡傞渶鍙湪浜у搧涓嚜琛屼慨鏀癸級
-		const taxInclusiveTotalPrice = (unitPrice * quantity).toFixed(2);
+		const taxInclusiveTotalPrice = (unitPrice * settlePieceArea * quantity).toFixed(2);
 		const taxExclusiveTotalPrice = proxy.calculateTaxExclusiveTotalPrice(taxInclusiveTotalPrice, taxRate);
 		return {
 			// 鍙拌处瀛楁
 			productCategory: p.product || p.productName || "",
 			specificationModel: p.specification || "",
-			unit: p.unit || "",
+			thickness: p.thickness,
 			quantity: quantity,
 			taxRate: taxRate,
 			taxInclusiveUnitPrice: unitPrice.toFixed(2),
@@ -1901,6 +2629,7 @@
 			settlePieceArea: 0,
 			settleTotalArea: 0,
 			processRequirement: "",
+			floorCode: "",
 			remark: "",
 			salesProductProcessList: [],
 		};
@@ -1952,8 +2681,21 @@
 	proxy.$refs["formRef"].validate((valid) => {
 		if (valid) {
 			console.log('productData.value--', productData.value)
+			// 琛屽唴缂栬緫鏈繚瀛樻椂涓嶅厑璁告彁浜わ紝閬垮厤鑴忔暟鎹�/涓存椂瀛楁杩涘叆鍚庣
+			const hasEditingRow = (productData.value || []).some((r) => r && r.__editing);
+			if (hasEditingRow) {
+				proxy.$modal.msgWarning("浜у搧淇℃伅瀛樺湪鏈繚瀛樼殑缂栬緫琛岋紝璇峰厛淇濆瓨鎴栧彇娑�");
+				return;
+			}
 			if (productData.value !== null && productData.value.length > 0) {
-				form.value.productData = proxy.HaveJson(productData.value);
+				const cleanedProducts = (productData.value || []).map((p) => {
+					if (!p || typeof p !== "object") return p;
+					const { __editing, __isNew, __backup, __productCategoryId, __tempKey, __otherAmountPopoverVisible, ...rest } = p;
+					rest.taxRate = Number(rest.taxRate ?? 0) || 0;
+					rest.quantity = Number(rest.quantity ?? 0) || 0;
+					return rest;
+				});
+				form.value.productData = proxy.HaveJson(cleanedProducts);
 			} else {
 				proxy.$modal.msgWarning("璇锋坊鍔犱骇鍝佷俊鎭�");
 				return;
@@ -1964,7 +2706,9 @@
 			}
 			form.value.tempFileIds = tempFileIds;
 			form.value.type = 1;
-			addOrUpdateSalesLedger(form.value).then((res) => {
+			const submitPayload = { ...form.value };
+			delete submitPayload.paymentMethod;
+			addOrUpdateSalesLedger(submitPayload).then((res) => {
 				proxy.$modal.msgSuccess("鎻愪氦鎴愬姛");
 				closeDia();
 				getList();
@@ -2007,10 +2751,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,19 +2806,28 @@
 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();
+			// 鎻愪氦鍏滃簳锛氱◣鐜�/鏁伴噺鏈~鏃舵寜鏁板瓧 0 浼犻��
+			productForm.value.taxRate = Number(productForm.value.taxRate ?? 0) || 0;
+			productForm.value.quantity = Number(productForm.value.quantity ?? 0) || 0;
 			// 鍏朵粬閲戦鍙彁浜� {id, processName, quantity}锛堝悗绔瓧娈碉細salesProductProcessList锛�
 			productForm.value.salesProductProcessList = (Array.isArray(productForm.value.salesProductProcessList)
-				? productForm.value.salesProductProcessList
-				: []
-			)
-				.map((it) => ({
-					id: it?.id,
-					processName: it?.processName ?? "",
-					quantity: Number(it?.quantity ?? 0) || 0,
-				}))
-				.filter((it) => it.id !== null && it.id !== undefined && it.id !== "");
+			? productForm.value.salesProductProcessList
+			: []
+		)
+			.map((it) => ({
+				id: it?.id,
+				processName: it?.processName ?? "",
+				unitPrice: Number(it?.unitPrice ?? 0) || 0,
+				quantity: Number(it?.quantity ?? 0) || 0,
+			}))
+			.filter((it) => it.id !== null && it.id !== undefined && it.id !== "");
 
 			if (operationType.value === "edit") {
 				submitProductEdit();
@@ -2109,9 +2869,18 @@
 	
 	if (operationType.value === "add") {
 		productSelectedRows.value.forEach((selectedRow) => {
-			const index = productData.value.findIndex(
-				(product) => product.id === selectedRow.id
-			);
+			const index = productData.value.findIndex((product) => {
+				if (!product || !selectedRow) return false;
+				// 鏂板琛� id 涓虹┖鏃讹紝鐢ㄤ复鏃� key 瀹氫綅
+				if (product.id != null && selectedRow.id != null) {
+					return String(product.id) === String(selectedRow.id);
+				}
+				return (
+					product.__tempKey &&
+					selectedRow.__tempKey &&
+					String(product.__tempKey) === String(selectedRow.__tempKey)
+				);
+			});
 			if (index !== -1) {
 				productData.value.splice(index, 1);
 			}
@@ -2248,364 +3017,118 @@
 		});
 };
 
-// 鎵撳嵃鍔熻兘
-const handlePrint = async () => {
-	if (selectedRows.value.length === 0) {
-		proxy.$modal.msgWarning("璇烽�夋嫨瑕佹墦鍗扮殑鏁版嵁");
+const handlePrintCommand = async (command) => {
+	if (command !== "finishedProcessCard" && command !== "salesOrder" && command !== "salesDeliveryNote") return;
+	if (command === "salesDeliveryNote") {
+		if (selectedRows.value.length === 0) {
+			proxy.$modal.msgWarning("璇疯嚦灏戦�夋嫨涓�鏉¢攢鍞彴璐︽暟鎹繘琛屾墦鍗�");
+			return;
+		}
+		const customerNames = Array.from(
+			new Set(selectedRows.value.map((item) => String(item?.customerName ?? "").trim()))
+		);
+		if (customerNames.length > 1) {
+			proxy.$modal.msgWarning("浠呮敮鎸佺浉鍚屽鎴峰悕绉扮殑閿�鍞彴璐﹀悎骞跺彂璐ф墦鍗�");
+			return;
+		}
+	} else if (selectedRows.value.length !== 1) {
+		proxy.$modal.msgWarning("璇烽�夋嫨涓�鏉¢攢鍞彴璐︽暟鎹繘琛屾墦鍗�");
 		return;
 	}
-	
-	// 鏄剧ず鍔犺浇鐘舵��
-	proxy.$modal.loading("姝e湪鑾峰彇浜у搧鏁版嵁锛岃绋嶅��...");
-	
-	try {
-		// 涓烘瘡涓�変腑鐨勯攢鍞彴璐﹁褰曟煡璇㈠搴旂殑浜у搧鏁版嵁
-		const printDataWithProducts = [];
-		
-		for (const row of selectedRows.value) {
-			try {
-				// 璋冪敤productList鎺ュ彛鏌ヨ浜у搧鏁版嵁
-				const productRes = await productList({ salesLedgerId: row.id, type: 1 });
-				
-				// 灏嗕骇鍝佹暟鎹暣鍚堝埌閿�鍞彴璐﹁褰曚腑
-				const rowWithProducts = {
-					...row,
-					products: productRes.data || []
-				};
-				
-				printDataWithProducts.push(rowWithProducts);
-			} catch (error) {
-				console.error(`鑾峰彇閿�鍞彴璐� ${row.id} 鐨勪骇鍝佹暟鎹け璐�:`, error);
-				// 鍗充娇鏌愪釜璁板綍鐨勪骇鍝佹暟鎹幏鍙栧け璐ワ紝涔熻鍖呭惈璇ヨ褰�
-				printDataWithProducts.push({
-					...row,
-					products: []
-				});
-			}
+
+	const selectedRow = selectedRows.value[0];
+	const selectedId = selectedRow?.id;
+	if (command === "salesDeliveryNote") {
+		const selectedIds = selectedRows.value
+			.map((item) => item?.id)
+			.filter((id) => id !== null && id !== undefined && id !== "");
+		if (selectedIds.length !== selectedRows.value.length) {
+			proxy.$modal.msgWarning("褰撳墠閫夋嫨鏁版嵁瀛樺湪缂哄皯ID鐨勮褰曪紝鏃犳硶鎵撳嵃");
+			return;
 		}
-		
-		printData.value = printDataWithProducts;
-		console.log('鎵撳嵃鏁版嵁锛堝寘鍚骇鍝侊級:', printData.value);
-		printPreviewVisible.value = true;
-		
+		const loadingText =
+			command === "salesOrder"
+				? "姝e湪鑾峰彇閿�鍞鍗曟暟鎹紝璇风◢鍊�..."
+				: command === "salesDeliveryNote"
+					? "姝e湪鑾峰彇閿�鍞彂璐у崟鏁版嵁锛岃绋嶅��..."
+					: "姝e湪鑾峰彇鐢熶骇娴佺▼鍗℃暟鎹紝璇风◢鍊�...";
+		proxy.$modal.loading(loadingText);
+		try {
+			const res = await getSalesInvoices(selectedIds);
+			const salesInvoiceData = res?.data ?? {};
+			printSalesDeliveryNote(salesInvoiceData, selectedRow);
+		} catch (error) {
+			console.error("鎵撳嵃閿�鍞彂璐у崟澶辫触:", error);
+			proxy.$modal.msgError("鎵撳嵃澶辫触锛岃绋嶅悗閲嶈瘯");
+		} finally {
+			proxy.$modal.closeLoading();
+		}
+		return;
+	}
+	if (!selectedId) {
+		proxy.$modal.msgWarning("褰撳墠閫夋嫨鏁版嵁缂哄皯ID锛屾棤娉曟墦鍗�");
+		return;
+	}
+
+	const loadingText =
+		command === "salesOrder"
+			? "姝e湪鑾峰彇閿�鍞鍗曟暟鎹紝璇风◢鍊�..."
+			: command === "salesDeliveryNote"
+				? "姝e湪鑾峰彇閿�鍞彂璐у崟鏁版嵁锛岃绋嶅��..."
+				: "姝e湪鑾峰彇鐢熶骇娴佺▼鍗℃暟鎹紝璇风◢鍊�...";
+	proxy.$modal.loading(loadingText);
+	try {
+		if (command === "salesOrder") {
+			const res = await getSalesOrder(selectedId);
+			const salesOrderData = res?.data ?? {};
+			printSalesOrder(salesOrderData);
+		} else {
+			const res = await getProcessCard(selectedId);
+			const processCardData = res?.data ?? {};
+			printFinishedProcessCard(processCardData);
+		}
 	} catch (error) {
-		console.error('鑾峰彇浜у搧鏁版嵁澶辫触:', error);
-		proxy.$modal.msgError("鑾峰彇浜у搧鏁版嵁澶辫触锛岃閲嶈瘯");
+		console.error(
+			command === "salesOrder"
+				? "鎵撳嵃閿�鍞鍗曞け璐�:"
+				: command === "salesDeliveryNote"
+					? "鎵撳嵃閿�鍞彂璐у崟澶辫触:"
+					: "鎵撳嵃鐢熶骇娴佺▼鍗″け璐�:",
+			error
+		);
+		proxy.$modal.msgError("鎵撳嵃澶辫触锛岃绋嶅悗閲嶈瘯");
 	} finally {
 		proxy.$modal.closeLoading();
 	}
 };
-// 鎵ц鎵撳嵃
-const executePrint = () => {
-	console.log('寮�濮嬫墽琛屾墦鍗帮紝鏁版嵁鏉℃暟:', printData.value.length);
-	console.log('鎵撳嵃鏁版嵁:', printData.value);
-	
-	// 鍒涘缓涓�涓柊鐨勬墦鍗扮獥鍙�
-	const printWindow = window.open('', '_blank', 'width=800,height=600');
-	
-	// 鏋勫缓鎵撳嵃鍐呭
-	let printContent = `
-    <!DOCTYPE html>
-    <html>
-    <head>
-      <meta charset="UTF-8">
-      <title>鎵撳嵃棰勮</title>
-      <style>
-        body {
-          margin: 0;
-          padding: 0;
-          font-family: "SimSun", serif;
-          background: white;
-        }
-                                                     .print-page {
-            width: 200mm;
-            height: 75mm;
-            padding: 10mm;
-            padding-left: 20mm;
-            background: white;
-            box-sizing: border-box;
-            page-break-after: always;
-            page-break-inside: avoid;
-          }
-         .print-page:last-child {
-           page-break-after: avoid;
-         }
-        .delivery-note {
-          width: 100%;
-          height: 100%;
-          font-size: 12px;
-          line-height: 1.2;
-          display: flex;
-          flex-direction: column;
-          color: #000;
-        }
-        .header {
-          text-align: center;
-          margin-bottom: 8px;
-        }
-        .company-name {
-          font-size: 18px;
-          font-weight: bold;
-          margin-bottom: 4px;
-        }
-        .document-title {
-          font-size: 16px;
-          font-weight: bold;
-        }
-        .info-section {
-          margin-bottom: 8px;
-          display: flex;
-          justify-content: space-between;
-          align-items: center;
-        }
-        .info-row {
-          line-height: 20px;
-        }
-        .label {
-          font-weight: bold;
-          width: 60px;
-          font-size: 12px;
-        }
-        .value {
-          margin-right: 20px;
-          min-width: 80px;
-          font-size: 12px;
-        }
-                 .table-section {
-                 margin-bottom: 40px;
-          //  flex: 0.6;
-         }
-        .product-table {
-          width: 100%;
-          border-collapse: collapse;
-          border: 1px solid #000;
-        }
-                 .product-table th, .product-table td {
-           border: 1px solid #000;
-           padding: 6px;
-           text-align: center;
-           font-size: 12px;
-           line-height: 1.4;
-         }
-        .product-table th {
-          font-weight: bold;
-        }
-        .total-value {
-          font-weight: bold;
-        }
-        .footer-section {
-          margin-top: auto;
-        }
-        .footer-row {
-          display: flex;
-          margin-bottom: 3px;
-          line-height: 22px;
-          justify-content: space-between;
-        }
-        .footer-item {
-          display: flex;
-          margin-right: 20px;
-        }
-        .footer-item .label {
-          font-weight: bold;
-          width: 80px;
-          font-size: 12px;
-        }
-        .footer-item .value {
-          min-width: 80px;
-          font-size: 12px;
-        }
-        .address-item .address-value {
-          min-width: 200px;
-        }
-        @media print {
-          body {
-            margin: 0;
-            padding: 0;
-          }
-                     .print-page {
-             margin: 0;
-             padding: 10mm;
-             /* padding-left: 20mm; */
-             page-break-inside: avoid;
-             page-break-after: always;
-           }
-           .print-page:last-child {
-             page-break-after: avoid;
-           }
-        }
-      </style>
-    </head>
-    <body>
-  `;
-	
-	// 涓烘瘡鏉℃暟鎹敓鎴愭墦鍗伴〉闈�
-	printData.value.forEach((item, index) => {
-		printContent += `
-      <div class="print-page">
-        <div class="delivery-note">
-          <div class="header">
-            <div class="document-title">闆跺敭鍙戣揣鍗�</div>
-          </div>
-          
-          <div class="info-section">
-            <div class="info-row">
-              <div>
-                <span class="label">鍙戣揣鏃ユ湡锛�</span>
-                <span class="value">${formatDate(item.createTime)}</span>
-              </div>
-              <div>
-                <span class="label">瀹㈡埛鍚嶇О锛�</span>
-                <span class="value">${item.customerName}</span>
-              </div>
-            </div>
-            <div class="info-row">
-              <span class="label">鍗曞彿锛�</span>
-              <span class="value">${item.salesContractNo || ''}</span>
-            </div>
-          </div>
 
-          <div class="table-section">
-            <table class="product-table">
-              <thead>
-                <tr>
-                  <th>浜у搧鍚嶇О</th>
-                  <th>瑙勬牸鍨嬪彿</th>
-                  <th>鍗曚綅</th>
-                  <th>鍗曚环</th>
-                  <th>闆跺敭鏁伴噺</th>
-                  <th>闆跺敭閲戦</th>
-                </tr>
-              </thead>
-              <tbody>
-                ${item.products && item.products.length > 0 ?
-			item.products.map(product => `
-                    <tr>
-                      <td>${product.productCategory || ''}</td>
-                      <td>${product.specificationModel || ''}</td>
-                      <td>${product.unit || ''}</td>
-                      <td>${product.taxInclusiveUnitPrice || '0'}</td>
-                      <td>${product.quantity || '0'}</td>
-                      <td>${product.taxInclusiveTotalPrice || '0'}</td>
-                    </tr>
-                  `).join('') :
-			'<tr><td colspan="6" style="text-align: center; color: #999;">鏆傛棤浜у搧鏁版嵁</td></tr>'
+const handlePrintLabel = async () => {
+	if (selectedRows.value.length !== 1) {
+		proxy.$modal.msgWarning("璇烽�夋嫨涓�鏉¢攢鍞彴璐︽暟鎹繘琛屾爣绛炬墦鍗�");
+		return;
+	}
+
+	const selectedId = selectedRows.value[0]?.id;
+	if (!selectedId) {
+		proxy.$modal.msgWarning("褰撳墠閫夋嫨鏁版嵁缂哄皯ID锛屾棤娉曟墦鍗版爣绛�");
+		return;
+	}
+
+	proxy.$modal.loading("姝e湪鑾峰彇鏍囩鏁版嵁锛岃绋嶅��...");
+	try {
+		const res = await getSalesLabel(selectedId);
+		const labelList = res?.data ?? [];
+		if (!Array.isArray(labelList) || labelList.length === 0) {
+			proxy.$modal.msgWarning("鏆傛棤鍙墦鍗版爣绛炬暟鎹�");
+			return;
 		}
-              </tbody>
-              <tfoot>
-                <tr>
-                  <td class="label">鍚堣</td>
-                  <td class="total-value"></td>
-                  <td class="total-value"></td>
-                  <td class="total-value"></td>
-                  <td class="total-value">${getTotalQuantityForPrint(item.products)}</td>
-                  <td class="total-value">${getTotalAmountForPrint(item.products)}</td>
-                </tr>
-              </tfoot>
-            </table>
-          </div>
-
-          <div class="footer-section">
-            <div class="footer-row">
-              <div class="footer-item">
-                <span class="label">鏀惰揣鐢佃瘽锛�</span>
-                <span class="value"></span>
-              </div>
-              <div class="footer-item">
-                <span class="label">鏀惰揣浜猴細</span>
-                <span class="value"></span>
-              </div>
-              <div class="footer-item address-item">
-                <span class="label">鏀惰揣鍦板潃锛�</span>
-                <span class="value address-value"></span>
-              </div>
-            </div>
-            <div class="footer-row">
-              <div class="footer-item">
-                <span class="label">鎿嶄綔鍛橈細</span>
-                <span class="value">${userStore.nickName || '鎾曞紑鍓�'}</span>
-              </div>
-              <div class="footer-item">
-                <span class="label">鎵撳嵃鏃ユ湡锛�</span>
-                <span class="value">${formatDateTime(new Date())}</span>
-              </div>
-            </div>
-          </div>
-        </div>
-      </div>
-    `;
-	});
-	
-	printContent += `
-    </body>
-    </html>
-  `;
-	
-	// 鍐欏叆鍐呭鍒版柊绐楀彛
-	printWindow.document.write(printContent);
-	printWindow.document.close();
-	
-	// 绛夊緟鍐呭鍔犺浇瀹屾垚鍚庢墦鍗�
-	printWindow.onload = () => {
-		setTimeout(() => {
-			printWindow.print();
-			printWindow.close();
-			printPreviewVisible.value = false;
-		}, 500);
-	};
-};
-// 鏍煎紡鍖栨棩鏈�
-const formatDate = (dateString) => {
-	if (!dateString) return getCurrentDate();
-	const date = new Date(dateString);
-	const year = date.getFullYear();
-	const month = String(date.getMonth() + 1).padStart(2, "0");
-	const day = String(date.getDate()).padStart(2, "0");
-	return `${year}/${month}/${day}`;
-};
-// 鏍煎紡鍖栨棩鏈熸椂闂�
-const formatDateTime = (date) => {
-	const year = date.getFullYear();
-	const month = String(date.getMonth() + 1).padStart(2, "0");
-	const day = String(date.getDate()).padStart(2, "0");
-	const hours = String(date.getHours()).padStart(2, "0");
-	const minutes = String(date.getMinutes()).padStart(2, "0");
-	const seconds = String(date.getSeconds()).padStart(2, "0");
-	return `${year}/${month}/${day} ${hours}:${minutes}:${seconds}`;
-};
-// 璁$畻浜у搧鎬绘暟閲�
-const getTotalQuantity = (products) => {
-	if (!products || products.length === 0) return '0';
-	const total = products.reduce((sum, product) => {
-		return sum + (parseFloat(product.quantity) || 0);
-	}, 0);
-	return total.toFixed(2);
-};
-
-// 璁$畻浜у搧鎬婚噾棰�
-const getTotalAmount = (products) => {
-	if (!products || products.length === 0) return '0';
-	const total = products.reduce((sum, product) => {
-		return sum + (parseFloat(product.taxInclusiveTotalPrice) || 0);
-	}, 0);
-	return total.toFixed(2);
-};
-
-// 鐢ㄤ簬鎵撳嵃鐨勮绠楀嚱鏁�
-const getTotalQuantityForPrint = (products) => {
-	if (!products || products.length === 0) return '0';
-	const total = products.reduce((sum, product) => {
-		return sum + (parseFloat(product.quantity) || 0);
-	}, 0);
-	return total.toFixed(2);
-};
-
-const getTotalAmountForPrint = (products) => {
-	if (!products || products.length === 0) return '0';
-	const total = products.reduce((sum, product) => {
-		return sum + (parseFloat(product.taxInclusiveTotalPrice) || 0);
-	}, 0);
-	return total.toFixed(2);
+		printSalesLabel(labelList);
+	} catch (error) {
+		console.error("鎵撳嵃鏍囩澶辫触:", error);
+		proxy.$modal.msgError("鎵撳嵃鏍囩澶辫触锛岃绋嶅悗閲嶈瘯");
+	} finally {
+		proxy.$modal.closeLoading();
+	}
 };
 
 const mathNum = () => {
@@ -2616,12 +3139,16 @@
 	if (!productForm.value.quantity) {
 		return;
 	}
-	// 鍚◣鎬讳环璁$畻
-	productForm.value.taxInclusiveTotalPrice =
-		proxy.calculateTaxIncludeTotalPrice(
-			productForm.value.taxInclusiveUnitPrice,
-			productForm.value.quantity
-		);
+	const settlePieceArea = parseFloat(productForm.value.settlePieceArea) || 1;
+	// 鍚◣鎬讳环璁$畻 = 鍗曚环 * 缁撶畻闈㈢Н * 鏁伴噺 + 鍏朵粬閲戦鎬诲拰
+	const basePrice = proxy.calculateTaxIncludeTotalPrice(
+		productForm.value.taxInclusiveUnitPrice * settlePieceArea,
+		productForm.value.quantity
+	);
+	const otherAmountTotal = (productForm.value.salesProductProcessList || []).reduce((total, item) => {
+		return total + (Number(item.unitPrice) || 0) * (Number(item.quantity) || 0);
+	}, 0);
+	productForm.value.taxInclusiveTotalPrice = (parseFloat(basePrice) + otherAmountTotal).toFixed(2);
 	if (productForm.value.taxRate) {
 		// 涓嶅惈绋庢�讳环璁$畻
 		productForm.value.taxExclusiveTotalPrice =
@@ -2682,15 +3209,12 @@
 	const computed = Number(computedPieceArea.toFixed(5));
 
 	productForm.value.actualPieceArea = computed;
-
-	// settlePieceArea锛氳嫢鐢ㄦ埛鏈~鍐�/涓�0锛屽垯榛樿浣跨敤瀹介珮璁$畻鍊�
-	const settlePieceRaw = Number(productForm.value.settlePieceArea ?? 0) || 0;
-	if (!settlePieceRaw) {
-		productForm.value.settlePieceArea = computed;
-	}
+	productForm.value.settlePieceArea = computed;
 
 	recalcPerimeterFromWidthHeight();
 	recalcAreaTotals();
+	// 闈㈢Н鏇存柊鍚庯紝閲嶆柊璁$畻鍚◣鎬讳环 = 鍗曚环 * 缁撶畻闈㈢Н * 鏁伴噺
+	calculateFromUnitPrice(true);
 };
 
 // 鏍规嵁鍚◣鎬讳环璁$畻鍚◣鍗曚环鍜屾暟閲�
@@ -2706,8 +3230,12 @@
 	
 	isCalculating.value = true;
 	
-	// 璁$畻鍚◣鍗曚环 = 鍚◣鎬讳环 / 鏁伴噺
-	productForm.value.taxInclusiveUnitPrice = (totalPrice / quantity).toFixed(2);
+	// 璁$畻鍚◣鍗曚环 = (鍚◣鎬讳环 - 鍏朵粬閲戦鎬诲拰) / 鏁伴噺
+	const otherAmountTotal = (productForm.value.salesProductProcessList || []).reduce((total, item) => {
+		return total + (Number(item.unitPrice) || 0) * (Number(item.quantity) || 0);
+	}, 0);
+	const basePrice = totalPrice - otherAmountTotal;
+	productForm.value.taxInclusiveUnitPrice = (basePrice / quantity).toFixed(2);
 	
 	// 濡傛灉鏈夌◣鐜囷紝璁$畻涓嶅惈绋庢�讳环
 	if (productForm.value.taxRate) {
@@ -2723,10 +3251,10 @@
 
 // 鏍规嵁涓嶅惈绋庢�讳环璁$畻鍚◣鍗曚环鍜屾暟閲�
 const calculateFromExclusiveTotalPrice = () => {
-	if (!productForm.value.taxRate) {
-		proxy.$modal.msgWarning("璇峰厛閫夋嫨绋庣巼");
-		return;
-	}
+	// if (!productForm.value.taxRate) {
+	// 	proxy.$modal.msgWarning("璇峰厛閫夋嫨绋庣巼");
+	// 	return;
+	// }
 	if (isCalculating.value) return;
 	
 	const exclusiveTotalPrice = parseFloat(productForm.value.taxExclusiveTotalPrice);
@@ -2744,32 +3272,41 @@
 	const inclusiveTotalPrice = exclusiveTotalPrice / (1 - taxRateDecimal);
 	productForm.value.taxInclusiveTotalPrice = inclusiveTotalPrice.toFixed(2);
 	
-	// 璁$畻鍚◣鍗曚环 = 鍚◣鎬讳环 / 鏁伴噺
-	productForm.value.taxInclusiveUnitPrice = (inclusiveTotalPrice / quantity).toFixed(2);
+	// 璁$畻鍚◣鍗曚环 = (鍚◣鎬讳环 - 鍏朵粬閲戦鎬诲拰) / 鏁伴噺
+	const otherAmountTotal = (productForm.value.salesProductProcessList || []).reduce((total, item) => {
+		return total + (Number(item.unitPrice) || 0) * (Number(item.quantity) || 0);
+	}, 0);
+	const basePrice = inclusiveTotalPrice - otherAmountTotal;
+	productForm.value.taxInclusiveUnitPrice = (basePrice / quantity).toFixed(2);
 	
 	isCalculating.value = false;
 };
 
 // 鏍规嵁鏁伴噺鍙樺寲璁$畻鎬讳环
 const calculateFromQuantity = () => {
-	if (!productForm.value.taxRate) {
-		proxy.$modal.msgWarning("璇峰厛閫夋嫨绋庣巼");
-		return;
-	}
+	// if (!productForm.value.taxRate) {
+	// 	proxy.$modal.msgWarning("璇峰厛閫夋嫨绋庣巼");
+	// 	return;
+	// }
 	if (isCalculating.value) return;
-	
+
 	const quantity = parseFloat(productForm.value.quantity);
 	const unitPrice = parseFloat(productForm.value.taxInclusiveUnitPrice);
-	
+	const settlePieceArea = parseFloat(productForm.value.settlePieceArea) || 1;
+
 	if (!quantity || quantity <= 0 || !unitPrice) {
 		return;
 	}
-	
+
 	isCalculating.value = true;
-	
-	// 璁$畻鍚◣鎬讳环
-	productForm.value.taxInclusiveTotalPrice = (unitPrice * quantity).toFixed(2);
-	
+
+	// 璁$畻鍚◣鎬讳环 = 鍗曚环 * 缁撶畻闈㈢Н * 鏁伴噺 + 鍏朵粬閲戦鎬诲拰
+	const basePrice = unitPrice * settlePieceArea * quantity;
+	const otherAmountTotal = (productForm.value.salesProductProcessList || []).reduce((total, item) => {
+		return total + (Number(item.unitPrice) || 0) * (Number(item.quantity) || 0);
+	}, 0);
+	productForm.value.taxInclusiveTotalPrice = (basePrice + otherAmountTotal).toFixed(2);
+
 	// 濡傛灉鏈夌◣鐜囷紝璁$畻涓嶅惈绋庢�讳环
 	if (productForm.value.taxRate) {
 		productForm.value.taxExclusiveTotalPrice =
@@ -2778,30 +3315,35 @@
 				productForm.value.taxRate
 			);
 	}
-	
+
 	isCalculating.value = false;
 };
 
 // 鏍规嵁鍚◣鍗曚环鍙樺寲璁$畻鎬讳环
-const calculateFromUnitPrice = () => {
-	if (!productForm.value.taxRate) {
-		proxy.$modal.msgWarning("璇峰厛閫夋嫨绋庣巼");
-		return;
-	}
+const calculateFromUnitPrice = (silent = false) => {
+	// if (!productForm.value.taxRate) {
+	// 	if (!silent) proxy.$modal.msgWarning("璇峰厛閫夋嫨绋庣巼");
+	// 	return;
+	// }
 	if (isCalculating.value) return;
-	
+
 	const quantity = parseFloat(productForm.value.quantity);
 	const unitPrice = parseFloat(productForm.value.taxInclusiveUnitPrice);
-	
+	const settlePieceArea = parseFloat(productForm.value.settlePieceArea) || 1;
+
 	if (!quantity || quantity <= 0 || !unitPrice) {
 		return;
 	}
-	
+
 	isCalculating.value = true;
-	
-	// 璁$畻鍚◣鎬讳环
-	productForm.value.taxInclusiveTotalPrice = (unitPrice * quantity).toFixed(2);
-	
+
+	// 璁$畻鍚◣鎬讳环 = 鍗曚环 * 缁撶畻闈㈢Н * 鏁伴噺 + 鍏朵粬閲戦鎬诲拰
+	const basePrice = unitPrice * settlePieceArea * quantity;
+	const otherAmountTotal = (productForm.value.salesProductProcessList || []).reduce((total, item) => {
+		return total + (Number(item.unitPrice) || 0) * (Number(item.quantity) || 0);
+	}, 0);
+	productForm.value.taxInclusiveTotalPrice = (basePrice + otherAmountTotal).toFixed(2);
+
 	// 濡傛灉鏈夌◣鐜囷紝璁$畻涓嶅惈绋庢�讳环
 	if (productForm.value.taxRate) {
 		productForm.value.taxExclusiveTotalPrice =
@@ -2810,16 +3352,16 @@
 				productForm.value.taxRate
 			);
 	}
-	
+
 	isCalculating.value = false;
 };
 
 // 鏍规嵁绋庣巼鍙樺寲璁$畻涓嶅惈绋庢�讳环
 const calculateFromTaxRate = () => {
-	if (!productForm.value.taxRate) {
-		proxy.$modal.msgWarning("璇峰厛閫夋嫨绋庣巼");
-		return;
-	}
+	// if (!productForm.value.taxRate) {
+	// 	proxy.$modal.msgWarning("璇峰厛閫夋嫨绋庣巼");
+	// 	return;
+	// }
 	if (isCalculating.value) return;
 	
 	const inclusiveTotalPrice = parseFloat(productForm.value.taxInclusiveTotalPrice);
@@ -2908,11 +3450,23 @@
  * @param row 琛屾暟鎹�
  */
 const canShip = (row) => {
+
 	// 浜у搧鐘舵�佸繀椤绘槸鍏呰冻锛坅pproveStatus === 1锛�
 	if (row.approveStatus !== 1) {
 		return false;
 	}
 	
+	// 濡傛灉鍚庣杩斿洖浜嗗彴璐︾骇鍙戣揣鐘舵�侊紙deliveryStatus锛�
+	// 1=宸插彂璐э紝鍒欑姝㈠啀娆″彂璐�
+	const deliveryStatus = row.deliveryStatus;
+	if (
+		deliveryStatus !== null &&
+		deliveryStatus !== undefined &&
+		String(deliveryStatus).trim() !== ""
+	) {
+		if (Number(deliveryStatus) === 1) return false;
+	}
+
 	// 鑾峰彇鍙戣揣鐘舵��
 	const shippingStatus = row.shippingStatus;
 	
@@ -2924,6 +3478,117 @@
 	// 鍙戣揣鐘舵�佸繀椤绘槸"寰呭彂璐�"鎴�"瀹℃牳鎷掔粷"
 	const statusStr = shippingStatus ? String(shippingStatus).trim() : '';
 	return statusStr === '寰呭彂璐�' || statusStr === '瀹℃牳鎷掔粷';
+};
+
+const handleBulkDelivery = async () => {
+	if (selectedRows.value.length === 0) {
+		proxy.$modal.msgWarning("璇烽�夋嫨鏁版嵁");
+		return;
+	}
+
+	// 鍙厑璁搞�愭湭鍙戣揣/瀹℃壒澶辫触銆戣繘鍏ュ彂璐ф祦绋�
+	const canDeliveryLedgers = selectedRows.value.filter((r) => {
+		const status = Number(r.deliveryStatus);
+		return status === 1 || status === 3;
+	});
+	if (canDeliveryLedgers.length === 0) {
+		proxy.$modal.msgWarning("浠呮湭鍙戣揣鎴栧鎵瑰け璐ョ殑鍙拌处鍙互鍙戣揣");
+		return;
+	}
+
+	// 宸插彂璐у彴璐︼細寮圭獥鎻愰啋锛屼笉鑳藉啀娆″彂璐э紙4 瑙嗕负宸插彂璐э級
+	const shippedLedgers = selectedRows.value.filter((r) => Number(r.deliveryStatus) === 4);
+	if (shippedLedgers.length === selectedRows.value.length) {
+		try {
+			await ElMessageBox.alert("鎵�閫夐攢鍞彴璐﹀潎宸插彂璐э紝涓嶈兘鍐嶆鍙戣揣銆�", "鎻愮ず", {
+				type: "warning",
+				confirmButtonText: "鐭ラ亾浜�",
+			});
+		} catch {
+			/* 鍏抽棴寮圭獥 */
+		}
+		return;
+	}
+	if (shippedLedgers.length > 0) {
+		try {
+			await ElMessageBox.alert(
+				"閫変腑鐨勯攢鍞彴璐︿腑鍖呭惈宸插彂璐ц褰曪紝宸插彂璐х殑涓嶈兘鍐嶆鍙戣揣锛岀郴缁熷皢浠呬负鏈彂璐у彴璐﹀鐞嗐��",
+				"鎻愮ず",
+				{
+					type: "warning",
+					confirmButtonText: "鐭ラ亾浜�",
+				}
+			);
+		} catch {
+			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) {
+
+			//濡傛灉宸茬粡鏄�滃鎵逛腑(2)鈥濇垨鈥滃凡鍙戣揣(4)鈥濓紝鍒欒烦杩囷紝涓嶅厑璁搁噸澶嶆搷浣�
+			const status = Number(ledger.deliveryStatus);
+			if (status === 2 || status === 4) {
+				console.warn(`鍙拌处缂栧彿 ${ledger.salesContractNo} 鐘舵�佷负 ${status}锛岃烦杩囧彂璐);
+				continue;
+			}
+
+			let products = [];
+			try {
+				const res = await productList({ salesLedgerId: ledger.id, type: 1 });
+				products = res?.data || [];
+			} catch (error) {
+				products = [];
+				console.error('璇锋眰鍙戠敓寮傚父', error);
+			}
+			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();
+	}
 };
 
 /**
@@ -2941,15 +3606,16 @@
 	});
 }
 
-// 鎵撳紑鍙戣揣寮规
+// 鎵撳紑鍙戣揣寮规锛堝崟鏉★級
 const openDeliveryForm = (row) => {
-	// 妫�鏌ユ槸鍚﹀彲浠ュ彂璐�
-	if (!canShip(row)) {
-		proxy.$modal.msgWarning("鍙湁鍦ㄤ骇鍝佺姸鎬佹槸鍏呰冻锛屽彂璐х姸鎬佹槸寰呭彂璐ф垨瀹℃牳鎷掔粷鐨勬椂鍊欐墠鍙互鍙戣揣");
+	// 鍙厑璁搞�愭湭鍙戣揣/瀹℃壒澶辫触銆戝彂璐э紱宸插彂璐�/瀹℃壒涓笉鍏佽
+	const status = Number(row.deliveryStatus);
+	if (status !== 1 && status !== 3) {
+		proxy.$modal.msgWarning("鍙湁鍙戣揣鐘舵�佷负鏈彂璐ф垨瀹℃壒澶辫触鐨勮褰曟墠鍙互鍙戣揣");
 		return;
 	}
-	
-	currentDeliveryRow.value = row;
+
+	currentDeliveryRows.value = [row];
   deliveryForm.value = {
     type: "璐ц溅",
   };
@@ -2972,13 +3638,27 @@
       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;
+      }
+
+      // 鎸夊彴璐︾淮搴﹀幓閲嶏紝姣忎釜 salesLedgerId 鍙皟鐢ㄤ竴娆″彂璐ф帴鍙�
+      const uniqueLedgerIds = [...new Set(targets.map((item) => item.salesLedgerId).filter(Boolean))];
+
+		const run = async () => {
+			for (const salesLedgerId of uniqueLedgerIds) {
+				await addShippingInfo({
+					salesLedgerId,
+					type: deliveryForm.value.type,
+					approveUserIds,
+				});
+			}
+		};
+
+      run()
         .then(() => {
           proxy.$modal.msgSuccess("鍙戣揣鎴愬姛");
           closeDeliveryDia();
@@ -2986,8 +3666,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 +3675,14 @@
                 });
               });
               Promise.all(loadPromises).then(() => {
-                // 鎭㈠灞曞紑鐘舵��
                 expandedRowKeys.value = currentExpandedKeys;
               });
             }
           });
         })
+        .catch(() => {
+          proxy.$modal.msgError("鍙戣揣澶辫触锛岃绋嶅悗閲嶈瘯");
+        });
     }
   });
 };
@@ -3010,7 +3691,7 @@
 const closeDeliveryDia = () => {
   proxy.resetForm("deliveryFormRef");
   deliveryFormVisible.value = false;
-  currentDeliveryRow.value = null;
+  currentDeliveryRows.value = [];
 };
 const currentFactoryName = ref("");
 const getCurrentFactoryName = async () => {
@@ -3019,6 +3700,9 @@
 };
 onMounted(() => {
 	getList();
+	customerList().then((res) => {
+		customerOption.value = res;
+	});
 	userListNoPage().then(res => {
 		userList.value = res.data;
 	})
@@ -3066,171 +3750,5 @@
 	display: flex;
 	justify-content: space-between;
 	margin-bottom: 10px;
-}
-.print-preview-dialog {
-	.el-dialog__body {
-		padding: 0;
-		max-height: 80vh;
-		overflow-y: auto;
-	}
-}
-
-.print-preview-container {
-	.print-preview-header {
-		padding: 15px;
-		border-bottom: 1px solid #e4e7ed;
-		text-align: center;
-		
-		.el-button {
-			margin: 0 10px;
-		}
-	}
-	
-	.print-preview-content {
-		padding: 20px;
-		background-color: #f5f5f5;
-		min-height: 400px;
-	}
-}
-
-.print-page {
-	width: 220mm;
-	height: 90mm;
-	padding: 10mm;
-	margin: 0 auto;
-	background: white;
-	box-shadow: 0 0 10px rgba(0, 0, 0, 0.1);
-	margin-bottom: 10px;
-	box-sizing: border-box;
-}
-
-.delivery-note {
-	width: 100%;
-	height: 100%;
-	font-family: "SimSun", serif;
-	font-size: 10px;
-	line-height: 1.2;
-	display: flex;
-	flex-direction: column;
-}
-
-.header {
-	text-align: center;
-	margin-bottom: 8px;
-	
-	.company-name {
-		font-size: 18px;
-		font-weight: bold;
-		margin-bottom: 4px;
-	}
-	
-	.document-title {
-		font-size: 16px;
-		font-weight: bold;
-	}
-}
-
-.info-section {
-	margin-bottom: 8px;
-	display: flex;
-	justify-content: space-between;
-	align-items: center;
-	
-	.info-row {
-		line-height: 20px;
-		
-		.label {
-			font-weight: bold;
-			width: 60px;
-			font-size: 14px;
-		}
-		
-		.value {
-			margin-right: 20px;
-			min-width: 80px;
-			font-size: 14px;
-		}
-	}
-}
-
-.table-section {
-	margin-bottom: 4px;
-	flex: 1;
-	
-	.product-table {
-		width: 100%;
-		border-collapse: collapse;
-		border: 1px solid #000;
-		
-		th, td {
-			border: 1px solid #000;
-			padding: 6px;
-			text-align: center;
-			font-size: 14px;
-			line-height: 1.4;
-		}
-		
-		th {
-			font-weight: bold;
-		}
-		
-		.total-label {
-			text-align: right;
-			font-weight: bold;
-		}
-		
-		.total-value {
-			font-weight: bold;
-		}
-	}
-}
-
-.footer-section {
-	.footer-row {
-		display: flex;
-		margin-bottom: 3px;
-		line-height: 20px;
-		justify-content: space-between;
-		
-		.footer-item {
-			display: flex;
-			margin-right: 20px;
-			
-			.label {
-				font-weight: bold;
-				width: 80px;
-				font-size: 14px;
-			}
-			
-			.value {
-				min-width: 80px;
-				font-size: 14px;
-			}
-			
-			&.address-item {
-				.address-value {
-					min-width: 200px;
-				}
-			}
-		}
-	}
-}
-
-@media print {
-	.app-container {
-		display: none;
-	}
-	
-	.print-page {
-		box-shadow: none;
-		margin: 0;
-		padding: 10mm;
-		padding-left: 20mm;
-		page-break-inside: avoid;
-		page-break-after: always;
-	}
-	.print-page:last-child {
-		page-break-after: avoid;
-	}
 }
 </style>

--
Gitblit v1.9.3