| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065 |
- <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',
- mask: true
- })
- setTimeout(() => {
- uni.navigateTo({
- 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
- }
- // 获取快递类型字典
- 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>
|