"use client"; import { useState, useEffect } from "react"; import Link from "next/link"; import { authFetch, useAuth } from "@/lib/auth"; import { XAxis, YAxis, Tooltip, ResponsiveContainer, ReferenceLine, Area, AreaChart } from "recharts"; function bjt(ms: number) { const d = new Date(ms + 8 * 3600 * 1000); return `${String(d.getUTCMonth() + 1).padStart(2, "0")}-${String(d.getUTCDate()).padStart(2, "0")} ${String(d.getUTCHours()).padStart(2, "0")}:${String(d.getUTCMinutes()).padStart(2, "0")}`; } function fmtPrice(p: number) { return p < 100 ? p.toFixed(4) : p.toLocaleString("en-US", { minimumFractionDigits: 1, maximumFractionDigits: 1 }); } function parseFactors(raw: any) { if (!raw) return null; if (typeof raw === "string") { try { return JSON.parse(raw); } catch { return null; } } return raw; } type StrategyTab = "v53_alt" | "v53_btc"; const STRATEGY_LABELS: Record = { v53_alt: { label: "๐ŸŸฃ ALT่ฝจ (v53_alt)", desc: "ETH / XRP / SOL ยท ๅ››ๅฑ‚่ฏ„ๅˆ† 55/25/15/5", coins: ["ETHUSDT", "XRPUSDT", "SOLUSDT"], badgeClass: "bg-purple-100 text-purple-700 border border-purple-200", }, v53_btc: { label: "๐Ÿ”ต BTC่ฝจ (v53_btc)", desc: "BTCUSDT ยท Gate-Control้€ป่พ‘", coins: ["BTCUSDT"], badgeClass: "bg-amber-100 text-amber-700 border border-amber-200", }, }; // โ”€โ”€โ”€ ๆœ€ๆ–ฐไฟกๅท็Šถๆ€ โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€ const ALT_COINS = ["ETHUSDT", "XRPUSDT", "SOLUSDT"]; const BTC_COINS = ["BTCUSDT"]; function LatestSignals({ strategy }: { strategy: StrategyTab }) { const coins = strategy === "v53_btc" ? BTC_COINS : ALT_COINS; const [signals, setSignals] = useState>({}); useEffect(() => { const f = async () => { for (const sym of coins) { const coin = sym.replace("USDT", ""); const strat = strategy === "v53_btc" ? "v53_btc" : "v53_alt"; try { const r = await authFetch(`/api/signals/signal-history?symbol=${coin}&limit=1&strategy=${strat}`); if (r.ok) { const j = await r.json(); if (j.data && j.data.length > 0) setSignals(prev => ({ ...prev, [sym]: j.data[0] })); } } catch {} } }; f(); const iv = setInterval(f, 15000); return () => clearInterval(iv); }, [strategy, coins]); const meta = STRATEGY_LABELS[strategy]; return (

ๆœ€ๆ–ฐไฟกๅท

