First commit of the new app

This commit is contained in:
2026-05-26 09:18:37 +01:00
parent 295d1bda21
commit b427fb0f85
110 changed files with 6483 additions and 833 deletions

5
app/(auth)/_layout.tsx Normal file
View File

@@ -0,0 +1,5 @@
import { Stack } from 'expo-router';
export default function AuthLayout() {
return <Stack screenOptions={{ headerShown: false }} />;
}

110
app/(auth)/login/index.tsx Normal file
View File

@@ -0,0 +1,110 @@
import { Ionicons } from '@expo/vector-icons';
import React, { useState } from 'react';
import {
Image,
ImageBackground,
KeyboardAvoidingView,
Platform,
Pressable,
ScrollView,
Text,
TextInput,
View,
StatusBar,
} from 'react-native';
import styles from '@/styles/screens/auth/login.styles';
import { useAuth } from '@/assets/contexts/useAuth';
import { router, type Href } from 'expo-router';
export default function Login() {
const [email, setEmail] = useState('');
const [password, setPassword] = useState('');
const [showPassword, setShowPassword] = useState(false);
const [localError, setLocalError] = useState<string | null>(null);
const { login, isLoading, error } = useAuth();
const handleLogin = async () => {
if (!email.trim() || !password.trim()) {
setLocalError('Preenche email e palavra-passe.');
return;
}
setLocalError(null);
try {
await login(email.trim(), password);
router.replace('/home' as Href);
} catch {
// O erro já é guardado no contexto; evitamos uncaught promise no ecrã.
}
};
return (
<KeyboardAvoidingView
style={styles.container}
behavior={Platform.OS === 'ios' ? 'padding' : 'height'}
keyboardVerticalOffset={Platform.OS === 'ios' ? 0 : 20}>
<StatusBar barStyle="light-content" />
<ScrollView
contentContainerStyle={styles.scrollContent}
keyboardShouldPersistTaps="handled"
showsVerticalScrollIndicator={false}
bounces={false}>
<ImageBackground
source={require('@/assets/images/banner-login.png')}
style={styles.hero}
imageStyle={styles.heroImage}>
<View style={styles.overlay} />
<View style={styles.heroContent}>
<Image source={require('@/assets/icons/logotipo-branco.png')} style={styles.logo} resizeMode="contain" />
<Text style={styles.title}>Login</Text>
<Text style={styles.subtitle}>Acede às tuas reservas, documentos e informaçoes de viagem.</Text>
</View>
</ImageBackground>
<View style={styles.formCard}>
<Text style={styles.label}>
Email<Text style={styles.required}>*</Text>
</Text>
<TextInput
value={email}
onChangeText={setEmail}
placeholder="hello@domain.pt"
placeholderTextColor="#9AA0A6"
autoCapitalize="none"
keyboardType="email-address"
style={styles.input}
/>
<Text style={styles.label}>
Palavra-passe<Text style={styles.required}>*</Text>
</Text>
<View style={styles.passwordWrapper}>
<TextInput
value={password}
onChangeText={setPassword}
placeholder="Palavra-passe"
placeholderTextColor="#9AA0A6"
secureTextEntry={!showPassword}
style={styles.passwordInput}
/>
<Pressable onPress={() => setShowPassword((prev) => !prev)} hitSlop={8}>
<Ionicons name={showPassword ? 'eye-off-outline' : 'eye-outline'} size={20} style={styles.eyeIcon} />
</Pressable>
</View>
<Pressable onPress={() => router.push('/recover' as Href)}>
<Text style={styles.forgot}>Esqueci-me da palavra-passe</Text>
</Pressable>
{!!(localError || error) && <Text style={styles.errorText}>{localError || error}</Text>}
<Pressable style={styles.loginButton} onPress={handleLogin} disabled={isLoading}>
<Text style={styles.loginButtonText}>Entrar</Text>
<Image source={require('@/assets/icons/seta-up.png')} style={styles.loginButtonIcon} />
</Pressable>
</View>
</ScrollView>
</KeyboardAvoidingView>
);
}

View File

@@ -0,0 +1,106 @@
import { Ionicons } from '@expo/vector-icons';
import { router, type Href } from 'expo-router';
import React, { useState } from 'react';
import {
Image,
ImageBackground,
KeyboardAvoidingView,
Modal,
Platform,
Pressable,
ScrollView,
Text,
TextInput,
View,
StatusBar,
} from 'react-native';
import styles from '@/styles/screens/auth/recover.styles';
export default function Recover() {
const [email, setEmail] = useState('');
const [localError, setLocalError] = useState<string | null>(null);
const [showSuccessModal, setShowSuccessModal] = useState(false);
const handleRecover = () => {
if (!email.trim()) {
setLocalError('Indica o teu email para continuar.');
return;
}
setLocalError(null);
setShowSuccessModal(true);
};
return (
<KeyboardAvoidingView
style={styles.container}
behavior={Platform.OS === 'ios' ? 'padding' : 'height'}
keyboardVerticalOffset={Platform.OS === 'ios' ? 0 : 20}>
<StatusBar barStyle="light-content" />
<ScrollView
contentContainerStyle={styles.scrollContent}
keyboardShouldPersistTaps="handled"
showsVerticalScrollIndicator={false}
bounces={false}>
<ImageBackground
source={require('@/assets/images/banner-login.png')}
style={styles.hero}
imageStyle={styles.heroImage}>
<View style={styles.overlay} />
<View style={styles.heroContent}>
<Image source={require('@/assets/icons/logotipo-branco.png')} style={styles.logo} resizeMode="contain" />
<Text style={styles.title}>Repor palavra-passe</Text>
<Text style={styles.subtitle}>Indica o teu email para receberes o link de recuperacao.</Text>
</View>
</ImageBackground>
<View style={styles.formCard}>
<Text style={styles.label}>
Email<Text style={styles.required}>*</Text>
</Text>
<TextInput
value={email}
onChangeText={setEmail}
placeholder="hello@domain.pt"
placeholderTextColor="#9AA0A6"
autoCapitalize="none"
keyboardType="email-address"
style={styles.input}
/>
{!!localError && <Text style={styles.errorText}>{localError}</Text>}
<Pressable style={styles.actionButton} onPress={handleRecover}>
<Text style={styles.actionButtonText}>Recuperar Palavra-passe</Text>
<Image source={require('@/assets/icons/seta-up.png')} style={styles.actionButtonIcon} />
</Pressable>
<Pressable onPress={() => router.replace('/login' as Href)}>
<Text style={styles.backLink}>Voltar ao Login</Text>
</Pressable>
</View>
</ScrollView>
<Modal visible={showSuccessModal} animationType="fade" transparent>
<View style={styles.modalOverlay}>
<View style={styles.modalCard}>
<View style={styles.modalIconCircle}>
<Ionicons name="mail-open-outline" size={22} style={styles.modalIcon} />
</View>
<Text style={styles.modalTitle}>Verifica o teu email</Text>
<Text style={styles.modalMessage}>Enviamos-te as instrucoes para recuperares a palavra-passe.</Text>
<Pressable
style={styles.modalCloseButton}
onPress={() => {
setShowSuccessModal(false);
router.replace('/login' as Href);
}}>
<Text style={styles.modalCloseText}>Ok</Text>
</Pressable>
</View>
</View>
</Modal>
</KeyboardAvoidingView>
);
}