[Android] DataBinding 사용해보기

Android DataBinding Sample

Posted by JungHoon-Park on May 22, 2019

Sample : Android Databinding Sample


Android DataBinding

DataBinding 이란?

xml 에 데이터를 바인딩하여 불필요한 코드를 줄이는 방법으로, 보통 MVVM 패턴을 구현 할 때 사용된다.

사용방법

1. build.gradle 파일에 dataBinding 요소를 추가

1
2
3
dataBinding {
    enabled = true
}

2. xml 수정하기

레이아웃 파일의 루트태그에 ‘layout’ 로 추가

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
<layout xmlns:android="http://schemas.android.com/apk/res/android">

    <data>
        <variable
            name="user"
            type="com.example.databinding_sample.data.User" />
    </data>

<LinearLayout
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical">

<TextView
    android:id="@+id/firstname_textView"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="@{user.firstName, default=defaults}" />

<TextView
    android:id="@+id/lastname_textView"
    android:layout_width="wrap_content"
    android:layout_height="wrap_content"
    android:text="@{user.lastName, default=defaults}" />
    
</layout>

data 내에 있는 user 변수는 이 레이아웃 내에서 사용할 수 있는 속성에 대한 설명이다. 레이아웃 내에 있는 식은 “@{}” 구문을 사용하여 특성 속성에 기록된다. 여기서 TextView의 텍스트는 사용자의 firstName 속성으로 설정된다.

3. xml 에 binding 할 데이터 클래스 생성

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
public class User {
    private final String firstName;
    private final String lastName;

    public User(String firstName, String lastName) {
        this.firstName = firstName;
        this.lastName = lastName;
    }

    public String getFirstName() {
        return firstName;
    }

    public String getLastName() {
        return lastName;
    }
}

TextView의 android:text 특성에 사용되는 식 @{user.firstName}은 전자의 클래스에 있는 firstName 필드와 후자의 클래스에 있는 getFirstName() 메서드에 액세스한다. 또는 firstName() 메서드가 존재할 경우에는 이 메서드로 해석하기도 한다.

4. 데이터 바인딩 하기

1
2
3
4
5
6
7
8
9
public class MainActivity extends AppCompatActivity{
    ActivityMainBinding binding;
    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
        User user = new User("JungHoon", "Park");
        binding.setUser(user);
}

위의 레이아웃 파일은 main_activity.xml이고, 따라서 생성된 클래스는 MainActivityBinding이었습니다. 이 클래스는 레이아웃 속성(예: user 변수)에서 레이아웃의 View까지 모든 바인딩을 유지하고 바인딩 식에 대해 값을 할당하는 방법을 알고 있습니다. 바인딩을 가장 쉽게 생성하는 방법은 다음과 같이 확장하는 동안 바인딩을 생성하는 것입니다.

또는

1
MainActivityBinding binding = MainActivityBinding.inflate(getLayoutInflater());

를 통해 뷰를 불러올 수 있다.

또는 ListView 어댑터나 RecyclerView 어댑터 내에서 데이터 바인딩 항목을 사용할때

1
ListItemBinding binding = ListItemBinding.inflate(layoutInflater, viewGroup, false);

or

1
ListItemBinding binding = DataBindingUtil.inflate(layoutInflater, R.layout.list_item, viewGroup, false);

을 통해 뷰를 불러올 수 있다.

< 참고 >

이벤트 처리하기

버튼 클릭시 아래의 함수를 실행시키고 싶다면

1
2
3
public void OnButtonClick(View view){
    Toast.makeText(this, "Button Click", Toast.LENGTH_SHORT).show();
}

xml 에서 함수를 onClick 에 지정해준다.

1
2
3
4
5
6
<Button
    android:id="@+id/btnSample"
    android:layout_width="match_parent"
    android:layout_height="wrap_content"
    android:onClick="@{activity::onButtonClick}"
    android:text="button"/>

xml 에서 activity 변수 지정

1
2
3
4
5
<data>
    <variable
        name="activity"
        type="com.example.databinding_sample.MainActivity" />
</data>

MainActivity 에서 사용하기

1
2
3
4
5
6
7
DatabindingActivityBinding binding;

@Override
protected void onCreate(@Nullable Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    binding = DataBindingUtil.setContentView(this, R.layout.activity_main);
    binding.setActivity(this);

setAcrtivity(this)는 우리가 위에서 지정한 activity라는 변수를 이 Class로 지정하겠다는 의미이다.

RecyclerView Adapter 에서 사용하기

1
2
3
4
@Override
public void onBindViewHolder(@NonNull BindingViewHolder holder, int position) {
    holder.binding.setUser(mList.get(position));
}
1
2
3
4
5
6
7
8
public class BindingViewHolder extends RecyclerView.ViewHolder {
    public RecyclerItemLayoutBinding binding;

    public BindingViewHolder(@NonNull View itemView) {
        super(itemView);
        binding = DataBindingUtil.bind(itemView);
    }
}

RecyclerView Adapter 에서 Glide 사용할 때

RecyclerView의 item xml 의 ImageView에서 첫번째 속성을 추가한다. 사용자가 직접 photo 라는 속성을 만든것이다.

1
2
3
4
5
6
7
8
9
<ImageView
    app:photo="{@photo.thumbnailUrl}"
    android:id="@+id/thumbnail_imageView"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:layout_gravity="center"
    android:layout_margin="7dp"
    android:scaleType="centerCrop"
    app:srcCompat="@drawable/ic_launcher_background" />

이후에 @BindingAdapter Class 를 생성해준다.

1
2
3
4
5
6
7
8
9
10
public class PhotoBindingAdapter{
    @BindingAdapter("photo")//xml 에서 사용자가 지정한 속성이름
    public static void setImage(ImageView view, String url){
        Glide.with(view)
                .load(url)
                .centerCrop()
                .placeholder(R.mipmap.ic_launcher)
                .into(view)
    }
}

평소에 onBindViewHolder 함수내에서 Glide 를 처리했었다면 BindingAdapter를 통해 RecyclerView Adapter의 코드 수를 줄이고, 분리하여 관리할 수 있다.

Layout 세부정보 처리하기

1
2
3
<data>
    <import type="android.view.View"/>
</data>
1
2
3
4
5
<TextView
   android:text="@{user.lastName}"
   android:layout_width="wrap_content"
   android:layout_height="wrap_content"
   android:visibility="@{user.isAdult ? View.VISIBLE : View.GONE}"/>