(null);
const [isMenuOpen, setIsMenuOpen] = useState(false);
const isSponsorMode = useSponsorMode();
const { isDemo, demoRole } = useDemoUserMode();
const shouldReduceMotion = useReducedMotion();
const primaryDriverId = useEffectiveDriverId();
// Load driver data only for non-demo users
useEffect(() => {
let cancelled = false;
async function loadDriver() {
if (!primaryDriverId || isDemo) {
if (!cancelled) {
setDriver(null);
}
return;
}
const dto = await driverService.findById(primaryDriverId);
if (!cancelled) {
setDriver(dto ? new DriverViewModelClass({ ...dto, avatarUrl: (dto as any).avatarUrl ?? null }) : null);
}
}
void loadDriver();
return () => {
cancelled = true;
};
}, [primaryDriverId, driverService, isDemo]);
const data = useMemo(() => {
if (!session?.user) {
return null;
}
// Demo users don't have real driver data
if (isDemo) {
return {
isDemo: true,
demoRole,
displayName: session.user.displayName,
email: session.user.email,
avatarUrl: session.user.avatarUrl,
};
}
if (!primaryDriverId || !driver) {
return null;
}
// Driver rating + rank are not exposed by the current API contract for the lightweight
// driver DTO used in the header. Keep it null until the API provides it.
const rating: number | null = null;
const rank: number | null = null;
const avatarSrc = driver.avatarUrl;
return {
driver,
avatarSrc,
rating,
rank,
isDemo: false,
demoRole: null,
};
}, [session, driver, primaryDriverId, isDemo, demoRole]);
// Close menu when clicking outside
useEffect(() => {
const handleClickOutside = (e: MouseEvent) => {
if (isMenuOpen) {
const target = e.target as HTMLElement;
if (!target.closest('[data-user-pill]')) {
setIsMenuOpen(false);
}
}
};
document.addEventListener('click', handleClickOutside);
return () => document.removeEventListener('click', handleClickOutside);
}, [isMenuOpen]);
// Logout handler for demo users
const handleLogout = async () => {
try {
// Call the logout API
await fetch('/api/auth/logout', { method: 'POST' });
// Clear any demo mode cookies
document.cookie = 'gridpilot_demo_mode=; path=/; expires=Thu, 01 Jan 1970 00:00:00 GMT';
// Redirect to home
window.location.href = '/';
} catch (error) {
console.error('Logout failed:', error);
window.location.href = '/';
}
};
// Demo user UI
if (isDemo && data?.isDemo) {
const roleLabel = {
'driver': 'Driver',
'sponsor': 'Sponsor',
'league-owner': 'League Owner',
'league-steward': 'League Steward',
'league-admin': 'League Admin',
'system-owner': 'System Owner',
'super-admin': 'Super Admin',
}[demoRole || 'driver'];
const roleColor = {
'driver': 'text-primary-blue',
'sponsor': 'text-performance-green',
'league-owner': 'text-purple-400',
'league-steward': 'text-amber-400',
'league-admin': 'text-red-400',
'system-owner': 'text-indigo-400',
'super-admin': 'text-pink-400',
}[demoRole || 'driver'];
return (
setIsMenuOpen((open) => !open)}
className="group flex items-center gap-3 rounded-full bg-gradient-to-r from-iron-gray to-deep-graphite border border-charcoal-outline px-3 py-1.5 hover:border-primary-blue/50 transition-all duration-200"
whileHover={{ scale: 1.02 }}
whileTap={{ scale: 0.98 }}
>
{/* Avatar */}
{data.avatarUrl ? (
) : (
DEMO
)}
{/* Info */}
{data.displayName}
{roleLabel}
{/* Chevron */}
{isMenuOpen && (
{/* Header */}
{data.avatarUrl ? (
) : (
DEMO
)}
{data.displayName}
{roleLabel}
Development account - not for production use
{/* Menu Items */}
Demo users have limited profile access
{/* Footer */}
Logout
)}
);
}
// Sponsor mode UI
if (isSponsorMode) {
return (
setIsMenuOpen((open) => !open)} />
{isMenuOpen && (
{/* Header */}
Acme Racing Co.
Sponsor Account
{/* Quick stats */}
{/* Menu Items */}
Sponsor portal is currently unavailable.
}
comingSoon={
Sponsor portal is coming soon.
}
>
setIsMenuOpen(false)}
>
Dashboard
setIsMenuOpen(false)}
>
My Sponsorships
setIsMenuOpen(false)}
>
Billing
setIsMenuOpen(false)}
>
Settings
{/* Footer */}
{
document.cookie = 'gridpilot_demo_mode=; path=/; expires=Thu, 01 Jan 1970 00:00:00 GMT';
window.location.reload();
}}
className="flex w-full items-center justify-between px-4 py-3 text-sm text-gray-500 hover:text-racing-red hover:bg-racing-red/5 transition-colors"
>
Exit Sponsor Mode
)}
);
}
if (!session) {
return (
Sign In
Get Started
);
}
if (!data || data.isDemo) {
return null;
}
// Type guard to ensure data has the required properties for regular driver
if (!data.driver || data.rating === undefined || data.rank === undefined) {
return null;
}
return (
setIsMenuOpen((open) => !open)}
/>
{isMenuOpen && (
setIsMenuOpen(false)}
>
Profile
setIsMenuOpen(false)}
>
Manage leagues
setIsMenuOpen(false)}
>
Liveries
setIsMenuOpen(false)}
>
Sponsorship Requests
setIsMenuOpen(false)}
>
Settings
)}
);
}