Managing Application State through Dependency Injection
After setting up the basic structure of our application with dependency injection, we want to explore how a well-designed project structure can work alongside state management through DI. This approach aligns with Flutter's recommended best practices for app architecture and state management.
Before we dive into state management, let's refactor our application to better separate concerns by extracting the app and home widgets into their own files. This restructuring will provide a cleaner foundation for implementing state management through dependency injection.
Refactoring the Application Structure
First, let's reorganize our code to separate the UI components from the dependency injection setup. This creates a more maintainable architecture as our application grows.
Extracting Widgets into Feature-Based Files
We'll extract our widgets from main.dart
into separate files using a
feature-based organization:
- Create a
src/features/app
directory for the application widget - Create a
src/features/home
directory for the home page widget - Move relevant code while maintaining dependencies and annotations
Here's how we'll refactor each file:
- main.dart - Will contain only the DI setup and application entry point:
import 'package:flutter/material.dart';
import 'package:inject_annotation/inject_annotation.dart';
import 'main.inject.dart' as g;
import 'src/features/app/app.dart';
void main() {
final mainComponent = MainComponent.create();
final app = mainComponent.myAppFactory.create();
runApp(app);
}
@component
abstract class MainComponent {
static const create = g.MainComponent$Component.create;
@inject
MyAppFactory get myAppFactory;
}
- my_app.dart - Contains the app widget and its factory:
import 'package:flutter/material.dart';
import 'package:inject_annotation/inject_annotation.dart';
import '../home/home.dart';
@assistedFactory
abstract class MyAppFactory {
MyApp create({Key? key});
}
class MyApp extends StatelessWidget {
@assistedInject
const MyApp({@assisted super.key});
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
useMaterial3: true,
),
home: const MyHomePage(title: 'Flutter Demo Home Page'),
);
}
}
- my_home.dart - Contains the home page widget:
import 'package:flutter/material.dart';
class MyHomePage extends StatefulWidget {
const MyHomePage({super.key, required this.title});
final String title;
@override
State<MyHomePage> createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> {
int _counter = 0;
void _incrementCounter() {
setState(() {
_counter++;
});
}
@override
Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
title: Text(widget.title),
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
const Text(
'You have pushed the button this many times:',
),
Text(
'$_counter',
style: Theme.of(context).textTheme.headlineMedium,
),
],
),
),
floatingActionButton: FloatingActionButton(
onPressed: _incrementCounter,
tooltip: 'Increment',
child: const Icon(Icons.add),
),
);
}
}
This refactoring creates a cleaner, more scalable project structure with these technical benefits:
- Separation of Concerns: Each file now has a distinct responsibility
- Improved Maintainability: Changes to one feature won't affect files for other features
- Better Dependency Management: Import paths clearly show dependencies between features
- Enhanced Discoverability: New team members can quickly locate components by feature
- DI-Friendly Organization: Structure supports targeted injection of dependencies where needed
The resulting directory structure looks like this:
flutter_demo/
└── lib/
├── main.dart (DI setup and entry point)
└── src/
└── features/
├── app/
│ └── my_app.dart (MyApp and MyAppFactory)
└── home/
└── my_home.dart (MyHomePage)
With this structure in place, we're now ready to implement state management through dependency injection.
Implementing State Management with Dependency Injection
Now that we have a clean project structure, we'll implement state
management using dependency injection principles. Our architecture will
introduce two new components: a CounterRepository
for data persistence
and a MyHomePageViewModel
to connect the UI with this data layer.
The CounterRepository
will handle the persistence of our counter value.
It will provide methods to store and retrieve the counter, effectively
abstracting the storage mechanism from the rest of the application. This
abstraction is powerful because it allows us to change the underlying
storage implementation without affecting the components that use it.
The MyHomePageViewModel
will serve as an intermediary between our UI and the
data layer. It will depend on the CounterRepository
(injected through our
DI system) and provide the UI with the state and methods it needs. This
creates a clean separation between how data is presented and how it's
stored or processed.
This architecture demonstrates one of the key benefits of dependency
injection: the ability to create a system where components depend on
abstractions rather than concrete implementations. When the
MyHomePageViewModel
receives its CounterRepository
through injection, it
doesn't need to know the details of how the repository is implemented or
how it's instantiated.
This separation creates more maintainable code because changes to one layer don't necessarily affect others. It also dramatically improves testability since each component can be tested in isolation with mock implementations of its dependencies. The repository could be reused across different features if needed, showcasing how DI promotes code reuse through proper component design.
After refactoring our application to use the MyHomeViewModel
and the
CounterRepository
for handling the counter value, the project structure
now looks like this:
flutter_demo/
└── lib/
├── main.dart (DI setup and entry point)
└── src/
├── data/
│ ├── repositories/
│ │ └── counter_repository.dart (Manages counter state)
│ └── services/
│ └── database.dart (Simulated database service)
└── features/
├── app/
│ └── my_app.dart (MyApp and MyAppFactory)
└── home/
├── my_home_page.dart (MyHomePage UI component)
└── my_home_page_view_model.dart (Home state management)
Making MyHomePage Injectable
To make our home page injectable, we create a factory that allows the DI system to instantiate it:
import 'package:flutter/material.dart';
import 'package:inject_annotation/inject_annotation.dart';
/// Factory to create the [MyHomePage] widget with its dependencies.
@assistedFactory
abstract class MyHomePageFactory {
MyHomePage create({Key? key, required String title});
}
/// The home page of the application with a simple counter.
class MyHomePage extends StatelessWidget {
@assistedInject
const MyHomePage({
@assisted super.key,
@assisted required this.title,
});
// Widget implementation...
}
-
We've created a factory (
MyHomePageFactory
) that allows the DI system to create instances ofMyHomePage
with all dependencies properly injected. -
We use
@assistedInject
to mark the constructor as an injection point, while marking runtime parameters likekey
andtitle
with@assisted
. -
Notice that we also changed it to be a
StatelessWidget
instead of aStatefulWidget
. When we later add the view model, we'll see how we use the view model to manage the state of the widget.
Connecting MyHomePage to MyApp
In the MyApp
widget, we inject the MyHomePageFactory
and use it to
create the home page:
import 'package:flutter/material.dart';
import 'package:inject_annotation/inject_annotation.dart';
import '../home/my_home_page.dart';
/// Factory to create the [MyApp] widget with the [MyHomePageFactory] injected.
@assistedFactory
abstract class MyAppFactory {
MyApp create({Key? key});
}
/// The root widget of the application.
/// The [MyHomePageFactory] is injected into the widget at compile-time.
class MyApp extends StatelessWidget {
@assistedInject
const MyApp({
@assisted super.key,
required this.homePageFactory,
});
final MyHomePageFactory homePageFactory;
@override
Widget build(BuildContext context) {
return MaterialApp(
title: 'Flutter Demo',
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
useMaterial3: true,
),
home: homePageFactory.create(title: 'Flutter Demo Home Page'),
);
}
}
This structure creates a clean dependency chain:
- The
MyApp
widget depends onMyHomePageFactory
(injected) - The
homePageFactory.create()
method is called to instantiate the home page
This approach demonstrates nested widget creation through dependency injection, allowing each component to receive its required dependencies.
By using factories, we ensure proper dependency injection throughout our
widget tree while still allowing for runtime parameters like title
and
key
.
This approach gives us the best of both worlds: compile-time dependency
injection with runtime flexibility.
Adding State Management with View Model
Now that we have our basic widget structure with dependency injection, we need to implement state management. We'll use a view model approach that provides a clean separation between UI and business logic.
Creating the View Model
First, let's create a view model class that handles the state and business logic for our counter feature:
import 'package:flutter/material.dart';
import 'package:inject_annotation/inject_annotation.dart';
import '../../data/repositories/counter_repository.dart';
import 'my_home_page.dart';
/// The view model for the [MyHomePage] widget.
@inject
class MyHomePageViewModel extends ChangeNotifier {
MyHomePageViewModel({required CounterRepository repository})
: _repository = repository;
final CounterRepository _repository;
int count = 0;
Future<void> increaseCount() async {
await _repository.increaseCount();
count = await _repository.count;
notifyListeners();
}
}
The view model has these important characteristics:
- It's marked with
@inject
so it can be created by the DI system - It extends
ChangeNotifier
to provide change notifications to the UI - It depends on
CounterRepository
, which is injected through its constructor - It manages state (the
count
variable) and provides a method to update it - It calls
notifyListeners()
when the state changes to trigger UI updates
This pattern creates a clear separation of concerns:
- The view model handles business logic and state management
- The repository handles data operations
- The UI focuses solely on presentation
Note the constructor implementation pattern:
MyHomePageViewModel({required CounterRepository repository})
: _repository = repository;
final CounterRepository _repository;
Rather than using the more concise this.repository
syntax and a public field,
we deliberately use a private field with manual assignment to enforce strict
encapsulation. This approach provides significant architectural benefits:
-
True Encapsulation: Dependencies like
_repository
remain truly private. If we use a public field, any component that received the view model could potentially access its repository directly. This would violate the encapsulation principle and make it difficult to change the implementation later without breaking existing code. -
Preventing Dependency Leakage: When a view model is injected into a UI component, we want to ensure the UI can only access the intended public API. Manual assignment to private fields creates a clear boundary that prevents dependency leakage across architectural layers.
-
Layer Isolation: This pattern supports the principle that each layer should only know about its immediate dependencies. The UI knows about the view model but should have no knowledge of or access to the repositories or services the view model uses.
This small syntax choice reinforces an important architectural principle: components should expose only what their consumers need and nothing more, maintaining clear boundaries between different layers of the application.
Injecting the View Model with ViewModelFactory
Now, let's update our MyHomePage
to use this view model with the
ViewModelFactory
pattern:
import 'package:flutter/material.dart';
import 'package:inject_annotation/inject_annotation.dart';
import 'package:inject_flutter/inject_flutter.dart';
import 'my_home_page_view_model.dart';
/// Factory to create the [MyHomePage] widget with the [MyHomePageViewModel] injected.
@assistedFactory
abstract class MyHomePageFactory {
MyHomePage create({Key? key, required String title});
}
/// The home page of the application with a simple counter.
/// The [viewModelFactory] is injected into the widget at compile-time, while the
/// [key] or [title] can be provided at runtime.
class MyHomePage extends StatelessWidget {
@assistedInject
const MyHomePage({
@assisted super.key,
@assisted required this.title,
required this.viewModelFactory,
});
final String title;
final ViewModelFactory<MyHomePageViewModel> viewModelFactory;
@override
Widget build(BuildContext context) {
return viewModelFactory(
builder: (context, viewModel, _) {
return Scaffold(
appBar: AppBar(
backgroundColor: Theme.of(context).colorScheme.inversePrimary,
title: Text(title),
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
const Text(
'You have pushed the button this many times:',
),
Text(
'${viewModel.count}',
style: Theme.of(context).textTheme.headlineMedium,
),
],
),
),
floatingActionButton: FloatingActionButton(
onPressed: viewModel.increaseCount,
tooltip: 'Increment',
child: const Icon(Icons.add),
),
);
},
);
}
}
How ViewModelFactory Works
The ViewModelFactory<T>
is a typedef for a function that returns a
ViewModelBuilder<T>
:
typedef ViewModelFactory<T extends ChangeNotifier> = ViewModelBuilder<T> Function({
Key? key,
required ViewModelWidgetBuilder<T> builder,
Widget? child,
});
When you call this function in the build method, it:
- Creates a ViewModelBuilder: This
StatefulWidget
handles the view model lifecycle - Passes Your Builder Function: Your UI-building logic receives the view model instance
- Manages View Model Creation: The view model is created when the
ViewModelBuilder
first builds - Handles View Model Disposal: When the
ViewModelBuilder
is disposed, it disposes the view model
The builder function pattern provides a clean way to access the view model's state and methods inside your UI code. By using this approach:
- The UI reacts to changes in the view model automatically
- Business logic stays in the view model
- The widget remains a simple
StatelessWidget
- Lifecycle management happens behind the scenes
This creates a maintainable architecture where each component has a clear responsibility.
Data Layer: Repository and Database
Let's explore how we implement dependency injection for our data layer components.
The Repository Pattern
Our CounterRepository
acts as a mediator between the UI layer and the
data storage:
import 'package:inject_annotation/inject_annotation.dart';
import '../services/database.dart';
/// Repository to manage the counter value.
/// Uses the [Database] to persist the counter value.
@inject
@singleton
class CounterRepository {
CounterRepository({required Database database}) : _database = database;
final Database _database;
Future<int> get count async => _database.selectCount();
Future<void> increaseCount() async {
final count = await _database.selectCount();
await _database.updateCount(count + 1);
}
}
Notice the @singleton
annotation - this tells our DI system to create
only one instance of the repository throughout the application.
This is crucial because:
- We want a single source of truth for data operations
- It ensures consistent state management across the application
- It avoids redundant database connections
The repository depends on the Database
, which is injected through its
constructor.
This creates a clean separation between data access logic and storage
implementation.
Providing the Database Through a Module
Since we are using a database from a third-party library that we cannot
annotate with @inject
, we use a module to provide it:
import 'package:inject_annotation/inject_annotation.dart';
/// Module to provide the database instance.
/// Modules are used to provide instances of classes from 3rd party libraries
/// that can't be annotated with [inject].
@module
class DataBaseModule {
@provides
@singleton
Database provideDatabase() => Database();
}
/// Simulates a 3rd party database library for demonstration purposes.
///
/// This class mimics what you might find in an actual database package
/// like Drift, Isar, or Hive, but with simplified functionality to focus
/// on dependency injection concepts. In a real app, you would replace this
/// with an actual database implementation.
///
/// Usage example:
/// ```dart
/// final db = Database();
/// await db.updateCount(5);
/// final value = await db.selectCount(); // Returns 5
/// ```
class Database {
/// In-memory storage for the counter value.
/// In a real database, this would be persisted to disk.
int _count = 0;
/// Simulates updating a record in the database.
///
/// In a real database, this would write to persistent storage.
Future<void> updateCount(int count) async {
_count = count;
}
/// Simulates reading a record from the database.
///
/// In a real database, this would fetch data from persistent storage.
Future<int> selectCount() {
return Future.value(_count);
}
}
The @module
annotation defines a class that provides dependencies.
The @provides
method tells inject.dart how to create an instance of the
Database
class.
By adding the @singleton
annotation, we ensure only one
database instance exists in our application.
This approach demonstrates how to integrate third-party libraries into your dependency injection system, even when you can't modify their source code.
View Models: Why They're Not Singletons
In contrast to the repository and database, view models are deliberately not marked as singletons. This is an important architectural decision:
- Lifecycle Alignment: Each view should have its own view model instance that matches its lifecycle
- State Isolation: Different instances of the same view should have isolated state
- Memory Efficiency: View models can be garbage collected when their associated view is disposed
If view models were singletons, all instances of a view would share the same state, creating unexpected behavior and potential memory leaks. By making each view model instance-specific while keeping the data layer as singletons, we create a clean hierarchy where stable infrastructure is shared while UI state remains isolated.
This pattern demonstrates a key strength of dependency injection: the ability to configure different scopes for different types of components in your application.
Conclusion
In this chapter, we've explored how dependency injection naturally complements state management in Flutter applications. By separating our application into clean layers — UI components, view models, repositories, and services — we've created a maintainable architecture that's both flexible and testable.
The key principles we've covered include:
- Using dependency injection to provide state management services
- Creating a clear separation between UI and business logic
- Implementing proper lifecycle management with ViewModelFactory
- Leveraging singletons for shared infrastructure like repositories and databases
Complete Example
You can find the complete source code for all examples in this chapter in
the examples/flutter_demo
folder of the inject.dart repository. This working implementation
demonstrates all the patterns and practices we've discussed.
Coming Next: Testing with Dependency Injection
In the next chapter, we'll explore one of the most powerful benefits of our architecture: testability. We'll show how to create a separate dependency graph for unit testing that allows us to:
- Test view models by injecting a
FakeCounterRepository
- Test repositories with a
FakeDatabase
implementation
This testing approach demonstrates how dependency injection doesn't just make your code more maintainable — it makes it substantially easier to verify and validate your application's behavior through automated testing.
By separating concerns and making dependencies explicit, we've built a foundation that will continue to pay dividends as your application grows in complexity.