In this tutorial, we’ll be discussing and implementing the Android MVVM Architectural Pattern in our Android Application. We’ve previously discussed the Android MVP Pattern.
Why do we need these patterns? Adding everything in a Single Activity or Fragment would lead to problems in testing and refactoring the code. Hence, the use of separation of code and clean architecture is recommended.
MVVM stands for Model, View, ViewModel.
The following flow illustrates the core MVVM Pattern. How does this differ from MVP?
There are two ways to implement MVVM in Android:
In this tutorial, we’ll be using Data Binding only. Data Binding Library was introduced by Google in order to bind data directly in the xml layout. For more info on Data Binding, refer this tutorial. We’ll be creating a simple Login Page Example Application that asks for user inputs. We’ll see how the ViewModel notifies the View when to show a Toast Message without keeping a reference of the View.
How is it possible to notify some class without having a reference of it? It can be done in three different ways:
Two-way Data Binding is a technique of binding your objects to your XML layouts such that the Object and the layout can both send data to each other. In our case, the ViewModel can send data to the layout and also observe changes. For this, we need a BindingAdapter
and custom attribute defined in the XML. The Binding Adapter would listen to changes in the attribute property. We’ll learn more about Two-way Data Binding through our example below.
Add the following code to your app’s build.gradle file:
android {
dataBinding {
enabled = true
}
}
This enables Data Binding in your Application.
Add the following dependencies in your build.gradle
file :
implementation 'android.arch.lifecycle:extensions:1.1.0'
The Model would hold the user’s email and password. The following User.java class does it:
package com.journaldev.androidmvvmbasics.model;
public class User {
private String email;
private String password;
public User(String email, String password) {
this.email = email;
this.password = password;
}
public void setEmail(String email) {
this.email = email;
}
public String getEmail() {
return email;
}
public void setPassword(String password) {
this.password = password;
}
public String getPassword() {
return password;
}
}
Two-way Data Binding allows us to bind objects in the XML layouts such that the object can send data to the layout, and vice versa. The Syntax for two way data binding is @={variable}
The code for the activity_main.xml is given below:
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="https://schemas.android.com/apk/res/android"
xmlns:bind="https://schemas.android.com/tools">
<data>
<variable
name="viewModel"
type="com.journaldev.androidmvvmbasics.viewmodels.LoginViewModel" />
</data>
<ScrollView
android:layout_width="match_parent"
android:layout_height="match_parent">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_gravity="center"
android:layout_margin="8dp"
android:orientation="vertical">
<EditText
android:id="@+id/inEmail"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="Email"
android:inputType="textEmailAddress"
android:padding="8dp"
android:text="@={viewModel.userEmail}" />
<EditText
android:id="@+id/inPassword"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:hint="Password"
android:inputType="textPassword"
android:padding="8dp"
android:text="@={viewModel.userPassword}" />
<Button
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="8dp"
android:onClick="@{()-> viewModel.onLoginClicked()}"
android:text="LOGIN"
bind:toastMessage="@{viewModel.toastMessage}" />
</LinearLayout>
</ScrollView>
</layout>
Data Binding requires us to set the layout tag at the top. Here our ViewModel binds the data to the View. ()-> viewModel.onLoginClicked()
invokes the Button click listener lambda defined in our ViewModel. The EditText updates the values in the Model (via View Model). bind:toastMessage="@{viewModel.toastMessage}"
is a custom attribute we’ve created for two-way data binding. Based on changes in the toastMessage in the ViewModel the BindingAdapter would get triggered in the View.
The code for the LoginViewModel.java is given below:
package com.journaldev.androidmvvmbasics.viewmodels;
import android.databinding.BaseObservable;
import android.databinding.Bindable;
import android.text.TextUtils;
import android.util.Patterns;
import com.android.databinding.library.baseAdapters.BR;
import com.journaldev.androidmvvmbasics.model.User;
public class LoginViewModel extends BaseObservable {
private User user;
private String successMessage = "Login was successful";
private String errorMessage = "Email or Password not valid";
@Bindable
private String toastMessage = null;
public String getToastMessage() {
return toastMessage;
}
private void setToastMessage(String toastMessage) {
this.toastMessage = toastMessage;
notifyPropertyChanged(BR.toastMessage);
}
public void setUserEmail(String email) {
user.setEmail(email);
notifyPropertyChanged(BR.userEmail);
}
@Bindable
public String getUserEmail() {
return user.getEmail();
}
@Bindable
public String getUserPassword() {
return user.getPassword();
}
public void setUserPassword(String password) {
user.setPassword(password);
notifyPropertyChanged(BR.userPassword);
}
public LoginViewModel() {
user = new User("","");
}
public void onLoginClicked() {
if (isInputDataValid())
setToastMessage(successMessage);
else
setToastMessage(errorMessage);
}
public boolean isInputDataValid() {
return !TextUtils.isEmpty(getUserEmail()) && Patterns.EMAIL_ADDRESS.matcher(getUserEmail()).matches() && getUserPassword().length() > 5;
}
}
The methods were called in the layout are implemented in the above code with the same signature. If the XML counterpart of the method doesn’t exist, we need to change the attribute to app:
. The above class can also extend ViewModel. But we need BaseObservable since it converts the data into streams and notifies when the toastMessage
property is changed. We need to define the getter and setter for the toastMessage custom attribute defined in the XML. Inside the setter, we notify the observer (which will be the View in our application) that the data has changed. The View(Our activity) can define the appropriate action.
BR class is auto-generated from data binding when you rebuild the project
The code for the MainActivity.java
class is given below:
package com.journaldev.androidmvvmbasics.views;
import android.databinding.BindingAdapter;
import android.databinding.DataBindingUtil;
import android.support.v7.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Toast;
import com.journaldev.androidmvvmbasics.R;
import com.journaldev.androidmvvmbasics.databinding.ActivityMainBinding;
import com.journaldev.androidmvvmbasics.viewmodels.LoginViewModel;
public class MainActivity extends AppCompatActivity {
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityMainBinding activityMainBinding = DataBindingUtil.setContentView(this, R.layout.activity_main);
activityMainBinding.setViewModel(new LoginViewModel());
activityMainBinding.executePendingBindings();
}
@BindingAdapter({"toastMessage"})
public static void runMe(View view, String message) {
if (message != null)
Toast.makeText(view.getContext(), message, Toast.LENGTH_SHORT).show();
}
}
Thanks to DataBinding, the ActivityMainBinding
class is auto-generated from the layout. The @BindingAdapter
method gets triggered whenever toastMessage attribute defined on the Button is changed. It must use the same attribute as defined in the XML and in the ViewModel. So in the above application, the ViewModel updates the Model by listening to the changes in the View. Also, the Model can update the view via the ViewModel using the notifyPropertyChanged
The output of the above application in action is given below: This brings an end to this tutorial on Android MVVM Using DataBinding. You can download the project from the link given below.
Thanks for learning with the DigitalOcean Community. Check out our offerings for compute, storage, networking, and managed databases.
While we believe that this content benefits our community, we have not yet thoroughly reviewed it. If you have any suggestions for improvements, please let us know by clicking the “report an issue“ button at the bottom of the tutorial.
Great tutorial. But i’m facing a little problem. When I rotate the phone, the data remains there… when I “rerotate” de phone, the data remains there, but the cursor comes back to initial point… Do you have some thoughts about it?
- Victor
This is not pure MVVM pattern, It is mixture of MVVM and MVP. In MVVM ViewModel doesn’t know about View, but here ViewModel has reference of View(Activity) through LoginResultCallback .
- Shubham
Hello, Is there a way i can subscribe to you, so that I would be notified whenever a new article related to Android/Java is posted. Your articles are great. Thanks Aakash
- Aakash Jain
hi,i do this step by step but it showed me a “duplicate class found in activity_main.xml” error! please help
- امیر
Hi, If I don’t use Databinding/Livedata, then will my app fall in MVVM pattern?
- Tushar
Found data binding errors. ****/ data binding error ****msg:Cannot find the setter for attribute ‘bind:toastMessage’ with parameter type java.lang.String on android.widget.Button
- Luciano Moura
Brother can you give me same example of mvvb and mvp to understand the patterns better ?
- Md. Rafsan Biswas
Hi Anupam, From past few days i have been searching for mvvm in android, then i have found yours code simple and understandable, but as the comments section says this is not pure mvvm, So, please can u provide the pure mvvm understandable example for simple login page. Please help ASAP, I am in extreme need of it. Thank you in advance.
- Rajesh
ViewModel: It acts as a link between the Model and the ViewModel. I think it should be “It acts as a link between the Model and the View”
- Pratik Butani
Very neat and simple to understand this tutorial. I though MVVM is very hard to learn, but with your support and tutorial it makes me learner. Thanks lot man!! But i am obstruction of below error… Please help to find this error Found data binding errors. ****/ data binding error ****msg:cannot find method afterEmailTextChanged(android.text.Editable) in class com.eyeraise.kot.model.login.LoginModel file:/Users/bala/Documents/Muthu/Source/eyeraise/kot/app/src/main/res/layout/activity_login.xml loc:27:64 - 27:104 ****\ data binding error ****
- Muthukumar Subramaniam