From 49c801c7510cf64efa000bde69842c5065e87fbc Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?=E8=B5=B5=E5=BF=A0=E6=9E=97?= <170083662@qq.com>
Date: Mon, 9 Mar 2026 12:48:02 +0800
Subject: [PATCH] =?UTF-8?q?refactor(order):=20=E9=87=8D=E6=9E=84=E8=AE=A2?=
=?UTF-8?q?=E5=8D=95=E7=8A=B6=E6=80=81=E5=A4=84=E7=90=86=E9=80=BB=E8=BE=91?=
=?UTF-8?q?=E5=B9=B6=E4=BC=98=E5=8C=96=E9=80=81=E6=B0=B4=E8=AE=A2=E5=8D=95?=
=?UTF-8?q?=E5=8A=9F=E8=83=BD?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
- 将订单状态相关工具函数提取到独立的 utils 文件中
- 统一订单状态文本和颜色显示逻辑
- 移除重复的状态判断函数
- 优化送水订单列表的数据过滤逻辑
- 添加订单编辑模式支持
- 实现订单修改和取消功能
- 修复订单状态判断中的数值转换问题
- 优化送水订单的时间选择组件
- 添加订单数据加载和验证逻辑
- 重构订单详情页的条件渲染逻辑
---
src/dealer/withdraw/index.tsx | 2 +-
src/shop/orderDetail/index.tsx | 36 +---
src/user/order/components/OrderList.tsx | 64 +------
src/user/ticket/index.tsx | 183 +++++++++-----------
src/user/ticket/use.tsx | 211 ++++++++++++++++++------
src/utils/shopOrderStatus.ts | 65 ++++++++
6 files changed, 313 insertions(+), 248 deletions(-)
create mode 100644 src/utils/shopOrderStatus.ts
diff --git a/src/dealer/withdraw/index.tsx b/src/dealer/withdraw/index.tsx
index e8274a9..c183b2b 100644
--- a/src/dealer/withdraw/index.tsx
+++ b/src/dealer/withdraw/index.tsx
@@ -491,7 +491,7 @@ const DealerWithdraw: React.FC = () => {
diff --git a/src/shop/orderDetail/index.tsx b/src/shop/orderDetail/index.tsx
index dca8ed1..ee2409e 100644
--- a/src/shop/orderDetail/index.tsx
+++ b/src/shop/orderDetail/index.tsx
@@ -8,6 +8,7 @@ import {listShopOrderGoods} from "@/api/shop/shopOrderGoods";
import {ShopOrderGoods} from "@/api/shop/shopOrderGoods/model";
import dayjs from "dayjs";
import PaymentCountdown from "@/components/PaymentCountdown";
+import {getShopOrderStatusText} from "@/utils/shopOrderStatus";
import './index.scss'
// 申请退款:支付成功后仅允许在指定时间窗内发起(前端展示层限制,后端仍应校验)
@@ -114,37 +115,6 @@ const OrderDetail = () => {
}
}
- const getOrderStatusText = (order: ShopOrder) => {
- // 优先检查订单状态
- if (order.orderStatus === 2) return '已取消';
- if (order.orderStatus === 3) return '取消中';
- if (order.orderStatus === 4) return '退款申请中';
- if (order.orderStatus === 5) return '退款被拒绝';
- if (order.orderStatus === 6) return '退款成功';
- if (order.orderStatus === 7) return '客户端申请退款';
-
- // 检查支付状态 (payStatus为boolean类型)
- if (!order.payStatus) return '待付款';
-
- // 已付款后检查发货状态
- if (order.deliveryStatus === 10) return '待发货';
- if (order.deliveryStatus === 20) {
- // 若订单有配送员,则以配送员送达时间作为“可确认收货”的依据
- if (order.riderId) {
- if (order.sendEndTime && order.orderStatus !== 1) return '待确认收货';
- return '配送中';
- }
- return '待收货';
- }
- if (order.deliveryStatus === 30) return '部分发货';
-
- // 最后检查订单完成状态
- if (order.orderStatus === 1) return '已完成';
- if (order.orderStatus === 0) return '未使用';
-
- return '未知状态';
- };
-
const getPayTypeText = (payType?: number) => {
switch (payType) {
case 0:
@@ -194,7 +164,7 @@ const OrderDetail = () => {
order.payStatus &&
order.orderStatus !== 1 &&
order.deliveryStatus === 20 &&
- (!order.riderId || !!order.sendEndTime)
+ (!order.riderId || Number(order.riderId) === 0 || !!order.sendEndTime)
return (
@@ -232,7 +202,7 @@ const OrderDetail = () => {
|
-
+ |
| | |
diff --git a/src/user/order/components/OrderList.tsx b/src/user/order/components/OrderList.tsx
index 485a328..0083a69 100644
--- a/src/user/order/components/OrderList.tsx
+++ b/src/user/order/components/OrderList.tsx
@@ -16,6 +16,7 @@ import {copyText} from "@/utils/common";
import PaymentCountdown from "@/components/PaymentCountdown";
import {PaymentType} from "@/utils/payment";
import {ErrorType, RequestError} from "@/utils/request";
+import {getShopOrderStatusColor, getShopOrderStatusText, isShopOrderCompleted} from "@/utils/shopOrderStatus";
// 判断订单是否支付已过期
const isPaymentExpired = (createTime: string, timeoutHours: number = 24): boolean => {
@@ -165,68 +166,11 @@ function OrderList(props: OrderListProps) {
};
// “已完成”应以订单状态为准;不要用商品ID等字段推断完成态,否则会造成 Tab(待发货/待收货) 与状态文案不同步
- const isOrderCompleted = (order: ShopOrder) => toNum(order.orderStatus) === 1;
+ const isOrderCompleted = (order: ShopOrder) => isShopOrderCompleted(order);
- // 获取订单状态文本
- const getOrderStatusText = (order: ShopOrder) => {
- const orderStatus = toNum(order.orderStatus);
- const deliveryStatus = toNum(order.deliveryStatus);
+ const getOrderStatusText = (order: ShopOrder) => getShopOrderStatusText(order);
- // 优先检查订单状态
- if (orderStatus === 2) return '已取消';
- if (orderStatus === 4) return '退款申请中';
- if (orderStatus === 5) return '退款被拒绝';
- if (orderStatus === 6) return '退款成功';
- if (orderStatus === 7) return '客户端申请退款';
- if (isOrderCompleted(order)) return '已完成';
-
- // 检查支付状态 (payStatus为boolean类型,false/0表示未付款,true/1表示已付款)
- if (!order.payStatus) return '等待买家付款';
-
- // 已付款后检查发货状态
- if (deliveryStatus === 10) return '待发货';
- if (deliveryStatus === 20) {
- // 若订单没有配送员,沿用原“待收货”语义
- if (!order.riderId || Number(order.riderId) === 0) return '待收货';
- // 配送员确认送达后(sendEndTime有值),才进入“待确认收货”
- if (order.sendEndTime && !isOrderCompleted(order)) return '待确认收货';
- return '配送中';
- }
- if (deliveryStatus === 30) return '部分发货';
-
- if (orderStatus === 0) return '未使用';
-
- return '未知状态';
- };
-
- // 获取订单状态颜色
- const getOrderStatusColor = (order: ShopOrder) => {
- const orderStatus = toNum(order.orderStatus);
- const deliveryStatus = toNum(order.deliveryStatus);
- // 优先检查订单状态
- if (orderStatus === 2) return 'text-gray-500'; // 已取消
- if (orderStatus === 4) return 'text-orange-500'; // 退款申请中
- if (orderStatus === 5) return 'text-red-500'; // 退款被拒绝
- if (orderStatus === 6) return 'text-green-500'; // 退款成功
- if (orderStatus === 7) return 'text-orange-500'; // 客户端申请退款
- if (isOrderCompleted(order)) return 'text-green-600'; // 已完成
-
- // 检查支付状态
- if (!order.payStatus) return 'text-orange-500'; // 等待买家付款
-
- // 已付款后检查发货状态
- if (deliveryStatus === 10) return 'text-blue-500'; // 待发货
- if (deliveryStatus === 20) {
- if (!order.riderId || Number(order.riderId) === 0) return 'text-purple-500'; // 待收货
- if (order.sendEndTime && !isOrderCompleted(order)) return 'text-purple-500'; // 待确认收货
- return 'text-blue-500'; // 配送中
- }
- if (deliveryStatus === 30) return 'text-blue-500'; // 部分发货
-
- if (orderStatus === 0) return 'text-gray-500'; // 未使用
-
- return 'text-gray-600'; // 默认颜色
- };
+ const getOrderStatusColor = (order: ShopOrder) => getShopOrderStatusColor(order);
// 使用后端统一的 statusFilter 进行筛选
const getOrderStatusParams = (index: string | number) => {
diff --git a/src/user/ticket/index.tsx b/src/user/ticket/index.tsx
index 325bb6e..9cccf18 100644
--- a/src/user/ticket/index.tsx
+++ b/src/user/ticket/index.tsx
@@ -1,4 +1,4 @@
-import { useRef, useState } from 'react';
+import { useState } from 'react';
import Taro, { useDidShow } from '@tarojs/taro';
import {
Button,
@@ -16,9 +16,8 @@ import {
import { View, Text, Image } from '@tarojs/components';
import { pageGltUserTicket } from '@/api/glt/gltUserTicket';
import type { GltUserTicket } from '@/api/glt/gltUserTicket/model';
-import { pageGltTicketOrder, updateGltTicketOrder } from '@/api/glt/gltTicketOrder';
+import { pageGltTicketOrder, removeGltTicketOrder, updateGltTicketOrder } from '@/api/glt/gltTicketOrder';
import type { GltTicketOrder } from '@/api/glt/gltTicketOrder/model';
-import { getShopUserAddress } from '@/api/shop/shopUserAddress';
import { BaseUrl } from '@/config/app';
import dayjs from "dayjs";
@@ -47,8 +46,6 @@ const UserTicketList = () => {
const [qrTicket, setQrTicket] = useState(null);
const [qrImageUrl, setQrImageUrl] = useState('');
- const addressCacheRef = useRef>({});
-
const getUserId = () => {
const raw = Taro.getStorageSync('UserId');
const id = Number(raw);
@@ -188,17 +185,16 @@ const UserTicketList = () => {
});
const resList = res?.list || [];
- const nextList = isRefresh ? resList : [...orderList, ...resList];
+ const safeList = resList.filter((o) => Number((o as any)?.deleted) !== 1);
+ const nextList = isRefresh ? safeList : [...orderList, ...safeList];
setOrderList(nextList);
- const count = typeof res?.count === 'number' ? res.count : nextList.length;
- setOrderTotal(count);
- setOrderHasMore(nextList.length < count);
+ const serverCount = typeof res?.count === 'number' ? res.count : undefined;
+ const total = typeof serverCount === 'number' ? serverCount : nextList.length;
+ setOrderTotal(total);
+ setOrderHasMore(typeof serverCount === 'number' ? nextList.length < serverCount : resList.length >= PAGE_SIZE);
- if (resList.length > 0) {
- setOrderPage(currentPage + 1);
- } else {
- setOrderHasMore(false);
- }
+ if (resList.length > 0) setOrderPage(currentPage + 1);
+ else setOrderHasMore(false);
} catch (error) {
console.error('获取送水订单失败:', error);
Taro.showToast({ title: '获取送水订单失败', icon: 'error' });
@@ -265,78 +261,60 @@ const UserTicketList = () => {
return d.isValid() ? d.format('YYYY年MM月DD日') : v;
};
- const parseLatLng = (latRaw?: unknown, lngRaw?: unknown) => {
- const lat = typeof latRaw === 'number' ? latRaw : parseFloat(String(latRaw ?? ''));
- const lng = typeof lngRaw === 'number' ? lngRaw : parseFloat(String(lngRaw ?? ''));
- if (!Number.isFinite(lat) || !Number.isFinite(lng)) return null;
- if (Math.abs(lat) > 90 || Math.abs(lng) > 180) return null;
- return { lat, lng };
+ const isTicketOrderPendingDelivery = (order: GltTicketOrder) => {
+ if (!order?.id) return false;
+ if (Number(order.status) === 1) return false;
+ if (Number((order as any)?.deleted) === 1) return false;
+ if (order.receiveConfirmTime || order.sendEndTime || order.sendStartTime) return false;
+
+ const ds = order.deliveryStatus;
+ if (typeof ds === 'number') return ds === 10;
+ return !!order.riderId;
};
- const handleNavigateToAddress = async (order: GltTicketOrder) => {
- try {
- // Prefer coordinates from backend if present (non-typed fields), otherwise fetch by addressId.
- const anyOrder = order as any;
- const direct =
- parseLatLng(anyOrder?.addressLat ?? anyOrder?.lat, anyOrder?.addressLng ?? anyOrder?.lng) ||
- parseLatLng(anyOrder?.receiverLat, anyOrder?.receiverLng);
-
- let coords = direct;
- let fullAddress: string | undefined = order.address || undefined;
-
- if (!coords && order.addressId) {
- const cached = addressCacheRef.current[order.addressId];
- if (cached) {
- coords = { lat: cached.lat, lng: cached.lng };
- fullAddress = fullAddress || cached.fullAddress;
- } else if (cached === null) {
- coords = null;
- } else {
- const addr = await getShopUserAddress(order.addressId);
- const parsed = parseLatLng(addr?.lat, addr?.lng);
- if (parsed) {
- coords = parsed;
- fullAddress = fullAddress || addr?.fullAddress || addr?.address || undefined;
- addressCacheRef.current[order.addressId] = { ...parsed, fullAddress };
- } else {
- addressCacheRef.current[order.addressId] = null;
- }
- }
- }
-
- if (!coords) {
- if (fullAddress) {
- await Taro.setClipboardData({ data: fullAddress });
- Taro.showToast({ title: '未配置定位,地址已复制', icon: 'none' });
- } else {
- Taro.showToast({ title: '暂无可导航的地址', icon: 'none' });
- }
- return;
- }
-
- Taro.openLocation({
- latitude: coords.lat,
- longitude: coords.lng,
- name: '收货地址',
- address: fullAddress || ''
- });
- } catch (e) {
- console.error('一键导航失败:', e);
- Taro.showToast({ title: '导航失败,请重试', icon: 'none' });
- }
- };
-
- const handleOneClickCall = async (order: GltTicketOrder) => {
- const phone = (order.riderPhone || order.storePhone || '').trim();
- if (!phone) {
- Taro.showToast({ title: '暂无可呼叫的电话', icon: 'none' });
+ const handleOrderModify = async (order: GltTicketOrder) => {
+ if (!order?.id) {
+ Taro.showToast({ title: '订单信息不完整', icon: 'none' });
return;
}
+ if (!isTicketOrderPendingDelivery(order)) {
+ Taro.showToast({ title: '仅“待配送”订单可修改', icon: 'none' });
+ return;
+ }
+ Taro.navigateTo({ url: `/user/ticket/use?orderId=${order.id}` });
+ };
+
+ const handleOrderCancel = async (order: GltTicketOrder) => {
+ if (!order?.id) {
+ Taro.showToast({ title: '订单信息不完整', icon: 'none' });
+ return;
+ }
+ if (!isTicketOrderPendingDelivery(order)) {
+ Taro.showToast({ title: '仅“待配送”订单可取消', icon: 'none' });
+ return;
+ }
+
+ const modal = await Taro.showModal({
+ title: '取消订单',
+ content: '确定要取消该订单吗?取消后无法恢复。',
+ confirmText: '确认取消'
+ });
+ if (!modal.confirm) return;
+
try {
- await Taro.makePhoneCall({ phoneNumber: phone });
+ Taro.showLoading({ title: '取消中...' });
+ try {
+ await updateGltTicketOrder({ id: order.id, deleted: 1 });
+ } catch (e) {
+ await removeGltTicketOrder(order.id);
+ }
+ Taro.showToast({ title: '订单已取消', icon: 'success' });
+ await reloadOrders(true);
} catch (e) {
- console.error('一键呼叫失败:', e);
- Taro.showToast({ title: '呼叫失败,请手动拨打', icon: 'none' });
+ console.error('取消送水订单失败:', e);
+ Taro.showToast({ title: '取消失败,请重试', icon: 'none' });
+ } finally {
+ Taro.hideLoading();
}
};
@@ -576,30 +554,29 @@ const UserTicketList = () => {
下单时间:{formatDateTime(item.createTime)}
- {(!!item.addressId || !!item.address || !!item.riderPhone || !!item.storePhone) ? (
+ {item.id ? (
- {(!!item.addressId || !!item.address) ? (
-
- ) : null}
- {(!!item.riderPhone || !!item.storePhone) ? (
-
- ) : null}
+
+
) : null}
{/*{item.storeName ? (*/}
diff --git a/src/user/ticket/use.tsx b/src/user/ticket/use.tsx
index 23cb3ff..7f698dd 100644
--- a/src/user/ticket/use.tsx
+++ b/src/user/ticket/use.tsx
@@ -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(MIN_START_QTY)
const [orderRemark, setOrderRemark] = useState('')
// Delivery date only (no hour/min selection).
- const [sendTime] = useState(() => dayjs().startOf('day').toDate())
+ const [sendTime, setSendTime] = useState(() => dayjs().startOf('day').toDate())
const [loading, setLoading] = useState(true)
const [error, setError] = useState('')
const [submitLoading, setSubmitLoading] = useState(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(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
}
@@ -1017,9 +1115,20 @@ const OrderConfirm = () => {
- {sendTimeText}
-
+ {
+ const v = (e as any)?.detail?.value
+ const d = dayjs(v)
+ if (d.isValid()) setSendTime(d.startOf('day').toDate())
+ }}
+ >
+
+ {sendTimeText}
+
+
+
)}
/>
|
diff --git a/src/utils/shopOrderStatus.ts b/src/utils/shopOrderStatus.ts
new file mode 100644
index 0000000..593db97
--- /dev/null
+++ b/src/utils/shopOrderStatus.ts
@@ -0,0 +1,65 @@
+import type { ShopOrder } from '@/api/shop/shopOrder/model';
+
+const toNum = (value: unknown): number | undefined => {
+ if (value === null || value === undefined || value === '') return undefined;
+ const n = Number(value);
+ return Number.isFinite(n) ? n : undefined;
+};
+
+export const isShopOrderCompleted = (order: Pick
): boolean =>
+ toNum(order?.orderStatus) === 1;
+
+export const getShopOrderStatusText = (order: ShopOrder): string => {
+ const orderStatus = toNum(order?.orderStatus);
+ const deliveryStatus = toNum(order?.deliveryStatus);
+ const riderId = toNum(order?.riderId);
+
+ if (orderStatus === 2) return '已取消';
+ if (orderStatus === 3) return '取消中';
+ if (orderStatus === 4) return '退款申请中';
+ if (orderStatus === 5) return '退款被拒绝';
+ if (orderStatus === 6) return '退款成功';
+ if (orderStatus === 7) return '客户端申请退款';
+ if (orderStatus === 1) return '已完成';
+
+ if (!order?.payStatus) return '等待买家付款';
+
+ if (deliveryStatus === 10) return '待发货';
+ if (deliveryStatus === 20) {
+ if (!riderId || riderId === 0) return '待收货';
+ if (order?.sendEndTime) return '待确认收货';
+ return '配送中';
+ }
+ if (deliveryStatus === 30) return '部分发货';
+
+ if (orderStatus === 0) return '未使用';
+ return '未知状态';
+};
+
+export const getShopOrderStatusColor = (order: ShopOrder): string => {
+ const orderStatus = toNum(order?.orderStatus);
+ const deliveryStatus = toNum(order?.deliveryStatus);
+ const riderId = toNum(order?.riderId);
+
+ if (orderStatus === 2) return 'text-gray-500'; // 已取消
+ if (orderStatus === 3) return 'text-orange-500'; // 取消中
+ if (orderStatus === 4) return 'text-orange-500'; // 退款申请中
+ if (orderStatus === 5) return 'text-red-500'; // 退款被拒绝
+ if (orderStatus === 6) return 'text-green-500'; // 退款成功
+ if (orderStatus === 7) return 'text-orange-500'; // 客户端申请退款
+ if (orderStatus === 1) return 'text-green-600'; // 已完成
+
+ if (!order?.payStatus) return 'text-orange-500'; // 等待买家付款
+
+ if (deliveryStatus === 10) return 'text-blue-500'; // 待发货
+ if (deliveryStatus === 20) {
+ if (!riderId || riderId === 0) return 'text-purple-500'; // 待收货
+ if (order?.sendEndTime) return 'text-purple-500'; // 待确认收货
+ return 'text-blue-500'; // 配送中
+ }
+ if (deliveryStatus === 30) return 'text-blue-500'; // 部分发货
+
+ if (orderStatus === 0) return 'text-gray-500'; // 未使用
+ return 'text-gray-600';
+};
+