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:
@@ -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');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
@@ -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'))
|
||||||
|
|||||||
@@ -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: [
|
||||||
|
|||||||
@@ -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
|
||||||
|
|||||||
Reference in New Issue
Block a user