feat: setup 'web' frontend

This commit is contained in:
Ryan Walters
2025-09-25 12:37:21 -05:00
parent c524fdb3e7
commit 55b31ba31e
19 changed files with 950 additions and 0 deletions

13
web/pages/+Head.tsx Normal file
View File

@@ -0,0 +1,13 @@
// https://vike.dev/Head
//# BATI.has("mantine")
import { ColorSchemeScript } from "@mantine/core";
export default function HeadDefault() {
return (
<>
<link rel="icon" href="/favicon.ico" />
<ColorSchemeScript />
</>
);
}

17
web/pages/+config.ts Normal file
View File

@@ -0,0 +1,17 @@
import type { Config } from "vike/types";
import vikeReact from "vike-react/config";
import Layout from "../layouts/LayoutDefault.js";
// Default config (can be overridden by pages)
// https://vike.dev/config
export default {
// https://vike.dev/Layout
Layout,
// https://vike.dev/head-tags
title: "Pac-Man",
description: "A Pac-Man game built with Rust and React.",
extends: vikeReact,
} satisfies Config;

View File

@@ -0,0 +1,6 @@
import type { OnPageTransitionEndAsync } from "vike/types";
export const onPageTransitionEnd: OnPageTransitionEndAsync = async () => {
console.log("Page transition end");
document.querySelector("body")?.classList.remove("page-is-transitioning");
};

View File

@@ -0,0 +1,6 @@
import type { OnPageTransitionStartAsync } from "vike/types";
export const onPageTransitionStart: OnPageTransitionStartAsync = async () => {
console.log("Page transition start");
document.querySelector("body")?.classList.add("page-is-transitioning");
};

View File

@@ -0,0 +1,19 @@
import { usePageContext } from "vike-react/usePageContext";
export default function Page() {
const { is404 } = usePageContext();
if (is404) {
return (
<>
<h1>Page Not Found</h1>
<p>This page could not be found.</p>
</>
);
}
return (
<>
<h1>Internal Error</h1>
<p>Something went wrong.</p>
</>
);
}

16
web/pages/index/+Page.tsx Normal file
View File

@@ -0,0 +1,16 @@
import { AspectRatio } from "@mantine/core";
export default function Page() {
return (
<div className="mt-4 flex justify-center h-[calc(100vh-120px)]">
<AspectRatio ratio={1.0 / 1.2} w="min(100vh * 1.2, 100vw)" maw="95vw">
<canvas
className="block border-1 border-yellow-400/50 w-full h-full"
style={{
boxShadow: "0 0 12px rgba(250, 204, 21, 0.35), 0 0 2px rgba(255, 255, 255, 0.25)",
}}
/>
</AspectRatio>
</div>
);
}

View File

