Compare commits
	
		
			No commits in common. "87254ef547145c7fea49022088bd96fbf13d4fa5" and "1dd49fa080a70090ba98ce076d3468e166149cdb" have entirely different histories.
		
	
	
		
			87254ef547
			...
			1dd49fa080
		
	
		
@ -29,28 +29,43 @@ Authentication Bloc for Flutter.
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
## Features
 | 
					## Features
 | 
				
			||||||
 | 
					
 | 
				
			||||||
- Wyatt Architecture
 | 
					- UserInterface
 | 
				
			||||||
- Entities:
 | 
					    * UserFirebase : FirebaseAuth user implementation
 | 
				
			||||||
    - Account : AccountModel -> Contains account information from provider
 | 
					- AuthenticationRepositoryInterface
 | 
				
			||||||
    - AccountWrapper : AccountWrapperModel -> Contains account and extra data.
 | 
					    * AuthenticationRepositoryFirebase : FirebaseAuth implementation
 | 
				
			||||||
- Data Sources:
 | 
					- ExceptionsInterface
 | 
				
			||||||
    - Local:
 | 
					    * ExceptionsFirebase : FirebaseAuth Exception parsing implementation
 | 
				
			||||||
        - Cached Authentication Data : AuthenticationCacheDataSourceImpl -> Provides a cache implementation
 | 
					- AuthenticationBloc
 | 
				
			||||||
    - Remote:
 | 
					    * Tracks every user changes
 | 
				
			||||||
        - Remote Authentication Data : AuthenticationFirebaseDataSourceImpl -> Provides a proxy to FirebaseAuth
 | 
					        - Right after the listener has been registered.
 | 
				
			||||||
- Repositories:
 | 
					        - When a user is signed in.
 | 
				
			||||||
    - AuthenticationRepository : AuthenticationRepositoryImpl -> Provides all authentication methods
 | 
					        - When the current user is signed out.
 | 
				
			||||||
- Features:
 | 
					        - When there is a change in the current user's token.
 | 
				
			||||||
    - Authentication:
 | 
					        - On `refresh()`
 | 
				
			||||||
        - AuthenticationBuilder : widget to build reactive view from authentication state
 | 
					    * Start/Stop listening on demand
 | 
				
			||||||
        - AuthenticationCubit : tracks every auth changes, have sign out capability.
 | 
					        - `start()` to listen to user changes
 | 
				
			||||||
    - SignUp:
 | 
					        - `stop()` to cancel listener
 | 
				
			||||||
        - SignUpCubit: implementation of a FormDataCubit from `wyatt_form_bloc` for the sign up
 | 
					- SignUpCubit
 | 
				
			||||||
    - SignIn:
 | 
					    * Handles email/password validation and password confirmation
 | 
				
			||||||
        - SignUpCubit: implementation of a FormDataCubit from `wyatt_form_bloc` for the sign in
 | 
					    * Handles register with email/password
 | 
				
			||||||
 | 
					    * Handles custom form fields thanks `wyatt_form_bloc`
 | 
				
			||||||
 | 
					        - Use `entries` to pass a `FormData` object
 | 
				
			||||||
 | 
					        - You can use several pre configured `FormInput` for validation
 | 
				
			||||||
 | 
					        - You can use `updateFormData()` to change FormData and validators during runtime (intersection, union, difference or replace)
 | 
				
			||||||
 | 
					- SignInCubit
 | 
				
			||||||
 | 
					    * Handles email/password validation
 | 
				
			||||||
 | 
					    * Handles login with email/password
 | 
				
			||||||
 | 
					- EmailVerificationCubit
 | 
				
			||||||
 | 
					    * Handles send email verification process
 | 
				
			||||||
 | 
					    * Handles email verification check
 | 
				
			||||||
 | 
					- PasswordResetCubit
 | 
				
			||||||
 | 
					    * Handles send password reset email process
 | 
				
			||||||
 | 
					- Builders
 | 
				
			||||||
 | 
					    * AuthenticationBuilder to build widgets on user state changes
 | 
				
			||||||
- Consistent
 | 
					- Consistent
 | 
				
			||||||
    * Every class have same naming convention
 | 
					    * Every class have same naming convention
 | 
				
			||||||
- Tested
 | 
					- Tested
 | 
				
			||||||
 | 
					    * Partially tested with *bloc_test*
 | 
				
			||||||
 | 
					
 | 
				
			||||||
