将 AdminRiskPage 改名为 AdminRiskIndexPage,并接入 RiskIndexConsole 组件。

This commit is contained in:
2026-05-11 11:52:53 +08:00
parent 78045de9a3
commit 0103d25426
16 changed files with 1033 additions and 15 deletions

73
src/api/admin-risk.ts Normal file
View File

@@ -0,0 +1,73 @@
import { adminRequest } from "@/lib/admin-http";
import { API_V1_PREFIX } from "./paths";
import type {
AdminRiskLockLogListData,
AdminRiskPoolListData,
AdminRiskPoolShowData,
} from "@/types/api/admin-risk";
const A = `${API_V1_PREFIX}/admin`;
export type AdminRiskPoolListQuery = {
page?: number;
per_page?: number;
sold_out_only?: boolean;
sort?: "usage_desc" | "locked_desc" | "remaining_asc" | "number_asc";
};
export async function getAdminRiskPools(
drawId: number,
q: AdminRiskPoolListQuery = {},
): Promise<AdminRiskPoolListData> {
return adminRequest.get<AdminRiskPoolListData>(`${A}/draws/${drawId}/risk-pools`, {
params: {
page: q.page,
per_page: q.per_page,
sold_out_only: q.sold_out_only === true ? 1 : undefined,
sort: q.sort,
},
});
}
export type AdminRiskLockLogQuery = {
page?: number;
per_page?: number;
action_type?: "lock" | "release";
normalized_number?: string;
};
export async function getAdminRiskPoolLockLogs(
drawId: number,
q: AdminRiskLockLogQuery = {},
): Promise<AdminRiskLockLogListData> {
return adminRequest.get<AdminRiskLockLogListData>(
`${A}/draws/${drawId}/risk-pool-lock-logs`,
{
params: {
page: q.page,
per_page: q.per_page,
action_type: q.action_type,
normalized_number: q.normalized_number,
},
},
);
}
export type AdminRiskPoolShowQuery = {
page?: number;
per_page?: number;
};
export async function getAdminRiskPoolDetail(
drawId: number,
number4d: string,
q: AdminRiskPoolShowQuery = {},
): Promise<AdminRiskPoolShowData> {
const encoded = encodeURIComponent(number4d);
return adminRequest.get<AdminRiskPoolShowData>(
`${A}/draws/${drawId}/risk-pools/${encoded}`,
{ params: { page: q.page, per_page: q.per_page } },
);
}

View File

