RecyclerView重新计算WRAP_CONTENT

时间:2016-03-15 19:56:14

标签: android android-recyclerview

对于版本23.2,我们现在可以将WRAP_CONTENT用于recyclerView高度,这很棒。我这样做,但是我想在将项目添加(或删除)到适配器后重新计算高度(从而增加或减少高度)。

我的特定RecyclerView以1项开头,然后在用户选择时添加项。所以我需要RecyclerView布局来增加高度,直到某一点。理想情况下,当列表增加或减少时,平滑动画会发生这种情况。

我们如何在布局后WRAP_CONTENT制作它?

尝试:

recyclerview.requestLayout();

recyclerview.invalidate();

4 个答案:

答案 0 :(得分:5)

我希望它可以与View.invalidate()一起使用。

如果这不起作用,请尝试在父视图上调用requestLayoutinvalidate

答案 1 :(得分:4)

RecyclerView如何根据新的LayoutManger

调整自身大小

RecyclerView小部件为创建列表和网格以及支持动画提供了一个先进而灵活的基础。此版本为LayoutManager API带来了令人兴奋的新功能:自动测量!这允许RecyclerView根据其内容的大小调整自身大小。这意味着现在可以使用以前不可用的方案,例如使用WRAP_CONTENT作为RecyclerView的维度。您会发现所有内置的LayoutManagers现在都支持自动测量。

由于此更改,请务必仔细检查项目视图的布局参数:现在将完全遵循先前忽略的布局参数(例如滚动方向中的MATCH_PARENT)。

如果你有一个自定义的LayoutManager没有扩展其中一个内置的LayoutManagers,这是一个选择加入的API - 你需要调用setAutoMeasureEnabled(true)以及进行一些细微的更改,详见方法的Javadoc。

请注意,尽管RecyclerView会为其子项设置动画,但它不会为自己的边界更改设置动画。如果您希望在RecyclerView边界更改时为其设置动画,则可以使用Transition API。

请阅读this

答案 2 :(得分:2)

选项I

你看到了answer吗?

它没有使用recyclerView的WRAP_CONTENT,但它可能有效。

您也可以使用链接中的layoutManager创建自己的自定义recyclerView(扩展RecyclerView)并覆盖onMeasure()方法。

选项II

尝试在绘制视图之前设置布局参数。我还没有检查它是否在recyclerView布局更改时被调用,但如果确实如此,那么它将起作用。这样的事情(在您的活动/片段onCreate() / onCreateView()方法中:

recyclerView.getViewTreeObserver().addOnPreDrawListener(new ViewTreeObserver.OnPreDrawListener() {
            @Override
            public boolean onPreDraw() { 
                recyclerView.getViewTreeObserver().removeOnPreDrawListener(this);

                YourParentLayoutType.LayoutParams params = (YourParentLayoutType.LayoutParams) recyclerView.getLayoutParams();
                params.height = YourParentLayoutType.LayoutParams.WRAP_CONTENT;
                recyclerView.setLayoutParams(params);

                return true;
            }
        });

在代码中使用recyclerView的父布局类型而不是YourParentLayoutType。 我不确定这会在布局刷新时起作用,但也许值得一试。

答案 3 :(得分:0)

使用此课程:

请使用23.2.1,因为23.2是错误的。

你是否尝试在notifyDataSetChanged适配器上调用recyclerview,如果你已将wrap_content指定为recyclerview的高度

,我认为它应该可以毫无问题地扩展

否则你可以使用这个类:

import android.content.Context;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.view.View;
import android.view.ViewGroup;

public class MyLinearLayoutManager extends LinearLayoutManager {

    public MyLinearLayoutManager(Context context, int orientation, boolean reverseLayout)    {
        super(context, orientation, reverseLayout);

    }



    private int[] mMeasuredDimension = new int[2];


    @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:
        }

        int widthDesired = Math.min(widthSize,width);
        setMeasuredDimension(widthDesired, height);
    }

    private void measureScrapChild(RecyclerView.Recycler recycler, int position, int widthSpec,
                                   int heightSpec, int[] measuredDimension) {
        View view = recycler.getViewForPosition(position);
        // 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);
        }
    }