Initial
This commit is contained in:
152
lib/BlindMasterScreens/Startup/create_user_screen.dart
Normal file
152
lib/BlindMasterScreens/Startup/create_user_screen.dart
Normal file
@@ -0,0 +1,152 @@
|
||||
import 'package:blind_master/BlindMasterResources/error_snackbar.dart';
|
||||
import 'package:blind_master/BlindMasterResources/secure_transmissions.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
import 'package:blind_master/BlindMasterResources/text_inputs.dart';
|
||||
import 'package:blind_master/BlindMasterResources/title_text.dart';
|
||||
|
||||
class CreateUserScreen extends StatefulWidget {
|
||||
const CreateUserScreen({super.key});
|
||||
|
||||
@override
|
||||
State<CreateUserScreen> createState() => _CreateUserScreenState();
|
||||
}
|
||||
|
||||
class _CreateUserScreenState extends State<CreateUserScreen> {
|
||||
|
||||
final emailController = TextEditingController();
|
||||
final nameController = TextEditingController();
|
||||
final passwordController = TextEditingController();
|
||||
|
||||
final _passFormKey = GlobalKey<FormState>();
|
||||
final _emailFormKey = GlobalKey<FormState>();
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
}
|
||||
|
||||
Future<void> attemptCreate() async{
|
||||
try {
|
||||
if (!_emailFormKey.currentState!.validate() || !_passFormKey.currentState!.validate()) {
|
||||
throw Exception('Invalid information entered!');
|
||||
}
|
||||
|
||||
final payload = {
|
||||
'name': nameController.text.trim(),
|
||||
'email': emailController.text.trim(),
|
||||
'password': passwordController.text,
|
||||
};
|
||||
|
||||
final response = await regularPost(payload, 'create_user');
|
||||
|
||||
if (response.statusCode == 201) {
|
||||
if(mounted) {
|
||||
ScaffoldMessenger.of(context).clearSnackBars();
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
SnackBar(
|
||||
backgroundColor: Colors.green[800],
|
||||
content: Text(
|
||||
"Account Successfully Created!",
|
||||
textAlign: TextAlign.center,
|
||||
style: TextStyle(fontSize: 15),
|
||||
),
|
||||
)
|
||||
);
|
||||
Navigator.pop(context);
|
||||
}
|
||||
} else {
|
||||
if (response.statusCode == 409) throw Exception('Email Already In Use!');
|
||||
throw Exception('Create failed: ${response.statusCode}');
|
||||
}
|
||||
|
||||
} catch(e) {
|
||||
if (!mounted) return;
|
||||
ScaffoldMessenger.of(context).clearSnackBars();
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
errorSnackbar(e)
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
String? confirmPasswordValidator(String? input) {
|
||||
if (input == passwordController.text) {
|
||||
return null;
|
||||
}
|
||||
else {
|
||||
return "Passwords do not match!";
|
||||
}
|
||||
}
|
||||
|
||||
String? emailValidator(String? input) {
|
||||
if (input == null || input.isEmpty) {
|
||||
return 'Email is required';
|
||||
}
|
||||
|
||||
final emailPattern = r'^[^@]+@[^@]+\.[^@]+$';
|
||||
|
||||
if (!RegExp(emailPattern).hasMatch(input)) {
|
||||
return 'Enter a valid email address';
|
||||
}
|
||||
return null;
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
body: Container(
|
||||
color: Theme.of(context).primaryColorLight,
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
TitleText("Create Account", txtClr: Colors.white),
|
||||
Container(
|
||||
padding: EdgeInsets.fromLTRB(40, 10, 40, 10),
|
||||
child: Column(
|
||||
children: [
|
||||
BlindMasterMainInput("Preferred Name (optional)", controller: nameController),
|
||||
SizedBox(height: 20),
|
||||
Form(
|
||||
key: _emailFormKey,
|
||||
autovalidateMode: AutovalidateMode.onUnfocus,
|
||||
child: BlindMasterMainInput(
|
||||
"Email",
|
||||
controller: emailController,
|
||||
validator: emailValidator,
|
||||
),
|
||||
),
|
||||
Form(
|
||||
key: _passFormKey,
|
||||
autovalidateMode: AutovalidateMode.onUserInteraction,
|
||||
child: Column(
|
||||
children: [
|
||||
BlindMasterMainInput(
|
||||
"Password",
|
||||
password: true,
|
||||
controller: passwordController
|
||||
),
|
||||
BlindMasterMainInput(
|
||||
"Confirm Password",
|
||||
validator: confirmPasswordValidator,
|
||||
password: true,
|
||||
)
|
||||
],
|
||||
)
|
||||
)
|
||||
],
|
||||
),
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: attemptCreate,
|
||||
child: Text(
|
||||
"Create"
|
||||
),
|
||||
)
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
126
lib/BlindMasterScreens/Startup/login_screen.dart
Normal file
126
lib/BlindMasterScreens/Startup/login_screen.dart
Normal file
@@ -0,0 +1,126 @@
|
||||
import 'package:blind_master/BlindMasterResources/secure_transmissions.dart';
|
||||
import 'package:blind_master/BlindMasterScreens/Startup/create_user_screen.dart';
|
||||
import 'package:blind_master/BlindMasterScreens/home_screen.dart';
|
||||
import 'package:blind_master/BlindMasterResources/error_snackbar.dart';
|
||||
import 'package:blind_master/BlindMasterResources/fade_transition.dart';
|
||||
import 'package:blind_master/BlindMasterResources/text_inputs.dart';
|
||||
import 'package:blind_master/BlindMasterResources/title_text.dart';
|
||||
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
||||
|
||||
import 'dart:convert';
|
||||
|
||||
import 'package:flutter/material.dart';
|
||||
|
||||
class LoginScreen extends StatefulWidget {
|
||||
const LoginScreen({super.key});
|
||||
|
||||
@override
|
||||
State<LoginScreen> createState() => _LoginScreenState();
|
||||
}
|
||||
|
||||
class _LoginScreenState extends State<LoginScreen> {
|
||||
final emailController = TextEditingController();
|
||||
final passwordController = TextEditingController();
|
||||
|
||||
bool inputWrong = false;
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
}
|
||||
|
||||
@override
|
||||
void dispose() {
|
||||
// Clean up the controller when the widget is removed from the
|
||||
// widget tree.
|
||||
emailController.dispose();
|
||||
passwordController.dispose();
|
||||
super.dispose();
|
||||
}
|
||||
|
||||
Future<void> attemptSignIn() async {
|
||||
final email = emailController.text.trim();
|
||||
final password = passwordController.text;
|
||||
|
||||
final payload = {
|
||||
'email': email,
|
||||
'password': password,
|
||||
}; // query parameters
|
||||
|
||||
try {
|
||||
final response = await regularPost(payload, 'login');
|
||||
if (response.statusCode != 200) {
|
||||
if (response.statusCode == 400) {throw Exception('Email and Password Necessary');}
|
||||
else if (response.statusCode == 500) {throw Exception('Server Error');}
|
||||
else {throw Exception('Incorrect email or password');}
|
||||
}
|
||||
|
||||
final body = json.decode(response.body) as Map<String, dynamic>;
|
||||
final token = body['token'] as String;
|
||||
|
||||
if (token.isEmpty) throw Exception('Token Not Received');
|
||||
final storage = FlutterSecureStorage();
|
||||
await storage.write(key: 'token', value: token);
|
||||
|
||||
} catch(e) {
|
||||
if (!mounted) return;
|
||||
ScaffoldMessenger.of(context).clearSnackBars();
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
errorSnackbar(e)
|
||||
);
|
||||
return;
|
||||
}
|
||||
|
||||
if (!mounted) return;
|
||||
Navigator.pushReplacement(
|
||||
context,
|
||||
fadeTransition(HomeScreen()),
|
||||
);
|
||||
}
|
||||
|
||||
void switchToCreate() {
|
||||
Navigator.push(
|
||||
context,
|
||||
MaterialPageRoute(builder: (context) => CreateUserScreen()),
|
||||
);
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
body: Container(
|
||||
color: Theme.of(context).primaryColorLight,
|
||||
child: Column(
|
||||
mainAxisAlignment: MainAxisAlignment.center,
|
||||
children: [
|
||||
TitleText("BlindMaster"),
|
||||
Container(
|
||||
padding: EdgeInsets.fromLTRB(40, 10, 40, 10),
|
||||
child: Column(
|
||||
children: [
|
||||
BlindMasterMainInput("Email", controller: emailController),
|
||||
BlindMasterMainInput("Password", controller: passwordController, password: true,),
|
||||
],
|
||||
),
|
||||
),
|
||||
Container(
|
||||
padding: EdgeInsets.only(bottom:10),
|
||||
child: ElevatedButton(
|
||||
onPressed: attemptSignIn,
|
||||
child: Text(
|
||||
"Log In"
|
||||
),
|
||||
)
|
||||
),
|
||||
ElevatedButton(
|
||||
onPressed: switchToCreate,
|
||||
child: Text(
|
||||
"Create Account"
|
||||
),
|
||||
),
|
||||
],
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
89
lib/BlindMasterScreens/Startup/splash_screen.dart
Normal file
89
lib/BlindMasterScreens/Startup/splash_screen.dart
Normal file
@@ -0,0 +1,89 @@
|
||||
import 'dart:async';
|
||||
import 'dart:convert';
|
||||
|
||||
import 'package:blind_master/BlindMasterResources/secure_transmissions.dart';
|
||||
import 'package:blind_master/BlindMasterScreens/home_screen.dart';
|
||||
import 'package:blind_master/BlindMasterScreens/Startup/login_screen.dart';
|
||||
import 'package:blind_master/BlindMasterResources/error_snackbar.dart';
|
||||
import 'package:blind_master/BlindMasterResources/fade_transition.dart';
|
||||
import 'package:flutter/material.dart';
|
||||
import 'package:flutter_secure_storage/flutter_secure_storage.dart';
|
||||
import 'package:http/http.dart' as http;
|
||||
|
||||
class SplashScreen extends StatefulWidget {
|
||||
const SplashScreen({super.key});
|
||||
|
||||
@override
|
||||
State<SplashScreen> createState() => _SplashScreenState();
|
||||
}
|
||||
|
||||
class _SplashScreenState extends State<SplashScreen> {
|
||||
Widget nextScreen = LoginScreen();
|
||||
|
||||
@override
|
||||
void initState() {
|
||||
super.initState();
|
||||
_routeNext();
|
||||
}
|
||||
|
||||
Future<void> _routeNext() async {
|
||||
await verifyToken();
|
||||
_animateBackgroundBasedOnTime();
|
||||
}
|
||||
|
||||
Future<void> verifyToken() async{
|
||||
final storage = FlutterSecureStorage();
|
||||
|
||||
try {
|
||||
http.Response? response = await secureGet('verify');
|
||||
if (response == null) {
|
||||
nextScreen = LoginScreen();
|
||||
return;
|
||||
}
|
||||
|
||||
if (response.statusCode == 200) {
|
||||
final body = json.decode(response.body) as Map<String, dynamic>;
|
||||
final newToken = body['token'];
|
||||
|
||||
if (newToken != null) {
|
||||
await storage.write(key: 'token', value: newToken); // ✅ Rotate
|
||||
}
|
||||
|
||||
nextScreen = HomeScreen();
|
||||
} else {
|
||||
nextScreen = LoginScreen();
|
||||
}
|
||||
} catch (e) {
|
||||
if (!mounted) return;
|
||||
ScaffoldMessenger.of(context).showSnackBar(
|
||||
errorSnackbar(e)
|
||||
);
|
||||
nextScreen = LoginScreen();
|
||||
}
|
||||
}
|
||||
|
||||
void _animateBackgroundBasedOnTime() {
|
||||
|
||||
// Optionally navigate to your main app after a short delay
|
||||
Timer(const Duration(seconds: 3), () {
|
||||
Navigator.pushReplacement(
|
||||
context,
|
||||
// MaterialPageRoute(builder: (context) => const MainAppScreen())
|
||||
fadeTransition(nextScreen),
|
||||
);
|
||||
});
|
||||
}
|
||||
|
||||
@override
|
||||
Widget build(BuildContext context) {
|
||||
return Scaffold(
|
||||
body: AnimatedContainer(
|
||||
duration: const Duration(milliseconds: 500),
|
||||
color: Theme.of(context).primaryColorLight,
|
||||
child: Center(
|
||||
child: Image.asset('assets/images/2xwhite.png', height: 250)
|
||||
),
|
||||
),
|
||||
);
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user