Próbuję dodać odstępy poniżej ostatniego wiersza elementu za RecyclerView
pomocą GridLayoutManager
. Użyłem ItemDecoration
do tego niestandardowego wypełnienia dolnego, gdy jego ostatni element wygląda następująco:
public class SpaceItemDecoration extends RecyclerView.ItemDecoration {
private int space;
private int bottomSpace = 0;
public SpaceItemDecoration(int space, int bottomSpace) {
this.space = space;
this.bottomSpace = bottomSpace;
}
public SpaceItemDecoration(int space) {
this.space = space;
this.bottomSpace = 0;
}
@Override
public void getItemOffsets(Rect outRect, View view,
RecyclerView parent, RecyclerView.State state) {
int childCount = parent.getChildCount();
final int itemPosition = parent.getChildAdapterPosition(view);
final int itemCount = state.getItemCount();
outRect.left = space;
outRect.right = space;
outRect.bottom = space;
outRect.top = space;
if (itemCount > 0 && itemPosition == itemCount - 1) {
outRect.bottom = bottomSpace;
}
}
}
Ale problem z tą metodą polega na tym, że zepsuła wysokość elementów w siatce w ostatnim rzędzie. Domyślam się, że GridLayoutManager
zmienia to wysokości elementów na podstawie odstępów w lewo. Jaki jest właściwy sposób osiągnięcia tego?
To zadziała poprawnie dla LinearLayoutManager
. Na wszelki wypadek jest GridLayoutManager
to problematyczne.
Jest to bardzo przydatne, jeśli masz FAB
na dole i potrzebujesz elementów w ostatnim rzędzie, aby przewijać powyżej FAB
, aby były widoczne.
android:scrollbarStyle="outsideOverlay"
Należy użyć opcji Dekoracja w widoku recyklera dla dolnego marginesu tylko w przypadku ostatniego elementu
recyclerView.addItemDecoration(MemberItemDecoration()) public class MemberItemDecoration extends RecyclerView.ItemDecoration { @Override public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) { // only for the last one if (parent.getChildAdapterPosition(view) == parent.getAdapter().getItemCount() - 1) { outRect.bottom = 50/* set your margin here */; } } }
źródło
Miałem podobny problem i odpowiedziałem na inny wątek w przepełnieniu stosu. Aby pomóc innym, którzy trafią na tę stronę, opublikuję ją ponownie tutaj.
Po przeczytaniu wszystkich innych odpowiedzi i stwierdziłem, że zmiany w układzie XML dla recyclinglerview działają zgodnie z oczekiwaniami w moim widoku recyklera:
android:paddingBottom="127px" android:clipToPadding="false" android:scrollbarStyle="outsideOverlay"
Kompletny układ wygląda następująco:
<android.support.v7.widget.RecyclerView android:id="@+id/library_list" android:layout_width="match_parent" android:layout_height="wrap_content" android:layout_marginStart="160px" android:layout_marginEnd="160px" tools:listitem="@layout/library_list_item" />
Aby zobaczyć efekt przed i po, zobacz link na androidblog.us: Dodawanie miejsca do końca przeglądu Androida
Daj mi znać, jak to działa.
David
źródło
Możesz użyć poniższego kodu, aby wykryć pierwszy i ostatni wiersz w widoku siatki i odpowiednio ustawić przesunięcie górne i dolne.
@Override public void getItemOffsets(Rect outRect, View view, RecyclerView parent, State state) { LayoutParams params = (LayoutParams) view.getLayoutParams(); int pos = params.getViewLayoutPosition(); int spanCount = mGridLayoutManager.getSpanCount(); boolean isFirstRow = pos < spanCount; boolean isLastRow = state.getItemCount() - 1 - pos < spanCount; if (isFirstRow) { outRect.top = top offset value here } if (isLastRow) { outRect.bottom = bottom offset value here } } // you also need to keep reference to GridLayoutManager to know the span count private final GridLayoutManager mGridLayoutManager;
źródło
Możesz dodać pustą stopkę do widoku recyklingu. Twoje wypełnienie będzie miało rozmiar twojej stopki.
@Override public Holder onCreateViewHolder( ViewGroup parent, int viewType) { if (viewType == FOOTER) { return new FooterHolder(); } View view = LayoutInflater.from(parent.getContext()).inflate(R.layout.item, parent, false); return new Holder(view); } @Override public void onBindViewHolder(final Holder holder, final int position) { //if footer if (position == items.getSize() - 1) { //do nothing return; } //do regular object bindding } @Override public int getItemViewType(int position) { return (position == items.getSize() - 1) ? FOOTER : ITEM_VIEW_TYPE_ITEM; } @Override public int getItemCount() { //add one for the footer return items.size() + 1; }
źródło
W przypadku takich rzeczy zaleca się rozwiązywanie za pomocą ItemDecoration, ponieważ są one do tego przeznaczone.
public class ListSpacingDecoration extends RecyclerView.ItemDecoration { private static final int VERTICAL = OrientationHelper.VERTICAL; private int orientation = -1; private int spanCount = -1; private int spacing; public ListSpacingDecoration(Context context, @DimenRes int spacingDimen) { spacing = context.getResources().getDimensionPixelSize(spacingDimen); } public ListSpacingDecoration(int spacingPx) { spacing = spacingPx; } @Override public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) { super.getItemOffsets(outRect, view, parent, state); if (orientation == -1) { orientation = getOrientation(parent); } if (spanCount == -1) { spanCount = getTotalSpan(parent); } int childCount = parent.getLayoutManager().getItemCount(); int childIndex = parent.getChildAdapterPosition(view); int itemSpanSize = getItemSpanSize(parent, childIndex); int spanIndex = getItemSpanIndex(parent, childIndex); /* INVALID SPAN */ if (spanCount < 1) return; setSpacings(outRect, parent, childCount, childIndex, itemSpanSize, spanIndex); } protected void setSpacings(Rect outRect, RecyclerView parent, int childCount, int childIndex, int itemSpanSize, int spanIndex) { if (isBottomEdge(parent, childCount, childIndex, itemSpanSize, spanIndex)) { outRect.bottom = spacing; } } @SuppressWarnings("all") protected int getTotalSpan(RecyclerView parent) { RecyclerView.LayoutManager mgr = parent.getLayoutManager(); if (mgr instanceof GridLayoutManager) { return ((GridLayoutManager) mgr).getSpanCount(); } else if (mgr instanceof StaggeredGridLayoutManager) { return ((StaggeredGridLayoutManager) mgr).getSpanCount(); } else if (mgr instanceof LinearLayoutManager) { return 1; } return -1; } @SuppressWarnings("all") protected int getItemSpanSize(RecyclerView parent, int childIndex) { RecyclerView.LayoutManager mgr = parent.getLayoutManager(); if (mgr instanceof GridLayoutManager) { return ((GridLayoutManager) mgr).getSpanSizeLookup().getSpanSize(childIndex); } else if (mgr instanceof StaggeredGridLayoutManager) { return 1; } else if (mgr instanceof LinearLayoutManager) { return 1; } return -1; } @SuppressWarnings("all") protected int getItemSpanIndex(RecyclerView parent, int childIndex) { RecyclerView.LayoutManager mgr = parent.getLayoutManager(); if (mgr instanceof GridLayoutManager) { return ((GridLayoutManager) mgr).getSpanSizeLookup().getSpanIndex(childIndex, spanCount); } else if (mgr instanceof StaggeredGridLayoutManager) { return childIndex % spanCount; } else if (mgr instanceof LinearLayoutManager) { return 0; } return -1; } @SuppressWarnings("all") protected int getOrientation(RecyclerView parent) { RecyclerView.LayoutManager mgr = parent.getLayoutManager(); if (mgr instanceof LinearLayoutManager) { return ((LinearLayoutManager) mgr).getOrientation(); } else if (mgr instanceof GridLayoutManager) { return ((GridLayoutManager) mgr).getOrientation(); } else if (mgr instanceof StaggeredGridLayoutManager) { return ((StaggeredGridLayoutManager) mgr).getOrientation(); } return VERTICAL; } protected boolean isBottomEdge(RecyclerView parent, int childCount, int childIndex, int itemSpanSize, int spanIndex) { if (orientation == VERTICAL) { return isLastItemEdgeValid((childIndex >= childCount - spanCount), parent, childCount, childIndex, spanIndex); } else { return (spanIndex + itemSpanSize) == spanCount; } } protected boolean isLastItemEdgeValid(boolean isOneOfLastItems, RecyclerView parent, int childCount, int childIndex, int spanIndex) { int totalSpanRemaining = 0; if (isOneOfLastItems) { for (int i = childIndex; i < childCount; i++) { totalSpanRemaining = totalSpanRemaining + getItemSpanSize(parent, i); } } return isOneOfLastItems && (totalSpanRemaining <= spanCount - spanIndex); } }
Skopiowałem tekst zredagowany z mojej oryginalnej odpowiedzi , która w rzeczywistości dotyczy równych odstępów, ale to ta sama koncepcja.
źródło
Możesz wziąć DividerItemDecoration.java jako przykład z kodu źródłowego i zamienić
for (int i = 0; i < childCount; i++)
z
for (int i = 0; i < childCount - 1; i++)
in drawVertical () i drawHorizontal ()
źródło