Skip to content

Latest commit

 

History

History
299 lines (217 loc) · 9.84 KB

README.md

File metadata and controls

299 lines (217 loc) · 9.84 KB

Android Data Binding Basic Sample

This sample showcases the following features of the Data Binding library:

  • Layout variables and expressions
  • Observability through Observable Fields, LiveData and Observable classes
  • Binding Adapters, Binding Methods and Binding Converters
  • Seamless integration with ViewModels

It shows common bad practices and their solutions in two different screens.

Features

Screenshot

Layout variables and expressions

With Data Binding you can write less boilerplate and repetitive code. It moves UI operations out of the activities and fragments to the XML layout.

For example, instead of setting text on a TextView in an activity:

TextView textView = findViewById(R.id.name);
textView.setText(user.name);

You assign the attribute to a variable, in the XML layout:

<TextView
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="@{user.name}" />

See ObservableFieldActivity.kt, ObservableFieldProfile.kt and observable_field_profile.xml for a simple example.

Observability

In order to update the UI automatically when the data changes, Data Binding lets you bind attributes with observable objects. You can choose between three mechanisms to achieve this: Observable fields, LiveData and Observable classes.

Observable fields

Types like ObservableBoolean, ObservableInt and the generic ObservableField replace the corresponding primitives to make them observable. Setting a new value on one of the Observable fields will update the layout automatically.

class ProfileObservableFieldsViewModel : ViewModel() {

    val likes = ObservableInt(0)

    fun onLike() {
        likes.increment()  // Equivalent to set(likes.get() + 1)
    }
}

In this example, when onLike is called, the number of likes is incremented and the UI is updated. There is no need to notify that the property changed.

LiveData

LiveData is an observable from Android Architecture Components that is lifecycle-aware.

The advantages over Observable Fields are that LiveData supports Transformations and it's compatible with other components and libraries, like Room and WorkManager.

class ProfileLiveDataViewModel : ViewModel() {
    private val _likes =  MutableLiveData(0)
    val likes: LiveData<Int> = _likes // Expose an immutable LiveData

    fun onLike() {
        _likes.value = (_likes.value ?: 0) + 1
    }
}

It requires an extra step done on the binding:

binding.lifecycleOwner = this  // use viewLifecycleOwner when assigning a fragment

Observable classes

For maximum flexibility and control, you can implement a fully observable class and decide when to update certain properties. This lets you create dependencies between properties and it's useful to dispatch partial UI updates, for example avoiding potential glitches (UI elements updating almost at the same time).

class ProfileObservableViewModel : ObservableViewModel() {
    val likes = ObservableInt(0)

    fun onLike() {
        likes.increment()
        notifyPropertyChanged(BR.popularity)
    }

    @Bindable
    fun getPopularity(): Popularity {
        return likes.get().let {
            when {
                it > 9 -> Popularity.STAR
                it > 4 -> Popularity.POPULAR
                else -> Popularity.NORMAL
            }
        }
    }
}

In this example, when onLike is called, the number of likes is incremented and the popularity property is notified of a potential change (popularity depends on likes). getPopularity is called by the library, returning a possible new value.

See ProfileObservableFieldsViewModel.kt for a complete example.

Binding adapters

Binding adapters let you customize or create layout attributes. For example, you can create an app:progressTint attribute for progress bars where you change the color of the progress indicator depending on an external value.

    @BindingAdapter("app:progressTint")
    @JvmStatic fun tintPopularity(view: ProgressBar, popularity: Popularity) {

        val color = getAssociatedColor(popularity, view.context)
        view.progressTintList = ColorStateList.valueOf(color)
    }

The binding is created in the XML layout with:

    <ProgressBar
        app:progressTint="@{viewmodel.popularity}" />

Using binding adapters lets you move UI calls from the activity to static methods, improving encapsulation.

You can also use multiple attributes in a Binding Adapter, see viewmodel_profile.xml for a complete example.

Binding methods and binding converters

Binding methods and binding converters let you reduce code if your binding adapters are very simple. You can read about them in the official guide.

For example, if an attribute's value needs to be passed to a method in the class:

@BindingAdapter("app:srcCompat")
@JvmStatic fun srcCompat(view: ImageView, @DrawableRes drawableId: Int) {
    view.setImageResource(drawable)
}

You can replace this with a Binding Method which can be added to any class in the project:

@BindingMethods(
        BindingMethod(type = ImageView::class,
                attribute = "app:srcCompat",
                method = "setImageDrawable"))

Binding Converters (use with caution)

In this sample we show a View depending on whether a number is zero. There are many options to do this. We're showing two, one in the observable_field_profile.xml and, the recommended way, in viewmodel_profile.xml.

The goal is to bind the view's visibility to the number of likes, but this won't work:

android:visibility="@{viewmodel.likes}"  <!-- Doesn't work as expected -->

The number of likes is an integer and the visibility attribute takes an integer (VISIBLE, GONE and INVISIBLE are 0, 4 and 8 respectively), so doing this would build, but the result would not be the expected.

A possible solution is:

android:visibility="@{viewmodel.likes == 0 ? View.GONE : View.VISIBLE}"/

But it adds a relatively complex expression to the layout.

Instead, you can create and import a utils class:

<data>
        <import type="com.example.android.databinding.basicsample.util.ConverterUtil" />
        ...
</data>

and use it from the View like so:

android:visibility="@{ConverterUtil.isZero(viewmodel.likes)}"  <!-- don't do this either -->

isZero returns a boolean and visibility takes an integer so in order to convert from boolean we can also define a BindingConversion:

    @BindingConversion
    @JvmStatic fun booleanToVisibility(isVisible: Boolean): Int {  // Risky! applies everywhere
        return if (isVisible) View.VISIBLE else View.GONE
    }

This conversion is unsafe because this binding conversion is not restricted to our case: it will convert all booleans to visibility integers when the attribute takes an integer.

Solution: As with every BindingConversion and BindingMethod, you can replace it with a Binding Adapter, which normally is much simpler:

    @BindingAdapter("app:hideIfZero")  // Recommended solution
    @JvmStatic fun hideIfZero(view: View, number: Int) {
        view.visibility = if (number == 0) View.GONE else View.VISIBLE
    }

and as shown in in viewmodel_profile.xml:

app:hideIfZero="@{viewmodel.likes}"

This defines a new custom attribute hideIfZero that can't be used accidentally.

As a rule of thumb it's preferable to create your our custom attributes using Data Binding adapters instead of adding logic to your binding expressions.

Sample app

This app shows a user's profile using two different screens to showcase different Data Binding features:

  • Main activity: Shows how a Data Binding layout lets you access Views without findViewById.

  • Observable field activity: In this screen the user can give "likes" to the profile and the UI reacts automatically to changes. However, the activity holds the logic that receives the user click and the actual profile data, which is not testable. Also, likes are reset when the user rotates the device and the layout contains documented common bad practices.

  • ViewModel activity: Using a ViewModel from the Architecture Components fixes the rotation problem and moves logic out of the activity. Also, the use of binding adapters changes the responsibility of the activity which is no longer the "view" and solely responsible for dealing with the lifecycle. Two ViewModels are suggested in ProfileObservableViewModel.kt: one based on observable fields and another implementing the observable interface.

License

Copyright 2018 The Android Open Source Project, Inc.

Licensed to the Apache Software Foundation (ASF) under one or more contributor license agreements. See the NOTICE file distributed with this work for additional information regarding copyright ownership. The ASF licenses this file to you under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at

http://www.apache.org/licenses/LICENSE-2.0

Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License.