Files
gridpilot.gg/apps/website/lib/builders/view-data/RacesViewDataBuilder.ts
Marc Mintel 046852703f
Some checks failed
Contract Testing / contract-tests (pull_request) Failing after 5m51s
Contract Testing / contract-snapshot (pull_request) Has been skipped
view data fixes
2026-01-24 12:14:08 +01:00

86 lines
3.0 KiB
TypeScript

/**
* Races View Data Builder
*
* Transforms API DTO to ViewData for templates.
*/
import { DateFormatter } from '@/lib/formatters/DateFormatter';
import { RaceStatusFormatter } from '@/lib/formatters/RaceStatusFormatter';
import { RelativeTimeFormatter } from '@/lib/formatters/RelativeTimeFormatter';
import type { RacesPageDataDTO } from '@/lib/types/generated/RacesPageDataDTO';
import type { RacesViewData, RaceViewData } from '@/lib/view-data/RacesViewData';
import { ViewDataBuilder } from "../../contracts/builders/ViewDataBuilder";
export class RacesViewDataBuilder {
/**
* Transform API DTO to ViewData
*
* @param apiDto - The DTO from the service
* @returns ViewData for the races page
*/
public static build(apiDto: RacesPageDataDTO): RacesViewData {
const now = new Date();
const races = (apiDto.races || []).map((race): RaceViewData => {
return {
id: race.id,
track: race.track,
car: race.car,
scheduledAt: race.scheduledAt,
scheduledAtLabel: DateFormatter.formatShort(race.scheduledAt),
timeLabel: DateFormatter.formatTime(race.scheduledAt),
relativeTimeLabel: RelativeTimeFormatter.format(race.scheduledAt, now),
status: race.status as RaceViewData['status'],
statusLabel: RaceStatusFormatter.getLabel(race.status),
statusVariant: RaceStatusFormatter.getVariant(race.status),
statusIconName: RaceStatusFormatter.getIconName(race.status),
sessionType: 'Race',
leagueId: race.leagueId,
leagueName: race.leagueName,
strengthOfField: race.strengthOfField ?? null,
isUpcoming: race.isUpcoming,
isLive: race.isLive,
isPast: race.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: 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,
};
}
}
RacesViewDataBuilder satisfies ViewDataBuilder<RacesPageDataDTO, RacesViewData>;