ScopedDependyMixin
This example demonstrates how to use the ScopedDependyMixin
to manage scoped dependencies in a Flutter app, such
as a LoggerService
& Example1State
, with a simple counter app. The goal is to show how to inject and use
dependencies in a scoped manner with ScopedDependyMixin
.
Step 1: Define the LoggerService
First, define an abstract class for the logger service:
abstract class LoggerService {
void log(String message);
}
Then, create an implementation of the logger service:
class ConsoleLoggerService extends LoggerService {
void log(String message) {
print('ConsoleLogger: $message');
}
}
Step 2: Create the Dependy Module
Next, create a module to provide the LoggerService
. This module will be used to register the ConsoleLoggerService
as
the implementation for LoggerService
.
Note: This module we assume it as a singleton module that does live across the app process lifespan
final example1ServicesModule = DependyModule(
providers: {
DependyProvider<LoggerService>(
(_) => ConsoleLoggerService(),
),
},
);
Step 3: Define the State Class with a ChangeNotifier
Now, create a ChangeNotifier
class to manage the app's state, which includes the counter. This class will also use the
LoggerService
to log actions like incrementing or decrementing the counter.
class Example1State with ChangeNotifier {
Example1State(this.loggerService);
final LoggerService loggerService;
int _counter = 0;
int get counter => _counter;
void incrementCounter() {
_counter++;
notifyListeners();
loggerService.log('incrementCounter $counter');
}
void decrementCounter() {
_counter--;
notifyListeners();
loggerService.log('decrementCounter $counter');
}
}
Step 4: Create the Main Flutter Application
Set up the main MyApp
widget and the MyHomePage
widget. This is where the ScopedDependyMixin
is applied,
allowing us to scope dependencies like Example1State
within this widget's lifecycle.
void main() {
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
Widget build(BuildContext context) {
return MaterialApp(
title: 'Example 1 (Using ScopedDependyMixin)',
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
useMaterial3: true,
),
home: const MyHomePage(),
);
}
}
Step 5: Apply ScopedDependyMixin
to the State
Class
Now, apply ScopedDependyMixin
to your State
class (_MyHomePageState
). This mixin gives you scoped dependency
management for your StatefulWidget
. Here’s where you will handle the state (with a counter) and inject the
dependencies (like Example1State
).
class MyHomePage extends StatefulWidget {
const MyHomePage({super.key});
State<MyHomePage> createState() => _MyHomePageState();
}
/// This class applies the [ScopedDependyMixin] to manage dependencies.
class _MyHomePageState extends State<MyHomePage> with ScopedDependyMixin {
Widget build(BuildContext context) {
return FutureBuilder(
/// [watchDependy] triggers a rebuild when the watched dependency changes.
future: watchDependy<Example1State>(),
builder: (context, snapshot) {
final state = snapshot.data;
return Scaffold(
appBar: AppBar(
backgroundColor: Theme
.of(context)
.colorScheme
.inversePrimary,
title: const Text('Example 1 (Using ScopedDependyMixin)'),
),
body: Center(
child: Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
const Text(
'You have pushed the button this many times:',
),
Text(
'${state?.counter}',
style: Theme
.of(context)
.textTheme
.headlineMedium,
),
],
),
),
floatingActionButton: FloatingActionButton(
onPressed: () async {
/// Call the increment function directly.
final state = await dependy<Example1State>();
state.incrementCounter();
},
tooltip: 'Increment',
child: const Icon(Icons.add),
),
);
},
);
}
/// This method returns the module scoped to the widget's lifecycle.
DependyModule moduleBuilder() {
return DependyModule(
providers: {
DependyProvider<Example1State>(
(dependy) async {
// Resolve LoggerService dependency
final logger = await dependy<LoggerService>();
// Return the Example1State with the resolved logger
return Example1State(logger);
},
dependsOn: {
LoggerService,
},
),
},
modules: {
example1ServicesModule,
},
);
}
}
Key Concepts
-
ScopedDependyMixin
: This mixin is added to theState
class to manage dependencies. It helps with dependency scoping, ensures dependencies are created, and handles rebuilding the widget when the state changes. -
watchDependy<Example1State>()
: This function retrieves theExample1State
from the dependency graph and rebuilds the widget when any changes occur in the state (like incrementing or decrementing the counter). -
dependy<Example1State>()
: This function is used to retrieve the dependency (Example1State
) without setting up a listener. It's useful for direct calls like incrementing the counter, where you don’t need to watch the state. -
moduleBuilder()
: This method is responsible for returning the scoped module that contains the providers and dependencies needed by the widget. It ensures that the dependencies are scoped to the widget’s lifecycle and will be disposed when the widget is removed from the tree.
Example Code
You can view the complete example code for this project on GitHub: example_1/app.dart