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}"/>
