bloc_enhancer
bloc_enhancer_gen
is a code generator package that complements bloc_enhancer
by automating the generation of streamlined Bloc event and state classes. This generator simplifies and enhances the development process, allowing for more intuitive and efficient state management.
Why bloc_enhancer_gen
?
The Problem
My biggest (prior) complaint for using bloc was naming. Generally, you have to prepend all public states and events with the bloc's name to avoid naming conflicts with other blocs (You can't have 2 Ready
states 🙄), then the names started to get long.
You'd also have to remember what the name of your event is to invoke bloc.add(YourBlocEven())
. You'd also have to remember what the name of your state is to check bloc.state is YourBlocState
. This is all fine and dandy for small projects, but as your project grows, it becomes a pain to manage.
There are packages that help you with mapping out your states... but dart only allows 80 characters per line, so nesting all of your (UI) code within a bloc.state.map
is not ideal.
The Solution
bloc_enhancer_gen
is a code generator that generates a set of extensions that allow you to invoke events and check states intuitively.
// Simplified events (you can use intellisense! 🙌)
bloc.events.createUser(name: 'John');
// Intuitive state type check and retrieval
if (bloc.state.asIfReady case final readyState?) {
// Handle the ready state
}
// or
if (bloc.state.isReady) {
final readyState = bloc.state.asReady;
// Handle the ready state
}
Getting Started
-
Add the
bloc_enhancer
,bloc_enhancer_gen
,build_runner
packages to yourpubspec.yaml
file:dependencies: bloc_enhancer: # latest version dev_dependencies: bloc_enhancer_gen: # latest version build_runner: # latest version
-
Configure the build runner to use
bloc_enhancer_gen
in yourbuild.yaml
file:targets: $default: builders: bloc_enhancer_gen: generate_for: - lib/blocs/*.dart options: auto_enhance: false # set to true to automatically enhance all Blocs
-
Feed Bloc to the generator using
@enhance
or the config inbuild.yaml
.
Using @enhance
import 'package:bloc_enhancer/bloc_enhancer.dart';
@enhance
class UserBloc extends Bloc<UserEvent, UserState> {
...
}
Using build.yaml
targets:
$default:
builders:
bloc_enhancer_gen:
generate_for:
- lib/**.dart
options:
auto_enhance: true
bloc_enhancer_gen
will read your events and your states.
// user_event.dart
class UserEvent {}
class _Create extends UserEvent {}
class _UpdateUser extends UserEvent {
const _UpdateUser.name(String this.name): email = null;
const _UpdateUser.email(String this.email): name = null;
final String? name;
final String? email;
}
// user_state.dart
class UserState {}
class _Ready extends UserState {
const Ready(this.user);
final User user;
}
class _Loading extends UserState {}
class _Error extends UserState {
const Error(this.message);
final String message;
}
-
Run the build command to generate the enhanced Bloc classes:
# using flutter flutter pub run build_runner build --delete-conflicting-outputs # using dart pub run build_runner build --delete-conflicting-outputs
Testing
Obviously we can't test private classes, so how do we test the enhanced blocs?
Enhanced blocs can generate "Factory" classes for both event & state classes. These classes will create new instances of the event or state class with all the necessary parameters.
Generate
Via createFactory
Annotation
To generate the factory classes, you can annotate the base event or state class with @createFactory
.
@createFactory
class UserEvent {}
class _Fetch extends UserEvent {}
@createFactory
class UserState {}
class _Ready extends UserState {}
Via create_event_factory
& create_state_factory
Options
You can also enable the factory generation via the build.yaml
file.
Note: This will generate factories for all events and states for every enhanced bloc.
targets:
$default:
builders:
bloc_enhancer_gen:
generate_for:
- lib/**.dart
options:
create_event_factory: true # default is false
create_state_factory: true # default is false
Usage
To use the newly generated factories, create a static instance of the in the base event or state class.
@createFactory
class UserEvent {
static const create = _$UserEventCreator();
}
@createFactory
class UserState {
static const create = _$UserStateCreator();
}
Now you can create new instances of the event or state classes within your tests.
final readyState = UserState.create.ready();
final fetchEvent = UserEvent.create.fetch();
If you don't want to create a certain class in the factory, you can exclude them by annotating the class with @ignore
. This also applies to constructs that you'd like to exclude too.
A note about build_runner
The build runner can be slow if you have a large project. It really starts to slow down with nested folders. To get the optimal performance, I recommend flattening your files that require generation as much as possible. This will allow the build runner to only scan the files that are necessary.
Slow 🚶
lib
└── blocs
├── user
│ ├── user_bloc.dart
│ ├── user_event.dart
│ └── user_state.dart
└── post
├── post_bloc.dart
├── post_event.dart
└── post_state.dart
Fast 🏃
lib
├── user_bloc.dart
├── user_event.dart
├── user_state.dart
├── post_bloc.dart
├── post_event.dart
└── post_state.dart
You can also use the exclude
option in build.yaml
to exclude folders that don't need to be scanned.
targets:
$default:
builders:
bloc_enhancer_gen:
generate_for:
include:
- lib/blocs/*.dart
exclude:
- lib/ui/**
generate_for.include
& generate_for.exclude
are not bound to this package. They are part of the build.yaml
format. You can read more here
License
This software is released under the Apache 2.0 license. www.apache.org/licenses/LICENSE-2.0
Libraries
- bloc_enhancer_gen
- Copyright 2024 CouchSurfing International Inc.
- models/settings
- Copyright 2024 CouchSurfing International Inc.
- models/settings_interface