init
This commit is contained in:
+957
@@ -0,0 +1,957 @@
|
||||
import { useEffect, useMemo, useRef, useState } from 'react'
|
||||
import Tree from '../Tree/Tree'
|
||||
import FallingLeaves from '../Tree/FallingLeaves'
|
||||
|
||||
const numberFormat = new Intl.NumberFormat('en-GB')
|
||||
const dateFormat = new Intl.DateTimeFormat('en-GB', {
|
||||
dateStyle: 'medium',
|
||||
timeStyle: 'short',
|
||||
})
|
||||
|
||||
const apiEndpoints = {
|
||||
teams: '/api/tss/leaderboard/teams?limit=100',
|
||||
resolve: (name) => `/api/tss/teams/resolve?name=${encodeURIComponent(name)}`,
|
||||
detail: (name) => `/api/tss/teams/${encodeURIComponent(name)}`,
|
||||
history: (name) => `/api/tss/teams/${encodeURIComponent(name)}/history`,
|
||||
games: (name) => `/api/tss/teams/${encodeURIComponent(name)}/games`,
|
||||
}
|
||||
|
||||
const navItems = [
|
||||
{ path: '/', label: 'Home' },
|
||||
{ path: '/teams', label: 'Team leaderboard' },
|
||||
{ path: '/battle-logs', label: 'Battle Logs' },
|
||||
]
|
||||
|
||||
async function fetchJson(path, signal) {
|
||||
const response = await fetch(path, {
|
||||
signal,
|
||||
headers: { Accept: 'application/json' },
|
||||
})
|
||||
const body = await response.json().catch(() => null)
|
||||
|
||||
if (!response.ok) {
|
||||
throw new Error(body?.error || `Request failed with ${response.status}`)
|
||||
}
|
||||
|
||||
return body
|
||||
}
|
||||
|
||||
function parseRoute(pathname = window.location.pathname) {
|
||||
if (pathname === '/') return { page: 'home', teamName: '' }
|
||||
if (pathname === '/teams') return { page: 'teams', teamName: '' }
|
||||
if (pathname.startsWith('/teams/')) {
|
||||
const teamName = decodeURIComponent(pathname.slice('/teams/'.length))
|
||||
return { page: 'team', teamName }
|
||||
}
|
||||
if (pathname === '/battle-logs' || pathname === '/live') return { page: 'battle-logs', teamName: '' }
|
||||
return { page: 'home', teamName: '' }
|
||||
}
|
||||
|
||||
function teamPath(name) {
|
||||
return `/teams/${encodeURIComponent(name)}`
|
||||
}
|
||||
|
||||
function formatNumber(value) {
|
||||
return numberFormat.format(Number(value || 0))
|
||||
}
|
||||
|
||||
function formatDate(timestamp) {
|
||||
if (!timestamp) return 'Unknown time'
|
||||
return dateFormat.format(new Date(Number(timestamp) * 1000))
|
||||
}
|
||||
|
||||
function bestTeamName(team) {
|
||||
return team?.tag_name || team?.short_name || team?.long_name || ''
|
||||
}
|
||||
|
||||
async function fetchRecentTssGames(teams, signal) {
|
||||
const teamNames = teams.map(bestTeamName).filter(Boolean).slice(0, 12)
|
||||
|
||||
if (!teamNames.length) {
|
||||
return { matches: [] }
|
||||
}
|
||||
|
||||
const responses = await Promise.allSettled(
|
||||
teamNames.map((name) => fetchJson(apiEndpoints.games(name), signal).then((data) => ({ name, data }))),
|
||||
)
|
||||
const bySession = new Map()
|
||||
|
||||
responses.forEach((result) => {
|
||||
if (result.status !== 'fulfilled') return
|
||||
|
||||
const { name, data } = result.value
|
||||
;(data.games || []).forEach((game) => {
|
||||
if (!game.session_id) return
|
||||
|
||||
const existing = bySession.get(game.session_id)
|
||||
const currentTimestamp = Number(game.timestamp || 0)
|
||||
if (existing && Number(existing.timestamp || 0) >= currentTimestamp) return
|
||||
|
||||
bySession.set(game.session_id, {
|
||||
...game,
|
||||
team_name: data.tag_name || name,
|
||||
long_name: data.long_name || '',
|
||||
})
|
||||
})
|
||||
})
|
||||
|
||||
return {
|
||||
matches: Array.from(bySession.values())
|
||||
.sort((a, b) => Number(b.timestamp || 0) - Number(a.timestamp || 0))
|
||||
.slice(0, 50),
|
||||
}
|
||||
}
|
||||
|
||||
function Stat({ label, value }) {
|
||||
return (
|
||||
<div className="border-l border-border pl-4">
|
||||
<p className="text-xs font-semibold uppercase tracking-wide text-text-soft">
|
||||
{label}
|
||||
</p>
|
||||
<p className="mt-1 text-2xl font-semibold text-text">{value}</p>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
function App() {
|
||||
const [route, setRoute] = useState(() => parseRoute())
|
||||
const [leaderboard, setLeaderboard] = useState({ status: 'idle', data: null, error: null })
|
||||
const [live, setLive] = useState({ status: 'idle', data: null, error: null })
|
||||
const [teamQuery, setTeamQuery] = useState('')
|
||||
const [searchHint, setSearchHint] = useState({ status: 'idle', name: '' })
|
||||
const [profile, setProfile] = useState({
|
||||
teamName: '',
|
||||
detail: { status: 'idle', data: null, error: null },
|
||||
history: { status: 'idle', data: null, error: null },
|
||||
games: { status: 'idle', data: null, error: null },
|
||||
})
|
||||
|
||||
function navigate(path) {
|
||||
window.history.pushState({}, '', path)
|
||||
setRoute(parseRoute(path))
|
||||
}
|
||||
|
||||
useEffect(() => {
|
||||
const onPopState = () => setRoute(parseRoute())
|
||||
window.addEventListener('popstate', onPopState)
|
||||
return () => window.removeEventListener('popstate', onPopState)
|
||||
}, [])
|
||||
|
||||
useEffect(() => {
|
||||
const title =
|
||||
route.page === 'team' && route.teamName
|
||||
? `${route.teamName} | Toothless' TSS Bot`
|
||||
: route.page === 'teams'
|
||||
? "Team leaderboard | Toothless' TSS Bot"
|
||||
: route.page === 'battle-logs'
|
||||
? "Battle Logs | Toothless' TSS Bot"
|
||||
: "Toothless' TSS Bot"
|
||||
|
||||
document.title = title
|
||||
}, [route.page, route.teamName])
|
||||
|
||||
useEffect(() => {
|
||||
const onKeyDown = (event) => {
|
||||
if (event.key === 'Escape') {
|
||||
navigate('/')
|
||||
}
|
||||
}
|
||||
|
||||
window.addEventListener('keydown', onKeyDown)
|
||||
return () => window.removeEventListener('keydown', onKeyDown)
|
||||
}, [])
|
||||
|
||||
useEffect(() => {
|
||||
const query = teamQuery.trim()
|
||||
if (query.length < 2) {
|
||||
setSearchHint({ status: 'idle', name: '' })
|
||||
return
|
||||
}
|
||||
|
||||
const controller = new AbortController()
|
||||
const timer = window.setTimeout(() => {
|
||||
setSearchHint({ status: 'loading', name: '' })
|
||||
fetchJson(apiEndpoints.resolve(query), controller.signal)
|
||||
.then((data) => {
|
||||
const name = data.tag_name || data.short_name || data.long_name || query
|
||||
setSearchHint({ status: 'ready', name })
|
||||
})
|
||||
.catch(() => {
|
||||
if (!controller.signal.aborted) {
|
||||
setSearchHint({ status: 'error', name: '' })
|
||||
}
|
||||
})
|
||||
}, 350)
|
||||
|
||||
return () => {
|
||||
window.clearTimeout(timer)
|
||||
controller.abort()
|
||||
}
|
||||
}, [teamQuery])
|
||||
|
||||
useEffect(() => {
|
||||
if (!['home', 'teams', 'team', 'battle-logs'].includes(route.page)) return
|
||||
if (leaderboard.status === 'ready' || leaderboard.status === 'loading') return
|
||||
|
||||
const controller = new AbortController()
|
||||
setLeaderboard({ status: 'loading', data: null, error: null })
|
||||
|
||||
fetchJson(apiEndpoints.teams, controller.signal)
|
||||
.then((data) => setLeaderboard({ status: 'ready', data, error: null }))
|
||||
.catch((error) => {
|
||||
if (!controller.signal.aborted) {
|
||||
setLeaderboard({ status: 'error', data: null, error: error.message })
|
||||
}
|
||||
})
|
||||
|
||||
return () => controller.abort()
|
||||
}, [leaderboard.status, route.page])
|
||||
|
||||
useEffect(() => {
|
||||
if (!['home', 'teams', 'team', 'battle-logs'].includes(route.page)) return
|
||||
|
||||
const controller = new AbortController()
|
||||
const timer = window.setInterval(() => {
|
||||
fetchJson(apiEndpoints.teams, controller.signal)
|
||||
.then((data) => setLeaderboard({ status: 'ready', data, error: null }))
|
||||
.catch((error) => {
|
||||
if (!controller.signal.aborted) {
|
||||
setLeaderboard((current) => ({ ...current, error: error.message }))
|
||||
}
|
||||
})
|
||||
}, 60000)
|
||||
|
||||
return () => {
|
||||
window.clearInterval(timer)
|
||||
controller.abort()
|
||||
}
|
||||
}, [route.page])
|
||||
|
||||
useEffect(() => {
|
||||
if (!['home', 'battle-logs'].includes(route.page)) return
|
||||
if (!teams.length) return
|
||||
|
||||
const controller = new AbortController()
|
||||
setLive((current) =>
|
||||
current.status === 'ready' ? current : { status: 'loading', data: null, error: null },
|
||||
)
|
||||
|
||||
fetchRecentTssGames(teams, controller.signal)
|
||||
.then((data) => setLive({ status: 'ready', data, error: null }))
|
||||
.catch((error) => {
|
||||
if (!controller.signal.aborted) {
|
||||
setLive({ status: 'error', data: null, error: error.message })
|
||||
}
|
||||
})
|
||||
|
||||
return () => controller.abort()
|
||||
}, [route.page, teams])
|
||||
|
||||
useEffect(() => {
|
||||
if (!['home', 'battle-logs'].includes(route.page)) return
|
||||
if (!teams.length) return
|
||||
|
||||
const controller = new AbortController()
|
||||
const timer = window.setInterval(() => {
|
||||
fetchRecentTssGames(teams, controller.signal)
|
||||
.then((data) => setLive({ status: 'ready', data, error: null }))
|
||||
.catch((error) => {
|
||||
if (!controller.signal.aborted) {
|
||||
setLive((current) => ({ ...current, error: error.message }))
|
||||
}
|
||||
})
|
||||
}, 15000)
|
||||
|
||||
return () => {
|
||||
window.clearInterval(timer)
|
||||
controller.abort()
|
||||
}
|
||||
}, [route.page, teams])
|
||||
|
||||
useEffect(() => {
|
||||
if (route.page !== 'team' || !route.teamName) return
|
||||
|
||||
const controller = new AbortController()
|
||||
setProfile({
|
||||
teamName: route.teamName,
|
||||
detail: { status: 'loading', data: null, error: null },
|
||||
history: { status: 'loading', data: null, error: null },
|
||||
games: { status: 'loading', data: null, error: null },
|
||||
})
|
||||
|
||||
Promise.allSettled([
|
||||
fetchJson(apiEndpoints.detail(route.teamName), controller.signal),
|
||||
fetchJson(apiEndpoints.history(route.teamName), controller.signal),
|
||||
fetchJson(apiEndpoints.games(route.teamName), controller.signal),
|
||||
]).then(([detailResult, historyResult, gamesResult]) => {
|
||||
if (controller.signal.aborted) return
|
||||
|
||||
setProfile({
|
||||
teamName: route.teamName,
|
||||
detail:
|
||||
detailResult.status === 'fulfilled'
|
||||
? { status: 'ready', data: detailResult.value, error: null }
|
||||
: { status: 'error', data: null, error: detailResult.reason.message },
|
||||
history:
|
||||
historyResult.status === 'fulfilled'
|
||||
? { status: 'ready', data: historyResult.value, error: null }
|
||||
: { status: 'error', data: null, error: historyResult.reason.message },
|
||||
games:
|
||||
gamesResult.status === 'fulfilled'
|
||||
? { status: 'ready', data: gamesResult.value, error: null }
|
||||
: { status: 'error', data: null, error: gamesResult.reason.message },
|
||||
})
|
||||
})
|
||||
|
||||
return () => controller.abort()
|
||||
}, [route.page, route.teamName])
|
||||
|
||||
useEffect(() => {
|
||||
if (route.page !== 'team' || !route.teamName) return
|
||||
|
||||
const controller = new AbortController()
|
||||
const timer = window.setInterval(() => {
|
||||
Promise.allSettled([
|
||||
fetchJson(apiEndpoints.detail(route.teamName), controller.signal),
|
||||
fetchJson(apiEndpoints.history(route.teamName), controller.signal),
|
||||
fetchJson(apiEndpoints.games(route.teamName), controller.signal),
|
||||
]).then(([detailResult, historyResult, gamesResult]) => {
|
||||
if (controller.signal.aborted) return
|
||||
|
||||
setProfile((current) => ({
|
||||
teamName: route.teamName,
|
||||
detail:
|
||||
detailResult.status === 'fulfilled'
|
||||
? { status: 'ready', data: detailResult.value, error: null }
|
||||
: current.detail,
|
||||
history:
|
||||
historyResult.status === 'fulfilled'
|
||||
? { status: 'ready', data: historyResult.value, error: null }
|
||||
: current.history,
|
||||
games:
|
||||
gamesResult.status === 'fulfilled'
|
||||
? { status: 'ready', data: gamesResult.value, error: null }
|
||||
: current.games,
|
||||
}))
|
||||
})
|
||||
}, 60000)
|
||||
|
||||
return () => {
|
||||
window.clearInterval(timer)
|
||||
controller.abort()
|
||||
}
|
||||
}, [route.page, route.teamName])
|
||||
|
||||
const teams = useMemo(
|
||||
() => leaderboard.data?.teams || leaderboard.data?.squadrons || [],
|
||||
[leaderboard.data],
|
||||
)
|
||||
const matches = live.data?.matches || []
|
||||
const topTeamName = bestTeamName(teams[0])
|
||||
const searchPlaceholder =
|
||||
searchHint.status === 'ready' ? `Found ${searchHint.name}` : topTeamName || 'Search teams'
|
||||
|
||||
async function handleTeamSearch(event) {
|
||||
event.preventDefault()
|
||||
const name = teamQuery.trim()
|
||||
if (!name) return
|
||||
|
||||
try {
|
||||
const resolved = await fetchJson(apiEndpoints.resolve(name))
|
||||
navigate(teamPath(resolved.tag_name || resolved.short_name || resolved.long_name || name))
|
||||
} catch {
|
||||
navigate(teamPath(name))
|
||||
}
|
||||
}
|
||||
|
||||
const activeNavPath =
|
||||
route.page === 'team'
|
||||
? '/teams'
|
||||
: route.page === 'battle-logs'
|
||||
? '/battle-logs'
|
||||
: window.location.pathname
|
||||
|
||||
return (
|
||||
<main className="min-h-screen bg-bg text-text">
|
||||
<header className="sticky top-0 z-50 border-b border-border bg-bg/90 backdrop-blur">
|
||||
<div className="mx-auto flex w-full max-w-7xl flex-col gap-3 px-5 py-3 sm:px-8 xl:flex-row xl:items-center">
|
||||
<button className="shrink-0 text-left" onClick={() => navigate('/')} type="button">
|
||||
<span className="text-lg font-bold tracking-tight">Toothless' TSS Bot</span>
|
||||
</button>
|
||||
|
||||
<nav className="flex flex-wrap items-center gap-1 rounded-lg bg-surface/70 p-1 xl:ml-8">
|
||||
{navItems.map((item) => (
|
||||
<button
|
||||
className={`rounded-md px-3 py-2 text-sm font-semibold transition ${activeNavPath === item.path
|
||||
? 'bg-text text-bg'
|
||||
: 'text-text-soft hover:bg-fury-white'
|
||||
}`}
|
||||
key={item.path}
|
||||
onClick={() => navigate(item.path)}
|
||||
type="button"
|
||||
>
|
||||
{item.label}
|
||||
</button>
|
||||
))}
|
||||
<a
|
||||
className="rounded-md px-3 py-2 text-sm font-semibold text-text-soft transition hover:bg-fury-white"
|
||||
href="https://sre.pawjob.us/"
|
||||
>
|
||||
SRE
|
||||
</a>
|
||||
</nav>
|
||||
|
||||
<form className="flex min-w-0 gap-2 xl:ml-auto xl:w-[360px]" onSubmit={handleTeamSearch}>
|
||||
<input
|
||||
className="min-w-0 flex-1 rounded-md border border-border bg-fury-white px-3 py-2 text-sm outline-none transition focus:border-ring focus:shadow-[0_0_0_3px_var(--color-shadow)]"
|
||||
placeholder={searchPlaceholder}
|
||||
value={teamQuery}
|
||||
onChange={(event) => setTeamQuery(event.target.value)}
|
||||
/>
|
||||
<button
|
||||
className="rounded-md bg-fury-cyan px-4 py-2 text-sm font-semibold text-text transition hover:bg-fury-aqua"
|
||||
type="submit"
|
||||
>
|
||||
Search
|
||||
</button>
|
||||
</form>
|
||||
</div>
|
||||
</header>
|
||||
|
||||
<section className="mx-auto w-full max-w-7xl px-5 sm:px-8">
|
||||
{route.page === 'home' ? (
|
||||
<Landing live={live} matches={matches} navigate={navigate} />
|
||||
) : null}
|
||||
{route.page === 'teams' ? (
|
||||
<TeamsPage leaderboard={leaderboard} navigate={navigate} teams={teams} />
|
||||
) : null}
|
||||
{route.page === 'team' ? (
|
||||
<TeamProfilePage
|
||||
navigate={navigate}
|
||||
profile={profile}
|
||||
requestedTeam={route.teamName}
|
||||
teams={teams}
|
||||
/>
|
||||
) : null}
|
||||
{route.page === 'battle-logs' ? <BattleLogsPage live={live} matches={matches} /> : null}
|
||||
</section>
|
||||
</main>
|
||||
)
|
||||
}
|
||||
|
||||
function Landing({ live, matches, navigate }) {
|
||||
const treeRef = useRef(null)
|
||||
|
||||
return (
|
||||
<div className="relative left-1/2 w-screen -translate-x-1/2 bg-bg">
|
||||
<div className="relative min-h-[calc(100vh-74px)] overflow-hidden px-5 sm:px-8">
|
||||
<PixelMountains />
|
||||
|
||||
<section className="relative z-10 mx-auto grid min-h-[calc(100vh-74px)] w-full max-w-7xl gap-8 pt-16 pb-10 lg:grid-cols-[1.05fr_0.95fr] lg:items-center">
|
||||
<div className="max-w-3xl">
|
||||
<p className="text-base font-semibold uppercase tracking-wide text-fury-cyan">
|
||||
BorisBot got nothin on THIS
|
||||
</p>
|
||||
<h1 className="mt-3 text-6xl font-bold tracking-normal sm:text-7xl lg:text-8xl">
|
||||
Toothless' TSS Bot
|
||||
</h1>
|
||||
<p className="mt-6 max-w-2xl text-xl leading-9 text-text-soft">
|
||||
Powered by Spectra. TSS analytics.
|
||||
</p>
|
||||
<div className="mt-8 flex flex-wrap gap-4">
|
||||
<button
|
||||
className="rounded-lg bg-text px-7 py-4 text-base font-semibold text-bg"
|
||||
onClick={() => navigate('/teams')}
|
||||
type="button"
|
||||
>
|
||||
Team leaderboard
|
||||
</button>
|
||||
<button
|
||||
className="rounded-lg border-2 border-ring px-7 py-4 text-base font-semibold text-fury-cyan"
|
||||
onClick={() => navigate('/battle-logs')}
|
||||
type="button"
|
||||
>
|
||||
Battle Logs
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="relative min-h-[520px] overflow-hidden">
|
||||
<FallingLeaves treeRef={treeRef} />
|
||||
<div className="absolute inset-0 z-[1] flex items-end justify-center pb-10 lg:pb-0">
|
||||
<Tree ref={treeRef} />
|
||||
</div>
|
||||
</div>
|
||||
</section>
|
||||
</div>
|
||||
|
||||
<RecentGamesSection live={live} matches={matches} navigate={navigate} />
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
function RecentGamesSection({ live, matches, navigate }) {
|
||||
const recentMatches = matches.slice(0, 6)
|
||||
|
||||
return (
|
||||
<section className="relative z-20 border-t border-border bg-fury-white px-5 py-10 sm:px-8">
|
||||
<div className="mx-auto w-full max-w-7xl">
|
||||
<div className="flex flex-col gap-3 sm:flex-row sm:items-end sm:justify-between">
|
||||
<div>
|
||||
<p className="text-sm font-semibold uppercase tracking-wide text-fury-cyan">
|
||||
Recent activity
|
||||
</p>
|
||||
<h2 className="mt-1 text-3xl font-bold">Latest games</h2>
|
||||
</div>
|
||||
<button
|
||||
className="w-fit rounded-lg border border-ring px-4 py-2 text-sm font-semibold text-fury-cyan transition hover:bg-surface"
|
||||
onClick={() => navigate('/battle-logs')}
|
||||
type="button"
|
||||
>
|
||||
View Battle Logs
|
||||
</button>
|
||||
</div>
|
||||
|
||||
<div className="mt-6 grid gap-4 lg:grid-cols-3">
|
||||
{recentMatches.map((match) => (
|
||||
<article
|
||||
className="rounded-lg border border-border bg-bg p-4 shadow-sm"
|
||||
key={match.session_id}
|
||||
>
|
||||
<div className="flex items-start justify-between gap-3">
|
||||
<div className="min-w-0">
|
||||
<h3 className="truncate text-lg font-semibold">
|
||||
{match.map_name || 'Unknown map'}
|
||||
</h3>
|
||||
<p className="mt-1 text-xs text-text-soft">{formatDate(match.endtime_unix)}</p>
|
||||
</div>
|
||||
<span className="shrink-0 rounded-md bg-surface px-2 py-1 text-xs font-semibold text-text-soft">
|
||||
{match.game_type || match.mode || 'SQB'}
|
||||
</span>
|
||||
</div>
|
||||
|
||||
<div className="mt-4 grid grid-cols-[1fr_auto_1fr] items-center gap-3 text-sm">
|
||||
<p className="truncate text-right font-semibold text-fury-cyan">
|
||||
{match.winning_tag || match.winning_squadron || 'Winner'}
|
||||
</p>
|
||||
<span className="text-xs font-semibold uppercase text-text-muted">vs</span>
|
||||
<p className="truncate font-semibold text-fury-violet">
|
||||
{match.losing_tag || match.losing_squadron || 'Loser'}
|
||||
</p>
|
||||
</div>
|
||||
</article>
|
||||
))}
|
||||
</div>
|
||||
|
||||
{!recentMatches.length ? (
|
||||
<p className="mt-6 rounded-lg border border-border bg-bg px-5 py-6 text-sm text-text-soft">
|
||||
{live.status === 'loading' ? 'Loading latest games' : live.error || 'No games returned'}
|
||||
</p>
|
||||
) : null}
|
||||
</div>
|
||||
</section>
|
||||
)
|
||||
}
|
||||
|
||||
function PixelMountains() {
|
||||
const canvasRef = useRef(null)
|
||||
|
||||
useEffect(() => {
|
||||
const canvas = canvasRef.current
|
||||
const ctx = canvas.getContext('2d')
|
||||
const WORLD_W = 1920
|
||||
const WORLD_H = 900
|
||||
|
||||
function interpolate(points, x) {
|
||||
for (let i = 0; i < points.length - 1; i++) {
|
||||
const [x0, y0] = points[i]
|
||||
const [x1, y1] = points[i + 1]
|
||||
if (x >= x0 && x <= x1) {
|
||||
const t = (x - x0) / Math.max(1, x1 - x0)
|
||||
return y0 + (y1 - y0) * t
|
||||
}
|
||||
}
|
||||
|
||||
return points.at(-1)[1]
|
||||
}
|
||||
|
||||
function drawMountain(points, color, jitter = 0) {
|
||||
const width = WORLD_W
|
||||
const height = WORLD_H
|
||||
ctx.fillStyle = color
|
||||
|
||||
for (let x = 0; x < width; x++) {
|
||||
const wave = jitter
|
||||
? Math.sin(x * 0.08) * jitter + Math.sin(x * 0.021) * jitter * 1.8
|
||||
: 0
|
||||
const y = Math.round(interpolate(points, x) + wave)
|
||||
ctx.fillRect(x, y, 1, height - y)
|
||||
}
|
||||
}
|
||||
|
||||
function draw() {
|
||||
const width = WORLD_W
|
||||
const height = WORLD_H
|
||||
|
||||
canvas.width = WORLD_W
|
||||
canvas.height = WORLD_H
|
||||
ctx.imageSmoothingEnabled = false
|
||||
ctx.clearRect(0, 0, WORLD_W, WORLD_H)
|
||||
|
||||
drawMountain(
|
||||
[
|
||||
[0, height * 0.82],
|
||||
[width * 0.12, height * 0.73],
|
||||
[width * 0.26, height * 0.66],
|
||||
[width * 0.39, height * 0.76],
|
||||
[width * 0.55, height * 0.58],
|
||||
[width * 0.64, height * 0.46],
|
||||
[width * 0.73, height * 0.62],
|
||||
[width * 0.86, height * 0.56],
|
||||
[width, height * 0.64],
|
||||
],
|
||||
'#fcfbcf',
|
||||
1.1,
|
||||
)
|
||||
|
||||
drawMountain(
|
||||
[
|
||||
[0, height * 0.86],
|
||||
[width * 0.15, height * 0.78],
|
||||
[width * 0.31, height * 0.81],
|
||||
[width * 0.43, height * 0.69],
|
||||
[width * 0.52, height * 0.76],
|
||||
[width * 0.62, height * 0.43],
|
||||
[width * 0.69, height * 0.31],
|
||||
[width * 0.78, height * 0.48],
|
||||
[width, height * 0.5],
|
||||
],
|
||||
'#fff2e6',
|
||||
1.6,
|
||||
)
|
||||
|
||||
drawMountain(
|
||||
[
|
||||
[0, height * 0.94],
|
||||
[width * 0.17, height * 0.9],
|
||||
[width * 0.32, height * 0.92],
|
||||
[width * 0.47, height * 0.83],
|
||||
[width * 0.58, height * 0.89],
|
||||
[width * 0.66, height * 0.61],
|
||||
[width * 0.71, height * 0.5],
|
||||
[width * 0.84, height * 0.7],
|
||||
[width, height * 0.68],
|
||||
],
|
||||
'#fee5cd',
|
||||
2.1,
|
||||
)
|
||||
|
||||
drawMountain(
|
||||
[
|
||||
[0, height * 0.98],
|
||||
[width * 0.17, height * 0.96],
|
||||
[width * 0.34, height * 0.99],
|
||||
[width * 0.48, height * 0.93],
|
||||
[width * 0.62, height * 0.97],
|
||||
[width * 0.72, height * 0.88],
|
||||
[width * 0.86, height * 0.95],
|
||||
[width, height * 0.93],
|
||||
],
|
||||
'#fdca9b',
|
||||
1.4,
|
||||
)
|
||||
}
|
||||
|
||||
draw()
|
||||
}, [])
|
||||
|
||||
return <canvas ref={canvasRef} className="pixel-mountains" aria-hidden="true" />
|
||||
}
|
||||
|
||||
function TeamsPage({ leaderboard, navigate, teams }) {
|
||||
return (
|
||||
<section className="space-y-6">
|
||||
<div>
|
||||
<h1 className="text-3xl font-bold">Team leaderboard</h1>
|
||||
<p className="mt-2 text-sm text-text-soft">
|
||||
{leaderboard.status === 'loading'
|
||||
? 'Loading leaderboard'
|
||||
: leaderboard.error || `${teams.length} teams returned`}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div className="overflow-hidden rounded-lg border border-border bg-fury-white shadow-sm">
|
||||
{teams.map((team, index) => {
|
||||
const name = bestTeamName(team)
|
||||
return (
|
||||
<button
|
||||
className="grid w-full gap-4 border-b border-surface px-5 py-4 text-left hover:bg-surface md:grid-cols-[4rem_1fr_repeat(4,auto)] md:items-center"
|
||||
key={`${name}-${team.clan_id || index}`}
|
||||
onClick={() => navigate(teamPath(name))}
|
||||
type="button"
|
||||
>
|
||||
<span className="text-sm font-semibold text-fury-cyan">#{index + 1}</span>
|
||||
<span className="min-w-0">
|
||||
<span className="block truncate text-lg font-semibold">{name}</span>
|
||||
<span className="block truncate text-xs text-text-soft">
|
||||
{team.long_name || team.short_name || 'Unresolved'}
|
||||
</span>
|
||||
</span>
|
||||
<span className="text-sm">{formatNumber(team.player_count)} players</span>
|
||||
<span className="text-sm">{formatNumber(team.total_battles)} battles</span>
|
||||
<span className="text-sm">{Number(team.win_rate || 0).toFixed(1)}% WR</span>
|
||||
<span className="text-sm font-semibold">
|
||||
{formatNumber(team.points?.total_points || team.total_kills)}
|
||||
</span>
|
||||
</button>
|
||||
)
|
||||
})}
|
||||
|
||||
{!teams.length ? (
|
||||
<p className="px-5 py-10 text-sm text-text-soft">
|
||||
{leaderboard.status === 'loading' ? 'Loading leaderboard' : 'No teams returned'}
|
||||
</p>
|
||||
) : null}
|
||||
</div>
|
||||
</section>
|
||||
)
|
||||
}
|
||||
|
||||
function TeamProfilePage({ navigate, profile, requestedTeam, teams }) {
|
||||
const detail = profile.detail.data
|
||||
const summary = detail?.team_summary || detail?.squadron_summary
|
||||
const players = detail?.players || []
|
||||
const games = profile.games.data?.games || []
|
||||
const history = profile.history.data?.history || []
|
||||
const ratingHourly = profile.history.data?.rating_hourly || []
|
||||
const latestRating = ratingHourly.at(-1)?.rating || summary?.points?.total_points
|
||||
const leaderboardTeam = teams.find((team) => bestTeamName(team) === requestedTeam)
|
||||
const displayName = detail?.tag_name || bestTeamName(leaderboardTeam) || requestedTeam
|
||||
const longName = detail?.long_name || leaderboardTeam?.long_name || ''
|
||||
|
||||
return (
|
||||
<section className="space-y-6">
|
||||
<button
|
||||
className="text-sm font-semibold text-fury-cyan hover:text-text"
|
||||
onClick={() => navigate('/teams')}
|
||||
type="button"
|
||||
>
|
||||
Back to leaderboard
|
||||
</button>
|
||||
|
||||
<div className="rounded-lg border border-border bg-fury-white p-6 shadow-sm">
|
||||
<div className="flex flex-col gap-4 lg:flex-row lg:items-start lg:justify-between">
|
||||
<div>
|
||||
<p className="text-sm font-semibold uppercase tracking-wide text-fury-cyan">
|
||||
Team profile
|
||||
</p>
|
||||
<h1 className="mt-1 text-4xl font-bold">{displayName}</h1>
|
||||
<p className="mt-2 text-sm text-text-soft">
|
||||
{profile.detail.error || longName || profile.detail.status}
|
||||
</p>
|
||||
</div>
|
||||
<div className="grid grid-cols-2 gap-3 text-sm sm:grid-cols-3">
|
||||
<span className="rounded-md bg-surface px-3 py-2 font-semibold">
|
||||
Rating {formatNumber(latestRating)}
|
||||
</span>
|
||||
<span className="rounded-md bg-surface px-3 py-2 font-semibold">
|
||||
Clan {detail?.clan_id || leaderboardTeam?.clan_id || 'n/a'}
|
||||
</span>
|
||||
<span className="rounded-md bg-surface px-3 py-2 font-semibold">
|
||||
{detail?.data_set || 'tss'}
|
||||
</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="mt-6 grid gap-5 sm:grid-cols-2 xl:grid-cols-5">
|
||||
<Stat label="Roster" value={formatNumber(summary?.player_count)} />
|
||||
<Stat label="Battles" value={formatNumber(summary?.total_battles)} />
|
||||
<Stat label="Wins" value={formatNumber(summary?.wins)} />
|
||||
<Stat label="Win rate" value={`${Number(summary?.win_rate || 0).toFixed(1)}%`} />
|
||||
<Stat label="KDR" value={Number(summary?.kdr || 0).toFixed(1)} />
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div className="grid gap-6 xl:grid-cols-[1.15fr_0.85fr]">
|
||||
<RosterTable players={players} status={profile.detail.status} />
|
||||
<RatingPanel history={history} ratingHourly={ratingHourly} status={profile.history.status} />
|
||||
</div>
|
||||
|
||||
<BattleResults games={games} status={profile.games.status} />
|
||||
</section>
|
||||
)
|
||||
}
|
||||
|
||||
function RosterTable({ players, status }) {
|
||||
const sortedPlayers = [...players].sort((a, b) => {
|
||||
return (b.total_kills || 0) - (a.total_kills || 0) || String(a.nick || '').localeCompare(b.nick || '')
|
||||
})
|
||||
|
||||
return (
|
||||
<div className="rounded-lg border border-border bg-fury-white shadow-sm">
|
||||
<div className="border-b border-surface px-5 py-4">
|
||||
<h2 className="text-lg font-semibold">Roster</h2>
|
||||
<p className="mt-1 text-sm text-text-soft">{formatNumber(sortedPlayers.length)} players</p>
|
||||
</div>
|
||||
<div className="max-h-[520px] overflow-auto">
|
||||
{sortedPlayers.map((player) => (
|
||||
<div
|
||||
className="grid gap-3 border-b border-surface px-5 py-3 text-sm md:grid-cols-[1fr_repeat(5,auto)] md:items-center"
|
||||
key={player.uid}
|
||||
>
|
||||
<div className="min-w-0">
|
||||
<p className="truncate font-semibold">{player.nick || player.uid}</p>
|
||||
<p className="text-xs text-text-soft">
|
||||
{player.uid} · {formatNumber(player.points || player.sqb_points)} pts
|
||||
</p>
|
||||
</div>
|
||||
<p>{formatNumber(player.total_battles)} battles</p>
|
||||
<p>{formatNumber(player.total_kills)} kills</p>
|
||||
<p>{Number(player.win_rate || 0).toFixed(1)}% WR</p>
|
||||
<p>{Number(player.kdr || 0).toFixed(1)} KDR</p>
|
||||
<p>{formatNumber(player.assists)} assists</p>
|
||||
</div>
|
||||
))}
|
||||
{!sortedPlayers.length ? (
|
||||
<p className="px-5 py-10 text-sm text-text-soft">
|
||||
{status === 'loading' ? 'Loading roster' : 'No roster rows returned'}
|
||||
</p>
|
||||
) : null}
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
function RatingPanel({ history, ratingHourly, status }) {
|
||||
const recentHistory = history.slice(-8)
|
||||
const firstRating = ratingHourly[0]?.rating || 0
|
||||
const latestRating = ratingHourly.at(-1)?.rating || 0
|
||||
const ratingChange = latestRating - firstRating
|
||||
|
||||
return (
|
||||
<div className="rounded-lg border border-border bg-fury-white shadow-sm">
|
||||
<div className="border-b border-surface px-5 py-4">
|
||||
<h2 className="text-lg font-semibold">History</h2>
|
||||
<p className="mt-1 text-sm text-text-soft">
|
||||
{ratingHourly.length ? `${formatNumber(ratingHourly.length)} rating snapshots` : status}
|
||||
</p>
|
||||
</div>
|
||||
<div className="space-y-5 p-5">
|
||||
<div className="grid gap-5 sm:grid-cols-2">
|
||||
<Stat label="Latest rating" value={formatNumber(latestRating)} />
|
||||
<Stat
|
||||
label="Rating change"
|
||||
value={`${ratingChange >= 0 ? '+' : ''}${formatNumber(ratingChange)}`}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="space-y-2">
|
||||
{recentHistory.map((item) => (
|
||||
<div
|
||||
className="grid grid-cols-[1fr_auto_auto] gap-3 rounded-md bg-surface px-3 py-2 text-sm"
|
||||
key={item.period}
|
||||
>
|
||||
<span className="font-semibold">{item.period}</span>
|
||||
<span>{formatNumber(item.battles)} battles</span>
|
||||
<span>{Number(item.win_rate || 0).toFixed(1)}% WR</span>
|
||||
</div>
|
||||
))}
|
||||
{!recentHistory.length ? (
|
||||
<p className="text-sm text-text-soft">
|
||||
{status === 'loading' ? 'Loading history' : 'No history rows returned'}
|
||||
</p>
|
||||
) : null}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
function BattleResults({ games, status }) {
|
||||
return (
|
||||
<div className="rounded-lg border border-border bg-fury-white shadow-sm">
|
||||
<div className="border-b border-surface px-5 py-4">
|
||||
<h2 className="text-lg font-semibold">Battle results</h2>
|
||||
<p className="mt-1 text-sm text-text-soft">{formatNumber(games.length)} battles returned</p>
|
||||
</div>
|
||||
<div className="max-h-[560px] overflow-auto">
|
||||
{games.map((game) => (
|
||||
<div
|
||||
className="grid gap-4 border-b border-surface px-5 py-4 md:grid-cols-[1fr_auto_repeat(5,auto)] md:items-center"
|
||||
key={game.session_id}
|
||||
>
|
||||
<div className="min-w-0">
|
||||
<p className="truncate font-semibold">{game.map_name || 'Unknown map'}</p>
|
||||
<p className="text-xs text-text-soft">
|
||||
{formatDate(game.timestamp)} · {game.session_id}
|
||||
</p>
|
||||
</div>
|
||||
<p
|
||||
className={`rounded-md px-3 py-1 text-sm font-semibold ${String(game.result).toLowerCase() === 'win'
|
||||
? 'bg-surface text-fury-cyan'
|
||||
: 'bg-fury-ice text-fury-violet'
|
||||
}`}
|
||||
>
|
||||
{game.result || 'Unknown'}
|
||||
</p>
|
||||
<p className="text-sm">{formatNumber(game.player_count)} players</p>
|
||||
<p className="text-sm">{formatNumber(game.stats?.ground_kills)} ground</p>
|
||||
<p className="text-sm">{formatNumber(game.stats?.air_kills)} air</p>
|
||||
<p className="text-sm">{formatNumber(game.stats?.assists)} assists</p>
|
||||
<p className="text-sm">{formatNumber(game.stats?.deaths)} deaths</p>
|
||||
</div>
|
||||
))}
|
||||
{!games.length ? (
|
||||
<p className="px-5 py-10 text-sm text-text-soft">
|
||||
{status === 'loading' ? 'Loading battle results' : 'No battle results returned'}
|
||||
</p>
|
||||
) : null}
|
||||
</div>
|
||||
</div>
|
||||
)
|
||||
}
|
||||
|
||||
function BattleLogsPage({ live, matches }) {
|
||||
return (
|
||||
<section className="space-y-6">
|
||||
<div>
|
||||
<h1 className="text-3xl font-bold">Battle Logs</h1>
|
||||
<p className="mt-2 text-sm text-text-soft">
|
||||
{live.status === 'loading' ? 'Loading battles' : live.error || `${matches.length} battles returned`}
|
||||
</p>
|
||||
</div>
|
||||
|
||||
<div className="overflow-hidden rounded-lg border border-border bg-fury-white shadow-sm">
|
||||
{matches.map((match) => (
|
||||
<div
|
||||
className="grid gap-4 border-b border-surface px-5 py-4 md:grid-cols-[1fr_1.2fr_auto]"
|
||||
key={match.session_id}
|
||||
>
|
||||
<div className="min-w-0">
|
||||
<p className="truncate font-semibold">{match.map_name || 'Unknown map'}</p>
|
||||
<p className="text-xs text-text-soft">{formatDate(match.endtime_unix)}</p>
|
||||
</div>
|
||||
<div className="grid grid-cols-[1fr_auto_1fr] items-center gap-3 text-sm">
|
||||
<p className="truncate text-right font-semibold text-fury-cyan">
|
||||
{match.winning_tag || match.winning_squadron || 'Winner'}
|
||||
</p>
|
||||
<span className="text-xs font-semibold uppercase text-text-muted">vs</span>
|
||||
<p className="truncate font-semibold text-fury-violet">
|
||||
{match.losing_tag || match.losing_squadron || 'Loser'}
|
||||
</p>
|
||||
</div>
|
||||
<p className="text-sm text-text-soft">{match.game_type || match.mode || 'SQB'}</p>
|
||||
</div>
|
||||
))}
|
||||
|
||||
{!matches.length ? (
|
||||
<p className="px-5 py-10 text-sm text-text-soft">
|
||||
{live.status === 'loading' ? 'Loading battles' : 'No battles returned'}
|
||||
</p>
|
||||
) : null}
|
||||
</div>
|
||||
</section>
|
||||
)
|
||||
}
|
||||
|
||||
export default App
|
||||
@@ -0,0 +1,10 @@
|
||||
import { StrictMode } from 'react'
|
||||
import { createRoot } from 'react-dom/client'
|
||||
import './styles.css'
|
||||
import App from './App.jsx'
|
||||
|
||||
createRoot(document.getElementById('root')).render(
|
||||
<StrictMode>
|
||||
<App />
|
||||
</StrictMode>,
|
||||
)
|
||||
+132
@@ -0,0 +1,132 @@
|
||||
@import "tailwindcss";
|
||||
|
||||
@theme {
|
||||
--color-cerulean-50: #e5f9ff;
|
||||
--color-cerulean-100: #ccf3ff;
|
||||
--color-cerulean-200: #99e7ff;
|
||||
--color-cerulean-300: #66dbff;
|
||||
--color-cerulean-400: #33cfff;
|
||||
--color-cerulean-500: #00c3ff;
|
||||
--color-cerulean-600: #009ccc;
|
||||
--color-cerulean-700: #007599;
|
||||
--color-cerulean-800: #004e66;
|
||||
--color-cerulean-900: #002733;
|
||||
--color-cerulean-950: #001b24;
|
||||
|
||||
--color-tropical-teal-50: #e5feff;
|
||||
--color-tropical-teal-100: #ccfcff;
|
||||
--color-tropical-teal-200: #99faff;
|
||||
--color-tropical-teal-300: #66f7ff;
|
||||
--color-tropical-teal-400: #33f5ff;
|
||||
--color-tropical-teal-500: #00f2ff;
|
||||
--color-tropical-teal-600: #00c2cc;
|
||||
--color-tropical-teal-700: #009199;
|
||||
--color-tropical-teal-800: #006166;
|
||||
--color-tropical-teal-900: #003033;
|
||||
--color-tropical-teal-950: #002224;
|
||||
|
||||
--color-light-yellow-50: #fefde7;
|
||||
--color-light-yellow-100: #fcfbcf;
|
||||
--color-light-yellow-200: #f9f69f;
|
||||
--color-light-yellow-300: #f7f26e;
|
||||
--color-light-yellow-400: #f4ee3e;
|
||||
--color-light-yellow-500: #f1e90e;
|
||||
--color-light-yellow-600: #c1bb0b;
|
||||
--color-light-yellow-700: #918c08;
|
||||
--color-light-yellow-800: #605d06;
|
||||
--color-light-yellow-900: #302f03;
|
||||
--color-light-yellow-950: #222102;
|
||||
|
||||
--color-soft-apricot-50: #fff2e6;
|
||||
--color-soft-apricot-100: #fee5cd;
|
||||
--color-soft-apricot-200: #fdca9b;
|
||||
--color-soft-apricot-300: #fdb068;
|
||||
--color-soft-apricot-400: #fc9636;
|
||||
--color-soft-apricot-500: #fb7b04;
|
||||
--color-soft-apricot-600: #c96303;
|
||||
--color-soft-apricot-700: #974a02;
|
||||
--color-soft-apricot-800: #643102;
|
||||
--color-soft-apricot-900: #321901;
|
||||
--color-soft-apricot-950: #231101;
|
||||
|
||||
--color-vibrant-coral-50: #fde9e8;
|
||||
--color-vibrant-coral-100: #fad3d1;
|
||||
--color-vibrant-coral-200: #f6a8a2;
|
||||
--color-vibrant-coral-300: #f17c74;
|
||||
--color-vibrant-coral-400: #ed5145;
|
||||
--color-vibrant-coral-500: #e82517;
|
||||
--color-vibrant-coral-600: #ba1e12;
|
||||
--color-vibrant-coral-700: #8b160e;
|
||||
--color-vibrant-coral-800: #5d0f09;
|
||||
--color-vibrant-coral-900: #2e0705;
|
||||
--color-vibrant-coral-950: #200503;
|
||||
|
||||
--color-bg: #fefde7;
|
||||
--color-surface: #fcfbcf;
|
||||
--color-surface-alt: #f9f69f;
|
||||
--color-fury-white: #fefde7;
|
||||
--color-fury-ice: #fcfbcf;
|
||||
--color-fury-blue: #f9f69f;
|
||||
--color-fury-glow: #fdb068;
|
||||
--color-fury-cyan: #e82517;
|
||||
--color-fury-aqua: #ed5145;
|
||||
--color-fury-violet: #fb7b04;
|
||||
--color-text: #000000;
|
||||
--color-text-soft: #555555;
|
||||
--color-text-muted: #888888;
|
||||
--color-border: #fee5cd;
|
||||
--color-ring: #ed5145;
|
||||
--color-shadow: rgba(232, 37, 23, 0.12);
|
||||
--color-success: #00f2ff;
|
||||
--color-warning: #f4ee3e;
|
||||
--color-danger: #e82517;
|
||||
}
|
||||
*,
|
||||
*::before,
|
||||
*::after {
|
||||
box-sizing: border-box;
|
||||
}
|
||||
|
||||
body {
|
||||
margin: 0;
|
||||
min-width: 320px;
|
||||
min-height: 100vh;
|
||||
font-family:
|
||||
Inter, ui-sans-serif, system-ui, -apple-system, BlinkMacSystemFont, "Segoe UI",
|
||||
sans-serif;
|
||||
}
|
||||
|
||||
.pixel-mountains {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
image-rendering: pixelated;
|
||||
object-fit: cover;
|
||||
object-position: center bottom;
|
||||
pointer-events: none;
|
||||
z-index: 0;
|
||||
}
|
||||
|
||||
.tree {
|
||||
width: min(100%, 500px);
|
||||
height: auto;
|
||||
display: block;
|
||||
background: #fefde7;
|
||||
border: 3px solid #fdca9b;
|
||||
border-radius: 16px;
|
||||
padding: 12px;
|
||||
box-shadow:
|
||||
0 4px 24px rgba(253, 202, 155, 0.35),
|
||||
0 1px 3px rgba(50, 25, 1, 0.08);
|
||||
}
|
||||
|
||||
.falling-leaves {
|
||||
position: absolute;
|
||||
inset: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
pointer-events: none;
|
||||
z-index: 2;
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user