In the realm of managing dependencies, simplicity and efficiency are key. This principle is reflected in ServiceWaiter
, a powerful Flutter library for managing dependencies, drawing inspiration from SwiftUI's dependency injection.
The ServiceWaiter
manages all the services in your application by providing a central location for registration and retrieval. It primarily consists of two parts:
- A
ServiceWaiter
class that manages dependencies. - A set of extensions on Flutter's
Widget
andBuildContext
classes for easy access to dependencies within widgets.
final class ServiceWaiter {
// Cache for storing instantiated dependencies
final _cache = <Type, Object>{};
// Factory for creating dependencies
final _factory = <Type, Object Function()>{};
...
}
The ServiceWaiter
class has a _cache
to store instantiated dependencies and a _factory
that holds functions responsible for creating the dependencies. Dependencies can be registered with the register
method, and retrieved with the get
method. This method checks the cache for an existing instance or creates a new one if not present. clearSingleton
and clear
methods are available to remove specific or all cached instances, respectively.
extension Environment on Widget { ... }
extension ContextDependency on BuildContext { ... }
The Environment
extension is defined on the Widget
class and provides a method to create a ServiceWaiterProvider
, passing an updater function. The ContextDependency
extension is defined on the BuildContext
class and offers a dependency
method to retrieve dependencies from the nearest ServiceWaiterProvider
in the widget tree. This retrieval is done recursively, ensuring that the method finds the closest instance of the required dependency.
Let's explore an example showing how ServiceWaiter
can elegantly handle passing dependencies to routes and manage a global authentication use-case.
class AuthUseCase {
bool isAuthenticated() {
// authentication logic here
}
}
class UserInfo {
// user information here
}
class AppSettings {
// app settings here
}
void main() {
runApp(
MaterialApp(
home: HomePage().environment(
(serviceWaiter) {
serviceWaiter.register((_) => AuthUseCase());
},
),
onGenerateRoute: (settings) {
switch (settings.name) {
case '/profile':
return MaterialPageRoute(
builder: (context) => ProfilePage().environment(
(serviceWaiter) {
serviceWaiter.register((_) => UserInfo());
},
),
);
case '/settings':
return MaterialPageRoute(
builder: (context) => SettingsPage().environment(
(serviceWaiter) {
serviceWaiter.register((_) => AppSettings());
},
),
);
default:
return MaterialPageRoute(builder: (context) => HomePage());
}
},
),
);
}
In this example, an AuthUseCase
dependency is registered in the HomePage
. Since AuthUseCase
will likely be needed throughout the application, registering it at the root of the widget tree makes it accessible everywhere. Notice how we can pass different dependencies to different routes via the constructor.
class ProfilePage extends StatelessWidget {
@override
Widget build(BuildContext context) {
final authUseCase = context.dependency<AuthUseCase>();
final userInfo = context.dependency<UserInfo>();
if (!authUseCase.isAuthenticated()) {
return
Text('Please log in to view your profile.');
}
// build profile details using userInfo
}
}
class SettingsPage extends StatelessWidget {
@override
Widget build(BuildContext context) {
final appSettings = context.dependency<AppSettings>();
// build settings page using appSettings
}
}
In ProfilePage
and SettingsPage
, we retrieve the dependencies using context.dependency<T>()
. This function will recursively attempt to find the nearest dependency in the widget tree that matches the requested type. As a result, each route can access its own dependencies, as well as those provided at the root level, like AuthUseCase
.
To wrap up, ServiceWaiter
presents a clean and efficient mechanism to handle dependencies in a Flutter application. This SwiftUI-inspired solution helps to keep your codebase organized and efficient, proving that the beauty of dependency injection isn't restricted to just one platform.