Onboarding
This commit is contained in:
@@ -1,278 +1,367 @@
|
||||
import React, { useEffect, useRef } from 'react';
|
||||
import {
|
||||
View,
|
||||
Text,
|
||||
StyleSheet,
|
||||
TouchableOpacity,
|
||||
Animated,
|
||||
Dimensions,
|
||||
Image,
|
||||
} from 'react-native';
|
||||
import { Ionicons } from '@expo/vector-icons';
|
||||
import { router } from 'expo-router';
|
||||
import { useApp } from '../context/AppContext';
|
||||
import { useColors } from '../constants/Colors';
|
||||
import { ThemeBackdrop } from '../components/ThemeBackdrop';
|
||||
|
||||
const { height: SCREEN_H, width: SCREEN_W } = Dimensions.get('window');
|
||||
|
||||
export default function OnboardingScreen() {
|
||||
const { isDarkMode, colorPalette, t } = useApp();
|
||||
const colors = useColors(isDarkMode, colorPalette);
|
||||
|
||||
const FEATURES = [
|
||||
{ icon: 'camera-outline' as const, label: t.onboardingFeatureScan },
|
||||
{ icon: 'notifications-outline' as const, label: t.onboardingFeatureReminder },
|
||||
{ icon: 'book-outline' as const, label: t.onboardingFeatureLexicon },
|
||||
];
|
||||
|
||||
// Entrance animations
|
||||
const logoAnim = useRef(new Animated.Value(0)).current;
|
||||
const logoScale = useRef(new Animated.Value(0.85)).current;
|
||||
const featuresAnim = useRef(new Animated.Value(0)).current;
|
||||
const buttonsAnim = useRef(new Animated.Value(0)).current;
|
||||
const featureAnims = useRef(FEATURES.map(() => new Animated.Value(0))).current;
|
||||
|
||||
useEffect(() => {
|
||||
Animated.sequence([
|
||||
Animated.parallel([
|
||||
Animated.timing(logoAnim, { toValue: 1, duration: 700, useNativeDriver: true }),
|
||||
Animated.spring(logoScale, { toValue: 1, tension: 50, friction: 8, useNativeDriver: true }),
|
||||
]),
|
||||
Animated.stagger(100, featureAnims.map(anim =>
|
||||
Animated.timing(anim, { toValue: 1, duration: 400, useNativeDriver: true })
|
||||
)),
|
||||
Animated.timing(buttonsAnim, { toValue: 1, duration: 400, useNativeDriver: true }),
|
||||
]).start();
|
||||
}, []);
|
||||
|
||||
return (
|
||||
<View style={[styles.container, { backgroundColor: colors.background }]}>
|
||||
<ThemeBackdrop colors={colors} />
|
||||
|
||||
{/* Logo-Bereich */}
|
||||
<Animated.View
|
||||
style={[
|
||||
styles.heroSection,
|
||||
{ opacity: logoAnim, transform: [{ scale: logoScale }] },
|
||||
]}
|
||||
>
|
||||
<View style={[styles.iconContainer, { shadowColor: colors.primary }]}>
|
||||
<Image
|
||||
source={require('../assets/icon.png')}
|
||||
style={styles.appIcon}
|
||||
resizeMode="cover"
|
||||
/>
|
||||
</View>
|
||||
|
||||
<Text style={[styles.appName, { color: colors.text }]}>GreenLens</Text>
|
||||
<Text style={[styles.tagline, { color: colors.textSecondary }]}>
|
||||
{t.onboardingTagline}
|
||||
</Text>
|
||||
</Animated.View>
|
||||
|
||||
{/* Feature-Liste */}
|
||||
<View style={styles.featuresSection}>
|
||||
{FEATURES.map((feat, i) => (
|
||||
<Animated.View
|
||||
key={feat.label}
|
||||
style={[
|
||||
styles.featureRow,
|
||||
{
|
||||
backgroundColor: colors.surface + '88', // Semi-transparent for backdrop effect
|
||||
borderColor: colors.border,
|
||||
opacity: featureAnims[i],
|
||||
transform: [{
|
||||
translateY: featureAnims[i].interpolate({
|
||||
inputRange: [0, 1],
|
||||
outputRange: [20, 0],
|
||||
}),
|
||||
}],
|
||||
},
|
||||
]}
|
||||
>
|
||||
<View style={[styles.featureIcon, { backgroundColor: colors.primary + '15' }]}>
|
||||
<Ionicons name={feat.icon} size={18} color={colors.primary} />
|
||||
</View>
|
||||
<Text style={[styles.featureText, { color: colors.text }]}>{feat.label}</Text>
|
||||
</Animated.View>
|
||||
))}
|
||||
</View>
|
||||
|
||||
{/* Buttons */}
|
||||
<Animated.View style={[styles.buttonsSection, { opacity: buttonsAnim }]}>
|
||||
<TouchableOpacity
|
||||
style={[styles.primaryBtn, { backgroundColor: colors.primary }]}
|
||||
onPress={() => router.push('/scanner')}
|
||||
activeOpacity={0.85}
|
||||
>
|
||||
<Ionicons name="scan" size={20} color={colors.onPrimary} />
|
||||
<Text style={[styles.primaryBtnText, { color: colors.onPrimary }]}>
|
||||
{t.onboardingScanBtn}
|
||||
</Text>
|
||||
</TouchableOpacity>
|
||||
|
||||
<View style={styles.authActions}>
|
||||
<TouchableOpacity
|
||||
style={[styles.secondaryBtn, { borderColor: colors.primary, backgroundColor: colors.surface }]}
|
||||
onPress={() => router.push('/auth/signup')}
|
||||
activeOpacity={0.82}
|
||||
>
|
||||
<Text style={[styles.secondaryBtnText, { color: colors.primary }]}>
|
||||
{t.onboardingRegister}
|
||||
</Text>
|
||||
</TouchableOpacity>
|
||||
|
||||
<TouchableOpacity
|
||||
style={[styles.secondaryBtn, { borderColor: colors.borderStrong, backgroundColor: colors.surface }]}
|
||||
onPress={() => router.push('/auth/login')}
|
||||
activeOpacity={0.82}
|
||||
>
|
||||
<Text style={[styles.secondaryBtnText, { color: colors.text }]}>
|
||||
{t.onboardingLogin}
|
||||
</Text>
|
||||
</TouchableOpacity>
|
||||
</View>
|
||||
|
||||
<TouchableOpacity
|
||||
style={[styles.plansBtn, { borderColor: colors.primary }]}
|
||||
onPress={() => router.push('/profile/billing')}
|
||||
activeOpacity={0.82}
|
||||
>
|
||||
<Ionicons name="pricetag-outline" size={16} color={colors.primary} />
|
||||
<Text style={[styles.plansBtnText, { color: colors.primary }]}>
|
||||
View Subscription Plans & Pricing
|
||||
</Text>
|
||||
</TouchableOpacity>
|
||||
|
||||
<Text style={[styles.disclaimer, { color: colors.textMuted }]}>
|
||||
{t.onboardingDisclaimer}
|
||||
</Text>
|
||||
</Animated.View>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
container: {
|
||||
flex: 1,
|
||||
paddingHorizontal: 32,
|
||||
paddingTop: SCREEN_H * 0.12,
|
||||
paddingBottom: 40,
|
||||
},
|
||||
heroSection: {
|
||||
alignItems: 'center',
|
||||
marginBottom: 40,
|
||||
},
|
||||
iconContainer: {
|
||||
width: 120,
|
||||
height: 120,
|
||||
borderRadius: 28,
|
||||
backgroundColor: '#fff',
|
||||
elevation: 8,
|
||||
shadowOffset: { width: 0, height: 4 },
|
||||
shadowOpacity: 0.15,
|
||||
shadowRadius: 12,
|
||||
marginBottom: 24,
|
||||
overflow: 'hidden',
|
||||
},
|
||||
appIcon: {
|
||||
width: '100%',
|
||||
height: '100%',
|
||||
},
|
||||
appName: {
|
||||
fontSize: 40,
|
||||
fontWeight: '900',
|
||||
letterSpacing: -1.5,
|
||||
marginBottom: 4,
|
||||
},
|
||||
tagline: {
|
||||
fontSize: 17,
|
||||
fontWeight: '500',
|
||||
opacity: 0.8,
|
||||
},
|
||||
featuresSection: {
|
||||
gap: 8,
|
||||
flex: 1,
|
||||
justifyContent: 'center',
|
||||
},
|
||||
featureRow: {
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
gap: 12,
|
||||
paddingHorizontal: 16,
|
||||
paddingVertical: 10,
|
||||
borderRadius: 16,
|
||||
borderWidth: 1,
|
||||
},
|
||||
featureIcon: {
|
||||
width: 36,
|
||||
height: 36,
|
||||
borderRadius: 10,
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
},
|
||||
featureText: {
|
||||
flex: 1,
|
||||
fontSize: 13,
|
||||
fontWeight: '600',
|
||||
letterSpacing: 0.1,
|
||||
},
|
||||
buttonsSection: {
|
||||
gap: 16,
|
||||
marginTop: 20,
|
||||
},
|
||||
primaryBtn: {
|
||||
height: 58,
|
||||
borderRadius: 20,
|
||||
flexDirection: 'row',
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
gap: 12,
|
||||
elevation: 4,
|
||||
shadowColor: '#000',
|
||||
shadowOffset: { width: 0, height: 2 },
|
||||
shadowOpacity: 0.1,
|
||||
shadowRadius: 4,
|
||||
},
|
||||
primaryBtnText: {
|
||||
fontSize: 17,
|
||||
fontWeight: '700',
|
||||
},
|
||||
authActions: {
|
||||
flexDirection: 'row',
|
||||
gap: 12,
|
||||
},
|
||||
secondaryBtn: {
|
||||
flex: 1,
|
||||
height: 54,
|
||||
borderRadius: 20,
|
||||
borderWidth: 1.5,
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
},
|
||||
secondaryBtnText: {
|
||||
fontSize: 15,
|
||||
fontWeight: '600',
|
||||
},
|
||||
plansBtn: {
|
||||
height: 48,
|
||||
borderRadius: 16,
|
||||
borderWidth: 1.5,
|
||||
flexDirection: 'row',
|
||||
justifyContent: 'center',
|
||||
alignItems: 'center',
|
||||
gap: 8,
|
||||
},
|
||||
plansBtnText: {
|
||||
fontSize: 14,
|
||||
fontWeight: '600',
|
||||
},
|
||||
disclaimer: {
|
||||
fontSize: 12,
|
||||
textAlign: 'center',
|
||||
opacity: 0.6,
|
||||
marginTop: 8,
|
||||
},
|
||||
});
|
||||
|
||||
import React from 'react';
|
||||
import {
|
||||
Image,
|
||||
ImageBackground,
|
||||
SafeAreaView,
|
||||
StyleSheet,
|
||||
Text,
|
||||
TouchableOpacity,
|
||||
View,
|
||||
useWindowDimensions,
|
||||
} from 'react-native';
|
||||
import { Ionicons } from '@expo/vector-icons';
|
||||
import { router } from 'expo-router';
|
||||
import Svg, { Path } from 'react-native-svg';
|
||||
import { useApp } from '../context/AppContext';
|
||||
|
||||
type Feature = {
|
||||
icon: keyof typeof Ionicons.glyphMap;
|
||||
title: string;
|
||||
description: string;
|
||||
};
|
||||
|
||||
export default function OnboardingScreen() {
|
||||
const { t } = useApp();
|
||||
const { height, width } = useWindowDimensions();
|
||||
const compact = height < 760;
|
||||
const sheetTop = compact ? 142 : 156;
|
||||
const waveHeight = compact ? 148 : 170;
|
||||
const bodyOffset = waveHeight - 2;
|
||||
const contentTop = compact ? 94 : 108;
|
||||
|
||||
const features: Feature[] = [
|
||||
{
|
||||
icon: 'scan-outline',
|
||||
title: t.welcomeFeatureIdentifyTitle,
|
||||
description: t.welcomeFeatureIdentifyDesc,
|
||||
},
|
||||
{
|
||||
icon: 'notifications-outline',
|
||||
title: t.welcomeFeatureReminderTitle,
|
||||
description: t.welcomeFeatureReminderDesc,
|
||||
},
|
||||
{
|
||||
icon: 'book-outline',
|
||||
title: t.welcomeFeatureLibraryTitle,
|
||||
description: t.welcomeFeatureLibraryDesc,
|
||||
},
|
||||
];
|
||||
|
||||
return (
|
||||
<View style={styles.container}>
|
||||
<ImageBackground
|
||||
source={require('../assets/welcome_botanical_hero.png')}
|
||||
style={styles.heroImage}
|
||||
imageStyle={styles.heroImageContent}
|
||||
resizeMode="cover"
|
||||
>
|
||||
<View style={styles.heroShadeTop} />
|
||||
<View style={styles.heroShadeBottom} />
|
||||
<SafeAreaView style={styles.safeArea}>
|
||||
<View style={[styles.brandRow, compact && styles.brandRowCompact]}>
|
||||
<Image
|
||||
source={require('../assets/icon.png')}
|
||||
style={styles.logo}
|
||||
resizeMode="cover"
|
||||
/>
|
||||
<Text style={styles.brandName}>
|
||||
Green<Text style={styles.brandAccent}>Lens</Text>
|
||||
</Text>
|
||||
</View>
|
||||
</SafeAreaView>
|
||||
</ImageBackground>
|
||||
|
||||
<View style={[styles.sheet, { top: sheetTop }]}>
|
||||
<Svg
|
||||
width={width}
|
||||
height={waveHeight}
|
||||
viewBox={`0 0 ${width} ${waveHeight}`}
|
||||
preserveAspectRatio="none"
|
||||
style={styles.sheetWave}
|
||||
>
|
||||
<Path
|
||||
d={`M0 34 C ${width * 0.08} 76 ${width * 0.14} 82 ${width * 0.24} 82 C ${width * 0.38} 82 ${width * 0.52} 82 ${width * 0.64} 82 C ${width * 0.78} 86 ${width * 0.88} 132 ${width} 156 L ${width} ${waveHeight} L 0 ${waveHeight} Z`}
|
||||
fill="#fbfaf3"
|
||||
/>
|
||||
</Svg>
|
||||
<View style={[styles.sheetBody, { top: bodyOffset }]} />
|
||||
<View
|
||||
style={[
|
||||
styles.sheetContent,
|
||||
{ top: contentTop },
|
||||
compact && styles.sheetContentCompact,
|
||||
]}
|
||||
>
|
||||
<Text style={[styles.headline, compact && styles.headlineCompact]}>
|
||||
{t.welcomeHeadline}
|
||||
</Text>
|
||||
<Text style={styles.subheadline}>{t.welcomeSubheadline}</Text>
|
||||
|
||||
<View style={styles.features}>
|
||||
{features.map((feature, index) => (
|
||||
<View
|
||||
key={feature.title}
|
||||
style={[
|
||||
styles.featureRow,
|
||||
index === features.length - 1 && styles.featureRowLast,
|
||||
]}
|
||||
>
|
||||
<View style={styles.featureIcon}>
|
||||
<Ionicons name={feature.icon} size={22} color="#a6d66f" />
|
||||
</View>
|
||||
<View style={styles.featureCopy}>
|
||||
<Text style={styles.featureTitle}>{feature.title}</Text>
|
||||
<Text style={styles.featureDescription}>{feature.description}</Text>
|
||||
</View>
|
||||
</View>
|
||||
))}
|
||||
</View>
|
||||
|
||||
<TouchableOpacity
|
||||
style={styles.demoButton}
|
||||
onPress={() => router.push('/scanner')}
|
||||
activeOpacity={0.86}
|
||||
>
|
||||
<Ionicons name="scan" size={25} color="#f8f7ef" />
|
||||
<Text style={styles.demoButtonText}>{t.welcomeDemoScan}</Text>
|
||||
<Ionicons name="chevron-forward" size={26} color="#f8f7ef" />
|
||||
</TouchableOpacity>
|
||||
|
||||
<View style={styles.authRow}>
|
||||
<TouchableOpacity
|
||||
style={styles.authButton}
|
||||
onPress={() => router.push('/auth/signup')}
|
||||
activeOpacity={0.82}
|
||||
>
|
||||
<Text style={styles.authButtonText}>{t.onboardingRegister}</Text>
|
||||
</TouchableOpacity>
|
||||
|
||||
<TouchableOpacity
|
||||
style={[styles.authButton, styles.loginButton]}
|
||||
onPress={() => router.push('/auth/login')}
|
||||
activeOpacity={0.82}
|
||||
>
|
||||
<Text style={[styles.authButtonText, styles.loginButtonText]}>
|
||||
{t.onboardingLogin}
|
||||
</Text>
|
||||
</TouchableOpacity>
|
||||
</View>
|
||||
|
||||
<TouchableOpacity
|
||||
style={styles.subscriptionLink}
|
||||
onPress={() => router.push('/profile/billing')}
|
||||
activeOpacity={0.8}
|
||||
>
|
||||
<Ionicons name="leaf-outline" size={21} color="#4b7c31" />
|
||||
<Text style={styles.subscriptionText}>{t.welcomeSubscriptionPlans}</Text>
|
||||
<Ionicons name="chevron-forward" size={20} color="#4b7c31" />
|
||||
</TouchableOpacity>
|
||||
|
||||
<Text style={styles.legalText}>{t.welcomeLegal}</Text>
|
||||
</View>
|
||||
</View>
|
||||
</View>
|
||||
);
|
||||
}
|
||||
|
||||
const styles = StyleSheet.create({
|
||||
container: {
|
||||
flex: 1,
|
||||
backgroundColor: '#0a110b',
|
||||
},
|
||||
heroImage: {
|
||||
height: '60%',
|
||||
minHeight: 430,
|
||||
},
|
||||
heroImageContent: {
|
||||
backgroundColor: '#0a110b',
|
||||
transform: [{ scale: 1.04 }],
|
||||
},
|
||||
heroShadeTop: {
|
||||
...StyleSheet.absoluteFillObject,
|
||||
backgroundColor: 'rgba(0,0,0,0.08)',
|
||||
},
|
||||
heroShadeBottom: {
|
||||
position: 'absolute',
|
||||
left: 0,
|
||||
right: 0,
|
||||
bottom: 0,
|
||||
height: 190,
|
||||
backgroundColor: 'rgba(7,12,7,0.2)',
|
||||
},
|
||||
safeArea: {
|
||||
flex: 1,
|
||||
},
|
||||
brandRow: {
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
gap: 15,
|
||||
paddingHorizontal: 30,
|
||||
paddingTop: 62,
|
||||
},
|
||||
brandRowCompact: {
|
||||
paddingTop: 42,
|
||||
},
|
||||
logo: {
|
||||
width: 68,
|
||||
height: 68,
|
||||
borderRadius: 16,
|
||||
backgroundColor: '#fff',
|
||||
},
|
||||
brandName: {
|
||||
color: '#f8f7ef',
|
||||
fontSize: 36,
|
||||
fontWeight: '900',
|
||||
},
|
||||
brandAccent: {
|
||||
color: '#9bc76e',
|
||||
},
|
||||
sheet: {
|
||||
position: 'absolute',
|
||||
left: 0,
|
||||
right: 0,
|
||||
bottom: 0,
|
||||
},
|
||||
sheetWave: {
|
||||
position: 'absolute',
|
||||
left: 0,
|
||||
right: 0,
|
||||
top: 0,
|
||||
},
|
||||
sheetBody: {
|
||||
position: 'absolute',
|
||||
left: 0,
|
||||
right: 0,
|
||||
bottom: 0,
|
||||
backgroundColor: '#fbfaf3',
|
||||
},
|
||||
sheetContent: {
|
||||
position: 'absolute',
|
||||
left: 0,
|
||||
right: 0,
|
||||
bottom: 0,
|
||||
paddingHorizontal: 24,
|
||||
paddingBottom: 10,
|
||||
},
|
||||
sheetContentCompact: {
|
||||
paddingHorizontal: 22,
|
||||
paddingBottom: 8,
|
||||
},
|
||||
headline: {
|
||||
color: '#101c12',
|
||||
fontSize: 40,
|
||||
lineHeight: 43,
|
||||
fontWeight: '900',
|
||||
marginBottom: 6,
|
||||
maxWidth: 310,
|
||||
},
|
||||
headlineCompact: {
|
||||
fontSize: 34,
|
||||
lineHeight: 37,
|
||||
},
|
||||
subheadline: {
|
||||
color: '#5f625d',
|
||||
fontSize: 15,
|
||||
lineHeight: 19,
|
||||
fontWeight: '500',
|
||||
marginBottom: 11,
|
||||
},
|
||||
features: {
|
||||
marginBottom: 10,
|
||||
},
|
||||
featureRow: {
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
gap: 11,
|
||||
paddingVertical: 6,
|
||||
borderBottomWidth: StyleSheet.hairlineWidth,
|
||||
borderBottomColor: 'rgba(16,28,18,0.14)',
|
||||
},
|
||||
featureRowLast: {
|
||||
borderBottomWidth: 0,
|
||||
},
|
||||
featureIcon: {
|
||||
width: 46,
|
||||
height: 46,
|
||||
borderRadius: 10,
|
||||
alignItems: 'center',
|
||||
justifyContent: 'center',
|
||||
backgroundColor: '#173817',
|
||||
},
|
||||
featureCopy: {
|
||||
flex: 1,
|
||||
},
|
||||
featureTitle: {
|
||||
color: '#101c12',
|
||||
fontSize: 16,
|
||||
fontWeight: '800',
|
||||
marginBottom: 2,
|
||||
},
|
||||
featureDescription: {
|
||||
color: '#696b65',
|
||||
fontSize: 13,
|
||||
lineHeight: 16,
|
||||
fontWeight: '500',
|
||||
},
|
||||
demoButton: {
|
||||
height: 60,
|
||||
borderRadius: 7,
|
||||
backgroundColor: '#437824',
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
justifyContent: 'space-between',
|
||||
paddingHorizontal: 18,
|
||||
marginBottom: 8,
|
||||
},
|
||||
demoButtonText: {
|
||||
color: '#f8f7ef',
|
||||
fontSize: 21,
|
||||
fontWeight: '800',
|
||||
},
|
||||
authRow: {
|
||||
flexDirection: 'row',
|
||||
gap: 8,
|
||||
marginBottom: 8,
|
||||
},
|
||||
authButton: {
|
||||
flex: 1,
|
||||
height: 50,
|
||||
borderRadius: 7,
|
||||
borderWidth: 1.4,
|
||||
borderColor: '#4b7c31',
|
||||
alignItems: 'center',
|
||||
justifyContent: 'center',
|
||||
},
|
||||
loginButton: {
|
||||
borderColor: '#101c12',
|
||||
},
|
||||
authButtonText: {
|
||||
color: '#4b7c31',
|
||||
fontSize: 17,
|
||||
fontWeight: '700',
|
||||
},
|
||||
loginButtonText: {
|
||||
color: '#101c12',
|
||||
},
|
||||
subscriptionLink: {
|
||||
minHeight: 24,
|
||||
flexDirection: 'row',
|
||||
alignItems: 'center',
|
||||
justifyContent: 'center',
|
||||
gap: 8,
|
||||
marginBottom: 7,
|
||||
},
|
||||
subscriptionText: {
|
||||
color: '#4b7c31',
|
||||
fontSize: 15,
|
||||
fontWeight: '800',
|
||||
textAlign: 'center',
|
||||
},
|
||||
legalText: {
|
||||
color: '#6b6d68',
|
||||
fontSize: 11,
|
||||
lineHeight: 14,
|
||||
fontWeight: '500',
|
||||
textAlign: 'center',
|
||||
},
|
||||
});
|
||||
|
||||
Reference in New Issue
Block a user