Skip to main content

Eager Module (Synchronous)

This example demonstrates how to use the EagerDependyModule to manage eagerly initialized services, like a LoggerService and CounterService, in a simple counter app. The goal is to show how to set up and use services that are available synchronously, even though their creation is asynchronous.

Setting Up Dependencies

First, define the LoggerService and CounterService used in this example.

LoggerService and ConsoleLoggerService

abstract class LoggerService {
void log(String message);
}

class ConsoleLoggerService extends LoggerService {

void log(String message) {
print('ConsoleLogger: $message');
}
}

CounterService and CounterServiceImpl

abstract class CounterService {
int get counter;

void increment();
}

class CounterServiceImpl implements CounterService {
CounterServiceImpl(this._counter, this.loggerService);

final LoggerService loggerService;
int _counter;


int get counter => _counter;


void increment() {
_counter++;
loggerService.log('Incremented counter to $_counter');
}
}

Creating the EagerDependyModule

We create an EagerDependyModule to initialize and provide instances of services like LoggerService and CounterService synchronously.

late final EagerDependyModule dependy;

Future<void> _initDependy() async {
dependy = await DependyModule(
providers: {
DependyProvider<LoggerService>(
(_) => ConsoleLoggerService(),
),
DependyProvider<CounterService>(
(dependy) async {
final loggerService = await dependy<LoggerService>();

return CounterServiceImpl(5, loggerService);
},
dependsOn: {
LoggerService,
},
),
},
).asEager(); // Use asEager to eagerly initialize dependencies and allow synchronous access
}

Main App Setup

In the main function, we initialize the dependy module synchronously and run the app.

Future<void> main() async {
// Initialize the global [dependy] and make it synchronous
await _initDependy();

runApp(const MyApp());
}

The MyApp Widget

The main app structure remains the same as in other examples.

class MyApp extends StatefulWidget {
const MyApp({super.key});


State<MyApp> createState() => _MyAppState();
}

class _MyAppState extends State<MyApp> {

Widget build(BuildContext context) {
return MaterialApp(
title: 'Example 7 (EagerDependyModule)',
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(
seedColor: Colors.redAccent,
),
useMaterial3: true,
),
home: const MyHomePage(),
);
}
}

MyHomePage Widget

This is where the main UI resides. It includes a counter value displayed using CounterView and a button ( CounterButton) to increment the counter.

class MyHomePage extends StatelessWidget {
const MyHomePage({super.key});


Widget build(BuildContext context) {
return Scaffold(
appBar: AppBar(
backgroundColor: Theme
.of(context)
.colorScheme
.inversePrimary,
title: const Text(
'Example 7 (EagerDependyModule)',
),
),
body: const Center(
child: CounterView(),
),
floatingActionButton: const CounterButton(),
);
}
}

CounterButton Widget

The CounterButton widget increments the counter when pressed. It synchronously accesses both the LoggerService and CounterService to log actions and update the counter.

class CounterButton extends StatelessWidget {
const CounterButton({super.key});


Widget build(BuildContext context) {
return FloatingActionButton(
onPressed: () {
// Access LoggerService synchronously
final loggerService = dependy<LoggerService>();
loggerService.log('CounterButton onPressed');

// Access CounterService synchronously
final counterService = dependy<CounterService>();
counterService.increment();
},
tooltip: 'Increment',
child: const Icon(Icons.add),
);
}
}

CounterView Widget

The CounterView widget displays the current counter value. It listens for updates using DependyNotifierListener, which listens for changes in the CounterService.

class CounterView extends StatelessWidget {
const CounterView({super.key});


Widget build(BuildContext context) {
final counterService = dependy<CounterService>();

return DependyNotifierListener(
notifier: counterService, // Listen for updates from CounterService
builder: (context, notifier) {
return Column(
mainAxisAlignment: MainAxisAlignment.center,
children: [
const Text(
'You have pushed the button this many times:',
),
Text(
'${counterService.counter}',
style: Theme
.of(context)
.textTheme
.headlineMedium,
),
],
);
},
);
}
}

Key Concepts

EagerDependyModule

  • EagerDependyModule is used to initialize services eagerly, making them immediately available for synchronous access throughout the app.
  • The asEager() method is used to eagerly resolve and initialize dependencies, allowing them to be accessed synchronously via dependy<>().

Synchronous Access to Dependencies

In this example, both the LoggerService and CounterService are resolved synchronously. This ensures that they are immediately available when needed, without waiting for asynchronous resolution.

DependyNotifierListener

  • DependyNotifierListener listens for changes in a dependency and rebuilds the widget whenever that dependency is updated.
  • In this example, it listens for changes in the CounterService and updates the UI when the counter value changes.

Example Code

You can view the complete example code for this project on GitHub: example_7/app.dart