Quickstart Guide
This chapter provides a hands-on introduction to using inject.dart in a real Flutter project. We'll build a simple application from scratch to demonstrate how dependency injection works in practice.
By the end of this chapter, you'll understand how to:
- Set up a Flutter project with inject.dart
- Define injectable classes and providers
- Create and use a component
- Access your dependencies throughout the application
Let's begin by creating a new Flutter project which provides a minimal starting point without unnecessary boilerplate code:
flutter create flutter_demo
This creates a basic Flutter project structure with just the essential files. Next, let's navigate into the project directory:
cd flutter_demo
Now we're ready to start implementing dependency injection with inject.dart!
Adding Dependencies
First, we need to add inject.dart and related packages to our project:
flutter pub add inject_annotation dev:inject_generator dev:build_runner
This command adds three important packages:
inject_annotation
: The core package that provides annotations like@inject
,@provides
, and@singleton
inject_generator
: The code generation package that processes the annotations and generates the dependency injection codebuild_runner
: Dart's standard build system that runs the code generators
The dev:
prefix indicates that inject_generator and build_runner are
development dependencies, which means they're only used during development
and won't be included in your production app.
After running this command, you should see the dependencies added to your
pubspec.yaml
file. The next step is to create our first injectable
classes!
Creating the Component
After adding dependencies, our first step is to create the root component that will serve as the entry point for our dependency graph.
In dependency injection, a component acts as a container that knows how to
create and provide the objects in your application. Let's create our
MainComponent
in the main.dart file:
import 'package:flutter/material.dart';
import 'package:inject_annotation/inject_annotation.dart';
void main() {
// We'll update this later to use our component
// For now, keep the default main method
runApp(const MyApp());
}
@component
abstract class MainComponent {
@inject
MyAppFactory get myAppFactory;
}
This MainComponent
is the root of our dependency graph. The @component
annotation tells inject.dart that this class will be our dependency
container. Inside the component, we define methods that return the types we
want to inject, like MyAppFactory
.
Creating a Factory for MyApp
Next, we need to create a factory for our main application widget. Add the
MyAppFactory
class above the MyApp
widget and add the @assistedInject
and @assisted
annotations to the MyApp
class:
@assistedFactory
abstract class MyAppFactory {
MyApp create({Key? key});
}
class MyApp extends StatelessWidget {
@assistedInject
const MyApp({@assisted super.key});
// the rest remains unchanged for now
}
These annotations tell inject.dart how to create instances of MyApp
:
@assistedFactory
creates a factory interface that will instantiateMyApp
with its dependencies@assistedInject
marks the constructor as the injection point@assisted
indicates parameters that are provided at runtime rather than from the dependency graph
Here, we're using @assistedInject
to create a factory for our MyApp
widget. This special annotation generates a factory that can create MyApp
instances with both injected dependencies (which we'll add later) and
runtime parameters (like key
).
The MyAppFactory
will allow us to create MyApp
instances with all
necessary dependencies automatically injected, while still allowing us to
pass in runtime values like the optional key
parameter.
Generating the Code
Now that we've set up our component and factory, it's time to generate the actual dependency injection code:
dart run build_runner build
This command processes our annotations and generates the necessary
implementation code. If successful, you'll see output indicating that files
were generated, including main.inject.dart
.
Once the generation is complete, we need to import the generated code and add a convenience factory method to our component:
import 'package:flutter/material.dart';
import 'package:inject_annotation/inject_annotation.dart';
// Import the generated code with a prefix
import 'main.inject.dart' as g;
void main() {
// We'll update this later to use our component
// For now, keep the default main method
runApp(const MyApp());
}
@component
abstract class MainComponent {
// Add a static factory method that references the generated code
static const create = g.MainComponent$Component.create;
@inject
MyAppFactory get myAppFactory;
}
The g.MainComponent$Component.create
reference points to the actual
component implementation generated by inject.dart. By adding this static
factory method, we make it easy to instantiate our component elsewhere in
the code without needing to directly reference the generated file.
This pattern gives us a clean API while keeping the implementation details
hidden in the generated code. The g
prefix helps distinguish between our
code and the generated code.
Troubleshooting the Code Generation
Bad state
When running the code generator, you might encounter this error:
Bad state: package:flutter_demo/main.dart:
component class must declare at least one @inject-annotated provider
This happens because a component must provide at least one injectable type.
To fix this, make sure you've added the @inject
annotation to at least
one getter in your component:
@component
abstract class MainComponent {
static const create = g.MainComponent$Component.create;
@inject
MyAppFactory get myAppFactory;
}
The @inject
annotation tells inject.dart that this getter should be
treated as a provider method. Every component needs at least one provider
to be valid, as a component without providers wouldn't serve any purpose in
a dependency injection system.
Could not find a way to provide
Another possible error you might encounter is:
Could not find a way to provide "MyAppFactory" for component "MainComponent".
This error occurs when inject.dart can't figure out how to create an
instance of the type your component is trying to provide. In this specific
case, it happens because the @assistedInject
annotation is missing from
the MyApp
constructor.
MainComponent
is trying to provide MyAppFactory
, but inject.dart
doesn't know how to create it because there's no constructor marked with
@assistedInject
that matches the factory's creation method signature.
Remember that for assisted injection to work properly:
- The abstract factory must be annotated with
@assistedFactory
- The class constructor must be annotated with
@assistedInject
- Runtime parameters must be annotated with
@assisted
All three parts need to be present for the code generator to successfully create the implementation.
Using the Component
With our dependency injection setup complete and code generated, we can now use the component to create an instance of our app:
void main() {
final mainComponent = MainComponent.create();
final app = mainComponent.myAppFactory.create();
runApp(app);
}
This three-line implementation achieves several important things:
- It initializes our dependency graph by creating the
MainComponent
- It uses the component to get the
MyAppFactory
and create our app instance - It ensures all dependencies are properly injected throughout the application
This approach provides a clear entry point for our dependency injection system. The component acts as the "source of truth" for all dependencies, and by using it to create our app, we establish a clean architectural boundary that makes our code more maintainable and testable.
As we add more dependencies to our application, they'll automatically be injected without any changes needed to this initialization code. This is the power of using a well-structured dependency injection system — the application startup remains clean while the dependency graph can grow more complex beneath the surface.
Conclusion
In this chapter, we've covered the fundamentals of setting up inject.dart in a Flutter application. You've learned how to:
- Set up a new Flutter project with the necessary dependencies
- Create a component to serve as the root of your dependency graph
- Define injectable factories and classes
- Generate the dependency injection code
- Use the component to create and inject dependencies
Complete Example
You can find the complete source code for this quickstart guide in the
examples/flutter_demo
folder of the inject.dart repository.
This example contains all the code we've discussed
in this chapter, properly structured and ready to run.
Reviewing the complete example may help solidify your understanding of how all the pieces fit together in a working application.
Coming Next: Managing Application State
In the next chapter, we'll build on these fundamentals by exploring how to manage application state through dependency injection. We'll refactor our application to follow best practices for state management, introducing repositories, view models, and services that work together through injected dependencies.
This more advanced implementation will demonstrate how dependency injection facilitates clean architecture by separating concerns and making dependencies explicit, resulting in code that's more maintainable, testable, and scalable.