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