Files
gridpilot.gg/apps/website/app/leagues/[id]/page.tsx
Marc Mintel 9ac74f5046
Some checks failed
CI / lint-typecheck (pull_request) Failing after 12s
CI / tests (pull_request) Has been skipped
CI / contract-tests (pull_request) Has been skipped
CI / e2e-tests (pull_request) Has been skipped
CI / comment-pr (pull_request) Has been skipped
CI / commit-types (pull_request) Has been skipped
code quality
2026-01-26 17:22:01 +01:00

104 lines
3.1 KiB
TypeScript

import { notFound } from 'next/navigation';
import { LeagueOverviewTemplate } from '@/templates/LeagueOverviewTemplate';
import { LeagueDetailPageQuery } from '@/lib/page-queries/LeagueDetailPageQuery';
import { LeagueDetailViewDataBuilder } from '@/lib/builders/view-data/LeagueDetailViewDataBuilder';
import { ErrorBanner } from '@/ui/ErrorBanner';
import { Metadata } from 'next';
import { MetadataHelper } from '@/lib/seo/MetadataHelper';
import { JsonLd } from '@/ui/JsonLd';
import { SessionGateway } from '@/lib/gateways/SessionGateway';
interface Props {
params: Promise<{ id: string }>;
}
export async function generateMetadata({ params }: Props): Promise<Metadata> {
const { id } = await params;
const result = await LeagueDetailPageQuery.execute(id);
if (result.isErr()) {
return MetadataHelper.generate({
title: 'League Not Found',
description: 'The requested league could not be found on GridPilot.',
path: `/leagues/${id}`,
});
}
const data = result.unwrap();
const league = data.league;
return MetadataHelper.generate({
title: league.name,
description: league.description || `Join ${league.name} on GridPilot. Professional iRacing league with automated results, standings, and obsessive attention to detail.`,
path: `/leagues/${id}`,
});
}
export default async function Page({ params }: Props) {
const { id } = await params;
// Get current user session
const sessionGateway = new SessionGateway();
const session = await sessionGateway.getSession();
const currentDriverId = session?.user?.primaryDriverId;
// Execute the PageQuery
const result = await LeagueDetailPageQuery.execute(id);
// Handle different result types
if (result.isErr()) {
const error = result.getError();
switch (error) {
case 'notFound':
notFound();
case 'redirect':
// In a real app, this would redirect to login
notFound();
default:
// Return error state
return (
<ErrorBanner
title="Load Failed"
message="Failed to load league details"
variant="error"
/>
);
}
}
const data = result.unwrap();
const league = data.league;
// Determine if current user is owner or admin
const isOwnerOrAdmin = currentDriverId
? currentDriverId === league.ownerId ||
data.memberships.members?.some((m: any) => m.driverId === currentDriverId && m.role === 'admin')
: false;
// Build ViewData using the builder
const viewData = LeagueDetailViewDataBuilder.build({
league: data.league,
owner: data.owner,
scoringConfig: data.scoringConfig,
memberships: data.memberships,
races: data.races,
sponsors: data.sponsors,
});
const jsonLd = {
'@context': 'https://schema.org',
'@type': 'SportsOrganization',
name: league.name,
description: league.description,
url: `https://gridpilot.com/leagues/${league.id}`,
};
return (
<>
<JsonLd data={jsonLd} />
<LeagueOverviewTemplate viewData={viewData} isOwnerOrAdmin={isOwnerOrAdmin} />
</>
);
}