@@ -0,0 +1,294 @@
import { useState } from "react";
import { Container, Title, Table, Tabs, Avatar, Text, Group, Badge, Stack, Paper } from "@mantine/core";
import { IconTrophy, IconCalendar } from "@tabler/icons-react";
interface LeaderboardEntry {
id: number;
rank: number;
name: string;
score: number;
duration: string;
levelCount: number;
submittedAt: string;
avatar?: string;
}
const mockGlobalData: LeaderboardEntry[] = [
{
id: 1,
rank: 1,
name: "PacMaster2024",
score: 125000,
duration: "45:32",
levelCount: 12,
submittedAt: "2 hours ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=PacMaster2024",
},
{
id: 2,
rank: 2,
name: "GhostHunter",
score: 118750,
duration: "42:18",
levelCount: 11,
submittedAt: "5 hours ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=GhostHunter",
},
{
id: 3,
rank: 3,
name: "DotCollector",
score: 112500,
duration: "38:45",
levelCount: 10,
submittedAt: "1 day ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=DotCollector",
},
{
id: 4,
rank: 4,
name: "MazeRunner",
score: 108900,
duration: "41:12",
levelCount: 10,
submittedAt: "2 days ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=MazeRunner",
},
{
id: 5,
rank: 5,
name: "PowerPellet",
score: 102300,
duration: "36:28",
levelCount: 9,
submittedAt: "3 days ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=PowerPellet",
},
{
id: 6,
rank: 6,
name: "CherryPicker",
score: 98750,
duration: "39:15",
levelCount: 9,
submittedAt: "4 days ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=CherryPicker",
},
{
id: 7,
rank: 7,
name: "BlinkyBeater",
score: 94500,
duration: "35:42",
levelCount: 8,
submittedAt: "5 days ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=BlinkyBeater",
},
{
id: 8,
rank: 8,
name: "PinkyPac",
score: 91200,
duration: "37:55",
levelCount: 8,
submittedAt: "1 week ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=PinkyPac",
},
{
id: 9,
rank: 9,
name: "InkyDestroyer",
score: 88800,
duration: "34:18",
levelCount: 8,
submittedAt: "1 week ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=InkyDestroyer",
},
{
id: 10,
rank: 10,
name: "ClydeChaser",
score: 85600,
duration: "33:45",
levelCount: 7,
submittedAt: "1 week ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=ClydeChaser",
},
];
const mockMonthlyData: LeaderboardEntry[] = [
{
id: 1,
rank: 1,
name: "JanuaryChamp",
score: 115000,
duration: "43:22",
levelCount: 11,
submittedAt: "1 day ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=JanuaryChamp",
},
{
id: 2,
rank: 2,
name: "NewYearPac",
score: 108500,
duration: "40:15",
levelCount: 10,
submittedAt: "3 days ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=NewYearPac",
},
{
id: 3,
rank: 3,
name: "WinterWarrior",
score: 102000,
duration: "38:30",
levelCount: 10,
submittedAt: "5 days ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=WinterWarrior",
},
{
id: 4,
rank: 4,
name: "FrostyPac",
score: 98500,
duration: "37:45",
levelCount: 9,
submittedAt: "1 week ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=FrostyPac",
},
{
id: 5,
rank: 5,
name: "IceBreaker",
score: 95200,
duration: "36:12",
levelCount: 9,
submittedAt: "1 week ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=IceBreaker",
},
{
id: 6,
rank: 6,
name: "SnowPac",
score: 91800,
duration: "35:28",
levelCount: 8,
submittedAt: "2 weeks ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=SnowPac",
},
{
id: 7,
rank: 7,
name: "BlizzardBeast",
score: 88500,
duration: "34:15",
levelCount: 8,
submittedAt: "2 weeks ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=BlizzardBeast",
},
{
id: 8,
rank: 8,
name: "ColdSnap",
score: 85200,
duration: "33:42",
levelCount: 8,
submittedAt: "3 weeks ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=ColdSnap",
},
{
id: 9,
rank: 9,
name: "FrozenFury",
score: 81900,
duration: "32:55",
levelCount: 7,
submittedAt: "3 weeks ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=FrozenFury",
},
{
id: 10,
rank: 10,
name: "ArcticAce",
score: 78600,
duration: "31:18",
levelCount: 7,
submittedAt: "4 weeks ago",
avatar: "https://api.dicebear.com/7.x/avataaars/svg?seed=ArcticAce",
},
];
function LeaderboardTable({ data }: { data: LeaderboardEntry[] }) {
const rows = data.map((entry) => (
<Table.Tr
key={entry.id}
style={{
backgroundColor: "#000",
marginBottom: "8px",
}}
>
<Table.Td>
<Group gap="sm">
<Avatar src={entry.avatar} size="md" radius="sm" alt={entry.name} />
<Stack gap={2}>
<Text fw={600} size="lg" c="yellow.4">
{entry.name}
</Text>
<Text size="xs" c="dimmed">
{entry.submittedAt}
</Text>
</Stack>
</Group>
</Table.Td>
<Table.Td>
<Text fw={500} size="lg" c="yellow.3">
{entry.score.toLocaleString()}
</Text>
</Table.Td>
<Table.Td>
<Text size="md" c="gray.3">
{entry.duration}
</Text>
</Table.Td>
<Table.Td>Level {entry.levelCount}</Table.Td>
</Table.Tr>
));
return (
<Table>
<Table.Tbody>{rows}</Table.Tbody>
</Table>
);
}
export default function Page() {
const [activeTab, setActiveTab] = useState<string | null>("global");
return (
<Container size="md" py="xl">
<Stack gap="xl">
<Paper shadow="lg" p="xl" radius="md" bg="none">
<Tabs value={activeTab} onChange={setActiveTab}>
<Tabs.List>
<Tabs.Tab value="global" leftSection={<IconTrophy size={16} />}>
Global
</Tabs.Tab>
<Tabs.Tab value="monthly" leftSection={<IconCalendar size={16} />}>
Monthly
</Tabs.Tab>
</Tabs.List>
<Tabs.Panel value="global" pt="md">
<LeaderboardTable data={mockGlobalData} />
</Tabs.Panel>
<Tabs.Panel value="monthly" pt="md">
<LeaderboardTable data={mockMonthlyData} />
</Tabs.Panel>
</Tabs>
</Paper>
</Stack>
</Container>
);
}