'use client'; import { useState, useEffect, useCallback } from 'react'; import { useParams } from 'next/navigation'; import Image from 'next/image'; import Link from 'next/link'; import Card from '@/components/ui/Card'; import Button from '@/components/ui/Button'; import Breadcrumbs from '@/components/layout/Breadcrumbs'; import SponsorInsightsCard, { useSponsorMode, MetricBuilders, SlotTemplates } from '@/components/sponsors/SponsorInsightsCard'; import { getImageService } from '@/lib/di-container'; import { TeamMembersPresenter } from '@/lib/presenters/TeamMembersPresenter'; import TeamRoster from '@/components/teams/TeamRoster'; import TeamStandings from '@/components/teams/TeamStandings'; import TeamAdmin from '@/components/teams/TeamAdmin'; import JoinTeamButton from '@/components/teams/JoinTeamButton'; import { getGetTeamDetailsUseCase, getGetTeamMembersUseCase, getTeamMembershipRepository, } from '@/lib/di-container'; import { useEffectiveDriverId } from '@/lib/currentDriver'; import type { Team } from '@gridpilot/racing'; import { Users, Trophy, TrendingUp, Star, Zap } from 'lucide-react'; type TeamRole = 'owner' | 'manager' | 'driver'; interface TeamMembership { driverId: string; role: TeamRole; joinedAt: Date; } type Tab = 'overview' | 'roster' | 'standings' | 'admin'; export default function TeamDetailPage() { const params = useParams(); const teamId = params.id as string; const [team, setTeam] = useState(null); const [memberships, setMemberships] = useState([]); const [activeTab, setActiveTab] = useState('overview'); const [loading, setLoading] = useState(true); const [isAdmin, setIsAdmin] = useState(false); const currentDriverId = useEffectiveDriverId(); const isSponsorMode = useSponsorMode(); const loadTeamData = useCallback(async () => { setLoading(true); try { const detailsUseCase = getGetTeamDetailsUseCase(); const membersUseCase = getGetTeamMembersUseCase(); await detailsUseCase.execute(teamId, currentDriverId); const detailsPresenter = detailsUseCase.presenter; const detailsViewModel = detailsPresenter ? (detailsPresenter as any).getViewModel?.() as { team: Team } | null : null; if (!detailsViewModel) { setTeam(null); setMemberships([]); setIsAdmin(false); return; } const teamMembersPresenter = new TeamMembersPresenter(); await membersUseCase.execute({ teamId }, teamMembersPresenter); const membersViewModel = teamMembersPresenter.getViewModel(); const teamMemberships: TeamMembership[] = (membersViewModel?.members ?? []).map((m) => ({ driverId: m.driverId, role: m.role as TeamRole, joinedAt: new Date(m.joinedAt), })); const adminStatus = teamMemberships.some( (m: TeamMembership) => m.driverId === currentDriverId && (m.role === 'owner' || m.role === 'manager'), ) ?? false; setTeam(detailsViewModel.team); setMemberships(teamMemberships); setIsAdmin(adminStatus); } finally { setLoading(false); } }, [teamId, currentDriverId]); useEffect(() => { void loadTeamData(); }, [loadTeamData]); const handleUpdate = () => { loadTeamData(); }; const handleRemoveMember = async (driverId: string) => { if (!confirm('Are you sure you want to remove this member?')) { return; } try { const membershipRepo = getTeamMembershipRepository(); const performer = await membershipRepo.getMembership(teamId, currentDriverId); if (!performer || (performer.role !== 'owner' && performer.role !== 'manager')) { throw new Error('Only owners or managers can remove members'); } const membership = await membershipRepo.getMembership(teamId, driverId); if (!membership) { throw new Error('Member not found'); } if (membership.role === 'owner') { throw new Error('Cannot remove the team owner'); } await membershipRepo.removeMembership(teamId, driverId); handleUpdate(); } catch (error) { alert(error instanceof Error ? error.message : 'Failed to remove member'); } }; const handleChangeRole = async (driverId: string, newRole: TeamRole) => { try { const membershipRepo = getTeamMembershipRepository(); const performer = await membershipRepo.getMembership(teamId, currentDriverId); if (!performer || (performer.role !== 'owner' && performer.role !== 'manager')) { throw new Error('Only owners or managers can update roles'); } const membership = await membershipRepo.getMembership(teamId, driverId); if (!membership) { throw new Error('Member not found'); } if (membership.role === 'owner') { throw new Error('Cannot change the owner role'); } await membershipRepo.saveMembership({ ...membership, role: newRole, }); handleUpdate(); } catch (error) { alert(error instanceof Error ? error.message : 'Failed to change role'); } }; if (loading) { return (
Loading team...
); } if (!team) { return (

Team Not Found

The team you're looking for doesn't exist or has been disbanded.

); } const tabs: { id: Tab; label: string; visible: boolean }[] = [ { id: 'overview', label: 'Overview', visible: true }, { id: 'roster', label: 'Roster', visible: true }, { id: 'standings', label: 'Standings', visible: true }, { id: 'admin', label: 'Admin', visible: isAdmin }, ]; const visibleTabs = tabs.filter(tab => tab.visible); // Build sponsor insights for team const teamMetrics = [ MetricBuilders.members(memberships.length), MetricBuilders.reach(memberships.length * 15), MetricBuilders.races(team.leagues.length * 8), MetricBuilders.engagement(82), ]; return (
{/* Breadcrumb */} {/* Sponsor Insights Card - Consistent placement at top */} {isSponsorMode && team && ( )}
{team.name}

{team.name}

{team.tag}

{team.description}

{memberships.length} {memberships.length === 1 ? 'member' : 'members'} Created {new Date(team.createdAt).toLocaleDateString()} {team.leagues.length > 0 && ( <> {team.leagues.length} {team.leagues.length === 1 ? 'league' : 'leagues'} )}
{visibleTabs.map((tab) => ( ))}
{activeTab === 'overview' && (

About

{team.description}

Quick Stats

Recent Activity

No recent activity to display
)} {activeTab === 'roster' && ( )} {activeTab === 'standings' && ( )} {activeTab === 'admin' && isAdmin && ( )}
); } function StatItem({ label, value, color }: { label: string; value: string; color: string }) { return (
{label} {value}
); }