AKTUALIZACJA 1
Od Android Support Library 23.2.0 dodano metodę setAutoMeasureEnabled(true)
dla LayoutManagers. To sprawia, że RecyclerView zawija zawartość i działa jak urok.
http://android-developers.blogspot.ru/2016/02/android-support-library-232.html
Dodaj więc coś takiego:
LayoutManager layoutManager = new LinearLayoutManager(this);
layoutManager.setAutoMeasureEnabled(true);
recyclerView.setLayoutManager(layoutManager);
recyclerView.setNestedScrollingEnabled(false);
AKTUALIZACJA 2
Ponieważ 27.1.0 setAutoMeasureEnabled
jest przestarzałe, należy zapewnić niestandardową implementację LayoutManager z przesłoniętą metodąisAutoMeasureEnabled()
Ale po wielu przypadkach użycia RecyclerView zdecydowanie nie zaleca się używania go w trybie owijania , ponieważ nie jest to przeznaczone do tego celu. Spróbuj przefiltrować cały układ za pomocą zwykłego pojedynczego RecyclerView z kilkoma typami przedmiotów. Lub użyj podejścia z LinearLayout, które opisałem poniżej jako ostateczność
Stara odpowiedź (niezalecana)
Możesz użyć w RecyclerView
środku NestedScrollView
. Przede wszystkim powinieneś wdrożyć własny niestandardowy LinearLayoutManager
, który powoduje RecyclerView
zawinięcie jego zawartości. Na przykład:
public class WrappingLinearLayoutManager extends LinearLayoutManager
{
public WrappingLinearLayoutManager(Context context) {
super(context);
}
private int[] mMeasuredDimension = new int[2];
@Override
public boolean canScrollVertically() {
return false;
}
@Override
public void onMeasure(RecyclerView.Recycler recycler, RecyclerView.State state,
int widthSpec, int heightSpec) {
final int widthMode = View.MeasureSpec.getMode(widthSpec);
final int heightMode = View.MeasureSpec.getMode(heightSpec);
final int widthSize = View.MeasureSpec.getSize(widthSpec);
final int heightSize = View.MeasureSpec.getSize(heightSpec);
int width = 0;
int height = 0;
for (int i = 0; i < getItemCount(); i++) {
if (getOrientation() == HORIZONTAL) {
measureScrapChild(recycler, i,
View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED),
heightSpec,
mMeasuredDimension);
width = width + mMeasuredDimension[0];
if (i == 0) {
height = mMeasuredDimension[1];
}
} else {
measureScrapChild(recycler, i,
widthSpec,
View.MeasureSpec.makeMeasureSpec(0, View.MeasureSpec.UNSPECIFIED),
mMeasuredDimension);
height = height + mMeasuredDimension[1];
if (i == 0) {
width = mMeasuredDimension[0];
}
}
}
switch (widthMode) {
case View.MeasureSpec.EXACTLY:
width = widthSize;
case View.MeasureSpec.AT_MOST:
case View.MeasureSpec.UNSPECIFIED:
}
switch (heightMode) {
case View.MeasureSpec.EXACTLY:
height = heightSize;
case View.MeasureSpec.AT_MOST:
case View.MeasureSpec.UNSPECIFIED:
}
setMeasuredDimension(width, height);
}
private void measureScrapChild(RecyclerView.Recycler recycler, int position, int widthSpec,
int heightSpec, int[] measuredDimension) {
View view = recycler.getViewForPosition(position);
if (view.getVisibility() == View.GONE) {
measuredDimension[0] = 0;
measuredDimension[1] = 0;
return;
}
// For adding Item Decor Insets to view
super.measureChildWithMargins(view, 0, 0);
RecyclerView.LayoutParams p = (RecyclerView.LayoutParams) view.getLayoutParams();
int childWidthSpec = ViewGroup.getChildMeasureSpec(
widthSpec,
getPaddingLeft() + getPaddingRight() + getDecoratedLeft(view) + getDecoratedRight(view),
p.width);
int childHeightSpec = ViewGroup.getChildMeasureSpec(
heightSpec,
getPaddingTop() + getPaddingBottom() + getDecoratedTop(view) + getDecoratedBottom(view),
p.height);
view.measure(childWidthSpec, childHeightSpec);
// Get decorated measurements
measuredDimension[0] = getDecoratedMeasuredWidth(view) + p.leftMargin + p.rightMargin;
measuredDimension[1] = getDecoratedMeasuredHeight(view) + p.bottomMargin + p.topMargin;
recycler.recycleView(view);
}
}
Następnie użyj tego LayoutManager
dla swojegoRecyclerView
recyclerView.setLayoutManager(new WrappingLinearLayoutManager(getContext()));
Ale powinieneś także wywołać te dwie metody:
recyclerView.setNestedScrollingEnabled(false);
recyclerView.setHasFixedSize(false);
Tutaj setNestedScrollingEnabled(false)
wyłącz przewijanie dla RecyclerView
, aby nie przechwytywał zdarzenia przewijania NestedScrollView
. I setHasFixedSize(false)
ustal, że zmiany w treści adaptera mogą zmienić rozmiar plikuRecyclerView
Ważna uwaga: w niektórych przypadkach to rozwiązanie jest trochę wadliwe i ma problemy z wydajnością, więc jeśli masz wiele elementów RecyclerView
, polecam skorzystanie z niestandardowej LinearLayout
implementacji widoku listy, stwórz dla niego analog i adapter. zachowuj się jak ListView
lubRecyclerView