Skip to main content

ViewModel with Jetpack Compose

 Compose uses remember API to store object in memory. Stored value is returned during recomposition . remember helps us retain data across recompostion , but when configuration changes happen all stored values are lost . One way to overcome this is to use rememberSaveable . rememberSaveable saves any value that can be saved in a Bundle , so it will survive configuration changes. But when we are using lot of data , for example a list we can cannot use a rememberSavble beacuse there is limit on amount of data that can be stored in Bundle . So we use ViweModel.

ViewModel provide the ui state and access to the business logic located in other layers of the app. It also survives configuration changes. ViewModel handles events coming from the UI or other layers of the app and updates the state holder based on the events.

We need to add the following dependency in our app level build.gradle to use ViewModel

implementation "androidx.lifecycle:lifecycle-viewmodel-compose:2.4.1"
First create a ViewModel class , then we can access the ViewModel from the composable using viewModel() function.
class DemoViewModel : ViewModel() { ......}

@Composable
fun DemoFun1(
    viewModel: DemoViewModel = viewModel()
    //cereates a new ViewModel
) {
   ....}
@Composable
fun DemoFun2(
    viewModel: DemoViewModel = viewModel()
    //returns same instance as in DemoFun1
) {
   ....}
We should  call viewModel() function from screen level composable , that is close to root composable called from activity or fragment . We should never pass down ViewModel instances to other Composables , pass only data and fuctions. viewModel() creates a new ViewModel or returns a existing ViewModel in the given scope. ViewModel is retained as long as the scope is alive.
This is a demo application which shows how to use ViewModel with Jetpack Compose . Screenshots of the app  




This is our MainActivity
package com.arun.androidtutsforu.democomposeviewmodel

import android.os.Bundle
import androidx.activity.ComponentActivity
import androidx.activity.compose.setContent
import androidx.compose.foundation.layout.*
import androidx.compose.material.*
import androidx.compose.runtime.Composable
import androidx.compose.runtime.getValue
import androidx.compose.runtime.mutableStateOf
import androidx.compose.runtime.saveable.rememberSaveable
import androidx.compose.runtime.setValue
import androidx.compose.ui.Alignment
import androidx.compose.ui.Modifier
import androidx.compose.ui.graphics.Color
import androidx.compose.ui.text.font.FontWeight
import androidx.compose.ui.tooling.preview.Preview
import androidx.compose.ui.unit.dp
import androidx.compose.ui.unit.sp
import androidx.lifecycle.ViewModel
import androidx.lifecycle.viewmodel.compose.viewModel
import com.arun.androidtutsforu.democomposeviewmodel.ui.theme.DemoComposeViewModelTheme

class MainActivity : ComponentActivity() {
    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContent {
            DemoComposeViewModelTheme {
                Surface(
                    modifier = Modifier.fillMaxSize(),
                    color = MaterialTheme.colors.background
                ) {
                    CalculateScreen()
                }
            }
        }
    }
}
@Composable
fun CalculateScreen(composeViewModel: ComposeViewModel = viewModel()){
    Column(
        modifier = Modifier.padding(16.dp),
        horizontalAlignment = Alignment.CenterHorizontally
    ) {
        Text(
            text = "JetPack Compose Demo",
            fontWeight = FontWeight.Bold,
            fontSize = 20.sp,
            color = Color.Blue
        )
        Spacer(modifier = Modifier.height(32.dp))
        TextField(
            value =composeViewModel.firstNum,
            onValueChange ={composeViewModel.changeFirstNum(it)} ,
            label = { Text(text = "Enter First Num")}
        )
        Spacer(modifier = Modifier.height(16.dp))
        TextField(
            value = composeViewModel.secondNum,
            onValueChange = {composeViewModel.changeSecondNum(it)},
            label = { Text(text = "Enter Second Num")}
        )
        Spacer(modifier = Modifier.height(16.dp))
        Text(
            text = "The sum is ${composeViewModel.sum}",
            fontWeight = FontWeight.Bold,
            fontSize = 20.sp
        )
        Button(onClick = { composeViewModel.valuculate() }) {
            Text(text = "Calculate")
        }
    }
}
This is our ViewModel , ComposeViewModel
package com.arun.androidtutsforu.democomposeviewmodel
import androidx.compose.runtime.mutableStateOf
import androidx.lifecycle.ViewModel
class ComposeViewModel : ViewModel() {
    private val _firstNum = mutableStateOf("")
    val firstNum :String
    get() = _firstNum.value
    private val _secondNum = mutableStateOf("")
    val secondNum :String
        get() = _secondNum.value
    private var _sum = mutableStateOf(0)
    val sum:Int
      get() =_sum.value

