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 {
final prefs = await SharedPreferences.getInstance(); try {
final isDark = prefs.getBool(_prefKey) ?? false; final prefs = await SharedPreferences.getInstance();
themeMode.value = isDark ? ThemeMode.dark : ThemeMode.light; final isDark = prefs.getBool(_prefKey) ?? false;
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 {
final prefs = await SharedPreferences.getInstance(); try {
final hasEmail = prefs.getString('email') != null && prefs.getString('email')!.isNotEmpty; final prefs = await SharedPreferences.getInstance();
setState(() => _loggedIn = hasEmail); final hasEmail = prefs.getString('email') != null && prefs.getString('email')!.isNotEmpty;
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,107 +223,125 @@ 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(
width: double.infinity, offset: const Offset(0, -28),
decoration: BoxDecoration( child: Container(
color: theme.scaffoldBackgroundColor, width: double.infinity,
borderRadius: const BorderRadius.vertical( decoration: BoxDecoration(
top: Radius.circular(28), color: theme.scaffoldBackgroundColor,
), borderRadius: const BorderRadius.vertical(
boxShadow: [ top: Radius.circular(28),
BoxShadow(
color: Colors.black.withOpacity(0.08),
blurRadius: 20,
offset: const Offset(0, -6),
), ),
], boxShadow: [
), BoxShadow(
child: Column( color: Colors.black.withOpacity(0.08),
crossAxisAlignment: CrossAxisAlignment.start, blurRadius: 20,
children: [ offset: const Offset(0, -6),
_buildTitleSection(theme), ),
_buildAboutSection(theme),
if (_event!.latitude != null && _event!.longitude != null) ...[
_buildVenueSection(theme),
_buildGetDirectionsButton(theme),
], ],
if (_event!.importantInfo.isNotEmpty) ),
_buildImportantInfoSection(theme), child: Column(
if (_event!.importantInfo.isEmpty && crossAxisAlignment: CrossAxisAlignment.start,
(_event!.importantInformation ?? '').isNotEmpty) children: [
_buildImportantInfoFallback(theme), _buildTitleSection(theme),
const SizedBox(height: 100), _buildAboutSection(theme),
], if (_event!.latitude != null && _event!.longitude != null) ...[
_buildVenueSection(theme),
_buildGetDirectionsButton(theme),
],
if (_event!.importantInfo.isNotEmpty)
_buildImportantInfoSection(theme),
if (_event!.importantInfo.isEmpty &&
(_event!.importantInformation ?? '').isNotEmpty)
_buildImportantInfoFallback(theme),
const SizedBox(height: 100),
],
),
), ),
), ),
], ],
), ),
), ),
// ── 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: 16, left: 0,
right: 16, right: 0,
child: Row( child: Container(
children: [ padding: EdgeInsets.only(
_squareIconButton( top: topPadding + 10,
icon: Icons.arrow_back, bottom: 10,
onTap: () => Navigator.pop(context), left: 16,
right: 16,
),
decoration: BoxDecoration(
gradient: LinearGradient(
begin: Alignment.topCenter,
end: Alignment.bottomCenter,
colors: [
Colors.black.withOpacity(0.5),
Colors.black.withOpacity(0.0),
],
), ),
// Pill-shaped page indicators (centered) ),
Expanded( child: Row(
child: _imageUrls.length > 1 children: [
? Row( _squareIconButton(
mainAxisAlignment: MainAxisAlignment.center, icon: Icons.arrow_back,
children: List.generate(_imageUrls.length, (i) { onTap: () => Navigator.pop(context),
final active = i == _currentPage; ),
return AnimatedContainer( // Pill-shaped page indicators (centered)
duration: const Duration(milliseconds: 300), Expanded(
margin: const EdgeInsets.symmetric(horizontal: 3), child: _imageUrls.length > 1
width: active ? 18 : 8, ? Row(
height: 6, mainAxisAlignment: MainAxisAlignment.center,
decoration: BoxDecoration( children: List.generate(_imageUrls.length, (i) {
color: active final active = i == _currentPage;
? Colors.white return AnimatedContainer(
: Colors.white.withOpacity(0.45), duration: const Duration(milliseconds: 300),
borderRadius: BorderRadius.circular(3), margin: const EdgeInsets.symmetric(horizontal: 3),
), width: active ? 18 : 8,
); height: 6,
}), decoration: BoxDecoration(
) color: active
: const SizedBox.shrink(), ? Colors.white
), : Colors.white.withOpacity(0.45),
_squareIconButton( borderRadius: BorderRadius.circular(3),
icon: Icons.ios_share_outlined, ),
onTap: _shareEvent, );
), }),
const SizedBox(width: 10), )
_squareIconButton( : const SizedBox.shrink(),
icon: _wishlisted ? Icons.favorite : Icons.favorite_border, ),
iconColor: _wishlisted ? Colors.redAccent : Colors.white, _squareIconButton(
onTap: () => setState(() => _wishlisted = !_wishlisted), icon: Icons.ios_share_outlined,
), onTap: _shareEvent,
], ),
const SizedBox(width: 10),
_squareIconButton(
icon: _wishlisted ? Icons.favorite : Icons.favorite_border,
iconColor: _wishlisted ? Colors.redAccent : Colors.white,
onTap: () => setState(() => _wishlisted = !_wishlisted),
),
],
),
), ),
), ),
], ],
@@ -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,26 +659,66 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
height: 280, height: 280,
child: Stack( child: Stack(
children: [ children: [
GoogleMap( // Use static map image on web (Google Maps JS SDK not configured),
initialCameraPosition: CameraPosition( // native GoogleMap widget on mobile
target: LatLng(lat, lng), if (kIsWeb)
zoom: 15, GestureDetector(
), onTap: _viewLargerMap,
mapType: _mapType, child: Container(
markers: { decoration: BoxDecoration(
Marker( color: Colors.grey.shade200,
markerId: const MarkerId('event'), borderRadius: BorderRadius.circular(20),
position: LatLng(lat, lng), ),
infoWindow: InfoWindow(title: venueLabel), 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,
),
),
],
),
),
),
),
],
),
), ),
}, )
myLocationButtonEnabled: false, else
zoomControlsEnabled: false, GoogleMap(
scrollGesturesEnabled: true, initialCameraPosition: CameraPosition(
rotateGesturesEnabled: false, target: LatLng(lat, lng),
tiltGesturesEnabled: false, zoom: 15,
onMapCreated: (c) => _mapController = c, ),
), mapType: _mapType,
markers: {
Marker(
markerId: const MarkerId('event'),
position: LatLng(lat, lng),
infoWindow: InfoWindow(title: venueLabel),
),
},
myLocationButtonEnabled: false,
zoomControlsEnabled: false,
scrollGesturesEnabled: true,
rotateGesturesEnabled: false,
tiltGesturesEnabled: false,
onMapCreated: (c) => _mapController = c,
),
// "View larger map" top left // "View larger map" top left
Positioned( Positioned(
@@ -691,36 +750,38 @@ class _LearnMoreScreenState extends State<LearnMoreScreen> {
), ),
), ),
// Map type toggle bottom left // Map type toggle bottom left (native only)
Positioned( if (!kIsWeb)
bottom: 12, Positioned(
left: 12, bottom: 12,
child: _mapControlButton( left: 12,
icon: _mapType == MapType.normal child: _mapControlButton(
? Icons.satellite_alt icon: _mapType == MapType.normal
: Icons.map_outlined, ? Icons.satellite_alt
onTap: () { : Icons.map_outlined,
setState(() { onTap: () {
_mapType = _mapType == MapType.normal setState(() {
? MapType.satellite _mapType = _mapType == MapType.normal
: MapType.normal; ? MapType.satellite
}); : MapType.normal;
}, });
},
),
), ),
),
// Map controls toggle bottom right // Map controls toggle bottom right (native only)
Positioned( if (!kIsWeb)
bottom: 12, Positioned(
right: 12, bottom: 12,
child: _mapControlButton( right: 12,
icon: Icons.open_with_rounded, child: _mapControlButton(
onTap: () => setState(() => _showMapControls = !_showMapControls), icon: Icons.open_with_rounded,
onTap: () => setState(() => _showMapControls = !_showMapControls),
),
), ),
),
// 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,149 +118,410 @@ 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)));
} }
void _showComingSoon() {
ScaffoldMessenger.of(context).showSnackBar(
const SnackBar(content: Text('Coming soon'), duration: Duration(seconds: 1)),
);
}
/// Glassmorphism pill-shaped input decoration
InputDecoration _glassInputDecoration({
required String hint,
required IconData prefixIcon,
Widget? suffixIcon,
}) {
const borderRadius = BorderRadius.all(Radius.circular(28));
return InputDecoration(
hintText: hint,
hintStyle: const TextStyle(color: _textHint, fontSize: 14),
prefixIcon: Padding(
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: [
icon,
const SizedBox(width: 8),
Text(
label,
style: const TextStyle(
color: _textWhite,
fontSize: 13,
fontWeight: FontWeight.w500,
),
),
],
),
),
),
);
}
@override @override
Widget build(BuildContext context) { Widget build(BuildContext context) {
const primary = Color(0xFF0B63D6);
final width = MediaQuery.of(context).size.width;
return Scaffold( return Scaffold(
// backgroundColor: primary, backgroundColor: _darkBg,
body: Container( body: Stack(
decoration: AppDecoration.blueGradient, children: [
child: SafeArea( // Video background
child: Center( if (_videoInitialized)
child: SingleChildScrollView( Positioned.fill(
padding: const EdgeInsets.symmetric(horizontal: 20, vertical: 18), child: FittedBox(
child: ConstrainedBox( fit: BoxFit.cover,
constraints: BoxConstraints(maxWidth: width < 720 ? width : 720), child: SizedBox(
child: Column( width: _videoController.value.size.width,
crossAxisAlignment: CrossAxisAlignment.stretch, height: _videoController.value.size.height,
children: [ child: VideoPlayer(_videoController),
const SizedBox(height: 6), ),
const Text('Welcome', style: TextStyle(fontSize: 34, fontWeight: FontWeight.bold, color: Colors.white)), ),
const SizedBox(height: 6), ),
const Text('Sign in to continue', style: TextStyle(color: Colors.white70, fontSize: 16)),
const SizedBox(height: 26),
// HERO card // Dark gradient overlay for readability
Hero( Positioned.fill(
tag: 'headerCard', child: Container(
flightShuttleBuilder: (flightContext, animation, flightDirection, fromContext, toContext) { decoration: BoxDecoration(
return Material( gradient: LinearGradient(
color: Colors.transparent, begin: Alignment.topCenter,
child: ScaleTransition( end: Alignment.bottomCenter,
scale: animation.drive(Tween(begin: 0.98, end: 1.0).chain(CurveTween(curve: Curves.easeOut))), colors: [
child: fromContext.widget, 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),
},
child: Material( // Heading
type: MaterialType.transparency, const Center(
child: Container( child: Text(
width: double.infinity, 'Log In, Start Your\nJourney',
decoration: AppDecoration.blueGradientRounded(20).copyWith( textAlign: TextAlign.center,
boxShadow: const [BoxShadow(color: Colors.black26, blurRadius: 18, offset: Offset(0, 8))], style: TextStyle(
color: _textWhite,
fontSize: 28,
fontWeight: FontWeight.bold,
fontStyle: FontStyle.italic,
height: 1.2,
),
),
), ),
padding: const EdgeInsets.fromLTRB(18, 18, 18, 18), const SizedBox(height: 32),
child: Column(
crossAxisAlignment: CrossAxisAlignment.center, // Email label
const Padding(
padding: EdgeInsets.only(left: 4, bottom: 8),
child: Text(
'Email',
style: TextStyle(color: _textMuted, fontSize: 13),
),
),
// Email input
TextFormField(
controller: _emailCtrl,
focusNode: _emailFocus,
keyboardType: TextInputType.emailAddress,
autofillHints: const [AutofillHints.email],
style: const TextStyle(color: _textWhite, fontSize: 14),
cursorColor: Colors.white54,
decoration: _glassInputDecoration(
hint: 'Enter your email',
prefixIcon: Icons.mail_outline_rounded,
),
validator: _emailValidator,
textInputAction: TextInputAction.next,
onFieldSubmitted: (_) {
FocusScope.of(context).requestFocus(_passFocus);
},
),
const SizedBox(height: 18),
// Password label
const Padding(
padding: EdgeInsets.only(left: 4, bottom: 8),
child: Text(
'Password',
style: TextStyle(color: _textMuted, fontSize: 13),
),
),
// Password input
TextFormField(
controller: _passCtrl,
focusNode: _passFocus,
obscureText: _obscurePassword,
style: const TextStyle(color: _textWhite, fontSize: 14),
cursorColor: Colors.white54,
decoration: _glassInputDecoration(
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,
textInputAction: TextInputAction.done,
onFieldSubmitted: (_) => _performLogin(),
),
const SizedBox(height: 14),
// Remember me + Forgot Password row
Row(
mainAxisAlignment: MainAxisAlignment.spaceBetween,
children: [ children: [
const SizedBox(height: 8), // Remember me
const Text('Eventify', style: TextStyle(color: Colors.white, fontSize: 22, fontWeight: FontWeight.bold)), GestureDetector(
const SizedBox(height: 12), onTap: () => setState(() => _rememberMe = !_rememberMe),
child: Row(
// white card inside the blue card — now uses Form mainAxisSize: MainAxisSize.min,
Form( children: [
key: _formKey, Container(
child: Container( width: 18,
decoration: BoxDecoration(color: Colors.white, borderRadius: BorderRadius.circular(12)), height: 18,
padding: const EdgeInsets.symmetric(horizontal: 12, vertical: 14), decoration: BoxDecoration(
child: Column( borderRadius: BorderRadius.circular(4),
children: [ border: Border.all(color: _glassBorder),
// Email color: _rememberMe ? Colors.white24 : Colors.transparent,
TextFormField(
controller: _emailCtrl,
focusNode: _emailFocus,
keyboardType: TextInputType.emailAddress,
autofillHints: const [AutofillHints.email],
decoration: InputDecoration(
labelText: 'Email',
border: InputBorder.none,
prefixIcon: Icon(Icons.email, color: primary),
),
validator: _emailValidator,
textInputAction: TextInputAction.next,
onFieldSubmitted: (_) {
FocusScope.of(context).requestFocus(_passFocus);
},
), ),
const Divider(), child: _rememberMe
// Password ? const Icon(Icons.check, size: 14, color: _textWhite)
TextFormField( : null,
controller: _passCtrl, ),
focusNode: _passFocus, const SizedBox(width: 8),
obscureText: true, const Text(
decoration: InputDecoration( 'Remember me',
labelText: 'Password', style: TextStyle(color: _textMuted, fontSize: 12),
border: InputBorder.none, ),
prefixIcon: Icon(Icons.lock, color: primary), ],
),
validator: _passwordValidator,
textInputAction: TextInputAction.done,
onFieldSubmitted: (_) => _performLogin(),
),
],
),
), ),
), ),
// Forgot Password
const SizedBox(height: 18), GestureDetector(
onTap: _showComingSoon,
// Login button child: const Text(
SizedBox( 'Forgot Password?',
width: double.infinity, style: TextStyle(color: _textMuted, fontSize: 12),
child: ElevatedButton(
onPressed: _loading ? null : _performLogin,
style: ElevatedButton.styleFrom(
backgroundColor: Colors.white,
foregroundColor: primary,
padding: const EdgeInsets.symmetric(vertical: 14),
shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(10)),
),
child: _loading
? SizedBox(height: 18, width: 18, child: CircularProgressIndicator(strokeWidth: 2, color: primary))
: const Text('Login', style: TextStyle(fontSize: 16, fontWeight: FontWeight.w600)),
), ),
), ),
const SizedBox(height: 8),
], ],
), ),
), const SizedBox(height: 24),
),
),
const SizedBox(height: 18), // Login button — dark gradient pill
Center( SizedBox(
child: Column( width: double.infinity,
children: [ child: Container(
const SizedBox(height: 8), decoration: BoxDecoration(
TextButton( borderRadius: BorderRadius.circular(28),
onPressed: _openRegister, gradient: const LinearGradient(
child: const Text("Don't have an account? Register", style: TextStyle(color: Colors.white)), colors: [Color(0xFF2A2A2A), Color(0xFF1A1A1A)],
),
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
? const SizedBox(
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),
// "Or continue with" divider
Row(
children: [
Expanded(child: Divider(color: Colors.white.withOpacity(0.12))),
Padding(
padding: const EdgeInsets.symmetric(horizontal: 14),
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,
),
),
),
],
),
), ),
], ],
), ),
), ),
], ),
), ),
), ),
), ),
), ],
), ),
), );
); }
}
} }
/// Register screen calls backend register endpoint via AuthService.register /// Register screen calls backend register endpoint via AuthService.register