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

A Flutter implementation of MVVM plus support for sharing business logic across widgets.

mvvm_plus (mvvm+) #

mvvm+ logo

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

MVVM+ employs ChangeNotifiers and cherry picks the best bits of provider, get_it, and mvvm (plush 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 an immutable rendering (called the View) and a presentation model (called the View 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:

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

Views and View Models #

With MVVM+ you extend the View the way your extend a StatelessWidget widget. E.g., 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); // <- state is stored in your custom "viewModel" instance
  }
}

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

class MyWidgetViewModel extends ViewModel {
  String someText;
}

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 provides initState() and dispose() which is 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 the 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 rebuild
  }
}

Or 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);   // <- calls buildView on new value for counter
  }
}

Retrieving View Models 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 View Model with the "register" parameter of the ViewModel constructor (similar to how get_it works):

class MyOtherWidget extends View<MyOtherWidgetViewModel> {
  MyOtherWidget(super.key) : super(
    viewModelBuilder: () => MyOtherWidgetViewModel(
      register: true, // <- registers the View Model so other widgets and models can access
    ),
  );
}

Widgets and models can then "get" the registered View Model with the ViewModel function get:

final otherViewModel = get<MyOtherWidgetViewModel>();

Like get_it, MVVM+ uses singletons that are not managed by InheritedWidget. So, widgets don't need to be children of a View widget to get its registered View Model. This is a big plus for use cases where the accessed View Model is not an ancestor.

Unlike get_it 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 View Models of the same type, just give each View Model instance a unique name:

class MyOtherWidget extends View<MyOtherWidgetViewModel> {
  MyOtherWidget(super.key) : super(
    viewModelBuilder: () => MyOtherWidgetViewModel(
      register: true,
      name: 'Header', // <- distinguishes from other registered View Models of the same type
    ),
  );
}

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 #

The ViewModel constructor optionally registers a View Model, but sometimes you want registered models that are not associated with Views. MVVM+ uses Registrar under the hood which has widget that you can add to the widget tree:

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

The Registar 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 View Models #

ViewModel adds a get method that retrieves models registered by another ViewModels. (Or registered with a Registrar widget)

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

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

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

listenTo adds the buildView method as a listener to queue View to build every time MyOtherWidgetViewModel.notifyListeners() 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:

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

Either way, listeners passed to 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, except if your ViewModel is registered--any listeners to your registered ViewModel will be called on notifyListeners but not on buildView. So, to eliminate unnecessary 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 sharing an entire ViewModel or service you can register a ValueNotifier instead using Registrar:

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

And get or listenTo to the ValueNotifier the same 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

A Flutter implementation of MVVM plus 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