refactor: update environment configuration and enhance notification handling

- Refactored ecosystem configuration to utilize .env for sensitive variables, improving security and flexibility.
- Replaced direct notification button implementation with a dedicated PlayerNotificationBell component for better code organization.
- Updated various screens to integrate the new notification component and adjusted prefetch settings for links to optimize performance.
- Added new translations for notifications and wallet logs to enhance user experience across multiple languages.
This commit is contained in:
2026-06-01 09:29:02 +08:00
parent 9f43d07778
commit 10bee1b857
16 changed files with 241 additions and 59 deletions

View File

@@ -38,16 +38,10 @@ module.exports = {
env: {
NODE_ENV: "production",
PORT: "3800",
// Laravel 根地址(无尾部 /);同机部署填 http://127.0.0.1:8000
LOTTERY_API_UPSTREAM: "http://127.0.0.1:8000",
// 构建时需存在;运行时可留空若已在 build 时写入
// NEXT_PUBLIC_MAIN_SITE_URL: "https://main.yourdomain.com",
},
// PM2 5.2+:可把变量放在 .env由 PM2 注入(需 npm run build 前也有一份供 Next 编译)
// env_file: path.join(APP_CWD, ".env"),
// LOTTERY_API_UPSTREAM、NEXT_PUBLIC_* 写在 .env勿在此硬编码 LOTTERY_API_UPSTREAM
env_file: path.join(APP_CWD, ".env"),
},
],
};

View File

@@ -0,0 +1,107 @@
"use client";
import { Bell } from "lucide-react";
import Link from "next/link";
import { useState, type ReactElement } from "react";
import { useTranslation } from "react-i18next";
import { Popover, PopoverContent, PopoverTrigger } from "@/components/ui/popover";
import { logTypeLabel } from "@/features/wallet/wallet-logs-block";
import { usePendingWalletReconcile } from "@/hooks/use-pending-wallet-reconcile";
import { playerHeaderControl } from "@/lib/player-spacing";
import { formatMinorAsCurrency } from "@/lib/money";
import { cn } from "@/lib/utils";
type PlayerNotificationBellProps = {
className?: string;
};
/** 顶栏铃铛待对账划转等提醒Popover 右上角展开) */
export function PlayerNotificationBell({ className }: PlayerNotificationBellProps): ReactElement {
const { t } = useTranslation("common");
const { t: tp } = useTranslation("player");
const [open, setOpen] = useState(false);
const { pending, hasPending, loading, refresh } = usePendingWalletReconcile();
return (
<Popover
open={open}
onOpenChange={(next) => {
setOpen(next);
if (next) void refresh();
}}
>
<PopoverTrigger
render={
<button
type="button"
className={cn(
playerHeaderControl,
"relative size-8 rounded-full text-[#1d57b7] hover:bg-[#f4f7fb]",
className,
)}
aria-label={t("navigation.notifications")}
>
<Bell className="size-4" aria-hidden />
{hasPending ? (
<span className="absolute right-1.5 top-1.5 size-1.5 rounded-full bg-[#ff143d]" />
) : null}
</button>
}
/>
<PopoverContent
align="end"
sideOffset={8}
className="w-[min(20rem,calc(100vw-1.5rem))] border-[#dce7f7] p-0 shadow-[0_16px_40px_rgba(15,23,42,0.14)]"
>
<div className="border-b border-[#edf2f9] px-3 py-2.5">
<p className="text-sm font-black text-[#0b3f96]">{t("navigation.notifications")}</p>
</div>
{loading && pending.length === 0 ? (
<p className="px-3 py-6 text-center text-xs text-slate-500">{tp("actions.loading")}</p>
) : null}
{!loading && pending.length === 0 ? (
<p className="px-3 py-6 text-center text-xs text-slate-500">
{tp("notifications.empty")}
</p>
) : null}
{pending.length > 0 ? (
<div className="max-h-[min(50vh,320px)] overflow-y-auto p-2">
<div className="rounded-xl border border-amber-200 bg-amber-50 px-3 py-3">
<p className="text-sm font-black text-amber-700">{tp("wallet.pendingTitle")}</p>
<p className="mt-1 text-xs leading-relaxed text-amber-800/90">
{tp("wallet.pendingDescription")}
</p>
<ul className="mt-2 space-y-2">
{pending.map((p) => (
<li
key={p.transfer_no}
className="flex flex-wrap items-baseline justify-between gap-1 border-b border-dashed border-amber-200/80 py-2 last:border-0"
>
<span className="text-sm text-amber-950">
{logTypeLabel(p.type, tp)}{" "}
{formatMinorAsCurrency(p.amount, p.currency_code)}
</span>
<span className="text-xs font-semibold text-amber-700">
{tp("wallet.pendingStatus")}
</span>
</li>
))}
</ul>
</div>
<Link
href="/wallet/logs"
className="mt-2 block rounded-lg px-2 py-2 text-center text-xs font-bold text-[#0b56b7] hover:bg-[#f8fbff]"
onClick={() => setOpen(false)}
>
{tp("wallet.logs", { defaultValue: "查看流水" })}
</Link>
</div>
) : null}
</PopoverContent>
</Popover>
);
}

