forked from gxwebsoft/mp-10550
refactor(order): 重构订单状态处理逻辑并优化送水订单功能
- 将订单状态相关工具函数提取到独立的 utils 文件中 - 统一订单状态文本和颜色显示逻辑 - 移除重复的状态判断函数 - 优化送水订单列表的数据过滤逻辑 - 添加订单编辑模式支持 - 实现订单修改和取消功能 - 修复订单状态判断中的数值转换问题 - 优化送水订单的时间选择组件 - 添加订单数据加载和验证逻辑 - 重构订单详情页的条件渲染逻辑
This commit is contained in:
@@ -1,6 +1,6 @@
|
||||
import { useEffect, useMemo, useRef, useState } from 'react'
|
||||
import Taro, { useDidShow } from '@tarojs/taro'
|
||||
import { View, Text } from '@tarojs/components'
|
||||
import { View, Text, Picker } from '@tarojs/components'
|
||||
import {
|
||||
Button,
|
||||
Cell,
|
||||
@@ -25,8 +25,8 @@ import type {ShopStore} from "@/api/shop/shopStore/model";
|
||||
import {getShopStore, listShopStore} from "@/api/shop/shopStore";
|
||||
import {getSelectedStoreFromStorage, saveSelectedStoreToStorage} from "@/utils/storeSelection";
|
||||
import type { GltUserTicket } from '@/api/glt/gltUserTicket/model'
|
||||
import { listGltUserTicket } from '@/api/glt/gltUserTicket'
|
||||
import { addGltTicketOrder } from '@/api/glt/gltTicketOrder'
|
||||
import { getGltUserTicket, listGltUserTicket } from '@/api/glt/gltUserTicket'
|
||||
import { addGltTicketOrder, getGltTicketOrder, updateGltTicketOrder } from '@/api/glt/gltTicketOrder'
|
||||
import { pageGltTicketOrder } from '@/api/glt/gltTicketOrder'
|
||||
import type { GltTicketOrder } from '@/api/glt/gltTicketOrder/model'
|
||||
import type { ShopStoreRider } from '@/api/shop/shopStoreRider/model'
|
||||
@@ -44,7 +44,7 @@ const OrderConfirm = () => {
|
||||
const [quantity, setQuantity] = useState<number>(MIN_START_QTY)
|
||||
const [orderRemark, setOrderRemark] = useState<string>('')
|
||||
// Delivery date only (no hour/min selection).
|
||||
const [sendTime] = useState<Date>(() => dayjs().startOf('day').toDate())
|
||||
const [sendTime, setSendTime] = useState<Date>(() => dayjs().startOf('day').toDate())
|
||||
const [loading, setLoading] = useState<boolean>(true)
|
||||
const [error, setError] = useState<string>('')
|
||||
const [submitLoading, setSubmitLoading] = useState<boolean>(false)
|
||||
@@ -89,11 +89,21 @@ const OrderConfirm = () => {
|
||||
|
||||
const router = Taro.getCurrentInstance().router;
|
||||
const goodsId = router?.params?.goodsId;
|
||||
const orderId = router?.params?.orderId;
|
||||
const numericGoodsId = useMemo(() => {
|
||||
const n = goodsId ? Number(goodsId) : undefined
|
||||
return typeof n === 'number' && Number.isFinite(n) ? n : undefined
|
||||
}, [goodsId])
|
||||
|
||||
const numericOrderId = useMemo(() => {
|
||||
const n = orderId ? Number(orderId) : undefined
|
||||
return typeof n === 'number' && Number.isFinite(n) && n > 0 ? n : undefined
|
||||
}, [orderId])
|
||||
|
||||
const isEditMode = !!numericOrderId
|
||||
const [editingOrder, setEditingOrder] = useState<GltTicketOrder | null>(null)
|
||||
const editingInitRef = useRef(false)
|
||||
|
||||
const userId = useMemo(() => {
|
||||
const raw = Taro.getStorageSync('UserId')
|
||||
const id = Number(raw)
|
||||
@@ -300,8 +310,19 @@ const OrderConfirm = () => {
|
||||
|
||||
const maxQuantity = useMemo(() => {
|
||||
const stockMax = goods?.stock ?? 999
|
||||
return Math.max(0, Math.min(stockMax, availableTicketTotal))
|
||||
}, [availableTicketTotal, goods?.stock])
|
||||
if (!isEditMode) return Math.max(0, Math.min(stockMax, availableTicketTotal))
|
||||
|
||||
const original = Number(editingOrder?.totalNum ?? 0)
|
||||
const originalSafe = Number.isFinite(original) ? original : 0
|
||||
const ticketId = Number(editingOrder?.userTicketId ?? 0)
|
||||
const ticketIdSafe = Number.isFinite(ticketId) && ticketId > 0 ? ticketId : undefined
|
||||
const rawTicket = ticketIdSafe ? (tickets || []).find(t => Number(t?.id) === ticketIdSafe) : undefined
|
||||
if (!rawTicket) return Math.max(0, Math.min(stockMax, originalSafe))
|
||||
|
||||
const avail = getTicketAvailableQty(rawTicket)
|
||||
const upper = Math.max(0, avail + originalSafe)
|
||||
return Math.max(0, Math.min(stockMax, upper))
|
||||
}, [availableTicketTotal, editingOrder?.totalNum, editingOrder?.userTicketId, goods?.stock, isEditMode, tickets])
|
||||
|
||||
const canStartOrder = useMemo(() => {
|
||||
return maxQuantity >= MIN_START_QTY
|
||||
@@ -623,13 +644,16 @@ const OrderConfirm = () => {
|
||||
const onSubmit = async () => {
|
||||
if (submitLoading) return
|
||||
if (deliveryRangeCheckingRef.current) return
|
||||
if (!goods?.goodsId) return
|
||||
|
||||
// 基础校验
|
||||
if (!userId) {
|
||||
Taro.showToast({ title: '请先登录', icon: 'none' })
|
||||
return
|
||||
}
|
||||
if (isEditMode && !editingOrder?.id) {
|
||||
Taro.showToast({ title: '订单信息加载中,请稍后重试', icon: 'none' })
|
||||
return
|
||||
}
|
||||
if (!address?.id) {
|
||||
Taro.showToast({ title: '请选择收货地址', icon: 'none' })
|
||||
return
|
||||
@@ -672,7 +696,7 @@ const OrderConfirm = () => {
|
||||
Taro.showToast({ title: '未找到可配送门店,请先在首页选择门店或联系管理员配置门店坐标', icon: 'none' })
|
||||
return
|
||||
}
|
||||
if (availableTicketTotal <= 0) {
|
||||
if (!isEditMode && availableTicketTotal <= 0) {
|
||||
Taro.showToast({ title: '暂无可用水票', icon: 'none' })
|
||||
return
|
||||
}
|
||||
@@ -682,11 +706,15 @@ const OrderConfirm = () => {
|
||||
Taro.showToast({ title: '请选择送水数量', icon: 'none' })
|
||||
return
|
||||
}
|
||||
if (finalQty > availableTicketTotal) {
|
||||
if (!isEditMode && finalQty > availableTicketTotal) {
|
||||
Taro.showToast({ title: '水票可用次数不足', icon: 'none' })
|
||||
return
|
||||
}
|
||||
if (goods.stock !== undefined && finalQty > goods.stock) {
|
||||
if (isEditMode && finalQty > maxQuantity) {
|
||||
Taro.showToast({ title: '水票可用次数不足', icon: 'none' })
|
||||
return
|
||||
}
|
||||
if (goods?.stock !== undefined && finalQty > goods.stock) {
|
||||
Taro.showToast({ title: '商品库存不足', icon: 'none' })
|
||||
return
|
||||
}
|
||||
@@ -704,8 +732,10 @@ const OrderConfirm = () => {
|
||||
if (!ok) return
|
||||
|
||||
const confirmRes = await Taro.showModal({
|
||||
title: '确认下单',
|
||||
content: `配送时间:${sendTimeText}\n将使用 ${finalQty} 张水票下单(优先使用可用数量少的水票),送水 ${finalQty} 桶,是否确认?`
|
||||
title: isEditMode ? '确认修改' : '确认下单',
|
||||
content: isEditMode
|
||||
? `配送时间:${sendTimeText}\n送水数量:${finalQty} 桶\n是否确认修改?`
|
||||
: `配送时间:${sendTimeText}\n将使用 ${finalQty} 张水票下单(优先使用可用数量少的水票),送水 ${finalQty} 桶,是否确认?`
|
||||
})
|
||||
if (!confirmRes.confirm) return
|
||||
|
||||
@@ -713,51 +743,59 @@ const OrderConfirm = () => {
|
||||
setSubmitLoading(true)
|
||||
Taro.showLoading({ title: '提交中...' })
|
||||
|
||||
// Best-effort auto dispatch rider. If it fails, backend/manual dispatch can still handle it.
|
||||
const autoRider = storeForOrder.id ? await resolveAutoRiderForStore(storeForOrder.id) : null
|
||||
|
||||
// Split this "delivery" into multiple ticket orders (backend order model binds to one userTicketId).
|
||||
// Consume tickets with smaller available qty first.
|
||||
let remain = finalQty
|
||||
let created = 0
|
||||
for (const t of ticketsToConsume) {
|
||||
if (remain <= 0) break
|
||||
const avail = getTicketAvailableQty(t)
|
||||
const useQty = Math.min(remain, avail)
|
||||
if (useQty <= 0) continue
|
||||
await addGltTicketOrder({
|
||||
userTicketId: Number(t.id),
|
||||
storeId: storeForOrder.id,
|
||||
if (isEditMode) {
|
||||
await updateGltTicketOrder({
|
||||
id: editingOrder?.id,
|
||||
addressId: address.id,
|
||||
totalNum: useQty,
|
||||
totalNum: finalQty,
|
||||
buyerRemarks: orderRemark,
|
||||
sendTime: dayjs(sendTime).startOf('day').format('YYYY-MM-DD HH:mm:ss'),
|
||||
// Backend may take userId from token; pass-through is harmless if backend ignores it.
|
||||
userId,
|
||||
riderId: Number.isFinite(Number(autoRider?.userId)) ? Number(autoRider?.userId) : undefined,
|
||||
riderName: autoRider?.realName,
|
||||
riderPhone: autoRider?.mobile,
|
||||
comments: goods.name ? `立即送水:${goods.name}` : '立即送水'
|
||||
sendTime: dayjs(sendTime).startOf('day').format('YYYY-MM-DD HH:mm:ss')
|
||||
})
|
||||
remain -= useQty
|
||||
created += 1
|
||||
}
|
||||
} else {
|
||||
// Best-effort auto dispatch rider. If it fails, backend/manual dispatch can still handle it.
|
||||
const autoRider = storeForOrder.id ? await resolveAutoRiderForStore(storeForOrder.id) : null
|
||||
|
||||
if (remain > 0) {
|
||||
// Ticket counts might have changed between loading and submission.
|
||||
throw new Error('水票可用次数不足,请刷新后重试')
|
||||
// Split this "delivery" into multiple ticket orders (backend order model binds to one userTicketId).
|
||||
// Consume tickets with smaller available qty first.
|
||||
let remain = finalQty
|
||||
for (const t of ticketsToConsume) {
|
||||
if (remain <= 0) break
|
||||
const avail = getTicketAvailableQty(t)
|
||||
const useQty = Math.min(remain, avail)
|
||||
if (useQty <= 0) continue
|
||||
await addGltTicketOrder({
|
||||
userTicketId: Number(t.id),
|
||||
storeId: storeForOrder.id,
|
||||
addressId: address.id,
|
||||
totalNum: useQty,
|
||||
buyerRemarks: orderRemark,
|
||||
sendTime: dayjs(sendTime).startOf('day').format('YYYY-MM-DD HH:mm:ss'),
|
||||
// Backend may take userId from token; pass-through is harmless if backend ignores it.
|
||||
userId,
|
||||
riderId: Number.isFinite(Number(autoRider?.userId)) ? Number(autoRider?.userId) : undefined,
|
||||
riderName: autoRider?.realName,
|
||||
riderPhone: autoRider?.mobile,
|
||||
comments: goods?.name ? `立即送水:${goods.name}` : '立即送水'
|
||||
})
|
||||
remain -= useQty
|
||||
}
|
||||
|
||||
if (remain > 0) {
|
||||
// Ticket counts might have changed between loading and submission.
|
||||
throw new Error('水票可用次数不足,请刷新后重试')
|
||||
}
|
||||
}
|
||||
|
||||
await loadUserTickets()
|
||||
|
||||
Taro.showToast({ title: created > 1 ? '下单成功(已拆分多张水票)' : '下单成功', icon: 'success' })
|
||||
Taro.showToast({ title: isEditMode ? '修改成功' : '下单成功', icon: 'success' })
|
||||
setTimeout(() => {
|
||||
// 跳转到“我的送水订单”
|
||||
Taro.redirectTo({ url: '/user/ticket/index?tab=order' })
|
||||
}, 800)
|
||||
} catch (e: any) {
|
||||
console.error('水票下单失败:', e)
|
||||
Taro.showToast({ title: e?.message || '下单失败', icon: 'none' })
|
||||
console.error(isEditMode ? '送水订单修改失败:' : '水票下单失败:', e)
|
||||
Taro.showToast({ title: e?.message || (isEditMode ? '修改失败' : '下单失败'), icon: 'none' })
|
||||
} finally {
|
||||
Taro.hideLoading()
|
||||
setSubmitLoading(false)
|
||||
@@ -772,11 +810,28 @@ const OrderConfirm = () => {
|
||||
if (!opts?.silent) setLoading(true)
|
||||
setError('')
|
||||
|
||||
const [goodsRes, addressRes] = await Promise.all([
|
||||
numericGoodsId ? getShopGoods(numericGoodsId) : Promise.resolve(null),
|
||||
listShopUserAddress({ isDefault: true })
|
||||
const [addressRes, editingOrderRes, goodsByParam] = await Promise.all([
|
||||
listShopUserAddress({ isDefault: true }),
|
||||
numericOrderId ? getGltTicketOrder(numericOrderId) : Promise.resolve(null),
|
||||
numericGoodsId ? getShopGoods(numericGoodsId) : Promise.resolve(null)
|
||||
])
|
||||
|
||||
let goodsRes = goodsByParam
|
||||
if (!goodsRes && editingOrderRes?.userTicketId) {
|
||||
const ticketId = Number(editingOrderRes.userTicketId)
|
||||
if (Number.isFinite(ticketId) && ticketId > 0) {
|
||||
try {
|
||||
const ticket = await getGltUserTicket(ticketId)
|
||||
const gid = Number((ticket as any)?.goodsId)
|
||||
if (Number.isFinite(gid) && gid > 0) {
|
||||
goodsRes = await getShopGoods(gid)
|
||||
}
|
||||
} catch (e) {
|
||||
console.error('加载订单关联商品失败:', e)
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// 设置商品信息
|
||||
if (goodsRes) {
|
||||
setGoods(goodsRes)
|
||||
@@ -788,6 +843,49 @@ const OrderConfirm = () => {
|
||||
setAddress(addressRes[0])
|
||||
}
|
||||
|
||||
if (numericOrderId && editingOrderRes && !editingInitRef.current) {
|
||||
editingInitRef.current = true
|
||||
setEditingOrder(editingOrderRes)
|
||||
Taro.setNavigationBarTitle({ title: '订单确认' })
|
||||
|
||||
const ds = editingOrderRes.deliveryStatus
|
||||
const hasProgress = !!editingOrderRes.sendStartTime || !!editingOrderRes.sendEndTime || !!editingOrderRes.receiveConfirmTime
|
||||
const isPending =
|
||||
Number((editingOrderRes as any)?.deleted) !== 1 &&
|
||||
Number(editingOrderRes.status) !== 1 &&
|
||||
!hasProgress &&
|
||||
(ds === 10 || (typeof ds !== 'number' && !!editingOrderRes.riderId))
|
||||
if (!isPending) {
|
||||
Taro.showToast({ title: '该订单当前不可修改', icon: 'none' })
|
||||
setTimeout(() => {
|
||||
Taro.navigateBack()
|
||||
}, 600)
|
||||
return
|
||||
}
|
||||
|
||||
const initQty = Number(editingOrderRes.totalNum ?? MIN_START_QTY)
|
||||
setQuantity(Number.isFinite(initQty) && initQty > 0 ? initQty : MIN_START_QTY)
|
||||
setOrderRemark(String(editingOrderRes.buyerRemarks || ''))
|
||||
const st = parseTime(editingOrderRes.sendTime)
|
||||
if (st) setSendTime(st.startOf('day').toDate())
|
||||
|
||||
const addrId = Number(editingOrderRes.addressId)
|
||||
const addrIdSafe = Number.isFinite(addrId) && addrId > 0 ? addrId : undefined
|
||||
if (addrIdSafe) {
|
||||
const hit = addressRes?.find(a => Number(a?.id) === addrIdSafe)
|
||||
if (hit?.id) {
|
||||
setAddress(hit)
|
||||
} else {
|
||||
try {
|
||||
const addr = await getShopUserAddress(addrIdSafe)
|
||||
if (addr?.id) setAddress(addr)
|
||||
} catch (e) {
|
||||
console.error('加载订单收货地址失败:', e)
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Load ticket-order history to enforce "address can be modified once per 30 days".
|
||||
// If currently locked, force using last ticket-order address (snapshot) to avoid getting stuck with a new default address.
|
||||
try {
|
||||
@@ -946,7 +1044,7 @@ const OrderConfirm = () => {
|
||||
}
|
||||
|
||||
// 加载状态
|
||||
if (loading || !goods) {
|
||||
if (loading) {
|
||||
return <OrderConfirmSkeleton/>
|
||||
}
|
||||
|
||||
@@ -1017,9 +1115,20 @@ const OrderConfirm = () => {
|
||||
<Cell
|
||||
title={'配送时间'}
|
||||
extra={(
|
||||
<View className={'flex items-center gap-2'}>
|
||||
<View className={'text-gray-900'}>{sendTimeText}</View>
|
||||
</View>
|
||||
<Picker
|
||||
mode="date"
|
||||
value={dayjs(sendTime).format('YYYY-MM-DD')}
|
||||
onChange={(e) => {
|
||||
const v = (e as any)?.detail?.value
|
||||
const d = dayjs(v)
|
||||
if (d.isValid()) setSendTime(d.startOf('day').toDate())
|
||||
}}
|
||||
>
|
||||
<View className={'flex items-center gap-2'}>
|
||||
<View className={'text-gray-900'}>{sendTimeText}</View>
|
||||
<ArrowRight className={'text-gray-400'} size={14} />
|
||||
</View>
|
||||
</Picker>
|
||||
)}
|
||||
/>
|
||||
</CellGroup>
|
||||
|
||||
Reference in New Issue
Block a user