mvvm_plus 0.2.0 copy "mvvm_plus: ^0.2.0" to clipboard
mvvm_plus: ^0.2.0 copied to clipboard

MVVM+ is a lightweight Flutter state management library that implements MVVM, plus adds support for sharing business logic across widgets.

mvvm_plus (mvvm+) #

mvvm plus logo

MVVM+ is a lightweight Flutter implementation of MVVM, plus support for sharing business logic across widgets.

MVVM+ employs ChangeNotifiers and cherry picks the best bits of Provider, GetIt, and MVVM (plus adds a few bits of its own), so will be familiar to most Flutter developers.

Model-View-View Model (MVVM) #

As with all MVVM implementations, MVVM+ divides responsibilities into a UI object (called the View), business logic associated with the View (called the View Model), and business logic that spans two or more View Models (called the Model):

mvvm flow

States are mutated in the View Model and the Model, but not the View. With MVVM+, the View is a Flutter widget and the View Model is a Dart model.

MVVM+ goals:

  • Clearly separate business logic from UI.
  • Optionally support access to View Models from anywhere in the widget tree.
  • Work well alone or with other state management packages (BLoC, RxDart, Provider, GetIt, ...).
  • Be scalable and performant, so suitable for both indy and production apps.
  • Be simple.
  • Be small.

Views and View Models #

Your ViewModel class is a Dart class that inherits from ViewModel:

class MyWidgetViewModel extends ViewModel {
  String someText;
}

For your View class, you give the super constructor a builder for your ViewModel (via the "viewModelBuilder" parameter) and extend the View class the same way you extend StatelessWidget widget--you override the build function:

class MyWidget extends View<MyWidgetViewModel> {
  MyWidget({super.key}) : super(viewModelBuilder: () => MyWidgetViewModel());
  @override
  Widget build(BuildContext context) {
    return Text(viewModel.someText); // <- your "viewModel" instance
  }
}

Views are frequently nested and can be large, like an app page, feature, or even an entire app. Or small, like a password field or a button.

Like the Flutter State class associated with StatefulWidget, the ViewModel class has initState and dispose member functions which are handy for subscribing to and canceling listeners to streams, subjects, change notifiers, etc.:

class MyWidgetViewModel extends ViewModel {
  @override
  initState() {
    super.initState();
    _streamSubscription = Services.someStream.listen(myListener);
  }
  @override
  void dispose() {
    _streamSubscription.cancel();
    super.dispose();
  }
  late final StreamSubscription<bool> _streamSubscription;
}

Rebuilding a View #

ViewModel includes a buildView method for rebuilding the View. You can call it explicitly:

class MyWidgetViewModel extends ViewModel {
  int counter;
  void incrementCounter() {
    counter++;
    buildView(); // <- queues View to build
  }
}

Or use buildView as a listener to bind the ViewModel to the View with a ValueNotifier:

class MyWidgetViewModel extends ViewModel {
  final counter = ValueNotifier<int>(0);
  void initState() {
    super.initState();
    counter.addListener(buildView); // <- binds ViewModel to View
  }
}

Retrieving ViewModels from anywhere #

Occasionally you need to access another widget's ViewModel instance (e.g., if it's an ancestor or on another branch of the widget tree). This is accomplished by "registering" the ViewModel with the "register" parameter of the ViewModel constructor (similar to how GetIt works):

class MyOtherWidget extends View<MyOtherWidgetViewModel> {
  MyOtherWidget(super.key) : super(
    viewModelBuilder: () => MyOtherWidgetViewModel(
      register: true, // <- registers ViewModel instance
    ),
  );
}

ViewModels can then "get" the other registered ViewModel:

final otherViewModel = get<MyOtherWidgetViewModel>();

Like GetIt, registered ViewModels are not managed by InheritedWidget. So, widgets don't need to be children of a View widget to get its registered ViewModel. This is a big plus for use cases where the accessed ViewModel is not an ancestor.

Unlike GetIt the lifecycle of all ViewModel instances (including registered) are bound to the lifecycle of the View instances that instantiated them. So, when a View instance is removed from the widget tree, its ViewModel is disposed.

On rare occasions when you need to register multiple ViewModels of the same type, just give each ViewModel instance a unique name:

class MyOtherWidget extends View<MyOtherWidgetViewModel> {
  MyOtherWidget(super.key) : super(
    viewModelBuilder: () => MyOtherWidgetViewModel(
      register: true,
      name: 'Header', // <- unique name
    ),
  );
}

and then get the ViewModel by type and name:

final headerText = get<MyOtherWidgetViewModel>(name: 'Header').someText;
final footerText = get<MyOtherWidgetViewModel>(name: 'Footer').someText;

Adding additional ChangeNotifiers #

Sometimes you want registered models that are not associated with Views. MVVM+ uses the Registrar package under the hood which has a widget named "Registrar" that you can add to the widget tree:

Registrar<MyModel>(
  builder: () => MyModel(),
  child: MyWidget(),
);

The Registrar widget registers the model when added to the widget tree and unregisters it when removed. To register multiple models with a single widget, check out MultiRegistrar.

Listening to other widget's ViewModels #

The ViewModel get method retrieves registered ViewModels but does not listen for future changes. For that, use listenTo from within your ViewModel:

final text = listenTo<MyOtherWidgetViewModel>().someText;

listenTo performs a one-time add of the buildView method as a listener that is called every time the notifyListeners method of MyOtherWidgetViewModel is called. If you want to do more than just queue a build, you can give listenTo a listener function:

listenTo<MyWidgetViewModel>(listener: myListener);

If you want to rebuild your View after your custom listener finishes, just call buildView within your listener:

void myListener() {
  // do some stuff
  buildView(); 
}

Either way, listeners added by listenTo are automatically removed when your ViewModel instance is disposed.

notifyListeners vs buildView #

When your View and ViewModel classes are instantiated, buildView is added as a listener to your ViewModel. So, calling buildView or notifyListeners from within your ViewModel will both rebuild your View. So, what's the difference between calling buildView and notifyListeners? Nothing, unless your ViewModel is registered--any listeners to your registered ViewModel will be called on notifyListeners but not on buildView. So, to eliminate unnecessary View builds, it is a best practice to use buildView unless your use case requires listeners to be notified of a change.

ValueNotifiers #

If you want more granularity than registering an entire ViewModel or service, you can register a ValueNotifier instead:

Registrar.register<MyValueNotifier>(instance: myValueNotifier);

And get or listenTo to the ValueNotifier the same way as a registered ViewModel (because they are both subclasses of ChangeNotifier):

final myValue = listenTo<MyValueNotifier>().value;

That's it! #

The example app demos much of the above functionality and shows how small and organized MVVM+ classes typically are.

If you have questions or suggestions on anything MVVM+, please do not hesitate to contact me.

35
likes
0
pub points
65%
popularity

Publisher

verified publisherrichardcoutts.com

MVVM+ is a lightweight Flutter state management library that implements MVVM, plus adds support for sharing business logic across widgets.

Repository (GitHub)
View/report issues

License

unknown (LICENSE)

Dependencies

cupertino_icons, equatable, flutter, registrar

More

Packages that depend on mvvm_plus