| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066 |
- <template>
- <view class="order-container">
- <!-- 用户信息卡片 -->
- <view class="info-card">
- <view class="address-item">
- <view class="img-status-text">
- <image src="/static/img/icon-send.png" mode="" class="img"></image>
- </view>
- <view class="user-info">
- <AddressInfo v-if="addressSend.addressId" :address="addressSend" />
- <view v-else class="create-btn" @click="handleAddAddress('sender')">新建寄件人</view>
- </view>
- <view class="img-status-text right" @click="openAddressBook('sender')">
- <image src="/static/img/create-order-address.png" mode="" class="address-image"></image>
- <view class="status-text">地址薄</view>
- </view>
- </view>
- <view class="address-item">
- <view class="img-status-text">
- <image src="/static/img/create-order-change.png" mode="" class="img-change"></image>
- </view>
- <view class="user-info">
- <view class="line"></view>
- </view>
- </view>
- <view class="address-item">
- <view class="img-status-text">
- <image src="/static/img/icon-receive.png" mode="" class="img"></image>
- </view>
- <view class="user-info">
- <AddressInfo v-if="addressReceive.addressId" :address="addressReceive" />
- <view v-else class="create-btn" @click="handleAddAddress('receiver')">新建收件人</view>
- </view>
- <view class="img-status-text right" @click="openAddressBook('receiver')">
- <image src="/static/img/create-order-address.png" mode="" class="address-image"></image>
- <view class="status-text">地址薄</view>
- </view>
- </view>
- </view>
- <view class="pickup-title">上门取件</view>
- <!-- 物品信息 -->
- <view class="goods-card">
- <view class="goods-item">
- <view class="item-label required">期望上门时间</view>
- <view class="time-value" @click="handleTimeClick">
- <text class="value">{{ selectedTimeLabel }}</text>
- <u-icon class="arrow" name='arrow-right' size="18"></u-icon>
- </view>
- </view>
- <view class="goods-item">
- <view class="item-label required">快递类型</view>
- <view class="item-control">
- <picker :range="expressTypes" :range-key="'label'" @change="onExpressTypeChange"
- :value="expressTypeIndex" :disabled="!expressTypes.length">
- <view class="picker-text">
- {{ expressTypes[expressTypeIndex]?.label || (expressTypes.length ? '请选择快递类型' : '加载中...') }}
- </view>
- </picker>
- <u-icon class="arrow" name='arrow-right' size="18"></u-icon>
- </view>
- </view>
- <view class="goods-item">
- <view class="item-label required">物品信息</view>
- <view class="item-control">
- <input class="input-field" placeholder="请输入物品信息" placeholder-class="placeholder" v-model="goodsInfo"
- maxlength="50" @input="onGoodsInfoInput" />
- </view>
- </view>
- <!-- 总体重 -->
- <view class="goods-item">
- <view class="item-label">总体重(KG)</view>
- <view class="item-control btn">
- <view class="control-btn minus" @click="decreaseWeight">-</view>
- <view class="control-value">
- <input class="input-field" placeholder="" placeholder-class="placeholder" :value="weight"
- @input="handleWeightInput" type="digit" />
- </view>
- <view class="control-btn plus" @click="increaseWeight">+</view>
- </view>
- </view>
- <!-- 总体积 -->
- <view class="goods-item">
- <view class="item-label">总体积(m³)</view>
- <view class="item-control btn">
- <view class="control-btn minus" @click="decreaseVolume">-</view>
- <view class="control-value">
- <input class="input-field" placeholder="" placeholder-class="placeholder" :value="volume"
- @input="handleVolumeInput" type="digit" />
- </view>
- <view class="control-btn plus" @click="increaseVolume">+</view>
- </view>
- </view>
- <!-- 件数 -->
- <view class="goods-item">
- <view class="item-label">件数(件)</view>
- <view class="item-control btn">
- <view class="control-btn minus" @click="decreaseQuantity">-</view>
- <view class="control-value">
- <input class="input-field" placeholder="" placeholder-class="placeholder" :value="quantity"
- @input="handleQuantityInput" type="number" />
- </view>
- <view class="control-btn plus" @click="increaseQuantity">+</view>
- </view>
- </view>
- </view>
- <view class="pickup-title">增值服务</view>
- <!-- 增值服务卡片 -->
- <view class="goods-card">
- <view class="goods-item">
- <view class="item-label">包装服务</view>
- <view class="item-control">
- <switch color="#007AFF" :checked="valueServices.isPack" @change="onPackagingChange" />
- </view>
- </view>
- <view class="goods-item">
- <view class="item-label">保价</view>
- <view class="item-control">
- <switch color="#007AFF" :checked="!!insuranceAmount" @change="onInsuranceChange" />
- </view>
- </view>
- <view v-if="insuranceAmount !== '' && insuranceAmount !== null" class="goods-item">
- <view class="item-label">保价金额(元)</view>
- <view class="item-control">
- <input class="input-field" placeholder="请输入保价金额" placeholder-class="placeholder"
- v-model="insuranceAmount" type="digit" maxlength="10" @input="validateInsuranceAmount" />
- </view>
- </view>
- <!-- 超长超重(顺丰/京东) -->
- <view class="goods-item">
- <view class="item-label">超长超重</view>
- <view class="item-control">
- <switch color="#007AFF" :checked="valueServices.isOverLongWeight" @change="onOverweightChange" />
- </view>
- </view>
- <!-- 签单返还 -->
- <view class="goods-item">
- <view class="item-label">签单返还</view>
- <view class="item-control">
- <switch color="#007AFF" :checked="valueServices.isReceiptCollect" @change="onSignReturnChange" />
- </view>
- </view>
- <!-- 顺丰:签单返还类型选择 -->
- <view v-if="product === '2' && valueServices.isReceiptCollect" class="goods-item">
- <view class="item-label">返还类型</view>
- <view class="item-control">
- <picker :range="receiptReturnTypes" :range-key="'label'" @change="onReceiptReturnTypeChange"
- :value="receiptReturnTypeIndex">
- <view class="picker-text">{{ receiptReturnTypes[receiptReturnTypeIndex]?.label || '请选择返还类型' }}
- </view>
- </picker>
- <u-icon class="arrow" name='arrow-right' size="18"></u-icon>
- </view>
- </view>
- <!-- 打木架(仅顺丰) -->
- <view v-if="product === '2'" class="goods-item">
- <view class="item-label">打木架</view>
- <view class="item-control">
- <switch color="#007AFF" :checked="valueServices.isWoodenCrate" @change="onWoodenFrameChange" />
- </view>
- </view>
- </view>
- <!-- 协议同意 -->
- <view class="agreement-card">
- <view class="agreement-item">
- <checkbox color="#1B64F0" :checked="agreed" @click="toggleAgreement" style="transform:scale(0.8)" />
- <text class="agreement-text">我已理解并同意遵守《快件服务协议》</text>
- </view>
- <view class="price-notice">
- 实际费用以快递员核实为准,四舍五入取整
- </view>
- </view>
- <!-- 下单按钮 -->
- <view class="submit-btn" :class="{ disabled: !agreed }" :disabled="!agreed" @click="submitOrder">
- 提交订单
- </view>
- <!-- 安全区域占位 -->
- <view class="safe-area"></view>
- <!-- 时间选择弹窗 -->
- <TimePopup :visible="showTimePicker" @close="showTimePicker = false" @confirm="handleTimeConfirm" />
- </view>
- </template>
- <script setup>
- import {
- ref,
- computed,
- onMounted,
- onUnmounted,
- watch,
- } from 'vue'
- import {
- onLoad
- } from '@dcloudio/uni-app'
- import TimePopup from './components/TimePopup.vue'
- import AddressInfo from '@/components/AddressInfo.vue'
- import {
- createOrder,
- dictList
- } from '../../api/order'
- import {
- getDefaultAddress
- } from '../../api/address'
- // ==================== 数据存储 ====================
- const getAddressFromStorage = (key) => {
- const address = uni.getStorageSync(key)
- return address || {
- id: '',
- name: '',
- phone: '',
- address: '',
- provinceName: '',
- cityName: '',
- countyName: '',
- isDefault: false
- }
- }
- const saveAddressToStorage = (key, address) => {
- uni.setStorageSync(key, address)
- }
- // ==================== 页面参数 ====================
- const product = ref('1') // 1-京东,2-顺丰
- // ==================== 地址信息 ====================
- const addressSend = ref(getAddressFromStorage('senderAddress'))
- const addressReceive = ref(getAddressFromStorage('receiverAddress'))
- // ==================== 物品信息 ====================
- const goodsInfo = ref('')
- const weight = ref(1)
- const volume = ref(0.001)
- const quantity = ref(1)
- // ==================== 时间选择 ====================
- const showTimePicker = ref(false)
- const selectedTimeData = ref(null)
- const selectedTimeLabel = computed(() => {
- if (!selectedTimeData.value) return '请选择时间'
- const {
- dateLabel,
- timeLabel,
- isToday,
- isImmediate
- } = selectedTimeData.value
- if (isToday && isImmediate) return '一小时内'
- return `${dateLabel} ${timeLabel}`
- })
- // ==================== 快递类型字典(动态获取)====================
- const jdDictList = ref([]) // 京东快递类型字典
- const sfDictList = ref([]) // 顺丰快递类型字典
- const expressTypeIndex = ref(0)
- const expressTypes = ref([]) // 当前显示的快递类型列表(已转换为 {label, value} 格式)
- // 从后端获取字典数据
- const fetchDictData = async () => {
- try {
- uni.showLoading({
- title: '加载字典...'
- })
- // 并发请求两个字典
- const [jdRes, sfRes] = await Promise.all([
- dictList('jd_logistics_product_code'),
- dictList('sf_logistics_product_code')
- ])
- if (jdRes.code === 200) {
- jdDictList.value = jdRes.data || []
- }
- if (sfRes.code === 200) {
- sfDictList.value = sfRes.data || []
- }
- updateExpressTypes()
- } catch (error) {
- console.error('获取快递类型字典失败', error)
- uni.showToast({
- title: '字典加载失败',
- icon: 'none'
- })
- } finally {
- uni.hideLoading()
- }
- }
- // 根据 product 更新 expressTypes
- const updateExpressTypes = () => {
- let sourceList = []
- if (product.value === '1') {
- sourceList = jdDictList.value
- } else if (product.value === '2') {
- sourceList = sfDictList.value
- }
- // 转换为 picker 需要的格式:{ label: dictLabel, value: dictValue }
- expressTypes.value = sourceList.map(item => ({
- label: item.dictLabel,
- value: item.dictValue
- }))
- // 重置选中索引
- expressTypeIndex.value = 0
- }
- // ==================== 增值服务 ====================
- const valueServices = ref({
- isPack: false,
- isOverLongWeight: false,
- isReceiptCollect: false,
- isWoodenCrate: false
- })
- // 保价金额
- const insuranceAmount = ref('')
- const insuranceAmountError = ref('')
- // 签单返还类型(顺丰)
- const receiptReturnTypeIndex = ref(0)
- const receiptReturnTypes = ref([{
- label: '电子凭证',
- value: 'electronic'
- },
- {
- label: '纸质凭证',
- value: 'paper'
- }
- ])
- // ==================== 协议 ====================
- const agreed = ref(false)
- // ==================== 表单操作 ====================
- // 重量
- const increaseWeight = () => {
- weight.value = parseFloat((weight.value + 0.5).toFixed(3))
- if (weight.value > 100) weight.value = 100
- }
- const decreaseWeight = () => {
- weight.value = parseFloat((weight.value - 0.5).toFixed(3))
- if (weight.value < 0.5) weight.value = 0.5
- }
- const handleWeightInput = (e) => {
- let value = e.detail.value.replace(/[^\d.]/g, '')
- const parts = value.split('.')
- if (parts.length > 2) value = parts[0] + '.' + parts.slice(1).join('')
- if (value.includes('.')) {
- const decimal = value.split('.')[1]
- if (decimal && decimal.length > 3) value = parseFloat(value).toFixed(3)
- }
- const num = parseFloat(value) || 0.5
- weight.value = Math.max(0.5, Math.min(100, num))
- }
- // 体积
- const increaseVolume = () => {
- volume.value = parseFloat((volume.value + 0.001).toFixed(3))
- if (volume.value > 10) volume.value = 10
- }
- const decreaseVolume = () => {
- volume.value = parseFloat((volume.value - 0.001).toFixed(3))
- if (volume.value < 0.001) volume.value = 0.001
- }
- const handleVolumeInput = (e) => {
- let value = e.detail.value.replace(/[^\d.]/g, '')
- const parts = value.split('.')
- if (parts.length > 2) value = parts[0] + '.' + parts.slice(1).join('')
- if (value.includes('.')) {
- const decimal = value.split('.')[1]
- if (decimal && decimal.length > 3) value = parseFloat(value).toFixed(3)
- }
- const num = parseFloat(value) || 0.001
- volume.value = Math.max(0.001, Math.min(10, num))
- }
- // 件数
- const increaseQuantity = () => {
- quantity.value += 1
- if (quantity.value > 99) quantity.value = 99
- }
- const decreaseQuantity = () => {
- quantity.value -= 1
- if (quantity.value < 1) quantity.value = 1
- }
- const handleQuantityInput = (e) => {
- let value = e.detail.value.replace(/\D/g, '')
- const int = parseInt(value) || 1
- quantity.value = Math.max(1, Math.min(99, int))
- }
- // 物品信息
- const onGoodsInfoInput = (e) => {
- goodsInfo.value = e.detail.value
- if (goodsInfo.value.length > 50) goodsInfo.value = goodsInfo.value.substring(0, 50)
- }
- // 快递类型
- const onExpressTypeChange = (e) => {
- expressTypeIndex.value = e.detail.value
- }
- // 增值服务开关
- const onPackagingChange = (e) => {
- valueServices.value.isPack = e.detail.value
- }
- const onOverweightChange = (e) => {
- valueServices.value.isOverLongWeight = e.detail.value
- }
- const onWoodenFrameChange = (e) => {
- valueServices.value.isWoodenCrate = e.detail.value
- }
- const onInsuranceChange = (e) => {
- if (e.detail.value) {
- insuranceAmount.value = '0'
- } else {
- insuranceAmount.value = ''
- insuranceAmountError.value = ''
- }
- }
- const onSignReturnChange = (e) => {
- valueServices.value.isReceiptCollect = e.detail.value
- if (!e.detail.value) {
- receiptReturnTypeIndex.value = 0
- }
- }
- const onReceiptReturnTypeChange = (e) => {
- receiptReturnTypeIndex.value = e.detail.value
- }
- // 保价金额验证
- const validateInsuranceAmount = () => {
- if (!insuranceAmount.value) return true
- const amount = parseFloat(insuranceAmount.value)
- if (isNaN(amount) || amount < 0) {
- insuranceAmountError.value = '保价金额必须为数字且≥0'
- return false
- }
- if (amount > 1000000) {
- insuranceAmountError.value = '保价金额不能超过100万'
- return false
- }
- const decimal = insuranceAmount.value.toString().split('.')[1]
- if (decimal && decimal.length > 2) {
- insuranceAmountError.value = '最多两位小数'
- return false
- }
- insuranceAmountError.value = ''
- return true
- }
- // 时间选择
- const handleTimeClick = () => {
- showTimePicker.value = true
- }
- const handleTimeConfirm = (time) => {
- selectedTimeData.value = time
- showTimePicker.value = false
- }
- // 地址簿相关
- const openAddressBook = (type) => {
- uni.navigateTo({
- url: `/pages/address/address_list?addType=${type}&from=order`
- })
- }
- const handleAddAddress = (type) => {
- uni.navigateTo({
- url: `/pages/address/edit?addType=${type}&from=order`
- })
- }
- // 协议切换
- const toggleAgreement = () => {
- agreed.value = !agreed.value
- }
- // ==================== 表单验证(增强) ====================
- const validateForm = () => {
- // 地址
- if (!addressSend.value.addressId) {
- uni.showToast({
- title: '请选择寄件人地址',
- icon: 'none'
- });
- return false
- }
- if (!addressReceive.value.addressId) {
- uni.showToast({
- title: '请选择收件人地址',
- icon: 'none'
- });
- return false
- }
- // // 寄件人信息完整性
- // if (!addressSend.value.contactName || !addressSend.value.contactPhone) {
- // uni.showToast({
- // title: '寄件人信息不完整',
- // icon: 'none'
- // });
- // return false
- // }
- // if (!/^1[3-9]\d{9}$/.test(addressSend.value.contactPhone)) {
- // uni.showToast({
- // title: '寄件人手机号格式错误',
- // icon: 'none'
- // });
- // return false
- // }
- // if (addressSend.value.contactName.length < 1 || addressSend.value.contactName.length > 20) {
- // uni.showToast({
- // title: '寄件人姓名长度1-20字符',
- // icon: 'none'
- // });
- // return false
- // }
- // if (!addressSend.value.detailedAddress) {
- // uni.showToast({
- // title: '请填写寄件人人详细地址',
- // icon: 'none'
- // });
- // return false
- // }
- // if (!addressReceive.value.contactName || !addressReceive.value.contactPhone) {
- // uni.showToast({
- // title: '收件人信息不完整',
- // icon: 'none'
- // });
- // return false
- // }
- // if (!/^1[3-9]\d{9}$/.test(addressReceive.value.contactPhone)) {
- // uni.showToast({
- // title: '收件人手机号格式错误',
- // icon: 'none'
- // });
- // return false
- // }
- // if (addressReceive.value.contactName.length < 1 || addressReceive.value.contactName.length > 20) {
- // uni.showToast({
- // title: '收件人姓名长度1-20字符',
- // icon: 'none'
- // });
- // return false
- // }
- // if (!addressReceive.value.detailedAddress) {
- // uni.showToast({
- // title: '请填写收件人详细地址',
- // icon: 'none'
- // });
- // return false
- // }
- // 时间
- if (!selectedTimeData.value) {
- uni.showToast({
- title: '请选择期望上门时间',
- icon: 'none'
- });
- return false
- }
- // 快递类型
- if (!expressTypes.value.length) {
- uni.showToast({
- title: '快递类型字典加载中,请稍后',
- icon: 'none'
- });
- return false
- }
- if (!expressTypes.value[expressTypeIndex.value]) {
- uni.showToast({
- title: '请选择快递类型',
- icon: 'none'
- });
- return false
- }
- // 物品信息
- if (!goodsInfo.value.trim()) {
- uni.showToast({
- title: '请输入物品信息',
- icon: 'none'
- });
- return false
- }
- if (goodsInfo.value.length > 50) {
- uni.showToast({
- title: '物品信息不能超过50字',
- icon: 'none'
- });
- return false
- }
- // 重量/体积/件数范围
- if (weight.value < 0.1 || weight.value > 100) {
- uni.showToast({
- title: '重量范围0.1-100KG',
- icon: 'none'
- });
- return false
- }
- if (volume.value < 0.001 || volume.value > 10) {
- uni.showToast({
- title: '体积范围0.001-10m³',
- icon: 'none'
- });
- return false
- }
- if (quantity.value < 1 || quantity.value > 99) {
- uni.showToast({
- title: '件数范围1-99件',
- icon: 'none'
- });
- return false
- }
- // 保价金额验证
- if (insuranceAmount.value !== '' && insuranceAmount.value !== null) {
- if (!validateInsuranceAmount()) {
- uni.showToast({
- title: insuranceAmountError.value || '保价金额无效',
- icon: 'none'
- })
- return false
- }
- }
- // 顺丰签单返还类型
- if (product.value === '2' && valueServices.value.isReceiptCollect && !receiptReturnTypes.value[
- receiptReturnTypeIndex.value]) {
- uni.showToast({
- title: '请选择签单返还类型',
- icon: 'none'
- });
- return false
- }
- // 协议
- if (!agreed.value) {
- uni.showToast({
- title: '请同意服务协议',
- icon: 'none'
- });
- return false
- }
- return true
- }
- // ==================== 提交订单 ====================
- const submitOrder = async () => {
- if (!validateForm()) return
- // 构建增值服务JSON
- const addedService = {
- isPack: valueServices.value.isPack || null,
- guaranteeMoney: insuranceAmount.value && insuranceAmount.value !== '' ? parseFloat(insuranceAmount
- .value) : null,
- isReceiptCollect: product.value === '2' ?
- (valueServices.value.isReceiptCollect ? receiptReturnTypes.value[receiptReturnTypeIndex.value]
- ?.value : null) :
- (valueServices.value.isReceiptCollect ? true : null),
- isOverLongWeight: valueServices.value.isOverLongWeight || null,
- isWoodenCrate: valueServices.value.isWoodenCrate || null
- }
- // 移除null字段
- Object.keys(addedService).forEach(key => {
- if (addedService[key] === null || addedService[key] === undefined) {
- delete addedService[key]
- }
- })
- const orderData = {
- orderType: product.value,
- senderName: addressSend.value.contactName,
- senderPhone: addressSend.value.contactPhone,
- senderProvince: addressSend.value.provinceName,
- senderCity: addressSend.value.cityName,
- senderCounty: addressSend.value.countyName,
- senderAddress: addressSend.value.detailedAddress,
- receiverName: addressReceive.value.contactName,
- receiverPhone: addressReceive.value.contactPhone,
- receiverProvince: addressReceive.value.provinceName,
- receiverCity: addressReceive.value.cityName,
- receiverCounty: addressReceive.value.countyName,
- receiverAddress: addressReceive.value.detailedAddress,
- goodsName: goodsInfo.value,
- goodsWeight: weight.value,
- goodsVolume: volume.value,
- goodsQty: quantity.value,
- sendStartTime: selectedTimeData.value.startTime,
- sendEndTime: selectedTimeData.value.endTime,
- productCode: expressTypes.value[expressTypeIndex.value]?.value, // 直接从动态字典取 dictValue
- addedService: JSON.stringify(addedService)
- }
- uni.showLoading({
- title: '提交中...'
- })
- try {
- const res = await createOrder(orderData)
- uni.hideLoading()
- if (res.code === 200) {
- uni.showToast({
- title: '下单成功',
- icon: 'success'
- })
- setTimeout(() => {
- uni.switchTab({
- url: '/pages/order/index'
- })
- }, 1000)
- } else {
- uni.showToast({
- title: res.msg || '下单失败',
- icon: 'none'
- })
- }
- } catch (error) {
- uni.hideLoading()
- uni.showToast({
- title: '网络错误,请重试',
- icon: 'none'
- })
- console.error('订单提交失败', error)
- }
- }
- // ==================== 获取默认寄件地址 ====================
- const fetchDefaultSenderAddress = async () => {
- try {
- const res = await getDefaultAddress()
- if (res.code === 200 && res.data) {
- addressSend.value = res.data
- addressSend.value.defaultFlag = res.data.defaultFlag === '1'
- saveAddressToStorage('senderAddress', res.data)
- }
- } catch (error) {
- console.error('获取默认地址失败', error)
- }
- }
- // ==================== 生命周期 ====================
- onLoad((option) => {
- if (option.product) {
- product.value = option.product
- }
- // 获取默认寄件地址
- fetchDefaultSenderAddress()
- // 获取快递类型字典
- fetchDictData()
- })
- onMounted(() => {
- // 监听地址选择返回
- uni.$on('addressSelected', (data) => {
- if (data.type === 'sender') {
- addressSend.value = data.address
- saveAddressToStorage('senderAddress', data.address)
- } else if (data.type === 'receiver') {
- addressReceive.value = data.address
- saveAddressToStorage('receiverAddress', data.address)
- }
- })
- })
- // 组件卸载时取消监听
- onUnmounted(() => {
- uni.$off('addressSelected')
- })
- </script>
- <style lang="scss" scoped>
- /* 完全保持原有样式,未做任何修改 */
- .order-container {
- min-height: 100vh;
- background-color: #F5F7FA;
- padding: 20rpx;
- padding-bottom: 180rpx;
- box-sizing: border-box;
- }
- .info-card {
- background: #fff;
- border-radius: 32rpx;
- padding: 20rpx;
- margin-bottom: 20rpx;
- .address-item {
- display: flex;
- align-items: center;
- &:first-child {
- padding-bottom: 20rpx;
- }
- &:last-child {
- padding-top: 20rpx;
- }
- }
- .user-info {
- flex: 1;
- margin-left: 26rpx;
- .line {
- width: 590rpx;
- height: 3rpx;
- background-color: #F1F3F8;
- }
- .create-btn {
- height: 88rpx;
- line-height: 88rpx;
- font-size: 28rpx;
- color: #1B64F0;
- }
- }
- .right {
- margin-left: 20rpx;
- }
- .img-status-text {
- align-self: center;
- text-align: center;
- flex-shrink: 0;
- .img {
- width: 56rpx;
- height: 56rpx;
- }
- .img-change {
- width: 40rpx;
- height: 40rpx;
- }
- .address-image {
- width: 48rpx;
- height: 48rpx;
- }
- .status-text {
- font-size: 28rpx;
- color: #333;
- font-weight: 400;
- }
- }
- }
- .pickup-title {
- font-size: 32rpx;
- font-weight: 600;
- color: #333;
- margin-bottom: 20rpx;
- }
- .goods-card {
- background-color: #fff;
- border-radius: 32rpx;
- padding: 0rpx 20rpx;
- margin-bottom: 20rpx;
- }
- .goods-item {
- display: flex;
- justify-content: space-between;
- align-items: center;
- height: 100rpx;
- border-bottom: 1rpx solid #F1F3F8;
- &:last-child {
- border-bottom: none;
- margin-bottom: 0;
- }
- .item-label {
- font-size: 28rpx;
- color: #666;
- font-weight: 400;
- &.required::before {
- content: '*';
- color: #ff4444;
- margin-right: 8rpx;
- }
- }
- .item-control {
- display: flex;
- align-items: center;
- .picker-text {
- font-size: 28rpx;
- color: #333;
- text-align: right;
- margin-right: 8rpx;
- }
- &.btn {
- background: #fff;
- border: 1rpx solid #dcdfe6;
- border-radius: 5rpx;
- height: 59rpx;
- width: 266rpx;
- box-sizing: border-box;
- }
- .control-btn {
- width: 56rpx;
- height: 56rpx;
- background: #F5F7FA;
- border-radius: 4rpx;
- display: flex;
- align-items: center;
- justify-content: center;
- font-size: 36rpx;
- color: #666;
- &.minus {
- color: #999;
- border-right: 1rpx solid #dcdfe6;
- }
- &.plus {
- color: #999;
- border-left: 1rpx solid #dcdfe6;
- }
- }
- .control-value {
- width: 154rpx;
- text-align: center;
- font-size: 24rpx;
- color: #333;
- .input-field {
- width: 100%;
- height: 100rpx;
- line-height: 100rpx;
- padding: 0 24rpx;
- font-size: 28rpx;
- color: #333;
- text-align: center;
- &::placeholder {
- color: #999;
- }
- }
- }
- }
- .input-field {
- width: 100%;
- height: 100rpx;
- line-height: 100rpx;
- padding: 0 24rpx;
- font-size: 28rpx;
- color: #333;
- text-align: right;
- &::placeholder {
- color: #999;
- }
- }
- .time-value {
- display: flex;
- align-items: center;
- font-size: 28rpx;
- color: #333;
- line-height: 88rpx;
- height: 88rpx;
- .value {
- margin-right: 16rpx;
- }
- .arrow {
- margin-left: 32rpx;
- color: #999;
- font-size: 28rpx;
- font-weight: bold;
- }
- }
- .arrow {
- color: #999;
- }
- }
- .agreement-card {
- background-color: #fff;
- border-radius: 32rpx;
- padding: 30rpx 20rpx;
- margin-bottom: 20rpx;
- .agreement-item {
- display: flex;
- align-items: center;
- margin-bottom: 20rpx;
- .agreement-text {
- font-size: 28rpx;
- color: #333;
- margin-left: 10rpx;
- }
- }
- .price-notice {
- font-size: 24rpx;
- color: #ff7d00;
- text-align: center;
- }
- }
- .submit-btn {
- position: fixed;
- bottom: 40rpx;
- left: 30rpx;
- right: 30rpx;
- width: 686rpx;
- height: 88rpx;
- background: #1B64F0;
- border-radius: 44rpx;
- display: flex;
- justify-content: center;
- align-items: center;
- font-size: 32rpx;
- color: #FFFFFF;
- line-height: 88rpx;
- text-align: center;
- z-index: 10;
- &.disabled {
- background: #f5f5f5;
- color: #999;
- }
- }
- .safe-area {
- height: 140rpx;
- }
- </style>
|