feat(integration): 为集成站点与开奖管理新增 AdminPermissionGate 权限控制

使用 AdminPermissionGate 包裹集成站点与开奖相关组件,根据权限进行访问控制。
新增集成管理与开奖管理相关权限常量。
更新相关 UI 组件以适配权限校验逻辑,提升系统安全性与用户体验。
增强国际化支持,在英文、尼泊尔语与中文语言包中新增集成相关文案。
This commit is contained in:
2026-05-27 16:51:48 +08:00
parent 5eabbcf0ee
commit 788c7998eb
24 changed files with 276 additions and 64 deletions

View File

@@ -1,6 +1,8 @@
import { AdminPermissionGate } from "@/components/admin/admin-permission-gate";
import { ModuleScaffold } from "@/components/admin/module-scaffold";
import { IntegrationSitesConsole } from "@/modules/integration/integration-sites-console";
import { buildPageMetadata } from "@/lib/page-metadata";
import { PRD_INTEGRATION_ACCESS_ANY } from "@/lib/admin-prd";
import type { Metadata } from "next";
export const metadata: Metadata = buildPageMetadata("config", "integrationSites.title");
@@ -8,7 +10,9 @@ export const metadata: Metadata = buildPageMetadata("config", "integrationSites.
export default function AdminIntegrationSitesPage() {
return (
<ModuleScaffold>
<IntegrationSitesConsole />
<AdminPermissionGate requiredAny={PRD_INTEGRATION_ACCESS_ANY}>
<IntegrationSitesConsole />
</AdminPermissionGate>
</ModuleScaffold>
);
}

View File

@@ -1,4 +1,6 @@
import { AdminPermissionGate } from "@/components/admin/admin-permission-gate";
import { DrawFinanceConsole } from "@/modules/draws/draw-finance-console";
import { PRD_DRAW_ACCESS_ANY } from "@/lib/admin-prd";
import { buildPageMetadata } from "@/lib/page-metadata";
import type { Metadata } from "next";
@@ -8,5 +10,9 @@ export default async function AdminDrawFinancePage(props: {
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
return <DrawFinanceConsole drawId={drawId} />;
return (
<AdminPermissionGate requiredAny={PRD_DRAW_ACCESS_ANY}>
<DrawFinanceConsole drawId={drawId} />
</AdminPermissionGate>
);
}

View File

@@ -1,8 +1,14 @@
import { AdminPermissionGate } from "@/components/admin/admin-permission-gate";
import { DrawDetailConsole } from "@/modules/draws/draw-detail-console";
import { PRD_DRAW_ACCESS_ANY } from "@/lib/admin-prd";
export default async function AdminDrawDetailPage(props: {
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
return <DrawDetailConsole drawId={drawId} />;
return (
<AdminPermissionGate requiredAny={PRD_DRAW_ACCESS_ANY}>
<DrawDetailConsole drawId={drawId} />
</AdminPermissionGate>
);
}

View File

@@ -1,8 +1,14 @@
import { AdminPermissionGate } from "@/components/admin/admin-permission-gate";
import { DrawResultsConsole } from "@/modules/draws/draw-results-console";
import { PRD_DRAW_ACCESS_ANY } from "@/lib/admin-prd";
export default async function AdminDrawResultsPage(props: {
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
return <DrawResultsConsole drawId={drawId} />;
return (
<AdminPermissionGate requiredAny={PRD_DRAW_ACCESS_ANY}>
<DrawResultsConsole drawId={drawId} />
</AdminPermissionGate>
);
}

View File

@@ -1,8 +1,14 @@
import { AdminPermissionGate } from "@/components/admin/admin-permission-gate";
import { DrawReviewConsole } from "@/modules/draws/draw-review-console";
import { PRD_DRAW_ACCESS_ANY } from "@/lib/admin-prd";
export default async function AdminDrawReviewPage(props: {
params: Promise<{ drawId: string }>;
}) {
const { drawId } = await props.params;
return <DrawReviewConsole drawId={drawId} />;
return (
<AdminPermissionGate requiredAny={PRD_DRAW_ACCESS_ANY}>
<DrawReviewConsole drawId={drawId} />
</AdminPermissionGate>
);
}

View File

@@ -1,5 +1,7 @@
import { ModuleScaffold } from "@/components/admin/module-scaffold";
import { AdminPermissionGate } from "@/components/admin/admin-permission-gate";
import { DrawsIndexConsole } from "@/modules/draws/draws-index-console";
import { PRD_DRAW_ACCESS_ANY } from "@/lib/admin-prd";
import { buildPageMetadata } from "@/lib/page-metadata";
import type { Metadata } from "next";
@@ -7,8 +9,10 @@ export const metadata: Metadata = buildPageMetadata("draws", "statusListTitle");
export default function AdminDrawsPage() {
return (
<ModuleScaffold>
<DrawsIndexConsole />
</ModuleScaffold>
<AdminPermissionGate requiredAny={PRD_DRAW_ACCESS_ANY}>
<ModuleScaffold>
<DrawsIndexConsole />
</ModuleScaffold>
</AdminPermissionGate>
);
}

View File

@@ -1,5 +1,7 @@
import { AdminPermissionGate } from "@/components/admin/admin-permission-gate";
import { InvalidSettlementBatchId } from "@/modules/settlement/invalid-settlement-batch-id";
import { SettlementBatchDetailsConsole } from "@/modules/settlement/settlement-batch-details-console";
import { PRD_PAYOUT_ACCESS_ANY } from "@/lib/admin-prd";
import { buildPageMetadata } from "@/lib/page-metadata";
import type { Metadata } from "next";
@@ -14,5 +16,9 @@ export default async function AdminSettlementBatchDetailsPage(props: {
return <InvalidSettlementBatchId />;
}
return <SettlementBatchDetailsConsole batchId={id} />;
return (
<AdminPermissionGate requiredAny={PRD_PAYOUT_ACCESS_ANY}>
<SettlementBatchDetailsConsole batchId={id} />
</AdminPermissionGate>
);
}

View File

@@ -1,9 +1,15 @@
import { AdminPermissionGate } from "@/components/admin/admin-permission-gate";
import { SettlementBatchesConsole } from "@/modules/settlement/settlement-batches-console";
import { PRD_PAYOUT_ACCESS_ANY } from "@/lib/admin-prd";
import { buildPageMetadata } from "@/lib/page-metadata";
import type { Metadata } from "next";
export const metadata: Metadata = buildPageMetadata("settlement", "batchList");
export default function AdminSettlementBatchesPage() {
return <SettlementBatchesConsole />;
return (
<AdminPermissionGate requiredAny={PRD_PAYOUT_ACCESS_ANY}>
<SettlementBatchesConsole />
</AdminPermissionGate>
);
}

View File

@@ -40,6 +40,8 @@ const NAV_TRANSLATION_KEYS: Record<string, string> = {
tickets: "tickets",
audit: "audit",
settings: "settings",
integration: "integration",
config: "config",
};
const RULES_ROUTE_LABELS: Record<string, string> = {
@@ -55,6 +57,16 @@ const SETTINGS_ROUTE_LABELS: Record<string, string> = {
currencies: "currencies.title",
};
const CONFIG_ROUTE_LABELS: Record<string, string> = {
"integration-sites": "integrationSites.title",
plays: "nav.items.plays",
odds: "nav.items.odds",
rebate: "nav.items.rebate",
jackpot: "nav.items.jackpot",
"risk-cap": "nav.items.risk-cap",
wallet: "wallet.title",
};
function titleCase(value: string): string {
return value
.split("-")
@@ -146,6 +158,11 @@ export function AdminBreadcrumb() {
ns: "config",
defaultValue: titleCase(subSegment),
});
} else if (businessSegment === "config" && subSegment) {
const key = CONFIG_ROUTE_LABELS[subSegment];
subLabel = key
? t(key, { ns: "config", defaultValue: titleCase(subSegment) })
: titleCase(subSegment);
} else {
subLabel = subSegment
? t(`subnav.${subSegment}`, {

View File

@@ -4,6 +4,7 @@ import { useCallback, useEffect, useState } from "react";
import { getAdminIntegrationSites } from "@/api/admin-integration-sites";
import { adminHasAnyPermission } from "@/lib/admin-permissions";
import { PRD_INTEGRATION_ACCESS_ANY } from "@/lib/admin-prd";
import { useAdminProfile } from "@/stores/admin-session";
export type AdminSiteCodeOption = {
@@ -21,10 +22,7 @@ export function useAdminSiteCodeOptions(): {
reload: () => Promise<void>;
} {
const profile = useAdminProfile();
const canLoad = adminHasAnyPermission(profile?.permissions, [
"prd.integration.view",
"prd.integration.manage",
]);
const canLoad = adminHasAnyPermission(profile?.permissions, PRD_INTEGRATION_ACCESS_ANY);
const [sites, setSites] = useState<AdminSiteCodeOption[]>([]);
const [loading, setLoading] = useState(false);

View File

@@ -147,7 +147,9 @@
"tickets": "Ticket list",
"audit": "Audit Logs",
"settings": "Settings",
"account": "Account settings"
"account": "Account settings",
"integration": "Integration sites",
"config": "Operations config"
},
"sidebar": {
"workspace": "Workspace"

View File

@@ -147,7 +147,9 @@
"tickets": "टिकट सूची",
"audit": "अडिट लग",
"settings": "सेटिङ",
"account": "खाता सेटिङ"
"account": "खाता सेटिङ",
"integration": "मुख्य साइट एकीकरण",
"config": "सञ्चालन कन्फिगरेसन"
},
"sidebar": {
"workspace": "कार्यस्थान"

View File

@@ -29,7 +29,77 @@
"jackpotTitle": "Jackpot",
"jackpotDesc": "पूल प्यारामिटर र लेजर",
"riskCapTitle": "जोखिम क्याप",
"riskCapDesc": "नम्बर क्याप र ओगट उपस्थिति"
"riskCapDesc": "नम्बर क्याप र ओगट उपस्थिति",
"integrationTitle": "मुख्य साइट एकीकरण",
"integrationDesc": "site_code, JWT गोप्य, पार्टनर वालेट URL र iframe श्वेतसूची"
},
"integrationSites": {
"title": "मुख्य साइट एकीकरण साइटहरू",
"description": "एडमिनमा पार्टनर एकीकरण सेटिङ मिलाउनुहोस्। site_code सिर्जना पछि परिवर्तन गर्न मिल्दैन।",
"create": "नयाँ साइट",
"edit": "सम्पादन",
"save": "बचत",
"saving": "बचत हुँदैछ…",
"cancel": "रद्द",
"copy": "प्रतिलिपि",
"loading": "लोड हुँदैछ…",
"empty": "कुनै एकीकरण साइट छैन",
"loadFailed": "एकीकरण साइट लोड असफल",
"saveFailed": "बचत असफल",
"createSuccess": "साइट {{code}} सिर्जना भयो",
"updateSuccess": "साइट {{code}} अद्यावधिक भयो",
"connectivityTest": "जडान परीक्षण",
"connectivityTitle": "पार्टनर वालेट जडान परीक्षण",
"connectivityDescription": "परीक्षण खेलाडीबाट साइट {{code}} को balance API कल गर्नुहोस्।",
"connectivityPlayerId": "परीक्षण site_player_id",
"connectivityRun": "परीक्षण सुरु",
"connectivityRunning": "परीक्षण हुँदैछ…",
"connectivitySuccess": "जडान सफल",
"connectivityFailed": "जडान असफल",
"exportParams": "प्यारामिटर निर्यात",
"exportSuccess": "{{code}} को प्यारामिटर चिट्ठा निर्यात भयो",
"exportFailed": "निर्यात असफल",
"rotateSecrets": "गोप्य कुञ्जी पुनः सिर्जना",
"rotateSuccess": "साइट {{code}} का गोप्य कुञ्जी पुनः सिर्जना भयो",
"rotateFailed": "गोप्य कुञ्जी पुनः सिर्जना असफल",
"rotateConfirmTitle": "गोप्य कुञ्जी पुनः सिर्जना गर्ने?",
"rotateConfirmDescription": "साइट {{code}} का नयाँ SSO र वालेट कुञ्जी सिर्जना हुन्छ। पुराना कुञ्जी तुरुन्त अमान्य हुन्छन्।",
"rotateConfirm": "पुष्टि",
"secretsTitle": "गोप्य कुञ्जी अहिले नै सुरक्षित राख्नुहोस्",
"secretsDescription": "साइट {{code}} का गोप्य कुञ्जी एक पटक मात्र देखिन्छ।",
"secretsDismiss": "सुरक्षित गरिसके",
"copied": "{{field}} प्रतिलिपि भयो",
"copyFailed": "प्रतिलिपि असफल",
"noPermission": "एकीकरण साइट हेर्ने अनुमति छैन",
"codeImmutable": "site_code सिर्जना पछि परिवर्तन गर्न मिल्दैन",
"statusEnabled": "सक्रिय",
"statusDisabled": "निष्क्रिय",
"dialogCreateTitle": "नयाँ एकीकरण साइट",
"dialogEditTitle": "एकीकरण साइट सम्पादन",
"dialogDescription": "पार्टनरले अनुकूल URL नभएसम्म पूर्वनिर्धारित वालेट path प्रयोग गर्न सकिन्छ।",
"form": {
"required": "साइट नाम अनिवार्य छ",
"codeRequired": "site_code अनिवार्य छ"
},
"columns": {
"code": "site_code",
"name": "नाम",
"status": "स्थिति",
"walletUrl": "वालेट API",
"actions": "कार्य"
},
"fields": {
"code": "site_code",
"name": "साइट नाम",
"currency": "पूर्वनिर्धारित मुद्रा",
"status": "स्थिति",
"walletApiUrl": "पार्टनर वालेट आधार URL",
"lotteryH5BaseUrl": "लटरी H5 आधार URL (वैकल्पिक)",
"iframeOrigins": "iframe श्वेतसूची (प्रति लाइन एक origin)",
"notes": "टिप्पणी",
"ssoSecret": "SSO गोप्य",
"walletApiKey": "वालेट API कुञ्जी"
}
},
"versionStatus": {
"active": "सक्रिय",

View File

@@ -147,7 +147,9 @@
"tickets": "注单列表",
"audit": "审计日志",
"settings": "系统设置",
"account": "账号设置"
"account": "账号设置",
"integration": "主站接入站点",
"config": "运营配置"
},
"sidebar": {
"workspace": "工作台"

View File

@@ -0,0 +1,41 @@
import { PRD_INTEGRATION_MANAGE, PRD_INTEGRATION_VIEW } from "@/lib/admin-prd";
export type AdminPermissionBundleKey = "view" | "manage" | "audit" | "export" | "privilege";
export type AdminPageKey = "integration-sites";
/**
* “页面权限包”是把运营/管理员能理解的词汇(查看/管理/审核/导出/特权)
* 映射到系统真实用的 `prd.*` 权限 slug。
*
* 目前只落地 integration-sites其它页面按同样方式逐步接入。
*/
export const ADMIN_PERMISSION_BUNDLES = {
"integration-sites": {
view: [PRD_INTEGRATION_VIEW] as const,
manage: [PRD_INTEGRATION_MANAGE] as const,
audit: [] as const,
// 导出接口的资源鉴权仍落在 view/manage因此这里复用 view。
export: [PRD_INTEGRATION_VIEW] as const,
privilege: [] as const,
},
} satisfies Record<AdminPageKey, Record<AdminPermissionBundleKey, readonly string[]>>;
export const ADMIN_PAGE_REQUIRED_ANY = {
"integration-sites": [
...ADMIN_PERMISSION_BUNDLES["integration-sites"].view,
...ADMIN_PERMISSION_BUNDLES["integration-sites"].manage,
] as const,
} satisfies Record<AdminPageKey, readonly string[]>;
export function getAdminPageRequiredAny(page: AdminPageKey): readonly string[] {
return ADMIN_PAGE_REQUIRED_ANY[page];
}
export function getAdminPageBundle(
page: AdminPageKey,
bundle: AdminPermissionBundleKey,
): readonly string[] {
return ADMIN_PERMISSION_BUNDLES[page][bundle] ?? [];
}

View File

@@ -10,6 +10,10 @@ export const PRD_PLAYER_FREEZE_MANAGE = "prd.player_freeze.manage" as const;
export const PRD_CURRENCY_MANAGE = "prd.currency.manage" as const;
/** 接入站点integration-sites */
export const PRD_INTEGRATION_VIEW = "prd.integration.view" as const;
export const PRD_INTEGRATION_MANAGE = "prd.integration.manage" as const;
export const PRD_WALLET_RECONCILE_MANAGE = "prd.wallet_reconcile.manage" as const;
export const PRD_WALLET_RECONCILE_VIEW = "prd.wallet_reconcile.view" as const;
export const PRD_WALLET_RECONCILE_VIEW_CS = "prd.wallet_reconcile.view_cs" as const;
@@ -105,8 +109,25 @@ export const PRD_RULES_ODDS_ACCESS_ANY = [
PRD_REBATE_VIEW,
] as const;
/** 开奖页面入口 */
export const PRD_DRAW_ACCESS_ANY = [
PRD_DRAW_RESULT_VIEW,
PRD_DRAW_RESULT_MANAGE,
PRD_DRAW_REOPEN_MANAGE,
] as const;
/** 封顶配置页 */
export const PRD_RISK_CAP_ACCESS_ANY = [PRD_RISK_CAP_MANAGE, PRD_RISK_CAP_VIEW] as const;
/** Jackpot 配置页 */
export const PRD_JACKPOT_ACCESS_ANY = [PRD_JACKPOT_MANAGE, PRD_JACKPOT_VIEW] as const;
/** 派彩 / 结算页面入口 */
export const PRD_PAYOUT_ACCESS_ANY = [
PRD_PAYOUT_VIEW,
PRD_PAYOUT_REVIEW,
PRD_PAYOUT_MANAGE,
] as const;
/** 接入站点配置页 */
export const PRD_INTEGRATION_ACCESS_ANY = [PRD_INTEGRATION_VIEW, PRD_INTEGRATION_MANAGE] as const;

View File

@@ -6,6 +6,7 @@ import { ChevronRight } from "lucide-react";
import { ModuleScaffold } from "@/components/admin/module-scaffold";
import { Card, CardContent, CardDescription, CardHeader, CardTitle } from "@/components/ui/card";
import { PRD_INTEGRATION_ACCESS_ANY } from "@/lib/admin-prd";
import { useAdminProfile } from "@/stores/admin-session";
import { adminHasAnyPermission } from "@/lib/admin-permissions";
@@ -45,7 +46,7 @@ const HUB_CARDS: HubCard[] = [
href: "/admin/config/integration-sites",
titleKey: "hub.integrationTitle",
descKey: "hub.integrationDesc",
requiredAny: ["prd.integration.view", "prd.integration.manage"],
requiredAny: PRD_INTEGRATION_ACCESS_ANY,
},
];

View File

@@ -27,6 +27,7 @@ import type { AdminDrawFinanceSummaryData } from "@/types/api/admin-draw-finance
import { toast } from "sonner";
import { useAdminCurrencyCatalog } from "@/hooks/use-admin-currency-catalog";
import { useAdminDateTimeFormatter } from "@/hooks/use-admin-datetime-formatter";
import { useConfirmAction } from "@/hooks/use-confirm-action";
import { useExportLabels } from "@/hooks/use-export-labels";
import { formatAdminMinorUnits } from "@/lib/money";
@@ -44,6 +45,7 @@ export function DrawFinanceConsole({ drawId }: { drawId: string }): React.ReactE
PRD_PAYOUT_REVIEW,
]);
const [data, setData] = useState<AdminDrawFinanceSummaryData | null>(null);
const formatTs = useAdminDateTimeFormatter();
const exportLabels = useExportLabels("drawFinance", { drawNo: data?.draw_no ?? drawId });
const [err, setErr] = useState<string | null>(null);
const [loading, setLoading] = useState(true);
@@ -219,8 +221,8 @@ export function DrawFinanceConsole({ drawId }: { drawId: string }): React.ReactE
<TableCell className="text-center tabular-nums text-xs">
{formatMoney(b.total_jackpot_payout_amount)}
</TableCell>
<TableCell className="font-mono text-[11px] text-muted-foreground">
{b.finished_at ?? "—"}
<TableCell className="whitespace-nowrap text-xs text-muted-foreground">
{formatTs(b.finished_at)}
</TableCell>
</TableRow>
))}

View File

@@ -15,6 +15,7 @@ import {
TableHeader,
TableRow,
} from "@/components/ui/table";
import { useAdminDateTimeFormatter } from "@/hooks/use-admin-datetime-formatter";
import { cn } from "@/lib/utils";
import { adminHasAnyPermission } from "@/lib/admin-permissions";
import { useAdminProfile } from "@/stores/admin-session";
@@ -103,6 +104,7 @@ export function DrawResultsConsole({ drawId }: { drawId: string }) {
function BatchTable({ batch }: { batch: AdminDrawBatchRow }) {
const { t } = useTranslation("draws");
const formatDt = useAdminDateTimeFormatter();
return (
<Card>
<CardHeader className="pb-2">
@@ -112,7 +114,7 @@ function BatchTable({ batch }: { batch: AdminDrawBatchRow }) {
source: batch.source_type === "manual" ? t("manualEntry") : t("rng"),
})}{" "}
· {t("rngSummary", { hash: batch.rng_seed_hash ?? "—" })} ·{" "}
{t("confirmedAt", { time: batch.confirmed_at ?? "—" })}
{t("confirmedAt", { time: formatDt(batch.confirmed_at) })}
</p>
</CardHeader>
<CardContent className="overflow-x-auto pt-0">

View File

@@ -36,11 +36,15 @@ import {
} from "@/components/ui/table";
import { Textarea } from "@/components/ui/textarea";
import { adminHasAnyPermission } from "@/lib/admin-permissions";
import { getAdminPageBundle } from "@/lib/admin-permission-bundles";
import { useAdminProfile } from "@/stores/admin-session";
import { LotteryApiBizError } from "@/types/api/errors";
import type {
AdminIntegrationSiteCreatePayload,
AdminIntegrationSiteConnectivityResult,
AdminIntegrationSiteUpdatePayload,
AdminIntegrationSiteRow,
AdminIntegrationSiteDetail,
AdminIntegrationSiteSecrets,
AdminIntegrationSiteWithSecrets,
} from "@/types/api/admin-integration-site";
@@ -86,7 +90,7 @@ function textToOrigins(text: string): string[] {
.filter(Boolean);
}
function rowToForm(row: AdminIntegrationSiteRow & Partial<FormState>): FormState {
function rowToForm(row: AdminIntegrationSiteDetail): FormState {
return {
code: row.code,
name: row.name,
@@ -103,7 +107,12 @@ function rowToForm(row: AdminIntegrationSiteRow & Partial<FormState>): FormState
};
}
function formToPayload(form: FormState, includeCode: boolean) {
function formToPayload(form: FormState, includeCode: true): AdminIntegrationSiteCreatePayload;
function formToPayload(form: FormState, includeCode: false): AdminIntegrationSiteUpdatePayload;
function formToPayload(
form: FormState,
includeCode: boolean,
): AdminIntegrationSiteCreatePayload | AdminIntegrationSiteUpdatePayload {
const base = {
name: form.name.trim(),
currency_code: form.currency_code.trim() || "NPR",
@@ -128,11 +137,10 @@ function formToPayload(form: FormState, includeCode: boolean) {
export function IntegrationSitesConsole() {
const { t } = useTranslation("config");
const profile = useAdminProfile();
const canView = adminHasAnyPermission(profile?.permissions, [
"prd.integration.view",
"prd.integration.manage",
]);
const canManage = adminHasAnyPermission(profile?.permissions, ["prd.integration.manage"]);
const canManage = adminHasAnyPermission(
profile?.permissions,
getAdminPageBundle("integration-sites", "manage"),
);
const [items, setItems] = useState<AdminIntegrationSiteRow[]>([]);
const [loading, setLoading] = useState(true);
@@ -158,12 +166,6 @@ export function IntegrationSitesConsole() {
const [exportBusyId, setExportBusyId] = useState<number | null>(null);
const load = useCallback(async () => {
if (!canView) {
setItems([]);
setLoading(false);
return;
}
setLoading(true);
try {
const data = await getAdminIntegrationSites();
@@ -175,7 +177,7 @@ export function IntegrationSitesConsole() {
} finally {
setLoading(false);
}
}, [canView, t]);
}, [t]);
useEffect(() => {
queueMicrotask(() => {
@@ -334,14 +336,6 @@ export function IntegrationSitesConsole() {
}
}
if (!canView) {
return (
<AdminPageCard title={t("integrationSites.title")}>
<p className="text-sm text-muted-foreground">{t("integrationSites.noPermission")}</p>
</AdminPageCard>
);
}
return (
<>
<AdminPageCard
@@ -377,13 +371,12 @@ export function IntegrationSitesConsole() {
<TableCell>{row.name}</TableCell>
<TableCell>
<AdminStatusBadge
tone={row.status === 1 ? "success" : "muted"}
label={
row.status === 1
? t("integrationSites.statusEnabled")
: t("integrationSites.statusDisabled")
}
/>
tone={row.status === 1 ? "success" : "neutral"}
>
{row.status === 1
? t("integrationSites.statusEnabled")
: t("integrationSites.statusDisabled")}
</AdminStatusBadge>
</TableCell>
<TableCell className="max-w-[240px] truncate text-xs text-muted-foreground">
{row.wallet_api_url ?? "—"}
@@ -407,9 +400,16 @@ export function IntegrationSitesConsole() {
>
{t("integrationSites.exportParams")}
</Button>
<Button type="button" variant="outline" size="sm" onClick={() => void openEdit(row)}>
{t("integrationSites.edit")}
</Button>
{canManage ? (
<Button
type="button"
variant="outline"
size="sm"
onClick={() => void openEdit(row)}
>
{t("integrationSites.edit")}
</Button>
) : null}
{canManage ? (
<Button
type="button"

View File

@@ -286,9 +286,10 @@ export function JackpotPoolsConsole({ embedded = false }: JackpotPoolsConsolePro
<Label>{t("adjustmentDirection")}</Label>
<Select
value={adj.direction}
onValueChange={(value: "increase" | "decrease") =>
updateAdjustmentDraft(p.id, { direction: value })
}
onValueChange={(value: "increase" | "decrease" | null) => {
if (value === null) return;
updateAdjustmentDraft(p.id, { direction: value });
}}
>
<SelectTrigger className="w-full min-w-0 sm:max-w-[12rem]">
<SelectValue>

View File

@@ -305,7 +305,10 @@ export function PlayersConsole(): React.ReactElement {
{canChooseSite ? (
<div className="admin-list-field">
<Label className="sm:w-20 sm:shrink-0">{t("filterSite")}</Label>
<Select value={siteCode || "__all__"} onValueChange={(v) => setSiteCode(v === "__all__" ? "" : v)}>
<Select
value={siteCode || "__all__"}
onValueChange={(v) => setSiteCode(v === "__all__" ? "" : v ?? "")}
>
<SelectTrigger className="w-full sm:w-[12rem]">
<SelectValue placeholder={t("filterAllSites")} />
</SelectTrigger>

View File

@@ -73,6 +73,8 @@ import {
} from "@/components/ui/table";
import { useAdminCurrencyCatalog } from "@/hooks/use-admin-currency-catalog";
import { useAdminDateTimeFormatter } from "@/hooks/use-admin-datetime-formatter";
import { formatAdminInstant } from "@/lib/admin-datetime";
import { getAdminRequestLocale } from "@/lib/admin-locale";
import { cn } from "@/lib/utils";
import { formatAdminMinorUnits } from "@/lib/money";
import { LotteryApiBizError } from "@/types/api/errors";
@@ -249,6 +251,10 @@ function normalizeFilenamePart(value: string): string {
return value.trim().replace(/[\\/:*?"<>|\s]+/g, "-").replace(/-+/g, "-").replace(/^-|-$/g, "");
}
function formatExportInstant(iso: string | null | undefined): ExportCell {
return formatAdminInstant(iso, { locale: getAdminRequestLocale() });
}
function toCsvValue(value: ExportCell): string {
if (value == null) {
return "";
@@ -369,7 +375,7 @@ function drawRowsFromSummary(summary: AdminDrawFinanceSummaryData): ExportRow[]
total_win_count: batch.total_win_count,
total_payout_amount: batch.total_payout_amount,
total_jackpot_payout_amount: batch.total_jackpot_payout_amount,
finished_at: batch.finished_at,
finished_at: formatExportInstant(batch.finished_at),
})),
];
}
@@ -575,8 +581,8 @@ export function ReportsConsole() {
status: item.status,
external_ref_no: item.external_ref_no,
fail_reason: item.fail_reason,
created_at: item.created_at,
finished_at: item.finished_at,
created_at: formatExportInstant(item.created_at),
finished_at: formatExportInstant(item.finished_at),
}));
setResult({
key: "player_transfer",
@@ -625,7 +631,7 @@ export function ReportsConsole() {
ticket_no: item.ticket_no,
play_code: item.play_code,
player_id: item.player_id,
created_at: item.created_at,
created_at: formatExportInstant(item.created_at),
})),
];
setResult({
@@ -737,7 +743,7 @@ export function ReportsConsole() {
target_id: item.target_id,
ip: item.ip,
user_agent: item.user_agent,
created_at: item.created_at,
created_at: formatExportInstant(item.created_at),
}));
setResult({
key: "admin_audit",

View File

@@ -194,7 +194,7 @@ export function PlayerTicketsConsole(): React.ReactElement {
onValueChange={(v) =>
setDraft((current) => ({
...current,
siteCode: v === "__all__" ? "" : v,
siteCode: v === "__all__" ? "" : (v ?? ""),
}))
}
>