ОНОВЛЕННЯ 1
Оскільки бібліотека підтримки Android 23.2.0, для LayoutManagers був доданий метод setAutoMeasureEnabled(true)
. Це робить RecyclerView загортати його вміст і працює як шарм.
http://android-developers.blogspot.ru/2016/02/android-support-library-232.html
Тому просто додайте щось подібне:
LayoutManager layoutManager = new LinearLayoutManager(this);
layoutManager.setAutoMeasureEnabled(true);
recyclerView.setLayoutManager(layoutManager);
recyclerView.setNestedScrollingEnabled(false);
ОНОВЛЕННЯ 2
Оскільки 27.1.0 setAutoMeasureEnabled
застаріло, то вам слід забезпечити власну реалізацію LayoutManager за допомогою методу, що перекриваєтьсяisAutoMeasureEnabled()
Але після багатьох випадків використання RecyclerView я настійно рекомендую не використовувати його в режимі обгортання , оскільки це не те, для чого призначений. Спробуйте переробляти весь ваш макет, використовуючи звичайний єдиний RecyclerView з кількома типами елементів. Або скористайтеся підходом з LinearLayout, який я описав нижче як крайній захід
Стара відповідь (не рекомендується)
Можна використовувати RecyclerView
всередину NestedScrollView
. Перш за все, ви повинні реалізувати свій власний звичай LinearLayoutManager
, це змушує вас RecyclerView
обгортати його вміст. Наприклад:
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);
}
}
Після цього використовуйте це LayoutManager
для свогоRecyclerView
recyclerView.setLayoutManager(new WrappingLinearLayoutManager(getContext()));
Але вам також слід назвати ці два методи:
recyclerView.setNestedScrollingEnabled(false);
recyclerView.setHasFixedSize(false);
Тут setNestedScrollingEnabled(false)
вимкніть прокрутку для RecyclerView
, так що вона не перехоплює подію прокрутки з NestedScrollView
. І setHasFixedSize(false)
визначте, що зміни вмісту адаптера можуть змінювати розмірRecyclerView
Важлива примітка: Це рішення має невелику помилку в деяких випадках і має проблеми з виконанням, тому, якщо у вас є багато елементів, RecyclerView
я рекомендую скористатись LinearLayout
власною реалізацією списку списку, створіть для нього аналог адаптера та зробіть його поводитись як ListView
абоRecyclerView