欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

Android实战RecyclerView头部尾部添加方法示例

程序员文章站 2023-11-27 12:50:46
最近开启sdk manager,突然发现android7.0的都有了,这迭代升级还真快。不过国内普遍手机还是停留在4.4+,多则是是处于5.0版本的。android5.0变...

最近开启sdk manager,突然发现android7.0的都有了,这迭代升级还真快。不过国内普遍手机还是停留在4.4+,多则是是处于5.0版本的。android5.0变化非常大,引入material design,加强权限管理、减少功耗...好像扯远了0 0。现在直接进入主题。在这里先感谢读者的支持!!

listview是有addheaderview和 addfooterview两个方法的.

但是作为官方推荐的listview的升级版recyclerview缺无法实现这两个方法。

那么如果使用recyclerview实现这两个方法的效果该怎么做呢?

网上查询了很久,试过各种各样的实现方式,终于让我发现一个还不错的实现方法,那么就给大家推荐一下。

笔者前阵子写了一个万能适配器,提供了上拉加载、上拉刷新的基础功能,重要的是一个基础baseadapter能够支持listview与recyclerview,后期提供传送门,现在我打算一步骤一步骤讲下我的实现思路。
实战recyclerview头部尾部添加方法

效果图如下:

Android实战RecyclerView头部尾部添加方法示例

Android实战RecyclerView头部尾部添加方法示例

一、前提

首先listview与recyclerview两者非常相似,两者提供view都是依赖适配器。只不过就是5.0版本推出recyclerview后,google将adapter和viewholder做了一系列的优化和封装。不像之前为了复用listview里面的converview,要类似在getview里面实现下列的代码:

Android实战RecyclerView头部尾部添加方法示例

上面代码看起来挺眼熟吧~

二、对比recyclerview,google进行的优化

在recyclerview依赖的适配器中,无论是适配器还是viewholder,从源码我们可以看出,都存在recyclerview的匿名内部类。相对于listview,recyclerview内置了多级缓存、recyclerviewpool(从线程的角度,可以理解成类似线程池的东西,即多个recyclerview可以公用一个view)、viewholder(已经实现了复用,相对于listview的baseadapter中getview方法需要开发者自己引入复用问题方便很多)等等。这里我们简单说下两个方法:

public viewholder oncreateviewholder(viewgroup parent, int viewtype)
public void onbindviewholder(viewholder holder, int position)

在以前的baseadapter中,所有视图加载、数据绑定以及复用,都需要我们直接在getview里面进行操作。oncreateviewholder负责视图加载并且内部完成复用,onbindviewholder负责数据绑定并且内部完成一系列的缓存机制。这里满足了视图层与逻辑层的分离,典型的mvp模式。

三、recyclerview的头部与尾部实现

recyclerview不像listview拥有addheaderview()与addfooterview()的方法简单添加头部尾部即可,而且recyclerview也没有像listview的列表点击监听方法(setitemonclicklistener),这里我也不明白为什么官方会取消了这些独有的属性,不过我们依然可以在onbindviewholder方法中进行事件绑定!

具体头部与尾部实现方法,这里有个诀窍,这里先看一个方法:

public int getitemviewtype(int position)

getitemviewtype方法是在执行oncreateviewholder(viewgroup parent, int viewtype)前回调用viewtype,目的是为了根据viewtype不同创建不同的视图。我们可以通过在oncreateviewholder创建视图的时候,对viewtype进行判断,如果添加了头部,在position = 0的时候回调头部的viewtype给oncreateviewholder,从而创建头部。尾部创建方法于此类同,直接看下代码,适配器的实现:

package cn.wsy.recyclerdemo;

import android.content.context;
import android.support.v7.widget.gridlayoutmanager;
import android.support.v7.widget.recyclerview;
import android.view.layoutinflater;
import android.view.view;
import android.view.viewgroup;
import android.widget.textview;

import java.util.arraylist;
import java.util.list;

/**
 * created by wsy on 2016/8/4.
 */
public class myadapter extends recyclerview.adapter<myadapter.myholder> {

  private recyclerview mrecyclerview;

  private list<string> data = new arraylist<>();
  private context mcontext;

  private view view_footer;
  private view view_header;

  //type
  private int type_normal = 1000;
  private int type_header = 1001;
  private int type_footer = 1002;

  public myadapter(list<string> data, context mcontext) {
    this.data = data;
    this.mcontext = mcontext;
  }

  @override
  public myadapter.myholder oncreateviewholder(viewgroup parent, int viewtype) {
    if (viewtype == type_footer) {
      return new myholder(view_footer);
    } else if (viewtype == type_header) {
      return new myholder(view_header);
    } else {
      return new myholder(getlayout(r.layout.item_list_layout));
    }
  }

