Я работаю над фрагментом ViewPager, который использует FragmentStatePagerAdapter в качестве адаптера.Данные Адаптера - это объект List, который поступает из базы данных, и я наблюдаю за запросом с помощью MediatorLiveData.MediatorLiveData объединяет шесть разных списков, которые у меня есть на один день, и превращает их в один список, который используется адаптером.
Я хочу добавить элемент в список к определенному индексу и динамически обновлять пользовательский интерфейс.Наблюдатель уведомляет адаптер, когда элемент добавляется, и обновление работает нормально, однако, когда я пытаюсь сделать это для определенного индекса, вызов notifyDataSetChanged () вызывает IndexOutOfBoundsError.
Я инициализирую адаптер со списком какследует:
public MealDetailsPagerAdapter(FragmentManager fm, List<Long> list, long date, MealViewModel vm) {
super(fm);
this.mealIdList = list;
this.date = date;
this.model = vm;
}
Если MealViewModel не имеет отношения к этому вопросу, он используется для фрагмента, который создает адаптер.Изменения списка выполняются другой моделью представления.
Вот код, который работает правильно:
public void changeItems(List<Long> list, long date) {
if(this.date != date){
this.mealIdList = list;
notifyDataSetChanged();
return;
}
mealIdList.addAll(list);
mealIdList = removeDuplicates(mealIdList);
System.out.println(Arrays.toString(mealIdList.toArray()));
notifyDataSetChanged();
}
И обозреватель, который вызывает его:
@Override
public void onActivityCreated(@Nullable Bundle savedInstanceState) {
super.onActivityCreated(savedInstanceState);
mViewModel.getMealIdsInDay().observe(getViewLifecycleOwner(), longs -> {
myAdapter.changeItems(longs, mCurrentIndexDay);
if(isResumed){
mViewPager.setCurrentItem(myAdapter.getPageIndexForMealId(mViewModel.getMealId()));
isResumed=false;
}
updateIndicator(mViewPager);
});
}
Где isResumed isпо умолчанию false, однако, если пользователь добавляет новую еду, isResumed изменяется на true, а текущая позиция viewPager изменяется на позицию созданной еды.
Однако, с рабочим кодом, позиция созданной еды всегда будетв конце списка адаптера из-за addAll ().Я хочу добавить блюдо в определенную позицию, но если я получаю индекс с помощью mViewPager.getCurrentItem () и отправляю его в метод следующим образом:
mealIdList.addAll(index, list);
, сам addAll работает, но notifyDataSetChanged ()вызывает IndexOutOfBoundsError.
Вот полная трассировка стека:
E/AndroidRuntime: FATAL EXCEPTION: main
Process: fi.seehowyoueat.shye.debug, PID: 14148
java.lang.IndexOutOfBoundsException: Index: 2, Size: 2
at java.util.ArrayList.set(ArrayList.java:453)
at androidx.fragment.app.FragmentStatePagerAdapter.destroyItem(FragmentStatePagerAdapter.java:147)
at androidx.viewpager.widget.ViewPager.populate(ViewPager.java:1212)
at androidx.viewpager.widget.ViewPager.setCurrentItemInternal(ViewPager.java:669)
at androidx.viewpager.widget.ViewPager.setCurrentItemInternal(ViewPager.java:631)
at androidx.viewpager.widget.ViewPager.dataSetChanged(ViewPager.java:1086)
at androidx.viewpager.widget.ViewPager$PagerObserver.onChanged(ViewPager.java:3097)
at androidx.viewpager.widget.PagerAdapter.notifyDataSetChanged(PagerAdapter.java:291)
at fi.octo3.shye.view.viewpagers.MealDetailsPagerAdapter.changeTheItems(MealDetailsPagerAdapter.java:87)
at fi.octo3.shye.fragments.MealDetailsFragment.lambda$onActivityCreated$0(MealDetailsFragment.java:223)
at fi.octo3.shye.fragments.-$$Lambda$MealDetailsFragment$XB4Svnx84FE6kVa5Gzle01e8F3o.onChanged(Unknown Source:4)
at androidx.lifecycle.LiveData.considerNotify(LiveData.java:131)
at androidx.lifecycle.LiveData.dispatchingValue(LiveData.java:149)
at androidx.lifecycle.LiveData.setValue(LiveData.java:307)
at androidx.lifecycle.MutableLiveData.setValue(MutableLiveData.java:50)
at fi.octo3.shye.models.viewmodel.-$$Lambda$2CouvY7DQv4NA0nk6EMoH6jUavw.onChanged(Unknown Source:4)
at androidx.lifecycle.MediatorLiveData$Source.onChanged(MediatorLiveData.java:152)
at androidx.lifecycle.LiveData.considerNotify(LiveData.java:131)
at androidx.lifecycle.LiveData.dispatchingValue(LiveData.java:149)
at androidx.lifecycle.LiveData.setValue(LiveData.java:307)
at androidx.lifecycle.LiveData$1.run(LiveData.java:91)
at android.os.Handler.handleCallback(Handler.java:873)
at android.os.Handler.dispatchMessage(Handler.java:99)
at android.os.Looper.loop(Looper.java:280)
at android.app.ActivityThread.main(ActivityThread.java:6748)
at java.lang.reflect.Method.invoke(Native Method)
at com.android.internal.os.RuntimeInit$MethodAndArgsCaller.run(RuntimeInit.java:493)
at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:858)
Глядя на это, кажется, что проблема как-то вызвана LiveData, но я не уверен, как исправитьЭто.Если вы хотите взглянуть на метод, который обновляет MediatorLiveData, который у меня есть, он находится здесь:
private void refresh(long key){
if(groupsAndMealsMap.get(key) != null){
//remove source and then value from map
liveDataMerger.removeSource(groupsAndMealsMap.get(key));
groupsAndMealsMap.remove(key);
//add new value to map and add it as a source
groupsAndMealsMap.append(key, mealIdsInGroup);
liveDataMerger.addSource(groupsAndMealsMap.get(key), liveDataMerger::setValue);
}
}
refresh () вызывается addItem (), который получает обновленный список блюд из базы данных (список, являющийся едыIdsInGroup) и liveDataMerger состоит из шести объектов LiveData>.
Любая помощь приветствуется.Спасибо!
РЕДАКТИРОВАТЬ
Вот метод addItem (), так как вы можете видеть, как исполнитель службы ожидает выполнения операции, прежде чем перейти к обновлению () метод.
public void addItem(Meal meal, long mealGroupId){
ExecutorService service = Executors.newCachedThreadPool();
service.execute(() -> {
setMealId(db.mealDao().insertMealIntoGroup(meal, mealGroupId));
mealIdsInGroup = db.mealDao().loadMealsWithinGroup(mealGroupId);
});
service.shutdown();
try {
service.awaitTermination(Long.MAX_VALUE, TimeUnit.NANOSECONDS);
} catch (InterruptedException e) {
e.printStackTrace();
}
refresh(mealGroupId);
}