Architecture Components - I'm not a purist but ...

  • Post by Hannes Dorfmann
  • Jun 25, 2017

At I/O 2017 Google surprised us with a new initiative: Architecture Components. I really appreciate this initiative. In this blog post I would like to share my thoughts about ViewModel and some pitfalls you might stumble upon when using ViewModel and LiveData by taking a look at the official Google samples.

The State Problem

It is not a secret that I’m a big fan of Model-View-Intent (MVI). This is just a matter of personal preference one could argue and I’m not disagreeing. I like MVI because state management, unidirectional data flow and immutability is a core part of the pattern. On the other hand MVVM and MVP don’t talk about state management although it can be (and should) applied in these patterns too. For example Laimonas Turauskas from Instacart wrote a blog post how they do state management in their MVP based app. Fortunately, Google’s MVVM samples show how to do that with ViewModel and LiveData too. Take a look at this example which is an app that let’s you browse Github. It’s a very simplified Github client that uses Githubs REST API to display some data about git repositories and users.

public enum Status {
    SUCCESS,
    ERROR,
    LOADING
}

public class Resource<T> {

    public final Status status;
    public final String message;
    public final T data;

    public Resource(@NonNull Status status, @Nullable T data, @Nullable String message) {
        this.status = status;
        this.data = data;
        this.message = message;
    }

    public static <T> Resource<T> success(@Nullable T data) {
        return new Resource<>(SUCCESS, data, null);
    }

    public static <T> Resource<T> error(String msg, @Nullable T data) {
        return new Resource<>(ERROR, data, msg);
    }

    public static <T> Resource<T> loading(@Nullable T data) {
        return new Resource<>(LOADING, data, null);
    }
}

A Resource is used for example to wrap a http response i.e. Resource<ApiResponse> (ApiResponse holds the json parsed data returned from github.com). Btw. if you use Kotlin, you should take a look at sealed classes.

I’m super happy to see such state management related code in official Google samples and encourage developers to build your apps with proper state management in mind. However, in the same Google example we can scroll through a infinite list of search results (uses pagination). Some interesting observation:

public class SearchViewModel extends ViewModel {
  ...
  public LiveData<Resource<List<Repo>>> getResults() { ... }
  public LiveData<LoadMoreState> getLoadMoreStatus() { ... }
}

getResults() is used to represent the search results whereas getLoadingMoreStatus() is used for consecutive pagingation.

Why not just one LiveData<State> getState()? Wouldn’t it make state management especially in a multi threaded environment more predictable and deterministic?

I’m not a purist but it leaves a stale after-taste …

The mantra problem

LiveData seems to be a very simplified version of RxJava’s Observable. LiveData is lifecycle aware so that we as developer don’t have to unsubscribe explicitly as we have to do with RxJava’s Observable. Both implement the Observer pattern. LiveData doesn’t provide all this fancy functional programming operators as RxJava does although architecture components provides some functional concepts via Transformations helper class such as Transformations.map() and Transformations.switchMap().

Quite often I hear developers saying how much they like functional reactive programming and pure functions without side effects but then we write code like mutableLiveData.setValue(...).

I’m not a purist but it leaves a stale after-taste …

The state restoration race

Rebecca Franks who seems to be working on a sample app (with LiveData and Architecture Components) asked an interesting question in a public slack chat about a Google TODO app example, more specific about the screen where the user can create a new TODO Task:

Is there a reason why the Fragment passes information back to the ViewModel when calling saveTask(title, description). The ViewModel has the title and description already, so does that need to be passed into the saveTask(title, description) method explicitly?

The code she is referring to can be found here. At first glance the answer seems to be obvious: “No, there is no reason to pass that information back from View to the ViewModel because the ViewModel already holds this information”. Is that always true?

Well, ViewModel is “only” kept across screen orientation changes. On the other hand UI widgets like EditText store and restore their own state. Do you see the conflict potential? Just for fun, let’s also add android data binding into this equation: the problem is that data binding will override the value of the restored UI widget (Update: this problem might be addressed in the future).

I’m not really answering Rebecca’s question because the previous answer that has seemed to be so obvious is actually not that obvious anymore. So which one is the source of truth? Should the ViewModel’s state be persisted too (see this post by Danny Preussler and official docs)? Which one wins the state restoration race?

I’m not a purist but it leaves a stale after-taste …

The “SingleLiveEvent” problem

Fabio Collini raised another very interesting question about the Google’s sample github client app that is powered by the new Architecture Components:

Sometimes the ViewModel needs to invoke an action using an Activity (for example to show a snack bar or to navigate to a new activity). Using MVP it’s easy to implement it because the presenter has a reference to the Activity. The ViewModel doesn’t contain a reference to the Activity, what’s the best way to implement this feature?

What he means is an “action” that takes place only one time like displaying a SnackBar or a Toast. If you would model this problem in MVVM you would have a LiveData<String> getSnackBarErrorMessage() in your ViewModel and the View subscribes to it. If the emitted String (error message) is null, don’t display a SnackBar, otherwise display it. The problem he is facing is that this flag must be cleared somehow otherwise the SnackBar is shown again after a screen orientation change because LiveData is emitting the latest (cached) value when the view (re)subscribes to it.

In the Google samples they have added a class called SingleLiveEvent extends LiveData. The idea is that once an event has been dispatched it sets the internal value to null doesn’t dispatch the value (Update: I was wrong, it doesn’t set the value internally to null, thanks Jose Alcérreca for the hint). This prevents the SnackBar to appear a second time after screen orientation change.

Well, SingleLiveEvent seems to solve this problem, but isn’t it just hiding the real underlying problem (hint: state management)?

Additionally, if we introduce something like a SingeLiveEvent because of showing something only once (i.e SnackBar) wouldn’t we then leak an implementation detail from the View Layer down to underlying layers like ViewModel or business logic? What if one day we would like to change the way such a message is displayed from SnackBar to TextView, which layers would we have to change? Only the View (UI) layer? Btw. I have commented on that issue too and suggested an alternative solution

I’m not a purist but it leaves a stale after-taste …

Conclusion: I’m a purist

As you see, even in such small sample apps like a Github client or a TODO app we face some tricky edge cases. From my point of view this is not related to any particular architectural pattern. It doesn’t matter if you prefer MVVM with the new Architecture Components or MVP or MVI or whatever else. This kind of problems are just the symptoms of a much deeper issue we face but maybe haven’t thought about it yet: proper state management. I guess I am a purist when it comes to state management.

Google’s Architecture Components are awesome and this blog post is by no means a critique on that great library nor on the examples Google (and many other external contributors) provided us. I just think that with the introduction of Architecture Components we (including Android Developers from around the world and Google Developers) have a very unique chance to create examples that can be considered as a reference implementation. These are great starting point for any Junior Android developer. By agreeing on the importance of state management, regardless if you are a purist or not, such a reference implementation could help a lot developers avoiding pain points in the future from lessons we have already learned the hard way in the past. Let’s reflect that in official Google examples. They are open source, let’s contribute!