StaggeredGridLayoutManager переупорядочивает элементы
Мои элементы имеют разную компоновку, и иногда я создаю пробел или пробелы поверх RecyclerView, перетаскивая элементы в макете. Когда я возвращаюсь к началу RecyclerView, некоторые элементы переупорядочиваются, и пробелы заполняются ими. Поведение фиксируется здесь:
Заполнение пробелов в порядке. Также хорошо, когда элементы меняют положение во время перетаскивания. Проблема в том, что когда дело доходит до прокрутки наверх. Изменение порядка происходит, когда прокрутка заканчивается - я не касаюсь экрана.
ЭТО КАК Я СОЗДАЮ РЕЦИКЛЕРВИС И МЕНЕДЖЕР
recyclerView = (RecyclerView)layout.findViewById(R.id.recycler_view);
stagaggeredGridLayoutManager =
new StaggeredGridLayoutManager(
getResources().getConfiguration().orientation == Configuration.ORIENTATION_LANDSCAPE ? 3 : 2, StaggeredGridLayoutManager.VERTICAL);
recyclerView.setLayoutManager(stagaggeredGridLayoutManager);
stagaggeredList = getListItemData();
StatisticsGridViewAdapter rcAdapter = new StatisticsGridViewAdapter(this.getActivity(), stagaggeredList, recyclerView);
ItemTouchHelper.Callback callback = new StatisticsTouchHelperCallback(rcAdapter);
touchHelper = new ItemTouchHelper(callback);
touchHelper.attachToRecyclerView(recyclerView);
recyclerView.setAdapter(rcAdapter);
ItemTouchHelper класс
public class StatisticsTouchHelperCallback extends ItemTouchHelper.Callback {
private final ItemTouchHelperAdapter touchHelperAdapter;
@Override
public boolean isLongPressDragEnabled() {
return true;
}
@Override
public boolean isItemViewSwipeEnabled() {
return false;
}
@Override
public int getMovementFlags(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder) {
int dragFlags = ItemTouchHelper.UP | ItemTouchHelper.DOWN |
ItemTouchHelper.LEFT | ItemTouchHelper.RIGHT;
int swipeFlags = ItemTouchHelper.START | ItemTouchHelper.END;
return makeMovementFlags(dragFlags, swipeFlags);
}
@Override
public boolean onMove(RecyclerView recyclerView, RecyclerView.ViewHolder viewHolder,
RecyclerView.ViewHolder target) {
touchHelperAdapter.onItemMove(viewHolder.getAdapterPosition(), target.getAdapterPosition());
return false;
}
@Override
public void onSwiped(RecyclerView.ViewHolder viewHolder, int direction) {
touchHelperAdapter.onItemDismiss(viewHolder.getAdapterPosition());
}
public StatisticsTouchHelperCallback(ItemTouchHelperAdapter adapter) {
touchHelperAdapter = adapter;
}
}
RecyclerView адаптер
public class StatisticsGridViewAdapter extends RecyclerView.Adapter<StatisticsGridItemViewHolder> implements ItemTouchHelperAdapter {
private List<Statistic> itemList;
private Context context;
@Override
public StatisticsGridItemViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
View layoutView = LayoutInflater.from(parent.getContext()).inflate(itemList.get(viewType).getStatisticType().getStringResourcesId(), null);
StatisticsGridItemViewHolder rcv = new StatisticsGridItemViewHolder(layoutView, viewType);
return rcv;
}
@Override
public void onBindViewHolder(final StatisticsGridItemViewHolder holder, int position) {
holder.getMenu().setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
PopupMenu popup = new PopupMenu(v.getContext(), holder.getMenu());
popup.getMenuInflater().inflate(R.menu.statistics__widget__popup_menu, popup.getMenu());
Menu menu = popup.getMenu();
MenuItem item = menu.add(R.string.statistics__widget__menu_item__pin_to_dashboard);
//item.setOnMenuItemClickListener()
popup.show();
}
});
}
@Override
public int getItemCount() {
return this.itemList.size();
}
@Override
public boolean onItemMove(int fromPosition, int toPosition) {
if (fromPosition < toPosition) {
for (int i = fromPosition; i < toPosition; i++) {
Collections.swap(itemList, i, i + 1);
}
} else {
for (int i = fromPosition; i > toPosition; i--) {
Collections.swap(itemList, i, i - 1);
}
}
notifyItemMoved(fromPosition, toPosition);
return true;
}
@Override
public void onItemDismiss(int position) {
itemList.remove(position);
notifyItemRemoved(position);
}
@Override
public int getItemViewType(int position) {return itemList.get(position).getStatisticType().getPosition();
}
public StatisticsGridViewAdapter(Context context, List<Statistic> itemList) {
this.itemList = itemList;
this.context = context;
}
}
interface ItemTouchHelperAdapter {
boolean onItemMove(int fromPosition, int toPosition);
void onItemDismiss(int position);
}
Прямо сейчас я использую 12 разных макетов с одинаковой структурой. Только высота отличается для имитации конечных виджетов.
Пример макета предмета
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout
xmlns:android="http://schemas.android.com/apk/res/android"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content" >
<LinearLayout
android:id="@+id/statistics__widget__main_layout"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:background="@drawable/order_tile_final"
android:layout_marginLeft="5dp"
android:layout_marginRight="5dp"
android:layout_marginTop="2dp"
android:layout_marginBottom="7dp"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:layout_marginLeft="17dp"
android:orientation="horizontal">
<TextView
android:id="@+id/widget_title"
android:layout_width="0dp"
android:layout_height="wrap_content"
android:text="@string/statistics__widget__forms_filled_title"
style="@style/tablet_common_font__main_3"
android:paddingTop="10dp"
android:layout_weight="1"/>
<ImageView
android:id="@+id/widget_menu"
android:layout_width="wrap_content"
android:layout_height="23dp"
android:layout_marginTop="7dp"
android:paddingTop="5dp"
android:layout_marginRight="8dp"
android:src="@drawable/a_menu_dark"/>
</LinearLayout>
<TextView
android:id="@+id/widget_value"
android:layout_width="wrap_content"
android:layout_height="wrap_content"
style="@style/tablet_common_font__headline_2"
android:layout_marginLeft="17dp"
android:layout_marginBottom="6dp"/>
</LinearLayout>
</LinearLayout>
4 ответа
Я думаю, что это обычное поведение StaggeredGridLayoutManager:
Когда состояние прокрутки изменяется на SCROLL_STATE_IDLE, StaggeredGrid проверит, есть ли пробелы в элементах из-за полного диапазона. Если он найдет, он перестроит и переместит элементы в правильные позиции с анимацией.
Вы можете остановить поведение заполнения пробелов, изменив стратегию на GAP_HANDLING_NONE:
stagaggeredGridLayoutManager.setGapStrategy(
StaggeredGridLayoutManager.GAP_HANDLING_NONE);
Я работал над Recyclerview прежде, если я понял вашу проблему, я думаю, что следующая вещь решит проблему.
@Override
public boolean onItemMove(int fromPosition, int toPosition) {
if (fromPosition < toPosition) {
for (int i = fromPosition; i < toPosition; i++) {
Collections.swap(itemList, i, i + 1);
notifyItemMoved(i, i+1);
}
} else {
for (int i = fromPosition; i > toPosition; i--) {
Collections.swap(itemList, i, i - 1);
notifyItemMoved(i, i-1);
}
}
return true;
}
Дайте мне знать, если какие-либо проблемы, или я не прав в понимании вашей проблемы.
Best Regds
Вы должны расширить представление Recycler, чтобы включить функцию переупорядочения.
Пожалуйста, перейдите по следующей ссылке: https://gist.github.com/mohlendo/68b7e2f89d0b1b354abe
А для полного примера используйте следующую ссылку: https://github.com/mohlendo/ReorderRecyclerView
Лучшее решение - создать свой StaggeredLayoutMenager
class YourStaggeredLayoutManager(count:Int, orientation:Int) : StaggeredGridLayoutManager(count,orientation) {
override fun onItemsUpdated(
recyclerView: RecyclerView,
positionStart: Int,
itemCount: Int,
payload: Any?
) {
}
override fun onItemsUpdated(recyclerView: RecyclerView, positionStart: Int, itemCount: Int) {
}
}