View File

@@ -3,10 +3,11 @@
import Link from "next/link";
import type { ReactNode } from "react";
import { Bell, ChevronLeft } from "lucide-react";
import { ChevronLeft } from "lucide-react";
import { useTranslation } from "react-i18next";
import { LanguageSwitcher } from "@/components/language-switcher";
import { PlayerNotificationBell } from "@/components/layout/player-notification-bell";
import {
playerHeaderControl,
playerPageHeader,
@@ -79,17 +80,7 @@ export function PlayerPanel({
"rounded-full border border-[#e4eaf4] bg-[#f8fafc] [&_button]:h-8 [&_button]:gap-1 [&_button]:px-2 [&_button]:py-0 [&_button]:text-xs",
)}
/>
<button
type="button"
className={cn(
playerHeaderControl,
"relative size-8 rounded-full text-[#1d57b7] hover:bg-[#f4f7fb]",
)}
aria-label={t("navigation.notifications")}
>
<Bell className="size-4" aria-hidden />
<span className="absolute right-1.5 top-1.5 size-1.5 rounded-full bg-[#ff143d]" />
</button>
<PlayerNotificationBell />
</div>
</header>
{children}

View File

@@ -1,12 +1,12 @@
"use client";
import { Bell } from "lucide-react";
import Image from "next/image";
import Link from "next/link";
import { useTranslation } from "react-i18next";
import { CurrencySwitcher } from "@/components/currency-switcher";
import { LanguageSwitcher } from "@/components/language-switcher";
import { PlayerNotificationBell } from "@/components/layout/player-notification-bell";
import { HallBettingGrid } from "@/features/hall/hall-betting-grid";
import { useActivePlayerCurrency } from "@/hooks/use-active-player-currency";
import { HallDrawPanel } from "@/features/hall/hall-draw-panel";
@@ -21,7 +21,6 @@ import { cn } from "@/lib/utils";
* 下注大厅:钱包条 §4 + 当期期号 §4.2(封盘置灰 / 倒计时错误色 / WS+轮询);玩法目录 §12.3;下注表格 §13.3。
*/
export function HallScreen() {
const { t } = useTranslation("common");
const { t: tp } = useTranslation("player");
const drawLive = useHallDrawLive();
const { activeCurrency } = useActivePlayerCurrency();
@@ -69,17 +68,7 @@ export function HallScreen() {
>
{tp("nav.rules")}
</Link>
<button
type="button"
className={cn(
playerHeaderControl,
"relative size-8 rounded-full text-[#1d57b7] hover:bg-[#f4f7fb]",
)}
aria-label={t("navigation.notifications")}
>
<Bell className="size-4" aria-hidden />
<span className="absolute right-1.5 top-1.5 size-1.5 rounded-full bg-[#ff143d]" />
</button>
<PlayerNotificationBell />
</div>
</header>

View File

@@ -432,6 +432,7 @@ export function TicketOrderDetailScreen({ ticketNo }: { ticketNo: string }) {
{data.draw_no ? (
<Link
href={`/results/${encodeURIComponent(data.draw_no)}`}
prefetch={false}
className={cn(
"inline-flex h-11 min-w-[140px] flex-1 items-center justify-center rounded-xl bg-[#07459f] px-4 text-sm font-bold text-white shadow-sm transition-colors hover:bg-[#063b88]",
)}

View File

@@ -17,6 +17,7 @@ import { useTranslation } from "react-i18next";
import { getPublicCurrencies } from "@/api/currency";
import { getPlayerMe, getPlayerPing } from "@/api/player";
import { isInIframe } from "@/components/iframe-bridge";
import { LanguageSwitcher } from "@/components/language-switcher";
import { Button, buttonVariants } from "@/components/ui/button";
import { cn } from "@/lib/utils";
@@ -25,6 +26,8 @@ import { LotteryApiBizError } from "@/types/api/errors";
const RETRY_ATTEMPTS = 3;
const RETRY_DELAY_MS = 2000;
/** 嵌入主站 iframe 时,等待父页 postMessage 下发 token 的最长时间 */
const IFRAME_TOKEN_WAIT_MS = 15_000;
const MAIN_SITE_URL = process.env.NEXT_PUBLIC_MAIN_SITE_URL?.trim() ?? "";
@@ -134,11 +137,18 @@ export function EntryGate() {
const doEntry = useCallback(async () => {
if (!effectiveToken) {
// 主站 iframetoken 由 MAIN_INIT_TOKEN 稍后到达,勿先闪「授权失败」
if (typeof window !== "undefined" && isInIframe() && !tokenFromUrl) {
return;
}
setPhase("failed");
setFailureDetails([{ code: "NO_TOKEN", detailKey: "errors.noTokenDetail" }]);
return;
}
setPhase("loading");
setFailureDetails([]);
if (tokenFromUrl) {
setBearerToken(tokenFromUrl);
stripSearchParamFromBrowserUrl("token");
@@ -260,11 +270,32 @@ export function EntryGate() {
useEffect(() => {
if (sessionExpired) return;
const embedded = typeof window !== "undefined" && isInIframe() && !tokenFromUrl;
if (embedded && !effectiveToken) {
setPhase("loading");
return;
}
const tmr = window.setTimeout(() => {
void doEntry();
}, 300);
return () => window.clearTimeout(tmr);
}, [doEntry, sessionExpired]);
}, [doEntry, sessionExpired, effectiveToken, tokenFromUrl]);
useEffect(() => {
if (sessionExpired) return;
if (tokenFromUrl || effectiveToken) return;
if (typeof window === "undefined" || !isInIframe()) return;
const tmr = window.setTimeout(() => {
if (usePlayerSessionStore.getState().bearerToken?.trim()) return;
setFailureDetails([{ code: "NO_TOKEN", detailKey: "errors.noTokenDetail" }]);
setPhase("failed");
}, IFRAME_TOKEN_WAIT_MS);
return () => window.clearTimeout(tmr);
}, [sessionExpired, effectiveToken, tokenFromUrl]);
return (
<div className="relative flex min-h-dvh flex-col bg-white">

View File

@@ -163,6 +163,7 @@ export function DrawResultDetailScreen({ drawNo }: DrawResultDetailScreenProps)
{data.previous_draw_no ? (
<Link
href={`/results/${encodeURIComponent(data.previous_draw_no)}`}
prefetch={false}
className={cn(
buttonVariants({ variant: "outline", size: "sm" }),
"min-w-[5rem] rounded-full border-[#dce7f7] bg-white text-[#0b56b7] hover:bg-[#f1f6ff]",
@@ -188,6 +189,7 @@ export function DrawResultDetailScreen({ drawNo }: DrawResultDetailScreenProps)
{data.next_draw_no ? (
<Link
href={`/results/${encodeURIComponent(data.next_draw_no)}`}
prefetch={false}
className={cn(
buttonVariants({ variant: "outline", size: "sm" }),
"min-w-[5rem] rounded-full border-[#dce7f7] bg-white text-[#0b56b7] hover:bg-[#f1f6ff]",

View File

@@ -252,6 +252,7 @@ export function DrawResultsListScreen() {
</div>
<Link
href={`/results/${encodeURIComponent(featured.draw_no)}`}
prefetch={false}
className="inline-flex h-8 shrink-0 items-center justify-center rounded-full border border-[#dce7f7] bg-white px-3 text-sm font-semibold text-[#0b56b7] transition-colors hover:bg-[#f1f6ff]"
>
{t("results.openDetail", { defaultValue: "查看详情" })}
@@ -273,6 +274,7 @@ export function DrawResultsListScreen() {
<Link
key={row.draw_no}
href={`/results/${encodeURIComponent(row.draw_no)}`}
prefetch={false}
className="block rounded-xl border border-[#e5edf8] bg-white p-3 shadow-[0_8px_24px_rgba(15,23,42,0.05)] transition-colors hover:border-[#b9ccf6]"
>
<div className="flex items-start justify-between gap-3">

View File

@@ -48,7 +48,7 @@ type WalletLogsBlockProps = {
title?: string;
};
/** 待对账 + 类型筛选 + 列表(供钱包主页与 `/wallet/logs` 共用 */
/** 类型筛选 + 列表(待对账见顶栏通知铃铛 */
export function WalletLogsBlock({
logs,
logsLoading,
@@ -74,29 +74,6 @@ export function WalletLogsBlock({
return (
<>
{logs && logs.pending_reconcile.length > 0 ? (
<section className="rounded-xl border border-amber-200 bg-amber-50 px-3 py-3">
<p className="text-sm font-black text-amber-700">{t("wallet.pendingTitle")}</p>
<p className="mt-1 text-xs text-amber-700/80">
{t("wallet.pendingDescription")}
</p>
<div className="mt-2 space-y-2 text-sm">
{logs.pending_reconcile.map((p) => (
<div
key={p.transfer_no}
className="flex flex-wrap items-baseline justify-between gap-1 border-b border-dashed border-border py-2 last:border-0"
>
<span className="text-muted-foreground">
{logTypeLabel(p.type, t)}{" "}
{formatMinorAsCurrency(p.amount, p.currency_code)}
</span>
<span className="text-xs text-amber-700">{t("wallet.pendingStatus")}</span>
</div>
))}
</div>
</section>
) : null}
<section className="space-y-3">
<div className="flex flex-col gap-2">
<h2 className="text-sm font-black text-[#0b3f96]">{resolvedTitle}</h2>

View File

@@ -7,6 +7,7 @@ import { getWalletLogs } from "@/api/wallet";
import { Button } from "@/components/ui/button";
import { PlayerPanel } from "@/components/layout/player-panel";
import { WalletLogsBlock } from "@/features/wallet/wallet-logs-block";
import { dispatchWalletLogsRefresh } from "@/hooks/use-pending-wallet-reconcile";
import { useActivePlayerCurrency } from "@/hooks/use-active-player-currency";
import { PLAYER_CURRENCY_CHANGE_EVENT } from "@/lib/player-currency-preference";
import { formatWalletClientError } from "@/lib/wallet-api-error";
@@ -49,6 +50,7 @@ export function WalletLogsScreen() {
? { ...nextLogs, items: [...current.items, ...nextLogs.items] }
: nextLogs,
);
dispatchWalletLogsRefresh(nextLogs.pending_reconcile ?? []);
} catch (e) {
setError(formatWalletClientError(e, t));
if (!append) {

View File

@@ -14,6 +14,7 @@ import {
TransferOutDialog,
} from "@/features/wallet/wallet-transfer-dialogs";
import { WalletLogsBlock } from "@/features/wallet/wallet-logs-block";
import { dispatchWalletLogsRefresh } from "@/hooks/use-pending-wallet-reconcile";
import { useActivePlayerCurrency } from "@/hooks/use-active-player-currency";
import { formatMinorAsCurrency } from "@/lib/money";
import { PLAYER_CURRENCY_CHANGE_EVENT } from "@/lib/player-currency-preference";
@@ -49,6 +50,7 @@ export function WalletScreen() {
? { ...nextLogs, items: [...current.items, ...nextLogs.items] }
: nextLogs,
);
dispatchWalletLogsRefresh(nextLogs.pending_reconcile ?? []);
return nextLogs;
}, [currency, filter]);

View File

@@ -0,0 +1,72 @@
"use client";
import { useCallback, useEffect, useState } from "react";
import { getWalletLogs } from "@/api/wallet";
import { usePlayerSessionStore } from "@/stores/player-session-store";
import type { WalletPendingTransfer } from "@/types/api/wallet-logs";
export const WALLET_LOGS_REFRESH_EVENT = "lottery:wallet-logs-refreshed";
type WalletLogsRefreshDetail = {
pending?: WalletPendingTransfer[];
};
export function usePendingWalletReconcile(): {
pending: WalletPendingTransfer[];
hasPending: boolean;
loading: boolean;
refresh: () => Promise<void>;
} {
const bearerToken = usePlayerSessionStore((s) => s.bearerToken);
const [pending, setPending] = useState<WalletPendingTransfer[]>([]);
const [loading, setLoading] = useState(false);
const refresh = useCallback(async (): Promise<void> => {
if (!bearerToken?.trim()) {
setPending([]);
return;
}
setLoading(true);
try {
const data = await getWalletLogs({ page: 1, size: 1 });
setPending(data.pending_reconcile ?? []);
} catch {
setPending([]);
} finally {
setLoading(false);
}
}, [bearerToken]);
useEffect(() => {
void refresh();
function onWalletLogsRefreshed(event: Event): void {
const detail = (event as CustomEvent<WalletLogsRefreshDetail>).detail;
if (Array.isArray(detail?.pending)) {
setPending(detail.pending);
return;
}
void refresh();
}
window.addEventListener(WALLET_LOGS_REFRESH_EVENT, onWalletLogsRefreshed);
return () => window.removeEventListener(WALLET_LOGS_REFRESH_EVENT, onWalletLogsRefreshed);
}, [refresh]);
return {
pending,
hasPending: pending.length > 0,
loading,
refresh,
};
}
export function dispatchWalletLogsRefresh(pending: WalletPendingTransfer[]): void {
if (typeof window === "undefined") return;
window.dispatchEvent(
new CustomEvent<WalletLogsRefreshDetail>(WALLET_LOGS_REFRESH_EVENT, {
detail: { pending },
}),
);
}

View File

@@ -26,6 +26,9 @@
"rules": "Rules",
"wallet": "Wallet"
},
"notifications": {
"empty": "No notifications"
},
"panel": {
"home": "Home"
},
@@ -367,6 +370,7 @@
"flowsTitle": "Wallet logs",
"totalRecords": "{{total}} records",
"emptyLogs": "No wallet logs",
"noMoreLogs": "No more transactions",
"balanceAfter": "Balance after",
"wsBalanceUpdated": "Balance {{change}} ({{reason}})",
"wsReason": {

View File

@@ -26,6 +26,9 @@
"rules": "नियम",
"wallet": "वालेट"
},
"notifications": {
"empty": "कुनै सूचना छैन"
},
"panel": {
"home": "गृह"
},
@@ -367,6 +370,7 @@
"flowsTitle": "वालेट लग",
"totalRecords": "{{total}} रेकर्ड",
"emptyLogs": "वालेट लग छैन",
"noMoreLogs": "थप लेनदेन छैन",
"balanceAfter": "पछि बाँकी ब्यालेन्स",
"wsBalanceUpdated": "ब्यालेन्स {{change}} ({{reason}})",
"wsReason": {

View File

@@ -29,7 +29,7 @@
"retryProgress": "正在重试({{current}}/{{total}})…"
},
"failure": {
"title": "授权失败111",
"title": "授权失败",
"subtitle": "无法完成授权,请重试。",
"detailsTitle": "失败详情",
"table": {

View File

@@ -26,6 +26,9 @@
"rules": "规则",
"wallet": "钱包"
},
"notifications": {
"empty": "暂无通知"
},
"panel": {
"home": "首页"
},
@@ -368,6 +371,7 @@
"flowsTitle": "资金流水",
"totalRecords": "共 {{total}} 条记录",
"emptyLogs": "暂无流水",
"noMoreLogs": "没有更多流水",
"balanceAfter": "变更后余额",
"wsBalanceUpdated": "余额 {{change}}{{reason}}",
"wsReason": {