feat(admin, i18n): enhance admin dashboard and user management with new features and translations

Added the ability to filter admin dashboard data by site code and agent node ID, improving data retrieval capabilities. Introduced new functions for fetching dashboard data based on these parameters. Updated the admin users and roles management components to reflect these changes. Enhanced multi-language support by adding new translations for agent management and permission levels in English, Nepali, and Chinese, ensuring a consistent user experience across the admin interface.
This commit is contained in:
2026-06-03 10:07:51 +08:00
parent b15e377187
commit ce27a3ec8a
66 changed files with 1361 additions and 720 deletions

View File

@@ -16,7 +16,6 @@ import {
postAdminRejectSettlementBatch,
} from "@/api/admin-settlement";
import { AdminListPaginationFooter } from "@/components/admin/admin-list-pagination-footer";
import { AdminAgentFilter } from "@/components/admin/admin-agent-filter";
import { AdminAgentIdentityCells, AdminAgentIdentityHeads } from "@/components/admin/admin-agent-columns";
import { AdminPlayerIdentityCells, AdminPlayerIdentityHeads } from "@/components/admin/admin-player-identity-columns";
import { AdminStatusBadge } from "@/components/admin/admin-status-badge";
@@ -90,8 +89,7 @@ export function SettlementBatchDetailsConsole({ batchId }: Props) {
const [err, setErr] = useState<string | null>(null);
const [page, setPage] = useState(1);
const [perPage, setPerPage] = useState(10);
const [agentNodeId, setAgentNodeId] = useState<number | undefined>(undefined);
const [appliedAgentNodeId, setAppliedAgentNodeId] = useState<number | undefined>(undefined);
const [appliedAgentNodeId] = useState<number | undefined>(undefined);
const [acting, setActing] = useState<string | null>(null);
const [pendingAction, setPendingAction] = useState<SettlementAction | null>(null);
const [reviewRemark, setReviewRemark] = useState("");
@@ -343,24 +341,6 @@ export function SettlementBatchDetailsConsole({ batchId }: Props) {
<CardContent>
{details ? (
<>
<div className="mb-4 flex flex-wrap items-end gap-3">
<AdminAgentFilter
id="settlement-details-agent-filter"
className="w-[14rem]"
value={agentNodeId}
onChange={setAgentNodeId}
/>
<Button
type="button"
size="sm"
onClick={() => {
setAppliedAgentNodeId(agentNodeId);
setPage(1);
}}
>
{t("search", { ns: "common", defaultValue: "Search" })}
</Button>
</div>
<Table id={`settlement-details-table-${batchId}`}>
<TableHeader>
<TableRow>

View File

@@ -14,7 +14,6 @@ import {
postAdminPayoutSettlementBatch,
postAdminRejectSettlementBatch,
} from "@/api/admin-settlement";
import { AdminAgentFilter } from "@/components/admin/admin-agent-filter";
import { AdminListPaginationFooter } from "@/components/admin/admin-list-pagination-footer";
import { AdminRowActionsMenu } from "@/components/admin/admin-row-actions-menu";
import { AdminStatusBadge } from "@/components/admin/admin-status-badge";
@@ -104,8 +103,6 @@ export function SettlementBatchesConsole() {
const [appliedDrawNo, setAppliedDrawNo] = useState("");
const [draftStatus, setDraftStatus] = useState(STATUS_ALL);
const [appliedStatus, setAppliedStatus] = useState(STATUS_ALL);
const [agentNodeId, setAgentNodeId] = useState<number | undefined>(undefined);
const [appliedAgentNodeId, setAppliedAgentNodeId] = useState<number | undefined>(undefined);
const [page, setPage] = useState(1);
const [perPage, setPerPage] = useState(10);
const [actingId, setActingId] = useState<number | null>(null);
@@ -124,7 +121,6 @@ export function SettlementBatchesConsole() {
appliedStatus === STATUS_ALL || appliedStatus.trim() === ""
? undefined
: appliedStatus.trim(),
agent_node_id: appliedAgentNodeId,
});
setData(d);
} catch (e) {
@@ -133,16 +129,15 @@ export function SettlementBatchesConsole() {
} finally {
setLoading(false);
}
}, [page, perPage, appliedDrawNo, appliedStatus, appliedAgentNodeId]);
}, [page, perPage, appliedDrawNo, appliedStatus]);
useAsyncEffect(() => {
void load();
}, [page, perPage, appliedDrawNo, appliedStatus, appliedAgentNodeId]);
}, [page, perPage, appliedDrawNo, appliedStatus]);
const applyFilters = () => {
setAppliedDrawNo(draftDrawNo);
setAppliedStatus(draftStatus);
setAppliedAgentNodeId(agentNodeId);
setPage(1);
};
@@ -201,12 +196,6 @@ export function SettlementBatchesConsole() {
<CardTitle className="admin-list-title">{t("batchList")}</CardTitle>
</div>
<div className="admin-list-toolbar">
<AdminAgentFilter
id="settlement-batches-agent-filter"
className="admin-list-field sm:w-[14rem]"
value={agentNodeId}
onChange={setAgentNodeId}
/>
<div className="admin-list-field">
<Label htmlFor="sb-draw-no" className="sm:w-10 sm:shrink-0">
{t("drawNo")}
@@ -260,7 +249,7 @@ export function SettlementBatchesConsole() {
<TableHead className="text-center">{t("platformProfit")}</TableHead>
<TableHead>{t("reviewStatus")}</TableHead>
<TableHead>{t("status")}</TableHead>
<TableHead />
<TableHead className="sticky right-0 z-20 bg-muted w-14 text-center shadow-[-1px_0_0_rgba(203,213,225,0.7)]">{t("table.actions", { ns: "common" })}</TableHead>
</TableRow>
</TableHeader>
<TableBody>
@@ -300,7 +289,7 @@ export function SettlementBatchesConsole() {
{settlementStatusText(row.status, t)}
</AdminStatusBadge>
</TableCell>
<TableCell className="text-center">
<TableCell className="sticky right-0 z-10 bg-card text-center shadow-[-1px_0_0_rgba(203,213,225,0.7)]">
<AdminRowActionsMenu
busy={actingId === row.id}
actions={[
@@ -310,6 +299,11 @@ export function SettlementBatchesConsole() {
icon: Eye,
href: `/admin/settlement-batches/${row.id}/details`,
},
{
key: "report",
label: t("viewReport", { defaultValue: "查看报表" }),
href: `/admin/reports?draw_no=${encodeURIComponent(row.draw_no ?? "")}`,
},
{
key: "approve",
label: t("pass"),