PagingDataAdapter
public abstract class PagingDataAdapter<T extends Object, VH extends RecyclerView.ViewHolder> extends RecyclerView.Adapter
RecyclerView.Adapter base class for presenting paged data from PagingDatas in a RecyclerView.
This class is a convenience wrapper around AsyncPagingDataDiffer that implements common default behavior for item counting, and listening to update events.
To present a Pager, use collectLatest to observe Pager.flow and call submitData whenever a new PagingData is emitted.
If using RxJava and LiveData extensions on Pager, use the non-suspending overload of submitData, which accepts a Lifecycle.
PagingDataAdapter listens to internal PagingData loading events as pages are loaded, and uses DiffUtil on a background thread to compute fine grained updates as updated content in the form of new PagingData objects are received.
State Restoration: To be able to restore RecyclerView state (e.g. scroll position) after a configuration change / application recreate, PagingDataAdapter calls RecyclerView.Adapter.setStateRestorationPolicy with RecyclerView.Adapter.StateRestorationPolicy.PREVENT upon initialization and waits for the first page to load before allowing state restoration. Any other call to RecyclerView.Adapter.setStateRestorationPolicy by the application will disable this logic and will rely on the user set value.
val USER_COMPARATOR = object : DiffUtil.ItemCallback<User>() {
override fun areItemsTheSame(oldItem: User, newItem: User): Boolean =
// User ID serves as unique ID
oldItem.userId == newItem.userId
override fun areContentsTheSame(oldItem: User, newItem: User): Boolean =
// Compare full contents (note: Java users should call .equals())
oldItem == newItem
}
class UserAdapter : PagingDataAdapter<User, UserViewHolder>(USER_COMPARATOR) {
override fun onCreateViewHolder(parent: ViewGroup, viewType: Int): UserViewHolder {
return UserViewHolder.create(parent)
}
override fun onBindViewHolder(holder: UserViewHolder, position: Int) {
val repoItem = getItem(position)
// Note that item may be null, ViewHolder must support binding null item as placeholder
holder.bind(repoItem)
}
}
Summary
Public fields |
|
|---|---|
final @NonNull Flow<@NonNull CombinedLoadStates> |
A hot |
final @NonNull Flow<Unit> |
A hot |
Public constructors |
|
|---|---|
<T extends Object, VH extends RecyclerView.ViewHolder> PagingDataAdapter( |
|
Public methods |
|
|---|---|
final void |
addLoadStateListener(Add a |
final void |
addOnPagesUpdatedListener(@NonNull Function0<Unit> listener)Add a listener which triggers after the pages presented to the UI are updated, even if the actual items presented don't change. |
int |
Returns the total number of items in the data set held by the adapter. |
final long |
getItemId(int position)Note: |
final @Nullable T |
Returns the presented item at the specified position, without notifying Paging of the item access that would normally trigger page loads. |
final void |
refresh()Refresh the data presented by this |
final void |
removeLoadStateListener(Remove a previously registered |
final void |
removeOnPagesUpdatedListener(@NonNull Function0<Unit> listener)Remove a previously registered listener for new |
final void |
retry()Retry any failed load requests that would result in a |
final void |
setHasStableIds(boolean hasStableIds)Stable ids are unsupported by |
void |
Sets the state restoration strategy for the Adapter. |
final @NonNull ItemSnapshotList<@NonNull T> |
snapshot()Returns a new |
final void |
submitData(@NonNull PagingData<@NonNull T> pagingData)Present a |
final void |
submitData(Present a |
final @NonNull ConcatAdapter |
withLoadStateFooter(@NonNull LoadStateAdapter<@NonNull ?> footer)Create a |
final @NonNull ConcatAdapter |
withLoadStateHeader(@NonNull LoadStateAdapter<@NonNull ?> header)Create a |
final @NonNull ConcatAdapter |
withLoadStateHeaderAndFooter(Create a |
Inherited methods |
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
Public fields
loadStateFlow
@NonNull
public final @NonNull Flow<@NonNull CombinedLoadStates> loadStateFlow
A hot Flow of CombinedLoadStates that emits a snapshot whenever the loading state of the current PagingData changes.
This flow is conflated, so it buffers the last update to CombinedLoadStates and immediately delivers the current load states on collection.
onPagesUpdatedFlow
@NonNull
public final @NonNull Flow<Unit> onPagesUpdatedFlow
A hot Flow that emits after the pages presented to the UI are updated, even if the actual items presented don't change.
An update is triggered from one of the following:
-
submitDatais called and initial load completes, regardless of any differences in the loaded data -
A
Pageis inserted -
A
Pageis dropped
Note: This is a SharedFlow configured to replay 0 items with a buffer of size 64. If a collector lags behind page updates, it may trigger multiple times for each intermediate update that was presented while your collector was still working. To avoid this behavior, you can conflate this Flow so that you only receive the latest update, which is useful in cases where you are simply updating UI and don't care about tracking the exact number of page updates.
Public constructors
PagingDataAdapter
public final <T extends Object, VH extends RecyclerView.ViewHolder> PagingDataAdapter(
@NonNull DiffUtil.ItemCallback<@NonNull T> diffCallback,
@NonNull CoroutineDispatcher mainDispatcher,
@NonNull CoroutineDispatcher workerDispatcher
)
Public methods
addLoadStateListener
@NonNull
public final void addLoadStateListener(
@NonNull Function1<@NonNull CombinedLoadStates, Unit> listener
)
Add a CombinedLoadStates listener to observe the loading state of the current PagingData.
As new PagingData generations are submitted and displayed, the listener will be notified to reflect the current CombinedLoadStates.
val adapter = UserPagingAdapter()
adapter.addLoadStateListener {
// show a retry button outside the list when refresh hits an error
retryButton.isVisible = it.refresh is LoadState.Error
// swipeRefreshLayout displays whether refresh is occurring
swipeRefreshLayout.isRefreshing = it.refresh is LoadState.Loading
// show an empty state over the list when loading initially, before items are loaded
emptyState.isVisible = it.refresh is LoadState.Loading && adapter.itemCount == 0
}
| Parameters | |
|---|---|
@NonNull Function1<@NonNull CombinedLoadStates, Unit> listener |
|
| See also | |
|---|---|
removeLoadStateListener |
|
addOnPagesUpdatedListener
@NonNull
public final void addOnPagesUpdatedListener(@NonNull Function0<Unit> listener)
Add a listener which triggers after the pages presented to the UI are updated, even if the actual items presented don't change.
An update is triggered from one of the following:
-
submitDatais called and initial load completes, regardless of any differences in the loaded data -
A
Pageis inserted -
A
Pageis dropped
| See also | |
|---|---|
removeOnPagesUpdatedListener |
|
getItemCount
@NonNull
public int getItemCount()
Returns the total number of items in the data set held by the adapter.
| Returns | |
|---|---|
int |
The total number of items in this adapter. |
getItemId
@NonNull
public final long getItemId(int position)
Note: getItemId is final, because stable IDs are unnecessary and therefore unsupported.
PagingDataAdapter's async diffing means that efficient change animations are handled for you, without the performance drawbacks of RecyclerView.Adapter.notifyDataSetChanged. Instead, the diffCallback parameter of the PagingDataAdapter serves the same functionality - informing the adapter and RecyclerView how items are changed and moved.
peek
@Nullable
public final T peek(@IntRange(from = 0) int index)
Returns the presented item at the specified position, without notifying Paging of the item access that would normally trigger page loads.
| Parameters | |
|---|---|
@IntRange(from = 0) int index |
Index of the presented item to return, including placeholders. |
| Returns | |
|---|---|
T |
The presented item at position |
refresh
@NonNull
public final void refresh()
Refresh the data presented by this PagingDataAdapter.
refresh triggers the creation of a new PagingData with a new instance of PagingSource to represent an updated snapshot of the backing dataset. If a RemoteMediator is set, calling refresh will also trigger a call to RemoteMediator.load with LoadType to allow RemoteMediator</c