    fun changeFirstNum(number: String){
        _firstNum.value = number
    }
    fun changeSecondNum(number: String){
        _secondNum.value = number
    }
   fun valuculate() {
       val num1 = firstNum.toIntOrNull()?:0
       val num2 = secondNum.toIntOrNull()?:0
       _sum.value = num1 + num2
   }
}
Viewmodel provides all our UI related data.
    private val _firstNum = mutableStateOf("")
    val firstNum :String
     get() = _firstNum.value
    private val _secondNum = mutableStateOf("")
    val secondNum :String
     get() = _secondNum.value
    private var _sum = mutableStateOf(0)
    val sum:Int
      get() =_sum.value
Create a private _firstNum variable using mutableStateOf , and expose only firstNum as a String so it's not modified outside the ViewModel.
We have done the same with other variables. Only our ComposeViewModel can change the values of these varibles. We can only get the values of these varible from the MainActivity. We can not modify it from MainActivity.
...
onValueChange ={composeViewModel.changeFirstNum(it)} 
...
onValueChange = {composeViewModel.changeSecondNum(it)}
...
text = "The sum is ${composeViewModel.sum}"
...
Button(onClick = { composeViewModel.valuculate() })

This is how our MainActivity gets value from our ViewModel. 

You can download full source code of this demo app from github

Comments

Popular posts

Android CardView And SQLite example

SQLite Database Android provides several options to save persistent application data. SQlite database is ideal for saving repeating and structured data . Using Sqlite we can save structured data in a private database.  This is a simple application showing use of Sqlite database in android . This example shows how to perform Insert , select , update and delete operation in  SQlite database Screenshots of our sample application                                                                                                                      This is a Registration app. New user can register by clicking registration button . Registered  users are shown in cards below that button . we can update or delete registered users by clicking overflow menu in each card. 1.First we need to create database for our application .    a. Create a contract class .Contract class contains constants that defines names of Tables and columns of our database. Contract class allows us

Android List View using Custom Adapter and SQLite

following is a simple applicaton to create ListView using  Custom adapter.screenshot of the application  is like this . ListView is not used in android Anymore . We use  RecyclerView  and  CardView   in Android RecyclerView Demo is available on the following link http://androidtuts4u.blogspot.in/2017/04/android-recyclerview-example.html RecyclerView with Cardview Example is available on the following link http://androidtuts4u.blogspot.in/2017/09/android-card-view-example.html The ListView below the submit button is populated using Custom Adapter.Data is stored and retrieved using SQLite databsase. you can download the source code of this project from  google drive   https://drive.google.com/folderview?id=0BySLpWhqmbbdUXE5aTNhazludjQ&usp=sharing click on the above link ->sign into  your  google account ->add this to your google drive -> open it in google drive and download it. To create a simple application like this 1.  Create a class which extends  

Google Sign-in for Android App

This is demo application implementing Google Sign-in in android application. Screenshots of the application are when you click on the sign in button popup will be shown to select google account for sign in . After signing in , user can see his profile photo,name and Email id in the login screen . User can sign out using signout button . Disconnect button will provide user ability to disconnect their google account from the app. Before we start coding for our application , we need to turn on sign-in API for our app in google developer console. Go to this link to know  how to turn on google sign-in api for our app https://androidtuts4u.blogspot.com/2019/09/enabling-sign-in-api-in-google.html now we successfully configured Google Api console project .  we can start coding our application. I . first we need to configure build.gradle file we need to add two dependencies   1. google play service dependency for google sign-in.    2 . we are using third party library