{strategy}
{coins.map(sym => { const s = signals[sym]; const coin = sym.replace("USDT", ""); const ago = s?.ts ? Math.round((Date.now() - s.ts) / 60000) : null; const fc = s?.factors; return (
{coin} {s?.signal ? ( <> {s.signal === "LONG" ? "๐ŸŸข" : "๐Ÿ”ด"} {s.signal} {s.score}ๅˆ† {s.score >= 85 ? "ๅŠ ไป“" : s.score >= 75 ? "ๆ ‡ๅ‡†" : "่ง‚ๆœ›"} ) : ( ๆš‚ๆ— ไฟกๅท )}
{ago !== null && ( {ago < 60 ? `${ago}mๅ‰` : `${Math.round(ago / 60)}hๅ‰`} )}
{fc && strategy === "v53_alt" && (
ๆ–นๅ‘{fc.direction?.score ?? 0}/55 ๆ‹ฅๆŒค{fc.crowding?.score ?? 0}/25 ็Žฏๅขƒ{fc.environment?.score ?? 0}/15 ่พ…ๅŠฉ{fc.auxiliary?.score ?? 0}/5
)} {fc && strategy === "v53_btc" && (
{fc.gate_passed ? "โœ… Gate้€š่ฟ‡" : "โŒ Gateๅฆๅ†ณ"} {fc.block_reason && ( {fc.block_reason} )} OBI {((fc.obi_raw ?? 0) * 100).toFixed(2)}%
)}
); })}
); } // โ”€โ”€โ”€ ๆŽงๅˆถ้ขๆฟ โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€ function ControlPanel() { const [config, setConfig] = useState(null); const [saving, setSaving] = useState(false); useEffect(() => { (async () => { try { const r = await authFetch("/api/paper/config"); if (r.ok) setConfig(await r.json()); } catch {} })(); }, []); const toggle = async () => { setSaving(true); try { const r = await authFetch("/api/paper/config", { method: "POST", headers: { "Content-Type": "application/json" }, body: JSON.stringify({ enabled: !config.enabled }), }); if (r.ok) setConfig(await r.json().then((j: any) => j.config)); } catch {} finally { setSaving(false); } }; if (!config) return null; return (
{config.enabled ? "๐ŸŸข ่ฟ่กŒไธญ" : "โšช ๅทฒๅœๆญข"}
ๅˆๅง‹่ต„้‡‘: ${config.initial_balance?.toLocaleString()} ๅ•็ฌ”้ฃŽ้™ฉ: {(config.risk_per_trade * 100).toFixed(0)}% ๆœ€ๅคงๆŒไป“: {config.max_positions}
); } // โ”€โ”€โ”€ ๆ€ป่งˆ โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€ function SummaryCards({ strategy }: { strategy: StrategyTab }) { const [data, setData] = useState(null); useEffect(() => { const f = async () => { try { const r = await authFetch(`/api/paper/summary?strategy=${strategy}`); if (r.ok) setData(await r.json()); } catch {} }; f(); const iv = setInterval(f, 10000); return () => clearInterval(iv); }, [strategy]); if (!data) return
ๅŠ ่ฝฝไธญ...
; return (
{[ { label: "ๆ€ป็›ˆไบ(R)", value: `${data.total_pnl >= 0 ? "+" : ""}${data.total_pnl}R`, sub: `${data.total_pnl_usdt >= 0 ? "+" : ""}$${data.total_pnl_usdt}`, color: data.total_pnl >= 0 ? "text-emerald-600" : "text-red-500" }, { label: "่ƒœ็އ", value: `${data.win_rate}%`, sub: `ๅ…ฑ${data.total_trades}็ฌ”`, color: "text-slate-800" }, { label: "ๆŒไป“ไธญ", value: data.active_positions, sub: "ๆดป่ทƒไป“ไฝ", color: "text-blue-600" }, { label: "็›ˆไบๆฏ”", value: data.profit_factor, sub: "Profit Factor", color: "text-slate-800" }, { label: "ๅฝ“ๅ‰่ต„้‡‘", value: `$${data.balance?.toLocaleString()}`, sub: "่™šๆ‹Ÿไฝ™้ข", color: data.balance >= 10000 ? "text-emerald-600" : "text-red-500" }, { label: "็Šถๆ€", value: data.start_time ? "่ฟ่กŒไธญ โœ…" : "็ญ‰ๅพ…้ฆ–็ฌ”", sub: "signal accumulating", color: "text-slate-600" }, ].map(({ label, value, sub, color }) => (

{label}

{value}

{sub}

))}
); } // โ”€โ”€โ”€ ๅฝ“ๅ‰ๆŒไป“ โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€ function ActivePositions({ strategy }: { strategy: StrategyTab }) { const [positions, setPositions] = useState([]); const [wsPrices, setWsPrices] = useState>({}); const [paperRiskUsd, setPaperRiskUsd] = useState(200); const meta = STRATEGY_LABELS[strategy]; useEffect(() => { (async () => { try { const r = await authFetch("/api/paper/config"); if (r.ok) { const cfg = await r.json(); setPaperRiskUsd((cfg.initial_balance || 10000) * (cfg.risk_per_trade || 0.02)); } } catch {} })(); }, []); useEffect(() => { const f = async () => { try { const r = await authFetch(`/api/paper/positions?strategy=${strategy}`); if (r.ok) setPositions((await r.json()).data || []); } catch {} }; f(); const iv = setInterval(f, 10000); return () => clearInterval(iv); }, [strategy]); useEffect(() => { const streams = ["btcusdt", "ethusdt", "xrpusdt", "solusdt"].map(s => `${s}@aggTrade`).join("/"); const ws = new WebSocket(`wss://fstream.binance.com/stream?streams=${streams}`); ws.onmessage = (e) => { try { const msg = JSON.parse(e.data); if (msg.data) { const sym = msg.data.s; const price = parseFloat(msg.data.p); if (sym && price > 0) setWsPrices(prev => ({ ...prev, [sym]: price })); } } catch {} }; return () => ws.close(); }, []); if (positions.length === 0) return (
{meta.label} ๆš‚ๆ— ๆดป่ทƒๆŒไป“
); return (

ๅฝ“ๅ‰ๆŒไป“ โ— ๅฎžๆ—ถ

{positions.map((p: any) => { const sym = p.symbol?.replace("USDT", "") || ""; const holdMin = Math.round((Date.now() - p.entry_ts) / 60000); const currentPrice = wsPrices[p.symbol] || p.current_price || 0; const entry = p.entry_price || 0; const riskDist = p.risk_distance || Math.abs(entry - (p.sl_price || entry)) || 1; const tp1R = riskDist > 0 ? (p.direction === "LONG" ? ((p.tp1_price || 0) - entry) / riskDist : (entry - (p.tp1_price || 0)) / riskDist) : 0; const fullR = riskDist > 0 ? (p.direction === "LONG" ? (currentPrice - entry) / riskDist : (entry - currentPrice) / riskDist) : 0; const unrealR = p.tp1_hit ? 0.5 * tp1R + 0.5 * fullR : fullR; const unrealUsdt = unrealR * paperRiskUsd; return (
{p.direction === "LONG" ? "๐ŸŸข" : "๐Ÿ”ด"} {sym} {p.direction} {strategy} ่ฏ„ๅˆ†{p.score} ยท {p.tier === "heavy" ? "ๅŠ ไป“" : "ๆ ‡ๅ‡†"}
= 0 ? "text-emerald-600" : "text-red-500"}`}> {unrealR >= 0 ? "+" : ""}{unrealR.toFixed(2)}R = 0 ? "text-emerald-500" : "text-red-400"}`}> ({unrealUsdt >= 0 ? "+" : ""}${unrealUsdt.toFixed(0)}) {holdMin}m
ๅ…ฅๅœบ: ${fmtPrice(p.entry_price)} ็Žฐไปท: ${currentPrice ? fmtPrice(currentPrice) : "-"} TP1: ${fmtPrice(p.tp1_price)}{p.tp1_hit ? " โœ…" : ""} TP2: ${fmtPrice(p.tp2_price)} SL: ${fmtPrice(p.sl_price)}
); })}
); } // โ”€โ”€โ”€ ๆƒ็›Šๆ›ฒ็บฟ โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€ function EquityCurve({ strategy }: { strategy: StrategyTab }) { const [data, setData] = useState([]); useEffect(() => { const f = async () => { try { const r = await authFetch(`/api/paper/equity-curve?strategy=${strategy}`); if (r.ok) setData((await r.json()).data || []); } catch {} }; f(); const iv = setInterval(f, 30000); return () => clearInterval(iv); }, [strategy]); return (

