import 'package:aitrainer_app/model/cache.dart'; import 'package:aitrainer_app/service/logging.dart' as logging; import 'package:apple_sign_in/apple_sign_in.dart'; import 'package:firebase_auth/firebase_auth.dart'; import 'package:firebase_core/firebase_core.dart'; import 'package:flutter_facebook_auth/flutter_facebook_auth.dart'; import 'package:google_sign_in/google_sign_in.dart'; class FirebaseApi with logging.Logging { bool appleSignInAvailable = false; static FirebaseApi _instance; static final FirebaseAuth auth = FirebaseAuth.instance; static const String SIGN_IN_OK = "OK"; static const String SIGN_IN_NOT_FOUND = "user-not-found"; static const String SIGN_IN_WRONG_PWD = "wrong-password"; static const String REGISTER_WEAK_PWD = "weak-password"; static const String REGISTER_EMAIL_IN_USE = "email-already-in-use"; UserCredential userCredential; factory FirebaseApi() => _instance ?? FirebaseApi._internal(); FirebaseApi._internal() { _instance = this; } // Define an async function to initialize FlutterFire Future initializeFlutterFire() async { try { // Wait for Firebase to initialize and set `_initialized` state to true await Firebase.initializeApp(); this.appleSignInAvailable = await AppleSignIn.isAvailable(); } catch (e) { // Set `_error` state to true if Firebase initialization fails log("Error initializing Firebase"); } } Future signInEmail(String email, String password) async { if (email == null) { throw Exception("Please type an email address"); } if (password == null) { throw Exception("Password too short"); } String rc = SIGN_IN_OK; try { userCredential = await FirebaseAuth.instance.signInWithEmailAndPassword(email: email, password: password); Cache().firebaseUid = userCredential.user.uid; } on FirebaseAuthException catch (e) { if (e.code == 'user-not-found') { log('No user found for that email.'); rc = SIGN_IN_NOT_FOUND; } else if (e.code == 'wrong-password') { log('Wrong password provided for that user.'); rc = SIGN_IN_WRONG_PWD; throw Exception("Customer does not exist or the password is wrong"); } return e.code; } return rc; } Future registerEmail(String email, String password) async { String rc = SIGN_IN_OK; try { userCredential = await FirebaseAuth.instance.createUserWithEmailAndPassword(email: email, password: password); Cache().firebaseUid = userCredential.user.uid; } on FirebaseAuthException catch (e) { if (e.code == 'weak-password') { log('The password provided is too weak.'); rc = REGISTER_WEAK_PWD; throw Exception("Password too short"); } else if (e.code == 'email-already-in-use') { log('The account already exists for that email.'); rc = REGISTER_EMAIL_IN_USE; throw Exception("The email address has been registered already"); } } catch (e) { log(e); throw Exception(e.toString()); } return rc; } Future> signInWithApple() async { Map userData = Map(); final AuthorizationResult result = await AppleSignIn.performRequests([ AppleIdRequest(requestedScopes: [Scope.email, Scope.fullName]) ]); switch (result.status) { case AuthorizationStatus.authorized: print('User authorized'); break; case AuthorizationStatus.error: print('User error'); throw Exception("Apple Sign-In failed"); break; case AuthorizationStatus.cancelled: print('User cancelled'); throw Exception("Apple Sign-In cancelled"); break; } // Create an `OAuthCredential` from the credential returned by Apple. final oauthCredential = OAuthProvider("apple.com").credential( idToken: String.fromCharCodes(result.credential.identityToken), accessToken: String.fromCharCodes(result.credential.authorizationCode)); // Sign in the user with Firebase. If the nonce we generated earlier does // not match the nonce in `appleCredential.identityToken`, sign in will fail. UserCredential userCredential = await FirebaseAuth.instance.signInWithCredential(oauthCredential); log("userCredential: " + userCredential.toString()); log("Apple Credentials: " + result.credential.user.toString() + " state " + result.credential.state.toString() + " email " + userCredential.user.email); userData['email'] = userCredential.user.email; return userData; } Future> registerWithApple() async { Map userData = Map(); final AuthorizationResult result = await AppleSignIn.performRequests([ AppleIdRequest(requestedScopes: [Scope.email, Scope.fullName]) ]); switch (result.status) { case AuthorizationStatus.authorized: print('Apple User authorized'); break; case AuthorizationStatus.error: print('Apple User error'); throw Exception("Apple Sign-In failed"); break; case AuthorizationStatus.cancelled: print('User cancelled'); throw Exception("Apple Sign-In cancelled"); break; } // Create an `OAuthCredential` from the credential returned by Apple. final oauthCredential = OAuthProvider("apple.com").credential( idToken: String.fromCharCodes(result.credential.identityToken), accessToken: String.fromCharCodes(result.credential.authorizationCode)); // Sign in the user with Firebase. If the nonce we generated earlier does // not match the nonce in `appleCredential.identityToken`, sign in will fail. UserCredential userCredential = await FirebaseAuth.instance.signInWithCredential(oauthCredential); Cache().firebaseUid = userCredential.user.uid; userData['email'] = userCredential.user.email; return userData; } Future> signInWithGoogle() async { Map userData = Map(); try { // Trigger the authentication flow GoogleSignIn _googleSignIn = GoogleSignIn( scopes: [ 'email', 'https://www.googleapis.com/auth/contacts.readonly', ], ); final GoogleSignInAccount googleUser = await _googleSignIn.signIn(); if (googleUser == null) { throw Exception("Google Sign In failed"); } // Obtain the auth details from the request final GoogleSignInAuthentication googleAuth = await googleUser.authentication; // Create a new credential final GoogleAuthCredential credential = GoogleAuthProvider.credential( accessToken: googleAuth.accessToken, idToken: googleAuth.idToken, ); await FirebaseAuth.instance.signInWithCredential(credential); if (googleUser != null) { log("GoogleUser: " + googleUser.toString()); userData['email'] = googleUser.email; userData['id'] = googleUser.id; userData['name'] = googleUser.displayName; } } on Exception catch (ex) { log("Google exception: " + ex.toString()); throw Exception("Google Sign In failed"); } return userData; } Future> registerWithGoogle() async { Map userData = Map(); // Trigger the authentication flow GoogleSignIn _googleSignIn = GoogleSignIn( scopes: [ 'email', 'https://www.googleapis.com/auth/contacts.readonly', ], ); final GoogleSignInAccount googleUser = await _googleSignIn.signIn(); if (googleUser == null) { throw Exception("Google Sign In failed"); } // Obtain the auth details from the request final GoogleSignInAuthentication googleAuth = await googleUser.authentication; // Create a new credential final GoogleAuthCredential credential = GoogleAuthProvider.credential( accessToken: googleAuth.accessToken, idToken: googleAuth.idToken, ); final userCredential = await FirebaseAuth.instance.signInWithCredential(credential); log("Google credentials: " + credential.toString() + " GoogleUser: " + googleUser.toString()); Cache().firebaseUid = userCredential.user.uid; userData['email'] = googleUser.email; return userData; } Future> signInWithFacebook() async { Map userData; // by default the login method has the next permissions ['email','public_profile'] AccessToken accessToken = await FacebookAuth.instance.login(); if (accessToken != null) { log(accessToken.toJson().toString()); Cache().accessTokenFacebook = accessToken; // get the user data userData = await FacebookAuth.instance.getUserData(); Cache().firebaseUid = userData['id']; log(userData.toString()); } else { throw Exception("Facebook login was not successful"); } return userData; } Future> registerWithFacebook() async { Map userData; // by default the login method has the next permissions ['email','public_profile'] AccessToken accessToken = await FacebookAuth.instance.login(); if (accessToken != null) { Cache().accessTokenFacebook = accessToken; // get the user data userData = await FacebookAuth.instance.getUserData(); // Create a credential from the access token final FacebookAuthCredential facebookAuthCredential = FacebookAuthProvider.credential(accessToken.token); // Once signed in, return the UserCredential final userCredential = await FirebaseAuth.instance.signInWithCredential(facebookAuthCredential); Cache().firebaseUid = userCredential.user.uid; log(userData.toString()); } else { throw Exception("Facebook login was not successful"); } return userData; } Future logOutFacebook() async { await FacebookAuth.instance.logOut(); Cache().accessTokenFacebook = null; } Future signOut() async { await FirebaseAuth.instance.signOut(); } Future resetPassword(String email) async { await FirebaseAuth.instance.sendPasswordResetEmail(email: email); } }