feat: update login, event detail, theme, and API client

- Improved event detail page with carousel, map, and layout fixes
- Updated login screen with video background and glassmorphism
- API client development mode with mock responses
- Theme manager and main app updates

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
2026-03-14 08:57:25 +05:30
parent 4acf75902c
commit e0f34398c2
6 changed files with 662 additions and 268 deletions

View File

@@ -6,6 +6,8 @@ import '../storage/token_storage.dart';
class ApiClient { class ApiClient {
static const Duration _timeout = Duration(seconds: 30); static const Duration _timeout = Duration(seconds: 30);
// Set to true to enable mock/offline development mode (useful when backend is unavailable)
static const bool _developmentMode = true;
/// POST request /// POST request
/// ///
@@ -34,6 +36,30 @@ class ApiClient {
.timeout(_timeout); .timeout(_timeout);
} catch (e) { } catch (e) {
if (kDebugMode) debugPrint('ApiClient.post network error: $e'); if (kDebugMode) debugPrint('ApiClient.post network error: $e');
// Development mode: return mock responses for common endpoints on network errors
if (_developmentMode) {
if (url.contains('/user/login/')) {
if (kDebugMode) debugPrint('Development mode: returning mock login response');
final email = finalBody['username'] ?? 'test@example.com';
return {
'token': 'mock_token_${DateTime.now().millisecondsSinceEpoch}',
'username': email,
'email': email,
'phone_number': '+1234567890',
};
} else if (url.contains('/user/register/')) {
if (kDebugMode) debugPrint('Development mode: returning mock register response');
final email = finalBody['email'] ?? 'test@example.com';
return {
'token': 'mock_token_${DateTime.now().millisecondsSinceEpoch}',
'username': email,
'email': email,
'phone_number': finalBody['phone_number'] ?? '+1234567890',
};
}
}
throw Exception('Network error: $e'); throw Exception('Network error: $e');
} }

View File

@@ -10,9 +10,15 @@ class ThemeManager {
/// Call during app startup to load saved preference. /// Call during app startup to load saved preference.
static Future<void> init() async { static Future<void> init() async {
try {
final prefs = await SharedPreferences.getInstance(); final prefs = await SharedPreferences.getInstance();
final isDark = prefs.getBool(_prefKey) ?? false; final isDark = prefs.getBool(_prefKey) ?? false;
themeMode.value = isDark ? ThemeMode.dark : ThemeMode.light; themeMode.value = isDark ? ThemeMode.dark : ThemeMode.light;
} catch (e) {
// If SharedPreferences fails, default to light theme
print('Error initializing theme: $e');
themeMode.value = ThemeMode.light;
}
} }
/// Set theme and persist /// Set theme and persist

View File

@@ -39,9 +39,12 @@ class MyApp extends StatelessWidget {
}, },
); );
static const String _fontFamily = 'Gilroy';
ThemeData _lightTheme() { ThemeData _lightTheme() {
return ThemeData( return ThemeData(
brightness: Brightness.light, brightness: Brightness.light,
fontFamily: _fontFamily,
primarySwatch: primarySwatch, primarySwatch: primarySwatch,
scaffoldBackgroundColor: const Color(0xFFF7F5FB), scaffoldBackgroundColor: const Color(0xFFF7F5FB),
appBarTheme: const AppBarTheme( appBarTheme: const AppBarTheme(
@@ -61,9 +64,9 @@ class MyApp extends StatelessWidget {
} }
ThemeData _darkTheme() { ThemeData _darkTheme() {
// Basic dark theme based on your sample — tweak colors as desired
return ThemeData( return ThemeData(
brightness: Brightness.dark, brightness: Brightness.dark,
fontFamily: _fontFamily,
primarySwatch: primarySwatch, primarySwatch: primarySwatch,
scaffoldBackgroundColor: const Color(0xFF0B1220), scaffoldBackgroundColor: const Color(0xFF0B1220),
appBarTheme: const AppBarTheme( appBarTheme: const AppBarTheme(
@@ -75,7 +78,7 @@ class MyApp extends StatelessWidget {
style: ElevatedButton.styleFrom(backgroundColor: const Color(0xFF0B63D6)), style: ElevatedButton.styleFrom(backgroundColor: const Color(0xFF0B63D6)),
), ),
cardColor: const Color(0xFF0E1620), cardColor: const Color(0xFF0E1620),
textTheme: ThemeData.dark().textTheme, textTheme: ThemeData.dark().textTheme.apply(fontFamily: _fontFamily),
useMaterial3: false, useMaterial3: false,
); );
} }
@@ -115,9 +118,17 @@ class _StartupScreenState extends State<StartupScreen> {
} }
Future<void> _loadLoginState() async { Future<void> _loadLoginState() async {
try {
final prefs = await SharedPreferences.getInstance(); final prefs = await SharedPreferences.getInstance();
final hasEmail = prefs.getString('email') != null && prefs.getString('email')!.isNotEmpty; final hasEmail = prefs.getString('email') != null && prefs.getString('email')!.isNotEmpty;
setState(() => _loggedIn = hasEmail); setState(() => _loggedIn = hasEmail);
} catch (e) {
// If SharedPreferences fails (common on web with plugin issues), default to not logged in
print('Error loading login state: $e');
if (mounted) {
setState(() => _loggedIn = false);
}
}
} }
@override @override

View File

@@ -237,8 +237,8 @@ class _DesktopLoginScreenState extends State<DesktopLoginScreen> with SingleTick
), ),
), ),
const SizedBox(height: 12), const SizedBox(height: 12),
Row( Wrap(
mainAxisAlignment: MainAxisAlignment.spaceBetween, alignment: WrapAlignment.spaceBetween,
children: [ children: [
TextButton(onPressed: _openRegister, child: const Text("Don't have an account? Register")), TextButton(onPressed: _openRegister, child: const Text("Don't have an account? Register")),
TextButton(onPressed: () {}, child: const Text('Contact support')) TextButton(onPressed: () {}, child: const Text('Contact support'))

View File

@@ -1,6 +1,7 @@
// lib/screens/learn_more_screen.dart // lib/screens/learn_more_screen.dart
import 'dart:async'; import 'dart:async';
import 'dart:ui'; import 'dart:ui';
import 'package:flutter/foundation.dart' show kIsWeb;
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:google_maps_flutter/google_maps_flutter.dart'; import 'package:google_maps_flutter/google_maps_flutter.dart';
import 'package:share_plus/share_plus.dart'; import 'package:share_plus/share_plus.dart';
@@ -222,26 +223,25 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
} }
final screenHeight = MediaQuery.of(context).size.height; final screenHeight = MediaQuery.of(context).size.height;
final imageHeight = screenHeight * 0.50; final imageHeight = screenHeight * 0.45;
final overlap = 30.0; final topPadding = MediaQuery.of(context).padding.top;
return Scaffold( return Scaffold(
backgroundColor: theme.scaffoldBackgroundColor, backgroundColor: theme.scaffoldBackgroundColor,
body: Stack( body: Stack(
children: [ children: [
// ── LAYER 1: Image carousel (background) ── // ── Scrollable content (carousel + card scroll together) ──
_buildImageCarousel(theme, imageHeight),
// ── LAYER 2: Scrollable content with overlapping white card ──
SingleChildScrollView( SingleChildScrollView(
child: Column( child: Column(
crossAxisAlignment: CrossAxisAlignment.start, crossAxisAlignment: CrossAxisAlignment.start,
children: [ children: [
// Transparent spacer — shows the image behind // Image carousel (scrolls with content)
SizedBox(height: imageHeight - overlap), _buildImageCarousel(theme, imageHeight),
// White card with rounded top corners overlapping image // Content card with rounded top corners overlapping carousel
Container( Transform.translate(
offset: const Offset(0, -28),
child: Container(
width: double.infinity, width: double.infinity,
decoration: BoxDecoration( decoration: BoxDecoration(
color: theme.scaffoldBackgroundColor, color: theme.scaffoldBackgroundColor,
@@ -274,15 +274,33 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
], ],
), ),
), ),
),
], ],
), ),
), ),
// ── LAYER 3: Floating icon row (above scrollview so taps work) ── // ── Fixed top bar with back/share/heart buttons ──
Positioned( Positioned(
top: MediaQuery.of(context).padding.top + 10, top: 0,
left: 0,
right: 0,
child: Container(
padding: EdgeInsets.only(
top: topPadding + 10,
bottom: 10,
left: 16, left: 16,
right: 16, right: 16,
),
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
colors: [
Colors.black.withOpacity(0.5),
Colors.black.withOpacity(0.0),
],
),
),
child: Row( child: Row(
children: [ children: [
_squareIconButton( _squareIconButton(
@@ -325,6 +343,7 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
], ],
), ),
), ),
),
], ],
), ),
); );
@@ -492,7 +511,7 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
); );
} }
/// Square icon button with rounded corners and translucent white background /// Square icon button with rounded corners and prominent background
Widget _squareIconButton({ Widget _squareIconButton({
required IconData icon, required IconData icon,
required VoidCallback onTap, required VoidCallback onTap,
@@ -501,12 +520,12 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
return GestureDetector( return GestureDetector(
onTap: onTap, onTap: onTap,
child: Container( child: Container(
width: 42, width: 44,
height: 42, height: 44,
decoration: BoxDecoration( decoration: BoxDecoration(
color: Colors.white.withOpacity(0.2), color: Colors.black.withOpacity(0.35),
borderRadius: BorderRadius.circular(12), borderRadius: BorderRadius.circular(12),
border: Border.all(color: Colors.white.withOpacity(0.3)), border: Border.all(color: Colors.white.withOpacity(0.4)),
), ),
child: Icon(icon, color: iconColor, size: 22), child: Icon(icon, color: iconColor, size: 22),
), ),
@@ -640,6 +659,46 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
height: 280, height: 280,
child: Stack( child: Stack(
children: [ children: [
// Use static map image on web (Google Maps JS SDK not configured),
// native GoogleMap widget on mobile
if (kIsWeb)
GestureDetector(
onTap: _viewLargerMap,
child: Container(
decoration: BoxDecoration(
color: Colors.grey.shade200,
borderRadius: BorderRadius.circular(20),
),
child: Stack(
children: [
Positioned.fill(
child: Image.network(
'https://maps.googleapis.com/maps/api/staticmap?center=$lat,$lng&zoom=15&size=600x300&markers=color:red%7C$lat,$lng&key=',
fit: BoxFit.cover,
errorBuilder: (_, __, ___) => Container(
color: const Color(0xFFE8EAF6),
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
Icon(Icons.map_outlined, size: 48, color: theme.colorScheme.primary),
const SizedBox(height: 8),
Text(
'Tap to view on Google Maps',
style: TextStyle(
color: theme.colorScheme.primary,
fontWeight: FontWeight.w600,
),
),
],
),
),
),
),
],
),
),
)
else
GoogleMap( GoogleMap(
initialCameraPosition: CameraPosition( initialCameraPosition: CameraPosition(
target: LatLng(lat, lng), target: LatLng(lat, lng),
@@ -691,7 +750,8 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
), ),
), ),
// Map type toggle bottom left // Map type toggle bottom left (native only)
if (!kIsWeb)
Positioned( Positioned(
bottom: 12, bottom: 12,
left: 12, left: 12,
@@ -709,7 +769,8 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
), ),
), ),
// Map controls toggle bottom right // Map controls toggle bottom right (native only)
if (!kIsWeb)
Positioned( Positioned(
bottom: 12, bottom: 12,
right: 12, right: 12,
@@ -719,8 +780,8 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
), ),
), ),
// Directional pad overlay // Directional pad overlay (native only)
if (_showMapControls) if (!kIsWeb && _showMapControls)
Positioned.fill( Positioned.fill(
child: Container( child: Container(
decoration: BoxDecoration( decoration: BoxDecoration(
@@ -730,7 +791,6 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
child: Column( child: Column(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
// Top row: Up + Zoom In
Row( Row(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
@@ -744,7 +804,6 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
], ],
), ),
const SizedBox(height: 10), const SizedBox(height: 10),
// Middle row: Left + Right
Row( Row(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
@@ -758,7 +817,6 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
], ],
), ),
const SizedBox(height: 10), const SizedBox(height: 10),
// Bottom row: Down + Zoom Out + Close
Row( Row(
mainAxisAlignment: MainAxisAlignment.center, mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [

View File

@@ -1,9 +1,11 @@
// lib/screens/login_screen.dart // lib/screens/login_screen.dart
import 'dart:ui';
import 'package:flutter/foundation.dart' show kIsWeb;
import 'package:flutter/material.dart'; import 'package:flutter/material.dart';
import 'package:video_player/video_player.dart';
import '../features/auth/services/auth_service.dart'; import '../features/auth/services/auth_service.dart';
import 'home_screen.dart'; import 'home_screen.dart';
import '../core/app_decoration.dart';
class LoginScreen extends StatefulWidget { class LoginScreen extends StatefulWidget {
const LoginScreen({Key? key}) : super(key: key); const LoginScreen({Key? key}) : super(key: key);
@@ -22,9 +24,42 @@ class _LoginScreenState extends State<LoginScreen> {
final AuthService _auth = AuthService(); final AuthService _auth = AuthService();
bool _loading = false; bool _loading = false;
bool _obscurePassword = true;
bool _rememberMe = false;
late VideoPlayerController _videoController;
bool _videoInitialized = false;
// Glassmorphism color palette
static const _darkBg = Color(0xFF0A0A0A);
static const _glassBg = Color(0x1AFFFFFF); // 10% white
static const _glassBorder = Color(0x33FFFFFF); // 20% white
static const _inputBg = Color(0x14FFFFFF); // 8% white
static const _inputBorder = Color(0x26FFFFFF); // 15% white
static const _textWhite = Colors.white;
static const _textMuted = Color(0xFFAAAAAA);
static const _textHint = Color(0xFF888888);
@override
void initState() {
super.initState();
_initVideo();
}
Future<void> _initVideo() async {
_videoController = VideoPlayerController.networkUrl(
Uri.parse('assets/login-bg.mp4'),
);
await _videoController.initialize();
_videoController.setLooping(true);
_videoController.setVolume(0);
_videoController.play();
if (mounted) setState(() => _videoInitialized = true);
}
@override @override
void dispose() { void dispose() {
_videoController.dispose();
_emailCtrl.dispose(); _emailCtrl.dispose();
_passCtrl.dispose(); _passCtrl.dispose();
_emailFocus.dispose(); _emailFocus.dispose();
@@ -35,7 +70,6 @@ class _LoginScreenState extends State<LoginScreen> {
String? _emailValidator(String? v) { String? _emailValidator(String? v) {
if (v == null || v.trim().isEmpty) return 'Enter email'; if (v == null || v.trim().isEmpty) return 'Enter email';
final email = v.trim(); final email = v.trim();
// Basic email pattern check
final emailRegex = RegExp(r"^[^@]+@[^@]+\.[^@]+"); final emailRegex = RegExp(r"^[^@]+@[^@]+\.[^@]+");
if (!emailRegex.hasMatch(email)) return 'Enter a valid email'; if (!emailRegex.hasMatch(email)) return 'Enter a valid email';
return null; return null;
@@ -58,11 +92,9 @@ class _LoginScreenState extends State<LoginScreen> {
setState(() => _loading = true); setState(() => _loading = true);
try { try {
// AuthService.login now returns a UserModel and also persists profile info.
await _auth.login(email, password); await _auth.login(email, password);
if (!mounted) return; if (!mounted) return;
// small delay for UX
await Future.delayed(const Duration(milliseconds: 150)); await Future.delayed(const Duration(milliseconds: 150));
Navigator.of(context).pushReplacement(PageRouteBuilder( Navigator.of(context).pushReplacement(PageRouteBuilder(
@@ -86,75 +118,195 @@ class _LoginScreenState extends State<LoginScreen> {
Navigator.of(context).push(MaterialPageRoute(builder: (_) => const RegisterScreen(isDesktop: false))); Navigator.of(context).push(MaterialPageRoute(builder: (_) => const RegisterScreen(isDesktop: false)));
} }
@override void _showComingSoon() {
Widget build(BuildContext context) { ScaffoldMessenger.of(context).showSnackBar(
const primary = Color(0xFF0B63D6); const SnackBar(content: Text('Coming soon'), duration: Duration(seconds: 1)),
final width = MediaQuery.of(context).size.width; );
}
return Scaffold( /// Glassmorphism pill-shaped input decoration
// backgroundColor: primary, InputDecoration _glassInputDecoration({
body: Container( required String hint,
decoration: AppDecoration.blueGradient, required IconData prefixIcon,
child: SafeArea( Widget? suffixIcon,
child: Center( }) {
child: SingleChildScrollView( const borderRadius = BorderRadius.all(Radius.circular(28));
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 18), return InputDecoration(
child: ConstrainedBox( hintText: hint,
constraints: BoxConstraints(maxWidth: width < 720 ? width : 720), hintStyle: const TextStyle(color: _textHint, fontSize: 14),
child: Column( prefixIcon: Padding(
crossAxisAlignment: CrossAxisAlignment.stretch, padding: const EdgeInsets.only(left: 16, right: 8),
child: Icon(prefixIcon, color: _textMuted, size: 20),
),
prefixIconConstraints: const BoxConstraints(minWidth: 44),
suffixIcon: suffixIcon != null
? Padding(
padding: const EdgeInsets.only(right: 8),
child: suffixIcon,
)
: null,
filled: true,
fillColor: _inputBg,
contentPadding: const EdgeInsets.symmetric(horizontal: 20, vertical: 16),
border: OutlineInputBorder(
borderRadius: borderRadius,
borderSide: BorderSide(color: _inputBorder),
),
enabledBorder: OutlineInputBorder(
borderRadius: borderRadius,
borderSide: BorderSide(color: _inputBorder),
),
focusedBorder: OutlineInputBorder(
borderRadius: borderRadius,
borderSide: BorderSide(color: _glassBorder, width: 1.5),
),
errorBorder: OutlineInputBorder(
borderRadius: borderRadius,
borderSide: const BorderSide(color: Colors.redAccent),
),
focusedErrorBorder: OutlineInputBorder(
borderRadius: borderRadius,
borderSide: const BorderSide(color: Colors.redAccent, width: 1.5),
),
errorStyle: const TextStyle(color: Colors.redAccent, fontSize: 11),
);
}
/// Glassmorphism social button
Widget _socialButton({
required String label,
required Widget icon,
required VoidCallback onTap,
}) {
return Expanded(
child: GestureDetector(
onTap: onTap,
child: Container(
padding: const EdgeInsets.symmetric(vertical: 14),
decoration: BoxDecoration(
color: _inputBg,
borderRadius: BorderRadius.circular(28),
border: Border.all(color: _inputBorder),
),
child: Row(
mainAxisAlignment: MainAxisAlignment.center,
children: [ children: [
const SizedBox(height: 6), icon,
const Text('Welcome', style: TextStyle(fontSize: 34, fontWeight: FontWeight.bold, color: Colors.white)), const SizedBox(width: 8),
const SizedBox(height: 6), Text(
const Text('Sign in to continue', style: TextStyle(color: Colors.white70, fontSize: 16)), label,
const SizedBox(height: 26), style: const TextStyle(
color: _textWhite,
// HERO card fontSize: 13,
Hero( fontWeight: FontWeight.w500,
tag: 'headerCard', ),
flightShuttleBuilder: (flightContext, animation, flightDirection, fromContext, toContext) { ),
return Material( ],
color: Colors.transparent, ),
child: ScaleTransition( ),
scale: animation.drive(Tween(begin: 0.98, end: 1.0).chain(CurveTween(curve: Curves.easeOut))),
child: fromContext.widget,
), ),
); );
}, }
child: Material(
type: MaterialType.transparency, @override
child: Container( Widget build(BuildContext context) {
width: double.infinity, return Scaffold(
decoration: AppDecoration.blueGradientRounded(20).copyWith( backgroundColor: _darkBg,
boxShadow: const [BoxShadow(color: Colors.black26, blurRadius: 18, offset: Offset(0, 8))], body: Stack(
),
padding: const EdgeInsets.fromLTRB(18, 18, 18, 18),
child: Column(
crossAxisAlignment: CrossAxisAlignment.center,
children: [ children: [
const SizedBox(height: 8), // Video background
const Text('Eventify', style: TextStyle(color: Colors.white, fontSize: 22, fontWeight: FontWeight.bold)), if (_videoInitialized)
Positioned.fill(
child: FittedBox(
fit: BoxFit.cover,
child: SizedBox(
width: _videoController.value.size.width,
height: _videoController.value.size.height,
child: VideoPlayer(_videoController),
),
),
),
// Dark gradient overlay for readability
Positioned.fill(
child: Container(
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
colors: [
Colors.black.withOpacity(0.50),
Colors.black.withOpacity(0.65),
Colors.black.withOpacity(0.70),
],
stops: const [0.0, 0.4, 1.0],
),
),
),
),
// Main content
SafeArea(
child: Center(
child: SingleChildScrollView(
padding: const EdgeInsets.symmetric(horizontal: 24, vertical: 20),
child: ConstrainedBox(
constraints: const BoxConstraints(maxWidth: 400),
child: Form(
key: _formKey,
child: Column(
crossAxisAlignment: CrossAxisAlignment.start,
children: [
// Brand name
Center(
child: Text(
'Eventify',
style: TextStyle(
color: _textWhite.withOpacity(0.7),
fontSize: 16,
fontWeight: FontWeight.w500,
fontStyle: FontStyle.italic,
letterSpacing: 1.5,
),
),
),
const SizedBox(height: 12), const SizedBox(height: 12),
// white card inside the blue card — now uses Form // Heading
Form( const Center(
key: _formKey, child: Text(
child: Container( 'Log In, Start Your\nJourney',
decoration: BoxDecoration(color: Colors.white, borderRadius: BorderRadius.circular(12)), textAlign: TextAlign.center,
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 14), style: TextStyle(
child: Column( color: _textWhite,
children: [ fontSize: 28,
// Email fontWeight: FontWeight.bold,
fontStyle: FontStyle.italic,
height: 1.2,
),
),
),
const SizedBox(height: 32),
// Email label
const Padding(
padding: EdgeInsets.only(left: 4, bottom: 8),
child: Text(
'Email',
style: TextStyle(color: _textMuted, fontSize: 13),
),
),
// Email input
TextFormField( TextFormField(
controller: _emailCtrl, controller: _emailCtrl,
focusNode: _emailFocus, focusNode: _emailFocus,
keyboardType: TextInputType.emailAddress, keyboardType: TextInputType.emailAddress,
autofillHints: const [AutofillHints.email], autofillHints: const [AutofillHints.email],
decoration: InputDecoration( style: const TextStyle(color: _textWhite, fontSize: 14),
labelText: 'Email', cursorColor: Colors.white54,
border: InputBorder.none, decoration: _glassInputDecoration(
prefixIcon: Icon(Icons.email, color: primary), hint: 'Enter your email',
prefixIcon: Icons.mail_outline_rounded,
), ),
validator: _emailValidator, validator: _emailValidator,
textInputAction: TextInputAction.next, textInputAction: TextInputAction.next,
@@ -162,60 +314,199 @@ class _LoginScreenState extends State<LoginScreen> {
FocusScope.of(context).requestFocus(_passFocus); FocusScope.of(context).requestFocus(_passFocus);
}, },
), ),
const Divider(), const SizedBox(height: 18),
// Password
// Password label
const Padding(
padding: EdgeInsets.only(left: 4, bottom: 8),
child: Text(
'Password',
style: TextStyle(color: _textMuted, fontSize: 13),
),
),
// Password input
TextFormField( TextFormField(
controller: _passCtrl, controller: _passCtrl,
focusNode: _passFocus, focusNode: _passFocus,
obscureText: true, obscureText: _obscurePassword,
decoration: InputDecoration( style: const TextStyle(color: _textWhite, fontSize: 14),
labelText: 'Password', cursorColor: Colors.white54,
border: InputBorder.none, decoration: _glassInputDecoration(
prefixIcon: Icon(Icons.lock, color: primary), hint: 'Enter your password',
prefixIcon: Icons.lock_outline_rounded,
suffixIcon: IconButton(
icon: Icon(
_obscurePassword ? Icons.visibility_off_outlined : Icons.visibility_outlined,
color: _textMuted,
size: 20,
),
onPressed: () => setState(() => _obscurePassword = !_obscurePassword),
),
), ),
validator: _passwordValidator, validator: _passwordValidator,
textInputAction: TextInputAction.done, textInputAction: TextInputAction.done,
onFieldSubmitted: (_) => _performLogin(), onFieldSubmitted: (_) => _performLogin(),
), ),
const SizedBox(height: 14),
// Remember me + Forgot Password row
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [
// Remember me
GestureDetector(
onTap: () => setState(() => _rememberMe = !_rememberMe),
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
Container(
width: 18,
height: 18,
decoration: BoxDecoration(
borderRadius: BorderRadius.circular(4),
border: Border.all(color: _glassBorder),
color: _rememberMe ? Colors.white24 : Colors.transparent,
),
child: _rememberMe
? const Icon(Icons.check, size: 14, color: _textWhite)
: null,
),
const SizedBox(width: 8),
const Text(
'Remember me',
style: TextStyle(color: _textMuted, fontSize: 12),
),
], ],
), ),
), ),
// Forgot Password
GestureDetector(
onTap: _showComingSoon,
child: const Text(
'Forgot Password?',
style: TextStyle(color: _textMuted, fontSize: 12),
), ),
),
],
),
const SizedBox(height: 24),
const SizedBox(height: 18), // Login button — dark gradient pill
// Login button
SizedBox( SizedBox(
width: double.infinity, width: double.infinity,
child: ElevatedButton( child: Container(
onPressed: _loading ? null : _performLogin, decoration: BoxDecoration(
style: ElevatedButton.styleFrom( borderRadius: BorderRadius.circular(28),
backgroundColor: Colors.white, gradient: const LinearGradient(
foregroundColor: primary, colors: [Color(0xFF2A2A2A), Color(0xFF1A1A1A)],
padding: const EdgeInsets.symmetric(vertical: 14),
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10)),
), ),
border: Border.all(color: const Color(0x33FFFFFF)),
boxShadow: [
BoxShadow(
color: Colors.black.withOpacity(0.4),
blurRadius: 16,
offset: const Offset(0, 6),
),
],
),
child: Material(
color: Colors.transparent,
child: InkWell(
borderRadius: BorderRadius.circular(28),
onTap: _loading ? null : _performLogin,
child: Padding(
padding: const EdgeInsets.symmetric(vertical: 16),
child: Center(
child: _loading child: _loading
? SizedBox(height: 18, width: 18, child: CircularProgressIndicator(strokeWidth: 2, color: primary)) ? const SizedBox(
: const Text('Login', style: TextStyle(fontSize: 16, fontWeight: FontWeight.w600)), height: 20,
width: 20,
child: CircularProgressIndicator(
strokeWidth: 2,
color: _textWhite,
),
)
: const Text(
'Login',
style: TextStyle(
color: _textWhite,
fontSize: 16,
fontWeight: FontWeight.w600,
), ),
), ),
),
),
),
),
),
),
const SizedBox(height: 24),
const SizedBox(height: 8), // "Or continue with" divider
], Row(
),
),
),
),
const SizedBox(height: 18),
Center(
child: Column(
children: [ children: [
const SizedBox(height: 8), Expanded(child: Divider(color: Colors.white.withOpacity(0.12))),
TextButton( Padding(
onPressed: _openRegister, padding: const EdgeInsets.symmetric(horizontal: 14),
child: const Text("Don't have an account? Register", style: TextStyle(color: Colors.white)), child: Text(
'Or continue with',
style: TextStyle(
color: _textMuted.withOpacity(0.7),
fontSize: 12,
),
),
),
Expanded(child: Divider(color: Colors.white.withOpacity(0.12))),
],
),
const SizedBox(height: 18),
// Social buttons — side by side
Row(
children: [
_socialButton(
label: 'Google',
icon: const Text(
'G',
style: TextStyle(
fontSize: 18,
fontWeight: FontWeight.bold,
color: Color(0xFF4285F4),
),
),
onTap: _showComingSoon,
),
const SizedBox(width: 12),
_socialButton(
label: 'Apple',
icon: const Icon(Icons.apple, color: _textWhite, size: 20),
onTap: _showComingSoon,
),
],
),
const SizedBox(height: 28),
// Don't have an account? Create an account
Center(
child: Row(
mainAxisSize: MainAxisSize.min,
children: [
const Text(
"Don't have an account? ",
style: TextStyle(color: _textMuted, fontSize: 13),
),
GestureDetector(
onTap: _openRegister,
child: const Text(
'Create an account',
style: TextStyle(
color: _textWhite,
fontSize: 13,
fontWeight: FontWeight.w600,
decoration: TextDecoration.underline,
decorationColor: _textWhite,
),
),
), ),
], ],
), ),
@@ -227,8 +518,10 @@ class _LoginScreenState extends State<LoginScreen> {
), ),
), ),
), ),
],
),
); );
} }
} }
/// Register screen calls backend register endpoint via AuthService.register /// Register screen calls backend register endpoint via AuthService.register