website refactor

This commit is contained in:
2026-01-14 23:46:04 +01:00
parent c1a86348d7
commit 4a2d7d15a5
294 changed files with 5637 additions and 3418 deletions

View File

@@ -1,39 +1,100 @@
import { RacesViewData, RacesRace } from '@/lib/view-data/races/RacesViewData';
import type { RacesPageDataDTO } from '@/lib/types/generated/RacesPageDataDTO';
import type { RacesViewData, RaceViewData } from '@/lib/view-data/RacesViewData';
/**
* Races View Data Builder
*
* Transforms API DTO into ViewData for the races template.
* Deterministic, side-effect free.
*/
export class RacesViewDataBuilder {
static build(apiDto: any): RacesViewData {
const races = apiDto.races.map((race: any) => ({
id: race.id,
track: race.track,
car: race.car,
scheduledAt: race.scheduledAt,
status: race.status as 'scheduled' | 'running' | 'completed' | 'cancelled',
sessionType: 'race',
leagueId: race.leagueId,
leagueName: race.leagueName,
strengthOfField: race.strengthOfField ?? undefined,
isUpcoming: race.status === 'scheduled',
isLive: race.status === 'running',
isPast: race.status === 'completed',
}));
static build(apiDto: RacesPageDataDTO): RacesViewData {
const races = apiDto.races.map((race): RaceViewData => {
const scheduledAt = new Date(race.scheduledAt);
return {
id: race.id,
track: race.track,
car: race.car,
scheduledAt: race.scheduledAt,
scheduledAtLabel: scheduledAt.toLocaleDateString('en-US', {
weekday: 'short',
month: 'short',
day: 'numeric',
}),
timeLabel: scheduledAt.toLocaleTimeString('en-US', {
hour: '2-digit',
minute: '2-digit',
}),
relativeTimeLabel: this.getRelativeTime(scheduledAt),
status: race.status as RaceViewData['status'],
statusLabel: this.getStatusLabel(race.status),
sessionType: 'Race',
leagueId: race.leagueId,
leagueName: race.leagueName,
strengthOfField: race.strengthOfField ?? null,
isUpcoming: race.isUpcoming,
isLive: race.isLive,
isPast: race.isPast,
};
});
const totalCount = races.length;
const scheduledRaces = races.filter((r: RacesRace) => r.isUpcoming);
const runningRaces = races.filter((r: RacesRace) => r.isLive);
const completedRaces = races.filter((r: RacesRace) => r.isPast);
const leagues = Array.from(
new Map(
races
.filter(r => r.leagueId && r.leagueName)
.map(r => [r.leagueId, { id: r.leagueId!, name: r.leagueName! }])
).values()
);
const groupedRaces = new Map<string, RaceViewData[]>();
races.forEach((race) => {
const dateKey = race.scheduledAt.split('T')[0]!;
if (!groupedRaces.has(dateKey)) {
groupedRaces.set(dateKey, []);
}
groupedRaces.get(dateKey)!.push(race);
});
const racesByDate = Array.from(groupedRaces.entries()).map(([dateKey, dayRaces]) => ({
dateKey,
dateLabel: dayRaces[0]?.scheduledAtLabel || '',
races: dayRaces,
}));
return {
races,
totalCount,
scheduledRaces,
runningRaces,
completedRaces,
totalCount: races.length,
scheduledCount: races.filter(r => r.status === 'scheduled').length,
runningCount: races.filter(r => r.status === 'running').length,
completedCount: races.filter(r => r.status === 'completed').length,
leagues,
upcomingRaces: races.filter(r => r.isUpcoming).slice(0, 5),
liveRaces: races.filter(r => r.isLive),
recentResults: races.filter(r => r.isPast).slice(0, 5),
racesByDate,
};
}
}
private static getStatusLabel(status: string): string {
switch (status) {
case 'scheduled': return 'Scheduled';
case 'running': return 'LIVE';
case 'completed': return 'Completed';
case 'cancelled': return 'Cancelled';
default: return status;
}
}
private static getRelativeTime(date: Date): string {
const now = new Date();
const diffMs = date.getTime() - now.getTime();
const diffHours = Math.floor(diffMs / (1000 * 60 * 60));
const diffDays = Math.floor(diffMs / (1000 * 60 * 60 * 24));
if (diffMs < 0) return 'Past';
if (diffHours < 1) return 'Starting soon';
if (diffHours < 24) return `In ${diffHours}h`;
if (diffDays === 1) return 'Tomorrow';
if (diffDays < 7) return `In ${diffDays} days`;
return date.toLocaleDateString('en-US', {
weekday: 'short',
month: 'short',
day: 'numeric',
});
}
}