blob: 0c5a63b592f913c9b39f30ecac8092c543db6230 [file] [log] [blame]
/*
* Copyright (C) 2014 The Android Open Source Project
*
* Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except
* in compliance with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under the License
* is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express
* or implied. See the License for the specific language governing permissions and limitations under
* the License.
*/
package androidx.leanback.widget;
import android.util.Log;
import androidx.annotation.Nullable;
import androidx.recyclerview.widget.DiffUtil;
import androidx.recyclerview.widget.ListUpdateCallback;
import java.util.ArrayList;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
/**
* An {@link ObjectAdapter} implemented with an {@link ArrayList}.
*/
public class ArrayObjectAdapter extends ObjectAdapter {
private static final Boolean DEBUG = false;
private static final String TAG = "ArrayObjectAdapter";
private final List mItems = new ArrayList<Object>();
// To compute the payload correctly, we should use a temporary list to hold all the old items.
private final List mOldItems = new ArrayList<Object>();
// Un modifiable version of mItems;
private List mUnmodifiableItems;
/**
* Constructs an adapter with the given {@link PresenterSelector}.
*/
public ArrayObjectAdapter(PresenterSelector presenterSelector) {
super(presenterSelector);
}
/**
* Constructs an adapter that uses the given {@link Presenter} for all items.
*/
public ArrayObjectAdapter(Presenter presenter) {
super(presenter);
}
/**
* Constructs an adapter.
*/
public ArrayObjectAdapter() {
super();
}
@Override
public int size() {
return mItems.size();
}
@Override
public Object get(int index) {
return mItems.get(index);
}
/**
* Returns the index for the first occurrence of item in the adapter, or -1 if
* not found.
*
* @param item The item to find in the list.
* @return Index of the first occurrence of the item in the adapter, or -1
* if not found.
*/
public int indexOf(Object item) {
return mItems.indexOf(item);
}
/**
* Notify that the content of a range of items changed. Note that this is
* not same as items being added or removed.
*
* @param positionStart The position of first item that has changed.
* @param itemCount The count of how many items have changed.
*/
public void notifyArrayItemRangeChanged(int positionStart, int itemCount) {
notifyItemRangeChanged(positionStart, itemCount);
}
/**
* Adds an item to the end of the adapter.
*
* @param item The item to add to the end of the adapter.
*/
public void add(Object item) {
add(mItems.size(), item);
}
/**
* Inserts an item into this adapter at the specified index.
* If the index is > {@link #size} an exception will be thrown.
*
* @param index The index at which the item should be inserted.
* @param item The item to insert into the adapter.
*/
public void add(int index, Object item) {
mItems.add(index, item);
notifyItemRangeInserted(index, 1);
}
/**
* Adds the objects in the given collection to the adapter, starting at the
* given index. If the index is >= {@link #size} an exception will be thrown.
*
* @param index The index at which the items should be inserted.
* @param items A {@link Collection} of items to insert.
*/
public void addAll(int index, Collection items) {
int itemsCount = items.size();
if (itemsCount == 0) {
return;
}
mItems.addAll(index, items);
notifyItemRangeInserted(index, itemsCount);
}
/**
* Removes the first occurrence of the given item from the adapter.
*
* @param item The item to remove from the adapter.
* @return True if the item was found and thus removed from the adapter.
*/
public boolean remove(Object item) {
int index = mItems.indexOf(item);
if (index >= 0) {
mItems.remove(index);
notifyItemRangeRemoved(index, 1);
}
return index >= 0;
}
/**
* Moved the item at fromPosition to toPosition.
*
* @param fromPosition Previous position of the item.
* @param toPosition New position of the item.
*/
public void move(int fromPosition, int toPosition) {
if (fromPosition == toPosition) {
// no-op
return;
}
Object item = mItems.remove(fromPosition);
mItems.add(toPosition, item);
notifyItemMoved(fromPosition, toPosition);
}
/**
* Replaces item at position with a new item and calls notifyItemRangeChanged()
* at the given position. Note that this method does not compare new item to
* existing item.
*
* @param position The index of item to replace.
* @param item The new item to be placed at given position.
*/
public void replace(int position, Object item) {
mItems.set(position, item);
notifyItemRangeChanged(position, 1);
}
/**
* Removes a range of items from the adapter. The range is specified by giving
* the starting position and the number of elements to remove.
*
* @param position The index of the first item to remove.
* @param count The number of items to remove.
* @return The number of items removed.
*/
public int removeItems(int position, int count) {
int itemsToRemove = Math.min(count, mItems.size() - position);
if (itemsToRemove <= 0) {
return 0;
}
for (int i = 0; i < itemsToRemove; i++) {
mItems.remove(position);
}
notifyItemRangeRemoved(position, itemsToRemove);
return itemsToRemove;
}
/**
* Removes all items from this adapter, leaving it empty.
*/
public void clear() {
int itemCount = mItems.size();
if (itemCount == 0) {
return;
}
mItems.clear();
notifyItemRangeRemoved(0, itemCount);
}
/**
* Gets a read-only view of the list of object of this ArrayObjectAdapter.
*/
public <E> List<E> unmodifiableList() {
// The mUnmodifiableItems will only be created once as long as the content of mItems has not
// been changed.
if (mUnmodifiableItems == null) {
mUnmodifiableItems = Collections.unmodifiableList(mItems);
}
return mUnmodifiableItems;
}
@Override
public boolean isImmediateNotifySupported() {
return true;
}
ListUpdateCallback mListUpdateCallback;
/**
* Set a new item list to adapter. The DiffUtil will compute the difference and dispatch it to
* specified position.
*
* @param itemList List of new Items
* @param callback Optional DiffCallback Object to compute the difference between the old data
* set and new data set. When null, {@link #notifyChanged()} will be fired.
*/
public void setItems(final List itemList, final DiffCallback callback) {
if (callback == null) {
// shortcut when DiffCallback is not provided
mItems.clear();
mItems.addAll(itemList);
notifyChanged();
return;
}
mOldItems.clear();
mOldItems.addAll(mItems);
DiffUtil.DiffResult diffResult = DiffUtil.calculateDiff(new DiffUtil.Callback() {
@Override
public int getOldListSize() {
return mOldItems.size();
}
@Override
public int getNewListSize() {
return itemList.size();
}
@Override
public boolean areItemsTheSame(int oldItemPosition, int newItemPosition) {
return callback.areItemsTheSame(mOldItems.get(oldItemPosition),
itemList.get(newItemPosition));
}
@Override
public boolean areContentsTheSame(int oldItemPosition, int newItemPosition) {
return callback.areContentsTheSame(mOldItems.get(oldItemPosition),
itemList.get(newItemPosition));
}
@Nullable
@Override
public Object getChangePayload(int oldItemPosition, int newItemPosition) {
return callback.getChangePayload(mOldItems.get(oldItemPosition),
itemList.get(newItemPosition));
}
});
// update items.
mItems.clear();
mItems.addAll(itemList);
// dispatch diff result
if (mListUpdateCallback == null) {
mListUpdateCallback = new ListUpdateCallback() {
@Override
public void onInserted(int position, int count) {
if (DEBUG) {
Log.d(TAG, "onInserted");
}
notifyItemRangeInserted(position, count);
}
@Override
public void onRemoved(int position, int count) {
if (DEBUG) {
Log.d(TAG, "onRemoved");
}
notifyItemRangeRemoved(position, count);
}
@Override
public void onMoved(int fromPosition, int toPosition) {
if (DEBUG) {
Log.d(TAG, "onMoved");
}
notifyItemMoved(fromPosition, toPosition);
}
@Override
public void onChanged(int position, int count, Object payload) {
if (DEBUG) {
Log.d(TAG, "onChanged");
}
notifyItemRangeChanged(position, count, payload);
}
};
}
diffResult.dispatchUpdatesTo(mListUpdateCallback);
mOldItems.clear();
}
}