@@ -0,0 +1,18 @@
import { RiskPoolsConsole } from "@/modules/risk/risk-pools-console";
export default async function AdminRiskHotPage(props: {
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
const id = Number(drawId);
return (
<RiskPoolsConsole
drawId={id}
title="热门号码监控"
description="按「已占用 / 封顶」比例与已占用额排序,便于发现集中下注号码(产品文档 §13.4)。"
soldOutOnly={false}
defaultSort="usage_desc"
/>
);
}

View File

@@ -0,0 +1,20 @@
import { ModuleScaffold } from "@/components/admin/module-scaffold";
import { RiskDrawHeader } from "@/modules/risk/risk-draw-header";
import { RiskSubnav } from "@/modules/risk/risk-subnav";
export default async function AdminRiskDrawLayout(props: {
children: React.ReactNode;
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
const id = Number(drawId);
const safeId = Number.isFinite(id) ? id : 0;
return (
<ModuleScaffold className="w-full max-w-none">
<RiskDrawHeader drawId={safeId} />
<RiskSubnav drawId={drawId} />
{props.children}
</ModuleScaffold>
);
}

View File

@@ -0,0 +1,10 @@
import { RiskLockLogsConsole } from "@/modules/risk/risk-lock-logs-console";
export default async function AdminRiskOccupancyPage(props: {
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
const id = Number(drawId);
return <RiskLockLogsConsole drawId={id} />;
}

View File

@@ -0,0 +1,10 @@
import { RiskPoolDetailConsole } from "@/modules/risk/risk-pool-detail-console";
export default async function AdminRiskPoolDetailPage(props: {
params: Promise<{ drawId: string; number: string }>;
}) {
const { drawId, number } = await props.params;
const id = Number(drawId);
return <RiskPoolDetailConsole drawId={id} number4d={number} />;
}

View File

@@ -0,0 +1,19 @@
import { RiskPoolsConsole } from "@/modules/risk/risk-pools-console";
export default async function AdminRiskPoolsPage(props: {
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
const id = Number(drawId);
return (
<RiskPoolsConsole
drawId={id}
title="全部风险池"
description="本期已出现的全部号码赔付池事实表;可排序,点击「查看」进入单号码详情页。"
soldOutOnly={false}
defaultSort="number_asc"
allowSortChange
/>
);
}

View File

@@ -0,0 +1,18 @@
import { RiskPoolsConsole } from "@/modules/risk/risk-pools-console";
export default async function AdminRiskSoldOutPage(props: {
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
const id = Number(drawId);
return (
<RiskPoolsConsole
drawId={id}
title="售罄号码列表"
description="剩余赔付额度为 0 的 4D 号码;玩家侧将收到售罄拒单(产品文档 §6.4、§13.4)。"
soldOutOnly
defaultSort="number_asc"
/>
);
}

View File

@@ -1,21 +1,10 @@
import { ModuleScaffold } from "@/components/admin/module-scaffold";
import { riskModuleMeta } from "@/modules/risk/meta";
import type { Metadata } from "next";
import { RiskIndexConsole } from "@/modules/risk/risk-index-console";
export const metadata: Metadata = {
title: riskModuleMeta.title,
};
export default function AdminRiskPage() {
export default function AdminRiskIndexPage() {
return (
<ModuleScaffold>
<p className="text-sm text-zinc-500 dark:text-zinc-400">
{" "}
<code className="rounded bg-zinc-100 px-1 py-0.5 font-mono text-xs dark:bg-zinc-800">
src/modules/risk
</code>{" "}
</p>
<ModuleScaffold className="w-full max-w-none">
<RiskIndexConsole />
</ModuleScaffold>
);
}

8
src/lib/money.ts Normal file
View File

@@ -0,0 +1,8 @@
/** 后台列表统一:最小货币单位 → 主货币展示(默认 2 位小数,与钱包一致) */
export function formatAdminMinorUnits(minor: number, currencyCode = "NPR"): string {
const major = minor / 100;
return `${currencyCode} ${major.toLocaleString(undefined, {
minimumFractionDigits: 2,
maximumFractionDigits: 2,
})}`;
}

View File

@@ -0,0 +1,57 @@
"use client";
import { useCallback, useEffect, useState } from "react";
import { getAdminDraw } from "@/api/admin-draws";
import { DrawStatusBadge } from "@/modules/draws/draw-status-badge";
import { LotteryApiBizError } from "@/types/api/errors";
import type { AdminDrawShowData } from "@/types/api/admin-draws";
export function RiskDrawHeader({ drawId }: { drawId: number }) {
const [draw, setDraw] = useState<AdminDrawShowData | null>(null);
const [error, setError] = useState<string | null>(null);
const load = useCallback(async () => {
setError(null);
try {
const d = await getAdminDraw(drawId);
setDraw(d);
} catch (e) {
const msg =
e instanceof LotteryApiBizError ? e.message : "无法加载期号信息";
setError(msg);
setDraw(null);
}
}, [drawId]);
useEffect(() => {
queueMicrotask(() => {
void load();
});
}, [load]);
if (error) {
return <p className="text-sm text-destructive">{error}</p>;
}
if (!draw) {
return <p className="text-sm text-muted-foreground"></p>;
}
return (
<div className="mb-4 space-y-1">
<h1 className="text-xl font-semibold tracking-tight">
· {draw.draw_no}
</h1>
<p className="flex flex-wrap items-center gap-2 text-sm text-muted-foreground">
<span></span>
<DrawStatusBadge status={draw.status} />
<span className="text-xs opacity-80">{draw.hall_preview_status}</span>
</p>
<p className="text-xs text-muted-foreground">
`risk_pools` /
`risk_pool_lock_logs`
</p>
</div>
);
}

View File

@@ -0,0 +1,101 @@
"use client";
import Link from "next/link";
import { useCallback, useEffect, useState } from "react";
import { getAdminDraws } from "@/api/admin-draws";
import { buttonVariants } from "@/components/ui/button";
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from "@/components/ui/table";
import { DrawStatusBadge } from "@/modules/draws/draw-status-badge";
import { useAdminDateTimeFormatter } from "@/hooks/use-admin-datetime-formatter";
import { cn } from "@/lib/utils";
import { LotteryApiBizError } from "@/types/api/errors";
import type { AdminDrawListData, AdminDrawListItem } from "@/types/api/admin-draws";
export function RiskIndexConsole() {
const formatDt = useAdminDateTimeFormatter();
const [data, setData] = useState<AdminDrawListData | null>(null);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const load = useCallback(async () => {
setLoading(true);
setError(null);
try {
const d = await getAdminDraws({ page: 1, per_page: 50 });
setData(d);
} catch (e) {
const msg =
e instanceof LotteryApiBizError ? e.message : "加载期号列表失败";
setError(msg);
setData(null);
} finally {
setLoading(false);
}
}, []);
useEffect(() => {
queueMicrotask(() => {
void load();
});
}, [load]);
return (
<Card>
<CardHeader>
<CardTitle className="text-lg"></CardTitle>
<CardDescription>
§13.4
</CardDescription>
</CardHeader>
<CardContent className="space-y-4">
{error ? <p className="text-sm text-destructive">{error}</p> : null}
{loading ? (
<p className="text-sm text-muted-foreground"></p>
) : (
<div className="overflow-x-auto rounded-md border">
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead className="text-right"></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{(data?.items ?? []).map((row: AdminDrawListItem) => (
<TableRow key={row.id}>
<TableCell className="font-mono font-medium">{row.draw_no}</TableCell>
<TableCell>
<DrawStatusBadge status={row.status} />
</TableCell>
<TableCell className="text-sm text-muted-foreground">
{row.close_time ? formatDt(row.close_time) : "—"}
</TableCell>
<TableCell className="text-right">
<Link
href={`/admin/risk/draws/${row.id}/occupancy`}
className={cn(buttonVariants({ variant: "secondary", size: "sm" }))}
>
</Link>
</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</div>
)}
</CardContent>
</Card>
);
}

View File

@@ -0,0 +1,192 @@
"use client";
import { useCallback, useEffect, useState } from "react";
import { getAdminRiskPoolLockLogs } from "@/api/admin-risk";
import { AdminListPaginationFooter } from "@/components/admin/admin-list-pagination-footer";
import { Button } from "@/components/ui/button";
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
import { Input } from "@/components/ui/input";
import { Label } from "@/components/ui/label";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select";
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from "@/components/ui/table";
import { useAdminDateTimeFormatter } from "@/hooks/use-admin-datetime-formatter";
import { formatAdminMinorUnits } from "@/lib/money";
import { LotteryApiBizError } from "@/types/api/errors";
import type { AdminRiskLockLogListData, AdminRiskLockLogRow } from "@/types/api/admin-risk";
const ACTION_ALL = "__all__";
export function RiskLockLogsConsole({ drawId }: { drawId: number }) {
const formatDt = useAdminDateTimeFormatter();
const [page, setPage] = useState(1);
const [perPage, setPerPage] = useState(25);
const [data, setData] = useState<AdminRiskLockLogListData | null>(null);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const [draftNumber, setDraftNumber] = useState("");
const [appliedNumber, setAppliedNumber] = useState("");
const [draftAction, setDraftAction] = useState<string>(ACTION_ALL);
const [appliedAction, setAppliedAction] = useState<string>(ACTION_ALL);
const load = useCallback(async () => {
setLoading(true);
setError(null);
try {
const d = await getAdminRiskPoolLockLogs(drawId, {
page,
per_page: perPage,
normalized_number: appliedNumber.trim() === "" ? undefined : appliedNumber.trim(),
action_type:
appliedAction === ACTION_ALL
? undefined
: (appliedAction as "lock" | "release"),
});
setData(d);
} catch (e) {
const msg =
e instanceof LotteryApiBizError ? e.message : "加载占用流水失败";
setError(msg);
setData(null);
} finally {
setLoading(false);
}
}, [drawId, page, perPage, appliedAction, appliedNumber]);
useEffect(() => {
queueMicrotask(() => {
void load();
});
}, [load]);
return (
<Card>
<CardHeader>
<CardTitle className="text-lg"></CardTitle>
<CardDescription>
/ `risk_pool_lock_logs`
</CardDescription>
</CardHeader>
<CardContent className="space-y-4">
<div className="grid max-w-full gap-3 sm:grid-cols-[minmax(0,8rem)_minmax(0,10rem)_auto] sm:items-end">
<div className="space-y-1.5">
<Label htmlFor="risk-log-number">4 </Label>
<Input
id="risk-log-number"
inputMode="numeric"
maxLength={4}
value={draftNumber}
onChange={(e) => setDraftNumber(e.target.value.replace(/\D/g, "").slice(0, 4))}
placeholder="可选"
/>
</div>
<div className="space-y-1.5">
<Label htmlFor="risk-log-action"></Label>
<Select
modal={false}
value={draftAction}
onValueChange={(v) => {
if (v) setDraftAction(v);
}}
>
<SelectTrigger id="risk-log-action" size="sm" className="w-full">
<SelectValue />
</SelectTrigger>
<SelectContent>
<SelectItem value={ACTION_ALL}></SelectItem>
<SelectItem value="lock"> lock</SelectItem>
<SelectItem value="release"> release</SelectItem>
</SelectContent>
</Select>
</div>
<div className="flex gap-2 sm:justify-end">
<Button
type="button"
size="sm"
onClick={() => {
setAppliedNumber(draftNumber);
setAppliedAction(draftAction);
setPage(1);
}}
>
</Button>
</div>
</div>
{error ? <p className="text-sm text-destructive">{error}</p> : null}
{loading && !data ? (
<p className="text-sm text-muted-foreground"></p>
) : (
<>
<div className="overflow-x-auto rounded-md border">
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead className="text-right"></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{(data?.items ?? []).map((row: AdminRiskLockLogRow) => (
<TableRow key={row.id}>
<TableCell className="whitespace-nowrap text-xs text-muted-foreground">
{row.created_at ? formatDt(row.created_at) : "—"}
</TableCell>
<TableCell className="font-mono text-sm">{row.normalized_number}</TableCell>
<TableCell className="text-sm">{row.action_type}</TableCell>
<TableCell className="text-right text-sm tabular-nums">
{formatAdminMinorUnits(row.amount)}
</TableCell>
<TableCell className="text-xs text-muted-foreground">
{row.source_reason ?? "—"}
</TableCell>
<TableCell className="font-mono text-xs">{row.ticket_no ?? "—"}</TableCell>
<TableCell className="text-xs">{row.play_code ?? "—"}</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</div>
{data ? (
<AdminListPaginationFooter
selectId={`risk-logs-${drawId}`}
total={data.meta.total}
page={data.meta.current_page}
lastPage={data.meta.last_page}
perPage={data.meta.per_page}
loading={loading}
onPerPageChange={(n) => {
setPerPage(n);
setPage(1);
}}
onPageChange={setPage}
/>
) : null}
</>
)}
</CardContent>
</Card>
);
}

View File

@@ -0,0 +1,195 @@
"use client";
import Link from "next/link";
import { useCallback, useEffect, useState } from "react";
import { getAdminRiskPoolDetail } from "@/api/admin-risk";
import { AdminListPaginationFooter } from "@/components/admin/admin-list-pagination-footer";
import { buttonVariants } from "@/components/ui/button";
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from "@/components/ui/table";
import { useAdminDateTimeFormatter } from "@/hooks/use-admin-datetime-formatter";
import { formatAdminMinorUnits } from "@/lib/money";
import { cn } from "@/lib/utils";
import { LotteryApiBizError } from "@/types/api/errors";
import type { AdminRiskPoolDetailLogRow, AdminRiskPoolShowData } from "@/types/api/admin-risk";
export function RiskPoolDetailConsole({
drawId,
number4d,
}: {
drawId: number;
number4d: string;
}) {
const formatDt = useAdminDateTimeFormatter();
const [page, setPage] = useState(1);
const [perPage, setPerPage] = useState(20);
const [data, setData] = useState<AdminRiskPoolShowData | null>(null);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const load = useCallback(async () => {
setLoading(true);
setError(null);
try {
const d = await getAdminRiskPoolDetail(drawId, number4d, { page, per_page: perPage });
setData(d);
} catch (e) {
const msg =
e instanceof LotteryApiBizError ? e.message : "加载风险池详情失败";
setError(msg);
setData(null);
} finally {
setLoading(false);
}
}, [drawId, number4d, page, perPage]);
useEffect(() => {
queueMicrotask(() => {
void load();
});
}, [load]);
if (error && !data) {
return (
<Card className="border-destructive/40">
<CardHeader>
<CardTitle className="text-lg"></CardTitle>
<CardDescription>{error}</CardDescription>
</CardHeader>
<CardContent>
<Link
href={`/admin/risk/draws/${drawId}/pools`}
className={cn(buttonVariants({ variant: "outline", size: "sm" }))}
>
</Link>
</CardContent>
</Card>
);
}
if (loading && !data) {
return <p className="text-sm text-muted-foreground"></p>;
}
if (!data) {
return null;
}
const { pool, logs } = data;
return (
<div className="space-y-6">
<div className="flex flex-wrap items-center justify-between gap-2">
<Link
href={`/admin/risk/draws/${drawId}/pools`}
className={cn(buttonVariants({ variant: "ghost", size: "sm" }))}
>
</Link>
</div>
<Card>
<CardHeader>
<CardTitle className="text-lg">
<span className="font-mono">{pool.normalized_number}</span>
</CardTitle>
<CardDescription>
{data.draw_no} · `risk_pools` 0 §6.4
</CardDescription>
</CardHeader>
<CardContent className="grid gap-3 sm:grid-cols-2 lg:grid-cols-4">
<div className="rounded-lg border bg-muted/40 p-3">
<p className="text-xs text-muted-foreground"></p>
<p className="mt-1 font-mono text-sm font-medium tabular-nums">
{formatAdminMinorUnits(pool.total_cap_amount)}
</p>
</div>
<div className="rounded-lg border bg-muted/40 p-3">
<p className="text-xs text-muted-foreground"></p>
<p className="mt-1 font-mono text-sm font-medium tabular-nums">
{formatAdminMinorUnits(pool.locked_amount)}
</p>
</div>
<div className="rounded-lg border bg-muted/40 p-3">
<p className="text-xs text-muted-foreground"></p>
<p className="mt-1 font-mono text-sm font-medium tabular-nums">
{formatAdminMinorUnits(pool.remaining_amount)}
</p>
</div>
<div className="rounded-lg border bg-muted/40 p-3">
<p className="text-xs text-muted-foreground"></p>
<p className="mt-1 text-sm font-medium">{pool.is_sold_out ? "是" : "否"}</p>
<p className="mt-0.5 text-xs text-muted-foreground">
{" "}
{pool.usage_ratio != null ? `${(pool.usage_ratio * 100).toFixed(2)}%` : "—"}
</p>
</div>
</CardContent>
</Card>
<Card>
<CardHeader>
<CardTitle className="text-base"> / </CardTitle>
<CardDescription></CardDescription>
</CardHeader>
<CardContent className="space-y-4">
<div className="overflow-x-auto rounded-md border">
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead className="text-right"></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
<TableHead></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{logs.items.map((row: AdminRiskPoolDetailLogRow) => (
<TableRow key={row.id}>
<TableCell className="whitespace-nowrap text-xs text-muted-foreground">
{row.created_at ? formatDt(row.created_at) : "—"}
</TableCell>
<TableCell className="text-sm">{row.action_type}</TableCell>
<TableCell className="text-right text-sm tabular-nums">
{formatAdminMinorUnits(row.amount)}
</TableCell>
<TableCell className="text-xs text-muted-foreground">
{row.source_reason ?? "—"}
</TableCell>
<TableCell className="font-mono text-xs">{row.ticket_no ?? "—"}</TableCell>
<TableCell className="text-xs">{row.play_code ?? "—"}</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</div>
<AdminListPaginationFooter
selectId={`risk-pool-detail-${drawId}-${number4d}`}
total={logs.meta.total}
page={logs.meta.current_page}
lastPage={logs.meta.last_page}
perPage={logs.meta.per_page}
loading={loading}
onPerPageChange={(n) => {
setPerPage(n);
setPage(1);
}}
onPageChange={setPage}
/>
</CardContent>
</Card>
</div>
);
}

View File

@@ -0,0 +1,196 @@
"use client";
import Link from "next/link";
import { useCallback, useEffect, useState } from "react";
import { getAdminRiskPools } from "@/api/admin-risk";
import { AdminListPaginationFooter } from "@/components/admin/admin-list-pagination-footer";
import { buttonVariants } from "@/components/ui/button";
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
import { Label } from "@/components/ui/label";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select";
import {
Table,
TableBody,
TableCell,
TableHead,
TableHeader,
TableRow,
} from "@/components/ui/table";
import { formatAdminMinorUnits } from "@/lib/money";
import { cn } from "@/lib/utils";
import { LotteryApiBizError } from "@/types/api/errors";
import type { AdminRiskPoolListData, AdminRiskPoolRow } from "@/types/api/admin-risk";
const SORT_OPTIONS: { value: "usage_desc" | "locked_desc" | "remaining_asc" | "number_asc"; label: string }[] =
[
{ value: "usage_desc", label: "占用比 ↓(热门)" },
{ value: "locked_desc", label: "已占用额 ↓" },
{ value: "remaining_asc", label: "剩余额 ↑(紧俏)" },
{ value: "number_asc", label: "号码 ↑" },
];
type RiskPoolsConsoleProps = {
drawId: number;
title: string;
description: string;
soldOutOnly: boolean;
defaultSort: "usage_desc" | "locked_desc" | "remaining_asc" | "number_asc";
allowSortChange?: boolean;
};
export function RiskPoolsConsole({
drawId,
title,
description,
soldOutOnly,
defaultSort,
allowSortChange = false,
}: RiskPoolsConsoleProps) {
const [sort, setSort] = useState(defaultSort);
const [page, setPage] = useState(1);
const [perPage, setPerPage] = useState(25);
const [data, setData] = useState<AdminRiskPoolListData | null>(null);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const load = useCallback(async () => {
setLoading(true);
setError(null);
try {
const d = await getAdminRiskPools(drawId, {
page,
per_page: perPage,
sold_out_only: soldOutOnly,
sort,
});
setData(d);
} catch (e) {
const msg =
e instanceof LotteryApiBizError ? e.message : "加载风险池失败";
setError(msg);
setData(null);
} finally {
setLoading(false);
}
}, [drawId, page, perPage, soldOutOnly, sort]);
useEffect(() => {
queueMicrotask(() => {
void load();
});
}, [load]);
return (
<Card>
<CardHeader className="space-y-2">
<CardTitle className="text-lg">{title}</CardTitle>
<CardDescription>{description}</CardDescription>
{allowSortChange ? (
<div className="flex max-w-xs flex-col gap-2 sm:flex-row sm:items-end">
<div className="space-y-1.5">
<Label htmlFor="risk-pool-sort"></Label>
<Select
modal={false}
value={sort}
onValueChange={(v) => {
if (!v) return;
setSort(v as typeof sort);
setPage(1);
}}
>
<SelectTrigger id="risk-pool-sort" size="sm" className="w-full sm:w-52">
<SelectValue />
</SelectTrigger>
<SelectContent>
{SORT_OPTIONS.map((o) => (
<SelectItem key={o.value} value={o.value}>
{o.label}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
</div>
) : null}
</CardHeader>
<CardContent className="space-y-4">
{error ? <p className="text-sm text-destructive">{error}</p> : null}
{loading && !data ? (
<p className="text-sm text-muted-foreground"></p>
) : (
<>
<div className="overflow-x-auto rounded-md border">
<Table>
<TableHeader>
<TableRow>
<TableHead></TableHead>
<TableHead className="text-right"></TableHead>
<TableHead className="text-right"></TableHead>
<TableHead className="text-right"></TableHead>
<TableHead className="text-right"></TableHead>
<TableHead></TableHead>
<TableHead className="text-right"></TableHead>
</TableRow>
</TableHeader>
<TableBody>
{(data?.items ?? []).map((row: AdminRiskPoolRow) => (
<TableRow key={row.normalized_number}>
<TableCell className="font-mono font-medium">{row.normalized_number}</TableCell>
<TableCell className="text-right text-sm tabular-nums">
{formatAdminMinorUnits(row.total_cap_amount)}
</TableCell>
<TableCell className="text-right text-sm tabular-nums">
{formatAdminMinorUnits(row.locked_amount)}
</TableCell>
<TableCell className="text-right text-sm tabular-nums">
{formatAdminMinorUnits(row.remaining_amount)}
</TableCell>
<TableCell className="text-right text-sm tabular-nums">
{row.usage_ratio != null ? `${(row.usage_ratio * 100).toFixed(2)}%` : "—"}
</TableCell>
<TableCell>{row.is_sold_out ? "是" : "否"}</TableCell>
<TableCell className="text-right">
<Link
href={`/admin/risk/draws/${drawId}/pools/${row.normalized_number}`}
className={cn(
buttonVariants({ variant: "link", size: "sm" }),
"h-auto p-0",
)}
>
</Link>
</TableCell>
</TableRow>
))}
</TableBody>
</Table>
</div>
{data ? (
<AdminListPaginationFooter
selectId={`risk-pools-${drawId}-${soldOutOnly ? "so" : "all"}`}
total={data.meta.total}
page={data.meta.current_page}
lastPage={data.meta.last_page}
perPage={data.meta.per_page}
loading={loading}
onPerPageChange={(n) => {
setPerPage(n);
setPage(1);
}}
onPageChange={setPage}
/>
) : null}
</>
)}
</CardContent>
</Card>
);
}

View File

@@ -0,0 +1,46 @@
"use client";
import Link from "next/link";
import { usePathname } from "next/navigation";
import { buttonVariants } from "@/components/ui/button";
import { cn } from "@/lib/utils";
const segments = [
{ suffix: "/occupancy", key: "occupancy", label: "风险占用" },
{ suffix: "/hot", key: "hot", label: "热门号码" },
{ suffix: "/sold-out", key: "sold-out", label: "售罄列表" },
{ suffix: "/pools", key: "pools", label: "全部风险池" },
] as const;
export function RiskSubnav({ drawId }: { drawId: string }) {
const pathname = usePathname();
const base = `/admin/risk/draws/${drawId}`;
return (
<nav className="mb-6 flex flex-wrap gap-2 border-b border-border pb-3">
{segments.map(({ suffix, key, label }) => {
const href = `${base}${suffix}`;
const active =
pathname === href ||
(key === "pools" && pathname?.startsWith(`${base}/pools`));
return (
<Link
key={key}
href={href}
className={cn(buttonVariants({ variant: active ? "default" : "outline", size: "sm" }))}
>
{label}
</Link>
);
})}
<Link
href="/admin/risk"
className={cn(buttonVariants({ variant: "ghost", size: "sm" }), "ml-auto")}
>
</Link>
</nav>
);
}

View File

@@ -0,0 +1,66 @@
export type AdminRiskPoolListMeta = {
current_page: number;
per_page: number;
total: number;
last_page: number;
};
export type AdminRiskPoolRow = {
normalized_number: string;
total_cap_amount: number;
locked_amount: number;
remaining_amount: number;
sold_out_status: number;
is_sold_out: boolean;
usage_ratio: number | null;
version: number;
};
export type AdminRiskPoolListData = {
draw_id: number;
draw_no: string;
items: AdminRiskPoolRow[];
meta: AdminRiskPoolListMeta;
};
export type AdminRiskLockLogRow = {
id: number;
normalized_number: string;
action_type: string;
amount: number;
source_reason: string | null;
ticket_item_id: number | null;
ticket_no: string | null;
play_code: string | null;
player_id: number | null;
created_at: string | null;
};
export type AdminRiskLockLogListData = {
draw_id: number;
draw_no: string;
items: AdminRiskLockLogRow[];
meta: AdminRiskPoolListMeta;
};
export type AdminRiskPoolDetailLogRow = {
id: number;
action_type: string;
amount: number;
source_reason: string | null;
ticket_item_id: number | null;
ticket_no: string | null;
play_code: string | null;
player_id: number | null;
created_at: string | null;
};
export type AdminRiskPoolShowData = {
draw_id: number;
draw_no: string;
pool: AdminRiskPoolRow;
logs: {
items: AdminRiskPoolDetailLogRow[];
meta: AdminRiskPoolListMeta;
};
};