참고자료
developer.android.com/topic/libraries/architecture
androidwave.com/android-data-binding-recyclerview/
예제설계
- 앱 아키텍처는 MVVM을 따릅니다.
- 비동기적으로 수행되어 변경되는 데이타는 LiveData를 이용합니다.
- DataBinding을 이용해 소스코드를 간결하게 합니다.
- RecyclerView를 이용하여 리스트를 처리합니다.
위 설계 목표를 가지고 아래와 같은 예제를 직접 만들어보았습니다.
다수개의 항목들이 비동기적으로 로딩되고 로딩되는 항목들의 상태정보를 리스트로 보여주도록 해보겠습니다.
* 간단하므로 굳이 RecyclerView나 LiveData 등이 필요하지 않지만, 스터디 목적으로 끼워 맞춰보았습니다.
MainActivity.java
public class MainActivity extends AppCompatActivity {
private MainViewModel mainViewModel;
private ItemAdapter itemAdapter;
public ObservableBoolean isReadyToStart = new ObservableBoolean();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
ActivityMainBinding activityMainBinding = DataBindingUtil.setContentView(this, R.layout.activity_main);
activityMainBinding.setActivity(this);
RecyclerView recyclerView = activityMainBinding.viewItems;
recyclerView.setLayoutManager(new LinearLayoutManager(this));
recyclerView.setHasFixedSize(true);
itemAdapter = new ItemAdapter();
recyclerView.setAdapter(itemAdapter);
mainViewModel = new ViewModelProvider(this).get(MainViewModel.class);
mainViewModel.getMutableLiveDataItemsList().observe(this, (Observer<ArrayList<Item>>) items -> {
itemAdapter.setItemsList((ArrayList<Item>) items);
isReadyToStart.set(mainViewModel.isReadyToStart());
});
}
public void onStartButtonClicked(View view) {
mainViewModel.loadItems();
}
}
- MVVM의 View에 속하는 구현입니다. ViewModel인 mainViewModel만을 참조하므로 모듈화가 되어 있습니다.
- DataBinding을 이용하여 activity_main.xml layout과 연결되어 있기 때문에 소스가 간결합니다. 대신, activity_main.xml과 함께 살펴봐야 상세한 구현 내용을 파악할 수 있습니다.
- LiveData로 제공되는 항목들의 데이타 변화를 감지하여 그에 따른 동작을 수행합니다. 변화가 감지되면 RecyclerView의 항목을 재설정하고 시작버튼의 활성/비활성화를 설정합니다.
- 시작버튼이 눌리면 항목들의 로딩을 시작합니다.
activity_main.xml
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools"
>
<data>
<variable
name="activity"
type="com.jjjh.aacsample.MainActivity"/>
</data>
<androidx.constraintlayout.widget.ConstraintLayout
android:layout_width="match_parent"
android:layout_height="match_parent"
tools:context=".MainActivity"
tools:showIn="@layout/activity_main">
<Button
android:id="@+id/buttonStart"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginTop="16dp"
android:layout_marginStart="16dp"
android:layout_marginEnd="16dp"
android:enabled="@{activity.isReadyToStart}"
android:onClick="@{activity::onStartButtonClicked}"
android:padding="8dp"
android:text="Start"
tools:ignore="MissingConstraints" />
<androidx.recyclerview.widget.RecyclerView
android:id="@+id/viewItems"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginStart="8dp"
android:layout_marginEnd="8dp"
android:layout_marginBottom="8dp"
android:padding="4dp"
app:layout_constraintBottom_toBottomOf="parent"
app:layout_constraintEnd_toEndOf="parent"
app:layout_constraintHorizontal_bias="1.0"
app:layout_constraintStart_toStartOf="parent"
app:layout_constraintTop_toBottomOf="@+id/buttonStart"
app:layout_constraintVertical_bias="0.0" />
</androidx.constraintlayout.widget.ConstraintLayout>
</layout>
- <data>...</data>를 이용하여 MainActivity와의 DataBinding을 표시합니다.
- 버튼의 android:enabled를 MainActivity의 isReadyToStart와 바인딩하여 isReadyToStart가 set되는 값에 따라 버튼의 활성/비활성 상태가 변경됩니다.
- 버튼의 android:onClick을 MainActivity의 onStartButtonClicked로 바인딩하여서 버튼이 클릭될 때마다 MainActivity의 onStartButtonClicked가 수행됩니다.
MainViewModel.java
public class MainViewModel extends AndroidViewModel {
private final ItemRepository itemRepository;
public MainViewModel(@NonNull Application application) {
super(application);
itemRepository = new ItemRepository();
}
public LiveData<ArrayList<Item>> getMutableLiveDataItemsList() {
return itemRepository.getMutableLiveDataItemsList();
}
public void loadItems() {
itemRepository.loadItems();
}
public boolean isReadyToStart() {
return itemRepository.isReadyToStart();
}
}
- MVVM의 ViewModel에 해당합니다. ItemRepository만을 참조하도록 모듈화되어 있습니다.
ItemRepository.java
public class ItemRepository implements Item.OnStatusChangeListener {
private final ArrayList<Item> items = new ArrayList<>();
private final MutableLiveData<ArrayList<Item>> mutableLiveDataItemsList = new MutableLiveData<>();
public ItemRepository() {
items.clear();
items.add(new Item(this, "Item 1", "READY", 2));
items.add(new Item(this, "Item 2", "READY", 5));
items.add(new Item(this, "Item 3", "READY", 8));
notifyItemsChanged();
}
@Override
public void onStatusChanged(Item item, String status) {
notifyItemsChanged();
}
public MutableLiveData<ArrayList<Item>> getMutableLiveDataItemsList() {
return mutableLiveDataItemsList;
}
public void loadItems() {
for( Item item : items ) {
item.load();
}
}
public boolean isReadyToStart() {
for( Item item : items ) {
if ( "DOING".equals(item.getStatus()) ) {
return false;
}
}
return true;
}
private void notifyItemsChanged() {
if ( Looper.myLooper() == Looper.getMainLooper() ) {
mutableLiveDataItemsList.setValue(items);
} else {
mutableLiveDataItemsList.postValue(items);
}
}
}
- 초기 항목들로 3개의 항목을 생성했고 각각 2초, 5초, 8초후 로딩작업이 끝나는 것으로 설정했고 로딩작업이 끝나면 onStatusChanged가 콜백되도록 구성했습니다.
- 많은 예제에서 이 부분은 Retrofit이나 Room의 콜백을 사용합니다만 예제를 간단하게 하기 위해 실제 로딩없이 지연시간만 설정하였습니다.
- 데이타의 중복 없이 예제 앱 전체에서 유일한 데이타가 ArrayList<Item> items 입니다.
- MVVM의 Model에 해당하는 부분입니다. 유일한 데이타를 LiveData로 감쌌고 이를 View에서 observe하여 데이타의 변화에 대한 처리를 수행하기 때문에 View에 대한 고민없이 데이타만을 신경쓰면 됩니다.
- 상태는 "READY", "DOING", "DONE"으로 설정했습니다.
Item.java
public class Item {
private String title;
private String status;
private final OnStatusChangeListener listener;
private final int fakeLoadingTimeSeconds;
public interface OnStatusChangeListener {
void onStatusChanged(Item item, String status);
}
public Item(OnStatusChangeListener listener, String title, String status, int fakeLoadingTimeSeconds) {
this.listener = listener;
this.title = title;
this.status = status;
this.fakeLoadingTimeSeconds = fakeLoadingTimeSeconds;
}
public String getTitle() {
return title;
}
public void setTitle(String title) {
this.title = title;
}
public String getStatus() {
return status;
}
public void setStatus(String status) {
this.status = status;
listener.onStatusChanged(this, status);
}
@BindingAdapter({ "statusIcon" })
public static void loadStatusIcon(ImageView imageView, String status) {
if ("DONE".equals(status)) {
imageView.setImageResource(R.drawable.ic_baseline_done_24);
} else if ("DOING".equals(status)) {
imageView.setImageResource(R.drawable.ic_baseline_loop_24);
} else {
imageView.setImageResource(R.drawable.ic_baseline_block_24);
}
}
void load() {
setStatus("DOING");
new Thread(() -> {
try {
Thread.sleep(fakeLoadingTimeSeconds * 1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
setStatus("DONE");
}).start();
}
}
- 상태가 바뀌면 onStatusChanged 콜백을 호출해줍니다. 콜백이 되면 ItemRepository에서는 데이타가 변경되었다는 것을 LiveData set/post 하여 observer에게 알려줍니다.
- RecyclerView를 구성하는 각 항목에 대한 layout인 list_item.xml 과 DataBinding되어 있기 때문에 Item의 내부 값이 바뀔 때 마다 UI가 자동으로 변경됩니다.
list_item.xml
<?xml version="1.0" encoding="utf-8"?>
<layout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:bind="http://schemas.android.com/apk/res-auto"
xmlns:card_view="http://schemas.android.com/apk/res-auto"
xmlns:tools="http://schemas.android.com/tools">
<data>
<variable
name="item"
type="com.jjjh.aacsample.model.Item"/>
</data>
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:adjustViewBounds="true">
<androidx.cardview.widget.CardView
android:id="@+id/cardViewItem"
android:layout_width="match_parent"
android:layout_height="match_parent"
android:layout_gravity="center"
android:layout_margin="5dp"
android:elevation="3dp"
card_view:cardCornerRadius="1dp">
<androidx.constraintlayout.widget.ConstraintLayout
android:layout_width="match_parent"
android:layout_height="wrap_content">
<ImageView
android:id="@+id/imageViewStatus"
android:layout_width="80dp"
android:layout_height="80dp"
android:layout_marginBottom="8dp"
android:layout_marginStart="8dp"
android:layout_marginTop="8dp"
android:background="?attr/selectableItemBackgroundBorderless"
android:scaleType="fitXY"
bind:statusIcon="@{item.status}"
bind:layout_constraintBottom_toBottomOf="parent"
bind:layout_constraintStart_toStartOf="parent"
bind:layout_constraintTop_toTopOf="parent"
android:layout_marginLeft="8dp"/>
<TextView
android:id="@+id/textViewTitle"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:layout_below="@+id/imageViewStatus"
android:layout_marginEnd="8dp"
android:layout_marginStart="8dp"
android:layout_marginTop="8dp"
android:padding="4dp"
android:text="@{item.title}"
android:textSize="18sp"
bind:layout_constraintEnd_toEndOf="parent"
bind:layout_constraintStart_toEndOf="@+id/imageViewStatus"
bind:layout_constraintTop_toTopOf="parent"/>
</androidx.constraintlayout.widget.ConstraintLayout>
</androidx.cardview.widget.CardView>
</LinearLayout>
</layout>
- RecyclerView를 구성하는 리스트 항목 하나에 대한 layout입니다.
- <data>...</data>를 이용하여 Item과의 DataBinding을 표시합니다.
- Item의 title과 status가 변경되면 그에 따라 변경된 ImageView와 TextView가 표시됩니다.
ItemAdapter.java
public class ItemAdapter extends RecyclerView.Adapter<ItemAdapter.ItemViewHolder> {
private ArrayList<Item> items;
@NonNull
@Override
public ItemViewHolder onCreateViewHolder(@NonNull ViewGroup viewGroup, int i) {
ListItemBinding listItemBinding =
DataBindingUtil.inflate(LayoutInflater.from(viewGroup.getContext()),
R.layout.list_item, viewGroup, false);
return new ItemViewHolder(listItemBinding);
}
@Override
public void onBindViewHolder(@NonNull ItemViewHolder itemViewHolder, int i) {
Item currentItem = items.get(i);
itemViewHolder.listItemBinding.setItem(currentItem);
}
@Override
public int getItemCount() {
if ( items != null ) {
return items.size();
} else {
return 0;
}
}
public void setItemsList(ArrayList<Item> items) {
this.items = items;
notifyDataSetChanged();
}
static class ItemViewHolder extends RecyclerView.ViewHolder {
private final ListItemBinding listItemBinding;
public ItemViewHolder(@NonNull ListItemBinding listItemBinding) {
super(listItemBinding.getRoot());
this.listItemBinding = listItemBinding;
}
}
}
- RecyclerView adapter입니다. MainActivity에서 데이타의 변화를 observe하고 있다가 변화가 감지되면 setItemsList가 호출되어 RecyclerView가 업데이트됩니다.
- 데이타가 많은 경우 notifyDataSetChanged로 인한 비효율적인 동작이 예상되어 DiffUtil 적용이 필요해보입니다.
build.gradle (Module)
android {
...
dataBinding {
enabled = true
}
}
dependencies {
...
implementation "androidx.lifecycle:lifecycle-extensions:2.2.0"
}
- DataBinding을 사용하기 위해 android 부분에 dataBinding을 enable 하였습니다.
- ViewModel을 사용하기 위해 dependencies 부분에 관련 부분을 추가하였습니다.
'Android' 카테고리의 다른 글
여러 git project의 변경 내역 확인 방법 (0) | 2021.03.03 |
---|---|
ninja: error: 'blahblah.so.toc', needed by 'something.so', missing and no known rule to make it (0) | 2021.03.03 |
'adb'은(는) 내부 또는 외부 명령, 실행할 수 있는 프로그램, 또는 배치 파일이 아닙니다. (0) | 2021.03.03 |
adb install fail : INSTALL_FAILED_NO_MATCHING_ABIS (0) | 2021.02.15 |
Android에 설치된 앱이 Platform Key로 서명되었는지 확인하는 방법 (0) | 2021.02.15 |