Session 8: Frontend Stripe cutover, soccer pages, sport selector, grade result cards, beta badge
This commit is contained in:
+1
-1
File diff suppressed because one or more lines are too long
@@ -1,73 +1,100 @@
|
||||
import { NextRequest, NextResponse } from 'next/server';
|
||||
import { getUserFromRequest, jsonError } from '@/lib/auth-helpers';
|
||||
import { createPaymentLink, TIER_PRICING, type NexaPayTier } from '@/services/nexapay';
|
||||
import { getServiceRoleSupabase } from '@/lib/supabase';
|
||||
|
||||
export const dynamic = 'force-dynamic';
|
||||
|
||||
const VALID_TIERS = new Set<NexaPayTier>(['analyst', 'desk']);
|
||||
|
||||
async function resolveTier(req: NextRequest): Promise<NexaPayTier | null> {
|
||||
const url = new URL(req.url);
|
||||
const queryTier = url.searchParams.get('tier');
|
||||
if (queryTier && VALID_TIERS.has(queryTier as NexaPayTier)) return queryTier as NexaPayTier;
|
||||
if (req.method === 'POST') {
|
||||
try {
|
||||
const body = (await req.json().catch(() => ({}))) as { tier?: string };
|
||||
if (body.tier && VALID_TIERS.has(body.tier as NexaPayTier)) return body.tier as NexaPayTier;
|
||||
} catch {
|
||||
/* fall through */
|
||||
}
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
export async function GET(req: NextRequest) {
|
||||
return handle(req);
|
||||
}
|
||||
|
||||
export async function POST(req: NextRequest) {
|
||||
return handle(req);
|
||||
}
|
||||
const BACKEND_URL = process.env.BACKEND_URL || 'http://localhost:3000';
|
||||
const VALID_TIERS = new Set(['analyst', 'desk']);
|
||||
|
||||
/**
|
||||
* Checkout proxy — Next.js → Express → Stripe.
|
||||
*
|
||||
* Session 8 cutover: previously this route created NexaPay payment
|
||||
* links; now it forwards to the Express `/api/stripe/checkout` route
|
||||
* (Session 3.4 + 7i) which creates a Stripe Checkout Session
|
||||
* server-side. The browser never sees `sk_test_*` / `sk_live_*` —
|
||||
* only the resulting `https://checkout.stripe.com/...` redirect URL.
|
||||
*
|
||||
* Response shape — preserves the existing `{ url }` field so older
|
||||
* Pricing CTA code that read `.url` keeps working. Express returns
|
||||
* `{ checkout_url, session_id }`; we rename and forward both so
|
||||
* either field name resolves on the client.
|
||||
*/
|
||||
async function handle(req: NextRequest) {
|
||||
const user = await getUserFromRequest(req);
|
||||
if (!user) return jsonError(401, 'Log in to upgrade.');
|
||||
|
||||
const tier = await resolveTier(req);
|
||||
if (!tier) return jsonError(400, 'Pick a valid tier (analyst or desk).');
|
||||
|
||||
// Founder pricing eligibility — first 100 paid users overall
|
||||
let founderEligible = false;
|
||||
const sb = getServiceRoleSupabase();
|
||||
if (sb) {
|
||||
const { count } = await sb
|
||||
.from('user_profiles')
|
||||
.select('id', { count: 'exact', head: true })
|
||||
.eq('founder_pricing', true);
|
||||
founderEligible = (count ?? 0) < 100;
|
||||
// Tier resolution — query string for GET (button hrefs), body for POST.
|
||||
let tier: string | null = null;
|
||||
let founderCode: string | undefined;
|
||||
const url = new URL(req.url);
|
||||
const queryTier = url.searchParams.get('tier');
|
||||
if (queryTier) tier = queryTier;
|
||||
if (req.method === 'POST') {
|
||||
try {
|
||||
const body = (await req.json().catch(() => ({}))) as { tier?: string; founder_code?: string };
|
||||
if (body.tier) tier = body.tier;
|
||||
if (body.founder_code) founderCode = body.founder_code;
|
||||
} catch {
|
||||
/* tier may still be on the query string */
|
||||
}
|
||||
}
|
||||
if (!tier || !VALID_TIERS.has(tier)) {
|
||||
return jsonError(400, 'Pick a valid tier (analyst or desk).');
|
||||
}
|
||||
|
||||
const pricing = TIER_PRICING[tier];
|
||||
const amount = founderEligible ? pricing.founder : pricing.regular;
|
||||
// Forward to Express. The bearer token from the browser is the same
|
||||
// one Express's requireAuth verifies — no token rewriting on this hop.
|
||||
const authHeader = req.headers.get('authorization');
|
||||
if (!authHeader) return jsonError(401, 'Log in to upgrade.');
|
||||
|
||||
try {
|
||||
const link = await createPaymentLink({
|
||||
userId: user.id,
|
||||
tier,
|
||||
amount,
|
||||
description: `${pricing.label}${founderEligible ? ' (Founder)' : ''}`,
|
||||
founderPricing: founderEligible,
|
||||
const upstream = await fetch(`${BACKEND_URL}/api/stripe/checkout`, {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
Authorization: authHeader,
|
||||
},
|
||||
body: JSON.stringify({ tier, ...(founderCode ? { founder_code: founderCode } : {}) }),
|
||||
});
|
||||
|
||||
// For GET (used by Pricing CTA links), redirect directly.
|
||||
if (req.method === 'GET') {
|
||||
return NextResponse.redirect(link.url, { status: 303 });
|
||||
const data = (await upstream.json().catch(() => ({}))) as {
|
||||
checkout_url?: string;
|
||||
session_id?: string;
|
||||
error?: string;
|
||||
};
|
||||
|
||||
if (!upstream.ok) {
|
||||
return NextResponse.json(
|
||||
{ error: data.error || 'Checkout creation failed. Try again in a moment.' },
|
||||
{ status: upstream.status },
|
||||
);
|
||||
}
|
||||
|
||||
return NextResponse.json({ url: link.url, expires_at: link.expires_at, founder_pricing: founderEligible });
|
||||
} catch (err) {
|
||||
console.error('[checkout] NexaPay link failed', err);
|
||||
const checkoutUrl = data.checkout_url;
|
||||
if (!checkoutUrl) {
|
||||
// Defensive: Express returned 200 with no URL — should never happen,
|
||||
// but if it does we don't want to silently redirect to undefined.
|
||||
return jsonError(502, 'Checkout creation incomplete. Try again.');
|
||||
}
|
||||
|
||||
// GET requests (used by legacy <a> hrefs) redirect directly so a
|
||||
// plain link click flows to Stripe without JS.
|
||||
if (req.method === 'GET') {
|
||||
return NextResponse.redirect(checkoutUrl, { status: 303 });
|
||||
}
|
||||
|
||||
// POST returns JSON so the new Pricing onClick handler can navigate
|
||||
// explicitly (gives us a place to show loading state first).
|
||||
return NextResponse.json({
|
||||
url: checkoutUrl,
|
||||
checkout_url: checkoutUrl,
|
||||
session_id: data.session_id,
|
||||
});
|
||||
} catch {
|
||||
return jsonError(502, 'Payment processor is unreachable. Try again in a moment.');
|
||||
}
|
||||
}
|
||||
|
||||
export async function GET(req: NextRequest) { return handle(req); }
|
||||
export async function POST(req: NextRequest) { return handle(req); }
|
||||
|
||||
@@ -0,0 +1,44 @@
|
||||
import { NextRequest, NextResponse } from 'next/server';
|
||||
|
||||
export const dynamic = 'force-dynamic';
|
||||
|
||||
const BACKEND_URL = process.env.BACKEND_URL || 'http://localhost:3000';
|
||||
|
||||
// Frozen at the same set Express validates against
|
||||
// (`src/services/oddsService.js SOCCER_SPORT_KEYS`). Duplicated here so
|
||||
// a typo'd league bounces at the Next layer without burning a backend
|
||||
// round-trip.
|
||||
const VALID_LEAGUES = new Set([
|
||||
'wc', 'epl', 'laliga', 'bundesliga', 'seriea',
|
||||
'ligue1', 'ucl', 'mls', 'ligamx',
|
||||
]);
|
||||
|
||||
export async function GET(req: NextRequest, { params }: { params: Promise<{ league: string }> }) {
|
||||
const { league } = await params;
|
||||
const leagueLc = String(league || '').toLowerCase();
|
||||
if (!VALID_LEAGUES.has(leagueLc)) {
|
||||
return NextResponse.json(
|
||||
{ error: `Unknown soccer league. Valid: ${[...VALID_LEAGUES].join(', ')}.` },
|
||||
{ status: 400 },
|
||||
);
|
||||
}
|
||||
|
||||
// Pass the original query string through (filters: book, stat_type).
|
||||
const qs = req.nextUrl.search;
|
||||
try {
|
||||
const upstream = await fetch(`${BACKEND_URL}/api/odds/soccer/${leagueLc}${qs}`, {
|
||||
method: 'GET',
|
||||
headers: { Accept: 'application/json' },
|
||||
});
|
||||
const data = await upstream.json().catch(() => ({}));
|
||||
if (!upstream.ok) {
|
||||
return NextResponse.json(data, { status: upstream.status });
|
||||
}
|
||||
return NextResponse.json(data);
|
||||
} catch {
|
||||
return NextResponse.json(
|
||||
{ error: 'Odds service is unreachable. Try again in a moment.' },
|
||||
{ status: 502 },
|
||||
);
|
||||
}
|
||||
}
|
||||
@@ -9,18 +9,22 @@ const monthKey = () => new Date().toISOString().slice(0, 7) + '-01';
|
||||
const isSameMonth = (date: string | null | undefined) =>
|
||||
!!date && date.slice(0, 7) === new Date().toISOString().slice(0, 7);
|
||||
|
||||
const VALID_SPORTS = new Set(['NBA', 'MLB', 'WNBA']);
|
||||
const VALID_SPORTS = new Set(['NBA', 'MLB', 'WNBA', 'Soccer']);
|
||||
const VALID_DIRECTIONS = new Set(['over', 'under']);
|
||||
const VALID_NBA_STATS = new Set(['points', 'rebounds', 'assists', 'threes', 'blocks', 'steals', 'pra', 'turnovers']);
|
||||
const VALID_MLB_STATS = new Set([
|
||||
'strikeouts', 'hits_allowed', 'earned_runs', 'innings_pitched', 'walks_allowed',
|
||||
'hits', 'total_bases', 'rbi', 'runs', 'stolen_bases', 'home_runs', 'walks', 'singles', 'doubles',
|
||||
]);
|
||||
const VALID_SOCCER_STATS = new Set([
|
||||
'goals', 'assists', 'shots_on_target', 'shots', 'tackles',
|
||||
'cards', 'corners', 'saves', 'goals_conceded', 'passes', 'clean_sheet',
|
||||
]);
|
||||
|
||||
export const dynamic = 'force-dynamic';
|
||||
|
||||
interface ScanBody {
|
||||
sport: 'NBA' | 'MLB' | 'WNBA';
|
||||
sport: 'NBA' | 'MLB' | 'WNBA' | 'Soccer';
|
||||
player: string;
|
||||
stat: string;
|
||||
line: number;
|
||||
@@ -45,7 +49,10 @@ export async function POST(req: NextRequest) {
|
||||
return jsonError(400, 'Line must be a number between 0 and 500.');
|
||||
}
|
||||
|
||||
const validStats = body.sport === 'MLB' ? VALID_MLB_STATS : VALID_NBA_STATS;
|
||||
const validStats =
|
||||
body.sport === 'MLB' ? VALID_MLB_STATS :
|
||||
body.sport === 'Soccer' ? VALID_SOCCER_STATS :
|
||||
VALID_NBA_STATS;
|
||||
if (!validStats.has(body.stat)) {
|
||||
return jsonError(400, `Stat "${body.stat}" not supported for ${body.sport}.`);
|
||||
}
|
||||
|
||||
@@ -0,0 +1,414 @@
|
||||
'use client';
|
||||
|
||||
import { useCallback, useEffect, useState } from 'react';
|
||||
import { useRouter } from 'next/navigation';
|
||||
import SportSelector, { SoccerLeague, SportSelection } from '@/components/SportSelector';
|
||||
import SoccerGradeResult, { SoccerGradeResultProps } from '@/components/SoccerGradeResult';
|
||||
import { useAuth } from '@/contexts/AuthContext';
|
||||
|
||||
/**
|
||||
* /soccer — live soccer odds feed.
|
||||
*
|
||||
* Shows match cards for the selected league (defaults to World Cup
|
||||
* 2026). Each match expands to reveal player props grouped by stat
|
||||
* type. Clicking a prop hands it off to /scan for grading.
|
||||
*
|
||||
* Data path: this page → /api/odds/soccer/:league → Express
|
||||
* /api/odds/soccer/:league → odds-api. The Express route falls back
|
||||
* to cache when the API quota is low; the response carries `source:
|
||||
* 'cache' | 'live'` so we can tag the freshness.
|
||||
*/
|
||||
|
||||
interface NormalizedProp {
|
||||
player: string;
|
||||
stat_type: string;
|
||||
line: number;
|
||||
direction: 'over' | 'under';
|
||||
book: string;
|
||||
odds: number;
|
||||
game_time?: string;
|
||||
home_team?: string;
|
||||
away_team?: string;
|
||||
fetched_at?: string;
|
||||
}
|
||||
|
||||
interface GroupedProp {
|
||||
player: string;
|
||||
stat_type: string;
|
||||
line: number;
|
||||
game_time?: string;
|
||||
home_team?: string;
|
||||
away_team?: string;
|
||||
// best line per direction across books
|
||||
over?: { book: string; odds: number };
|
||||
under?: { book: string; odds: number };
|
||||
}
|
||||
|
||||
interface OddsResponse {
|
||||
sport: string;
|
||||
updated_at?: string;
|
||||
source?: string;
|
||||
quota_remaining?: number;
|
||||
props: GroupedProp[];
|
||||
message?: string;
|
||||
error?: string;
|
||||
}
|
||||
|
||||
// Group a flat props array by (player, stat_type, line) so each row
|
||||
// represents a SINGLE prop with both directions next to each other.
|
||||
// The Express response already does some grouping but ships per-direction
|
||||
// rows — collapse them.
|
||||
function groupProps(props: GroupedProp[]): GroupedProp[] {
|
||||
return props || [];
|
||||
}
|
||||
|
||||
// Group props under their match for the card layout.
|
||||
function groupByMatch(props: GroupedProp[]) {
|
||||
const matches = new Map<string, { home: string; away: string; time?: string; propsByStatType: Map<string, GroupedProp[]> }>();
|
||||
for (const p of props) {
|
||||
const home = p.home_team || '?';
|
||||
const away = p.away_team || '?';
|
||||
const key = `${home}__${away}__${p.game_time || ''}`;
|
||||
if (!matches.has(key)) {
|
||||
matches.set(key, { home, away, time: p.game_time, propsByStatType: new Map() });
|
||||
}
|
||||
const m = matches.get(key)!;
|
||||
const list = m.propsByStatType.get(p.stat_type) || [];
|
||||
list.push(p);
|
||||
m.propsByStatType.set(p.stat_type, list);
|
||||
}
|
||||
return Array.from(matches.values());
|
||||
}
|
||||
|
||||
const STAT_LABELS: Record<string, string> = {
|
||||
goals: 'Anytime / Total Goals',
|
||||
shots_on_target: 'Shots on Target',
|
||||
shots: 'Total Shots',
|
||||
tackles: 'Tackles',
|
||||
cards: 'Cards',
|
||||
corners: 'Corners',
|
||||
saves: 'Saves',
|
||||
goals_conceded: 'Goals Conceded',
|
||||
passes: 'Passes',
|
||||
clean_sheet: 'Clean Sheet',
|
||||
assists: 'Assists',
|
||||
};
|
||||
|
||||
function formatTime(iso?: string) {
|
||||
if (!iso) return '';
|
||||
try {
|
||||
const d = new Date(iso);
|
||||
return d.toLocaleString(undefined, {
|
||||
weekday: 'short', hour: 'numeric', minute: '2-digit',
|
||||
});
|
||||
} catch {
|
||||
return iso;
|
||||
}
|
||||
}
|
||||
|
||||
export default function SoccerOddsPage() {
|
||||
const router = useRouter();
|
||||
const { session } = useAuth();
|
||||
const [selection, setSelection] = useState<SportSelection>({ sport: 'Soccer', league: 'wc' });
|
||||
const [data, setData] = useState<OddsResponse | null>(null);
|
||||
const [loading, setLoading] = useState(true);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
const [expanded, setExpanded] = useState<Set<string>>(new Set());
|
||||
|
||||
const [scanning, setScanning] = useState(false);
|
||||
const [scanResult, setScanResult] = useState<SoccerGradeResultProps | null>(null);
|
||||
const [scanError, setScanError] = useState<string | null>(null);
|
||||
|
||||
const league: SoccerLeague = selection.league || 'wc';
|
||||
|
||||
// Redirect non-Soccer sport selections back to /scan — that page
|
||||
// owns NBA/MLB/WNBA. Soccer is the only one this page serves.
|
||||
useEffect(() => {
|
||||
if (selection.sport !== 'Soccer') {
|
||||
router.push('/scan');
|
||||
}
|
||||
}, [selection.sport, router]);
|
||||
|
||||
async function gradeProp(player: string, stat_type: string, lineVal: number) {
|
||||
setScanError(null);
|
||||
setScanResult(null);
|
||||
setScanning(true);
|
||||
try {
|
||||
const res = await fetch('/api/scan', {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
...(session?.access_token ? { Authorization: `Bearer ${session.access_token}` } : {}),
|
||||
},
|
||||
body: JSON.stringify({
|
||||
sport: 'Soccer',
|
||||
player,
|
||||
stat: stat_type,
|
||||
line: lineVal,
|
||||
direction: 'over',
|
||||
book: 'draftkings',
|
||||
}),
|
||||
});
|
||||
const body = (await res.json().catch(() => ({}))) as Record<string, unknown> & { error?: string };
|
||||
if (!res.ok) {
|
||||
setScanError(body.error || 'The engine hit a wall. Try that read again.');
|
||||
setScanning(false);
|
||||
return;
|
||||
}
|
||||
const result: SoccerGradeResultProps = {
|
||||
player,
|
||||
stat_type,
|
||||
line: lineVal,
|
||||
direction: 'over',
|
||||
league,
|
||||
grade: String(body.grade || 'C'),
|
||||
confidence: typeof body.confidence === 'number' ? body.confidence : undefined,
|
||||
edge_pct: typeof body.edge_pct === 'number' ? body.edge_pct : undefined,
|
||||
reasoning: (body.reasoning as SoccerGradeResultProps['reasoning']) || undefined,
|
||||
kill_conditions_triggered: (body.kill_conditions_triggered as SoccerGradeResultProps['kill_conditions_triggered']) || [],
|
||||
tier_gated: !!body.tier_gated,
|
||||
upgrade_hint: typeof body.upgrade_hint === 'string' ? body.upgrade_hint : undefined,
|
||||
onUpgradeClick: () => router.push('/#pricing'),
|
||||
onClose: () => setScanResult(null),
|
||||
};
|
||||
setScanResult(result);
|
||||
} catch {
|
||||
setScanError('Network error. Try again.');
|
||||
} finally {
|
||||
setScanning(false);
|
||||
}
|
||||
}
|
||||
|
||||
const fetchOdds = useCallback(async () => {
|
||||
setLoading(true);
|
||||
setError(null);
|
||||
try {
|
||||
const res = await fetch(`/api/odds/soccer/${league}`, { cache: 'no-store' });
|
||||
const body = (await res.json().catch(() => ({}))) as OddsResponse;
|
||||
if (!res.ok) {
|
||||
setError(body.error || 'Couldn’t load odds. Try again.');
|
||||
setData(null);
|
||||
} else {
|
||||
setData(body);
|
||||
}
|
||||
} catch {
|
||||
setError('Network error. Try again.');
|
||||
setData(null);
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
}, [league]);
|
||||
|
||||
useEffect(() => {
|
||||
fetchOdds();
|
||||
}, [fetchOdds]);
|
||||
|
||||
const matches = data ? groupByMatch(groupProps(data.props)) : [];
|
||||
|
||||
return (
|
||||
<main style={{ minHeight: '100vh', padding: '24px 16px 80px' }}>
|
||||
<div style={{ maxWidth: 1100, margin: '0 auto' }}>
|
||||
<header style={{ marginBottom: 24 }}>
|
||||
<h1 style={{ fontSize: 'clamp(24px, 3vw, 36px)', fontWeight: 700, letterSpacing: '-0.02em', marginBottom: 8 }}>
|
||||
Soccer odds
|
||||
</h1>
|
||||
<p style={{ color: 'var(--text-secondary)', fontSize: 14 }}>
|
||||
Live odds across our launch leagues. Click any prop to grade it through the VYNDR engine.
|
||||
</p>
|
||||
</header>
|
||||
|
||||
<div style={{ marginBottom: 20 }}>
|
||||
<SportSelector
|
||||
initialSport="Soccer"
|
||||
initialLeague={league}
|
||||
onChange={(sel) => setSelection(sel)}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{data?.source && (
|
||||
<p
|
||||
className="mono"
|
||||
style={{
|
||||
fontSize: 11,
|
||||
color: 'var(--text-tertiary)',
|
||||
marginBottom: 16,
|
||||
letterSpacing: '0.06em',
|
||||
textTransform: 'uppercase',
|
||||
}}
|
||||
>
|
||||
{data.updated_at ? `Updated ${formatTime(data.updated_at)} · ` : ''}
|
||||
source: {data.source}
|
||||
{typeof data.quota_remaining === 'number' ? ` · quota: ${data.quota_remaining}` : ''}
|
||||
</p>
|
||||
)}
|
||||
|
||||
{loading && (
|
||||
<div style={{ padding: 40, textAlign: 'center', color: 'var(--text-tertiary)' }}>Loading odds…</div>
|
||||
)}
|
||||
|
||||
{error && (
|
||||
<div
|
||||
role="alert"
|
||||
style={{
|
||||
padding: 16,
|
||||
border: '1px solid var(--grade-d, #ff5a5a)',
|
||||
color: 'var(--grade-d, #ff5a5a)',
|
||||
borderRadius: 8,
|
||||
marginBottom: 20,
|
||||
}}
|
||||
>
|
||||
{error}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{scanError && (
|
||||
<div
|
||||
role="alert"
|
||||
style={{
|
||||
padding: 12,
|
||||
border: '1px solid var(--grade-d, #ff5a5a)',
|
||||
color: 'var(--grade-d, #ff5a5a)',
|
||||
borderRadius: 6,
|
||||
marginBottom: 16,
|
||||
fontSize: 13,
|
||||
}}
|
||||
>
|
||||
{scanError}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{scanResult && (
|
||||
<SoccerGradeResult {...scanResult} />
|
||||
)}
|
||||
|
||||
{!loading && !error && matches.length === 0 && (
|
||||
<div
|
||||
className="surface"
|
||||
style={{
|
||||
padding: 32,
|
||||
border: '1px solid var(--border)',
|
||||
borderRadius: 8,
|
||||
textAlign: 'center',
|
||||
color: 'var(--text-secondary)',
|
||||
}}
|
||||
>
|
||||
No live matches with props in this league right now.
|
||||
{league !== 'wc' && (
|
||||
<p style={{ fontSize: 13, marginTop: 8, color: 'var(--text-tertiary)' }}>
|
||||
Off-season or between matchdays — World Cup props are running through July 19, 2026.
|
||||
</p>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div style={{ display: 'grid', gap: 16 }}>
|
||||
{matches.map((m, idx) => {
|
||||
const matchKey = `${m.home}-${m.away}-${idx}`;
|
||||
const isOpen = expanded.has(matchKey);
|
||||
return (
|
||||
<article
|
||||
key={matchKey}
|
||||
className="surface diagonal-cut"
|
||||
style={{
|
||||
padding: 20,
|
||||
border: '1px solid var(--border)',
|
||||
background: 'var(--bg-surface)',
|
||||
borderRadius: 8,
|
||||
}}
|
||||
>
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => {
|
||||
const next = new Set(expanded);
|
||||
if (next.has(matchKey)) next.delete(matchKey);
|
||||
else next.add(matchKey);
|
||||
setExpanded(next);
|
||||
}}
|
||||
style={{
|
||||
width: '100%',
|
||||
display: 'flex',
|
||||
justifyContent: 'space-between',
|
||||
alignItems: 'center',
|
||||
background: 'transparent',
|
||||
border: 0,
|
||||
cursor: 'pointer',
|
||||
color: 'inherit',
|
||||
padding: 0,
|
||||
textAlign: 'left',
|
||||
}}
|
||||
>
|
||||
<div>
|
||||
<div style={{ fontSize: 18, fontWeight: 700, letterSpacing: '-0.01em' }}>
|
||||
{m.away} <span style={{ color: 'var(--text-tertiary)' }}>vs</span> {m.home}
|
||||
</div>
|
||||
<div className="mono" style={{ fontSize: 11, color: 'var(--text-tertiary)', marginTop: 4, letterSpacing: '0.06em' }}>
|
||||
{formatTime(m.time)} · {m.propsByStatType.size} stat type(s)
|
||||
</div>
|
||||
</div>
|
||||
<span style={{ color: 'var(--text-secondary)', fontSize: 14 }}>{isOpen ? '−' : '+'}</span>
|
||||
</button>
|
||||
|
||||
{isOpen && (
|
||||
<div style={{ marginTop: 16, display: 'grid', gap: 12 }}>
|
||||
{Array.from(m.propsByStatType.entries()).map(([statType, list]) => (
|
||||
<section key={statType}>
|
||||
<h3
|
||||
className="mono"
|
||||
style={{
|
||||
fontSize: 11,
|
||||
color: 'var(--grade-a)',
|
||||
letterSpacing: '0.08em',
|
||||
textTransform: 'uppercase',
|
||||
marginBottom: 8,
|
||||
}}
|
||||
>
|
||||
{STAT_LABELS[statType] || statType}
|
||||
</h3>
|
||||
<ul style={{ display: 'grid', gap: 6 }}>
|
||||
{list.slice(0, 8).map((p, j) => (
|
||||
<li
|
||||
key={`${p.player}-${p.line}-${j}`}
|
||||
style={{
|
||||
display: 'flex',
|
||||
justifyContent: 'space-between',
|
||||
alignItems: 'center',
|
||||
padding: '8px 12px',
|
||||
border: '1px solid var(--border)',
|
||||
borderRadius: 6,
|
||||
background: 'var(--bg-elevated)',
|
||||
fontSize: 14,
|
||||
}}
|
||||
>
|
||||
<span style={{ fontWeight: 600 }}>{p.player}</span>
|
||||
<span className="mono" style={{ color: 'var(--text-secondary)', fontSize: 13 }}>
|
||||
{p.line.toFixed(1)}
|
||||
</span>
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => gradeProp(p.player, statType, p.line)}
|
||||
disabled={scanning}
|
||||
className="btn-ghost"
|
||||
style={{
|
||||
padding: '4px 12px',
|
||||
fontSize: 12,
|
||||
cursor: scanning ? 'not-allowed' : 'pointer',
|
||||
opacity: scanning ? 0.6 : 1,
|
||||
}}
|
||||
>
|
||||
{scanning ? '…' : 'Grade'}
|
||||
</button>
|
||||
</li>
|
||||
))}
|
||||
</ul>
|
||||
</section>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
</article>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
</div>
|
||||
</main>
|
||||
);
|
||||
}
|
||||
@@ -0,0 +1,39 @@
|
||||
import Link from 'next/link';
|
||||
|
||||
/**
|
||||
* Stripe cancel landing — Stripe sends users here when they bail out
|
||||
* of checkout. We don't gate, judge, or guilt; just acknowledge and
|
||||
* point them back to pricing.
|
||||
*/
|
||||
export default function UpgradeCancelPage() {
|
||||
return (
|
||||
<main style={{ minHeight: '100vh', display: 'flex', alignItems: 'center', justifyContent: 'center', padding: 24 }}>
|
||||
<article
|
||||
className="surface diagonal-cut"
|
||||
style={{
|
||||
maxWidth: 560,
|
||||
width: '100%',
|
||||
padding: 40,
|
||||
textAlign: 'center',
|
||||
border: '1px solid var(--border)',
|
||||
background: 'var(--bg-surface)',
|
||||
}}
|
||||
>
|
||||
<h1 style={{ fontSize: 28, fontWeight: 700, letterSpacing: '-0.02em', marginBottom: 12 }}>
|
||||
Checkout cancelled.
|
||||
</h1>
|
||||
<p style={{ fontSize: 15, color: 'var(--text-secondary)', marginBottom: 32 }}>
|
||||
Your account is unchanged. No card was charged.
|
||||
</p>
|
||||
<div style={{ display: 'flex', flexDirection: 'column', gap: 12 }}>
|
||||
<Link href="/#pricing" className="btn-primary" style={{ padding: 14, width: '100%' }}>
|
||||
Back to pricing
|
||||
</Link>
|
||||
<Link href="/scan" className="btn-ghost" style={{ padding: 14, width: '100%' }}>
|
||||
Keep using the free tier
|
||||
</Link>
|
||||
</div>
|
||||
</article>
|
||||
</main>
|
||||
);
|
||||
}
|
||||
@@ -0,0 +1,106 @@
|
||||
'use client';
|
||||
|
||||
import { Suspense, useEffect } from 'react';
|
||||
import Link from 'next/link';
|
||||
import { useSearchParams } from 'next/navigation';
|
||||
import { useAuth } from '@/contexts/AuthContext';
|
||||
|
||||
/**
|
||||
* Stripe checkout success landing.
|
||||
*
|
||||
* Stripe redirects to /upgrade/success?session_id={CHECKOUT_SESSION_ID}
|
||||
* after a completed checkout. Webhook events (handled server-side in
|
||||
* `src/services/stripeService.handleWebhookEvent`) write the user's
|
||||
* new tier into Supabase — there's nothing for the client to do here
|
||||
* besides confirm the redirect and refresh the auth context so the
|
||||
* new tier shows up on subsequent reads.
|
||||
*
|
||||
* We deliberately do NOT verify the session against Stripe from the
|
||||
* client (no secret key on the browser). The webhook is the source of
|
||||
* truth; this page just acknowledges the user landed.
|
||||
*/
|
||||
function SuccessInner() {
|
||||
const search = useSearchParams();
|
||||
const sessionId = search.get('session_id');
|
||||
const { refresh, profile } = useAuth();
|
||||
|
||||
useEffect(() => {
|
||||
// Force a profile re-read so the new tier flips in the UI without
|
||||
// requiring a manual sign-out/in.
|
||||
refresh();
|
||||
}, [refresh]);
|
||||
|
||||
const tierLabel = profile?.tier === 'desk' ? 'Desk' : profile?.tier === 'analyst' ? 'Analyst' : '';
|
||||
|
||||
return (
|
||||
<main style={{ minHeight: '100vh', display: 'flex', alignItems: 'center', justifyContent: 'center', padding: '24px' }}>
|
||||
<article
|
||||
className="surface diagonal-cut"
|
||||
style={{
|
||||
maxWidth: 560,
|
||||
width: '100%',
|
||||
padding: 40,
|
||||
textAlign: 'center',
|
||||
border: '1px solid var(--grade-a)',
|
||||
background: 'var(--bg-elevated)',
|
||||
boxShadow: '0 16px 48px var(--accent-glow)',
|
||||
}}
|
||||
>
|
||||
<div
|
||||
className="mono"
|
||||
style={{
|
||||
display: 'inline-block',
|
||||
padding: '4px 12px',
|
||||
background: 'var(--grade-a)',
|
||||
color: 'var(--bg-primary)',
|
||||
fontSize: 10,
|
||||
fontWeight: 800,
|
||||
letterSpacing: '0.08em',
|
||||
borderRadius: 999,
|
||||
textTransform: 'uppercase',
|
||||
marginBottom: 24,
|
||||
}}
|
||||
>
|
||||
Founder Pricing Locked
|
||||
</div>
|
||||
|
||||
<h1 style={{ fontSize: 32, fontWeight: 700, letterSpacing: '-0.02em', marginBottom: 12 }}>
|
||||
Welcome to VYNDR{tierLabel ? ` ${tierLabel}` : ''}.
|
||||
</h1>
|
||||
|
||||
<p style={{ fontSize: 16, color: 'var(--text-secondary)', marginBottom: 8 }}>
|
||||
Your beta pricing is locked for as long as you stay subscribed.
|
||||
</p>
|
||||
<p style={{ fontSize: 14, color: 'var(--text-tertiary)', marginBottom: 32 }}>
|
||||
Cancel anytime in your profile.
|
||||
</p>
|
||||
|
||||
{sessionId && (
|
||||
<p
|
||||
className="mono"
|
||||
style={{ fontSize: 11, color: 'var(--text-tertiary)', marginBottom: 24, wordBreak: 'break-all' }}
|
||||
>
|
||||
Session: {sessionId}
|
||||
</p>
|
||||
)}
|
||||
|
||||
<div style={{ display: 'flex', flexDirection: 'column', gap: 12 }}>
|
||||
<Link href="/scan" className="btn-primary" style={{ padding: 14, width: '100%' }}>
|
||||
Start scanning
|
||||
</Link>
|
||||
<Link href="/profile" className="btn-ghost" style={{ padding: 14, width: '100%' }}>
|
||||
View account
|
||||
</Link>
|
||||
</div>
|
||||
</article>
|
||||
</main>
|
||||
);
|
||||
}
|
||||
|
||||
export default function UpgradeSuccessPage() {
|
||||
return (
|
||||
<Suspense fallback={<main style={{ padding: 40, textAlign: 'center' }}>Loading…</main>}>
|
||||
<SuccessInner />
|
||||
</Suspense>
|
||||
);
|
||||
}
|
||||
@@ -47,10 +47,33 @@ export default function Nav() {
|
||||
>
|
||||
<a
|
||||
href="/"
|
||||
style={{ color: 'var(--text-0)', textDecoration: 'none', display: 'inline-flex', alignItems: 'center' }}
|
||||
style={{ color: 'var(--text-0)', textDecoration: 'none', display: 'inline-flex', alignItems: 'center', gap: 10 }}
|
||||
aria-label="VYNDR — home"
|
||||
>
|
||||
<Wordmark size={22} />
|
||||
{/* Session 8 — beta tag. Tiny, glitch-styled, sits next to
|
||||
the wordmark so it reads as part of the brand rather than
|
||||
a banner. Renders on every page that mounts Nav. */}
|
||||
<span
|
||||
className="mono"
|
||||
aria-label="Beta"
|
||||
style={{
|
||||
fontSize: 9,
|
||||
fontWeight: 800,
|
||||
letterSpacing: '0.14em',
|
||||
padding: '2px 5px',
|
||||
color: 'var(--grade-a)',
|
||||
border: '1px solid var(--grade-a)',
|
||||
borderRadius: 3,
|
||||
textTransform: 'uppercase',
|
||||
opacity: 0.85,
|
||||
lineHeight: 1,
|
||||
position: 'relative',
|
||||
top: -2,
|
||||
}}
|
||||
>
|
||||
BETA
|
||||
</span>
|
||||
</a>
|
||||
|
||||
<div className="nav-desktop" style={{ display: 'none', gap: 28, alignItems: 'center' }}>
|
||||
|
||||
+168
-77
@@ -1,6 +1,26 @@
|
||||
'use client';
|
||||
|
||||
const TIERS = [
|
||||
import { useState } from 'react';
|
||||
import { useRouter } from 'next/navigation';
|
||||
import { useAuth } from '@/contexts/AuthContext';
|
||||
|
||||
type TierId = 'free' | 'analyst' | 'desk';
|
||||
|
||||
interface TierConfig {
|
||||
id: TierId;
|
||||
name: string;
|
||||
price: string;
|
||||
originalPrice?: string;
|
||||
cadence: string;
|
||||
badge?: string;
|
||||
headline: string;
|
||||
cta: string;
|
||||
features: string[];
|
||||
locked: string[];
|
||||
highlight: boolean;
|
||||
}
|
||||
|
||||
const TIERS: TierConfig[] = [
|
||||
{
|
||||
id: 'free',
|
||||
name: 'Free',
|
||||
@@ -8,7 +28,6 @@ const TIERS = [
|
||||
cadence: '/mo',
|
||||
headline: 'Try the model. No card required.',
|
||||
cta: 'Start Free',
|
||||
ctaHref: '/signup',
|
||||
features: [
|
||||
'5 reads per month',
|
||||
'Grade letter + projection',
|
||||
@@ -31,7 +50,6 @@ const TIERS = [
|
||||
badge: 'Founder Access',
|
||||
headline: 'The full intelligence layer.',
|
||||
cta: 'Lock Founder Price',
|
||||
ctaHref: '/api/checkout?tier=analyst',
|
||||
features: [
|
||||
'Unlimited reads',
|
||||
'Full factor analysis (40+ signals)',
|
||||
@@ -54,7 +72,6 @@ const TIERS = [
|
||||
cadence: '/mo',
|
||||
headline: 'Everything. The professional setup.',
|
||||
cta: 'Go Desk',
|
||||
ctaHref: '/api/checkout?tier=desk',
|
||||
features: [
|
||||
'Everything in Analyst',
|
||||
'Alt line ladder + edge ranking',
|
||||
@@ -62,7 +79,6 @@ const TIERS = [
|
||||
'Real-time intelligence feed',
|
||||
'Parlay correlation analysis (phi)',
|
||||
'Consensus vs model comparison',
|
||||
'API access (coming Q3)',
|
||||
],
|
||||
locked: [],
|
||||
highlight: false,
|
||||
@@ -70,6 +86,51 @@ const TIERS = [
|
||||
];
|
||||
|
||||
export default function Pricing() {
|
||||
const router = useRouter();
|
||||
const { session, loading: authLoading } = useAuth();
|
||||
const [pending, setPending] = useState<TierId | null>(null);
|
||||
const [error, setError] = useState<string | null>(null);
|
||||
|
||||
async function startCheckout(tier: TierId) {
|
||||
setError(null);
|
||||
|
||||
// Free tier short-circuits — no checkout, just signup.
|
||||
if (tier === 'free') {
|
||||
router.push('/signup');
|
||||
return;
|
||||
}
|
||||
|
||||
// Anonymous → bounce to signup with a returnTo back to /#pricing.
|
||||
if (!session) {
|
||||
router.push('/signup?return=/%23pricing');
|
||||
return;
|
||||
}
|
||||
|
||||
setPending(tier);
|
||||
try {
|
||||
const res = await fetch('/api/checkout', {
|
||||
method: 'POST',
|
||||
headers: {
|
||||
'Content-Type': 'application/json',
|
||||
Authorization: `Bearer ${session.access_token}`,
|
||||
},
|
||||
body: JSON.stringify({ tier }),
|
||||
});
|
||||
const data = (await res.json().catch(() => ({}))) as { url?: string; error?: string };
|
||||
if (!res.ok || !data.url) {
|
||||
setError(data.error || 'Checkout creation failed. Try again in a moment.');
|
||||
setPending(null);
|
||||
return;
|
||||
}
|
||||
// Hand off to Stripe. The success_url returns the user to
|
||||
// /upgrade/success?session_id=… — no further client work needed.
|
||||
window.location.assign(data.url);
|
||||
} catch {
|
||||
setError('Network error. Try again.');
|
||||
setPending(null);
|
||||
}
|
||||
}
|
||||
|
||||
return (
|
||||
<section
|
||||
id="pricing"
|
||||
@@ -87,89 +148,119 @@ export default function Pricing() {
|
||||
Pricing built for bettors. Not for SaaS investors.
|
||||
</h2>
|
||||
<p style={{ fontSize: 17, color: 'var(--text-secondary)' }}>
|
||||
First 100 users lock $14.99/mo for life. This price dies at user 101.
|
||||
First 100 users lock $14.99/mo for life. Beta pricing — this price dies at user 101.
|
||||
</p>
|
||||
</header>
|
||||
|
||||
{error && (
|
||||
<div
|
||||
role="alert"
|
||||
style={{
|
||||
maxWidth: 720,
|
||||
margin: '0 auto 24px',
|
||||
padding: 14,
|
||||
border: '1px solid var(--grade-d, #ff5a5a)',
|
||||
color: 'var(--grade-d, #ff5a5a)',
|
||||
fontSize: 14,
|
||||
textAlign: 'center',
|
||||
borderRadius: 8,
|
||||
}}
|
||||
>
|
||||
{error}
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="pricing-grid" style={{ display: 'grid', gap: 24 }}>
|
||||
{TIERS.map((tier, i) => (
|
||||
<article
|
||||
key={tier.id}
|
||||
className={`surface diagonal-cut${tier.highlight ? ' diagonal-cut-strong' : ''} animate-fade-up stagger-${i + 1}`}
|
||||
style={{
|
||||
padding: 32,
|
||||
position: 'relative',
|
||||
border: tier.highlight ? '1px solid var(--grade-a)' : '1px solid var(--border)',
|
||||
background: tier.highlight ? 'var(--bg-elevated)' : 'var(--bg-surface)',
|
||||
boxShadow: tier.highlight ? '0 16px 48px var(--accent-glow)' : 'none',
|
||||
}}
|
||||
>
|
||||
{tier.badge && (
|
||||
<div
|
||||
className="mono"
|
||||
{TIERS.map((tier, i) => {
|
||||
const isPending = pending === tier.id;
|
||||
const isDisabled = authLoading || (pending !== null && !isPending);
|
||||
return (
|
||||
<article
|
||||
key={tier.id}
|
||||
className={`surface diagonal-cut${tier.highlight ? ' diagonal-cut-strong' : ''} animate-fade-up stagger-${i + 1}`}
|
||||
style={{
|
||||
padding: 32,
|
||||
position: 'relative',
|
||||
border: tier.highlight ? '1px solid var(--grade-a)' : '1px solid var(--border)',
|
||||
background: tier.highlight ? 'var(--bg-elevated)' : 'var(--bg-surface)',
|
||||
boxShadow: tier.highlight ? '0 16px 48px var(--accent-glow)' : 'none',
|
||||
}}
|
||||
>
|
||||
{tier.badge && (
|
||||
<div
|
||||
className="mono"
|
||||
style={{
|
||||
position: 'absolute',
|
||||
top: -12,
|
||||
left: 24,
|
||||
padding: '4px 12px',
|
||||
background: 'var(--grade-a)',
|
||||
color: 'var(--bg-primary)',
|
||||
fontSize: 10,
|
||||
fontWeight: 800,
|
||||
letterSpacing: '0.08em',
|
||||
borderRadius: 999,
|
||||
textTransform: 'uppercase',
|
||||
}}
|
||||
>
|
||||
{tier.badge}
|
||||
</div>
|
||||
)}
|
||||
<h3 style={{ fontSize: 14, fontWeight: 600, color: 'var(--text-secondary)', marginBottom: 4, textTransform: 'uppercase', letterSpacing: '0.08em' }}>
|
||||
{tier.name}
|
||||
</h3>
|
||||
<div style={{ display: 'flex', alignItems: 'baseline', gap: 8, marginBottom: 4 }}>
|
||||
<span className="mono" style={{ fontSize: 40, fontWeight: 800, color: 'var(--text-primary)', letterSpacing: '-0.03em' }}>
|
||||
{tier.price}
|
||||
</span>
|
||||
<span style={{ color: 'var(--text-tertiary)', fontSize: 14 }}>{tier.cadence}</span>
|
||||
{tier.originalPrice && (
|
||||
<span className="mono" style={{ fontSize: 13, color: 'var(--text-tertiary)', textDecoration: 'line-through' }}>
|
||||
{tier.originalPrice}
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
<p style={{ fontSize: 14, color: 'var(--text-secondary)', marginBottom: 24, minHeight: 42 }}>
|
||||
{tier.headline}
|
||||
</p>
|
||||
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => startCheckout(tier.id)}
|
||||
disabled={isDisabled}
|
||||
className={tier.highlight ? 'btn-primary' : 'btn-ghost'}
|
||||
style={{
|
||||
position: 'absolute',
|
||||
top: -12,
|
||||
left: 24,
|
||||
padding: '4px 12px',
|
||||
background: 'var(--grade-a)',
|
||||
color: 'var(--bg-primary)',
|
||||
fontSize: 10,
|
||||
fontWeight: 800,
|
||||
letterSpacing: '0.08em',
|
||||
borderRadius: 999,
|
||||
textTransform: 'uppercase',
|
||||
width: '100%',
|
||||
padding: 14,
|
||||
marginBottom: 24,
|
||||
cursor: isDisabled ? 'not-allowed' : 'pointer',
|
||||
opacity: isDisabled ? 0.6 : 1,
|
||||
}}
|
||||
>
|
||||
{tier.badge}
|
||||
</div>
|
||||
)}
|
||||
<h3 style={{ fontSize: 14, fontWeight: 600, color: 'var(--text-secondary)', marginBottom: 4, textTransform: 'uppercase', letterSpacing: '0.08em' }}>
|
||||
{tier.name}
|
||||
</h3>
|
||||
<div style={{ display: 'flex', alignItems: 'baseline', gap: 8, marginBottom: 4 }}>
|
||||
<span className="mono" style={{ fontSize: 40, fontWeight: 800, color: 'var(--text-primary)', letterSpacing: '-0.03em' }}>
|
||||
{tier.price}
|
||||
</span>
|
||||
<span style={{ color: 'var(--text-tertiary)', fontSize: 14 }}>{tier.cadence}</span>
|
||||
{tier.originalPrice && (
|
||||
<span className="mono" style={{ fontSize: 13, color: 'var(--text-tertiary)', textDecoration: 'line-through' }}>
|
||||
{tier.originalPrice}
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
<p style={{ fontSize: 14, color: 'var(--text-secondary)', marginBottom: 24, minHeight: 42 }}>
|
||||
{tier.headline}
|
||||
</p>
|
||||
{isPending ? 'Redirecting to Stripe…' : tier.cta}
|
||||
</button>
|
||||
|
||||
<a
|
||||
href={tier.ctaHref}
|
||||
className={tier.highlight ? 'btn-primary' : 'btn-ghost'}
|
||||
style={{ width: '100%', padding: 14, marginBottom: 24 }}
|
||||
>
|
||||
{tier.cta}
|
||||
</a>
|
||||
|
||||
<ul style={{ display: 'grid', gap: 10 }}>
|
||||
{tier.features.map((f) => (
|
||||
<li key={f} style={{ display: 'flex', gap: 10, fontSize: 14 }}>
|
||||
<span style={{ color: 'var(--grade-a)', fontWeight: 700 }} aria-hidden>+</span>
|
||||
<span style={{ color: 'var(--text-primary)' }}>{f}</span>
|
||||
</li>
|
||||
))}
|
||||
{tier.locked.map((f) => (
|
||||
<li key={f} style={{ display: 'flex', gap: 10, fontSize: 14, color: 'var(--text-tertiary)' }}>
|
||||
<span aria-hidden>—</span>
|
||||
<span>{f}</span>
|
||||
</li>
|
||||
))}
|
||||
</ul>
|
||||
</article>
|
||||
))}
|
||||
<ul style={{ display: 'grid', gap: 10 }}>
|
||||
{tier.features.map((f) => (
|
||||
<li key={f} style={{ display: 'flex', gap: 10, fontSize: 14 }}>
|
||||
<span style={{ color: 'var(--grade-a)', fontWeight: 700 }} aria-hidden>+</span>
|
||||
<span style={{ color: 'var(--text-primary)' }}>{f}</span>
|
||||
</li>
|
||||
))}
|
||||
{tier.locked.map((f) => (
|
||||
<li key={f} style={{ display: 'flex', gap: 10, fontSize: 14, color: 'var(--text-tertiary)' }}>
|
||||
<span aria-hidden>—</span>
|
||||
<span>{f}</span>
|
||||
</li>
|
||||
))}
|
||||
</ul>
|
||||
</article>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
|
||||
<p style={{ textAlign: 'center', fontSize: 13, color: 'var(--text-tertiary)', marginTop: 32 }}>
|
||||
Cancel anytime. No contracts. Card or Apple Pay or Google Pay — payments processed by NexaPay.
|
||||
Cancel anytime. No contracts. Card / Apple Pay / Google Pay — payments processed by Stripe (test mode while we onboard founders).
|
||||
</p>
|
||||
</div>
|
||||
|
||||
|
||||
@@ -0,0 +1,350 @@
|
||||
'use client';
|
||||
|
||||
import { useMemo } from 'react';
|
||||
|
||||
/**
|
||||
* Soccer result card — renders an analyze/prop response with
|
||||
* soccer-specific visual treatment. We can't surface raw feature
|
||||
* values (the backend response carries only `reasoning.summary` +
|
||||
* `kill_conditions_triggered` per the engine1 → legacy adapter), so
|
||||
* we parse the summary for known soccer-signal phrases and surface
|
||||
* each as a colored chip above the prose.
|
||||
*
|
||||
* Free-tier responses already arrive gated (the Session 7h
|
||||
* `applyTierGating` redacts `reasoning` and `kill_conditions`); we
|
||||
* just need to detect the `tier_gated` / `locked` markers and show
|
||||
* an upgrade CTA over the blurred content.
|
||||
*/
|
||||
|
||||
interface KillCondition {
|
||||
code: string;
|
||||
reason: string;
|
||||
locked?: boolean;
|
||||
}
|
||||
|
||||
interface Reasoning {
|
||||
summary?: string;
|
||||
steps?: unknown;
|
||||
locked?: boolean;
|
||||
}
|
||||
|
||||
export interface SoccerGradeResultProps {
|
||||
player: string;
|
||||
stat_type: string;
|
||||
line: number;
|
||||
direction: 'over' | 'under';
|
||||
league: string;
|
||||
grade: string;
|
||||
confidence?: number;
|
||||
edge_pct?: number;
|
||||
reasoning?: Reasoning;
|
||||
kill_conditions_triggered?: KillCondition[];
|
||||
tier_gated?: boolean;
|
||||
upgrade_hint?: string;
|
||||
onUpgradeClick?: () => void;
|
||||
onClose?: () => void;
|
||||
}
|
||||
|
||||
type SignalTone = 'positive' | 'caution' | 'warning' | 'neutral';
|
||||
|
||||
interface ParsedSignal {
|
||||
icon: string;
|
||||
label: string;
|
||||
detail: string;
|
||||
tone: SignalTone;
|
||||
}
|
||||
|
||||
const SIGNAL_TONE_STYLE: Record<SignalTone, { color: string; bg: string; border: string }> = {
|
||||
positive: { color: 'var(--grade-a)', bg: 'rgba(0,200,150,0.08)', border: 'rgba(0,200,150,0.40)' },
|
||||
caution: { color: 'var(--grade-c, #FFB347)', bg: 'rgba(255,179,71,0.08)', border: 'rgba(255,179,71,0.40)' },
|
||||
warning: { color: 'var(--grade-d, #ff5a5a)', bg: 'rgba(255,90,90,0.08)', border: 'rgba(255,90,90,0.40)' },
|
||||
neutral: { color: 'var(--text-secondary)', bg: 'transparent', border: 'var(--border)' },
|
||||
};
|
||||
|
||||
// Pattern-match the concrete sentences `buildSoccerReasoningLines`
|
||||
// emits in src/services/intelligence/analyzeViaEngine1.js. Order
|
||||
// matters — earlier patterns win when multiple match the same line.
|
||||
const SIGNAL_PATTERNS: Array<(line: string) => ParsedSignal | null> = [
|
||||
(line) => {
|
||||
const m = line.match(/scores ([\d.]+) goals per 90 minutes/i);
|
||||
if (m) return { icon: '⚽', label: 'Goals / 90', detail: `${m[1]}`, tone: 'positive' };
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
const m = line.match(/Expected goals \(xG\): ([\d.]+) per 90 — (.+)/i);
|
||||
if (m) {
|
||||
const trend = m[2].toLowerCase();
|
||||
const tone: SignalTone = trend.includes('regression') ? 'caution'
|
||||
: trend.includes('breakout') ? 'positive' : 'neutral';
|
||||
return { icon: '📊', label: 'xG / 90', detail: `${m[1]} — ${m[2]}`, tone };
|
||||
}
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
if (/Designated penalty taker/i.test(line)) {
|
||||
return { icon: '🎯', label: 'Penalty Taker', detail: '+0.15 goals/90 boost', tone: 'positive' };
|
||||
}
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
if (/Direct free-kick specialist/i.test(line)) {
|
||||
return { icon: '🏹', label: 'Free-Kick Taker', detail: 'shot/goal probability boost', tone: 'positive' };
|
||||
}
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
if (/corner taker/i.test(line)) {
|
||||
return { icon: '⛳', label: 'Corner Taker', detail: 'assist probability boost', tone: 'positive' };
|
||||
}
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
const m = line.match(/Match at ([\d,]+)ft altitude\.\s*(.+)/i);
|
||||
if (m) {
|
||||
const isAcclimated = /acclimated host/i.test(m[2]);
|
||||
return {
|
||||
icon: '🏔️',
|
||||
label: 'Altitude',
|
||||
detail: `${m[1]}ft — ${isAcclimated ? 'host acclimated' : 'visitor risk'}`,
|
||||
tone: isAcclimated ? 'neutral' : 'warning',
|
||||
};
|
||||
}
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
const m = line.match(/(.+?) averages ([\d.]+) cards per match/i);
|
||||
if (m) {
|
||||
const cardsPerGame = parseFloat(m[2]);
|
||||
const tone: SignalTone = cardsPerGame >= 5 ? 'caution' : 'neutral';
|
||||
return { icon: '🟨', label: `Referee: ${m[1].trim()}`, detail: `${m[2]} cards/match`, tone };
|
||||
}
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
const m = line.match(/Averaging only ([\d.]+) minutes per match/i);
|
||||
if (m) return { icon: '⏱️', label: 'Minutes', detail: `${m[1]}/90 — under-line discount`, tone: 'caution' };
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
const m = line.match(/(.+?) concedes ([\d.]+) goals per game/i);
|
||||
if (m) {
|
||||
const conceded = parseFloat(m[2]);
|
||||
const tone: SignalTone = conceded <= 0.8 ? 'warning' : conceded >= 1.6 ? 'positive' : 'neutral';
|
||||
return { icon: '🛡️', label: `Defense: ${m[1].trim()}`, detail: `${m[2]} GA/match`, tone };
|
||||
}
|
||||
return null;
|
||||
},
|
||||
(line) => {
|
||||
const m = line.match(/Tournament pedigree: (\d+) career World Cup goals/i);
|
||||
if (m) return { icon: '🏆', label: 'WC Pedigree', detail: `${m[1]} career goals`, tone: 'positive' };
|
||||
return null;
|
||||
},
|
||||
];
|
||||
|
||||
function parseSignals(summary: string | undefined): ParsedSignal[] {
|
||||
if (!summary) return [];
|
||||
const out: ParsedSignal[] = [];
|
||||
// The buildSoccerReasoningLines output is a single `lines.join(' ')`,
|
||||
// so split on period+space and trim. Some sentences contain periods
|
||||
// (e.g. "0.67 goals per 90"), so re-match conservatively.
|
||||
const fragments = summary.split(/(?<=\.)\s+(?=[A-Z⚽📊🎯🏹⛳🏔️🟨⏱️🛡️🏆])/);
|
||||
for (const frag of fragments) {
|
||||
for (const fn of SIGNAL_PATTERNS) {
|
||||
const sig = fn(frag);
|
||||
if (sig) {
|
||||
out.push(sig);
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return out;
|
||||
}
|
||||
|
||||
function gradeColor(grade: string): string {
|
||||
const g = (grade || '').trim().toUpperCase().charAt(0);
|
||||
if (g === 'A') return 'var(--grade-a)';
|
||||
if (g === 'B') return 'var(--grade-b, #4A9EFF)';
|
||||
if (g === 'C') return 'var(--grade-c, #FFB347)';
|
||||
return 'var(--grade-d, #ff5a5a)';
|
||||
}
|
||||
|
||||
export default function SoccerGradeResult(props: SoccerGradeResultProps) {
|
||||
const {
|
||||
player, stat_type, line, direction, league, grade, confidence, edge_pct,
|
||||
reasoning, kill_conditions_triggered, tier_gated, upgrade_hint,
|
||||
onUpgradeClick, onClose,
|
||||
} = props;
|
||||
|
||||
const signals = useMemo(() => parseSignals(reasoning?.summary), [reasoning?.summary]);
|
||||
const color = gradeColor(grade);
|
||||
const locked = !!tier_gated || !!reasoning?.locked;
|
||||
const kills = Array.isArray(kill_conditions_triggered) ? kill_conditions_triggered : [];
|
||||
|
||||
return (
|
||||
<article
|
||||
className="surface diagonal-cut"
|
||||
style={{
|
||||
padding: 24,
|
||||
border: `1px solid ${color}`,
|
||||
background: 'var(--bg-elevated)',
|
||||
borderRadius: 8,
|
||||
marginTop: 16,
|
||||
position: 'relative',
|
||||
}}
|
||||
data-testid="soccer-grade-result"
|
||||
>
|
||||
{onClose && (
|
||||
<button
|
||||
type="button"
|
||||
onClick={onClose}
|
||||
aria-label="Close"
|
||||
style={{
|
||||
position: 'absolute', top: 8, right: 12,
|
||||
background: 'transparent', border: 0, cursor: 'pointer',
|
||||
fontSize: 18, color: 'var(--text-tertiary)',
|
||||
}}
|
||||
>
|
||||
×
|
||||
</button>
|
||||
)}
|
||||
|
||||
<header style={{ display: 'flex', justifyContent: 'space-between', alignItems: 'flex-start', gap: 16, marginBottom: 16 }}>
|
||||
<div>
|
||||
<div style={{ fontSize: 18, fontWeight: 700 }}>{player}</div>
|
||||
<div
|
||||
className="mono"
|
||||
style={{ fontSize: 11, color: 'var(--text-tertiary)', marginTop: 4, textTransform: 'uppercase', letterSpacing: '0.06em' }}
|
||||
>
|
||||
{direction.toUpperCase()} {line.toFixed(1)} {stat_type.replace(/_/g, ' ')} · {league.toUpperCase()}
|
||||
</div>
|
||||
</div>
|
||||
<div style={{ textAlign: 'right' }}>
|
||||
<div className="mono" style={{ fontSize: 44, fontWeight: 800, color, lineHeight: 1, letterSpacing: '-0.04em' }}>
|
||||
{grade}
|
||||
</div>
|
||||
{typeof confidence === 'number' && (
|
||||
<div className="mono" style={{ fontSize: 11, color: 'var(--text-tertiary)', marginTop: 4 }}>
|
||||
{confidence.toFixed(0)}% conf
|
||||
{typeof edge_pct === 'number' && (
|
||||
<> · {edge_pct >= 0 ? '+' : ''}{edge_pct.toFixed(1)}% edge</>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
</header>
|
||||
|
||||
{signals.length > 0 && !locked && (
|
||||
<div style={{ display: 'flex', flexWrap: 'wrap', gap: 8, marginBottom: 16 }}>
|
||||
{signals.map((sig, idx) => {
|
||||
const style = SIGNAL_TONE_STYLE[sig.tone];
|
||||
return (
|
||||
<div
|
||||
key={`${sig.label}-${idx}`}
|
||||
style={{
|
||||
padding: '8px 12px',
|
||||
border: `1px solid ${style.border}`,
|
||||
background: style.bg,
|
||||
borderRadius: 6,
|
||||
fontSize: 12,
|
||||
display: 'flex',
|
||||
alignItems: 'baseline',
|
||||
gap: 6,
|
||||
maxWidth: '100%',
|
||||
}}
|
||||
>
|
||||
<span aria-hidden style={{ fontSize: 14 }}>{sig.icon}</span>
|
||||
<span
|
||||
className="mono"
|
||||
style={{
|
||||
color: style.color,
|
||||
fontWeight: 600,
|
||||
textTransform: 'uppercase',
|
||||
letterSpacing: '0.04em',
|
||||
fontSize: 10,
|
||||
}}
|
||||
>
|
||||
{sig.label}
|
||||
</span>
|
||||
<span style={{ color: 'var(--text-primary)' }}>{sig.detail}</span>
|
||||
</div>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
)}
|
||||
|
||||
{!locked && reasoning?.summary && (
|
||||
<p style={{ fontSize: 14, color: 'var(--text-secondary)', lineHeight: 1.6, marginBottom: 16 }}>
|
||||
{reasoning.summary}
|
||||
</p>
|
||||
)}
|
||||
|
||||
{locked && (
|
||||
<div
|
||||
style={{
|
||||
padding: 20,
|
||||
border: '1px dashed var(--border)',
|
||||
borderRadius: 6,
|
||||
background: 'rgba(0,0,0,0.20)',
|
||||
textAlign: 'center',
|
||||
marginBottom: 16,
|
||||
}}
|
||||
>
|
||||
<div
|
||||
className="mono"
|
||||
style={{
|
||||
filter: 'blur(4px)',
|
||||
userSelect: 'none',
|
||||
color: 'var(--text-tertiary)',
|
||||
fontSize: 13,
|
||||
marginBottom: 16,
|
||||
}}
|
||||
aria-hidden
|
||||
>
|
||||
⚽ Goals/90: 0.67 · 📊 xG: 0.52 — overperforming · 🏔️ altitude 7,349ft · 🟨 ref 4.7 cards/match · 🎯 penalty taker
|
||||
</div>
|
||||
<p style={{ fontSize: 13, color: 'var(--text-secondary)', marginBottom: 12 }}>
|
||||
{upgrade_hint || 'Unlock full intelligence — xG regression, altitude, referee, set-piece role.'}
|
||||
</p>
|
||||
<button
|
||||
type="button"
|
||||
onClick={onUpgradeClick}
|
||||
className="btn-primary"
|
||||
style={{ padding: '8px 18px', fontSize: 13 }}
|
||||
>
|
||||
Unlock full analysis
|
||||
</button>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{kills.length > 0 && (
|
||||
<section style={{ marginTop: 8 }}>
|
||||
<h3
|
||||
className="mono"
|
||||
style={{ fontSize: 10, color: 'var(--grade-d, #ff5a5a)', textTransform: 'uppercase', letterSpacing: '0.08em', marginBottom: 8 }}
|
||||
>
|
||||
Kill conditions ({kills.length})
|
||||
</h3>
|
||||
<ul style={{ display: 'grid', gap: 6 }}>
|
||||
{kills.map((k, idx) => (
|
||||
<li
|
||||
key={`${k.code}-${idx}`}
|
||||
style={{
|
||||
padding: '6px 10px',
|
||||
border: '1px solid rgba(255,90,90,0.30)',
|
||||
background: 'rgba(255,90,90,0.04)',
|
||||
borderRadius: 4,
|
||||
fontSize: 12,
|
||||
}}
|
||||
>
|
||||
<span className="mono" style={{ color: 'var(--grade-d, #ff5a5a)', fontWeight: 700, marginRight: 6 }}>
|
||||
{k.code}
|
||||
</span>
|
||||
<span style={{ color: 'var(--text-secondary)' }}>{k.reason}</span>
|
||||
</li>
|
||||
))}
|
||||
</ul>
|
||||
</section>
|
||||
)}
|
||||
</article>
|
||||
);
|
||||
}
|
||||
@@ -0,0 +1,190 @@
|
||||
'use client';
|
||||
|
||||
import { useState, useEffect } from 'react';
|
||||
|
||||
/**
|
||||
* SportSelector — pill tabs for the four launch verticals.
|
||||
*
|
||||
* Soccer reveals a secondary league pill row (WC default for the
|
||||
* tournament launch; EPL/La Liga/etc available year-round). The
|
||||
* selected `{ sport, league }` is emitted via `onChange` so the
|
||||
* parent owns the actual scan/odds state and can refetch on switch.
|
||||
*
|
||||
* The component is intentionally pure-UI — no fetches, no auth, no
|
||||
* persistence. A parent that wants the selection to stick should
|
||||
* pass `initialSport` / `initialLeague` from URL params or
|
||||
* localStorage.
|
||||
*/
|
||||
|
||||
export type Sport = 'NBA' | 'WNBA' | 'MLB' | 'Soccer';
|
||||
|
||||
// Soccer league codes match the GET /api/odds/soccer/:league path
|
||||
// segment AND the `SOCCER_LEAGUES` env on the backend. Source of truth
|
||||
// is `src/services/oddsService.js SOCCER_SPORT_KEYS`.
|
||||
export type SoccerLeague =
|
||||
| 'wc'
|
||||
| 'epl'
|
||||
| 'laliga'
|
||||
| 'bundesliga'
|
||||
| 'seriea'
|
||||
| 'ligue1'
|
||||
| 'ucl'
|
||||
| 'mls'
|
||||
| 'ligamx';
|
||||
|
||||
export interface SportSelection {
|
||||
sport: Sport;
|
||||
league?: SoccerLeague;
|
||||
}
|
||||
|
||||
const SPORTS: Array<{ id: Sport; label: string; status?: 'live' | 'beta' }> = [
|
||||
{ id: 'NBA', label: 'NBA', status: 'live' },
|
||||
{ id: 'WNBA', label: 'WNBA', status: 'live' },
|
||||
{ id: 'MLB', label: 'MLB', status: 'live' },
|
||||
{ id: 'Soccer', label: 'Soccer', status: 'beta' },
|
||||
];
|
||||
|
||||
const SOCCER_LEAGUES: Array<{ id: SoccerLeague; label: string; sub?: string }> = [
|
||||
{ id: 'wc', label: 'World Cup', sub: '2026' },
|
||||
{ id: 'epl', label: 'EPL' },
|
||||
{ id: 'laliga', label: 'La Liga' },
|
||||
{ id: 'bundesliga', label: 'Bundesliga' },
|
||||
{ id: 'seriea', label: 'Serie A' },
|
||||
{ id: 'ligue1', label: 'Ligue 1' },
|
||||
{ id: 'ucl', label: 'UCL' },
|
||||
{ id: 'mls', label: 'MLS' },
|
||||
{ id: 'ligamx', label: 'Liga MX' },
|
||||
];
|
||||
|
||||
interface Props {
|
||||
initialSport?: Sport;
|
||||
initialLeague?: SoccerLeague;
|
||||
onChange?: (selection: SportSelection) => void;
|
||||
}
|
||||
|
||||
export default function SportSelector({
|
||||
initialSport = 'NBA',
|
||||
initialLeague = 'wc',
|
||||
onChange,
|
||||
}: Props) {
|
||||
const [sport, setSport] = useState<Sport>(initialSport);
|
||||
const [league, setLeague] = useState<SoccerLeague>(initialLeague);
|
||||
|
||||
// Emit on every change so parents stay in sync. Effect (not inline
|
||||
// in setSport) so React batches both pieces of state correctly.
|
||||
useEffect(() => {
|
||||
if (onChange) {
|
||||
onChange(sport === 'Soccer' ? { sport, league } : { sport });
|
||||
}
|
||||
}, [sport, league, onChange]);
|
||||
|
||||
function selectSport(next: Sport) {
|
||||
setSport(next);
|
||||
}
|
||||
|
||||
return (
|
||||
<div data-testid="sport-selector" style={{ display: 'flex', flexDirection: 'column', gap: 12 }}>
|
||||
<div
|
||||
role="tablist"
|
||||
aria-label="Sport"
|
||||
style={{
|
||||
display: 'flex',
|
||||
flexWrap: 'wrap',
|
||||
gap: 8,
|
||||
}}
|
||||
>
|
||||
{SPORTS.map((s) => {
|
||||
const active = sport === s.id;
|
||||
return (
|
||||
<button
|
||||
key={s.id}
|
||||
type="button"
|
||||
role="tab"
|
||||
aria-selected={active}
|
||||
onClick={() => selectSport(s.id)}
|
||||
className="mono"
|
||||
style={{
|
||||
padding: '8px 16px',
|
||||
fontSize: 13,
|
||||
fontWeight: 700,
|
||||
letterSpacing: '0.04em',
|
||||
textTransform: 'uppercase',
|
||||
border: active ? '1px solid var(--grade-a)' : '1px solid var(--border)',
|
||||
background: active ? 'var(--grade-a)' : 'transparent',
|
||||
color: active ? 'var(--bg-primary)' : 'var(--text-primary)',
|
||||
cursor: 'pointer',
|
||||
borderRadius: 6,
|
||||
position: 'relative',
|
||||
transition: 'all 0.15s ease',
|
||||
}}
|
||||
>
|
||||
{s.label}
|
||||
{s.status === 'beta' && (
|
||||
<span
|
||||
style={{
|
||||
marginLeft: 6,
|
||||
fontSize: 9,
|
||||
padding: '1px 4px',
|
||||
background: active ? 'var(--bg-primary)' : 'var(--grade-a)',
|
||||
color: active ? 'var(--grade-a)' : 'var(--bg-primary)',
|
||||
borderRadius: 3,
|
||||
verticalAlign: 'middle',
|
||||
}}
|
||||
>
|
||||
BETA
|
||||
</span>
|
||||
)}
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
|
||||
{sport === 'Soccer' && (
|
||||
<div
|
||||
role="tablist"
|
||||
aria-label="Soccer league"
|
||||
style={{
|
||||
display: 'flex',
|
||||
flexWrap: 'wrap',
|
||||
gap: 6,
|
||||
padding: '8px 0 4px',
|
||||
borderTop: '1px solid var(--border)',
|
||||
}}
|
||||
>
|
||||
{SOCCER_LEAGUES.map((l) => {
|
||||
const active = league === l.id;
|
||||
return (
|
||||
<button
|
||||
key={l.id}
|
||||
type="button"
|
||||
role="tab"
|
||||
aria-selected={active}
|
||||
onClick={() => setLeague(l.id)}
|
||||
style={{
|
||||
padding: '6px 10px',
|
||||
fontSize: 12,
|
||||
fontWeight: 600,
|
||||
border: active ? '1px solid var(--grade-a)' : '1px solid var(--border)',
|
||||
background: active ? 'var(--bg-elevated)' : 'transparent',
|
||||
color: active ? 'var(--grade-a)' : 'var(--text-secondary)',
|
||||
cursor: 'pointer',
|
||||
borderRadius: 4,
|
||||
display: 'inline-flex',
|
||||
alignItems: 'baseline',
|
||||
gap: 4,
|
||||
}}
|
||||
>
|
||||
<span>{l.label}</span>
|
||||
{l.sub && (
|
||||
<span className="mono" style={{ fontSize: 10, opacity: 0.6 }}>
|
||||
{l.sub}
|
||||
</span>
|
||||
)}
|
||||
</button>
|
||||
);
|
||||
})}
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
Reference in New Issue
Block a user