  @override
  public void onbindviewholder(myholder holder, int position) {
    if (!isheaderview(position) && !isfooterview(position)) {
      if (haveheaderview()) position--;
      textview content = (textview) holder.itemview.findviewbyid(r.id.item_content);
      textview time = (textview) holder.itemview.findviewbyid(r.id.item_time);
      content.settext(data.get(position));
      time.settext("2016-1-1");
    }
  }

  @override
  public int getitemcount() {
    int count = (data == null ? 0 : data.size());
    if (view_footer != null) {
      count++;
    }

    if (view_header != null) {
      count++;
    }
    return count;
  }

  @override
  public int getitemviewtype(int position) {
    if (isheaderview(position)) {
      return type_header;
    } else if (isfooterview(position)) {
      return type_footer;
    } else {
      return type_normal;
    }
  }

  @override
  public void onattachedtorecyclerview(recyclerview recyclerview) {
    try {
      if (mrecyclerview == null && mrecyclerview != recyclerview) {
        mrecyclerview = recyclerview;
      }
      ifgridlayoutmanager();
    } catch (exception e) {
      e.printstacktrace();
    }
  }

  private view getlayout(int layoutid) {
    return layoutinflater.from(mcontext).inflate(layoutid, null);
  }

  public void addheaderview(view headerview) {
    if (haveheaderview()) {
      throw new illegalstateexception("hearview has already exists!");
    } else {
      //避免出现宽度自适应
      viewgroup.layoutparams params = new viewgroup.layoutparams(viewgroup.layoutparams.match_parent, viewgroup.layoutparams.wrap_content);
      headerview.setlayoutparams(params);
      view_header = headerview;
      ifgridlayoutmanager();
      notifyiteminserted(0);
    }

  }

  public void addfooterview(view footerview) {
    if (havefooterview()) {
      throw new illegalstateexception("footerview has already exists!");
    } else {
      viewgroup.layoutparams params = new viewgroup.layoutparams(viewgroup.layoutparams.match_parent, viewgroup.layoutparams.wrap_content);
      footerview.setlayoutparams(params);
      view_footer = footerview;
      ifgridlayoutmanager();
      notifyiteminserted(getitemcount() - 1);
    }
  }

  private void ifgridlayoutmanager() {
    if (mrecyclerview == null) return;
    final recyclerview.layoutmanager layoutmanager = mrecyclerview.getlayoutmanager();
    if (layoutmanager instanceof gridlayoutmanager) {
      final gridlayoutmanager.spansizelookup originalspansizelookup =
          ((gridlayoutmanager) layoutmanager).getspansizelookup();
      ((gridlayoutmanager) layoutmanager).setspansizelookup(new gridlayoutmanager.spansizelookup() {
        @override
        public int getspansize(int position) {
          return (isheaderview(position) || isfooterview(position)) ?
              ((gridlayoutmanager) layoutmanager).getspancount() :
              1;
        }
      });
    }
  }

  private boolean haveheaderview() {
    return view_header != null;
  }

  public boolean havefooterview() {
    return view_footer != null;
  }

  private boolean isheaderview(int position) {
    return haveheaderview() && position == 0;
  }

  private boolean isfooterview(int position) {
    return havefooterview() && position == getitemcount() - 1;
  }


  public static class myholder extends recyclerview.viewholder {

    public myholder(view itemview) {
      super(itemview);
    }
  }

}

四、实现方法

简单的初始化recycerview,以及设置适配器,如下:

  private void initrecyc() {
//    mrecyclerview.setlayoutmanager(new gridlayoutmanager(this,2));
    mrecyclerview.setlayoutmanager(new linearlayoutmanager(this));

    adapter = new myadapter(data, this);
    mrecyclerview.setadapter(adapter);

    adapter.addfooterview(layoutinflater.from(this).inflate(r.layout.item_footer_layout,null));
    adapter.addheaderview(layoutinflater.from(this).inflate(r.layout.item_header_layout,null));
  }

五、注意的问题

笔者在添加头部尾部的时候,发现在配置recyclerview,如果模式是配置gridlayoutmanager的时候,发现头部会跑到第一格,也就是不是自己想要独立一行的效果,这里贴上关键代码,可以解决(简单数学问题啦哈~):

  private void ifgridlayoutmanager() {
    if (mrecyclerview == null) return;
    final recyclerview.layoutmanager layoutmanager = mrecyclerview.getlayoutmanager();
    if (layoutmanager instanceof gridlayoutmanager) {
      final gridlayoutmanager.spansizelookup originalspansizelookup =
          ((gridlayoutmanager) layoutmanager).getspansizelookup();
      ((gridlayoutmanager) layoutmanager).setspansizelookup(new gridlayoutmanager.spansizelookup() {
        @override
        public int getspansize(int position) {
          return (isheaderview(position) || isfooterview(position)) ?
              ((gridlayoutmanager) layoutmanager).getspancount() :
              1;
        }
      });
    }
  }

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。