RepositoryProvider, a Flutter widget which provides a repository to its children.
BlocListener, a Flutter widget which invokes the listener code in response to state changes in the bloc.
Updating the UI based on a part of a bloc state with context.select.
Project Setup
We’ll start off by creating a brand new Flutter project
Next, we can install all of our dependencies
Authentication Repository
The first thing we’re going to do is create an authentication_repository package which will be responsible for managing the authentication domain.
We’ll start by creating a packages/authentication_repository directory at the root of the project which will contain all internal packages.
At a high level, the directory structure should look like this:
Next, we can create a pubspec.yaml for the authentication_repository package:
Next up, we need to implement the AuthenticationRepository class itself which will be in packages/authentication_repository/lib/src/authentication_repository.dart.
The AuthenticationRepository exposes a Stream of AuthenticationStatus updates which will be used to notify the application when a user signs in or out.
In addition, there are logIn and logOut methods which are stubbed for simplicity but can easily be extended to authenticate with FirebaseAuth for example or some other authentication provider.
Lastly, we need to create packages/authentication_repository/lib/authentication_repository.dart which will contain the public exports:
That’s it for the AuthenticationRepository, next we’ll work on the UserRepository.
User Repository
Just like with the AuthenticationRepository, we will create a user_repository package inside the packages directory.
Next, we’ll create the pubspec.yaml for the user_repository:
The user_repository will be responsible for the user domain and will expose APIs to interact with the current user.
The first thing we will define is the user model in packages/user_repository/lib/src/models/user.dart:
For simplicity, a user just has an id property but in practice we might have additional properties like firstName, lastName, avatarUrl, etc…
Next, we can create a models.dart in packages/user_repository/lib/src/models which will export all models so that we can use a single import state to import multiple models.
Now that the models have been defined, we can implement the UserRepository class in packages/user_repository/lib/src/user_repository.dart.
For this simple example, the UserRepository exposes a single method getUser which will retrieve the current user. We are stubbing this but in practice this is where we would query the current user from the backend.
Almost done with the user_repository package — the only thing left to do is to create the user_repository.dart file in packages/user_repository/lib which defines the public exports:
Now that we have the authentication_repository and user_repository packages complete, we can focus on the Flutter application.
Installing Dependencies
Let’s start by updating the generated pubspec.yaml at the root of our project:
We can install the dependencies by running:
Authentication Bloc
The AuthenticationBloc will be responsible for reacting to changes in the authentication state (exposed by the AuthenticationRepository) and will emit states we can react to in the presentation layer.
The implementation for the AuthenticationBloc is inside of lib/authentication because we treat authentication as a feature in our application layer.
authentication_event.dart
AuthenticationEvent instances will be the input to the AuthenticationBloc and will be processed and used to emit new AuthenticationState instances.
In this application, the AuthenticationBloc will be reacting to two different events:
AuthenticationSubscriptionRequested: initial event that notifies the bloc to subscribe to the AuthenticationStatus stream
AuthenticationLogoutPressed: notifies the bloc of a user logout action
Next, let’s take a look at the AuthenticationState.
authentication_state.dart
AuthenticationState instances will be the output of the AuthenticationBloc and will be consumed by the presentation layer.
The AuthenticationState class has three named constructors:
AuthenticationState.unknown(): the default state which indicates that the bloc does not yet know whether the current user is authenticated or not.
AuthenticationState.authenticated(): the state which indicates that the user is current authenticated.
AuthenticationState.unauthenticated(): the state which indicates that the user is current not authenticated.
Now that we have seen the AuthenticationEvent and AuthenticationState implementations let’s take a look at AuthenticationBloc.
authentication_bloc.dart
The AuthenticationBloc manages the authentication state of the application which is used to determine things like whether or not to start the user at a login page or a home page.
The AuthenticationBloc has a dependency on both the AuthenticationRepository and UserRepository and defines the initial state as AuthenticationState.unknown().
In the constructor body, AuthenticationEvent subclasses are mapped to their corresponding event handlers.
In the _onSubscriptionRequested event handler, the AuthenticationBloc uses emit.onEach to subscribe to the status stream of the AuthenticationRepository and emit a state in response to each AuthenticationStatus.
emit.onEach creates a stream subscription internally and takes care of canceling it when either AuthenticationBloc or the status stream is closed.
If the status stream emits an error, addError forwards the error and stackTrace to any BlocObserver listening.
When the status stream emits AuthenticationStatus.unknown or unauthenticated, the corresponding AuthenticationState is emitted.
When AuthenticationStatus.authenticated is emitted, the AuthentictionBloc queries the user via the UserRepository.
main.dart
Next, we can replace the default main.dart with:
App
app.dart will contain the root App widget for the entire application.
By default, BlocProvider is lazy and does not call create until the first time the Bloc is accessed. Since AuthenticationBloc should always subscribe to the AuthenticationStatus stream immediately (via the AuthenticationSubscriptionRequested event), we can explicitly opt out of this behavior by setting lazy: false.
AppView is a StatefulWidget because it maintains a GlobalKey which is used to access the NavigatorState. By default, AppView will render the SplashPage (which we will see later) and it uses BlocListener to navigate to different pages based on changes in the AuthenticationState.
Splash
The splash feature will just contain a simple view which will be rendered right when the app is launched while the app determines whether the user is authenticated.
Login
The login feature contains a LoginPage, LoginForm and LoginBloc and allows users to enter a username and password to log into the application.
Login Models
We are using package:formz to create reusable and standard models for the username and password.
Username
For simplicity, we are just validating the username to ensure that it is not empty but in practice you can enforce special character usage, length, etc…
Password
Again, we are just performing a simple check to ensure the password is not empty.
Models Barrel
Just like before, there is a models.dart barrel to make it easy to import the Username and Password models with a single import.
Login Bloc
The LoginBloc manages the state of the LoginForm and takes care validating the username and password input as well as the state of the form.
login_event.dart
In this application there are three different LoginEvent types:
LoginUsernameChanged: notifies the bloc that the username has been modified.
LoginPasswordChanged: notifies the bloc that the password has been modified.
LoginSubmitted: notifies the bloc that the form has been submitted.
login_state.dart
The LoginState will contain the status of the form as well as the username and password input states.
login_bloc.dart
The LoginBloc is responsible for reacting to user interactions in the LoginForm and handling the validation and submission of the form.
The LoginBloc has a dependency on the AuthenticationRepository because when the form is submitted, it invokes logIn. The initial state of the bloc is pure meaning neither the inputs nor the form has been touched or interacted with.
Whenever either the username or password change, the bloc will create a dirty variant of the Username/Password model and update the form status via the Formz.validate API.
When the LoginSubmitted event is added, if the current status of the form is valid, the bloc makes a call to logIn and updates the status based on the outcome of the request.
Next let’s take a look at the LoginPage and LoginForm.
Login Page
The LoginPage is responsible for exposing the Route as well as creating and providing the LoginBloc to the LoginForm.
Login Form
The LoginForm handles notifying the LoginBloc of user events and also responds to state changes using BlocBuilder and BlocListener.
BlocListener is used to show a SnackBar if the login submission fails. In addition, context.select is used to efficiently access specific parts of the LoginState for each widget, preventing unnecessary rebuilds. The onChanged callback is used to notify the LoginBloc of changes to the username/password.
The _LoginButton widget is only enabled if the status of the form is valid and a CircularProgressIndicator is shown in its place while the form is being submitted.
Home
Upon a successful logIn request, the state of the AuthenticationBloc will change to authenticated and the user will be navigated to the HomePage where we display the user’s id as well as a button to log out.
Home Page
The HomePage can access the current user id via context.select((AuthenticationBloc bloc) => bloc.state.user.id) and displays it via a Text widget. In addition, when the logout button is tapped, an AuthenticationLogoutRequested event is added to the AuthenticationBloc.
At this point we have a pretty solid login implementation and we have decoupled our presentation layer from the business logic layer by using Bloc.
The full source for this example (including unit and widget tests) can be found here.