ViewModel injection in ViewControllers with Storyboards

Wolox Engineering
Wolox
Published in
3 min readFeb 18, 2016

When applying MVVM architecture in iOS, we face some problems regarding the binding and unbinding of a ViewModel to its corresponding ViewController. We consider that the ideal situation would be to inject the view model on initialization, but when using storyboards, this isn’t possible. Having this in mind, how should we inject the ViewModel to its corresponding ViewController?

At Wolox, we use the following approach: The ViewController gets its ViewModel injected and when the view is loaded, we proceed to bind it. Given that ViewControllers cannot receive parameters on initialization when using storyboards, we are left with two options: either assuming that there will always be a ViewModel (explicit dereferenced property) or having the ViewModel as optional. We prefer the second approach, as it isn’t advisable to explicitly dereference a property (due to type safety).

So for each ViewController, we ended up with the same implementation:

We can read this as “when assigning a new viewModel, if we already had a viewModel, first unbind it. When it is assigned, if the view is loaded, proceed to bind the viewModel.”

The problem with this approach is that we end up adding these lines of code in every ViewController with an associated viewModel. It would be great to abstract this behaviour.

First idea

The first approach is to create a class ViewController which inherits from UIViewController and subclass it. There is a problem with this approach: we have many types of ViewControllers which already inherited from UIViewController (for example UITableViewController, UICollectionViewController, etc…), so this forces us to subclass each of them, which isn’t the best approach as we will end up with repeated code.

Second idea

One would then think, let’s use one of the coolest swift features: protocols. We create a ViewModelBindable protocol that looks like this:

Here we are saying that ViewModelBindable is a protocol that can only be implemented by a class (in our case, UIView or UIViewController) and that class has to define what type the viewModel is (typealias ViewModel), a property viewModel of type ViewModel, and how the ViewController will bind/unbind to that property.

Now that we could define the protocol, the next step will be adding the ViewModelBindable conformance for our ViewControllers. One would consider adding a default implementation of the protocol (yet another cool Swift feature), something like this:

Unfortunately, this doesn’t compile, we cannot have stored properties on extensions, only computed properties. So now we are facing the same problem as before: we created the protocol ViewModelBindable, but the viewModel remains in each ViewController.

Our solution

Luckily, there is a way to solve this and it is using a cool Objective-C feature: associated objects. This allows us to store the viewModel in an extension.

Note that for this to work, we need to box view models when they are structs (since associated objects only work for classes).

Isn’t it great? We abstracted the view model using protocols and associated objects. Now each ViewController only needs to define which type the viewModel is and how it should be bound.

For example:

Note that Xcode infers from the type signature of bindViewModel that for MyViewController ViewModel type is MyViewModel.

We could also overwrite the default implementation of unbindViewModel(), which may be useful for example when binding/unbinding cells.

Here’s the full implementation.

Now that we could abstract the binding/unbinding of view models, new interesting things come into light. For example, we are working on a tool which tracks the binding/unbinding of ViewModels for detecting possible memory leaks. Sounds great, doesn’t it?

Posted by Francisco Depascuali, iOS Developer at Wolox, @FranDepascuali, (francisco.depascuali@wolox.com.ar).

www.wolox.com.ar

--

--