RecyclerView的自定义间隔

自定义间隔样式需要 继承RecyclerView.ItemDecoration类,该类是个抽象类,主要有三个方法。
onDraw(Canvas c, RecyclerView parent, State state),在Item绘制之前被调用,该方法主要用于绘制间隔样式
onDrawOver(Canvas c, RecyclerView parent, State state),在Item绘制之前被调用,该方法主要用于绘制间隔样式
getItemOffsets(Rect outRect, View view, RecyclerView parent, State state),设置item的偏移量,偏移的部分用于填充间隔样式,在RecyclerView的onMesure()中会调用该方法。

onDraw()和onDrawOver()这两个方法都是用于绘制间隔样式,我们只需要复写其中一个方法即可。

添加分割线(线性布局)

import android.content.Context;
import android.content.res.TypedArray;
import android.graphics.Canvas;
import android.graphics.Rect;
import android.graphics.drawable.Drawable;
import android.support.v4.view.ViewCompat;
import android.support.v7.widget.LinearLayoutManager;
import android.support.v7.widget.RecyclerView;
import android.view.View;


/**
 * Created by Administrator on 2018/3/15.
 */


public class MyDividerItemDecoration extends RecyclerView.ItemDecoration {


    private static final int[] ATTRS = new int[]{
            android.R.attr.listDivider
    };
    public static final int HORIZONTAL_LIST = LinearLayoutManager.HORIZONTAL;
    public static final int VERTICAL_LIST = LinearLayoutManager.VERTICAL;
    /**
     * 用于绘制间隔样式
     */
    private Drawable mDivider;
    /**
     * 列表的方向,水平/竖直
     */
    private int mOrientation;




    public MyDividerItemDecoration(Context context, int orientation) {
        // 获取默认主题的属性
        final TypedArray a = context.obtainStyledAttributes(ATTRS);
        mDivider = a.getDrawable(0);
        a.recycle();
        setOrientation(orientation);
    }


    @Override
    public void onDraw(Canvas c, RecyclerView parent, RecyclerView.State state) {
        // 绘制间隔
        if (mOrientation == VERTICAL_LIST) {
            drawVertical(c, parent);
        } else {
            drawHorizontal(c, parent);
        }
    }


    @Override
    public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
        if (mOrientation == VERTICAL_LIST) {
            outRect.set(0, 0, 0, mDivider.getIntrinsicHeight());
        } else {
            outRect.set(0, 0, mDivider.getIntrinsicWidth(), 0);
        }
    }


    private void setOrientation(int orientation) {
        if (orientation != HORIZONTAL_LIST && orientation != VERTICAL_LIST) {
            throw new IllegalArgumentException("invalid orientation");
        }
        mOrientation = orientation;
    }


    /**
     * 绘制间隔
     */
    private void drawVertical(Canvas c, RecyclerView parent) {
        final int left = parent.getPaddingLeft();
        final int right = parent.getWidth() - parent.getPaddingRight();
        final int childCount = parent.getChildCount();
        for (int i = 0; i < childCount; i++) {
            final View child = parent.getChildAt(i);
            final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child
                    .getLayoutParams();
            final int top = child.getBottom() + params.bottomMargin +
                    Math.round(ViewCompat.getTranslationY(child));
            final int bottom = top + mDivider.getIntrinsicHeight();
            mDivider.setBounds(left, top, right, bottom);
            mDivider.draw(c);
        }
    }


    /**
     * 绘制间隔
     */
    private void drawHorizontal(Canvas c, RecyclerView parent) {
        final int top = parent.getPaddingTop();
        final int bottom = parent.getHeight() - parent.getPaddingBottom();
        final int childCount = parent.getChildCount();
        for (int i = 0; i < childCount; i++) {
            final View child = parent.getChildAt(i);
            final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child
                    .getLayoutParams();
            final int left = child.getRight() + params.rightMargin +
                    Math.round(ViewCompat.getTranslationX(child));
            final int right = left + mDivider.getIntrinsicHeight();
            mDivider.setBounds(left, top, right, bottom);
            mDivider.draw(c);
        }
    }
}

在这里我们采用系统主题(android.R.attr.listDivider)来设置成分隔线的,然后来获取尺寸,位置进行setBound(),绘制,接着通过outRect.set()来设置绘制整个区域范围。

改变分隔线样式

  <style name ="AppTheme" parent="AppBaseTheme">  
        <!-- All customizations that are NOT specific to a particular API-level can go here. -->  
        <item name= "android:listDivider">@drawable/divider </item >   

    </style >  

divider的内容如下

<shape xmlns:android="http://schemas.android.com/apk/res/android"  
    android:shape="rectangle" >  
        <!-- 填充的颜色 -->  
       <solid android:color ="@color/color_red"/>  
      <!--  线条大小 -->  
      <size android:height ="1dp" android:width ="1dp"/>  
</shape>  

添加分割线(表格布局和瀑布流布局)

public class DividerGridItemDecoration extends RecyclerView.ItemDecoration  
{  
     private static final int[] ATTRS = new int[] { android.R.attr.listDivider };  
     private Drawable mDivider;  
 
     public DividerGridItemDecoration(Context context)  
     {  
            final TypedArray a = context.obtainStyledAttributes(ATTRS );  
            mDivider = a.getDrawable(0);  
           a.recycle();  
     }  
  
