"use client"; import { useState, useEffect } from "react"; import Link from "next/link"; import { authFetch } from "@/lib/auth"; import { LineChart, Line, 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 SummaryCards() { const [data, setData] = useState(null); useEffect(() => { const f = async () => { try { const r = await authFetch("/api/paper/summary"); if (r.ok) setData(await r.json()); } catch {} }; f(); const iv = setInterval(f, 10000); return () => clearInterval(iv); }, []); if (!data) return
加载中...
; return (

总盈亏(R)

= 0 ? "text-emerald-600" : "text-red-500"}`}>{data.total_pnl >= 0 ? "+" : ""}{data.total_pnl}

胜率

{data.win_rate}%

总交易

{data.total_trades}

持仓中

{data.active_positions}

盈亏比(PF)

{data.profit_factor}

运行

{data.start_time ? "运行中 ✅" : "等待首笔"}

); } // ─── 当前持仓 ──────────────────────────────────────────────────── function ActivePositions() { const [positions, setPositions] = useState([]); useEffect(() => { const f = async () => { try { const r = await authFetch("/api/paper/positions"); if (r.ok) { const j = await r.json(); setPositions(j.data || []); } } catch {} }; f(); const iv = setInterval(f, 5000); return () => clearInterval(iv); }, []); if (positions.length === 0) return (
暂无活跃持仓
); return (

当前持仓

{positions.map((p: any) => { const sym = p.symbol?.replace("USDT", "") || ""; const holdMin = Math.round((Date.now() - p.entry_ts) / 60000); return (
{p.direction === "LONG" ? "🟢" : "🔴"} {sym} {p.direction} 评分{p.score} · {p.tier === "heavy" ? "加仓" : p.tier === "standard" ? "标准" : "轻仓"}
{holdMin}分钟
入场: ${fmtPrice(p.entry_price)} TP1: ${fmtPrice(p.tp1_price)}{p.tp1_hit ? " ✅" : ""} TP2: ${fmtPrice(p.tp2_price)} SL: ${fmtPrice(p.sl_price)}
); })}
); } // ─── 权益曲线 ──────────────────────────────────────────────────── function EquityCurve() { const [data, setData] = useState([]); useEffect(() => { const f = async () => { try { const r = await authFetch("/api/paper/equity-curve"); if (r.ok) { const j = await r.json(); setData(j.data || []); } } catch {} }; f(); const iv = setInterval(f, 30000); return () => clearInterval(iv); }, []); if (data.length < 2) return null; return (

权益曲线 (累计PnL)

bjt(v)} tick={{ fontSize: 10 }} /> `${v}R`} /> bjt(Number(v))} formatter={(v: any) => [`${v}R`, "累计PnL"]} />
); } // ─── 历史交易列表 ──────────────────────────────────────────────── type FilterSymbol = "all" | "BTC" | "ETH" | "XRP" | "SOL"; type FilterResult = "all" | "win" | "loss"; function TradeHistory() { const [trades, setTrades] = useState([]); const [symbol, setSymbol] = useState("all"); const [result, setResult] = useState("all"); useEffect(() => { const f = async () => { try { const r = await authFetch(`/api/paper/trades?symbol=${symbol}&result=${result}&limit=50`); if (r.ok) { const j = await r.json(); setTrades(j.data || []); } } catch {} }; f(); const iv = setInterval(f, 10000); return () => clearInterval(iv); }, [symbol, result]); return (

历史交易

{(["all", "BTC", "ETH", "XRP", "SOL"] as FilterSymbol[]).map(s => ( ))} | {(["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; return ( ); })}
币种 方向 入场 出场 PnL(R) 状态 分数 时间
{t.symbol?.replace("USDT", "")} {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} {t.score} {holdMin}m
)}
); } // ─── 统计面板 ──────────────────────────────────────────────────── function StatsPanel() { const [data, setData] = useState(null); useEffect(() => { const f = async () => { try { const r = await authFetch("/api/paper/stats"); if (r.ok) setData(await r.json()); } catch {} }; f(); const iv = setInterval(f, 30000); return () => clearInterval(iv); }, []); if (!data || data.error) return null; return (

详细统计

胜率

{data.win_rate}%

盈亏比

{data.win_loss_ratio}

平均盈利

+{data.avg_win}R

平均亏损

-{data.avg_loss}R

最大回撤

{data.mdd}R

夏普比率

{data.sharpe}

做多胜率

{data.long_win_rate}% ({data.long_count}笔)

做空胜率

{data.short_win_rate}% ({data.short_count}笔)

{data.by_symbol && Object.entries(data.by_symbol).map(([s, v]: [string, any]) => (
{s}胜率

{v.win_rate}% ({v.total}笔)

))} {data.by_tier && Object.entries(data.by_tier).map(([t, v]: [string, any]) => (
{t === "heavy" ? "加仓" : t === "standard" ? "标准" : "轻仓"}档

{v.win_rate}% ({v.total}笔)

))}
); } // ─── 主页面 ────────────────────────────────────────────────────── export default function PaperTradingPage() { const [isLoggedIn, setIsLoggedIn] = useState(false); useEffect(() => { const token = typeof window !== "undefined" ? localStorage.getItem("token") : null; setIsLoggedIn(!!token); }, []); if (!isLoggedIn) return (
🔒

请先登录查看模拟盘

登录
); return (

📊 模拟盘

V5.1信号引擎自动交易 · 实时追踪 · 数据驱动优化

); }