Files
gridpilot.gg/apps/website/components/onboarding/OnboardingWizard.tsx
2026-01-14 10:51:05 +01:00

229 lines
6.8 KiB
TypeScript

import { useState, FormEvent } from 'react';
import Card from '@/components/ui/Card';
import { StepIndicator } from '@/ui/StepIndicator';
import { PersonalInfoStep, PersonalInfo } from '@/ui/onboarding/PersonalInfoStep';
import { AvatarStep, AvatarInfo } from './AvatarStep';
import { OnboardingHeader } from '@/ui/onboarding/OnboardingHeader';
import { OnboardingHelpText } from '@/ui/onboarding/OnboardingHelpText';
import { OnboardingError } from '@/ui/onboarding/OnboardingError';
import { OnboardingNavigation } from '@/ui/onboarding/OnboardingNavigation';
import { OnboardingContainer } from '@/ui/onboarding/OnboardingContainer';
import { OnboardingCardAccent } from '@/ui/onboarding/OnboardingCardAccent';
import { OnboardingForm } from '@/ui/onboarding/OnboardingForm';
type OnboardingStep = 1 | 2;
interface FormErrors {
[key: string]: string | undefined;
firstName?: string;
lastName?: string;
displayName?: string;
country?: string;
facePhoto?: string;
avatar?: string;
submit?: string;
}
interface OnboardingWizardProps {
onCompleted: () => void;
onCompleteOnboarding: (data: {
firstName: string;
lastName: string;
displayName: string;
country: string;
timezone?: string;
}) => Promise<{ success: boolean; error?: string }>;
onGenerateAvatars: (params: {
facePhotoData: string;
suitColor: string;
}) => Promise<{ success: boolean; data?: { success: boolean; avatarUrls?: string[]; errorMessage?: string }; error?: string }>;
}
export function OnboardingWizard({ onCompleted, onCompleteOnboarding, onGenerateAvatars }: OnboardingWizardProps) {
const [step, setStep] = useState<OnboardingStep>(1);
const [errors, setErrors] = useState<FormErrors>({});
// Form state
const [personalInfo, setPersonalInfo] = useState<PersonalInfo>({
firstName: '',
lastName: '',
displayName: '',
country: '',
timezone: '',
});
const [avatarInfo, setAvatarInfo] = useState<AvatarInfo>({
facePhoto: null,
suitColor: 'blue',
generatedAvatars: [],
selectedAvatarIndex: null,
isGenerating: false,
isValidating: false,
});
// Validation
const validateStep = (currentStep: OnboardingStep): boolean => {
const newErrors: FormErrors = {};
if (currentStep === 1) {
if (!personalInfo.firstName.trim()) {
newErrors.firstName = 'First name is required';
}
if (!personalInfo.lastName.trim()) {
newErrors.lastName = 'Last name is required';
}
if (!personalInfo.displayName.trim()) {
newErrors.displayName = 'Display name is required';
} else if (personalInfo.displayName.length < 3) {
newErrors.displayName = 'Display name must be at least 3 characters';
}
if (!personalInfo.country) {
newErrors.country = 'Please select your country';
}
}
if (currentStep === 2) {
if (!avatarInfo.facePhoto) {
newErrors.facePhoto = 'Please upload a photo of your face';
}
if (avatarInfo.generatedAvatars.length > 0 && avatarInfo.selectedAvatarIndex === null) {
newErrors.avatar = 'Please select one of the generated avatars';
}
}
setErrors(newErrors);
return Object.keys(newErrors).length === 0;
};
const handleNext = () => {
const isValid = validateStep(step);
if (isValid && step < 2) {
setStep((step + 1) as OnboardingStep);
}
};
const handleBack = () => {
if (step > 1) {
setStep((step - 1) as OnboardingStep);
}
};
const generateAvatars = async () => {
if (!avatarInfo.facePhoto) {
setErrors({ ...errors, facePhoto: 'Please upload a photo first' });
return;
}
setAvatarInfo(prev => ({ ...prev, isGenerating: true, generatedAvatars: [], selectedAvatarIndex: null }));
const newErrors = { ...errors };
delete newErrors.avatar;
setErrors(newErrors);
try {
const result = await onGenerateAvatars({
facePhotoData: avatarInfo.facePhoto,
suitColor: avatarInfo.suitColor,
});
if (result.success && result.data?.success && result.data.avatarUrls) {
setAvatarInfo(prev => ({
...prev,
generatedAvatars: result.data!.avatarUrls!,
isGenerating: false,
}));
} else {
setErrors(prev => ({ ...prev, avatar: result.data?.errorMessage || result.error || 'Failed to generate avatars' }));
setAvatarInfo(prev => ({ ...prev, isGenerating: false }));
}
} catch (error) {
setErrors(prev => ({ ...prev, avatar: 'Failed to generate avatars. Please try again.' }));
setAvatarInfo(prev => ({ ...prev, isGenerating: false }));
}
};
const handleSubmit = async (e: FormEvent) => {
e.preventDefault();
// Validate step 2 - must have selected an avatar
if (!validateStep(2)) {
return;
}
if (avatarInfo.selectedAvatarIndex === null) {
setErrors({ ...errors, avatar: 'Please select an avatar' });
return;
}
setErrors({});
try {
const result = await onCompleteOnboarding({
firstName: personalInfo.firstName.trim(),
lastName: personalInfo.lastName.trim(),
displayName: personalInfo.displayName.trim(),
country: personalInfo.country,
timezone: personalInfo.timezone || undefined,
});
if (result.success) {
onCompleted();
} else {
setErrors({ submit: result.error || 'Failed to create profile' });
}
} catch (error) {
setErrors({ submit: 'Failed to create profile' });
}
};
// Loading state comes from the mutations
const loading = false; // This would be managed by the parent component
return (
<OnboardingContainer>
<OnboardingHeader
title="Welcome to GridPilot"
subtitle="Let us set up your racing profile"
emoji="🏁"
/>
<StepIndicator currentStep={step} />
<Card className="relative overflow-hidden">
<OnboardingCardAccent />
<OnboardingForm onSubmit={handleSubmit}>
{step === 1 && (
<PersonalInfoStep
personalInfo={personalInfo}
setPersonalInfo={setPersonalInfo}
errors={errors}
loading={loading}
/>
)}
{step === 2 && (
<AvatarStep
avatarInfo={avatarInfo}
setAvatarInfo={setAvatarInfo}
errors={errors}
setErrors={setErrors}
onGenerateAvatars={generateAvatars}
/>
)}
{errors.submit && <OnboardingError message={errors.submit} />}
<OnboardingNavigation
onBack={handleBack}
onNext={step < 2 ? handleNext : undefined}
isLastStep={step === 2}
canSubmit={avatarInfo.selectedAvatarIndex !== null}
loading={loading}
/>
</OnboardingForm>
</Card>
<OnboardingHelpText />
</OnboardingContainer>
);
}