Android TV перезагружает данные в BrowseFragment
Я пишу приложение для Android TV и столкнулся с этой проблемой. Я должен заменить все строки в BrowseFragment новыми при получении события. Что я пробовал:
mRowsAdapter.clear();
...
mRowsAdapter.add(add new rows here);
Где mRowsAdapter - это ArrayObjectAdapter с элементами ListRow. Также попробовал этот подход:
mRowsAdapter = new ArrayObjectAdapter(new ListRowPresenter());
setAdapter(mRowsAdapter);
mRowsAdapter.add(add new rows here);
И решения из этого вопроса. Ни один из них не работает. Я продолжаю получать IOBE. Вот журналы:
java.lang.IndexOutOfBoundsException: Index: 7, Size: 0
at java.util.ArrayList.get(ArrayList.java:411)
at android.support.v17.leanback.widget.ArrayObjectAdapter.get(ArrayObjectAdapter.java:56)
at android.support.v17.leanback.app.ListRowDataAdapter.get(ListRowDataAdapter.java:65)
at android.support.v17.leanback.widget.ItemBridgeAdapter.getItemViewType(ItemBridgeAdapter.java:248)
at android.support.v7.widget.RecyclerView$Recycler.validateViewHolderForOffsetPosition(RecyclerView.java:5046)
at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:5172)
at android.support.v7.widget.RecyclerView$Recycler.getViewForPosition(RecyclerView.java:5153)
at android.support.v17.leanback.widget.GridLayoutManager.getViewForPosition(GridLayoutManager.java:971)
at android.support.v17.leanback.widget.GridLayoutManager$2.createItem(GridLayoutManager.java:1464)
at android.support.v17.leanback.widget.SingleRow.appendVisibleItems(SingleRow.java:111)
at android.support.v17.leanback.widget.Grid.appendVisibleItems(Grid.java:371)
at android.support.v17.leanback.widget.GridLayoutManager.appendVisibleItems(GridLayoutManager.java:1702)
at android.support.v17.leanback.widget.GridLayoutManager.fastRelayout(GridLayoutManager.java:1764)
at android.support.v17.leanback.widget.GridLayoutManager.onLayoutChildren(GridLayoutManager.java:1888)
at android.support.v7.widget.RecyclerView.dispatchLayoutStep2(RecyclerView.java:3374)
at android.support.v7.widget.RecyclerView.dispatchLayout(RecyclerView.java:3183)
at android.support.v7.widget.RecyclerView.onLayout(RecyclerView.java:3627)
at android.view.View.layout(View.java:17519)
at android.view.ViewGroup.layout(ViewGroup.java:5612)
at android.support.v17.leanback.widget.ScaleFrameLayout.onLayout(ScaleFrameLayout.java:172)
at android.view.View.layout(View.java:17519)
at android.view.ViewGroup.layout(ViewGroup.java:5612)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:323)
at android.widget.FrameLayout.onLayout(FrameLayout.java:261)
at android.view.View.layout(View.java:17519)
at android.view.ViewGroup.layout(ViewGroup.java:5612)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:323)
at android.widget.FrameLayout.onLayout(FrameLayout.java:261)
at android.view.View.layout(View.java:17519)
at android.view.ViewGroup.layout(ViewGroup.java:5612)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:323)
at android.widget.FrameLayout.onLayout(FrameLayout.java:261)
at android.view.View.layout(View.java:17519)
at android.view.ViewGroup.layout(ViewGroup.java:5612)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:323)
at android.widget.FrameLayout.onLayout(FrameLayout.java:261)
at android.view.View.layout(View.java:17519)
at android.view.ViewGroup.layout(ViewGroup.java:5612)
at android.widget.LinearLayout.setChildFrame(LinearLayout.java:1741)
at android.widget.LinearLayout.layoutVertical(LinearLayout.java:1585)
at android.widget.LinearLayout.onLayout(LinearLayout.java:1494)
at android.view.View.layout(View.java:17519)
at android.view.ViewGroup.layout(ViewGroup.java:5612)
at android.widget.FrameLayout.layoutChildren(FrameLayout.java:323)
at android.widget.FrameLayout.onLayout(FrameLayout.java:261)
at com.android.internal.policy.DecorView.onLayout(DecorView.java:724)
at android.view.View.layout(View.java:17519)
at android.view.ViewGroup.layout(ViewGroup.java:5612)
at android.view.ViewRootImpl.performLayout(ViewRootImpl.java:2342)
at android.view.ViewRootImpl.performTraversals(ViewRootImpl.java:2069)
at android.view.ViewRootImpl.doTraversal(ViewRootImpl.java:1246)
at android.view.ViewRootImpl$TraversalRunnable.run(ViewRootImpl.java:6301)
at android.view.Choreographer$CallbackRecord.run(Choreographer.java:871)
at android.view.Choreographer.doCallbacks(Choreographer.java:683)
at android.view.Choreographer.doFrame(Choreographer.java:619)
com.cloudify E/AndroidRuntime: at android.view.Choreographer$FrameDisplayEventReceiver.run(Choreographer.java:857)
at android.os.Handler.handleCallback(Handler.java:751)
at android.os.Handler.dispatchMessage(Handler.java:95)
at android.os.Looper.loop(Looper.java:154)
at android.app.ActivityThread.main(ActivityThread.java:6077)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:865)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:755)
Пожалуйста помоги. Спасибо.
3 ответа
Попробуйте сделать это:
mRowsAdapter.clear (); mRowsAdapter.add (добавьте новые строки здесь); mRowsAdapter.notifyArrayItemRangeChanged (0, mRowsAdapter.size ());
Вы получаете индекс вне границ, потому что внутренний массив объектов в адаптере пытается использовать свой собственный размер, чтобы определить, куда добавить элементы.
Когда вы вызываете.add(), это внутренне делает это -
/**
* Adds an item to the end of the list.
*
* @param item The item to add to the end of the list.
*/
public void add(Object item) {
add(mItems.size(), item);
}
/**
* Inserts an item into this list at the specified index.
*
* @param index The index at which the item should be inserted.
* @param item The item to insert into the list.
*/
public void add(int index, Object item) {
mItems.add(index, item);
notifyItemRangeInserted(index, 1);
}
это означает, что то, что вы называете clear, временно оставляет что-то не в порядке, а внутренний индекс не соответствует тому, чем он является на самом деле, и arraylist выдает IOBE. Возможно, вместо этого попробуйте использовать add(0, yourArray).
РЕДАКТИРОВАТЬ:
попробуйте добавить какую-то проверку перед тем, как добавлять свои объекты. Что-то вроде:
if((myAdapter.size() - 1) > indexImTryingToAddAt)
{
myAdapter.addAll(indexImTryingToAddAt, arrayOfObjects);
}
else
{
Log.d("index is larger than current adapter", indexImTryingToAddAt+"");
}
Способ, которым я решил подобную проблему:
Вы не можете поместить объект с заданным индексом в ArrayObjectAdapter, если массив пуст.
https://developer.android.com/reference/android/support/v17/leanback/widget/ArrayObjectAdapter
void add (int index, Object item)
Вставляет элемент в этот адаптер по указанному индексу. Если индекс> size(), будет выдано исключение.
Я не знаю, насколько гладко мое решение, но вы можете сделать следующее:
Заполните массив "фиктивными" объектами
mRowsAdapter.add(0, new ListRow(new HeaderItem(""), new ArrayObjectAdapter()));
mRowsAdapter.add(1, new ListRow(new HeaderItem(""), new ArrayObjectAdapter()));
mRowsAdapter.add(2, new ListRow(new HeaderItem(""), new ArrayObjectAdapter()));
и когда вы сделаете обратные вызовы, замените их:
mRowsAdapter.replace(0, new ListRow(new HeaderItem("List1"), list1));
mRowsAdapter.replace(1, new ListRow(new HeaderItem("List2"), list2));
mRowsAdapter.replace(2, new ListRow(new HeaderItem("List3"), list3));
Таким образом, вы также можете добавить счетчик элементов в HeaderItem:
mRowsAdapter.replace(2, new ListRow(new HeaderItem("List3 (" + list3.size() + ")"), list3));