| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | <template> | 
					
						
							|  |  |  |  |   <div class="buy-details-page"> | 
					
						
							|  |  |  |  |     <!-- 导航栏 --> | 
					
						
							|  |  |  |  |     <nav class="navbar"> | 
					
						
							|  |  |  |  |       <div class="nav-left"> | 
					
						
							|  |  |  |  |         <el-button  | 
					
						
							|  |  |  |  |           type="text"  | 
					
						
							|  |  |  |  |           @click="$router.go(-1)" | 
					
						
							|  |  |  |  |           class="back-btn" | 
					
						
							|  |  |  |  |         > | 
					
						
							|  |  |  |  |           <el-icon><ArrowLeft /></el-icon> | 
					
						
							|  |  |  |  |         </el-button> | 
					
						
							|  |  |  |  |       </div> | 
					
						
							|  |  |  |  |       <div class="nav-center"> | 
					
						
							|  |  |  |  |         <h1 class="nav-title">确认订单</h1> | 
					
						
							|  |  |  |  |       </div> | 
					
						
							|  |  |  |  |       <div class="nav-right"> | 
					
						
							|  |  |  |  |       </div> | 
					
						
							|  |  |  |  |     </nav> | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     <div v-loading="loading" class="page-content"> | 
					
						
							|  |  |  |  |       <!-- 收货地址 --> | 
					
						
							|  |  |  |  |       <div class="address-section"> | 
					
						
							|  |  |  |  |         <div class="address-header"> | 
					
						
							|  |  |  |  |           <el-icon><Location /></el-icon> | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |           <span class="address-label">收货地址</span> | 
					
						
							|  |  |  |  |           <el-button  | 
					
						
							|  |  |  |  |             type="text"  | 
					
						
							|  |  |  |  |             @click="goToAddressManage" | 
					
						
							|  |  |  |  |             class="manage-address-btn" | 
					
						
							|  |  |  |  |           > | 
					
						
							|  |  |  |  |             管理地址 | 
					
						
							|  |  |  |  |           </el-button> | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |         </div> | 
					
						
							|  |  |  |  |         <div class="address-content"> | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |           <el-select  | 
					
						
							|  |  |  |  |             v-model="selectedAddressId" | 
					
						
							|  |  |  |  |             placeholder="请选择收货地址" | 
					
						
							|  |  |  |  |             class="address-select" | 
					
						
							|  |  |  |  |             @change="handleAddressChange" | 
					
						
							|  |  |  |  |           > | 
					
						
							|  |  |  |  |             <el-option | 
					
						
							|  |  |  |  |               v-for="address in addresses" | 
					
						
							|  |  |  |  |               :key="address.id" | 
					
						
							|  |  |  |  |               :label="`${address.recipientName} ${address.recipientPhone} ${address.province}${address.city}${address.district}${address.detailAddress}`" | 
					
						
							|  |  |  |  |               :value="address.id" | 
					
						
							|  |  |  |  |             > | 
					
						
							|  |  |  |  |               <div class="address-option"> | 
					
						
							|  |  |  |  |                 <div class="address-info"> | 
					
						
							|  |  |  |  |                   <span class="recipient-info">{{ address.recipientName }} {{ address.recipientPhone }}</span> | 
					
						
							|  |  |  |  |                   <el-tag v-if="address.isDefault" type="danger" size="small" class="default-tag">默认</el-tag> | 
					
						
							|  |  |  |  |                 </div> | 
					
						
							|  |  |  |  |                 <div class="address-detail">{{ address.province }}{{ address.city }}{{ address.district }}{{ address.detailAddress }}</div> | 
					
						
							|  |  |  |  |               </div> | 
					
						
							|  |  |  |  |             </el-option> | 
					
						
							|  |  |  |  |           </el-select> | 
					
						
							|  |  |  |  |           <div v-if="addresses.length === 0" class="no-address"> | 
					
						
							|  |  |  |  |             <span class="no-address-text">暂无收货地址</span> | 
					
						
							|  |  |  |  |             <el-button type="text" @click="goToAddressManage" class="add-address-btn"> | 
					
						
							|  |  |  |  |               立即添加 | 
					
						
							|  |  |  |  |             </el-button> | 
					
						
							|  |  |  |  |           </div> | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |         </div> | 
					
						
							|  |  |  |  |       </div> | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       <!-- 商品信息 --> | 
					
						
							|  |  |  |  |       <div class="product-section"> | 
					
						
							|  |  |  |  |         <div class="product-info"> | 
					
						
							|  |  |  |  |           <div class="product-image"> | 
					
						
							|  |  |  |  |             <img :src="product?.image || '/imgs/productdetail/商品主图.png'" alt="商品主图" /> | 
					
						
							|  |  |  |  |           </div> | 
					
						
							|  |  |  |  |           <div class="product-details"> | 
					
						
							|  |  |  |  |             <div class="product-price"> | 
					
						
							|  |  |  |  |               <span class="price-label">实付</span> | 
					
						
							|  |  |  |  |               <el-icon class="coin-icon"><Coin /></el-icon> | 
					
						
							|  |  |  |  |               <span class="price-value">{{ totalPrice }}</span> | 
					
						
							|  |  |  |  |             </div> | 
					
						
							|  |  |  |  |             <div class="quantity-selector"> | 
					
						
							|  |  |  |  |               <el-button size="small" @click="decreaseQuantity" :disabled="quantity <= 1">-</el-button> | 
					
						
							|  |  |  |  |               <span class="quantity">{{ quantity }}</span> | 
					
						
							|  |  |  |  |               <el-button size="small" @click="increaseQuantity">+</el-button> | 
					
						
							|  |  |  |  |             </div> | 
					
						
							|  |  |  |  |           </div> | 
					
						
							|  |  |  |  |         </div> | 
					
						
							|  |  |  |  |       </div> | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |       <!-- 动态规格选择 --> | 
					
						
							|  |  |  |  |       <div  | 
					
						
							|  |  |  |  |         v-for="(specOptions, specName) in specGroups"  | 
					
						
							|  |  |  |  |         :key="specName" | 
					
						
							|  |  |  |  |         class="spec-section" | 
					
						
							|  |  |  |  |       > | 
					
						
							|  |  |  |  |         <h3 class="section-title">{{ specName }} ({{ specOptions.length }})</h3> | 
					
						
							|  |  |  |  |         <div class="spec-grid"> | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |           <div  | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |             v-for="option in specOptions"  | 
					
						
							|  |  |  |  |             :key="option.id" | 
					
						
							|  |  |  |  |             class="spec-item" | 
					
						
							|  |  |  |  |             :class="{  | 
					
						
							|  |  |  |  |               active: selectedSpecs[specName]?.id === option.id, | 
					
						
							|  |  |  |  |               disabled: availableSpecs[specName] && !availableSpecs[specName][option.id] | 
					
						
							|  |  |  |  |             }" | 
					
						
							|  |  |  |  |             @click="selectSpec(specName, option)" | 
					
						
							|  |  |  |  |             :disabled="availableSpecs[specName] && !availableSpecs[specName][option.id]" | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |           > | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |             <span class="spec-label">{{ option.name }}</span> | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |           </div> | 
					
						
							|  |  |  |  |         </div> | 
					
						
							|  |  |  |  |       </div> | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |       <!-- 订单备注 --> | 
					
						
							|  |  |  |  |       <div class="note-section"> | 
					
						
							|  |  |  |  |         <h3 class="section-title">订单备注</h3> | 
					
						
							|  |  |  |  |         <div class="note-content" @click="showNoteEdit = true"> | 
					
						
							|  |  |  |  |           <span v-if="!showNoteEdit && !orderNote" class="note-placeholder">无备注</span> | 
					
						
							|  |  |  |  |           <span v-if="!showNoteEdit && orderNote" class="note-text">{{ orderNote }}</span> | 
					
						
							|  |  |  |  |           <el-input  | 
					
						
							|  |  |  |  |             v-if="showNoteEdit" | 
					
						
							|  |  |  |  |             v-model="orderNote" | 
					
						
							|  |  |  |  |             @blur="showNoteEdit = false" | 
					
						
							|  |  |  |  |             @keyup.enter="showNoteEdit = false" | 
					
						
							|  |  |  |  |             placeholder="请输入订单备注" | 
					
						
							|  |  |  |  |             class="note-input" | 
					
						
							|  |  |  |  |             autofocus | 
					
						
							|  |  |  |  |           /> | 
					
						
							|  |  |  |  |         </div> | 
					
						
							|  |  |  |  |       </div> | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     </div> | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     <!-- 底部操作按钮 --> | 
					
						
							|  |  |  |  |     <div class="bottom-actions"> | 
					
						
							|  |  |  |  |       <el-button  | 
					
						
							|  |  |  |  |         size="large" | 
					
						
							|  |  |  |  |         class="cart-button" | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |         @click="handleAddToCart" | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |         :disabled="!canPurchase" | 
					
						
							|  |  |  |  |       > | 
					
						
							|  |  |  |  |         加入购物车 | 
					
						
							|  |  |  |  |       </el-button> | 
					
						
							|  |  |  |  |       <el-button  | 
					
						
							|  |  |  |  |         type="primary"  | 
					
						
							|  |  |  |  |         size="large" | 
					
						
							|  |  |  |  |         class="buy-button" | 
					
						
							|  |  |  |  |         @click="handlePurchase" | 
					
						
							|  |  |  |  |         :disabled="!canPurchase" | 
					
						
							|  |  |  |  |       > | 
					
						
							|  |  |  |  |         立即购买 | 
					
						
							|  |  |  |  |       </el-button> | 
					
						
							|  |  |  |  |     </div> | 
					
						
							|  |  |  |  |   </div> | 
					
						
							|  |  |  |  | </template> | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | <script setup> | 
					
						
							|  |  |  |  | import { ref, computed, onMounted } from 'vue' | 
					
						
							|  |  |  |  | import { useRoute, useRouter } from 'vue-router' | 
					
						
							|  |  |  |  | import { ElMessage } from 'element-plus' | 
					
						
							|  |  |  |  | import {  | 
					
						
							|  |  |  |  |   ArrowLeft, | 
					
						
							|  |  |  |  |   Close, | 
					
						
							|  |  |  |  |   Location, | 
					
						
							|  |  |  |  |   Edit, | 
					
						
							|  |  |  |  |   Coin, | 
					
						
							|  |  |  |  |   ArrowRight | 
					
						
							|  |  |  |  | } from '@element-plus/icons-vue' | 
					
						
							|  |  |  |  | import api from '@/utils/api' | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | const route = useRoute() | 
					
						
							|  |  |  |  | const router = useRouter() | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // 响应式数据
 | 
					
						
							|  |  |  |  | const loading = ref(false) | 
					
						
							|  |  |  |  | const product = ref(null) | 
					
						
							|  |  |  |  | const quantity = ref(1) | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | const specGroups = ref({}) // 动态规格组
 | 
					
						
							|  |  |  |  | const selectedSpecs = ref({}) // 选中的规格值
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  | const addresses = ref([]) | 
					
						
							|  |  |  |  | const selectedAddressId = ref('') | 
					
						
							|  |  |  |  | const selectedAddress = ref(null) | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | const orderNote = ref('') | 
					
						
							|  |  |  |  | const showNoteEdit = ref(false) | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | const availableSpecs = ref({}) // 存储每个规格选项的可选状态
 | 
					
						
							|  |  |  |  | const validCombinations = ref([]) // 存储有效的规格组合键
 | 
					
						
							|  |  |  |  | const specIdToOrder = ref({}) // 规格ID到顺序编号的映射
 | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | // 计算属性
 | 
					
						
							|  |  |  |  | const totalPrice = computed(() => { | 
					
						
							|  |  |  |  |   if (!product.value) return 0 | 
					
						
							|  |  |  |  |   return product.value.points * quantity.value | 
					
						
							|  |  |  |  | }) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | const canPurchase = computed(() => { | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   const specNames = Object.keys(specGroups.value) | 
					
						
							|  |  |  |  |   const allSpecsSelected = specNames.every(specName => selectedSpecs.value[specName]) | 
					
						
							|  |  |  |  |   return allSpecsSelected && quantity.value > 0 | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | }) | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // 方法
 | 
					
						
							|  |  |  |  | const increaseQuantity = () => { | 
					
						
							|  |  |  |  |   if (product.value && quantity.value < product.value.stock) { | 
					
						
							|  |  |  |  |     quantity.value++ | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | const decreaseQuantity = () => { | 
					
						
							|  |  |  |  |   if (quantity.value > 1) { | 
					
						
							|  |  |  |  |     quantity.value-- | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | // 检查规格组合是否有效
 | 
					
						
							|  |  |  |  | const isValidCombination = (testSelection) => { | 
					
						
							|  |  |  |  |   const selectedIds = [] | 
					
						
							|  |  |  |  |   const specNames = Object.keys(specGroups.value) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 按规格名称顺序收集选中的规格ID,转换为顺序编号
 | 
					
						
							|  |  |  |  |   specNames.forEach(specName => { | 
					
						
							|  |  |  |  |     if (testSelection[specName]) { | 
					
						
							|  |  |  |  |       const specId = testSelection[specName].id | 
					
						
							|  |  |  |  |       const orderNumber = specIdToOrder.value[specId] | 
					
						
							|  |  |  |  |       selectedIds.push(orderNumber) | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |       selectedIds.push(null) // 未选择的规格用null占位
 | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   }) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 如果还没有选择完所有规格,检查部分选择是否与任何有效组合兼容
 | 
					
						
							|  |  |  |  |   if (selectedIds.includes(null)) { | 
					
						
							|  |  |  |  |     return validCombinations.value.some(combinationKey => { | 
					
						
							|  |  |  |  |       const keyParts = combinationKey.split('-').map(k => parseInt(k)) | 
					
						
							|  |  |  |  |        | 
					
						
							|  |  |  |  |       // 检查当前部分选择是否与这个combination_key兼容
 | 
					
						
							|  |  |  |  |       return selectedIds.every((selectedOrder, index) => { | 
					
						
							|  |  |  |  |         // 如果该位置未选择,则兼容
 | 
					
						
							|  |  |  |  |         if (selectedOrder === null) return true | 
					
						
							|  |  |  |  |         // 如果该位置已选择,检查是否匹配
 | 
					
						
							|  |  |  |  |         return selectedOrder === keyParts[index] | 
					
						
							|  |  |  |  |       }) | 
					
						
							|  |  |  |  |     }) | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 如果选择了所有规格,检查完整组合是否有效
 | 
					
						
							|  |  |  |  |   const combinationKey = selectedIds.join('-') | 
					
						
							|  |  |  |  |   return validCombinations.value.includes(combinationKey) | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // 更新可选规格状态
 | 
					
						
							|  |  |  |  | const updateAvailableSpecs = () => { | 
					
						
							|  |  |  |  |   const specNames = Object.keys(specGroups.value) | 
					
						
							|  |  |  |  |   const newAvailableSpecs = {} | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   specNames.forEach(specName => { | 
					
						
							|  |  |  |  |     newAvailableSpecs[specName] = {} | 
					
						
							|  |  |  |  |     specGroups.value[specName].forEach(option => { | 
					
						
							|  |  |  |  |       // 检查如果选择这个选项,是否存在有效的组合
 | 
					
						
							|  |  |  |  |       const testSelection = { ...selectedSpecs.value, [specName]: option } | 
					
						
							|  |  |  |  |       newAvailableSpecs[specName][option.id] = isValidCombination(testSelection) | 
					
						
							|  |  |  |  |     }) | 
					
						
							|  |  |  |  |   }) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   availableSpecs.value = newAvailableSpecs | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | // 选择规格
 | 
					
						
							|  |  |  |  | const selectSpec = (specName, option) => { | 
					
						
							|  |  |  |  |   // 检查该选项是否被禁用
 | 
					
						
							|  |  |  |  |   if (availableSpecs.value[specName] && !availableSpecs.value[specName][option.id]) { | 
					
						
							|  |  |  |  |     ElMessage.warning('该规格组合不可选,请选择其他规格') | 
					
						
							|  |  |  |  |     return | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   selectedSpecs.value[specName] = option | 
					
						
							|  |  |  |  |   console.log(`选择${specName}:`, option) | 
					
						
							|  |  |  |  |   console.log('当前选中的所有规格:', selectedSpecs.value) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 更新可选规格状态
 | 
					
						
							|  |  |  |  |   updateAvailableSpecs() | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | const getProductInfo = async () => { | 
					
						
							|  |  |  |  |   try { | 
					
						
							|  |  |  |  |     loading.value = true | 
					
						
							|  |  |  |  |     const productId = route.query.productId | 
					
						
							|  |  |  |  |     if (!productId) { | 
					
						
							|  |  |  |  |       ElMessage.error('商品信息缺失') | 
					
						
							|  |  |  |  |       router.go(-1) | 
					
						
							|  |  |  |  |       return | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |     const response = await api.get(`/products/${productId}`) | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |     const productData = response.data.data.product | 
					
						
							|  |  |  |  |     product.value = productData | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |     // 从商品规格中解析颜色分类和尺寸
 | 
					
						
							|  |  |  |  |     parseSpecifications(productData.specifications || []) | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |   } catch (error) { | 
					
						
							|  |  |  |  |     ElMessage.error('获取商品信息失败') | 
					
						
							|  |  |  |  |     router.go(-1) | 
					
						
							|  |  |  |  |   } finally { | 
					
						
							|  |  |  |  |     loading.value = false | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | // 解析商品规格信息,从spec_details中提取规格
 | 
					
						
							|  |  |  |  | const parseSpecifications = (specifications) => { | 
					
						
							|  |  |  |  |   console.log('原始规格数据:', specifications) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   const tempSpecGroups = {} | 
					
						
							|  |  |  |  |   const validCombinationKeys = [] // 存储有效的combination_key
 | 
					
						
							|  |  |  |  |   const specIdToOrderMap = {} // 规格ID到顺序编号的映射
 | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 遍历每个规格组合,提取combination_key
 | 
					
						
							|  |  |  |  |   specifications.forEach(spec => { | 
					
						
							|  |  |  |  |     if (spec.combination_key) { | 
					
						
							|  |  |  |  |       validCombinationKeys.push(spec.combination_key) | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |     // 遍历每个规格组合中的spec_details
 | 
					
						
							|  |  |  |  |     spec.spec_details.forEach(detail => { | 
					
						
							|  |  |  |  |       const specName = detail.spec_name | 
					
						
							|  |  |  |  |       const specValue = detail.value | 
					
						
							|  |  |  |  |        | 
					
						
							|  |  |  |  |       if (!tempSpecGroups[specName]) { | 
					
						
							|  |  |  |  |         tempSpecGroups[specName] = new Set() | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |        | 
					
						
							|  |  |  |  |       // 使用Set避免重复值
 | 
					
						
							|  |  |  |  |       tempSpecGroups[specName].add(JSON.stringify({ | 
					
						
							|  |  |  |  |         id: detail.id, | 
					
						
							|  |  |  |  |         name: specValue, | 
					
						
							|  |  |  |  |         label: specValue, | 
					
						
							|  |  |  |  |         description: specValue, | 
					
						
							|  |  |  |  |         spec_name_id: detail.spec_name_id, | 
					
						
							|  |  |  |  |         sort_order: detail.sort_order, | 
					
						
							|  |  |  |  |       })) | 
					
						
							|  |  |  |  |     }) | 
					
						
							|  |  |  |  |   }) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 转换Set为数组并解析JSON
 | 
					
						
							|  |  |  |  |   const finalSpecGroups = {} | 
					
						
							|  |  |  |  |   let orderCounter = 1 | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   Object.keys(tempSpecGroups).forEach(specName => { | 
					
						
							|  |  |  |  |     finalSpecGroups[specName] = Array.from(tempSpecGroups[specName]).map(item => JSON.parse(item)) | 
					
						
							|  |  |  |  |     // 按sort_order排序
 | 
					
						
							|  |  |  |  |     finalSpecGroups[specName].sort((a, b) => a.sort_order - b.sort_order) | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |     // 为每个规格选项分配顺序编号(从1开始)
 | 
					
						
							|  |  |  |  |     finalSpecGroups[specName].forEach(option => { | 
					
						
							|  |  |  |  |       specIdToOrderMap[option.id] = orderCounter++ | 
					
						
							|  |  |  |  |     }) | 
					
						
							|  |  |  |  |   }) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   specGroups.value = finalSpecGroups | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 存储有效的combination_key和ID映射,用于验证
 | 
					
						
							|  |  |  |  |   validCombinations.value = validCombinationKeys | 
					
						
							|  |  |  |  |   specIdToOrder.value = specIdToOrderMap | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   console.log('有效的规格组合键:', validCombinationKeys) | 
					
						
							|  |  |  |  |   console.log('规格ID到顺序编号映射:', specIdToOrderMap) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 初始化可选规格状态
 | 
					
						
							|  |  |  |  |   updateAvailableSpecs() | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 输出解析后的规格信息
 | 
					
						
							|  |  |  |  |   console.log('解析后的规格分组:', finalSpecGroups) | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | // 根据选中的规格组合找到对应的规格规则ID
 | 
					
						
							|  |  |  |  | const getSelectedSpecificationId = () => { | 
					
						
							|  |  |  |  |   const specNames = Object.keys(specGroups.value) | 
					
						
							|  |  |  |  |   const selectedIds = [] | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |    | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   // 按规格名称顺序收集选中的规格ID,转换为顺序编号
 | 
					
						
							|  |  |  |  |   specNames.forEach(specName => { | 
					
						
							|  |  |  |  |     if (selectedSpecs.value[specName]) { | 
					
						
							|  |  |  |  |       const specId = selectedSpecs.value[specName].id | 
					
						
							|  |  |  |  |       const orderNumber = specIdToOrder.value[specId] | 
					
						
							|  |  |  |  |       selectedIds.push(orderNumber) | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   }) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 生成combination_key
 | 
					
						
							|  |  |  |  |   const combinationKey = selectedIds.join('-') | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 在specifications数组中找到对应的规格规则
 | 
					
						
							|  |  |  |  |   const specification = product.value.specifications?.find(spec =>  | 
					
						
							|  |  |  |  |     spec.combination_key === combinationKey | 
					
						
							|  |  |  |  |   ) | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   return specification ? specification.id : null | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  | // 立即购买功能
 | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | const handlePurchase = async () => { | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   // 检查是否选择了所有必需的规格
 | 
					
						
							|  |  |  |  |   const specNames = Object.keys(specGroups.value) | 
					
						
							|  |  |  |  |   for (const specName of specNames) { | 
					
						
							|  |  |  |  |     if (!selectedSpecs.value[specName]) { | 
					
						
							|  |  |  |  |       ElMessage.warning(`请选择${specName}`) | 
					
						
							|  |  |  |  |       return | 
					
						
							|  |  |  |  |     } | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |   if (!selectedAddress.value) { | 
					
						
							|  |  |  |  |     ElMessage.error('请选择收货地址') | 
					
						
							|  |  |  |  |     return | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   // 获取选中规格对应的规格规则ID
 | 
					
						
							|  |  |  |  |   const specificationId = getSelectedSpecificationId() | 
					
						
							|  |  |  |  |   if (!specificationId) { | 
					
						
							|  |  |  |  |     ElMessage.error('所选规格组合无效,请重新选择') | 
					
						
							|  |  |  |  |     return | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |   try { | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |     // 创建单独的购买订单
 | 
					
						
							|  |  |  |  |     const orderData = { | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |       productId: product.value.id,           // 商品ID
 | 
					
						
							|  |  |  |  |       quantity: quantity.value,              // 购买数量
 | 
					
						
							|  |  |  |  |       specificationId: specificationId,      // 规格规则ID
 | 
					
						
							|  |  |  |  |       points: product.value.points,          // 商品积分价格
 | 
					
						
							|  |  |  |  |       name: product.value.name,              // 商品名称
 | 
					
						
							|  |  |  |  |       image: product.value.image,            // 商品图片
 | 
					
						
							|  |  |  |  |       stock: product.value.stock,            // 商品库存
 | 
					
						
							|  |  |  |  |       addressId: selectedAddress.value.id,   // 收货地址ID
 | 
					
						
							|  |  |  |  |       orderNote: orderNote.value             // 订单备注
 | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |     const response = await api.post('/cart/buy-now', orderData)//立即购买
 | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |      | 
					
						
							|  |  |  |  |     if (response.data.success) { | 
					
						
							|  |  |  |  |       const cartId = response.data.data.cartId | 
					
						
							|  |  |  |  |        | 
					
						
							|  |  |  |  |       // 跳转到支付页面
 | 
					
						
							|  |  |  |  |       router.push({ | 
					
						
							|  |  |  |  |         path: '/pay', | 
					
						
							|  |  |  |  |         query: { | 
					
						
							|  |  |  |  |           cartId: cartId | 
					
						
							|  |  |  |  |         } | 
					
						
							|  |  |  |  |       }) | 
					
						
							|  |  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |       throw new Error(response.data.message || '创建订单失败') | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } catch (error) { | 
					
						
							|  |  |  |  |     ElMessage.error(error.message || '操作失败,请重试') | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  | // 添加到购物车功能(新增)
 | 
					
						
							|  |  |  |  | const handleAddToCart = async () => { | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   // 检查是否选择了所有必需的规格
 | 
					
						
							|  |  |  |  |   const specNames = Object.keys(specGroups.value) | 
					
						
							|  |  |  |  |   for (const specName of specNames) { | 
					
						
							|  |  |  |  |     if (!selectedSpecs.value[specName]) { | 
					
						
							|  |  |  |  |       ElMessage.warning(`请选择${specName}`) | 
					
						
							|  |  |  |  |       return | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   // 获取选中规格对应的规格规则ID
 | 
					
						
							|  |  |  |  |   const specificationId = getSelectedSpecificationId() | 
					
						
							|  |  |  |  |   if (!specificationId) { | 
					
						
							|  |  |  |  |     ElMessage.error('所选规格组合无效,请重新选择') | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |     return | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							|  |  |  |  |   try { | 
					
						
							|  |  |  |  |     const cartItem = { | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |       productId: product.value.id,    // 商品ID
 | 
					
						
							|  |  |  |  |       quantity: quantity.value,       // 购买数量
 | 
					
						
							|  |  |  |  |       specificationId: specificationId, // 规格规则ID
 | 
					
						
							|  |  |  |  |       points: product.value.points,   // 商品积分价格
 | 
					
						
							|  |  |  |  |       name: product.value.name,       // 商品名称
 | 
					
						
							|  |  |  |  |       image: product.value.image,     // 商品图片
 | 
					
						
							|  |  |  |  |       stock: product.value.stock      // 商品库存
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  |     const response = await api.post('/cart/add', cartItem) | 
					
						
							|  |  |  |  |      | 
					
						
							|  |  |  |  |     if (response.data.success) { | 
					
						
							|  |  |  |  |       ElMessage.success('商品已加入购物车!') | 
					
						
							|  |  |  |  |       // 可以选择返回上一页或跳转到购物车页面
 | 
					
						
							|  |  |  |  |       router.go(-1) | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |       throw new Error(response.data.message || '添加到购物车失败') | 
					
						
							|  |  |  |  |     } | 
					
						
							|  |  |  |  |   } catch (error) { | 
					
						
							|  |  |  |  |     ElMessage.error(error.message || '添加到购物车失败,请重试') | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // 获取用户地址列表
 | 
					
						
							|  |  |  |  | const getAddressList = async () => { | 
					
						
							|  |  |  |  |   try { | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |     const response = await api.get('/addresses') | 
					
						
							|  |  |  |  |     console.log('获取地址列表响应:', response) | 
					
						
							|  |  |  |  |     if (response.data.success) { | 
					
						
							|  |  |  |  |       // 根据接口文档转换数据格式,与Address.vue保持一致
 | 
					
						
							|  |  |  |  |       const addressList = response.data.data || [] | 
					
						
							|  |  |  |  |       addresses.value = addressList.map(addr => ({ | 
					
						
							|  |  |  |  |         id: addr.id, | 
					
						
							|  |  |  |  |         recipientName: addr.receiver_name, | 
					
						
							|  |  |  |  |         recipientPhone: addr.receiver_phone, | 
					
						
							|  |  |  |  |         province: addr.province_name, | 
					
						
							|  |  |  |  |         city: addr.city_name, | 
					
						
							|  |  |  |  |         district: addr.district_name, | 
					
						
							|  |  |  |  |         detailAddress: addr.detailed_address, | 
					
						
							|  |  |  |  |         isDefault: addr.is_default, | 
					
						
							|  |  |  |  |         labelName: addr.label_name, | 
					
						
							|  |  |  |  |         labelColor: addr.label_color | 
					
						
							|  |  |  |  |       })) | 
					
						
							|  |  |  |  |        | 
					
						
							|  |  |  |  |       // 如果有默认地址,自动选中
 | 
					
						
							|  |  |  |  |       const defaultAddress = addresses.value.find(addr => addr.isDefault) | 
					
						
							|  |  |  |  |       if (defaultAddress) { | 
					
						
							|  |  |  |  |         selectedAddressId.value = defaultAddress.id | 
					
						
							|  |  |  |  |         selectedAddress.value = defaultAddress | 
					
						
							|  |  |  |  |       } | 
					
						
							|  |  |  |  |     } else { | 
					
						
							|  |  |  |  |       throw new Error(response.data.message || '获取地址列表失败') | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |     } | 
					
						
							|  |  |  |  |   } catch (error) { | 
					
						
							|  |  |  |  |     console.error('获取地址列表失败:', error) | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |     ElMessage.error(error.message || '获取地址列表失败') | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |   } | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // 处理地址选择变化
 | 
					
						
							|  |  |  |  | const handleAddressChange = (addressId) => { | 
					
						
							|  |  |  |  |   selectedAddress.value = addresses.value.find(addr => addr.id === addressId) | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | // 跳转到地址管理页面
 | 
					
						
							|  |  |  |  | const goToAddressManage = () => { | 
					
						
							|  |  |  |  |   router.push('/address') | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | // 生命周期
 | 
					
						
							|  |  |  |  | onMounted(() => { | 
					
						
							|  |  |  |  |   // 从URL参数获取初始数量
 | 
					
						
							|  |  |  |  |   const initialQuantity = route.query.quantity | 
					
						
							|  |  |  |  |   if (initialQuantity && !isNaN(initialQuantity)) { | 
					
						
							|  |  |  |  |     quantity.value = parseInt(initialQuantity) | 
					
						
							|  |  |  |  |   } | 
					
						
							|  |  |  |  |    | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   getProductInfo() // 商品信息中已包含规格信息,无需单独获取颜色分类和尺寸
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  |   getAddressList() | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | }) | 
					
						
							|  |  |  |  | </script> | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | <style scoped> | 
					
						
							|  |  |  |  | .buy-details-page { | 
					
						
							|  |  |  |  |   min-height: 100vh; | 
					
						
							|  |  |  |  |   background: #f5f5f5; | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   flex-direction: column; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .navbar { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   justify-content: space-between; | 
					
						
							|  |  |  |  |   padding: 12px 16px; | 
					
						
							|  |  |  |  |   background: white; | 
					
						
							|  |  |  |  |   border-bottom: 1px solid #eee; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .nav-title { | 
					
						
							|  |  |  |  |   font-size: 18px; | 
					
						
							|  |  |  |  |   font-weight: 500; | 
					
						
							|  |  |  |  |   margin: 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .back-btn { | 
					
						
							|  |  |  |  |   color: #333; | 
					
						
							|  |  |  |  |   padding: 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .page-content { | 
					
						
							|  |  |  |  |   flex: 1; | 
					
						
							|  |  |  |  |   padding: 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .address-section { | 
					
						
							|  |  |  |  |   background: white; | 
					
						
							|  |  |  |  |   padding: 16px; | 
					
						
							|  |  |  |  |   margin-bottom: 8px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .address-header { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   gap: 8px; | 
					
						
							|  |  |  |  |   margin-bottom: 8px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .address-label { | 
					
						
							|  |  |  |  |   font-weight: 500; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .edit-icon { | 
					
						
							|  |  |  |  |   margin-left: auto; | 
					
						
							|  |  |  |  |   color: #666; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .product-section { | 
					
						
							|  |  |  |  |   background: white; | 
					
						
							|  |  |  |  |   padding: 16px; | 
					
						
							|  |  |  |  |   margin-bottom: 8px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .product-info { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   gap: 12px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .product-image { | 
					
						
							|  |  |  |  |   width: 60px; | 
					
						
							|  |  |  |  |   height: 60px; | 
					
						
							|  |  |  |  |   border-radius: 8px; | 
					
						
							|  |  |  |  |   overflow: hidden; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .product-image img { | 
					
						
							|  |  |  |  |   width: 100%; | 
					
						
							|  |  |  |  |   height: 100%; | 
					
						
							|  |  |  |  |   object-fit: cover; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .product-details { | 
					
						
							|  |  |  |  |   flex: 1; | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   justify-content: space-between; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .product-price { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   gap: 4px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .price-label { | 
					
						
							|  |  |  |  |   font-size: 14px; | 
					
						
							|  |  |  |  |   color: #666; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .coin-icon { | 
					
						
							|  |  |  |  |   color: #ffae00; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .price-value { | 
					
						
							|  |  |  |  |   font-size: 18px; | 
					
						
							|  |  |  |  |   font-weight: bold; | 
					
						
							|  |  |  |  |   color: #ffae00; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .quantity-selector { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   gap: 12px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .quantity { | 
					
						
							|  |  |  |  |   font-size: 16px; | 
					
						
							|  |  |  |  |   min-width: 20px; | 
					
						
							|  |  |  |  |   text-align: center; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | .spec-section, | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | .note-section, | 
					
						
							|  |  |  |  | .payment-section { | 
					
						
							|  |  |  |  |   background: white; | 
					
						
							|  |  |  |  |   padding: 16px; | 
					
						
							|  |  |  |  |   margin-bottom: 8px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .section-title { | 
					
						
							|  |  |  |  |   font-size: 16px; | 
					
						
							|  |  |  |  |   font-weight: 500; | 
					
						
							|  |  |  |  |   margin: 0 0 12px 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | .spec-grid { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   flex-wrap: wrap; | 
					
						
							|  |  |  |  |   gap: 10px; | 
					
						
							|  |  |  |  |   margin-top: 10px; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | .spec-item { | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |   display: flex; | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   justify-content: center; | 
					
						
							|  |  |  |  |   padding: 10px 15px; | 
					
						
							|  |  |  |  |   border: 1px solid #e0e0e0; | 
					
						
							|  |  |  |  |   border-radius: 6px; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |   cursor: pointer; | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  |   transition: all 0.3s ease; | 
					
						
							|  |  |  |  |   min-width: 60px; | 
					
						
							|  |  |  |  |   text-align: center; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | .spec-item:hover { | 
					
						
							|  |  |  |  |   border-color: #ff6b35; | 
					
						
							|  |  |  |  |   background-color: #fff5f2; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | .spec-item.active { | 
					
						
							|  |  |  |  |   border-color: #ff6b35; | 
					
						
							|  |  |  |  |   background-color: #ff6b35; | 
					
						
							|  |  |  |  |   color: white; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | .spec-item.disabled { | 
					
						
							|  |  |  |  |   background-color: #f5f5f5; | 
					
						
							|  |  |  |  |   border-color: #e0e0e0; | 
					
						
							|  |  |  |  |   color: #ccc; | 
					
						
							|  |  |  |  |   cursor: not-allowed; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | .spec-item.disabled:hover { | 
					
						
							|  |  |  |  |   background-color: #f5f5f5; | 
					
						
							|  |  |  |  |   border-color: #e0e0e0; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-28 19:48:40 +08:00
										 |  |  |  | .spec-label { | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  |   font-size: 14px; | 
					
						
							|  |  |  |  |   font-weight: 500; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .note-content { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   justify-content: space-between; | 
					
						
							|  |  |  |  |   padding: 12px 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .note-placeholder { | 
					
						
							|  |  |  |  |   color: #999; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  | .address-select { | 
					
						
							|  |  |  |  |   width: 100%; | 
					
						
							|  |  |  |  |   margin-top: 10px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .manage-address-btn { | 
					
						
							|  |  |  |  |   color: #409eff; | 
					
						
							|  |  |  |  |   font-size: 14px; | 
					
						
							|  |  |  |  |   margin-left: auto; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .manage-address-btn:hover { | 
					
						
							|  |  |  |  |   color: #66b1ff; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .address-option { | 
					
						
							|  |  |  |  |   padding: 8px 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .address-info { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   justify-content: space-between; | 
					
						
							|  |  |  |  |   margin-bottom: 4px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .recipient-info { | 
					
						
							|  |  |  |  |   font-weight: 500; | 
					
						
							|  |  |  |  |   color: #303133; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .default-tag { | 
					
						
							|  |  |  |  |   margin-left: 8px; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  | .address-detail { | 
					
						
							|  |  |  |  |   color: #606266; | 
					
						
							|  |  |  |  |   font-size: 12px; | 
					
						
							|  |  |  |  |   line-height: 1.4; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .no-address { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   justify-content: center; | 
					
						
							|  |  |  |  |   padding: 20px; | 
					
						
							|  |  |  |  |   color: #909399; | 
					
						
							|  |  |  |  |   font-size: 14px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .no-address-text { | 
					
						
							|  |  |  |  |   margin-right: 8px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .add-address-btn { | 
					
						
							|  |  |  |  |   color: #409eff; | 
					
						
							|  |  |  |  |   font-size: 14px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .add-address-btn:hover { | 
					
						
							|  |  |  |  |   color: #66b1ff; | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .note-input { | 
					
						
							|  |  |  |  |   flex: 1; | 
					
						
							|  |  |  |  |   margin-right: 8px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .note-text { | 
					
						
							|  |  |  |  |   color: #333; | 
					
						
							|  |  |  |  |   flex: 1; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-27 10:16:48 +08:00
										 |  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2025-08-26 11:36:01 +08:00
										 |  |  |  | 
 | 
					
						
							|  |  |  |  | .note-content { | 
					
						
							|  |  |  |  |   cursor: pointer; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .payment-options { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   flex-direction: column; | 
					
						
							|  |  |  |  |   gap: 12px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .payment-option { | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   align-items: center; | 
					
						
							|  |  |  |  |   padding: 8px 0; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .bottom-actions { | 
					
						
							|  |  |  |  |   padding: 16px; | 
					
						
							|  |  |  |  |   background: white; | 
					
						
							|  |  |  |  |   border-top: 1px solid #eee; | 
					
						
							|  |  |  |  |   display: flex; | 
					
						
							|  |  |  |  |   gap: 12px; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .cart-button { | 
					
						
							|  |  |  |  |   flex: 1; | 
					
						
							|  |  |  |  |   height: 48px; | 
					
						
							|  |  |  |  |   background: white; | 
					
						
							|  |  |  |  |   border: 1px solid #ffae00; | 
					
						
							|  |  |  |  |   color: #ffae00; | 
					
						
							|  |  |  |  |   border-radius: 24px; | 
					
						
							|  |  |  |  |   font-size: 16px; | 
					
						
							|  |  |  |  |   font-weight: 500; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .cart-button:hover { | 
					
						
							|  |  |  |  |   background: #fff7e6; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .cart-button:disabled { | 
					
						
							|  |  |  |  |   background: #f5f5f5; | 
					
						
							|  |  |  |  |   border-color: #ccc; | 
					
						
							|  |  |  |  |   color: #ccc; | 
					
						
							|  |  |  |  |   cursor: not-allowed; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .buy-button { | 
					
						
							|  |  |  |  |   flex: 1; | 
					
						
							|  |  |  |  |   height: 48px; | 
					
						
							|  |  |  |  |   background: #ffae00; | 
					
						
							|  |  |  |  |   border: none; | 
					
						
							|  |  |  |  |   border-radius: 24px; | 
					
						
							|  |  |  |  |   font-size: 16px; | 
					
						
							|  |  |  |  |   font-weight: 500; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .buy-button:hover { | 
					
						
							|  |  |  |  |   background: #e69900; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | 
 | 
					
						
							|  |  |  |  | .buy-button:disabled { | 
					
						
							|  |  |  |  |   background: #ccc; | 
					
						
							|  |  |  |  |   cursor: not-allowed; | 
					
						
							|  |  |  |  | } | 
					
						
							|  |  |  |  | </style> |