ScrollView wrap_content 高度内的 RecyclerView 未按预期工作
RecyclerView within ScrollView wrap_content height not working as intended
我在 ScrollView
中嵌入了 RecyclerView
。这是完整的布局:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent">
<ScrollView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fillViewport="true"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:orientation="vertical"
android:layout_height="wrap_content">
<fragment
android:name="devarshi.sample.view.ProductPortfolioFragment"
android:id="@+id/fragmentProductPortfolio"
android:layout_width="match_parent"
android:layout_margin="10dp"
android:layout_height="230dp" />
<LinearLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:text="TextView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/textViewProductName" />
<TextView
android:text="TextView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/textViewProductDescription" />
<LinearLayout
android:id="@+id/linearLayoutProductDetails"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:text="@string/product_detail_title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/textViewProductDetails" />
</LinearLayout>
</LinearLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="match_parent"
android:id="@+id/linearyLayoutOtherProductDetails"
android:layout_height="100dp">
<LinearLayout
android:orientation="vertical"
android:layout_width="0dp"
android:layout_weight="0.6"
android:layout_height="wrap_content">
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:id="@+id/linearLayoutPrice"
android:layout_height="wrap_content">
<TextView
android:id="@+id/textViewProductPrice"
android:layout_width="wrap_content"
android:textSize="15sp"
android:textStyle="bold"
android:textColor="@color/colorDiscountedPrice"
android:layout_height="wrap_content"
android:text="1500"/>
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewOldProductPrice"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="200"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:layout_height="wrap_content">
<TextView
android:id="@+id/textViewQuantity"
android:layout_width="wrap_content"
android:textSize="15sp"
android:textStyle="bold"
android:textColor="@color/colorDiscountedPrice"
android:layout_height="wrap_content"
android:text="Quantity"/>
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewQuantityValue"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="200"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:layout_height="wrap_content">
<TextView
android:id="@+id/textViewBrand"
android:layout_width="wrap_content"
android:textSize="15sp"
android:textStyle="bold"
android:textColor="@color/colorDiscountedPrice"
android:layout_height="wrap_content"
android:text="Brand"/>
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewBrandValue"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="200"/>
</LinearLayout>
</LinearLayout>
<LinearLayout
android:orientation="vertical"
android:layout_width="0dp"
android:layout_weight="0.4"
android:layout_height="wrap_content">
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:id="@+id/linearLayoutFreeShipping"
android:layout_height="wrap_content">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:srcCompat="@drawable/product_detail_free_shipping_icon"
android:id="@+id/imageViewFreeShipping" />
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewFreeShipping"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="@string/product_detail_free_shipping"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:id="@+id/linearLayoutCashOnDelivery"
android:layout_height="wrap_content">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:srcCompat="@drawable/product_detail_cash_on_delivery_icon"
android:id="@+id/imageViewCashOnDelivery" />
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewCashOnDelivery"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="@string/product_detail_cash_on_delivery"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:id="@+id/linearLayoutUnit"
android:layout_height="wrap_content">
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewUnitLabel"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="@string/product_detail_unit"/>
<Button
android:text="-"
android:layout_width="@dimen/product_detail_counter_button_width"
android:layout_height="@dimen/product_detail_counter_button_height"
android:id="@+id/buttonDecrement" />
<TextView
android:id="@+id/textViewUnitValue"
android:layout_width="27dp"
android:textSize="14sp"
android:gravity="center"
android:textColor="@color/colorActualPrice"
android:layout_height="@dimen/product_detail_counter_button_height"
android:text="0"/>
<Button
android:text="+"
android:layout_width="@dimen/product_detail_counter_button_width"
android:layout_height="@dimen/product_detail_counter_button_height"
android:id="@+id/buttonIncrement" />
</LinearLayout>
</LinearLayout>
</LinearLayout>
<android.support.v7.widget.RecyclerView
android:id="@+id/recyclerViewProductDetails"
android:layout_width="match_parent"
android:layout_height="wrap_content">
</android.support.v7.widget.RecyclerView>
</LinearLayout>
</ScrollView>
</LinearLayout>
问题是 -
虽然我已将 RecyclerView
的 layout_height
设置为 wrap_content
,但它的高度部分包裹了内容,即理想情况下,在 wrap_content
之后,它的高度应该是 150dp
但目前正在 70dp
.
另一个有趣的场景是 -
如果我将 linearyLayoutOtherProductDetails
移动到 recyclerViewProductDetails
以下,那么它会准确地将 RecyclerView
高度显示为 150dp
。似乎当 RecyclerView
加载到布局上时,它正在根据当前在布局上可见的项目(其中)的数量计算高度,即高度等于 5 个项目而不是 9 个项目。
关于如何在第一种情况下让 RecyclerView
的高度完全包裹其内容有什么想法吗?
注:我用的是com.android.support:recyclerview-v7:25.0.0
制作 recyclerview 的 linearlayoutOtherProductdetails header(第一项)并删除 scrollview。因为另一个滚动视图内的滚动视图无法正常工作。
另一种解决方案是使用自定义布局管理器,将回收器视图的高度设置为其内容的高度。但这可能会导致滚动效果不佳。
在 ScrollView
中使用 RecyclerView
不是一个很好的做法。您可能会考虑将它们全部放在 NestedScrollView
中,这可能会满足您的目的。这是 documentation from developers android。
在您的情况下,您可以考虑将 linearyLayoutOtherProductDetails
作为 RecyclerView
的 header,这应该是最佳解决方案。删除 ScrollView
或 NestedScrollView
并将自定义布局添加为 header 或 RecyclerView
的页脚。
检查 my answer here 如何在 RecyclerView
中添加 header/footer。
我遇到了同样的问题,最后我发现在使用 UNSPECIFIED 模式的 heightMeasureSpec 而不是零尺寸时,RecyclerView 和 LinearLayoutManager 的测量逻辑可能存在错误。
我已将此错误报告给google issuetracker,您可以查看详细信息。
NestedScrollView 当它的子元素垂直边距不为零时,也可能导致同样的问题,因为它可能将 UNSPECIFIED 模式的 heightMeasureSpec 传递给它的子元素而不是零尺寸,然后子元素可能将类似的 heightMeasureSpec 传递给 RecyclerView。 NestedScrollView相关代码:
@Override
protected void measureChildWithMargins(View child, int parentWidthMeasureSpec, int widthUsed,
int parentHeightMeasureSpec, int heightUsed) {
final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
getPaddingLeft() + getPaddingRight() + lp.leftMargin + lp.rightMargin
+ widthUsed, lp.width);
final int childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(
lp.topMargin + lp.bottomMargin, MeasureSpec.UNSPECIFIED);
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
要修复此错误,您可以简单地使用下面继承的 RecyclerView:
public class FixRecyclerView extends RecyclerView {
public FixRecyclerView(@NonNull Context context) {
super(context);
}
public FixRecyclerView(@NonNull Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public FixRecyclerView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
@Override
protected void onMeasure(int widthSpec, int heightSpec) {
if (MeasureSpec.getMode(heightSpec) == MeasureSpec.UNSPECIFIED && MeasureSpec.getSize(heightSpec) != 0) {
super.onMeasure(widthSpec, MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED));
} else {
super.onMeasure(widthSpec, heightSpec);
}
}
}
我在 ScrollView
中嵌入了 RecyclerView
。这是完整的布局:
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:app="http://schemas.android.com/apk/res-auto"
android:orientation="vertical" android:layout_width="match_parent"
android:layout_height="match_parent">
<ScrollView
android:layout_width="match_parent"
android:layout_height="match_parent"
android:fillViewport="true"
android:orientation="vertical">
<LinearLayout
android:layout_width="match_parent"
android:orientation="vertical"
android:layout_height="wrap_content">
<fragment
android:name="devarshi.sample.view.ProductPortfolioFragment"
android:id="@+id/fragmentProductPortfolio"
android:layout_width="match_parent"
android:layout_margin="10dp"
android:layout_height="230dp" />
<LinearLayout
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:text="TextView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/textViewProductName" />
<TextView
android:text="TextView"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/textViewProductDescription" />
<LinearLayout
android:id="@+id/linearLayoutProductDetails"
android:orientation="vertical"
android:layout_width="match_parent"
android:layout_height="wrap_content">
<TextView
android:text="@string/product_detail_title"
android:layout_width="match_parent"
android:layout_height="wrap_content"
android:id="@+id/textViewProductDetails" />
</LinearLayout>
</LinearLayout>
<LinearLayout
android:orientation="horizontal"
android:layout_width="match_parent"
android:id="@+id/linearyLayoutOtherProductDetails"
android:layout_height="100dp">
<LinearLayout
android:orientation="vertical"
android:layout_width="0dp"
android:layout_weight="0.6"
android:layout_height="wrap_content">
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:id="@+id/linearLayoutPrice"
android:layout_height="wrap_content">
<TextView
android:id="@+id/textViewProductPrice"
android:layout_width="wrap_content"
android:textSize="15sp"
android:textStyle="bold"
android:textColor="@color/colorDiscountedPrice"
android:layout_height="wrap_content"
android:text="1500"/>
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewOldProductPrice"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="200"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:layout_height="wrap_content">
<TextView
android:id="@+id/textViewQuantity"
android:layout_width="wrap_content"
android:textSize="15sp"
android:textStyle="bold"
android:textColor="@color/colorDiscountedPrice"
android:layout_height="wrap_content"
android:text="Quantity"/>
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewQuantityValue"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="200"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:layout_height="wrap_content">
<TextView
android:id="@+id/textViewBrand"
android:layout_width="wrap_content"
android:textSize="15sp"
android:textStyle="bold"
android:textColor="@color/colorDiscountedPrice"
android:layout_height="wrap_content"
android:text="Brand"/>
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewBrandValue"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="200"/>
</LinearLayout>
</LinearLayout>
<LinearLayout
android:orientation="vertical"
android:layout_width="0dp"
android:layout_weight="0.4"
android:layout_height="wrap_content">
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:id="@+id/linearLayoutFreeShipping"
android:layout_height="wrap_content">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:srcCompat="@drawable/product_detail_free_shipping_icon"
android:id="@+id/imageViewFreeShipping" />
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewFreeShipping"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="@string/product_detail_free_shipping"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:id="@+id/linearLayoutCashOnDelivery"
android:layout_height="wrap_content">
<ImageView
android:layout_width="wrap_content"
android:layout_height="wrap_content"
app:srcCompat="@drawable/product_detail_cash_on_delivery_icon"
android:id="@+id/imageViewCashOnDelivery" />
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewCashOnDelivery"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="@string/product_detail_cash_on_delivery"/>
</LinearLayout>
<LinearLayout
android:layout_width="match_parent"
android:orientation="horizontal"
android:id="@+id/linearLayoutUnit"
android:layout_height="wrap_content">
<TextView
android:layout_marginLeft="5dp"
android:id="@+id/textViewUnitLabel"
android:layout_width="wrap_content"
android:textSize="14sp"
android:textColor="@color/colorActualPrice"
android:layout_height="wrap_content"
android:text="@string/product_detail_unit"/>
<Button
android:text="-"
android:layout_width="@dimen/product_detail_counter_button_width"
android:layout_height="@dimen/product_detail_counter_button_height"
android:id="@+id/buttonDecrement" />
<TextView
android:id="@+id/textViewUnitValue"
android:layout_width="27dp"
android:textSize="14sp"
android:gravity="center"
android:textColor="@color/colorActualPrice"
android:layout_height="@dimen/product_detail_counter_button_height"
android:text="0"/>
<Button
android:text="+"
android:layout_width="@dimen/product_detail_counter_button_width"
android:layout_height="@dimen/product_detail_counter_button_height"
android:id="@+id/buttonIncrement" />
</LinearLayout>
</LinearLayout>
</LinearLayout>
<android.support.v7.widget.RecyclerView
android:id="@+id/recyclerViewProductDetails"
android:layout_width="match_parent"
android:layout_height="wrap_content">
</android.support.v7.widget.RecyclerView>
</LinearLayout>
</ScrollView>
</LinearLayout>
问题是 -
虽然我已将 RecyclerView
的 layout_height
设置为 wrap_content
,但它的高度部分包裹了内容,即理想情况下,在 wrap_content
之后,它的高度应该是 150dp
但目前正在 70dp
.
另一个有趣的场景是 -
如果我将 linearyLayoutOtherProductDetails
移动到 recyclerViewProductDetails
以下,那么它会准确地将 RecyclerView
高度显示为 150dp
。似乎当 RecyclerView
加载到布局上时,它正在根据当前在布局上可见的项目(其中)的数量计算高度,即高度等于 5 个项目而不是 9 个项目。
关于如何在第一种情况下让 RecyclerView
的高度完全包裹其内容有什么想法吗?
注:我用的是com.android.support:recyclerview-v7:25.0.0
制作 recyclerview 的 linearlayoutOtherProductdetails header(第一项)并删除 scrollview。因为另一个滚动视图内的滚动视图无法正常工作。 另一种解决方案是使用自定义布局管理器,将回收器视图的高度设置为其内容的高度。但这可能会导致滚动效果不佳。
在 ScrollView
中使用 RecyclerView
不是一个很好的做法。您可能会考虑将它们全部放在 NestedScrollView
中,这可能会满足您的目的。这是 documentation from developers android。
在您的情况下,您可以考虑将 linearyLayoutOtherProductDetails
作为 RecyclerView
的 header,这应该是最佳解决方案。删除 ScrollView
或 NestedScrollView
并将自定义布局添加为 header 或 RecyclerView
的页脚。
检查 my answer here 如何在 RecyclerView
中添加 header/footer。
我遇到了同样的问题,最后我发现在使用 UNSPECIFIED 模式的 heightMeasureSpec 而不是零尺寸时,RecyclerView 和 LinearLayoutManager 的测量逻辑可能存在错误。
我已将此错误报告给google issuetracker,您可以查看详细信息。
NestedScrollView 当它的子元素垂直边距不为零时,也可能导致同样的问题,因为它可能将 UNSPECIFIED 模式的 heightMeasureSpec 传递给它的子元素而不是零尺寸,然后子元素可能将类似的 heightMeasureSpec 传递给 RecyclerView。 NestedScrollView相关代码:
@Override
protected void measureChildWithMargins(View child, int parentWidthMeasureSpec, int widthUsed,
int parentHeightMeasureSpec, int heightUsed) {
final MarginLayoutParams lp = (MarginLayoutParams) child.getLayoutParams();
final int childWidthMeasureSpec = getChildMeasureSpec(parentWidthMeasureSpec,
getPaddingLeft() + getPaddingRight() + lp.leftMargin + lp.rightMargin
+ widthUsed, lp.width);
final int childHeightMeasureSpec = MeasureSpec.makeMeasureSpec(
lp.topMargin + lp.bottomMargin, MeasureSpec.UNSPECIFIED);
child.measure(childWidthMeasureSpec, childHeightMeasureSpec);
}
要修复此错误,您可以简单地使用下面继承的 RecyclerView:
public class FixRecyclerView extends RecyclerView {
public FixRecyclerView(@NonNull Context context) {
super(context);
}
public FixRecyclerView(@NonNull Context context, @Nullable AttributeSet attrs) {
super(context, attrs);
}
public FixRecyclerView(@NonNull Context context, @Nullable AttributeSet attrs, int defStyle) {
super(context, attrs, defStyle);
}
@Override
protected void onMeasure(int widthSpec, int heightSpec) {
if (MeasureSpec.getMode(heightSpec) == MeasureSpec.UNSPECIFIED && MeasureSpec.getSize(heightSpec) != 0) {
super.onMeasure(widthSpec, MeasureSpec.makeMeasureSpec(0, MeasureSpec.UNSPECIFIED));
} else {
super.onMeasure(widthSpec, heightSpec);
}
}
}