ๆƒ็›Šๆ›ฒ็บฟ (็ดฏ่ฎกPnL)

{data.length < 2 ? (
V5.3 ๆš‚ๆ— ่ถณๅคŸๅކๅฒๆ•ฐๆฎ๏ผŒ็งฏ็ดฏไธญ...
) : (
bjt(v)} tick={{ fontSize: 10 }} /> `${v}R`} /> bjt(Number(v))} formatter={(v: any) => [`${v}R`, "็ดฏ่ฎกPnL"]} />
)}
); } // โ”€โ”€โ”€ ๅކๅฒไบคๆ˜“ โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€ type FilterResult = "all" | "win" | "loss"; function TradeHistory({ strategy }: { strategy: StrategyTab }) { const [trades, setTrades] = useState([]); const [result, setResult] = useState("all"); const meta = STRATEGY_LABELS[strategy]; useEffect(() => { const f = async () => { try { const r = await authFetch(`/api/paper/trades?result=${result}&strategy=${strategy}&limit=50`); if (r.ok) setTrades((await r.json()).data || []); } catch {} }; f(); const iv = setInterval(f, 10000); return () => clearInterval(iv); }, [result, strategy]); return (

ๅކๅฒไบคๆ˜“

{strategy} | {(["all", "win", "loss"] as FilterResult[]).map((r) => ( ))}
{trades.length === 0 ? (
ๆš‚ๆ— ไบคๆ˜“่ฎฐๅฝ•
) : ( {trades.map((t: any) => { const holdMin = t.exit_ts && t.entry_ts ? Math.round((t.exit_ts - t.entry_ts) / 60000) : 0; const factors = parseFactors(t.score_factors); const track = factors?.track || (t.symbol === "BTCUSDT" ? "BTC" : "ALT"); return ( ); })}
ๅธ็ง ๆ–นๅ‘ ๅ…ฅๅœบ ๅ‡บๅœบ PnL(R) ็Šถๆ€ ๅˆ†ๆ•ฐ ๆŒไป“
{t.symbol?.replace("USDT", "")} {track} {t.direction === "LONG" ? "๐ŸŸข" : "๐Ÿ”ด"} {t.direction} {fmtPrice(t.entry_price)} {t.exit_price ? fmtPrice(t.exit_price) : "-"} 0 ? "text-emerald-600" : t.pnl_r < 0 ? "text-red-500" : "text-slate-500"}`}> {t.pnl_r > 0 ? "+" : ""}{t.pnl_r?.toFixed(2)} {t.status === "tp" ? "ๆญข็›ˆ" : t.status === "sl" ? "ๆญขๆŸ" : t.status === "sl_be" ? "ไฟๆœฌ" : t.status === "timeout" ? "่ถ…ๆ—ถ" : t.status === "signal_flip" ? "็ฟป่ฝฌ" : t.status} {t.score} {holdMin}m
)}
); } // โ”€โ”€โ”€ ็ปŸ่ฎก้ขๆฟ โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€ function StatsPanel({ strategy }: { strategy: StrategyTab }) { const [data, setData] = useState(null); const [tab, setTab] = useState("ALL"); useEffect(() => { setTab("ALL"); }, [strategy]); useEffect(() => { const f = async () => { try { const r = await authFetch(`/api/paper/stats?strategy=${strategy}`); if (r.ok) setData(await r.json()); } catch {} }; f(); const iv = setInterval(f, 30000); return () => clearInterval(iv); }, [strategy]); if (!data || data.error) return (

่ฏฆ็ป†็ปŸ่ฎก

่ฏฅ่ง†ๅ›พๆš‚ๆ— ็ปŸ่ฎกๆ•ฐๆฎ๏ผŒ็ญ‰ๅพ…ไบคๆ˜“่ฎฐๅฝ•็งฏ็ดฏ
); const coinTabs = strategy === "v53_btc" ? ["ALL", "BTC"] : ["ALL", "ETH", "XRP", "SOL"]; const st = tab === "ALL" ? data : (data.by_symbol?.[tab] || null); return (

่ฏฆ็ป†็ปŸ่ฎก

{coinTabs.map(t => ( ))}
{st ? (
่ƒœ็އ

{st.win_rate}%

็›ˆไบๆฏ”

{st.win_loss_ratio}

ๅนณๅ‡็›ˆๅˆฉ

+{st.avg_win}R

ๅนณๅ‡ไบๆŸ

-{st.avg_loss}R

ๆœ€ๅคงๅ›žๆ’ค

{st.mdd}R

ๅคๆ™ฎๆฏ”็އ

{st.sharpe}

ๆ€ป็›ˆไบ

= 0 ? "text-emerald-600" : "text-red-500"}`}>{(st.total_pnl ?? 0) >= 0 ? "+" : ""}{st.total_pnl ?? "-"}R

