RecyclerView分组显示可实现粘性头部,并且在滑动过程中头部字体颜色产生渐变

废话不多说,先上图,左边图是一个RecyclerView就可实现,右边图是RecyclerView(LinearLayouManager)嵌套RecyclerView(GridLayouManager)来实现

          

刚开始我是先实现的粘性头部,但是我看到微信(IOS)的联系人列表在滑动的过程中头部的字体颜色会产生渐变,然后就也想实现这种效果(主要近段时间没啥任务,闲),经过一番努力之后,终于ojbk,出来了我想要的效果。

下面说说我实现的流程

一、绘制头部

这个头部其实就是我们经常给RecyclerView设置的分割线ItemDecoration,然后在绘制分割线的时,我们需要判断当前条目是否为分组内的第一条,如果是则需绘制头部(包括绘制背景和绘制字体),如果不是,则绘制一条普通的线即可。

1.实现分组

既然是分组显示,就需要对数据源进行改造,以第一副图为例,就是一个线性布局,只不过根据时间进行了分组,如果网络请求回来得数据源为List<Bean>,通过改造之后变为List<List<Bean>>,之所以要改成这种格式,是因为我需要定位每组的 头部 以及 尾部 ,这样我就知道哪一条需要绘制头部,并且在每组内最后一条(尾部)即将滑出去的时候不断改变头部的top值实现一种被下一个头部顶出去的效果。当适配器设置数据源的时候,还是以List<Bean>的方式去设置。

List<String> strList = new ArrayList<>();//给适配器要设置的数据源
List<GroupInfo> groupList = new ArrayList<>();//与strList长度相等,用于判断当前条目是否为头部/尾部
List<List<String>> mDatas = new ArrayList<>();//改造后的数据源

/*mData为我改造后的数据源,里面已有数据
* 通过双层for循环,就可以设置每组的头部以及尾部
* 用List<GroupInfo> groupList来接收并且得到适配器的数据源
*/
for (int i = 0; i < mDatas.size(); i++) {
            List<String> list = mDatas.get(i);
            for (int j = 0; j < list.size(); j++) {
                GroupInfo groupInfo = new GroupInfo(i, "第" + (i) + "组");
                if (j == 0) {
                    groupInfo.setHeader(true);
                } else {
                    groupInfo.setHeader(false);
                }
                if (j == list.size() - 1) {
                    groupInfo.setFooter(true);
                } else {
                    groupInfo.setFooter(false);
                }
                groupList.add(groupInfo);
                strList.add(mDatas.get(i).get(j));
            }
}

GroupInfo.java代码

public class GroupInfo {

    private String title;
    private boolean isHeader = false;
    private boolean isFooter = false;

    public String getTitle() {
        return title;
    }

    public void setTitle(String title) {
        this.title = title;
    }

    public boolean isHeader() {
        return isHeader;
    }

    public void setHeader(boolean header) {
        isHeader = header;
    }

    public boolean isFooter() {
        return isFooter;
    }

    public void setFooter(boolean footer) {
        isFooter = footer;
    }

}

2.实现自定义的ItemDecoration

继承RecyclerView的ItemDecoration,重写里面的 getItemOffsets()和onDrawOver()方法,onDraw和onDrawOver的区别在于前者

onDraw在绘制ItemView之前绘制,onDrawOver会在绘制ItemView之后绘制。
 

private GroupInfoCallBack groupInfoCallBack;

public interface GroupInfoCallBack {
     GroupInfo getGroupInfo(int position);
}

@Override
public void getItemOffsets(Rect outRect, View view, RecyclerView parent, RecyclerView.State state) {
        super.getItemOffsets(outRect, view, parent, state);
//通过view得到在适配器中的索引,再通过接口得到该索引所对应的GroupInfo,
//就可以判断当前条目是否为组内的头部/尾部,然后设置指定条目的偏移亮,就是给条目一定的装饰空间
        int position = parent.getChildAdapterPosition(view);
        if (groupInfoCallBack != null) {
            GroupInfo groupInfo = groupInfoCallBack.getGroupInfo(position);
            if (groupInfo != null && groupInfo.isHeader()) {
                outRect.top = headHeight;
            } else {
                outRect.top = divideHeight;
            }
        }
}


@Override
    public void onDrawOver(Canvas c, RecyclerView parent, RecyclerView.State state) {
        super.onDrawOver(c, parent, state);
        //parent为RecyclerView可视区域的
  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值