Android Recyclerview间距 均分 完美布局 支持任意列数和两端间距

最近项目中要在RecyclerView的Grid中均分间距,看了源码和一些文章,下面的这篇很有启发,在此引用一下,并且在他的基础上支持任意列数和两端间距。最后附上代码:

简介:

  App中,用到最多的设计就是列表形式的布局,而RecyclerView的出现,也完完全全的替代了之前的Listview、GridView,成为android控件中,用途最为广泛的widget之一,今天就来简单介绍一下,RecyclerView的一些布局方法。

实战:

  我们经常在设计稿中看到各式各样的列表,最通常的需求,就是对各个item进行排列,这就运用到了对分割线的处理。在RecyclerView中,有一个public void addItemDecoration(RecyclerView.ItemDecoration decor)的方法,RecyclerView.ItemDecoration这个类里包含了一个getItemOffsets的方法,我们就是要通过这个方法去对每个item分割,通过设置不同的offset来改变间距。以下图这个相册的为例:

 

Android Recyclerview间距 均分 完美布局 支持任意列数和两端间距

  让我们来简单地剖析一下布局,假设屏幕是720px,每个间距为20px,通过计算,每个item的长宽为165px。

Android Recyclerview间距 均分 完美布局 支持任意列数和两端间距

  这时候,很多人会考虑这样去设置

Android Recyclerview间距 均分 完美布局 支持任意列数和两端间距

  我这里是dp转px,720是2x,所以delta是10px,通过parent.getChildAdapterPosition(view)来判断是每一行的第几个,然后第一个item设置了左0右10,第二个左10右10,第三个10右10,第四个左10右0。嗯,理论上来看好像是这样的,动手试试。结果发现,这样的布局并没有到达上图设计的效果。这是为什么呢?

  首先,我们要了解RecyclerView的分割原理,当一个RecyclerView设置了一个GridLayoutManager(this,count),并且count为4的时候,实际上就是将屏幕均分为四份,每一份都是180px宽(以720px为例,我们只考虑左右,暂不考虑上下,原理是相同的),如果不设置ItemDecoration,那么默认item由左开始布置,也就是说,165px的View在它的布局中是这样婶儿的。

Android Recyclerview间距 均分 完美布局 支持任意列数和两端间距

  显而易见,无论我如何设置,item距边界最多就15px(left+right),如果要保证第一个item和第二个item间距为20px,那么我只需要将第二个item设置一个left为5px就可以达到想要的效果。第三第四个也同理。

Android Recyclerview间距 均分 完美布局 支持任意列数和两端间距

  所以正确的左右设置应该为

Android Recyclerview间距 均分 完美布局 支持任意列数和两端间距

 

在此基础上进行改进,支持任意列数和两端间距:

核心代码:

    @Override
    public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
        if (parent.getLayoutManager() instanceof GridLayoutManager) {
            GridLayoutManager layoutManager = (GridLayoutManager) parent.getLayoutManager();
            int spanCount = layoutManager.getSpanCount();
            int count = parent.getAdapter().getItemCount();
            int position = parent.getChildAdapterPosition(view);
            if (gapHSizePx < 0 || gapVSizePx < 0) {
                DisplayMetrics displayMetrics = new DisplayMetrics();
                parent.getDisplay().getMetrics(displayMetrics);
                gapHSizePx = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, gapHorizontalDp, displayMetrics);
                gapVSizePx = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, gapVerticalDp, displayMetrics);
                edgePaddingPx = (int) TypedValue.applyDimension(TypedValue.COMPLEX_UNIT_DIP, edgePaddingDp, displayMetrics);
                eachItemPaddingH = (edgePaddingPx * 2 + gapHSizePx * (spanCount - 1)) / spanCount;
            }
            outRect.top = gapVSizePx;
            outRect.bottom = 0;
            int visualPos = position + 1;
            if (visualPos % spanCount == 1) {
                //第一列
                outRect.left = edgePaddingPx;
                outRect.right = eachItemPaddingH - edgePaddingPx;
            } else if (visualPos % spanCount == 0) {
                //最后一列
                outRect.left = eachItemPaddingH - edgePaddingPx;
                outRect.right = edgePaddingPx;
            } else {
                outRect.left = gapHSizePx - preRect.right;
                outRect.right = eachItemPaddingH - outRect.left;
            }
            if (visualPos - spanCount <= 0) {
                //第一行
                outRect.top = 0;
            } else if (isLastRow(visualPos, spanCount, count)) {
                //最后一行
            }
            preRect = new Rect(outRect);
//            Log.w("GridAverageGapItem", "pos=" + position + "," + outRect.toShortString());
        } else {
            super.getItemOffsets(outRect, view, parent, state);
        }
    }

    private boolean isLastRow(int visualPos, int spanCount, int sectionItemCount) {
        int lastRowCount = sectionItemCount % spanCount;
        lastRowCount = lastRowCount == 0 ? spanCount : lastRowCount;
        return visualPos > sectionItemCount - lastRowCount;
    }

下载地址:GridAverageGapItemDecoration

另外如果项目中用到了Brvah,并且想自定义Section中的间距的话,看这篇:

BRVAH的Section中自定义间距,条目完美均分,支持Section顶部和底部间距