Onboarding
This commit is contained in:
@@ -1,5 +1,5 @@
|
||||
import React, { useMemo, useState } from 'react';
|
||||
import { StyleSheet, Text, TouchableOpacity, View } from 'react-native';
|
||||
import { ImageBackground, StyleSheet, Text, TouchableOpacity, View } from 'react-native';
|
||||
import { SafeAreaView } from 'react-native-safe-area-context';
|
||||
import { useRouter } from 'expo-router';
|
||||
import { Ionicons } from '@expo/vector-icons';
|
||||
@@ -9,18 +9,61 @@ import { useColors } from '../../constants/Colors';
|
||||
import { useApp } from '../../context/AppContext';
|
||||
import { OnboardingProgressService } from '../../services/onboardingProgressService';
|
||||
|
||||
const ONBOARDING_BACKGROUND = {
|
||||
light: '#fbfaf3',
|
||||
dark: '#0a110b',
|
||||
};
|
||||
|
||||
const EXPERIENCE_OPTIONS = [
|
||||
{ id: 'beginner', icon: 'leaf-outline' as const },
|
||||
{ id: 'intermediate', icon: 'sunny-outline' as const },
|
||||
{ id: 'advanced', icon: 'flask-outline' as const },
|
||||
];
|
||||
|
||||
const getExperienceScreenCopy = (language: 'de' | 'en' | 'es') => {
|
||||
if (language === 'de') {
|
||||
return {
|
||||
step: 'Schritt 3 von 4',
|
||||
heroBadge: 'Pflege-Tiefe',
|
||||
subtitles: {
|
||||
beginner: 'Klare Sprache, sichere Defaults, weniger Fachbegriffe.',
|
||||
intermediate: 'Praktische Schritte mit genug Kontext.',
|
||||
advanced: 'Mehr botanische Details und engere Diagnose.',
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
if (language === 'es') {
|
||||
return {
|
||||
step: 'Paso 3 de 4',
|
||||
heroBadge: 'Nivel de cuidado',
|
||||
subtitles: {
|
||||
beginner: 'Lenguaje claro y recomendaciones seguras.',
|
||||
intermediate: 'Pasos practicos con suficiente contexto.',
|
||||
advanced: 'Mas detalle botanico y diagnostico preciso.',
|
||||
},
|
||||
};
|
||||
}
|
||||
|
||||
return {
|
||||
step: 'Step 3 of 4',
|
||||
heroBadge: 'Care depth',
|
||||
subtitles: {
|
||||
beginner: 'Clear language, fewer assumptions, safer defaults.',
|
||||
intermediate: 'Practical care steps with enough detail.',
|
||||
advanced: 'More botanical context and tighter diagnosis.',
|
||||
},
|
||||
};
|
||||
};
|
||||
|
||||
export default function OnboardingExperienceScreen() {
|
||||
const router = useRouter();
|
||||
const posthog = usePostHog();
|
||||
const { session, isDarkMode, colorPalette, t } = useApp();
|
||||
const { session, isDarkMode, colorPalette, language, t } = useApp();
|
||||
const colors = useColors(isDarkMode, colorPalette);
|
||||
const screenBackground = isDarkMode ? ONBOARDING_BACKGROUND.dark : ONBOARDING_BACKGROUND.light;
|
||||
const [selectedLevel, setSelectedLevel] = useState<string | null>(null);
|
||||
const copy = getExperienceScreenCopy(language);
|
||||
|
||||
const levelLabels = useMemo(
|
||||
() => ({
|
||||
@@ -39,17 +82,29 @@ export default function OnboardingExperienceScreen() {
|
||||
posthog.capture('onboarding_experience_completed', {
|
||||
experience_level: level ?? 'skipped',
|
||||
});
|
||||
router.replace('/(tabs)');
|
||||
router.replace('/onboarding/health-check');
|
||||
};
|
||||
|
||||
return (
|
||||
<View style={[styles.container, { backgroundColor: colors.background }]}>
|
||||
<ThemeBackdrop colors={colors} />
|
||||
<View style={[styles.container, { backgroundColor: screenBackground }]}>
|
||||
{isDarkMode ? <ThemeBackdrop colors={colors} /> : null}
|
||||
<SafeAreaView style={styles.safeArea} edges={['top', 'left', 'right', 'bottom']}>
|
||||
<View style={styles.header}>
|
||||
<View style={[styles.headerIcon, { backgroundColor: colors.primarySoft }]}>
|
||||
<Ionicons name="sparkles-outline" size={26} color={colors.primaryDark} />
|
||||
<View style={[styles.stepPill, { backgroundColor: colors.primarySoft, borderColor: colors.border }]}>
|
||||
<Text style={[styles.stepLabel, { color: colors.primaryDark }]}>{copy.step}</Text>
|
||||
</View>
|
||||
<ImageBackground
|
||||
source={require('../../assets/onboarding_experience_mockup.png')}
|
||||
style={[styles.heroPreview, { borderColor: colors.border }]}
|
||||
imageStyle={styles.heroImage}
|
||||
resizeMode="cover"
|
||||
>
|
||||
<View style={[styles.heroOverlay, { backgroundColor: isDarkMode ? 'rgba(8, 14, 9, 0.4)' : 'rgba(251, 250, 243, 0.24)' }]} />
|
||||
<View style={[styles.heroMetric, { backgroundColor: colors.surface, borderColor: colors.border }]}>
|
||||
<Ionicons name="sparkles-outline" size={18} color={colors.primary} />
|
||||
<Text style={[styles.heroMetricText, { color: colors.text }]}>{copy.heroBadge}</Text>
|
||||
</View>
|
||||
</ImageBackground>
|
||||
<Text style={[styles.title, { color: colors.text }]}>{t.experienceOnboardingTitle}</Text>
|
||||
<Text style={[styles.subtitle, { color: colors.textSecondary }]}>{t.experienceOnboardingSubtitle}</Text>
|
||||
</View>
|
||||
@@ -73,7 +128,12 @@ export default function OnboardingExperienceScreen() {
|
||||
<View style={[styles.optionIcon, { backgroundColor: isActive ? colors.primary : colors.surfaceMuted }]}>
|
||||
<Ionicons name={option.icon} size={18} color={isActive ? colors.onPrimary : colors.textMuted} />
|
||||
</View>
|
||||
<Text style={[styles.optionLabel, { color: colors.text }]}>{levelLabels[option.id as keyof typeof levelLabels]}</Text>
|
||||
<View style={styles.optionCopy}>
|
||||
<Text style={[styles.optionLabel, { color: colors.text }]}>{levelLabels[option.id as keyof typeof levelLabels]}</Text>
|
||||
<Text style={[styles.optionSubtitle, { color: colors.textMuted }]}>
|
||||
{copy.subtitles[option.id as keyof typeof copy.subtitles]}
|
||||
</Text>
|
||||
</View>
|
||||
{isActive && <Ionicons name="checkmark-circle" size={18} color={colors.primary} />}
|
||||
</TouchableOpacity>
|
||||
);
|
||||
@@ -104,26 +164,35 @@ export default function OnboardingExperienceScreen() {
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
container: { flex: 1 },
|
||||
safeArea: { flex: 1, paddingHorizontal: 20, paddingTop: 24, paddingBottom: 20 },
|
||||
header: { alignItems: 'center', gap: 10, marginBottom: 28 },
|
||||
headerIcon: { width: 64, height: 64, borderRadius: 32, alignItems: 'center', justifyContent: 'center' },
|
||||
title: { fontSize: 28, fontWeight: '800', textAlign: 'center', lineHeight: 32 },
|
||||
subtitle: { fontSize: 14, textAlign: 'center', lineHeight: 20, maxWidth: 320 },
|
||||
options: { gap: 12, flex: 1 },
|
||||
safeArea: { flex: 1, paddingHorizontal: 20, paddingTop: 12, paddingBottom: 14 },
|
||||
header: { alignItems: 'center', gap: 9, marginBottom: 14 },
|
||||
stepPill: { borderWidth: 1, borderRadius: 999, paddingHorizontal: 12, paddingVertical: 7 },
|
||||
stepLabel: { fontSize: 12, fontWeight: '800', textTransform: 'uppercase', letterSpacing: 0.4 },
|
||||
heroPreview: { width: '100%', height: 175, borderRadius: 24, borderWidth: 1, overflow: 'hidden', justifyContent: 'flex-end', alignItems: 'flex-start' },
|
||||
heroImage: { borderRadius: 24 },
|
||||
heroOverlay: { ...StyleSheet.absoluteFillObject },
|
||||
heroMetric: { margin: 12, borderRadius: 999, borderWidth: 1, paddingHorizontal: 11, paddingVertical: 7, flexDirection: 'row', alignItems: 'center', gap: 6 },
|
||||
heroMetricText: { fontSize: 12, fontWeight: '800' },
|
||||
title: { fontSize: 25, fontWeight: '800', textAlign: 'center', lineHeight: 29 },
|
||||
subtitle: { fontSize: 13, textAlign: 'center', lineHeight: 18, maxWidth: 320 },
|
||||
options: { gap: 8, flex: 1 },
|
||||
optionCard: {
|
||||
minHeight: 64,
|
||||
borderRadius: 18,
|
||||
flex: 1,
|
||||
borderRadius: 15,
|
||||
borderWidth: 1.5,
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
paddingHorizontal: 16,
|
||||
gap: 12,
|
||||
paddingHorizontal: 14,
|
||||
paddingVertical: 16,
|
||||
gap: 10,
|
||||
},
|
||||
optionIcon: { width: 36, height: 36, borderRadius: 18, alignItems: 'center', justifyContent: 'center' },
|
||||
optionLabel: { flex: 1, fontSize: 15, fontWeight: '600' },
|
||||
footer: { flexDirection: 'row', gap: 12, marginTop: 16 },
|
||||
secondaryBtn: { flex: 1, height: 52, borderRadius: 16, borderWidth: 1.5, alignItems: 'center', justifyContent: 'center' },
|
||||
optionIcon: { width: 34, height: 34, borderRadius: 17, alignItems: 'center', justifyContent: 'center' },
|
||||
optionCopy: { flex: 1, gap: 3 },
|
||||
optionLabel: { fontSize: 14, fontWeight: '700' },
|
||||
optionSubtitle: { fontSize: 10.5, lineHeight: 14 },
|
||||
footer: { flexDirection: 'row', gap: 12, marginTop: 10 },
|
||||
secondaryBtn: { flex: 1, height: 50, borderRadius: 16, borderWidth: 1.5, alignItems: 'center', justifyContent: 'center' },
|
||||
secondaryBtnText: { fontSize: 15, fontWeight: '600' },
|
||||
primaryBtn: { flex: 1.2, height: 52, borderRadius: 16, alignItems: 'center', justifyContent: 'center' },
|
||||
primaryBtn: { flex: 1.2, height: 50, borderRadius: 16, alignItems: 'center', justifyContent: 'center' },
|
||||
primaryBtnText: { fontSize: 15, fontWeight: '700' },
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user