     @Override  
     public void onDraw(Canvas c, RecyclerView parent, State state)  
     {  
  
           drawHorizontal(c, parent);  
           drawVertical(c, parent);  
  
     }  
  
     private int getSpanCount(RecyclerView parent)  
     {  
            // 列数  
            int spanCount = -1;  
           LayoutManager layoutManager = parent.getLayoutManager();  
            if (layoutManager instanceof GridLayoutManager)  
           {  
  
                spanCount = ((GridLayoutManager) layoutManager).getSpanCount();  
           } else if (layoutManager instanceof StaggeredGridLayoutManager)  
           {  
                spanCount = ((StaggeredGridLayoutManager) layoutManager)  
                           .getSpanCount();  
           }  
            return spanCount;  
     }  
  
     public void drawHorizontal(Canvas c, RecyclerView parent)  
     {  
            int childCount = parent.getChildCount();  
            for ( int i = 0; i < childCount; i++)  
           {  
                 final View child = parent.getChildAt(i);  
                 final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child  
                           .getLayoutParams();  
                 final int left = child.getLeft() - params.leftMargin;  
                 final int right = child.getRight() + params.rightMargin  
                           + mDivider.getIntrinsicWidth();  
                 final int top = child.getBottom() + params.bottomMargin;  
                 final int bottom = top + mDivider.getIntrinsicHeight();  
                 mDivider.setBounds(left, top, right, bottom);  
                 mDivider.draw(c);  
           }  
     }  
  
     public void drawVertical(Canvas c, RecyclerView parent)  
     {  
            final int childCount = parent.getChildCount();  
            for ( int i = 0; i < childCount; i++)  
           {  
                 final View child = parent.getChildAt(i);  
  
                 final RecyclerView.LayoutParams params = (RecyclerView.LayoutParams) child  
                           .getLayoutParams();  
                 final int top = child.getTop() - params.topMargin;  
                 final int bottom = child.getBottom() + params.bottomMargin;  
                 final int left = child.getRight() + params.rightMargin;  
                 final int right = left + mDivider.getIntrinsicWidth();  
  
                 mDivider.setBounds(left, top, right, bottom);  
                 mDivider.draw(c);  
           }  
     }  
  
     private boolean isLastColum(RecyclerView parent, int pos, int spanCount,  
                 int childCount)  
     {  
           LayoutManager layoutManager = parent.getLayoutManager();  
            if (layoutManager instanceof GridLayoutManager)  
           {  
                 if ((pos + 1) % spanCount == 0) // 如果是最后一列,则不需要绘制右边  
                {  
                      return true;  
                }  
           } else if (layoutManager instanceof StaggeredGridLayoutManager)  
           {  
                 int orientation = ((StaggeredGridLayoutManager) layoutManager)  
                           .getOrientation();  
                 if (orientation == StaggeredGridLayoutManager.VERTICAL )  
                {  
                      if ((pos + 1) % spanCount == 0) // 如果是最后一列,则不需要绘制右边  
                     {  
                            return true;  
                     }  
                } else  
                {  
                     childCount = childCount - childCount % spanCount;  
                      if (pos >= childCount) // 如果是最后一列,则不需要绘制右边  
                            return true;  
                }  
           }  
            return false;  
     }  
  
     private boolean isLastRaw(RecyclerView parent, int pos, int spanCount,  
                 int childCount)  
     {  
           LayoutManager layoutManager = parent.getLayoutManager();  
            if (layoutManager instanceof GridLayoutManager)  
           {  
                childCount = childCount - childCount % spanCount;  
                 if (pos >= childCount) // 如果是最后一行,则不需要绘制底部  
                      return true;  
           } else if (layoutManager instanceof StaggeredGridLayoutManager)  
           {  
                 int orientation = ((StaggeredGridLayoutManager) layoutManager)  
                           .getOrientation();  
                 // StaggeredGridLayoutManager 且纵向滚动  
                 if (orientation == StaggeredGridLayoutManager.VERTICAL )  
                {  
                     childCount = childCount - childCount % spanCount;  
                      // 如果是最后一行,则不需要绘制底部  
                      if (pos >= childCount)  
                            return true;  
                } else  
                 // StaggeredGridLayoutManager 且横向滚动  
                {  
                      // 如果是最后一行,则不需要绘制底部  
                      if ((pos + 1) % spanCount == 0)  
                     {  
                            return true;  
                     }  
                }  
           }  
            return false;  
     }  
  
     @Override  
     public void getItemOffsets(Rect outRect, int itemPosition,  
                RecyclerView parent)  
     {  
            int spanCount = getSpanCount(parent);  
            int childCount = parent.getAdapter().getItemCount();  
            if (isLastRaw(parent, itemPosition, spanCount, childCount))// 如果是最后一行,则不需要绘制底部  
           {  
                outRect.set(0, 0, mDivider.getIntrinsicWidth(), 0);  
           } else if (isLastColum(parent, itemPosition, spanCount, childCount))// 如果是最后一列,则不需要绘制右边  
           {  
                outRect.set(0, 0, 0, mDivider.getIntrinsicHeight());  
           } else  
           {  
                outRect.set(0, 0, mDivider.getIntrinsicWidth(),  
                            mDivider.getIntrinsicHeight());  
           }  
     }  
}  

猜你喜欢

转载自blog.csdn.net/owen_bland/article/details/79574255