2026-03-01 03:04:24 +08:00
|
|
|
'use client';
|
|
|
|
|
|
|
|
|
|
import { useSearchParams } from 'next/navigation';
|
|
|
|
|
import { useState, useEffect, useCallback, Suspense } from 'react';
|
|
|
|
|
import OrderTable from '@/components/admin/OrderTable';
|
|
|
|
|
import OrderDetail from '@/components/admin/OrderDetail';
|
2026-03-01 20:12:32 +08:00
|
|
|
import PaginationBar from '@/components/PaginationBar';
|
2026-03-03 03:31:20 +08:00
|
|
|
import PayPageLayout from '@/components/PayPageLayout';
|
2026-03-01 03:04:24 +08:00
|
|
|
|
2026-03-01 17:58:08 +08:00
|
|
|
interface AdminOrder {
|
|
|
|
|
id: string;
|
|
|
|
|
userId: number;
|
|
|
|
|
userName: string | null;
|
|
|
|
|
userEmail: string | null;
|
2026-03-03 04:37:39 +08:00
|
|
|
userNotes: string | null;
|
2026-03-01 17:58:08 +08:00
|
|
|
amount: number;
|
|
|
|
|
status: string;
|
|
|
|
|
paymentType: string;
|
|
|
|
|
createdAt: string;
|
|
|
|
|
paidAt: string | null;
|
|
|
|
|
completedAt: string | null;
|
|
|
|
|
failedReason: string | null;
|
|
|
|
|
expiresAt: string;
|
2026-03-03 00:19:01 +08:00
|
|
|
srcHost: string | null;
|
2026-03-01 17:58:08 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
interface AdminOrderDetail extends AdminOrder {
|
|
|
|
|
rechargeCode: string;
|
|
|
|
|
paymentTradeNo: string | null;
|
|
|
|
|
refundAmount: number | null;
|
|
|
|
|
refundReason: string | null;
|
|
|
|
|
refundAt: string | null;
|
|
|
|
|
forceRefund: boolean;
|
|
|
|
|
failedAt: string | null;
|
|
|
|
|
updatedAt: string;
|
|
|
|
|
clientIp: string | null;
|
2026-03-02 20:40:16 +08:00
|
|
|
srcHost: string | null;
|
|
|
|
|
srcUrl: string | null;
|
2026-03-01 17:58:08 +08:00
|
|
|
paymentSuccess?: boolean;
|
|
|
|
|
rechargeSuccess?: boolean;
|
|
|
|
|
rechargeStatus?: string;
|
|
|
|
|
auditLogs: { id: string; action: string; detail: string | null; operator: string | null; createdAt: string }[];
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-01 03:04:24 +08:00
|
|
|
function AdminContent() {
|
|
|
|
|
const searchParams = useSearchParams();
|
|
|
|
|
const token = searchParams.get('token');
|
2026-03-03 03:31:20 +08:00
|
|
|
const theme = searchParams.get('theme') === 'dark' ? 'dark' : 'light';
|
|
|
|
|
const uiMode = searchParams.get('ui_mode') || 'standalone';
|
|
|
|
|
const isDark = theme === 'dark';
|
|
|
|
|
const isEmbedded = uiMode === 'embedded';
|
2026-03-01 03:04:24 +08:00
|
|
|
|
2026-03-01 17:58:08 +08:00
|
|
|
const [orders, setOrders] = useState<AdminOrder[]>([]);
|
2026-03-01 03:04:24 +08:00
|
|
|
const [total, setTotal] = useState(0);
|
|
|
|
|
const [page, setPage] = useState(1);
|
2026-03-01 20:04:49 +08:00
|
|
|
const [pageSize, setPageSize] = useState(20);
|
2026-03-01 03:04:24 +08:00
|
|
|
const [totalPages, setTotalPages] = useState(1);
|
|
|
|
|
const [statusFilter, setStatusFilter] = useState('');
|
|
|
|
|
const [loading, setLoading] = useState(true);
|
|
|
|
|
const [error, setError] = useState('');
|
|
|
|
|
|
2026-03-01 17:58:08 +08:00
|
|
|
const [detailOrder, setDetailOrder] = useState<AdminOrderDetail | null>(null);
|
2026-03-01 03:04:24 +08:00
|
|
|
|
|
|
|
|
const fetchOrders = useCallback(async () => {
|
|
|
|
|
if (!token) return;
|
|
|
|
|
setLoading(true);
|
|
|
|
|
try {
|
2026-03-01 20:04:49 +08:00
|
|
|
const params = new URLSearchParams({ token, page: String(page), page_size: String(pageSize) });
|
2026-03-01 03:04:24 +08:00
|
|
|
if (statusFilter) params.set('status', statusFilter);
|
|
|
|
|
|
|
|
|
|
const res = await fetch(`/api/admin/orders?${params}`);
|
|
|
|
|
if (!res.ok) {
|
|
|
|
|
if (res.status === 401) {
|
|
|
|
|
setError('管理员凭证无效');
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
throw new Error('请求失败');
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const data = await res.json();
|
|
|
|
|
setOrders(data.orders);
|
|
|
|
|
setTotal(data.total);
|
|
|
|
|
setTotalPages(data.total_pages);
|
|
|
|
|
} catch (e) {
|
|
|
|
|
setError('加载订单列表失败');
|
|
|
|
|
} finally {
|
|
|
|
|
setLoading(false);
|
|
|
|
|
}
|
2026-03-01 20:04:49 +08:00
|
|
|
}, [token, page, pageSize, statusFilter]);
|
2026-03-01 03:04:24 +08:00
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
fetchOrders();
|
|
|
|
|
}, [fetchOrders]);
|
|
|
|
|
|
|
|
|
|
if (!token) {
|
|
|
|
|
return (
|
2026-03-03 03:31:20 +08:00
|
|
|
<div className={`flex min-h-screen items-center justify-center p-4 ${isDark ? 'bg-slate-950' : 'bg-slate-50'}`}>
|
|
|
|
|
<div className="text-center text-red-500">
|
|
|
|
|
<p className="text-lg font-medium">缺少管理员凭证</p>
|
|
|
|
|
<p className="mt-2 text-sm text-gray-500">请从 Sub2API 平台正确访问管理页面</p>
|
|
|
|
|
</div>
|
2026-03-01 03:04:24 +08:00
|
|
|
</div>
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const handleRetry = async (orderId: string) => {
|
|
|
|
|
if (!confirm('确认重试充值?')) return;
|
|
|
|
|
try {
|
|
|
|
|
const res = await fetch(`/api/admin/orders/${orderId}/retry?token=${token}`, {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
});
|
|
|
|
|
if (res.ok) {
|
|
|
|
|
fetchOrders();
|
|
|
|
|
} else {
|
|
|
|
|
const data = await res.json();
|
|
|
|
|
setError(data.error || '重试失败');
|
|
|
|
|
}
|
|
|
|
|
} catch {
|
|
|
|
|
setError('重试请求失败');
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const handleCancel = async (orderId: string) => {
|
|
|
|
|
if (!confirm('确认取消该订单?')) return;
|
|
|
|
|
try {
|
|
|
|
|
const res = await fetch(`/api/admin/orders/${orderId}/cancel?token=${token}`, {
|
|
|
|
|
method: 'POST',
|
|
|
|
|
});
|
|
|
|
|
if (res.ok) {
|
|
|
|
|
fetchOrders();
|
|
|
|
|
} else {
|
|
|
|
|
const data = await res.json();
|
|
|
|
|
setError(data.error || '取消失败');
|
|
|
|
|
}
|
|
|
|
|
} catch {
|
|
|
|
|
setError('取消请求失败');
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const handleViewDetail = async (orderId: string) => {
|
|
|
|
|
try {
|
|
|
|
|
const res = await fetch(`/api/admin/orders/${orderId}?token=${token}`);
|
|
|
|
|
if (res.ok) {
|
|
|
|
|
const data = await res.json();
|
|
|
|
|
setDetailOrder(data);
|
|
|
|
|
}
|
|
|
|
|
} catch {
|
|
|
|
|
setError('加载订单详情失败');
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
const statuses = ['', 'PENDING', 'PAID', 'RECHARGING', 'COMPLETED', 'EXPIRED', 'CANCELLED', 'FAILED', 'REFUNDED'];
|
|
|
|
|
const statusLabels: Record<string, string> = {
|
|
|
|
|
'': '全部',
|
|
|
|
|
PENDING: '待支付',
|
|
|
|
|
PAID: '已支付',
|
|
|
|
|
RECHARGING: '充值中',
|
|
|
|
|
COMPLETED: '已完成',
|
|
|
|
|
EXPIRED: '已超时',
|
|
|
|
|
CANCELLED: '已取消',
|
|
|
|
|
FAILED: '充值失败',
|
|
|
|
|
REFUNDED: '已退款',
|
|
|
|
|
};
|
|
|
|
|
|
2026-03-04 17:06:27 +08:00
|
|
|
const navParams = new URLSearchParams();
|
|
|
|
|
if (token) navParams.set('token', token);
|
|
|
|
|
if (isDark) navParams.set('theme', 'dark');
|
|
|
|
|
if (isEmbedded) navParams.set('ui_mode', 'embedded');
|
|
|
|
|
|
|
|
|
|
const btnBase = [
|
|
|
|
|
'inline-flex items-center rounded-lg border px-3 py-1.5 text-xs font-medium transition-colors',
|
2026-03-05 23:10:44 +08:00
|
|
|
isDark
|
|
|
|
|
? 'border-slate-600 text-slate-200 hover:bg-slate-800'
|
|
|
|
|
: 'border-slate-300 text-slate-700 hover:bg-slate-100',
|
2026-03-04 17:06:27 +08:00
|
|
|
].join(' ');
|
|
|
|
|
|
2026-03-01 03:04:24 +08:00
|
|
|
return (
|
2026-03-03 03:31:20 +08:00
|
|
|
<PayPageLayout
|
|
|
|
|
isDark={isDark}
|
|
|
|
|
isEmbedded={isEmbedded}
|
|
|
|
|
maxWidth="full"
|
|
|
|
|
title="订单管理"
|
|
|
|
|
subtitle="查看和管理所有充值订单"
|
|
|
|
|
actions={
|
2026-03-04 17:06:27 +08:00
|
|
|
<>
|
|
|
|
|
<a href={`/admin/dashboard?${navParams}`} className={btnBase}>
|
|
|
|
|
数据概览
|
|
|
|
|
</a>
|
|
|
|
|
<button type="button" onClick={fetchOrders} className={btnBase}>
|
|
|
|
|
刷新
|
|
|
|
|
</button>
|
|
|
|
|
</>
|
2026-03-03 03:31:20 +08:00
|
|
|
}
|
|
|
|
|
>
|
2026-03-01 03:04:24 +08:00
|
|
|
{error && (
|
2026-03-05 23:10:44 +08:00
|
|
|
<div
|
|
|
|
|
className={`mb-4 rounded-lg border p-3 text-sm ${isDark ? 'border-red-800 bg-red-950/50 text-red-400' : 'border-red-200 bg-red-50 text-red-600'}`}
|
|
|
|
|
>
|
2026-03-01 03:04:24 +08:00
|
|
|
{error}
|
2026-03-03 03:31:20 +08:00
|
|
|
<button onClick={() => setError('')} className="ml-2 opacity-60 hover:opacity-100">
|
2026-03-01 17:58:08 +08:00
|
|
|
✕
|
|
|
|
|
</button>
|
2026-03-01 03:04:24 +08:00
|
|
|
</div>
|
|
|
|
|
)}
|
|
|
|
|
|
|
|
|
|
{/* Filters */}
|
|
|
|
|
<div className="mb-4 flex flex-wrap gap-2">
|
|
|
|
|
{statuses.map((s) => (
|
|
|
|
|
<button
|
|
|
|
|
key={s}
|
2026-03-01 17:58:08 +08:00
|
|
|
onClick={() => {
|
|
|
|
|
setStatusFilter(s);
|
|
|
|
|
setPage(1);
|
|
|
|
|
}}
|
2026-03-03 03:31:20 +08:00
|
|
|
className={[
|
|
|
|
|
'rounded-full px-3 py-1 text-sm transition-colors',
|
|
|
|
|
statusFilter === s
|
2026-03-05 23:10:44 +08:00
|
|
|
? isDark
|
|
|
|
|
? 'bg-indigo-500/30 text-indigo-200 ring-1 ring-indigo-400/40'
|
|
|
|
|
: 'bg-blue-600 text-white'
|
|
|
|
|
: isDark
|
|
|
|
|
? 'bg-slate-800 text-slate-400 hover:bg-slate-700'
|
|
|
|
|
: 'bg-gray-100 text-gray-600 hover:bg-gray-200',
|
2026-03-03 03:31:20 +08:00
|
|
|
].join(' ')}
|
2026-03-01 03:04:24 +08:00
|
|
|
>
|
|
|
|
|
{statusLabels[s]}
|
|
|
|
|
</button>
|
|
|
|
|
))}
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
{/* Table */}
|
2026-03-05 23:10:44 +08:00
|
|
|
<div
|
|
|
|
|
className={[
|
|
|
|
|
'rounded-xl border',
|
|
|
|
|
isDark ? 'border-slate-700 bg-slate-800/70' : 'border-slate-200 bg-white shadow-sm',
|
|
|
|
|
].join(' ')}
|
|
|
|
|
>
|
2026-03-01 03:04:24 +08:00
|
|
|
{loading ? (
|
2026-03-03 03:31:20 +08:00
|
|
|
<div className={`py-12 text-center ${isDark ? 'text-slate-400' : 'text-gray-500'}`}>加载中...</div>
|
2026-03-01 03:04:24 +08:00
|
|
|
) : (
|
2026-03-05 23:10:44 +08:00
|
|
|
<OrderTable
|
|
|
|
|
orders={orders}
|
|
|
|
|
onRetry={handleRetry}
|
|
|
|
|
onCancel={handleCancel}
|
|
|
|
|
onViewDetail={handleViewDetail}
|
|
|
|
|
dark={isDark}
|
|
|
|
|
/>
|
2026-03-01 03:04:24 +08:00
|
|
|
)}
|
|
|
|
|
</div>
|
|
|
|
|
|
2026-03-01 20:12:32 +08:00
|
|
|
<PaginationBar
|
|
|
|
|
page={page}
|
|
|
|
|
totalPages={totalPages}
|
|
|
|
|
total={total}
|
|
|
|
|
pageSize={pageSize}
|
|
|
|
|
loading={loading}
|
|
|
|
|
onPageChange={(p) => setPage(p)}
|
2026-03-05 23:10:44 +08:00
|
|
|
onPageSizeChange={(s) => {
|
|
|
|
|
setPageSize(s);
|
|
|
|
|
setPage(1);
|
|
|
|
|
}}
|
2026-03-03 03:31:20 +08:00
|
|
|
isDark={isDark}
|
2026-03-01 20:12:32 +08:00
|
|
|
/>
|
2026-03-01 03:04:24 +08:00
|
|
|
|
|
|
|
|
{/* Order Detail */}
|
2026-03-03 03:31:20 +08:00
|
|
|
{detailOrder && <OrderDetail order={detailOrder} onClose={() => setDetailOrder(null)} dark={isDark} />}
|
|
|
|
|
</PayPageLayout>
|
2026-03-01 03:04:24 +08:00
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
export default function AdminPage() {
|
|
|
|
|
return (
|
2026-03-01 17:58:08 +08:00
|
|
|
<Suspense
|
|
|
|
|
fallback={
|
|
|
|
|
<div className="flex min-h-screen items-center justify-center">
|
|
|
|
|
<div className="text-gray-500">加载中...</div>
|
|
|
|
|
</div>
|
|
|
|
|
}
|
|
|
|
|
>
|
2026-03-01 03:04:24 +08:00
|
|
|
<AdminContent />
|
|
|
|
|
</Suspense>
|
|
|
|
|
);
|
|
|
|
|
}
|