Redesign: phosphor-terminal identity and simplified 4-page structure
Deploy / lint (push) Successful in 6s
Deploy / test (push) Successful in 31s
Deploy / deploy (push) Successful in 22s

Information architecture (6 nav destinations -> 4):
- New Overview home: metric strip (live setups, high confidence,
  hit rate, expectancy), top-5 setups, watchlist pulse
- Market = Watchlist + Rankings merged as tabs; scoring weights
  moved into a collapsible disclosure
- Signals = Scanner + Performance merged as tabs (Setups | Track
  Record) with actions inside the panels
- Legacy routes redirect (/watchlist, /rankings, /scanner,
  /performance)

Visual identity:
- Warm ash-green dark palette replaces cold navy; citron lime
  accent replaces blue (Tailwind gray/blue remapped at config
  level so all components reskin)
- Primary buttons: lime with ink text; long/short stays
  emerald/red
- Typography: Bricolage Grotesque display, Instrument Sans body,
  IBM Plex Mono for all numerals incl. chart canvas labels
- Atmosphere: graph-paper grid + citron glow + film grain;
  pulsing brand dot; mono-numbered nav

Co-Authored-By: Claude Fable 5 <noreply@anthropic.com>
This commit is contained in:
2026-06-12 14:42:05 +02:00
parent 21ed83c56c
commit 9c6a0a72fa
20 changed files with 548 additions and 213 deletions
+217
View File
@@ -0,0 +1,217 @@
import { useMemo } from 'react';
import { Link } from 'react-router-dom';
import { useTrades } from '../hooks/useTrades';
import { useWatchlist } from '../hooks/useWatchlist';
import { usePerformance } from '../hooks/usePerformance';
import { Callout } from '../components/ui/Callout';
import { Section } from '../components/ui/Section';
import { SkeletonCard, SkeletonTable } from '../components/ui/Skeleton';
import { formatPrice } from '../lib/format';
import { recommendationActionLabel } from '../lib/recommendation';
import type { TradeSetup } from '../lib/types';
function fmtR(value: number | null): string {
if (value === null) return '—';
return `${value > 0 ? '+' : ''}${value.toFixed(2)}R`;
}
function rColor(value: number | null): string {
if (value === null) return 'text-gray-400';
if (value > 0) return 'text-emerald-400';
if (value < 0) return 'text-red-400';
return 'text-gray-300';
}
function Metric({ label, value, sub, valueClass = 'text-gray-100' }: {
label: string;
value: string;
sub?: string;
valueClass?: string;
}) {
return (
<div className="glass glass-hover p-5">
<p className="section-index">{label}</p>
<p className={`num mt-2 text-3xl font-semibold ${valueClass}`}>{value}</p>
{sub && <p className="mt-1 text-xs text-gray-500">{sub}</p>}
</div>
);
}
function DirectionTag({ direction }: { direction: string }) {
const isLong = direction === 'long';
return (
<span className={`num inline-block rounded px-1.5 py-0.5 text-[10px] font-semibold uppercase tracking-wider ${
isLong ? 'bg-emerald-500/15 text-emerald-400' : 'bg-red-500/15 text-red-400'
}`}>
{direction}
</span>
);
}
export default function DashboardPage() {
const trades = useTrades();
const watchlist = useWatchlist();
const performance = usePerformance();
const topSetups: TradeSetup[] = useMemo(
() => (trades.data ?? []).slice(0, 5),
[trades.data],
);
const highConfidenceCount = useMemo(
() => (trades.data ?? []).filter((t) => (t.confidence_score ?? 0) >= 70).length,
[trades.data],
);
const topWatchlist = useMemo(
() =>
[...(watchlist.data ?? [])]
.sort((a, b) => (b.composite_score ?? -1) - (a.composite_score ?? -1))
.slice(0, 6),
[watchlist.data],
);
const today = new Date().toLocaleDateString('en-US', {
weekday: 'long', month: 'long', day: 'numeric',
});
const stats = performance.data?.overall;
return (
<div className="space-y-8 animate-slide-up">
{/* Hero */}
<div>
<p className="num text-xs uppercase tracking-[0.22em] text-gray-500">{today}</p>
<h1 className="font-display mt-1 text-4xl font-bold tracking-tight text-gray-100">
Market overview
</h1>
</div>
{/* Metric strip */}
{(trades.isLoading || performance.isLoading) ? (
<div className="grid gap-4 sm:grid-cols-2 lg:grid-cols-4">
<SkeletonCard /><SkeletonCard /><SkeletonCard /><SkeletonCard />
</div>
) : (
<div className="grid gap-4 sm:grid-cols-2 lg:grid-cols-4">
<Metric
label="Live Setups"
value={String(trades.data?.length ?? 0)}
sub="latest per ticker & direction"
/>
<Metric
label="High Confidence"
value={String(highConfidenceCount)}
sub="confidence ≥ 70%"
valueClass={highConfidenceCount > 0 ? 'text-blue-300' : 'text-gray-100'}
/>
<Metric
label="Hit Rate"
value={stats?.hit_rate != null ? `${stats.hit_rate.toFixed(1)}%` : '—'}
sub={stats ? `${stats.wins}W / ${stats.losses}L evaluated` : 'no outcomes yet'}
/>
<Metric
label="Expectancy"
value={fmtR(stats?.avg_r ?? null)}
valueClass={rColor(stats?.avg_r ?? null)}
sub="average R per trade"
/>
</div>
)}
<div className="grid gap-8 xl:grid-cols-5">
{/* Top setups */}
<div className="xl:col-span-3">
<Section title="Top Setups" hint="by confidence">
{trades.isLoading && <SkeletonTable rows={5} cols={5} />}
{trades.isError && <Callout variant="error">Failed to load setups</Callout>}
{trades.data && topSetups.length === 0 && (
<Callout variant="empty">No active setups. Run the scanner from the Signals page.</Callout>
)}
{topSetups.length > 0 && (
<div className="glass overflow-x-auto">
<table className="w-full text-sm">
<thead>
<tr className="border-b border-white/[0.06] text-left text-xs uppercase tracking-wider text-gray-500">
<th className="px-4 py-3">Ticker</th>
<th className="px-4 py-3">Dir</th>
<th className="px-4 py-3 text-right">Entry</th>
<th className="px-4 py-3 text-right">R:R</th>
<th className="px-4 py-3 text-right">Conf.</th>
<th className="hidden px-4 py-3 md:table-cell">Action</th>
</tr>
</thead>
<tbody>
{topSetups.map((setup) => (
<tr key={setup.id} className="border-b border-white/[0.04] transition-colors duration-150 hover:bg-white/[0.03]">
<td className="px-4 py-3">
<Link to={`/ticker/${setup.symbol}`} className="font-medium text-blue-300 hover:text-blue-200 transition-colors">
{setup.symbol}
</Link>
</td>
<td className="px-4 py-3"><DirectionTag direction={setup.direction} /></td>
<td className="num px-4 py-3 text-right text-gray-200">{formatPrice(setup.entry_price)}</td>
<td className="num px-4 py-3 text-right text-gray-200">{setup.rr_ratio.toFixed(1)}:1</td>
<td className="num px-4 py-3 text-right text-gray-200">
{setup.confidence_score != null ? `${Math.round(setup.confidence_score)}%` : '—'}
</td>
<td className="hidden px-4 py-3 text-xs text-gray-400 md:table-cell">
{recommendationActionLabel(setup.recommended_action)}
</td>
</tr>
))}
</tbody>
</table>
<div className="border-t border-white/[0.04] px-4 py-2.5">
<Link to="/signals" className="text-xs font-medium text-blue-300 hover:text-blue-200 transition-colors">
All setups
</Link>
</div>
</div>
)}
</Section>
</div>
{/* Watchlist pulse */}
<div className="xl:col-span-2">
<Section title="Watchlist Pulse" hint="top by score">
{watchlist.isLoading && <SkeletonTable rows={6} cols={3} />}
{watchlist.isError && <Callout variant="error">Failed to load watchlist</Callout>}
{watchlist.data && topWatchlist.length === 0 && (
<Callout variant="empty">Watchlist is empty add tickers on the Market page.</Callout>
)}
{topWatchlist.length > 0 && (
<div className="glass overflow-hidden">
<ul className="divide-y divide-white/[0.04]">
{topWatchlist.map((entry) => (
<li key={entry.symbol}>
<Link
to={`/ticker/${entry.symbol}`}
className="flex items-center justify-between px-4 py-3 transition-colors duration-150 hover:bg-white/[0.03]"
>
<span className="font-medium text-gray-200">{entry.symbol}</span>
<span className="flex items-center gap-4">
{entry.rr_ratio != null && (
<span className="num text-xs text-gray-500">{entry.rr_ratio.toFixed(1)}:1</span>
)}
<span className="num text-sm font-semibold text-blue-300">
{entry.composite_score != null ? entry.composite_score.toFixed(0) : '—'}
</span>
</span>
</Link>
</li>
))}
</ul>
<div className="border-t border-white/[0.04] px-4 py-2.5">
<Link to="/market" className="text-xs font-medium text-blue-300 hover:text-blue-200 transition-colors">
Full watchlist
</Link>
</div>
</div>
)}
</Section>
</div>
</div>
</div>
);
}