ๆ€ป็ฌ”ๆ•ฐ

{st.total ?? data.total}

ๅšๅคš่ƒœ็އ

{st.long_win_rate}% ({st.long_count}็ฌ”)

ๅš็ฉบ่ƒœ็އ

{st.short_win_rate}% ({st.short_count}็ฌ”)

) : (
่ฏฅๅธ็งๆš‚ๆ— ๆ•ฐๆฎ
)}
); } // โ”€โ”€โ”€ ไธป้กต้ข โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€โ”€ export default function PaperTradingV53Page() { const { isLoggedIn, loading } = useAuth(); const [strategyTab, setStrategyTab] = useState("v53_alt"); const meta = STRATEGY_LABELS[strategyTab]; if (loading) return
ๅŠ ่ฝฝไธญ...
; if (!isLoggedIn) return (
๐Ÿ”’

่ฏทๅ…ˆ็™ปๅฝ•ๆŸฅ็œ‹ๆจกๆ‹Ÿ็›˜

็™ปๅฝ•
); return (

๐Ÿ“ˆ ๆจกๆ‹Ÿ็›˜ V5.3

ALT่ฝจ(ETH/XRP/SOL) + BTC็‹ฌ็ซ‹Gate-Control

{/* ็ญ–็•ฅTabๅˆ‡ๆข */}
{(Object.entries(STRATEGY_LABELS) as [StrategyTab, typeof STRATEGY_LABELS[StrategyTab]][]).map(([key, val]) => ( ))}
{meta.label} โ€” {meta.desc}
); }