Share Scope (ScopedDependyMixin)
In this example, we demonstrate how to use ScopedDependyMixin
in a StatefulWidget
to share a dependency scope across
multiple widgets. Unlike ScopedDependyProvider
(shown in the previous example), ScopedDependyMixin
achieves the same
goal of sharing the scope by using shareDependyScope()
inside a StatefulWidget
.
This example provides a CounterService
dependency that both CounterButton
and CounterView
widgets can access
directly from the shared scope.
Step 1: Define the CounterService
This service manages the counter state and provides methods to increment and retrieve the counter value.
abstract class CounterService {
int get counter;
void increment();
}
class SimpleCounterService implements CounterService {
int _counter = 0;
int get counter => _counter;
void increment() {
_counter++;
}
}
Step 2: Create the Dependy Module
Set up a module that provides SimpleCounterService
as the CounterService
implementation.
final example4ServicesModule = DependyModule(
providers: {
DependyProvider<CounterService>(
(_) => SimpleCounterService(),
),
},
);
Step 3: Set Up the Main Application
void main() {
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
Widget build(BuildContext context) {
return MaterialApp(
title: 'Example 4 (Share Scope ScopedDependyMixin)',
theme: ThemeData(
colorScheme: ColorScheme.fromSeed(seedColor: Colors.deepPurple),
useMaterial3: true,
),
home: const MyHomePage(),
);
}
}
Step 4: Use ScopedDependyMixin
in MyHomePage
In the MyHomePage
widget, which is a StatefulWidget
, we apply ScopedDependyMixin
and use shareDependyScope()
to
make the dependency scope available to child widgets.
class MyHomePage extends StatefulWidget {
const MyHomePage({super.key});
State<MyHomePage> createState() => _MyHomePageState();
}
class _MyHomePageState extends State<MyHomePage> with ScopedDependyMixin {
Widget build(BuildContext context) {
/// Using [shareDependyScope] here achieves the same result as `ScopedDependyProvider` with `shareScope: true`.
return shareDependyScope(
child: Scaffold(
appBar: AppBar(
backgroundColor: Theme
.of(context)
.colorScheme
.inversePrimary,
title: const Text('Example 4 (Share Scope ScopedDependyMixin)'),
),
body: const Center(
/// [CounterView] can access the shared scope directly, without needing props.
child: CounterView(),
),
/// [CounterButton] can also use the shared scope to access [CounterService].
floatingActionButton: const CounterButton(),
),
);
}
DependyModule moduleBuilder() {
/// Only provide additional dependencies if necessary. Here, `example4ServicesModule`
/// is added to the module list for this widget.
return DependyModule(
providers: {},
modules: {
example4ServicesModule,
},
);
}
}
Step 5: Define the CounterButton
Widget
The CounterButton
widget increments the counter by interacting with the shared CounterService
dependency scope.
Important Note:
CounterButton
uses getDependyScope(context)
to access the dependency without setting up a listener. This is
efficient for non-observing tasks like calling the increment
method.
class CounterButton extends StatelessWidget {
const CounterButton({super.key});
Widget build(BuildContext context) {
final scope = getDependyScope(context);
return FloatingActionButton(
onPressed: () async {
final counterService = await scope.dependy<CounterService>();
counterService.increment();
},
tooltip: 'Increment',
child: const Icon(Icons.add),
);
}
}
Step 6: Define the CounterView
Widget
The CounterView
widget displays the current counter value by accessing and observing the CounterService
dependency
for changes.
Important Note:
Here, scope.watchDependy<CounterService>()
is used, setting up a listener to automatically update CounterView
when
the counter value changes.
class CounterView extends StatelessWidget {
const CounterView({super.key});
Widget build(BuildContext context) {
return ScopedDependyConsumer(
builder: (context, scope) {
return FutureBuilder(
future: scope.watchDependy<CounterService>(),
builder: (context, snapshot) {
final counterService = snapshot.data;
return Column(
mainAxisAlignment: MainAxisAlignment.center,
children: <Widget>[
const Text(
'You have pushed the button this many times:',
),
Text(
'${counterService?.counter}',
style: Theme
.of(context)
.textTheme
.headlineMedium,
),
],
);
},
);
},
);
}
}
Key Concepts
-
ScopedDependyMixin
: Applied in aStatefulWidget
to share a dependency scope. UsingshareDependyScope()
insidebuild()
makes the scope accessible to all descendant widgets. -
ScopedDependyConsumer
: Used in bothCounterButton
andCounterView
to manage dependency access and widget rebuilds. It allows each widget to interact with dependencies without causing unnecessary rebuilds of other parts of the widget tree. -
Why use
.dependy
inCounterButton
andwatchDependy
inCounterView
?.dependy<CounterService>()
: Retrieves the dependency without observing changes. Ideal forCounterButton
since it only needs to callincrement
..watchDependy<CounterService>()
: Sets up a listener onCounterService
, causingCounterView
to rebuild whenever the counter updates. This approach is essential for displaying the latest counter value.
-
Isolating Rebuilds:
Using
ScopedDependyConsumer
isolates rebuilds within each widget (CounterButton
andCounterView
). Although we could usegetDependyScope(context)
and callscope.watchDependy<CounterService>()
directly, this would set up the listener at the top level, causing the entire provider to rebuild. By usingScopedDependyConsumer
, we control rebuilds within each widget individually. -
DependyModule
withexample4ServicesModule
: Defines the services and dependencies for the application. The module includesexample4ServicesModule
, allowingCounterService
to be accessible in the shared scope without redefinition.
Example Code
The complete code for this example can be found on GitHub: example_4/app.dart