## Getting started
 | 
					## Getting started
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -62,4 +77,227 @@ import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
## Usage
 | 
					## Usage
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// TODO
 | 
					Create an authentication repository:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					final AuthenticationRepositoryInterface _authenticationRepository = AuthenticationRepositoryFirebase();
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Create an authentication cubit:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					final AuthenticationCubit _authenticationCubit = AuthenticationCubit(
 | 
				
			||||||
 | 
					    authenticationRepository: _authenticationRepository,
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Create a sign up cubit:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					final SignUpCubit _signUpCubit = SignUpCubit(
 | 
				
			||||||
 | 
					    authenticationRepository: _authenticationRepository,
 | 
				
			||||||
 | 
					    authenticationCubit: _authenticationCubit,
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					You can use `AuthenticationBloc` to route your app.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					return MultiRepositoryProvider(
 | 
				
			||||||
 | 
					    providers: [
 | 
				
			||||||
 | 
					        RepositoryProvider<AuthenticationRepositoryInterface>(
 | 
				
			||||||
 | 
					            create: (context) => _authenticationRepository,
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					    ],
 | 
				
			||||||
 | 
					    child: MultiBlocProvider(
 | 
				
			||||||
 | 
					        providers: [
 | 
				
			||||||
 | 
					            BlocProvider<AuthenticationCubit>(
 | 
				
			||||||
 | 
					                create: (context) => _authenticationCubit..init(),
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
 | 
					            BlocProvider<SignUpCubit>(
 | 
				
			||||||
 | 
					                create: (context) => _signUpCubit,
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
 | 
					        ],
 | 
				
			||||||
 | 
					        child: const AppView(),
 | 
				
			||||||
 | 
					    ),
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					> Don't forget to call `init()` on authentication cubit.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					And in `AppView` use an `AuthenticationBuilder`:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					AuthenticationBuilder(
 | 
				
			||||||
 | 
					    unknown: (context) => const LoadingPage(),
 | 
				
			||||||
 | 
					    unauthenticated: (context) => const LoginPage(),
 | 
				
			||||||
 | 
					    authenticated: (context, user, userData) => const HomePage(),
 | 
				
			||||||
 | 
					)
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					To create a `SignInCubit` you'll need the same `AuthenticationRepository`, you can use the `context`:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					BlocProvider(
 | 
				
			||||||
 | 
					    create: (_) => SignInCubit(context.read<AuthenticationRepositoryInterface>()),
 | 
				
			||||||
 | 
					    child: const LoginForm(),
 | 
				
			||||||
 | 
					),
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					> In practice it's better to create it in the main `MultiBlocProvider` because the LoginPage can be destroyed, and cubit closed, before login flow ends
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					## Recipes
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Password confirmation
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					In this recipe we'll se how to create a custom `FormEntry` to confirm password.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					First, create an entry at the SignUpCubit creation:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					SignUpCubit _signUpCubit = SignUpCubit(
 | 
				
			||||||
 | 
					    authenticationRepository: _authenticationRepository,
 | 
				
			||||||
 | 
					    authenticationCubit: _authenticationCubit,
 | 
				
			||||||
 | 
					    entries: const FormData([
 | 
				
			||||||
 | 
					        FormEntry('form_field_confirmPassword', ConfirmedPassword.pure()),
 | 
				
			||||||
 | 
					    ]),
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Then, in the sign up form, create an input for password confirmation:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					- `ConfirmedPassword` validator need password value and confirm password value to compare.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					return BlocBuilder<SignUpCubit, SignUpState>(
 | 
				
			||||||
 | 
					    builder: (context, state) {
 | 
				
			||||||
 | 
					        return TextField(
 | 
				
			||||||
 | 
					            onChanged: (confirmPassword) => context
 | 
				
			||||||
 | 
					                .read<SignUpCubit>()
 | 
				
			||||||
 | 
					                .dataChanged(
 | 
				
			||||||
 | 
					                    'form_field_confirmPassword',
 | 
				
			||||||
 | 
					                    ConfirmedPassword.dirty(
 | 
				
			||||||
 | 
					                        password: context.read<SignUpCubit>().state.password.value,
 | 
				
			||||||
 | 
					                        value: confirmPassword,
 | 
				
			||||||
 | 
					                    ),
 | 
				
			||||||
 | 
					                ),
 | 
				
			||||||
 | 
					            obscureText: true,
 | 
				
			||||||
 | 
					            decoration: InputDecoration(
 | 
				
			||||||
 | 
					                labelText: 'confirm password',
 | 
				
			||||||
 | 
					                errorText: state.data!.input('form_field_confirmPassword').invalid
 | 
				
			||||||
 | 
					                    ? 'passwords do not match'
 | 
				
			||||||
 | 
					                    : null,
 | 
				
			||||||
 | 
					            ),
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					> `form_field_confirmPassword` is the field identifier used in all application to retrieve data. You can use a constant to avoid typos.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					You'll need to update password input to update confirm state on password update !
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					return BlocBuilder<SignUpCubit, SignUpState>(
 | 
				
			||||||
 | 
					    builder: (context, state) {
 | 
				
			||||||
 | 
					    return TextField(
 | 
				
			||||||
 | 
					        onChanged: (password) {
 | 
				
			||||||
 | 
					        context.read<SignUpCubit>().passwordChanged(password);
 | 
				
			||||||
 | 
					        context.read<SignUpCubit>().dataChanged(
 | 
				
			||||||
 | 
					                'form_field_confirmPassword',
 | 
				
			||||||
 | 
					                ConfirmedPassword.dirty(
 | 
				
			||||||
 | 
					                password: password,
 | 
				
			||||||
 | 
					                value: context
 | 
				
			||||||
 | 
					                    .read<SignUpCubit>()
 | 
				
			||||||
 | 
					                    .state
 | 
				
			||||||
 | 
					                    .data!
 | 
				
			||||||
 | 
					                    .input('form_field_confirmPassword')
 | 
				
			||||||
 | 
					                    .value,
 | 
				
			||||||
 | 
					                ),
 | 
				
			||||||
 | 
					            );
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        obscureText: true,
 | 
				
			||||||
 | 
					        decoration: InputDecoration(
 | 
				
			||||||
 | 
					            labelText: 'password',
 | 
				
			||||||
 | 
					            errorText: state.password.invalid ? 'invalid password' : null,
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
 | 
					    },
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					> Here you call standard `passwordChanged()` AND `dataChanged()`.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					And voilà !
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					### Create Firestore Document on Sign Up
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					In this recipe we'll se how to create a Firestore Document on sign up success.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					First create a callback function:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					Future<void> onSignUpSuccess(SignUpState state, String? uid) async {
 | 
				
			||||||
 | 
					    if (uid != null) {
 | 
				
			||||||
 | 
					        final user = {
 | 
				
			||||||
 | 
					            'uid': uid,
 | 
				
			||||||
 | 
					            'email': state.email.value,
 | 
				
			||||||
 | 
					            ...state.data.toMap(),
 | 
				
			||||||
 | 
					        };
 | 
				
			||||||
 | 
					        await FirebaseFirestore.instance.collection('users').doc(uid).set(user);
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Then create SignUpCubit with custom entries and register callback:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					SignUpCubit _signUpCubit = SignUpCubit(
 | 
				
			||||||
 | 
					    authenticationRepository: _authenticationRepository,
 | 
				
			||||||
 | 
					    authenticationCubit: _authenticationCubit,
 | 
				
			||||||
 | 
					    entries: const FormData([
 | 
				
			||||||
 | 
					        FormEntry('form_field_name', Name.pure(), fieldName: 'name'),
 | 
				
			||||||
 | 
					        FormEntry('form_field_phone', Phone.pure(), fieldName: 'phone'),
 | 
				
			||||||
 | 
					        FormEntry('form_field_confirmPassword', ConfirmedPassword.pure(), export: false),
 | 
				
			||||||
 | 
					    ]),
 | 
				
			||||||
 | 
					    onSignUpSuccess: onSignUpSuccess,
 | 
				
			||||||
 | 
					);
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					> Use `fieldName` and `export` to control `.toMap()` result on FormData ! Useful to disable exportation of sensible data like passwords.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Create widgets for each inputs:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					class _PhoneInput extends StatelessWidget {
 | 
				
			||||||
 | 
					    @override
 | 
				
			||||||
 | 
					    Widget build(BuildContext context) {
 | 
				
			||||||
 | 
					        return BlocBuilder<SignUpCubit, SignUpState>(
 | 
				
			||||||
 | 
					            builder: (context, state) {
 | 
				
			||||||
 | 
					                return TextField(
 | 
				
			||||||
 | 
					                onChanged: (phone) => context
 | 
				
			||||||
 | 
					                    .read<SignUpCubit>()
 | 
				
			||||||
 | 
					                    .dataChanged('form_field_phone', Phone.dirty(phone)),
 | 
				
			||||||
 | 
					                keyboardType: TextInputType.phone,
 | 
				
			||||||
 | 
					                decoration: InputDecoration(
 | 
				
			||||||
 | 
					                    labelText: 'phone',
 | 
				
			||||||
 | 
					                    helperText: '',
 | 
				
			||||||
 | 
					                    errorText: state.data!.input('form_field_phone').invalid
 | 
				
			||||||
 | 
					                        ? 'invalid phone'
 | 
				
			||||||
 | 
					                        : null,
 | 
				
			||||||
 | 
					                    ),
 | 
				
			||||||
 | 
					                );
 | 
				
			||||||
 | 
					            },
 | 
				
			||||||
 | 
					        );
 | 
				
			||||||
 | 
					    }
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					> Create widgets for Name and ConfirmedPassword too.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					Then add a sign up button with:
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					```dart
 | 
				
			||||||
 | 
					context.read<SignUpCubit>().signUpFormSubmitted()
 | 
				
			||||||
 | 
					```
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					And voilà, a document with `uid` as id, and fields `email`, `name`, `phone`, `uid` will be create in `users` collection.
 | 
				
			||||||
@ -16,7 +16,6 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
import 'package:get_it/get_it.dart';
 | 
					import 'package:get_it/get_it.dart';
 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
final getIt = GetIt.I;
 | 
					final getIt = GetIt.I;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -24,28 +23,7 @@ abstract class GetItInitializer {
 | 
				
			|||||||
  static Future<void> init() async {
 | 
					  static Future<void> init() async {
 | 
				
			||||||
    getIt
 | 
					    getIt
 | 
				
			||||||
      ..registerLazySingleton<AuthenticationRemoteDataSource>(
 | 
					      ..registerLazySingleton<AuthenticationRemoteDataSource>(
 | 
				
			||||||
        () => AuthenticationMockDataSourceImpl(registeredAccounts: [
 | 
					        () => AuthenticationFirebaseDataSourceImpl(),
 | 
				
			||||||
          Pair(
 | 
					 | 
				
			||||||
            AccountModel(
 | 
					 | 
				
			||||||
              uid: '1',
 | 
					 | 
				
			||||||
              emailVerified: true,
 | 
					 | 
				
			||||||
              isAnonymous: false,
 | 
					 | 
				
			||||||
              providerId: 'wyatt',
 | 
					 | 
				
			||||||
              email: 'toto@test.fr',
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            'toto1234',
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          Pair(
 | 
					 | 
				
			||||||
            AccountModel(
 | 
					 | 
				
			||||||
              uid: '2',
 | 
					 | 
				
			||||||
              emailVerified: false,
 | 
					 | 
				
			||||||
              isAnonymous: false,
 | 
					 | 
				
			||||||
              providerId: 'wyatt',
 | 
					 | 
				
			||||||
              email: 'tata@test.fr',
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            'tata1234',
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
        ]),
 | 
					 | 
				
			||||||
      )
 | 
					      )
 | 
				
			||||||
      ..registerLazySingleton<AuthenticationCacheDataSource<int>>(
 | 
					      ..registerLazySingleton<AuthenticationCacheDataSource<int>>(
 | 
				
			||||||
        () => AuthenticationCacheDataSourceImpl<int>(),
 | 
					        () => AuthenticationCacheDataSourceImpl<int>(),
 | 
				
			||||||
 | 
				
			|||||||
@ -3,11 +3,12 @@
 | 
				
			|||||||
// -----
 | 
					// -----
 | 
				
			||||||
// File: sign_in_form.dart
 | 
					// File: sign_in_form.dart
 | 
				
			||||||
// Created Date: 19/08/2022 15:24:37
 | 
					// Created Date: 19/08/2022 15:24:37
 | 
				
			||||||
// Last Modified: Fri Nov 11 2022
 | 
					// Last Modified: Thu Nov 10 2022
 | 
				
			||||||
// -----
 | 
					// -----
 | 
				
			||||||
// Copyright (c) 2022
 | 
					// Copyright (c) 2022
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import 'package:flutter/material.dart';
 | 
					import 'package:flutter/material.dart';
 | 
				
			||||||
 | 
					import 'package:flutter_bloc/flutter_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -72,12 +73,16 @@ class SignInForm extends StatelessWidget {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  Widget build(BuildContext context) {
 | 
					  Widget build(BuildContext context) {
 | 
				
			||||||
    return SignInListener<int>(
 | 
					    return BlocListener<SignInCubit<int>, SignInState>(
 | 
				
			||||||
      onError: (context, status, errorMessage) => ScaffoldMessenger.of(context)
 | 
					      listener: (context, state) {
 | 
				
			||||||
 | 
					        if (state.status.isSubmissionFailure) {
 | 
				
			||||||
 | 
					          ScaffoldMessenger.of(context)
 | 
				
			||||||
            ..hideCurrentSnackBar()
 | 
					            ..hideCurrentSnackBar()
 | 
				
			||||||
            ..showSnackBar(
 | 
					            ..showSnackBar(
 | 
				
			||||||
          SnackBar(content: Text(errorMessage ?? 'Sign In Failure')),
 | 
					              SnackBar(content: Text(state.errorMessage ?? 'Sign In Failure')),
 | 
				
			||||||
        ),
 | 
					            );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
      child: SingleChildScrollView(
 | 
					      child: SingleChildScrollView(
 | 
				
			||||||
        child: Column(
 | 
					        child: Column(
 | 
				
			||||||
          children: [
 | 
					          children: [
 | 
				
			||||||
 | 
				
			|||||||
@ -3,12 +3,13 @@
 | 
				
			|||||||
// -----
 | 
					// -----
 | 
				
			||||||
// File: sign_up_form.dart
 | 
					// File: sign_up_form.dart
 | 
				
			||||||
// Created Date: 19/08/2022 14:41:08
 | 
					// Created Date: 19/08/2022 14:41:08
 | 
				
			||||||
// Last Modified: Fri Nov 11 2022
 | 
					// Last Modified: Thu Nov 10 2022
 | 
				
			||||||
// -----
 | 
					// -----
 | 
				
			||||||
// Copyright (c) 2022
 | 
					// Copyright (c) 2022
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import 'package:example_router/core/constants/form_field.dart';
 | 
					import 'package:example_router/core/constants/form_field.dart';
 | 
				
			||||||
import 'package:flutter/material.dart' hide FormField;
 | 
					import 'package:flutter/material.dart' hide FormField;
 | 
				
			||||||
 | 
					import 'package:flutter_bloc/flutter_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -109,13 +110,16 @@ class SignUpForm extends StatelessWidget {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  Widget build(BuildContext context) {
 | 
					  Widget build(BuildContext context) {
 | 
				
			||||||
    return SignUpListener<int>(
 | 
					    return BlocListener<SignUpCubit<int>, SignUpState>(
 | 
				
			||||||
        onError: (context, status, errorMessage) =>
 | 
					      listener: (context, state) {
 | 
				
			||||||
 | 
					        if (state.status.isSubmissionFailure) {
 | 
				
			||||||
          ScaffoldMessenger.of(context)
 | 
					          ScaffoldMessenger.of(context)
 | 
				
			||||||
            ..hideCurrentSnackBar()
 | 
					            ..hideCurrentSnackBar()
 | 
				
			||||||
            ..showSnackBar(
 | 
					            ..showSnackBar(
 | 
				
			||||||
                SnackBar(content: Text(errorMessage ?? 'Sign Up Failure')),
 | 
					              SnackBar(content: Text(state.errorMessage ?? 'Sign Up Failure')),
 | 
				
			||||||
              ),
 | 
					            );
 | 
				
			||||||
 | 
					        }
 | 
				
			||||||
 | 
					      },
 | 
				
			||||||
      child: SingleChildScrollView(
 | 
					      child: SingleChildScrollView(
 | 
				
			||||||
        child: Column(
 | 
					        child: Column(
 | 
				
			||||||
          children: [
 | 
					          children: [
 | 
				
			||||||
@ -128,6 +132,7 @@ class SignUpForm extends StatelessWidget {
 | 
				
			|||||||
            _SignUpButton(),
 | 
					            _SignUpButton(),
 | 
				
			||||||
          ],
 | 
					          ],
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        ));
 | 
					      ),
 | 
				
			||||||
 | 
					    );
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -17,5 +17,4 @@
 | 
				
			|||||||
abstract class AuthFormName {
 | 
					abstract class AuthFormName {
 | 
				
			||||||
  static const String signUpForm = 'wyattSignUpForm';
 | 
					  static const String signUpForm = 'wyattSignUpForm';
 | 
				
			||||||
  static const String signInForm = 'wyattSignInForm';
 | 
					  static const String signInForm = 'wyattSignInForm';
 | 
				
			||||||
  static const String passwordResetForm = 'wyattPasswordResetForm';
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -23,9 +23,6 @@ abstract class AuthenticationFailureInterface extends AppException
 | 
				
			|||||||
  String code;
 | 
					  String code;
 | 
				
			||||||
  String msg;
 | 
					  String msg;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  String get message => msg;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  AuthenticationFailureInterface(this.code, this.msg);
 | 
					  AuthenticationFailureInterface(this.code, this.msg);
 | 
				
			||||||
  AuthenticationFailureInterface.fromCode(this.code)
 | 
					  AuthenticationFailureInterface.fromCode(this.code)
 | 
				
			||||||
      : msg = 'An unknown error occurred.';
 | 
					      : msg = 'An unknown error occurred.';
 | 
				
			||||||
@ -249,10 +246,3 @@ abstract class SignOutFailureInterface extends AuthenticationFailureInterface {
 | 
				
			|||||||
  /// {@macro sign_out_failure}
 | 
					  /// {@macro sign_out_failure}
 | 
				
			||||||
  SignOutFailureInterface.fromCode(super.code) : super.fromCode();
 | 
					  SignOutFailureInterface.fromCode(super.code) : super.fromCode();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					 | 
				
			||||||
abstract class GetIdTokenFailureInterface
 | 
					 | 
				
			||||||
    extends AuthenticationFailureInterface {
 | 
					 | 
				
			||||||
  GetIdTokenFailureInterface(super.code, super.msg);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  GetIdTokenFailureInterface.fromCode(super.code) : super.fromCode();
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -270,10 +270,3 @@ class SignOutFailureFirebase extends SignOutFailureInterface {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  SignOutFailureFirebase.fromCode(super.code) : super.fromCode();
 | 
					  SignOutFailureFirebase.fromCode(super.code) : super.fromCode();
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					 | 
				
			||||||
class GetIdTokenFailureFirebase extends GetIdTokenFailureInterface {
 | 
					 | 
				
			||||||
  GetIdTokenFailureFirebase([String? code, String? msg])
 | 
					 | 
				
			||||||
      : super(code ?? 'unknown', msg ?? 'An unknown error occurred.');
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  GetIdTokenFailureFirebase.fromCode(super.code) : super.fromCode();
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -16,4 +16,3 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
export 'local/authentication_cache_data_source_impl.dart';
 | 
					export 'local/authentication_cache_data_source_impl.dart';
 | 
				
			||||||
export 'remote/authentication_firebase_data_source_impl.dart';
 | 
					export 'remote/authentication_firebase_data_source_impl.dart';
 | 
				
			||||||
export 'remote/authentication_mock_data_source_impl.dart';
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -15,6 +15,7 @@
 | 
				
			|||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import 'package:firebase_auth/firebase_auth.dart';
 | 
					import 'package:firebase_auth/firebase_auth.dart';
 | 
				
			||||||
 | 
					import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -27,18 +28,7 @@ class AuthenticationFirebaseDataSourceImpl
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  Account _mapper(User user) => AccountModel(
 | 
					  Account _mapper(User user) => AccountModel(
 | 
				
			||||||
        uid: user.uid,
 | 
					        uid: user.uid,
 | 
				
			||||||
        emailVerified: user.emailVerified,
 | 
					 | 
				
			||||||
        isAnonymous: user.isAnonymous,
 | 
					 | 
				
			||||||
        providerId: user.providerData.first.providerId,
 | 
					 | 
				
			||||||
        creationTime: user.metadata.creationTime,
 | 
					 | 
				
			||||||
        lastSignInTime: user.metadata.lastSignInTime,
 | 
					 | 
				
			||||||
        isNewUser: (user.metadata.creationTime != null &&
 | 
					 | 
				
			||||||
                user.metadata.lastSignInTime != null)
 | 
					 | 
				
			||||||
            ? user.metadata.lastSignInTime! == user.metadata.creationTime!
 | 
					 | 
				
			||||||
            : null,
 | 
					 | 
				
			||||||
        email: user.email,
 | 
					        email: user.email,
 | 
				
			||||||
        phoneNumber: user.phoneNumber,
 | 
					 | 
				
			||||||
        photoURL: user.photoURL,
 | 
					 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
@ -65,8 +55,6 @@ class AuthenticationFirebaseDataSourceImpl
 | 
				
			|||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// {@macro signup}
 | 
					 | 
				
			||||||
  Future<Account> signUp({
 | 
					  Future<Account> signUp({
 | 
				
			||||||
    required String email,
 | 
					    required String email,
 | 
				
			||||||
    required String password,
 | 
					    required String password,
 | 
				
			||||||
@ -107,10 +95,9 @@ class AuthenticationFirebaseDataSourceImpl
 | 
				
			|||||||
      } else {
 | 
					      } else {
 | 
				
			||||||
        throw Exception(); // Get caught just after.
 | 
					        throw Exception(); // Get caught just after.
 | 
				
			||||||
      }
 | 
					      }
 | 
				
			||||||
    } on FirebaseAuthException catch (e) {
 | 
					 | 
				
			||||||
      throw GetIdTokenFailureFirebase.fromCode(e.code);
 | 
					 | 
				
			||||||
    } catch (_) {
 | 
					    } catch (_) {
 | 
				
			||||||
      throw GetIdTokenFailureFirebase();
 | 
					      // TODO(hpcl): implement a non ambiguous exception for this case
 | 
				
			||||||
 | 
					      throw ServerException();
 | 
				
			||||||
    }
 | 
					    }
 | 
				
			||||||
  }
 | 
					  }
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -120,83 +107,4 @@ class AuthenticationFirebaseDataSourceImpl
 | 
				
			|||||||
        final Account? account = (user.isNotNull) ? _mapper(user!) : null;
 | 
					        final Account? account = (user.isNotNull) ? _mapper(user!) : null;
 | 
				
			||||||
        return account;
 | 
					        return account;
 | 
				
			||||||
      });
 | 
					      });
 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> confirmPasswordReset({
 | 
					 | 
				
			||||||
    required String code,
 | 
					 | 
				
			||||||
    required String newPassword,
 | 
					 | 
				
			||||||
  }) async {
 | 
					 | 
				
			||||||
    try {
 | 
					 | 
				
			||||||
      await _firebaseAuth.confirmPasswordReset(
 | 
					 | 
				
			||||||
        code: code,
 | 
					 | 
				
			||||||
        newPassword: newPassword,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    } on FirebaseAuthException catch (e) {
 | 
					 | 
				
			||||||
      throw ConfirmPasswordResetFailureFirebase.fromCode(e.code);
 | 
					 | 
				
			||||||
    } catch (_) {
 | 
					 | 
				
			||||||
      throw ConfirmPasswordResetFailureFirebase();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> sendEmailVerification() async {
 | 
					 | 
				
			||||||
    try {
 | 
					 | 
				
			||||||
      await _firebaseAuth.currentUser!.sendEmailVerification();
 | 
					 | 
				
			||||||
    } on FirebaseAuthException catch (e) {
 | 
					 | 
				
			||||||
      throw SendEmailVerificationFailureFirebase.fromCode(e.code);
 | 
					 | 
				
			||||||
    } catch (_) {
 | 
					 | 
				
			||||||
      throw SendEmailVerificationFailureFirebase();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> sendPasswordResetEmail({required String email}) async {
 | 
					 | 
				
			||||||
    try {
 | 
					 | 
				
			||||||
      await _firebaseAuth.sendPasswordResetEmail(email: email);
 | 
					 | 
				
			||||||
    } on FirebaseAuthException catch (e) {
 | 
					 | 
				
			||||||
      throw SendPasswordResetEmailFailureFirebase.fromCode(e.code);
 | 
					 | 
				
			||||||
    } catch (_) {
 | 
					 | 
				
			||||||
      throw SendPasswordResetEmailFailureFirebase();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<Account> signInAnonymously() async {
 | 
					 | 
				
			||||||
    try {
 | 
					 | 
				
			||||||
      final userCredential = await _firebaseAuth.signInAnonymously();
 | 
					 | 
				
			||||||
      final user = userCredential.user;
 | 
					 | 
				
			||||||
      if (user.isNotNull) {
 | 
					 | 
				
			||||||
        return _mapper(user!);
 | 
					 | 
				
			||||||
      } else {
 | 
					 | 
				
			||||||
        throw Exception(); // Get caught just after.
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
    } on FirebaseAuthException catch (e) {
 | 
					 | 
				
			||||||
      throw SignInAnonymouslyFailureFirebase.fromCode(e.code);
 | 
					 | 
				
			||||||
    } catch (_) {
 | 
					 | 
				
			||||||
      throw SignInAnonymouslyFailureFirebase();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<bool> verifyPasswordResetCode({required String code}) async {
 | 
					 | 
				
			||||||
    try {
 | 
					 | 
				
			||||||
      final email = await _firebaseAuth.verifyPasswordResetCode(code);
 | 
					 | 
				
			||||||
      return email.isNotNullOrEmpty;
 | 
					 | 
				
			||||||
    } on FirebaseAuthException catch (e) {
 | 
					 | 
				
			||||||
      throw VerifyPasswordResetCodeFailureFirebase.fromCode(e.code);
 | 
					 | 
				
			||||||
    } catch (_) {
 | 
					 | 
				
			||||||
      throw VerifyPasswordResetCodeFailureFirebase();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> refresh() async {
 | 
					 | 
				
			||||||
    try {
 | 
					 | 
				
			||||||
      await _firebaseAuth.currentUser!.reload();
 | 
					 | 
				
			||||||
    } on FirebaseAuthException catch (e) {
 | 
					 | 
				
			||||||
      throw RefreshFailureFirebase.fromCode(e.code);
 | 
					 | 
				
			||||||
    } catch (_) {
 | 
					 | 
				
			||||||
      throw RefreshFailureFirebase();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -1,204 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'dart:async';
 | 
					 | 
				
			||||||
import 'dart:math';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class AuthenticationMockDataSourceImpl extends AuthenticationRemoteDataSource {
 | 
					 | 
				
			||||||
  Pair<Account, String>? _connectedMock;
 | 
					 | 
				
			||||||
  Pair<Account, String>? _registeredMock;
 | 
					 | 
				
			||||||
  final StreamController<Account?> _streamAccount = StreamController();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  final List<Pair<Account, String>>? registeredAccounts;
 | 
					 | 
				
			||||||
  final String idToken;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  AuthenticationMockDataSourceImpl({
 | 
					 | 
				
			||||||
    this.idToken = 'fake-id-token',
 | 
					 | 
				
			||||||
    this.registeredAccounts,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  Future<void> _randomDelay() async {
 | 
					 | 
				
			||||||
    await Future<void>.delayed(
 | 
					 | 
				
			||||||
      Duration(milliseconds: Random().nextInt(400) + 200),
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
    return;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> confirmPasswordReset({
 | 
					 | 
				
			||||||
    required String code,
 | 
					 | 
				
			||||||
    required String newPassword,
 | 
					 | 
				
			||||||
  }) async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<String> getIdentityToken() async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
    return idToken;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> refresh() async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
    if (_connectedMock.isNull) {
 | 
					 | 
				
			||||||
      throw RefreshFailureFirebase();
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    final refresh = DateTime.now();
 | 
					 | 
				
			||||||
    final mock = (_connectedMock?.left as AccountModel?)
 | 
					 | 
				
			||||||
        ?.copyWith(lastSignInTime: refresh);
 | 
					 | 
				
			||||||
    _connectedMock = _connectedMock?.copyWith(left: mock);
 | 
					 | 
				
			||||||
    _streamAccount.add(mock);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> sendEmailVerification() async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
    if (_connectedMock.isNotNull) {
 | 
					 | 
				
			||||||
      final refresh = DateTime.now();
 | 
					 | 
				
			||||||
      final mock = (_connectedMock?.left as AccountModel?)?.copyWith(
 | 
					 | 
				
			||||||
        emailVerified: false,
 | 
					 | 
				
			||||||
        lastSignInTime: refresh,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
      _streamAccount.add(mock);
 | 
					 | 
				
			||||||
      _connectedMock = _connectedMock?.copyWith(left: mock);
 | 
					 | 
				
			||||||
      return;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    throw SendEmailVerificationFailureFirebase();
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> sendPasswordResetEmail({required String email}) async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
    if (registeredAccounts.isNotNull) {
 | 
					 | 
				
			||||||
      final accounts =
 | 
					 | 
				
			||||||
          registeredAccounts?.where((pair) => pair.left?.email == email);
 | 
					 | 
				
			||||||
      if (accounts.isNotNullOrEmpty) {
 | 
					 | 
				
			||||||
        return;
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    if (_registeredMock.isNotNull) {
 | 
					 | 
				
			||||||
      if (_registeredMock?.left?.email != email) {
 | 
					 | 
				
			||||||
        throw SendPasswordResetEmailFailureFirebase();
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      return;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    throw SendPasswordResetEmailFailureFirebase();
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<Account> signInAnonymously() async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
    final creation = DateTime.now();
 | 
					 | 
				
			||||||
    final mock = AccountModel(
 | 
					 | 
				
			||||||
      uid: 'mock-id-anom',
 | 
					 | 
				
			||||||
      emailVerified: false,
 | 
					 | 
				
			||||||
      isAnonymous: true,
 | 
					 | 
				
			||||||
      providerId: 'wyatt',
 | 
					 | 
				
			||||||
      creationTime: creation,
 | 
					 | 
				
			||||||
      lastSignInTime: creation,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
    _streamAccount.add(mock);
 | 
					 | 
				
			||||||
    _connectedMock = _connectedMock?.copyWith(left: mock);
 | 
					 | 
				
			||||||
    return Future.value(mock);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<Account> signInWithEmailAndPassword({
 | 
					 | 
				
			||||||
    required String email,
 | 
					 | 
				
			||||||
    required String password,
 | 
					 | 
				
			||||||
  }) async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
    if (registeredAccounts.isNotNull) {
 | 
					 | 
				
			||||||
      final accounts =
 | 
					 | 
				
			||||||
          registeredAccounts?.where((pair) => pair.left?.email == email);
 | 
					 | 
				
			||||||
      if (accounts.isNotNullOrEmpty) {
 | 
					 | 
				
			||||||
        final account = accounts?.first;
 | 
					 | 
				
			||||||
        if (account?.right != password) {
 | 
					 | 
				
			||||||
          throw SignInWithCredentialFailureFirebase.fromCode('wrong-password');
 | 
					 | 
				
			||||||
        }
 | 
					 | 
				
			||||||
        _streamAccount.add(account!.left);
 | 
					 | 
				
			||||||
        _connectedMock = account.copyWith();
 | 
					 | 
				
			||||||
        return account.left!;
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    if (_registeredMock.isNotNull) {
 | 
					 | 
				
			||||||
      if (_registeredMock?.left?.email != email) {
 | 
					 | 
				
			||||||
        throw SignInWithCredentialFailureFirebase.fromCode('user-not-found');
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      if (_registeredMock?.right != password) {
 | 
					 | 
				
			||||||
        throw SignInWithCredentialFailureFirebase.fromCode('wrong-password');
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
      _streamAccount.add(_registeredMock!.left);
 | 
					 | 
				
			||||||
      _connectedMock = _registeredMock!.copyWith();
 | 
					 | 
				
			||||||
      return _registeredMock!.left!;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    throw SignInWithCredentialFailureFirebase();
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<void> signOut() async {
 | 
					 | 
				
			||||||
    _connectedMock = null;
 | 
					 | 
				
			||||||
    _streamAccount.add(null);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<Account> signUp({
 | 
					 | 
				
			||||||
    required String email,
 | 
					 | 
				
			||||||
    required String password,
 | 
					 | 
				
			||||||
  }) async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
    if (registeredAccounts.isNotNull) {
 | 
					 | 
				
			||||||
      final accounts =
 | 
					 | 
				
			||||||
          registeredAccounts?.where((pair) => pair.left?.email == email);
 | 
					 | 
				
			||||||
      if (accounts.isNotNullOrEmpty) {
 | 
					 | 
				
			||||||
        throw SignUpWithEmailAndPasswordFailureFirebase.fromCode(
 | 
					 | 
				
			||||||
          'email-already-in-use',
 | 
					 | 
				
			||||||
        );
 | 
					 | 
				
			||||||
      }
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    if (_registeredMock?.left?.email == email) {
 | 
					 | 
				
			||||||
      throw SignUpWithEmailAndPasswordFailureFirebase.fromCode(
 | 
					 | 
				
			||||||
        'email-already-in-use',
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
    final creation = DateTime.now();
 | 
					 | 
				
			||||||
    final mock = AccountModel(
 | 
					 | 
				
			||||||
      uid: 'mock-id-email',
 | 
					 | 
				
			||||||
      emailVerified: false,
 | 
					 | 
				
			||||||
      isAnonymous: false,
 | 
					 | 
				
			||||||
      providerId: 'wyatt',
 | 
					 | 
				
			||||||
      email: email,
 | 
					 | 
				
			||||||
      creationTime: creation,
 | 
					 | 
				
			||||||
      lastSignInTime: creation,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
    _streamAccount.add(mock);
 | 
					 | 
				
			||||||
    _registeredMock = Pair(mock, password);
 | 
					 | 
				
			||||||
    return Future.value(mock);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Stream<Account?> streamAccount() => _streamAccount.stream.asBroadcastStream();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Future<bool> verifyPasswordResetCode({required String code}) async {
 | 
					 | 
				
			||||||
    await _randomDelay();
 | 
					 | 
				
			||||||
    return true;
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,4 +1,3 @@
 | 
				
			|||||||
// ignore_for_file: public_member_api_docs, sort_constructors_first
 | 
					 | 
				
			||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					// Copyright (C) 2022 WYATT GROUP
 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					// Please see the AUTHORS file for details.
 | 
				
			||||||
//
 | 
					//
 | 
				
			||||||
@ -17,71 +16,12 @@
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/domain/entities/account.dart';
 | 
					import 'package:wyatt_authentication_bloc/src/domain/entities/account.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class AccountModel extends Account {
 | 
					class AccountModel implements Account {
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  final String uid;
 | 
					  final String uid;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  final String? email;
 | 
					  final String? email;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  AccountModel({required this.uid, required this.email});
 | 
				
			||||||
  final DateTime? creationTime;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  final bool emailVerified;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  final bool isAnonymous;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  final bool? isNewUser;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  final DateTime? lastSignInTime;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  final String? phoneNumber;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  final String? photoURL;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  final String providerId;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  AccountModel({
 | 
					 | 
				
			||||||
    required this.uid,
 | 
					 | 
				
			||||||
    required this.emailVerified,
 | 
					 | 
				
			||||||
    required this.isAnonymous,
 | 
					 | 
				
			||||||
    required this.providerId,
 | 
					 | 
				
			||||||
    this.lastSignInTime,
 | 
					 | 
				
			||||||
    this.creationTime,
 | 
					 | 
				
			||||||
    this.isNewUser,
 | 
					 | 
				
			||||||
    this.email,
 | 
					 | 
				
			||||||
    this.phoneNumber,
 | 
					 | 
				
			||||||
    this.photoURL,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  AccountModel copyWith({
 | 
					 | 
				
			||||||
    String? uid,
 | 
					 | 
				
			||||||
    String? email,
 | 
					 | 
				
			||||||
    DateTime? creationTime,
 | 
					 | 
				
			||||||
    bool? emailVerified,
 | 
					 | 
				
			||||||
    bool? isAnonymous,
 | 
					 | 
				
			||||||
    bool? isNewUser,
 | 
					 | 
				
			||||||
    DateTime? lastSignInTime,
 | 
					 | 
				
			||||||
    String? phoneNumber,
 | 
					 | 
				
			||||||
    String? photoURL,
 | 
					 | 
				
			||||||
    String? providerId,
 | 
					 | 
				
			||||||
  }) => AccountModel(
 | 
					 | 
				
			||||||
      uid: uid ?? this.uid,
 | 
					 | 
				
			||||||
      email: email ?? this.email,
 | 
					 | 
				
			||||||
      creationTime: creationTime ?? this.creationTime,
 | 
					 | 
				
			||||||
      emailVerified: emailVerified ?? this.emailVerified,
 | 
					 | 
				
			||||||
      isAnonymous: isAnonymous ?? this.isAnonymous,
 | 
					 | 
				
			||||||
      isNewUser: isNewUser ?? this.isNewUser,
 | 
					 | 
				
			||||||
      lastSignInTime: lastSignInTime ?? this.lastSignInTime,
 | 
					 | 
				
			||||||
      phoneNumber: phoneNumber ?? this.phoneNumber,
 | 
					 | 
				
			||||||
      photoURL: photoURL ?? this.photoURL,
 | 
					 | 
				
			||||||
      providerId: providerId ?? this.providerId,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -1,4 +1,3 @@
 | 
				
			|||||||
// ignore_for_file: public_member_api_docs, sort_constructors_first
 | 
					 | 
				
			||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					// Copyright (C) 2022 WYATT GROUP
 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					// Please see the AUTHORS file for details.
 | 
				
			||||||
//
 | 
					//
 | 
				
			||||||
@ -25,12 +24,4 @@ class AccountWrapperModel<T> extends AccountWrapper<T> {
 | 
				
			|||||||
  final T? data;
 | 
					  final T? data;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  AccountWrapperModel(this.account, this.data);
 | 
					  AccountWrapperModel(this.account, this.data);
 | 
				
			||||||
 | 
					 | 
				
			||||||
  AccountWrapperModel<T> copyWith({
 | 
					 | 
				
			||||||
    Account? account,
 | 
					 | 
				
			||||||
    T? data,
 | 
					 | 
				
			||||||
  }) => AccountWrapperModel<T>(
 | 
					 | 
				
			||||||
      account ?? this.account,
 | 
					 | 
				
			||||||
      data ?? this.data,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -132,7 +132,7 @@ class AuthenticationRepositoryImpl<T extends Object>
 | 
				
			|||||||
            );
 | 
					            );
 | 
				
			||||||
            await dataResult.foldAsync(
 | 
					            await dataResult.foldAsync(
 | 
				
			||||||
              _authenticationLocalDataSource.storeData,
 | 
					              _authenticationLocalDataSource.storeData,
 | 
				
			||||||
              (error) async => error,
 | 
					              (error) => throw error,
 | 
				
			||||||
            );
 | 
					            );
 | 
				
			||||||
          }
 | 
					          }
 | 
				
			||||||
          return account;
 | 
					          return account;
 | 
				
			||||||
@ -208,70 +208,4 @@ class AuthenticationRepositoryImpl<T extends Object>
 | 
				
			|||||||
          AccountWrapperModel<T>(account, null),
 | 
					          AccountWrapperModel<T>(account, null),
 | 
				
			||||||
        );
 | 
					        );
 | 
				
			||||||
      });
 | 
					      });
 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureResult<void> confirmPasswordReset({
 | 
					 | 
				
			||||||
    required String code,
 | 
					 | 
				
			||||||
    required String newPassword,
 | 
					 | 
				
			||||||
  }) =>
 | 
					 | 
				
			||||||
      Result.tryCatchAsync<void, AppException, AppException>(
 | 
					 | 
				
			||||||
        () async {
 | 
					 | 
				
			||||||
          await _authenticationRemoteDataSource.confirmPasswordReset(
 | 
					 | 
				
			||||||
            code: code,
 | 
					 | 
				
			||||||
            newPassword: newPassword,
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        (error) => error,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureResult<void> sendEmailVerification() =>
 | 
					 | 
				
			||||||
      Result.tryCatchAsync<void, AppException, AppException>(
 | 
					 | 
				
			||||||
        () async {
 | 
					 | 
				
			||||||
          await _authenticationRemoteDataSource.sendEmailVerification();
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        (error) => error,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureResult<void> sendPasswordResetEmail({required String email}) =>
 | 
					 | 
				
			||||||
      Result.tryCatchAsync<void, AppException, AppException>(
 | 
					 | 
				
			||||||
        () async {
 | 
					 | 
				
			||||||
          await _authenticationRemoteDataSource.sendPasswordResetEmail(
 | 
					 | 
				
			||||||
            email: email,
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        (error) => error,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureResult<Account> signInAnonymously() =>
 | 
					 | 
				
			||||||
      Result.tryCatchAsync<Account, AppException, AppException>(
 | 
					 | 
				
			||||||
        () async {
 | 
					 | 
				
			||||||
          final account =
 | 
					 | 
				
			||||||
              await _authenticationRemoteDataSource.signInAnonymously();
 | 
					 | 
				
			||||||
          return account;
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        (error) => error,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureResult<bool> verifyPasswordResetCode({required String code}) =>
 | 
					 | 
				
			||||||
      Result.tryCatchAsync<bool, AppException, AppException>(
 | 
					 | 
				
			||||||
        () async {
 | 
					 | 
				
			||||||
          final response = await _authenticationRemoteDataSource
 | 
					 | 
				
			||||||
              .verifyPasswordResetCode(code: code);
 | 
					 | 
				
			||||||
          return response;
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        (error) => error,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureResult<void> refresh() =>
 | 
					 | 
				
			||||||
      Result.tryCatchAsync<void, AppException, AppException>(
 | 
					 | 
				
			||||||
        () async {
 | 
					 | 
				
			||||||
          await _authenticationRemoteDataSource.refresh();
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        (error) => error,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -30,22 +30,7 @@ abstract class AuthenticationRemoteDataSource extends BaseRemoteDataSource {
 | 
				
			|||||||
 | 
					
 | 
				
			||||||
  Future<void> signOut();
 | 
					  Future<void> signOut();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  Future<void> refresh();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  Stream<Account?> streamAccount();
 | 
					  Stream<Account?> streamAccount();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  Future<String> getIdentityToken();
 | 
					  Future<String> getIdentityToken();
 | 
				
			||||||
 | 
					 | 
				
			||||||
  Future<void> sendEmailVerification();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  Future<void> sendPasswordResetEmail({required String email});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  Future<void> confirmPasswordReset({
 | 
					 | 
				
			||||||
    required String code,
 | 
					 | 
				
			||||||
    required String newPassword,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  Future<bool> verifyPasswordResetCode({required String code});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  Future<Account> signInAnonymously();
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -14,10 +14,9 @@
 | 
				
			|||||||
// You should have received a copy of the GNU General Public License
 | 
					// You should have received a copy of the GNU General Public License
 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import 'package:equatable/equatable.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
					import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
abstract class Account extends Equatable implements Entity {
 | 
					abstract class Account extends Entity {
 | 
				
			||||||
  /// The user's unique ID.
 | 
					  /// The user's unique ID.
 | 
				
			||||||
  String get uid;
 | 
					  String get uid;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
@ -25,59 +24,4 @@ abstract class Account extends Equatable implements Entity {
 | 
				
			|||||||
  ///
 | 
					  ///
 | 
				
			||||||
  /// Will be `null` if signing in anonymously.
 | 
					  /// Will be `null` if signing in anonymously.
 | 
				
			||||||
  String? get email;
 | 
					  String? get email;
 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// Returns whether the users email address has been verified.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// To send a verification email, see `SendEmailVerification`.
 | 
					 | 
				
			||||||
  bool get emailVerified;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// Returns whether the user is a anonymous.
 | 
					 | 
				
			||||||
  bool get isAnonymous;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// Returns the users account creation time.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// When this account was created as dictated by the server clock.
 | 
					 | 
				
			||||||
  DateTime? get creationTime;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// When the user last signed in as dictated by the server clock.
 | 
					 | 
				
			||||||
  DateTime? get lastSignInTime;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// Returns the users phone number.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// This property will be `null` if the user has not signed in or been has
 | 
					 | 
				
			||||||
  /// their phone number linked.
 | 
					 | 
				
			||||||
  String? get phoneNumber;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// Returns a photo URL for the user.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// This property will be populated if the user has signed in or been linked
 | 
					 | 
				
			||||||
  /// with a 3rd party OAuth provider (such as Google).
 | 
					 | 
				
			||||||
  String? get photoURL;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// The provider ID for the user.
 | 
					 | 
				
			||||||
  String get providerId;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// Whether the user account has been recently created.
 | 
					 | 
				
			||||||
  bool? get isNewUser;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  List<Object?> get props => [
 | 
					 | 
				
			||||||
        uid,
 | 
					 | 
				
			||||||
        email,
 | 
					 | 
				
			||||||
        emailVerified,
 | 
					 | 
				
			||||||
        isAnonymous,
 | 
					 | 
				
			||||||
        creationTime,
 | 
					 | 
				
			||||||
        lastSignInTime,
 | 
					 | 
				
			||||||
        phoneNumber,
 | 
					 | 
				
			||||||
        photoURL,
 | 
					 | 
				
			||||||
        providerId,
 | 
					 | 
				
			||||||
        isNewUser,
 | 
					 | 
				
			||||||
      ];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  String toString() => 'AccountModel(uid: $uid, email: $email, '
 | 
					 | 
				
			||||||
      'creationTime: $creationTime, emailVerified: $emailVerified, '
 | 
					 | 
				
			||||||
      'isAnonymous: $isAnonymous, isNewUser: $isNewUser, lastSignInTime: '
 | 
					 | 
				
			||||||
      '$lastSignInTime, phoneNumber: $phoneNumber, photoURL: $photoURL, '
 | 
					 | 
				
			||||||
      'providerId: $providerId)';
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -14,17 +14,10 @@
 | 
				
			|||||||
// You should have received a copy of the GNU General Public License
 | 
					// You should have received a copy of the GNU General Public License
 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import 'package:equatable/equatable.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
					import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/domain/entities/account.dart';
 | 
					import 'package:wyatt_authentication_bloc/src/domain/entities/account.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
abstract class AccountWrapper<T> extends Equatable implements Entity {
 | 
					abstract class AccountWrapper<T> extends Entity {
 | 
				
			||||||
  Account? get account;
 | 
					  Account? get account;
 | 
				
			||||||
  T? get data;
 | 
					  T? get data;
 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  List<Object?> get props => [account, data];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  String toString() => 'AccountWrapper($account, data: $data)';
 | 
					 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
				
			|||||||
@ -22,83 +22,18 @@ import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
				
			|||||||
abstract class AuthenticationRepository<T> extends BaseRepository {
 | 
					abstract class AuthenticationRepository<T> extends BaseRepository {
 | 
				
			||||||
  FormRepository get formRepository;
 | 
					  FormRepository get formRepository;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /// {@template signup}
 | 
					 | 
				
			||||||
  /// Creates a new user with the provided [email] and [password].
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Returns the newly created user's unique identifier.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Throws a SignUpWithEmailAndPasswordFailureInterface if
 | 
					 | 
				
			||||||
  /// an exception occurs.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  FutureResult<Account> signUp({
 | 
					  FutureResult<Account> signUp({
 | 
				
			||||||
    required String email,
 | 
					    required String email,
 | 
				
			||||||
    required String password,
 | 
					    required String password,
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /// {@template send_email_verification}
 | 
					 | 
				
			||||||
  /// Sends verification email to the account email.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Throws a SendEmailVerificationFailureInterface if an exception occurs.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  FutureResult<void> sendEmailVerification();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// {@template send_password_reset_email}
 | 
					 | 
				
			||||||
  /// Sends a password reset email to the provided [email].
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Throws a SendPasswordResetEmailFailureInterface if an exception occurs.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  FutureResult<void> sendPasswordResetEmail({required String email});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// {@template confirm_password_reset}
 | 
					 | 
				
			||||||
  /// Confirms the password reset with the provided [newPassword] and [code].
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Throws a ConfirmPasswordResetFailureInterface if an exception occurs.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  FutureResult<void> confirmPasswordReset({
 | 
					 | 
				
			||||||
    required String code,
 | 
					 | 
				
			||||||
    required String newPassword,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// {@template verify_password_reset_code}
 | 
					 | 
				
			||||||
  /// Verify password reset code.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Throws a VerifyPasswordResetCodeFailureInterface if an exception occurs.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  FutureResult<bool> verifyPasswordResetCode({required String code});
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// {@template signin_anom}
 | 
					 | 
				
			||||||
  /// Sign in anonymously.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Throws a SignInAnonymouslyFailureInterface if an exception occurs.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  FutureResult<Account> signInAnonymously();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// {@template signin_pwd}
 | 
					 | 
				
			||||||
  /// Signs in with the provided [email] and [password].
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Throws a SignInWithEmailAndPasswordFailureInterface if
 | 
					 | 
				
			||||||
  /// an exception occurs.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  FutureResult<Account> signInWithEmailAndPassword({
 | 
					  FutureResult<Account> signInWithEmailAndPassword({
 | 
				
			||||||
    required String email,
 | 
					    required String email,
 | 
				
			||||||
    required String password,
 | 
					    required String password,
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  /// {@template signout}
 | 
					 | 
				
			||||||
  /// Signs out the current user.
 | 
					 | 
				
			||||||
  /// It also clears the cache and the associated data.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  FutureResult<void> signOut();
 | 
					  FutureResult<void> signOut();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  FutureResult<void> refresh();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  /// {@template stream_account}
 | 
					 | 
				
			||||||
  /// Stream of [AccountWrapper] which will emit the current account when
 | 
					 | 
				
			||||||
  /// the authentication state changes.
 | 
					 | 
				
			||||||
  ///
 | 
					 | 
				
			||||||
  /// Emits [AccountWrapper] with null [Account] if the user is not 
 | 
					 | 
				
			||||||
  /// authenticated.
 | 
					 | 
				
			||||||
  /// {@endtemplate}
 | 
					 | 
				
			||||||
  Stream<FutureResult<AccountWrapper<T>>> streamAccount();
 | 
					  Stream<FutureResult<AccountWrapper<T>>> streamAccount();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  FutureResult<String> getIdentityToken();
 | 
					  FutureResult<String> getIdentityToken();
 | 
				
			||||||
 | 
				
			|||||||
@ -40,16 +40,16 @@ class AuthenticationCubit<Extra> extends Cubit<AuthenticationState<Extra>> {
 | 
				
			|||||||
      accountFutureResult.fold(
 | 
					      accountFutureResult.fold(
 | 
				
			||||||
        (value) {
 | 
					        (value) {
 | 
				
			||||||
          if (value.account.isNotNull) {
 | 
					          if (value.account.isNotNull) {
 | 
				
			||||||
            emit(AuthenticationState<Extra>.authenticated(value));
 | 
					            emit(AuthenticationState.authenticated(value));
 | 
				
			||||||
            return;
 | 
					            return;
 | 
				
			||||||
          }
 | 
					          }
 | 
				
			||||||
          _authenticationRepository.destroyCache();
 | 
					          _authenticationRepository.destroyCache();
 | 
				
			||||||
          emit(AuthenticationState<Extra>.unauthenticated());
 | 
					          emit(const AuthenticationState.unauthenticated());
 | 
				
			||||||
          return;
 | 
					          return;
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        (error) {
 | 
					        (error) {
 | 
				
			||||||
          _authenticationRepository.destroyCache();
 | 
					          _authenticationRepository.destroyCache();
 | 
				
			||||||
          emit(AuthenticationState<Extra>.unauthenticated());
 | 
					          emit(const AuthenticationState.unauthenticated());
 | 
				
			||||||
          return;
 | 
					          return;
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
				
			|||||||
@ -35,7 +35,7 @@ class AuthenticationState<Extra> extends Equatable {
 | 
				
			|||||||
      : this._(status: AuthenticationStatus.unauthenticated);
 | 
					      : this._(status: AuthenticationStatus.unauthenticated);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  List<Object?> get props => [status, accountWrapper];
 | 
					  List<Object?> get props => [status];
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  @override
 | 
					  @override
 | 
				
			||||||
  String toString() =>
 | 
					  String toString() =>
 | 
				
			||||||
 | 
				
			|||||||
@ -1,65 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:flutter/material.dart';
 | 
					 | 
				
			||||||
import 'package:flutter_bloc/flutter_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/features/email_verification/cubit/email_verification_cubit.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class EmailVerificationBuilder<Extra> extends StatelessWidget {
 | 
					 | 
				
			||||||
  const EmailVerificationBuilder({
 | 
					 | 
				
			||||||
    required this.verified,
 | 
					 | 
				
			||||||
    required this.notVerified,
 | 
					 | 
				
			||||||
    required this.onError,
 | 
					 | 
				
			||||||
    this.customBuilder,
 | 
					 | 
				
			||||||
    super.key,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  final Widget Function(BuildContext context) verified;
 | 
					 | 
				
			||||||
  final Widget Function(BuildContext context) notVerified;
 | 
					 | 
				
			||||||
  final Widget Function(
 | 
					 | 
				
			||||||
    BuildContext context,
 | 
					 | 
				
			||||||
    FormStatus status,
 | 
					 | 
				
			||||||
    String? errorMessage,
 | 
					 | 
				
			||||||
  ) onError;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  final Widget Function(BuildContext context, EmailVerificationState)?
 | 
					 | 
				
			||||||
      customBuilder;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Widget build(BuildContext context) =>
 | 
					 | 
				
			||||||
      BlocBuilder<EmailVerificationCubit<Extra>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        builder: (context, state) {
 | 
					 | 
				
			||||||
          if (customBuilder != null) {
 | 
					 | 
				
			||||||
            return customBuilder!(context, state);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          if (state.status.isSubmissionFailure ||
 | 
					 | 
				
			||||||
              state.status.isSubmissionCanceled) {
 | 
					 | 
				
			||||||
            return onError(
 | 
					 | 
				
			||||||
              context,
 | 
					 | 
				
			||||||
              state.status,
 | 
					 | 
				
			||||||
              state.errorMessage,
 | 
					 | 
				
			||||||
            );
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          if (state.isVerified) {
 | 
					 | 
				
			||||||
            return verified(context);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
          return notVerified(context);
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,77 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'dart:async';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:equatable/equatable.dart';
 | 
					 | 
				
			||||||
import 'package:flutter_bloc/flutter_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/domain/repositories/repositories.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
part 'email_verification_state.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class EmailVerificationCubit<Extra> extends Cubit<EmailVerificationState> {
 | 
					 | 
				
			||||||
  final AuthenticationRepository<Extra> _authenticationRepository;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  EmailVerificationCubit({
 | 
					 | 
				
			||||||
    required AuthenticationRepository<Extra> authenticationRepository,
 | 
					 | 
				
			||||||
  })  : _authenticationRepository = authenticationRepository,
 | 
					 | 
				
			||||||
        super(const EmailVerificationState());
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  FutureOr<void> sendEmailVerification() async {
 | 
					 | 
				
			||||||
    emit(state.copyWith(status: FormStatus.submissionInProgress));
 | 
					 | 
				
			||||||
    final response = await _authenticationRepository.sendEmailVerification();
 | 
					 | 
				
			||||||
    emit(
 | 
					 | 
				
			||||||
      response.fold(
 | 
					 | 
				
			||||||
        (value) => state.copyWith(status: FormStatus.submissionSuccess),
 | 
					 | 
				
			||||||
        (error) => state.copyWith(
 | 
					 | 
				
			||||||
          errorMessage: error.message,
 | 
					 | 
				
			||||||
          status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  FutureOr<void> checkEmailVerification() async {
 | 
					 | 
				
			||||||
    emit(state.copyWith(status: FormStatus.submissionInProgress));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    final refresh = await _authenticationRepository.refresh();
 | 
					 | 
				
			||||||
    if (refresh.isErr) {
 | 
					 | 
				
			||||||
      final refreshError = refresh.err!;
 | 
					 | 
				
			||||||
      emit(
 | 
					 | 
				
			||||||
        EmailVerificationState(
 | 
					 | 
				
			||||||
          errorMessage: refreshError.message,
 | 
					 | 
				
			||||||
          status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
      return;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    final currentAccount = await _authenticationRepository.getAccount();
 | 
					 | 
				
			||||||
    emit(
 | 
					 | 
				
			||||||
      currentAccount.fold(
 | 
					 | 
				
			||||||
        (value) => state.copyWith(
 | 
					 | 
				
			||||||
          isVerified: value.emailVerified,
 | 
					 | 
				
			||||||
          status: FormStatus.submissionSuccess,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        (error) => EmailVerificationState(
 | 
					 | 
				
			||||||
          errorMessage: error.message,
 | 
					 | 
				
			||||||
          status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,49 +0,0 @@
 | 
				
			|||||||
// ignore_for_file: public_member_api_docs, sort_constructors_first
 | 
					 | 
				
			||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
part of 'email_verification_cubit.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class EmailVerificationState extends Equatable {
 | 
					 | 
				
			||||||
  final FormStatus status;
 | 
					 | 
				
			||||||
  final bool isVerified;
 | 
					 | 
				
			||||||
  final String? errorMessage;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  const EmailVerificationState({
 | 
					 | 
				
			||||||
    this.isVerified = false,
 | 
					 | 
				
			||||||
    this.status = FormStatus.pure,
 | 
					 | 
				
			||||||
    this.errorMessage,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  EmailVerificationState copyWith({
 | 
					 | 
				
			||||||
    FormStatus? status,
 | 
					 | 
				
			||||||
    bool? isVerified,
 | 
					 | 
				
			||||||
    String? errorMessage,
 | 
					 | 
				
			||||||
  }) =>
 | 
					 | 
				
			||||||
      EmailVerificationState(
 | 
					 | 
				
			||||||
        status: status ?? this.status,
 | 
					 | 
				
			||||||
        isVerified: isVerified ?? this.isVerified,
 | 
					 | 
				
			||||||
        errorMessage: errorMessage ?? this.errorMessage,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  List<Object?> get props => [status, isVerified, errorMessage];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  String toString() => 'EmailVerificationState(status: ${status.name} '
 | 
					 | 
				
			||||||
      '${(errorMessage != null) ? " [$errorMessage]" : ""}, '
 | 
					 | 
				
			||||||
      'isVerified: $isVerified)';
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,18 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export 'builder/email_verification_builder.dart';
 | 
					 | 
				
			||||||
export 'cubit/email_verification_cubit.dart';
 | 
					 | 
				
			||||||
@ -15,7 +15,5 @@
 | 
				
			|||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export 'authentication/authentication.dart';
 | 
					export 'authentication/authentication.dart';
 | 
				
			||||||
export 'email_verification/email_verification.dart';
 | 
					 | 
				
			||||||
export 'password_reset/password_reset.dart';
 | 
					 | 
				
			||||||
export 'sign_in/sign_in.dart';
 | 
					export 'sign_in/sign_in.dart';
 | 
				
			||||||
export 'sign_up/sign_up.dart';
 | 
					export 'sign_up/sign_up.dart';
 | 
				
			||||||
 | 
				
			|||||||
@ -1,138 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'dart:async';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/core/constants/form_field.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/core/constants/form_name.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/domain/repositories/authentication_repository.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
part 'password_reset_state.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class PasswordResetCubit<Extra> extends FormDataCubit<PasswordResetState> {
 | 
					 | 
				
			||||||
  final AuthenticationRepository<Extra> _authenticationRepository;
 | 
					 | 
				
			||||||
  FormRepository get _formRepository =>
 | 
					 | 
				
			||||||
      _authenticationRepository.formRepository;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  PasswordResetCubit({
 | 
					 | 
				
			||||||
    required AuthenticationRepository<Extra> authenticationRepository,
 | 
					 | 
				
			||||||
  })  : _authenticationRepository = authenticationRepository,
 | 
					 | 
				
			||||||
        super(
 | 
					 | 
				
			||||||
          PasswordResetState(
 | 
					 | 
				
			||||||
            form: authenticationRepository.formRepository
 | 
					 | 
				
			||||||
                .accessForm(AuthFormName.passwordResetForm),
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
        );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  String get formName => AuthFormName.passwordResetForm;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  void emailChanged(String value) {
 | 
					 | 
				
			||||||
    final Email email = Email.dirty(value);
 | 
					 | 
				
			||||||
    dataChanged(AuthFormField.email, email);
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureOr<void> dataChanged<Value>(
 | 
					 | 
				
			||||||
    String key,
 | 
					 | 
				
			||||||
    FormInputValidator<Value, ValidationError> dirtyValue,
 | 
					 | 
				
			||||||
  ) {
 | 
					 | 
				
			||||||
    final form = _formRepository.accessForm(formName).clone();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    try {
 | 
					 | 
				
			||||||
      form.updateValidator(key, dirtyValue);
 | 
					 | 
				
			||||||
      _formRepository.updateForm(form);
 | 
					 | 
				
			||||||
    } catch (e) {
 | 
					 | 
				
			||||||
      rethrow;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    emit(
 | 
					 | 
				
			||||||
      state.copyWith(form: form, status: form.validate()),
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureOr<void> reset() {
 | 
					 | 
				
			||||||
    final form = state.form.reset();
 | 
					 | 
				
			||||||
    _formRepository.updateForm(form);
 | 
					 | 
				
			||||||
    emit(
 | 
					 | 
				
			||||||
      state.copyWith(form: form, status: form.validate()),
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureOr<void> submit() async {
 | 
					 | 
				
			||||||
    if (!state.status.isValidated) {
 | 
					 | 
				
			||||||
      return;
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    emit(state.copyWith(status: FormStatus.submissionInProgress));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    final form = _formRepository.accessForm(formName);
 | 
					 | 
				
			||||||
    final email = form.valueOf<String?>(AuthFormField.email);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    if (email.isNullOrEmpty) {
 | 
					 | 
				
			||||||
      emit(
 | 
					 | 
				
			||||||
        state.copyWith(
 | 
					 | 
				
			||||||
          errorMessage: 'An error occured while retrieving data from the form.',
 | 
					 | 
				
			||||||
          status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    final response = await _authenticationRepository.sendPasswordResetEmail(
 | 
					 | 
				
			||||||
      email: email!,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    emit(
 | 
					 | 
				
			||||||
      response.fold(
 | 
					 | 
				
			||||||
        (value) => state.copyWith(status: FormStatus.submissionSuccess),
 | 
					 | 
				
			||||||
        (error) => state.copyWith(
 | 
					 | 
				
			||||||
          errorMessage: error.message,
 | 
					 | 
				
			||||||
          status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureOr<void> update(
 | 
					 | 
				
			||||||
    WyattForm form, {
 | 
					 | 
				
			||||||
    SetOperation operation = SetOperation.replace,
 | 
					 | 
				
			||||||
  }) {
 | 
					 | 
				
			||||||
    final WyattForm current = _formRepository.accessForm(formName).clone();
 | 
					 | 
				
			||||||
    final WyattForm newForm = operation.operation.call(current, form);
 | 
					 | 
				
			||||||
    _formRepository.updateForm(newForm);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    emit(
 | 
					 | 
				
			||||||
      state.copyWith(
 | 
					 | 
				
			||||||
        form: newForm,
 | 
					 | 
				
			||||||
        status: newForm.validate(),
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  FutureOr<void> validate() {
 | 
					 | 
				
			||||||
    emit(
 | 
					 | 
				
			||||||
      state.copyWith(
 | 
					 | 
				
			||||||
        status: _formRepository.accessForm(formName).validate(),
 | 
					 | 
				
			||||||
      ),
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
  }
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,45 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
part of 'password_reset_cubit.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class PasswordResetState extends FormDataState {
 | 
					 | 
				
			||||||
  Email get email => form.validatorOf(AuthFormField.email);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  const PasswordResetState({
 | 
					 | 
				
			||||||
    required super.form,
 | 
					 | 
				
			||||||
    super.status = FormStatus.pure,
 | 
					 | 
				
			||||||
    super.errorMessage,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  PasswordResetState copyWith({
 | 
					 | 
				
			||||||
    WyattForm? form,
 | 
					 | 
				
			||||||
    FormStatus? status,
 | 
					 | 
				
			||||||
    String? errorMessage,
 | 
					 | 
				
			||||||
  }) =>
 | 
					 | 
				
			||||||
      PasswordResetState(
 | 
					 | 
				
			||||||
        form: form ?? this.form,
 | 
					 | 
				
			||||||
        status: status ?? this.status,
 | 
					 | 
				
			||||||
        errorMessage: errorMessage ?? this.errorMessage,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  List<Object> get props => [email, status];
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  String toString() => 'PasswordResetState(status: ${status.name} '
 | 
					 | 
				
			||||||
      '${(errorMessage != null) ? " [$errorMessage]" : ""}, $form)';
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,17 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
export 'cubit/password_reset_cubit.dart';
 | 
					 | 
				
			||||||
@ -1,66 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:flutter/material.dart';
 | 
					 | 
				
			||||||
import 'package:flutter_bloc/flutter_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/features/sign_in/sign_in.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class SignInListener<Extra> extends StatelessWidget {
 | 
					 | 
				
			||||||
  const SignInListener({
 | 
					 | 
				
			||||||
    required this.child,
 | 
					 | 
				
			||||||
    this.onProgress,
 | 
					 | 
				
			||||||
    this.onSuccess,
 | 
					 | 
				
			||||||
    this.onError,
 | 
					 | 
				
			||||||
    this.customBuilder,
 | 
					 | 
				
			||||||
    super.key,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  final void Function(BuildContext context)? onProgress;
 | 
					 | 
				
			||||||
  final void Function(BuildContext context)? onSuccess;
 | 
					 | 
				
			||||||
  final void Function(
 | 
					 | 
				
			||||||
    BuildContext context,
 | 
					 | 
				
			||||||
    FormStatus status,
 | 
					 | 
				
			||||||
    String? errorMessage,
 | 
					 | 
				
			||||||
  )? onError;
 | 
					 | 
				
			||||||
  final void Function(BuildContext context, SignInState state)? customBuilder;
 | 
					 | 
				
			||||||
  final Widget child;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Widget build(BuildContext context) =>
 | 
					 | 
				
			||||||
      BlocListener<SignInCubit<Extra>, SignInState>(
 | 
					 | 
				
			||||||
        listener: (context, state) {
 | 
					 | 
				
			||||||
          if (customBuilder != null) {
 | 
					 | 
				
			||||||
            return customBuilder!(context, state);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          if (onSuccess != null &&
 | 
					 | 
				
			||||||
              state.status == FormStatus.submissionSuccess) {
 | 
					 | 
				
			||||||
            return onSuccess!(context);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
          if (onProgress != null &&
 | 
					 | 
				
			||||||
              state.status == FormStatus.submissionInProgress) {
 | 
					 | 
				
			||||||
            return onProgress!(context);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
          if (onError != null &&
 | 
					 | 
				
			||||||
              (state.status == FormStatus.submissionCanceled ||
 | 
					 | 
				
			||||||
                  state.status == FormStatus.submissionFailure)) {
 | 
					 | 
				
			||||||
            return onError!(context, state.status, state.errorMessage);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        child: child,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -15,4 +15,3 @@
 | 
				
			|||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export 'cubit/sign_in_cubit.dart';
 | 
					export 'cubit/sign_in_cubit.dart';
 | 
				
			||||||
export 'listener/sign_in_listener.dart';
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -1,66 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:flutter/material.dart';
 | 
					 | 
				
			||||||
import 'package:flutter_bloc/flutter_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/src/features/sign_up/cubit/sign_up_cubit.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class SignUpListener<Extra> extends StatelessWidget {
 | 
					 | 
				
			||||||
  const SignUpListener({
 | 
					 | 
				
			||||||
    required this.child,
 | 
					 | 
				
			||||||
    this.onProgress,
 | 
					 | 
				
			||||||
    this.onSuccess,
 | 
					 | 
				
			||||||
    this.onError,
 | 
					 | 
				
			||||||
    this.customBuilder,
 | 
					 | 
				
			||||||
    super.key,
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  final void Function(BuildContext context)? onProgress;
 | 
					 | 
				
			||||||
  final void Function(BuildContext context)? onSuccess;
 | 
					 | 
				
			||||||
  final void Function(
 | 
					 | 
				
			||||||
    BuildContext context,
 | 
					 | 
				
			||||||
    FormStatus status,
 | 
					 | 
				
			||||||
    String? errorMessage,
 | 
					 | 
				
			||||||
  )? onError;
 | 
					 | 
				
			||||||
  final void Function(BuildContext context, SignUpState state)? customBuilder;
 | 
					 | 
				
			||||||
  final Widget child;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  @override
 | 
					 | 
				
			||||||
  Widget build(BuildContext context) =>
 | 
					 | 
				
			||||||
      BlocListener<SignUpCubit<Extra>, SignUpState>(
 | 
					 | 
				
			||||||
        listener: (context, state) {
 | 
					 | 
				
			||||||
          if (customBuilder != null) {
 | 
					 | 
				
			||||||
            return customBuilder!(context, state);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
          if (onSuccess != null &&
 | 
					 | 
				
			||||||
              state.status == FormStatus.submissionSuccess) {
 | 
					 | 
				
			||||||
            return onSuccess!(context);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
          if (onProgress != null &&
 | 
					 | 
				
			||||||
              state.status == FormStatus.submissionInProgress) {
 | 
					 | 
				
			||||||
            return onProgress!(context);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
          if (onError != null &&
 | 
					 | 
				
			||||||
              (state.status == FormStatus.submissionCanceled ||
 | 
					 | 
				
			||||||
                  state.status == FormStatus.submissionFailure)) {
 | 
					 | 
				
			||||||
            return onError!(context, state.status, state.errorMessage);
 | 
					 | 
				
			||||||
          }
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        child: child,
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -15,4 +15,3 @@
 | 
				
			|||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
				
			||||||
 | 
					
 | 
				
			||||||
export 'cubit/sign_up_cubit.dart';
 | 
					export 'cubit/sign_up_cubit.dart';
 | 
				
			||||||
export 'listener/sign_up_listener.dart';
 | 
					 | 
				
			||||||
 | 
				
			|||||||
@ -23,17 +23,22 @@ dependencies:
 | 
				
			|||||||
  twitter_login: ^4.2.3
 | 
					  twitter_login: ^4.2.3
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  wyatt_form_bloc:
 | 
					  wyatt_form_bloc:
 | 
				
			||||||
    hosted: https://git.wyatt-studio.fr/api/packages/Wyatt-FOSS/pub/
 | 
					    git:
 | 
				
			||||||
    version: 0.1.0+1
 | 
					      url: https://git.wyatt-studio.fr/Wyatt-FOSS/wyatt-packages
 | 
				
			||||||
 | 
					      ref: wyatt_form_bloc-v0.1.0+1
 | 
				
			||||||
 | 
					      path: packages/wyatt_form_bloc
 | 
				
			||||||
  
 | 
					  
 | 
				
			||||||
  wyatt_architecture:
 | 
					  wyatt_architecture:
 | 
				
			||||||
    hosted: https://git.wyatt-studio.fr/api/packages/Wyatt-FOSS/pub/
 | 
					    git:
 | 
				
			||||||
    version: 0.0.2
 | 
					      url: https://git.wyatt-studio.fr/Wyatt-FOSS/wyatt-packages
 | 
				
			||||||
 | 
					      ref: wyatt_architecture-v0.0.2-dev.0
 | 
				
			||||||
 | 
					      path: packages/wyatt_architecture
 | 
				
			||||||
  
 | 
					  
 | 
				
			||||||
  wyatt_type_utils:
 | 
					  wyatt_type_utils:
 | 
				
			||||||
    hosted: https://git.wyatt-studio.fr/api/packages/Wyatt-FOSS/pub/
 | 
					    hosted: https://git.wyatt-studio.fr/api/packages/Wyatt-FOSS/pub/
 | 
				
			||||||
    version: 0.0.3+1
 | 
					    version: 0.0.3+1
 | 
				
			||||||
    
 | 
					    
 | 
				
			||||||
 | 
					
 | 
				
			||||||
dev_dependencies:
 | 
					dev_dependencies:
 | 
				
			||||||
  flutter_test:
 | 
					  flutter_test:
 | 
				
			||||||
    sdk: flutter
 | 
					    sdk: flutter
 | 
				
			||||||
 | 
				
			|||||||
@ -17,114 +17,100 @@
 | 
				
			|||||||
import 'package:bloc_test/bloc_test.dart';
 | 
					import 'package:bloc_test/bloc_test.dart';
 | 
				
			||||||
import 'package:flutter_test/flutter_test.dart';
 | 
					import 'package:flutter_test/flutter_test.dart';
 | 
				
			||||||
import 'package:mocktail/mocktail.dart';
 | 
					import 'package:mocktail/mocktail.dart';
 | 
				
			||||||
import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
class MockAuthenticationRepository extends Mock
 | 
					class MockAuthenticationRepository extends Mock
 | 
				
			||||||
    implements AuthenticationRepository<int> {}
 | 
					    implements AuthenticationRepository {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class MockAccount extends Mock implements Account {}
 | 
					class MockUser extends Mock implements User {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void main() {
 | 
					void main() {
 | 
				
			||||||
  group('AuthenticationCubit<T>', () {
 | 
					  group('AuthenticationCubit<T>', () {
 | 
				
			||||||
    final MockAccount account = MockAccount();
 | 
					    final MockUser user = MockUser();
 | 
				
			||||||
    final AccountWrapper<int> wrapper = AccountWrapperModel(account, 10);
 | 
					    late AuthenticationRepository authenticationRepository;
 | 
				
			||||||
    late AuthenticationRepository<int> authenticationRepository;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    setUp(() {
 | 
					    setUp(() {
 | 
				
			||||||
      authenticationRepository = MockAuthenticationRepository();
 | 
					      authenticationRepository = MockAuthenticationRepository();
 | 
				
			||||||
      when(() => authenticationRepository.streamAccount()).thenAnswer(
 | 
					      when(() => authenticationRepository.user).thenAnswer(
 | 
				
			||||||
        (_) => const Stream.empty(),
 | 
					        (_) => const Stream.empty(),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					      when(() => authenticationRepository.cubitStatus).thenAnswer(
 | 
				
			||||||
 | 
					        (_) => Stream.fromIterable([AuthCubitStatus.stoped]),
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
      when(
 | 
					      when(
 | 
				
			||||||
        () => authenticationRepository.getAccount(),
 | 
					        () => authenticationRepository.currentUser,
 | 
				
			||||||
      ).thenAnswer((_) async => Ok(account));
 | 
					      ).thenReturn(user);
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    test('initial auth state is `unknown`', () {
 | 
					    test('initial auth state is `unknown`', () {
 | 
				
			||||||
      expect(
 | 
					      expect(
 | 
				
			||||||
        AuthenticationCubit<int>(
 | 
					        AuthenticationCubit<void>(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ).state,
 | 
					        ).state,
 | 
				
			||||||
        const AuthenticationState<Never>.unknown(),
 | 
					        const AuthenticationState<Never>.unknown(),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('ListenForAuthenticationChanges', () {
 | 
					    test('initial cubit status is `stoped`', () async {
 | 
				
			||||||
      blocTest<AuthenticationCubit<int>, AuthenticationState<int>>(
 | 
					      expect(
 | 
				
			||||||
        'emits authenticated when stream contains account',
 | 
					        await AuthenticationCubit<void>(
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.streamAccount()).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => Stream.fromIterable([
 | 
					 | 
				
			||||||
              Future.value(
 | 
					 | 
				
			||||||
                Ok(wrapper),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ]),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => AuthenticationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ).status,
 | 
				
			||||||
        seed: () => const AuthenticationState.unknown(),
 | 
					        AuthCubitStatus.stoped,
 | 
				
			||||||
        expect: () => [AuthenticationState<int>.authenticated(wrapper)],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<AuthenticationCubit<int>, AuthenticationState<int>>(
 | 
					 | 
				
			||||||
        'emits unauthenticated when account stream is empty',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => authenticationRepository.destroyCache(),
 | 
					 | 
				
			||||||
          ).thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.streamAccount()).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => Stream.fromIterable([
 | 
					 | 
				
			||||||
              Future.value(
 | 
					 | 
				
			||||||
                Ok(AccountWrapperModel(null, 1)),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ]),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => AuthenticationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => const AuthenticationState.unknown(),
 | 
					 | 
				
			||||||
        expect: () => [const AuthenticationState<int>.unauthenticated()],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<AuthenticationCubit<int>, AuthenticationState<int>>(
 | 
					 | 
				
			||||||
        'emits unauthenticated when there is an error in stream',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => authenticationRepository.destroyCache(),
 | 
					 | 
				
			||||||
          ).thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.streamAccount()).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => Stream.fromIterable([
 | 
					 | 
				
			||||||
              Future.value(
 | 
					 | 
				
			||||||
                Err(ServerException()),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ]),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => AuthenticationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => const AuthenticationState.unknown(),
 | 
					 | 
				
			||||||
        expect: () => [const AuthenticationState<int>.unauthenticated()],
 | 
					 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('SignOut', () {
 | 
					    group('UserChanged', () {
 | 
				
			||||||
      blocTest<AuthenticationCubit<int>, AuthenticationState<int>>(
 | 
					      blocTest<AuthenticationCubit<void>, AuthenticationState<void>>(
 | 
				
			||||||
 | 
					        'emits authenticated when user is not empty',
 | 
				
			||||||
 | 
					        setUp: () {
 | 
				
			||||||
 | 
					          when(() => user.isNotEmpty).thenReturn(true);
 | 
				
			||||||
 | 
					          when(() => authenticationRepository.user).thenAnswer(
 | 
				
			||||||
 | 
					            (_) => Stream.value(user),
 | 
				
			||||||
 | 
					          );
 | 
				
			||||||
 | 
					          when(() => authenticationRepository.cubitStatus).thenAnswer(
 | 
				
			||||||
 | 
					            (_) => Stream.value(AuthCubitStatus.started),
 | 
				
			||||||
 | 
					          );
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        build: () => AuthenticationCubit(
 | 
				
			||||||
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					        )..init(),
 | 
				
			||||||
 | 
					        seed: () => const AuthenticationState.unknown(),
 | 
				
			||||||
 | 
					        expect: () => [AuthenticationState<void>.authenticated(user, null)],
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					      blocTest<AuthenticationCubit<void>, AuthenticationState<void>>(
 | 
				
			||||||
 | 
					        'emits unauthenticated when user is empty',
 | 
				
			||||||
 | 
					        setUp: () {
 | 
				
			||||||
 | 
					          when(() => user.isEmpty).thenReturn(true);
 | 
				
			||||||
 | 
					          when(() => user.isNotEmpty).thenReturn(false);
 | 
				
			||||||
 | 
					          when(() => authenticationRepository.user).thenAnswer(
 | 
				
			||||||
 | 
					            (_) => Stream.value(user),
 | 
				
			||||||
 | 
					          );
 | 
				
			||||||
 | 
					          when(() => authenticationRepository.cubitStatus).thenAnswer(
 | 
				
			||||||
 | 
					            (_) => Stream.value(AuthCubitStatus.started),
 | 
				
			||||||
 | 
					          );
 | 
				
			||||||
 | 
					        },
 | 
				
			||||||
 | 
					        build: () => AuthenticationCubit(
 | 
				
			||||||
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					        )..init(),
 | 
				
			||||||
 | 
					        seed: () => const AuthenticationState.unknown(),
 | 
				
			||||||
 | 
					        expect: () => [const AuthenticationState<Never>.unauthenticated()],
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    group('LogoutRequested', () {
 | 
				
			||||||
 | 
					      blocTest<AuthenticationCubit<void>, AuthenticationState<void>>(
 | 
				
			||||||
        'invokes signOut',
 | 
					        'invokes signOut',
 | 
				
			||||||
        setUp: () {
 | 
					        setUp: () {
 | 
				
			||||||
          when(
 | 
					          when(
 | 
				
			||||||
            () => authenticationRepository.signOut(),
 | 
					            () => authenticationRepository.signOut(),
 | 
				
			||||||
          ).thenAnswer((_) async => const Ok(null));
 | 
					          ).thenAnswer((_) async {});
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        build: () => AuthenticationCubit(
 | 
					        build: () => AuthenticationCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.signOut(),
 | 
					        act: (cubit) => cubit.logOut(),
 | 
				
			||||||
        verify: (_) {
 | 
					        verify: (_) {
 | 
				
			||||||
          verify(() => authenticationRepository.signOut()).called(1);
 | 
					          verify(() => authenticationRepository.signOut()).called(1);
 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
 | 
				
			|||||||
@ -18,7 +18,7 @@ import 'package:flutter_test/flutter_test.dart';
 | 
				
			|||||||
import 'package:mocktail/mocktail.dart';
 | 
					import 'package:mocktail/mocktail.dart';
 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class MockAccount extends Mock implements Account {}
 | 
					class MockUser extends Mock implements User {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void main() {
 | 
					void main() {
 | 
				
			||||||
  group('AuthenticationState', () {
 | 
					  group('AuthenticationState', () {
 | 
				
			||||||
@ -27,33 +27,29 @@ void main() {
 | 
				
			|||||||
        const AuthenticationState<void> state =
 | 
					        const AuthenticationState<void> state =
 | 
				
			||||||
            AuthenticationState.unauthenticated();
 | 
					            AuthenticationState.unauthenticated();
 | 
				
			||||||
        expect(state.status, AuthenticationStatus.unauthenticated);
 | 
					        expect(state.status, AuthenticationStatus.unauthenticated);
 | 
				
			||||||
        expect(state.accountWrapper, null);
 | 
					        expect(state.user, null);
 | 
				
			||||||
      });
 | 
					      });
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('authenticated', () {
 | 
					    group('authenticated', () {
 | 
				
			||||||
      test('has correct status', () {
 | 
					      test('has correct status', () {
 | 
				
			||||||
        final MockAccount account = MockAccount();
 | 
					        final MockUser user = MockUser();
 | 
				
			||||||
        final AuthenticationState<void> state =
 | 
					        final AuthenticationState<void> state =
 | 
				
			||||||
            AuthenticationState.authenticated(
 | 
					            AuthenticationState.authenticated(user, null);
 | 
				
			||||||
          AccountWrapperModel<void>(account, null),
 | 
					 | 
				
			||||||
        );
 | 
					 | 
				
			||||||
        expect(state.status, AuthenticationStatus.authenticated);
 | 
					        expect(state.status, AuthenticationStatus.authenticated);
 | 
				
			||||||
        expect(state.accountWrapper?.account, account);
 | 
					        expect(state.user, user);
 | 
				
			||||||
      });
 | 
					      });
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('authenticated with extra data', () {
 | 
					    group('authenticated with extra data', () {
 | 
				
			||||||
      test('has correct status', () {
 | 
					      test('has correct status', () {
 | 
				
			||||||
        final MockAccount account = MockAccount();
 | 
					        final MockUser user = MockUser();
 | 
				
			||||||
        const String extra = 'AwesomeExtraData';
 | 
					        const String extra = 'AwesomeExtraData';
 | 
				
			||||||
        final AuthenticationState<String> state =
 | 
					        final AuthenticationState<String> state =
 | 
				
			||||||
            AuthenticationState.authenticated(
 | 
					            AuthenticationState.authenticated(user, extra);
 | 
				
			||||||
          AccountWrapperModel(account, extra),
 | 
					 | 
				
			||||||
        );
 | 
					 | 
				
			||||||
        expect(state.status, AuthenticationStatus.authenticated);
 | 
					        expect(state.status, AuthenticationStatus.authenticated);
 | 
				
			||||||
        expect(state.accountWrapper?.account, account);
 | 
					        expect(state.user, user);
 | 
				
			||||||
        expect(state.accountWrapper?.data, extra);
 | 
					        expect(state.extra, extra);
 | 
				
			||||||
      });
 | 
					      });
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
 | 
				
			|||||||
@ -1,249 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:bloc_test/bloc_test.dart';
 | 
					 | 
				
			||||||
import 'package:flutter_test/flutter_test.dart';
 | 
					 | 
				
			||||||
import 'package:mocktail/mocktail.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockAuthenticationRepository extends Mock
 | 
					 | 
				
			||||||
    implements AuthenticationRepository<int> {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockAuthenticationCubit extends Mock implements AuthenticationCubit<int> {
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockAccount extends Mock implements Account {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockFormRepository extends Mock implements FormRepository {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void main() {
 | 
					 | 
				
			||||||
  group('EmailVerificationCubit<T>', () {
 | 
					 | 
				
			||||||
    late MockAccount account;
 | 
					 | 
				
			||||||
    late AuthenticationRepository<int> authenticationRepository;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    setUp(() {
 | 
					 | 
				
			||||||
      authenticationRepository = MockAuthenticationRepository();
 | 
					 | 
				
			||||||
      when(
 | 
					 | 
				
			||||||
        () => authenticationRepository.getAccount(),
 | 
					 | 
				
			||||||
      ).thenAnswer((_) async => Ok(account));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      when(
 | 
					 | 
				
			||||||
        () => authenticationRepository.refresh(),
 | 
					 | 
				
			||||||
      ).thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      account = MockAccount();
 | 
					 | 
				
			||||||
      when(
 | 
					 | 
				
			||||||
        () => account.emailVerified,
 | 
					 | 
				
			||||||
      ).thenAnswer((_) => true);
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    test('initial state is `false`', () {
 | 
					 | 
				
			||||||
      expect(
 | 
					 | 
				
			||||||
        EmailVerificationCubit<int>(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ).state,
 | 
					 | 
				
			||||||
        const EmailVerificationState(),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    group('SendVerificationEmail', () {
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'invokes sendEmailVerification,',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.sendEmailVerification())
 | 
					 | 
				
			||||||
              .thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.sendEmailVerification(),
 | 
					 | 
				
			||||||
        verify: (_) {
 | 
					 | 
				
			||||||
          verify(() => authenticationRepository.sendEmailVerification())
 | 
					 | 
				
			||||||
              .called(1);
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'emits success',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.sendEmailVerification())
 | 
					 | 
				
			||||||
              .thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => const EmailVerificationState(),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.sendEmailVerification(),
 | 
					 | 
				
			||||||
        expect: () => [
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            status: FormStatus.submissionSuccess,
 | 
					 | 
				
			||||||
          )
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'emits failure',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.sendEmailVerification())
 | 
					 | 
				
			||||||
              .thenAnswer((_) async => Err(ServerException('erreur')));
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => const EmailVerificationState(),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.sendEmailVerification(),
 | 
					 | 
				
			||||||
        expect: () => [
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            errorMessage: 'erreur',
 | 
					 | 
				
			||||||
            status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
          )
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    group('CheckEmailVerification', () {
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'invokes refresh,',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => authenticationRepository.refresh(),
 | 
					 | 
				
			||||||
          ).thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.checkEmailVerification(),
 | 
					 | 
				
			||||||
        verify: (_) {
 | 
					 | 
				
			||||||
          verify(() => authenticationRepository.refresh()).called(1);
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'invokes emailVerified,',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => authenticationRepository.refresh(),
 | 
					 | 
				
			||||||
          ).thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
          when(() => account.emailVerified).thenAnswer((_) => false);
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.checkEmailVerification(),
 | 
					 | 
				
			||||||
        verify: (_) {
 | 
					 | 
				
			||||||
          verify(() => account.emailVerified).called(1);
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'emits success with true if verified',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.refresh())
 | 
					 | 
				
			||||||
              .thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => const EmailVerificationState(),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.checkEmailVerification(),
 | 
					 | 
				
			||||||
        expect: () => [
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            isVerified: true,
 | 
					 | 
				
			||||||
            status: FormStatus.submissionSuccess,
 | 
					 | 
				
			||||||
          )
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'emits success with false if not verified',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.refresh())
 | 
					 | 
				
			||||||
              .thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
          when(() => account.emailVerified).thenAnswer((_) => false);
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => const EmailVerificationState(),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.checkEmailVerification(),
 | 
					 | 
				
			||||||
        expect: () => [
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            status: FormStatus.submissionSuccess,
 | 
					 | 
				
			||||||
          )
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'emits failure on refresh error',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.refresh())
 | 
					 | 
				
			||||||
              .thenAnswer((_) async => Err(ServerException('erreur')));
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => const EmailVerificationState(),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.checkEmailVerification(),
 | 
					 | 
				
			||||||
        expect: () => [
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            errorMessage: 'erreur',
 | 
					 | 
				
			||||||
            status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
          )
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<EmailVerificationCubit<int>, EmailVerificationState>(
 | 
					 | 
				
			||||||
        'emits failure on get account error',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(() => authenticationRepository.getAccount())
 | 
					 | 
				
			||||||
              .thenAnswer((_) async => Err(ServerException('erreur')));
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => EmailVerificationCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => const EmailVerificationState(),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.checkEmailVerification(),
 | 
					 | 
				
			||||||
        expect: () => [
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          const EmailVerificationState(
 | 
					 | 
				
			||||||
            errorMessage: 'erreur',
 | 
					 | 
				
			||||||
            status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
          )
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,48 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:flutter_test/flutter_test.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void main() {
 | 
					 | 
				
			||||||
  group('EmailVerificationState', () {
 | 
					 | 
				
			||||||
    test('supports value comparisons', () {
 | 
					 | 
				
			||||||
      expect(
 | 
					 | 
				
			||||||
        const EmailVerificationState(isVerified: true),
 | 
					 | 
				
			||||||
        const EmailVerificationState(isVerified: true),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    test('returns same object when no properties are passed', () {
 | 
					 | 
				
			||||||
      expect(
 | 
					 | 
				
			||||||
        const EmailVerificationState(isVerified: true).copyWith(),
 | 
					 | 
				
			||||||
        const EmailVerificationState(isVerified: true),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    test('returns object with updated status when status is passed', () {
 | 
					 | 
				
			||||||
      expect(
 | 
					 | 
				
			||||||
        const EmailVerificationState(isVerified: true)
 | 
					 | 
				
			||||||
            .copyWith(status: FormStatus.invalid),
 | 
					 | 
				
			||||||
        const EmailVerificationState(
 | 
					 | 
				
			||||||
          isVerified: true,
 | 
					 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,335 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:bloc_test/bloc_test.dart';
 | 
					 | 
				
			||||||
import 'package:flutter_test/flutter_test.dart';
 | 
					 | 
				
			||||||
import 'package:mocktail/mocktail.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockAuthenticationRepository extends Mock
 | 
					 | 
				
			||||||
    implements AuthenticationRepository<int> {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockAuthenticationCubit extends Mock implements AuthenticationCubit<int> {
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockAccount extends Mock implements Account {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockFormRepository extends Mock implements FormRepository {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void main() {
 | 
					 | 
				
			||||||
  const String invalidEmailString = 'invalid';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  const String validEmailString = 'test@gmail.com';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  group('PasswordResetCubit', () {
 | 
					 | 
				
			||||||
    final WyattForm form = WyattFormImpl(
 | 
					 | 
				
			||||||
      [
 | 
					 | 
				
			||||||
        FormInput(AuthFormField.email, const Email.pure()),
 | 
					 | 
				
			||||||
        FormInput(AuthFormField.password, const Password.pure())
 | 
					 | 
				
			||||||
      ],
 | 
					 | 
				
			||||||
      name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    late MockFormRepository formRepository;
 | 
					 | 
				
			||||||
    late AuthenticationRepository<int> authenticationRepository;
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    setUp(() {
 | 
					 | 
				
			||||||
      authenticationRepository = MockAuthenticationRepository();
 | 
					 | 
				
			||||||
      formRepository = MockFormRepository();
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      when(
 | 
					 | 
				
			||||||
        () => authenticationRepository.sendPasswordResetEmail(
 | 
					 | 
				
			||||||
          email: any(named: 'email'),
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
      ).thenAnswer((_) async => const Ok(null));
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      when(
 | 
					 | 
				
			||||||
        () => authenticationRepository.formRepository,
 | 
					 | 
				
			||||||
      ).thenAnswer((_) => formRepository);
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      when(
 | 
					 | 
				
			||||||
        () => formRepository.accessForm(AuthFormName.passwordResetForm),
 | 
					 | 
				
			||||||
      ).thenAnswer((_) => form);
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    test('initial state is pure', () {
 | 
					 | 
				
			||||||
      expect(
 | 
					 | 
				
			||||||
        PasswordResetCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ).state,
 | 
					 | 
				
			||||||
        PasswordResetState(form: form),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    group('emailChanged', () {
 | 
					 | 
				
			||||||
      blocTest<PasswordResetCubit<int>, PasswordResetState>(
 | 
					 | 
				
			||||||
        'emits [invalid] when email is invalid',
 | 
					 | 
				
			||||||
        build: () => PasswordResetCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.emailChanged(invalidEmailString),
 | 
					 | 
				
			||||||
        expect: () => <PasswordResetState>[
 | 
					 | 
				
			||||||
          PasswordResetState(
 | 
					 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(invalidEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.invalid,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<PasswordResetCubit<int>, PasswordResetState>(
 | 
					 | 
				
			||||||
        'emits [valid] when email is valid',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.passwordResetForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.pure(),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => PasswordResetCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => PasswordResetState(
 | 
					 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.pure(),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.emailChanged(validEmailString),
 | 
					 | 
				
			||||||
        expect: () => <PasswordResetState>[
 | 
					 | 
				
			||||||
          PasswordResetState(
 | 
					 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.valid,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    group('submit', () {
 | 
					 | 
				
			||||||
      blocTest<PasswordResetCubit<int>, PasswordResetState>(
 | 
					 | 
				
			||||||
        'does nothing when status is not validated',
 | 
					 | 
				
			||||||
        build: () => PasswordResetCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					 | 
				
			||||||
        expect: () => const <PasswordResetState>[],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<PasswordResetCubit<int>, PasswordResetState>(
 | 
					 | 
				
			||||||
        'calls sendPasswordResetEmail with correct email',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.passwordResetForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => PasswordResetCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => PasswordResetState(
 | 
					 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					 | 
				
			||||||
        verify: (_) {
 | 
					 | 
				
			||||||
          verify(
 | 
					 | 
				
			||||||
            () => authenticationRepository.sendPasswordResetEmail(
 | 
					 | 
				
			||||||
              email: validEmailString,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          ).called(1);
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<PasswordResetCubit<int>, PasswordResetState>(
 | 
					 | 
				
			||||||
        'emits [submissionInProgress, submissionSuccess] '
 | 
					 | 
				
			||||||
        'when sendPasswordResetEmail succeeds',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.passwordResetForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => PasswordResetCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => PasswordResetState(
 | 
					 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					 | 
				
			||||||
        expect: () => <PasswordResetState>[
 | 
					 | 
				
			||||||
          PasswordResetState(
 | 
					 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          PasswordResetState(
 | 
					 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionSuccess,
 | 
					 | 
				
			||||||
          )
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
      blocTest<PasswordResetCubit<int>, PasswordResetState>(
 | 
					 | 
				
			||||||
        'emits [submissionInProgress, submissionFailure] '
 | 
					 | 
				
			||||||
        'when sendPasswordResetEmail fails',
 | 
					 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => authenticationRepository.sendPasswordResetEmail(
 | 
					 | 
				
			||||||
              email: any(named: 'email'),
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          ).thenAnswer((_) async => Err(ServerException()));
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.passwordResetForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => PasswordResetCubit(
 | 
					 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        seed: () => PasswordResetState(
 | 
					 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					 | 
				
			||||||
        expect: () => <PasswordResetState>[
 | 
					 | 
				
			||||||
          PasswordResetState(
 | 
					 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          PasswordResetState(
 | 
					 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionFailure,
 | 
					 | 
				
			||||||
          )
 | 
					 | 
				
			||||||
        ],
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -1,56 +0,0 @@
 | 
				
			|||||||
// Copyright (C) 2022 WYATT GROUP
 | 
					 | 
				
			||||||
// Please see the AUTHORS file for details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is free software: you can redistribute it and/or modify
 | 
					 | 
				
			||||||
// it under the terms of the GNU General Public License as published by
 | 
					 | 
				
			||||||
// the Free Software Foundation, either version 3 of the License, or
 | 
					 | 
				
			||||||
// any later version.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// This program is distributed in the hope that it will be useful,
 | 
					 | 
				
			||||||
// but WITHOUT ANY WARRANTY; without even the implied warranty of
 | 
					 | 
				
			||||||
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | 
					 | 
				
			||||||
// GNU General Public License for more details.
 | 
					 | 
				
			||||||
//
 | 
					 | 
				
			||||||
// You should have received a copy of the GNU General Public License
 | 
					 | 
				
			||||||
// along with this program. If not, see <https://www.gnu.org/licenses/>.
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
import 'package:flutter_test/flutter_test.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
void main() {
 | 
					 | 
				
			||||||
  final WyattForm form = WyattFormImpl(
 | 
					 | 
				
			||||||
    [
 | 
					 | 
				
			||||||
      FormInput(AuthFormField.email, const Email.pure()),
 | 
					 | 
				
			||||||
    ],
 | 
					 | 
				
			||||||
    name: AuthFormName.passwordResetForm,
 | 
					 | 
				
			||||||
  );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
  group('PasswordResetState', () {
 | 
					 | 
				
			||||||
    test('supports value comparisons', () {
 | 
					 | 
				
			||||||
      expect(
 | 
					 | 
				
			||||||
        PasswordResetState(
 | 
					 | 
				
			||||||
          form: form,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        PasswordResetState(form: form),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    test('returns same object when no properties are passed', () {
 | 
					 | 
				
			||||||
      expect(
 | 
					 | 
				
			||||||
        PasswordResetState(form: form).copyWith(),
 | 
					 | 
				
			||||||
        PasswordResetState(form: form),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    test('returns object with updated status when status is passed', () {
 | 
					 | 
				
			||||||
      expect(
 | 
					 | 
				
			||||||
        PasswordResetState(form: form).copyWith(status: FormStatus.invalid),
 | 
					 | 
				
			||||||
        PasswordResetState(
 | 
					 | 
				
			||||||
          form: form,
 | 
					 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					 | 
				
			||||||
  });
 | 
					 | 
				
			||||||
}
 | 
					 | 
				
			||||||
@ -17,61 +17,46 @@
 | 
				
			|||||||
import 'package:bloc_test/bloc_test.dart';
 | 
					import 'package:bloc_test/bloc_test.dart';
 | 
				
			||||||
import 'package:flutter_test/flutter_test.dart';
 | 
					import 'package:flutter_test/flutter_test.dart';
 | 
				
			||||||
import 'package:mocktail/mocktail.dart';
 | 
					import 'package:mocktail/mocktail.dart';
 | 
				
			||||||
import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
class MockAuthenticationRepository extends Mock
 | 
					class MockAuthenticationRepository extends Mock
 | 
				
			||||||
    implements AuthenticationRepository<int> {}
 | 
					    implements AuthenticationRepository {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class MockAuthenticationCubit extends Mock implements AuthenticationCubit<int> {
 | 
					class MockAuthenticationCubit extends Mock
 | 
				
			||||||
}
 | 
					    implements AuthenticationCubit<void> {}
 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockAccount extends Mock implements Account {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockFormRepository extends Mock implements FormRepository {}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
void main() {
 | 
					void main() {
 | 
				
			||||||
  const String invalidEmailString = 'invalid';
 | 
					  const String invalidEmailString = 'invalid';
 | 
				
			||||||
 | 
					  const Email invalidEmail = Email.dirty(invalidEmailString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  const String validEmailString = 'test@gmail.com';
 | 
					  const String validEmailString = 'test@gmail.com';
 | 
				
			||||||
 | 
					  const Email validEmail = Email.dirty(validEmailString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  const String invalidPasswordString = 'invalid';
 | 
					  const String invalidPasswordString = 'invalid';
 | 
				
			||||||
 | 
					  const Password invalidPassword = Password.dirty(invalidPasswordString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  const String validPasswordString = 't0pS3cret1234';
 | 
					  const String validPasswordString = 't0pS3cret1234';
 | 
				
			||||||
 | 
					  const Password validPassword = Password.dirty(validPasswordString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  group('SignInCubit', () {
 | 
					  group('SignInCubit', () {
 | 
				
			||||||
    final MockAccount account = MockAccount();
 | 
					    late AuthenticationRepository authenticationRepository;
 | 
				
			||||||
    final WyattForm form = WyattFormImpl(
 | 
					    late AuthenticationCubit<void> authenticationCubit;
 | 
				
			||||||
      [
 | 
					 | 
				
			||||||
        FormInput(AuthFormField.email, const Email.pure()),
 | 
					 | 
				
			||||||
        FormInput(AuthFormField.password, const Password.pure())
 | 
					 | 
				
			||||||
      ],
 | 
					 | 
				
			||||||
      name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    late MockFormRepository formRepository;
 | 
					 | 
				
			||||||
    late AuthenticationRepository<int> authenticationRepository;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    setUp(() {
 | 
					    setUp(() {
 | 
				
			||||||
      authenticationRepository = MockAuthenticationRepository();
 | 
					      authenticationRepository = MockAuthenticationRepository();
 | 
				
			||||||
      formRepository = MockFormRepository();
 | 
					      authenticationCubit = MockAuthenticationCubit();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      when(
 | 
					      when(
 | 
				
			||||||
        () => authenticationRepository.signInWithEmailAndPassword(
 | 
					        () => authenticationRepository.signInWithEmailAndPassword(
 | 
				
			||||||
          email: any(named: 'email'),
 | 
					          email: any(named: 'email'),
 | 
				
			||||||
          password: any(named: 'password'),
 | 
					          password: any(named: 'password'),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
      ).thenAnswer((_) async => Ok(account));
 | 
					      ).thenAnswer((_) async {});
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      when(
 | 
					      when(
 | 
				
			||||||
        () => authenticationRepository.formRepository,
 | 
					        () => authenticationCubit.start(),
 | 
				
			||||||
      ).thenAnswer((_) => formRepository);
 | 
					      ).thenReturn(true);
 | 
				
			||||||
 | 
					 | 
				
			||||||
      when(
 | 
					 | 
				
			||||||
        () => formRepository.accessForm(AuthFormName.signInForm),
 | 
					 | 
				
			||||||
      ).thenAnswer((_) => form);
 | 
					 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    test('initial state is SignInState', () {
 | 
					    test('initial state is SignInState', () {
 | 
				
			||||||
@ -79,90 +64,33 @@ void main() {
 | 
				
			|||||||
        SignInCubit(
 | 
					        SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ).state,
 | 
					        ).state,
 | 
				
			||||||
        SignInState(form: form),
 | 
					        const SignInState(),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('emailChanged', () {
 | 
					    group('emailChanged', () {
 | 
				
			||||||
      blocTest<SignInCubit<int>, SignInState>(
 | 
					      blocTest<SignInCubit, SignInState>(
 | 
				
			||||||
        'emits [invalid] when email/password are invalid',
 | 
					        'emits [invalid] when email/password are invalid',
 | 
				
			||||||
        build: () => SignInCubit(
 | 
					        build: () => SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.emailChanged(invalidEmailString),
 | 
					        act: (cubit) => cubit.emailChanged(invalidEmailString),
 | 
				
			||||||
        expect: () => <SignInState>[
 | 
					        expect: () => const <SignInState>[
 | 
				
			||||||
          SignInState(
 | 
					          SignInState(email: invalidEmail, status: FormStatus.invalid),
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(invalidEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(AuthFormField.password, const Password.pure())
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.invalid,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignInCubit<int>, SignInState>(
 | 
					      blocTest<SignInCubit, SignInState>(
 | 
				
			||||||
        'emits [valid] when email/password are valid',
 | 
					        'emits [valid] when email/password are valid',
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signInForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.pure(),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => SignInCubit(
 | 
					        build: () => SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignInState(
 | 
					        seed: () => const SignInState(password: validPassword),
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.pure(),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.emailChanged(validEmailString),
 | 
					        act: (cubit) => cubit.emailChanged(validEmailString),
 | 
				
			||||||
        expect: () => <SignInState>[
 | 
					        expect: () => const <SignInState>[
 | 
				
			||||||
          SignInState(
 | 
					          SignInState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					            email: validEmail,
 | 
				
			||||||
              [
 | 
					            password: validPassword,
 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.valid,
 | 
					            status: FormStatus.valid,
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
@ -170,145 +98,58 @@ void main() {
 | 
				
			|||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('passwordChanged', () {
 | 
					    group('passwordChanged', () {
 | 
				
			||||||
      blocTest<SignInCubit<int>, SignInState>(
 | 
					      blocTest<SignInCubit, SignInState>(
 | 
				
			||||||
        'emits [invalid] when email/password are invalid',
 | 
					        'emits [invalid] when email/password are invalid',
 | 
				
			||||||
        build: () => SignInCubit(
 | 
					        build: () => SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.passwordChanged(invalidPasswordString),
 | 
					        act: (cubit) => cubit.passwordChanged(invalidPasswordString),
 | 
				
			||||||
        expect: () => <SignInState>[
 | 
					        expect: () => const <SignInState>[
 | 
				
			||||||
          SignInState(
 | 
					          SignInState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					            password: invalidPassword,
 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.pure(),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(invalidPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.invalid,
 | 
					            status: FormStatus.invalid,
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignInCubit<int>, SignInState>(
 | 
					      blocTest<SignInCubit, SignInState>(
 | 
				
			||||||
        'emits [valid] when email/password are valid',
 | 
					        'emits [valid] when email/password are valid',
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signInForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.pure(),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => SignInCubit(
 | 
					        build: () => SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignInState(
 | 
					        seed: () => const SignInState(email: validEmail),
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.pure(),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        act: (cubit) => cubit.passwordChanged(validPasswordString),
 | 
					        act: (cubit) => cubit.passwordChanged(validPasswordString),
 | 
				
			||||||
        expect: () => <SignInState>[
 | 
					        expect: () => const <SignInState>[
 | 
				
			||||||
          SignInState(
 | 
					          SignInState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					            email: validEmail,
 | 
				
			||||||
              [
 | 
					            password: validPassword,
 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.valid,
 | 
					            status: FormStatus.valid,
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('submit', () {
 | 
					    group('logInWithCredentials', () {
 | 
				
			||||||
      blocTest<SignInCubit<int>, SignInState>(
 | 
					      blocTest<SignInCubit, SignInState>(
 | 
				
			||||||
        'does nothing when status is not validated',
 | 
					        'does nothing when status is not validated',
 | 
				
			||||||
        build: () => SignInCubit(
 | 
					        build: () => SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					        act: (cubit) => cubit.signInWithEmailAndPassword(),
 | 
				
			||||||
        expect: () => const <SignInState>[],
 | 
					        expect: () => const <SignInState>[],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignInCubit<int>, SignInState>(
 | 
					      blocTest<SignInCubit, SignInState>(
 | 
				
			||||||
        'calls signInWithEmailAndPassword with correct email/password',
 | 
					        'calls signInWithEmailAndPassword with correct email/password',
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signInForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => SignInCubit(
 | 
					        build: () => SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignInState(
 | 
					        seed: () => const SignInState(
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					          status: FormStatus.valid,
 | 
				
			||||||
 | 
					          email: validEmail,
 | 
				
			||||||
 | 
					          password: validPassword,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					        act: (cubit) => cubit.signInWithEmailAndPassword(),
 | 
				
			||||||
        verify: (_) {
 | 
					        verify: (_) {
 | 
				
			||||||
          verify(
 | 
					          verify(
 | 
				
			||||||
            () => authenticationRepository.signInWithEmailAndPassword(
 | 
					            () => authenticationRepository.signInWithEmailAndPassword(
 | 
				
			||||||
@ -319,85 +160,33 @@ void main() {
 | 
				
			|||||||
        },
 | 
					        },
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignInCubit<int>, SignInState>(
 | 
					      blocTest<SignInCubit, SignInState>(
 | 
				
			||||||
        'emits [submissionInProgress, submissionSuccess] '
 | 
					        'emits [submissionInProgress, submissionSuccess] '
 | 
				
			||||||
        'when signInWithEmailAndPassword succeeds',
 | 
					        'when signInWithEmailAndPassword succeeds',
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signInForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => SignInCubit(
 | 
					        build: () => SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignInState(
 | 
					        seed: () => const SignInState(
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					          status: FormStatus.valid,
 | 
				
			||||||
 | 
					          email: validEmail,
 | 
				
			||||||
 | 
					          password: validPassword,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					        act: (cubit) => cubit.signInWithEmailAndPassword(),
 | 
				
			||||||
        expect: () => <SignInState>[
 | 
					        expect: () => const <SignInState>[
 | 
				
			||||||
          SignInState(
 | 
					          SignInState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					            status: FormStatus.submissionInProgress,
 | 
				
			||||||
 | 
					            email: validEmail,
 | 
				
			||||||
 | 
					            password: validPassword,
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
          SignInState(
 | 
					          SignInState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionSuccess,
 | 
					            status: FormStatus.submissionSuccess,
 | 
				
			||||||
 | 
					            email: validEmail,
 | 
				
			||||||
 | 
					            password: validPassword,
 | 
				
			||||||
          )
 | 
					          )
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignInCubit<int>, SignInState>(
 | 
					      blocTest<SignInCubit, SignInState>(
 | 
				
			||||||
        'emits [submissionInProgress, submissionFailure] '
 | 
					        'emits [submissionInProgress, submissionFailure] '
 | 
				
			||||||
        'when signInWithEmailAndPassword fails',
 | 
					        'when signInWithEmailAndPassword fails',
 | 
				
			||||||
        setUp: () {
 | 
					        setUp: () {
 | 
				
			||||||
@ -406,77 +195,27 @@ void main() {
 | 
				
			|||||||
              email: any(named: 'email'),
 | 
					              email: any(named: 'email'),
 | 
				
			||||||
              password: any(named: 'password'),
 | 
					              password: any(named: 'password'),
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
          ).thenAnswer((_) async => Err(ServerException()));
 | 
					          ).thenThrow(Exception('oops'));
 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signInForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        build: () => SignInCubit(
 | 
					        build: () => SignInCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignInState(
 | 
					        seed: () => const SignInState(
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					          status: FormStatus.valid,
 | 
				
			||||||
 | 
					          email: validEmail,
 | 
				
			||||||
 | 
					          password: validPassword,
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					        act: (cubit) => cubit.signInWithEmailAndPassword(),
 | 
				
			||||||
        expect: () => <SignInState>[
 | 
					        expect: () => const <SignInState>[
 | 
				
			||||||
          SignInState(
 | 
					          SignInState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					            status: FormStatus.submissionInProgress,
 | 
				
			||||||
 | 
					            email: validEmail,
 | 
				
			||||||
 | 
					            password: validPassword,
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
          SignInState(
 | 
					          SignInState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionFailure,
 | 
					            status: FormStatus.submissionFailure,
 | 
				
			||||||
 | 
					            email: validEmail,
 | 
				
			||||||
 | 
					            password: validPassword,
 | 
				
			||||||
          )
 | 
					          )
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
				
			|||||||
@ -19,38 +19,36 @@ import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			|||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void main() {
 | 
					void main() {
 | 
				
			||||||
  final WyattForm form = WyattFormImpl(
 | 
					  const Email email = Email.dirty('email');
 | 
				
			||||||
    [
 | 
					  const Password password = Password.dirty('password');
 | 
				
			||||||
      FormInput(AuthFormField.email, const Email.pure()),
 | 
					 | 
				
			||||||
      FormInput(AuthFormField.password, const Password.pure())
 | 
					 | 
				
			||||||
    ],
 | 
					 | 
				
			||||||
    name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
  );
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  group('SignInState', () {
 | 
					  group('SignInState', () {
 | 
				
			||||||
    test('supports value comparisons', () {
 | 
					    test('supports value comparisons', () {
 | 
				
			||||||
      expect(
 | 
					      expect(const SignInState(), const SignInState());
 | 
				
			||||||
        SignInState(
 | 
					 | 
				
			||||||
          form: form,
 | 
					 | 
				
			||||||
        ),
 | 
					 | 
				
			||||||
        SignInState(form: form),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    test('returns same object when no properties are passed', () {
 | 
					    test('returns same object when no properties are passed', () {
 | 
				
			||||||
      expect(
 | 
					      expect(const SignInState().copyWith(), const SignInState());
 | 
				
			||||||
        SignInState(form: form).copyWith(),
 | 
					 | 
				
			||||||
        SignInState(form: form),
 | 
					 | 
				
			||||||
      );
 | 
					 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    test('returns object with updated status when status is passed', () {
 | 
					    test('returns object with updated status when status is passed', () {
 | 
				
			||||||
      expect(
 | 
					      expect(
 | 
				
			||||||
        SignInState(form: form).copyWith(status: FormStatus.invalid),
 | 
					        const SignInState().copyWith(status: FormStatus.pure),
 | 
				
			||||||
        SignInState(
 | 
					        const SignInState(),
 | 
				
			||||||
          form: form,
 | 
					      );
 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					    });
 | 
				
			||||||
        ),
 | 
					
 | 
				
			||||||
 | 
					    test('returns object with updated email when email is passed', () {
 | 
				
			||||||
 | 
					      expect(
 | 
				
			||||||
 | 
					        const SignInState().copyWith(email: email),
 | 
				
			||||||
 | 
					        const SignInState(email: email),
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    test('returns object with updated password when password is passed', () {
 | 
				
			||||||
 | 
					      expect(
 | 
				
			||||||
 | 
					        const SignInState().copyWith(password: password),
 | 
				
			||||||
 | 
					        const SignInState(password: password),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
  });
 | 
					  });
 | 
				
			||||||
 | 
				
			|||||||
@ -17,298 +17,163 @@
 | 
				
			|||||||
import 'package:bloc_test/bloc_test.dart';
 | 
					import 'package:bloc_test/bloc_test.dart';
 | 
				
			||||||
import 'package:flutter_test/flutter_test.dart';
 | 
					import 'package:flutter_test/flutter_test.dart';
 | 
				
			||||||
import 'package:mocktail/mocktail.dart';
 | 
					import 'package:mocktail/mocktail.dart';
 | 
				
			||||||
import 'package:wyatt_architecture/wyatt_architecture.dart';
 | 
					 | 
				
			||||||
import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
					import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
				
			||||||
import 'package:wyatt_type_utils/wyatt_type_utils.dart';
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
class MockAuthenticationRepository extends Mock
 | 
					class MockAuthenticationRepository extends Mock
 | 
				
			||||||
    implements AuthenticationRepository<int> {}
 | 
					    implements AuthenticationRepository {}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
class MockAuthenticationCubit extends Mock implements AuthenticationCubit<int> {
 | 
					class MockAuthenticationCubit extends Mock
 | 
				
			||||||
}
 | 
					    implements AuthenticationCubit<void> {}
 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockAccount extends Mock implements Account {}
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
class MockFormRepository extends Mock implements FormRepository {}
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
void main() {
 | 
					void main() {
 | 
				
			||||||
  const String invalidEmailString = 'invalid';
 | 
					  const String invalidEmailString = 'invalid';
 | 
				
			||||||
 | 
					  const Email invalidEmail = Email.dirty(invalidEmailString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  const String validEmailString = 'test@gmail.com';
 | 
					  const String validEmailString = 'test@gmail.com';
 | 
				
			||||||
 | 
					  const Email validEmail = Email.dirty(validEmailString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  const String invalidPasswordString = 'invalid';
 | 
					  const String invalidPasswordString = 'invalid';
 | 
				
			||||||
 | 
					  const Password invalidPassword = Password.dirty(invalidPasswordString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  const String validPasswordString = 't0pS3cret1234';
 | 
					  const String validPasswordString = 't0pS3cret1234';
 | 
				
			||||||
 | 
					  const Password validPassword = Password.dirty(validPasswordString);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
  group('SignUpCubit', () {
 | 
					  group('SignUpCubit', () {
 | 
				
			||||||
    final MockAccount account = MockAccount();
 | 
					    late AuthenticationRepository authenticationRepository;
 | 
				
			||||||
    final WyattForm form = WyattFormImpl(
 | 
					    late AuthenticationCubit<void> authenticationCubit;
 | 
				
			||||||
      [
 | 
					 | 
				
			||||||
        FormInput(AuthFormField.email, const Email.pure()),
 | 
					 | 
				
			||||||
        FormInput(AuthFormField.password, const Password.pure())
 | 
					 | 
				
			||||||
      ],
 | 
					 | 
				
			||||||
      name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
    );
 | 
					 | 
				
			||||||
 | 
					 | 
				
			||||||
    late MockFormRepository formRepository;
 | 
					 | 
				
			||||||
    late AuthenticationRepository<int> authenticationRepository;
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
    setUp(() {
 | 
					    setUp(() {
 | 
				
			||||||
      authenticationRepository = MockAuthenticationRepository();
 | 
					      authenticationRepository = MockAuthenticationRepository();
 | 
				
			||||||
      formRepository = MockFormRepository();
 | 
					      authenticationCubit = MockAuthenticationCubit();
 | 
				
			||||||
 | 
					 | 
				
			||||||
      when(
 | 
					      when(
 | 
				
			||||||
        () => authenticationRepository.signUp(
 | 
					        () => authenticationRepository.signUp(
 | 
				
			||||||
          email: any(named: 'email'),
 | 
					          email: any(named: 'email'),
 | 
				
			||||||
          password: any(named: 'password'),
 | 
					          password: any(named: 'password'),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
      ).thenAnswer((_) async => Ok(account));
 | 
					      ).thenAnswer((_) async => 'uid');
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      when(
 | 
					      when(
 | 
				
			||||||
        () => authenticationRepository.formRepository,
 | 
					        () => authenticationCubit.start(),
 | 
				
			||||||
      ).thenAnswer((_) => formRepository);
 | 
					      ).thenReturn(true);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      when(
 | 
					      when(
 | 
				
			||||||
        () => formRepository.accessForm(AuthFormName.signUpForm),
 | 
					        () => authenticationCubit.stop(),
 | 
				
			||||||
      ).thenAnswer((_) => form);
 | 
					      ).thenReturn(true);
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    test('initial state is SignUpState', () {
 | 
					    test('initial state is SignUpState', () {
 | 
				
			||||||
      expect(
 | 
					      expect(
 | 
				
			||||||
        SignUpCubit(
 | 
					        SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ).state,
 | 
					        ).state,
 | 
				
			||||||
        SignUpState(form: form),
 | 
					        const SignUpState(data: FormData.empty()),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('emailChanged', () {
 | 
					    group('emailChanged', () {
 | 
				
			||||||
      blocTest<SignUpCubit<int>, SignUpState>(
 | 
					      blocTest<SignUpCubit, SignUpState>(
 | 
				
			||||||
        'emits [invalid] when email/password are invalid',
 | 
					        'emits [invalid] when email/password are invalid',
 | 
				
			||||||
        build: () => SignUpCubit(
 | 
					        build: () => SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.emailChanged(invalidEmailString),
 | 
					        act: (cubit) => cubit.emailChanged(invalidEmailString),
 | 
				
			||||||
        expect: () => <SignUpState>[
 | 
					        expect: () => <SignUpState>[
 | 
				
			||||||
          SignUpState(
 | 
					          const SignUpState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					            email: invalidEmail,
 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(invalidEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(AuthFormField.password, const Password.pure())
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.invalid,
 | 
					            status: FormStatus.invalid,
 | 
				
			||||||
 | 
					            data: FormData.empty(),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignUpCubit<int>, SignUpState>(
 | 
					      blocTest<SignUpCubit, SignUpState>(
 | 
				
			||||||
        'emits [valid] when email/password are valid',
 | 
					        'emits [valid] when email/password are valid',
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signUpForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.pure(),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => SignUpCubit(
 | 
					        build: () => SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignUpState(
 | 
					        seed: () => const SignUpState(
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					          password: validPassword,
 | 
				
			||||||
            [
 | 
					          data: FormData.empty(),
 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.pure(),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.emailChanged(validEmailString),
 | 
					        act: (cubit) => cubit.emailChanged(validEmailString),
 | 
				
			||||||
        expect: () => <SignUpState>[
 | 
					        expect: () => <SignUpState>[
 | 
				
			||||||
          SignUpState(
 | 
					          const SignUpState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					            email: validEmail,
 | 
				
			||||||
              [
 | 
					            password: validPassword,
 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.valid,
 | 
					            status: FormStatus.valid,
 | 
				
			||||||
 | 
					            data: FormData.empty(),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('passwordChanged', () {
 | 
					    group('passwordChanged', () {
 | 
				
			||||||
      blocTest<SignUpCubit<int>, SignUpState>(
 | 
					      blocTest<SignUpCubit, SignUpState>(
 | 
				
			||||||
        'emits [invalid] when email/password are invalid',
 | 
					        'emits [invalid] when email/password are invalid',
 | 
				
			||||||
        build: () => SignUpCubit(
 | 
					        build: () => SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.passwordChanged(invalidPasswordString),
 | 
					        act: (cubit) => cubit.passwordChanged(invalidPasswordString),
 | 
				
			||||||
        expect: () => <SignUpState>[
 | 
					        expect: () => <SignUpState>[
 | 
				
			||||||
          SignUpState(
 | 
					          const SignUpState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					            password: invalidPassword,
 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.pure(),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(invalidPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.invalid,
 | 
					            status: FormStatus.invalid,
 | 
				
			||||||
 | 
					            data: FormData.empty(),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignUpCubit<int>, SignUpState>(
 | 
					      blocTest<SignUpCubit, SignUpState>(
 | 
				
			||||||
        'emits [valid] when email/password are valid',
 | 
					        'emits [valid] when email/password are valid',
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signUpForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.pure(),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => SignUpCubit(
 | 
					        build: () => SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignUpState(
 | 
					        seed: () => const SignUpState(
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					          email: validEmail,
 | 
				
			||||||
            [
 | 
					          data: FormData.empty(),
 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.pure(),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.passwordChanged(validPasswordString),
 | 
					        act: (cubit) => cubit.passwordChanged(validPasswordString),
 | 
				
			||||||
        expect: () => <SignUpState>[
 | 
					        expect: () => <SignUpState>[
 | 
				
			||||||
          SignUpState(
 | 
					          const SignUpState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					            email: validEmail,
 | 
				
			||||||
              [
 | 
					            password: validPassword,
 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.valid,
 | 
					            status: FormStatus.valid,
 | 
				
			||||||
 | 
					            data: FormData.empty(),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    group('submit', () {
 | 
					    group('signUpFormSubmitted', () {
 | 
				
			||||||
      blocTest<SignUpCubit<int>, SignUpState>(
 | 
					      blocTest<SignUpCubit, SignUpState>(
 | 
				
			||||||
        'does nothing when status is not validated',
 | 
					        'does nothing when status is not validated',
 | 
				
			||||||
        build: () => SignUpCubit(
 | 
					        build: () => SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					        act: (cubit) => cubit.signUpFormSubmitted(),
 | 
				
			||||||
        expect: () => const <SignUpState>[],
 | 
					        expect: () => const <SignUpState>[],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignUpCubit<int>, SignUpState>(
 | 
					      blocTest<SignUpCubit, SignUpState>(
 | 
				
			||||||
        'calls signUp with correct email/password',
 | 
					        'calls signUp with correct email/password/confirmedPassword',
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signUpForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => SignUpCubit(
 | 
					        build: () => SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignUpState(
 | 
					        seed: () => const SignUpState(
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					          status: FormStatus.valid,
 | 
				
			||||||
 | 
					          email: validEmail,
 | 
				
			||||||
 | 
					          password: validPassword,
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					        act: (cubit) => cubit.signUpFormSubmitted(),
 | 
				
			||||||
        verify: (_) {
 | 
					        verify: (_) {
 | 
				
			||||||
          verify(
 | 
					          verify(
 | 
				
			||||||
            () => authenticationRepository.signUp(
 | 
					            () => authenticationRepository.signUp(
 | 
				
			||||||
@ -319,85 +184,37 @@ void main() {
 | 
				
			|||||||
        },
 | 
					        },
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignUpCubit<int>, SignUpState>(
 | 
					      blocTest<SignUpCubit, SignUpState>(
 | 
				
			||||||
        'emits [submissionInProgress, submissionSuccess] '
 | 
					        'emits [submissionInProgress, submissionSuccess] '
 | 
				
			||||||
        'when signUp succeeds',
 | 
					        'when signUp succeeds',
 | 
				
			||||||
        setUp: () {
 | 
					 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signUpForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					 | 
				
			||||||
        build: () => SignUpCubit(
 | 
					        build: () => SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignUpState(
 | 
					        seed: () => const SignUpState(
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					          status: FormStatus.valid,
 | 
				
			||||||
 | 
					          email: validEmail,
 | 
				
			||||||
 | 
					          password: validPassword,
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					        act: (cubit) => cubit.signUpFormSubmitted(),
 | 
				
			||||||
        expect: () => <SignUpState>[
 | 
					        expect: () => <SignUpState>[
 | 
				
			||||||
          SignUpState(
 | 
					          const SignUpState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					            status: FormStatus.submissionInProgress,
 | 
				
			||||||
 | 
					            email: validEmail,
 | 
				
			||||||
 | 
					            password: validPassword,
 | 
				
			||||||
 | 
					            data: FormData.empty(),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
          SignUpState(
 | 
					          const SignUpState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionSuccess,
 | 
					            status: FormStatus.submissionSuccess,
 | 
				
			||||||
 | 
					            email: validEmail,
 | 
				
			||||||
 | 
					            password: validPassword,
 | 
				
			||||||
 | 
					            data: FormData.empty(),
 | 
				
			||||||
          )
 | 
					          )
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
					
 | 
				
			||||||
      blocTest<SignUpCubit<int>, SignUpState>(
 | 
					      blocTest<SignUpCubit, SignUpState>(
 | 
				
			||||||
        'emits [submissionInProgress, submissionFailure] '
 | 
					        'emits [submissionInProgress, submissionFailure] '
 | 
				
			||||||
        'when signUp fails',
 | 
					        'when signUp fails',
 | 
				
			||||||
        setUp: () {
 | 
					        setUp: () {
 | 
				
			||||||
@ -406,77 +223,31 @@ void main() {
 | 
				
			|||||||
              email: any(named: 'email'),
 | 
					              email: any(named: 'email'),
 | 
				
			||||||
              password: any(named: 'password'),
 | 
					              password: any(named: 'password'),
 | 
				
			||||||
            ),
 | 
					            ),
 | 
				
			||||||
          ).thenAnswer((_) async => Err(ServerException()));
 | 
					          ).thenThrow(Exception('oops'));
 | 
				
			||||||
          when(
 | 
					 | 
				
			||||||
            () => formRepository.accessForm(AuthFormName.signUpForm),
 | 
					 | 
				
			||||||
          ).thenAnswer(
 | 
					 | 
				
			||||||
            (_) => WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
          );
 | 
					 | 
				
			||||||
        },
 | 
					        },
 | 
				
			||||||
        build: () => SignUpCubit(
 | 
					        build: () => SignUpCubit(
 | 
				
			||||||
          authenticationRepository: authenticationRepository,
 | 
					          authenticationRepository: authenticationRepository,
 | 
				
			||||||
 | 
					          formData: const FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        seed: () => SignUpState(
 | 
					        seed: () => const SignUpState(
 | 
				
			||||||
          form: WyattFormImpl(
 | 
					 | 
				
			||||||
            [
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.email,
 | 
					 | 
				
			||||||
                const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
              ),
 | 
					 | 
				
			||||||
              FormInput(
 | 
					 | 
				
			||||||
                AuthFormField.password,
 | 
					 | 
				
			||||||
                const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
              )
 | 
					 | 
				
			||||||
            ],
 | 
					 | 
				
			||||||
            name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
          ),
 | 
					 | 
				
			||||||
          status: FormStatus.valid,
 | 
					          status: FormStatus.valid,
 | 
				
			||||||
 | 
					          email: validEmail,
 | 
				
			||||||
 | 
					          password: validPassword,
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        act: (cubit) => cubit.submit(),
 | 
					        act: (cubit) => cubit.signUpFormSubmitted(),
 | 
				
			||||||
        expect: () => <SignUpState>[
 | 
					        expect: () => <SignUpState>[
 | 
				
			||||||
          SignUpState(
 | 
					          const SignUpState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionInProgress,
 | 
					            status: FormStatus.submissionInProgress,
 | 
				
			||||||
 | 
					            email: validEmail,
 | 
				
			||||||
 | 
					            password: validPassword,
 | 
				
			||||||
 | 
					            data: FormData.empty(),
 | 
				
			||||||
          ),
 | 
					          ),
 | 
				
			||||||
          SignUpState(
 | 
					          const SignUpState(
 | 
				
			||||||
            form: WyattFormImpl(
 | 
					 | 
				
			||||||
              [
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.email,
 | 
					 | 
				
			||||||
                  const Email.dirty(validEmailString),
 | 
					 | 
				
			||||||
                ),
 | 
					 | 
				
			||||||
                FormInput(
 | 
					 | 
				
			||||||
                  AuthFormField.password,
 | 
					 | 
				
			||||||
                  const Password.dirty(validPasswordString),
 | 
					 | 
				
			||||||
                )
 | 
					 | 
				
			||||||
              ],
 | 
					 | 
				
			||||||
              name: AuthFormName.signUpForm,
 | 
					 | 
				
			||||||
            ),
 | 
					 | 
				
			||||||
            status: FormStatus.submissionFailure,
 | 
					            status: FormStatus.submissionFailure,
 | 
				
			||||||
 | 
					            email: validEmail,
 | 
				
			||||||
 | 
					            password: validPassword,
 | 
				
			||||||
 | 
					            data: FormData.empty(),
 | 
				
			||||||
          )
 | 
					          )
 | 
				
			||||||
        ],
 | 
					        ],
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
 | 
				
			|||||||
@ -19,37 +19,93 @@ import 'package:wyatt_authentication_bloc/wyatt_authentication_bloc.dart';
 | 
				
			|||||||
import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
					import 'package:wyatt_form_bloc/wyatt_form_bloc.dart';
 | 
				
			||||||
 | 
					
 | 
				
			||||||
void main() {
 | 
					void main() {
 | 
				
			||||||
  final WyattForm form = WyattFormImpl(
 | 
					  const Email email = Email.dirty('email');
 | 
				
			||||||
    [
 | 
					  const String passwordString = 'password';
 | 
				
			||||||
      FormInput(AuthFormField.email, const Email.pure()),
 | 
					  const Password password = Password.dirty(passwordString);
 | 
				
			||||||
      FormInput(AuthFormField.password, const Password.pure())
 | 
					 | 
				
			||||||
    ],
 | 
					 | 
				
			||||||
    name: AuthFormName.signInForm,
 | 
					 | 
				
			||||||
  );
 | 
					 | 
				
			||||||
 | 
					
 | 
				
			||||||
  group('SignUpState', () {
 | 
					  group('SignUpState', () {
 | 
				
			||||||
    test('supports value comparisons', () {
 | 
					    test('supports value comparisons', () {
 | 
				
			||||||
      expect(
 | 
					      expect(
 | 
				
			||||||
        SignUpState(
 | 
					        const SignUpState(
 | 
				
			||||||
          form: form,
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
        SignUpState(form: form),
 | 
					 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    test('returns same object when no properties are passed', () {
 | 
					    test('returns same object when no properties are passed', () {
 | 
				
			||||||
      expect(
 | 
					      expect(
 | 
				
			||||||
        SignUpState(form: form).copyWith(),
 | 
					        const SignUpState(
 | 
				
			||||||
        SignUpState(form: form),
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ).copyWith(),
 | 
				
			||||||
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
    test('returns object with updated status when status is passed', () {
 | 
					    test('returns object with updated status when status is passed', () {
 | 
				
			||||||
      expect(
 | 
					      expect(
 | 
				
			||||||
        SignUpState(form: form).copyWith(status: FormStatus.invalid),
 | 
					        const SignUpState(
 | 
				
			||||||
        SignUpState(
 | 
					          data: FormData.empty(),
 | 
				
			||||||
          form: form,
 | 
					        ).copyWith(status: FormStatus.pure),
 | 
				
			||||||
          status: FormStatus.invalid,
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    test('returns object with updated email when email is passed', () {
 | 
				
			||||||
 | 
					      expect(
 | 
				
			||||||
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ).copyWith(email: email),
 | 
				
			||||||
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          email: email,
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    test('returns object with updated password when password is passed', () {
 | 
				
			||||||
 | 
					      expect(
 | 
				
			||||||
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ).copyWith(password: password),
 | 
				
			||||||
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          password: password,
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					      );
 | 
				
			||||||
 | 
					    });
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					    test(
 | 
				
			||||||
 | 
					        'returns object with updated data'
 | 
				
			||||||
 | 
					        ' when data is passed', () {
 | 
				
			||||||
 | 
					      expect(
 | 
				
			||||||
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          data: FormData.empty(),
 | 
				
			||||||
 | 
					        ).copyWith(
 | 
				
			||||||
 | 
					          data: const FormData(
 | 
				
			||||||
 | 
					            [
 | 
				
			||||||
 | 
					              FormInput(
 | 
				
			||||||
 | 
					                'field',
 | 
				
			||||||
 | 
					                Name.pure(),
 | 
				
			||||||
 | 
					              ),
 | 
				
			||||||
 | 
					            ],
 | 
				
			||||||
 | 
					          ),
 | 
				
			||||||
 | 
					        ),
 | 
				
			||||||
 | 
					        const SignUpState(
 | 
				
			||||||
 | 
					          data: FormData(
 | 
				
			||||||
 | 
					            [
 | 
				
			||||||
 | 
					              FormInput(
 | 
				
			||||||
 | 
					                'field',
 | 
				
			||||||
 | 
					                Name.pure(),
 | 
				
			||||||
 | 
					              ),
 | 
				
			||||||
 | 
					            ],
 | 
				
			||||||
 | 
					          ),
 | 
				
			||||||
        ),
 | 
					        ),
 | 
				
			||||||
      );
 | 
					      );
 | 
				
			||||||
    });
 | 
					    });
 | 
				
			||||||
 | 
				
			|||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user