Skip to content
Shreyas Patil's Blog

🙅‍♂️ Don't let ViewModel know about framework level dependencies

Cover image for 🙅‍♂️ Don't let ViewModel know about framework level dependencies

Hey Android developers 👋, Most of today’s developers are adopting MVVM architecture primarily for Android app development 👨‍💻. While adopting these things, anyone can do certain mistakes (which is absolutely fine). In this article, we are gonna see how to follow good practices with ViewModels in Android and how some decisions can make us helpless. Okay, let’s start 🏃‍♂️.


👁 Overview

Let’s understand the concept of ViewModel and what’s its purpose.

🤷‍♂️ What’s ViewModel?

In MVVM architecture, it suggests separating the data presentation logic (Views or UI) from the core business logic part of the application. ViewModel is such a component whose work is to execute core business and prepare required data for UI. So that ViewModel can be used by Activities/Fragments.

🤔 Why ViewModel shouldn’t hold framework references?

As we discussed earlier that ViewModel is lifecycle aware but View is not. Many developers also pass the Activity/View references to the ViewModel which is the most common case that developers make the ViewModel framework aware. But ViewModel should not be treated like this. If the View reference remains in ViewModel and it gets destroyed then ViewModel still hold that reference which may lead to the memory leaks 😮. Thus, the purpose for which ViewModel came into the picture is broken 💔.


🥽 Continue…

As you can see in the title, I’ve mentioned “Don’t let ViewModel know about FRAMEWORK LEVEL DEPENDENCIES”. So let’s discuss what’s exactly Framework Level Dependencies.

So as you might be aware of AndroidViewModel and have used already for some rare cases.

Note: AndroidViewModel is a ViewModel that has Application Context awareness.

Now when we are introducing Context in ViewModel, the last point 🔴 mentioned above is breaking. Why? Because Context is a part of Android’s framework. Then it decreases testability, modularity and maintainability of the codebase. No matter our application is small scale or large scale but having tests is always good because it acts as a safeguard as it gets expanding. That thing is breaking by using AndroidViewModel. Because it doesn’t let us write unit tests for that part of code (integration testing is possible only) ☹. After all, everyone knows how tricky it is to control memory leaks from the usage of Context if something goes wrong.

Why do we need AndroidViewModel? 🤔

For handling framework supported tasks in ViewModel sometimes we may need it. Example: File management, storage, WorkManager APIs, etc. But it’s completely up to us to implement business without using it.

Let’s see the example implementation to understand the problems…


👨‍💻 Example Implementation

Okay, so we have to develop a simple android app that stores a user session and performs simple login/logout actions. Now it’s obvious that we’ll need to use SharedPreferences. So let’s see how we can implement it.

❌ Example using AndroidViewModel way

I’m skipping UI code and will be only showing the ViewModel code. So let’s create UserViewModel. Need to inherit AndroidViewModel and create SharedPreferences for storing a user session. Also, let’s implement business for setting a user session.

Have you seen this code 😕? Let’s discuss key issues with this.

👎 Disadvantages of this approach

Isn’t it painful? 🤕 Let’s see how can we make it better 🦸‍♂️.


✔ Example using ViewModel way

So after coming from the previous approach, we need a solution that will be modular, easy to maintain, easy to plug whenever needed. So we can extract the logic of session management into a separate class 😃.

Let’s create a SessionManager. Create and implement operations/business logic of session management.

Now, it’s time to refactor and clean up UserViewModel 🧹.

😃 Can you see the difference? Now it has inherited core ViewModel. Too much code is now removed.

👍 Advantages of this approach

👎 Disadvantages of this approach

🙄 Obviously NOT 😅.


🧪 Testing

After using the best approach, UserViewModel is ready for testing. Let’s write code to test the code 😂.

image.png


Cool! That’s all. I hope this article helped to understand the pitfalls of AndroidViewModel.

If you need to have a look at both approaches, refer to this GitHub Repository. Here you’ll find both approaches. Also, if you want to take a look at just diff only ➕➖ then refer to this Pull Request which is a change from the first approach to the second.


Thank you! 😀

“Sharing is caring”


📚 References



Previous Post
Navigating Screens in Jetpack Compose (DevFest India 2021 - Mobile Track)
Next Post
Observing Live connectivity status in Jetpack Compose way!