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

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

ListView是有addHeaderView和 addFooterView两个方法的.

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

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

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

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

效果图如下:

一、前提

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

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

二、对比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;
        }
      });
    }
  }

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

(0)

相关推荐

  • Android中封装RecyclerView实现添加头部和底部示例代码

    前言 我们大家都知道ListView具有添加头部和添加底部的方法,但是RecyclerView并没有这样子的方法.所以RecyclerView是不能添加底部和头部的,但是能不能仿造ListView来实现RecyclerView添加头部和底部呢?答案当然是可行的.本文就来给大家介绍了关于Android封装RecyclerView添加头部和底部的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 首先看下实现的效果: 代码如下: public class WrapMyRecy

  • Android RecyclerView添加头部和底部的方法

    如果只是想添加头部,可是使用GitHub里面这个项目,它可以为LinearLayoutManager,GridLayoutManager ,StaggeredGridLayoutManager布局的RecyclerView添加header.使用起来也十分简单: 只需将RecyclerViewHeader布局放在RecyclerView的上层. <FrameLayout android:layout_width="match_parent" android:layout_heigh

  • Android RecyclerView添加头部和底部实例详解

    Android RecyclerView添加头部和底部实例详解 如果只是想添加头部,可是使用GitHub里面这个项目,它可以为LinearLayoutManager,GridLayoutManager ,StaggeredGridLayoutManager布局的RecyclerView添加header.使用起来也十分简单: 只需将RecyclerViewHeader布局放在RecyclerView的上层. <FrameLayout android:layout_width="match_p

  • Android实现RecyclerView添加分割线的简便方法

    1.前言 刚开始学习RecyclerView的时候我跟着一个视频学的,当时添加分割线是从外面导入一个Java类,然后使用里面的函数来创建分割线的,所以一直以来我都是这样做的.直到前几天才无意中发现,原来v7包中提供了一个DividerItemDecoration类,利用它,我们可以很简单地实现RecyclerView的分割线!那还等什么呢?赶紧用起来. 2.创建一个简单的RecyclerView 首先当然应该来一个RecyclerView,这里就不再赘述了,随便造点数据就好.直接上代码: pub

  • Recyclerview添加头布局和尾布局、item点击事件详解

    简介: 本篇博客主要包括recyclerview添加多种布局以及添加头布局和尾布局,还有item点击事件 思路: 主要重写Recyclerview.Adapter中的一些方法 1.public int getItemCount()  item熟练  +2(头布局和尾布局) 2.public int getItemViewType(int position)   判断position 设置itemType 3.创建不同的ViewHolder,分别用来加载头布局,正常布局,尾布局 4.public

  • 使用RecyclerView添加Header和Footer的方法

    RecyclerView与ListView原理是类似的:都是仅仅维护少量的View并且可以展示大量的数据集.RecyclerView用以下两种方式简化了数据的展示和处理: 使用LayoutManager来确定每一个item的排列方式. 为增加和删除项目提供默认的动画效果. RecyclerView虽然作为ListView的替代者有着较好的性能提升,但是ListView的一些常用功能却没有提供,比如我们平时会经常用到的addHeaderView,addFooterView,既然RecyclerVi

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

    最近开启SDK Manager,突然发现android7.0的都有了,这迭代升级还真快.不过国内普遍手机还是停留在4.4+,多则是是处于5.0版本的.Android5.0变化非常大,引入material design,加强权限管理.减少功耗...好像扯远了0 0.现在直接进入主题.在这里先感谢读者的支持!! ListView是有addHeaderView和 addFooterView两个方法的. 但是作为官方推荐的ListView的升级版RecyclerView缺无法实现这两个方法. 那么如果使

  • Android中RecyclerView实现Item添加和删除的代码示例

    本文介绍了Android中RecyclerView实现Item添加和删除的代码示例,分享给大家,具体如下: 先上效果图: RecyclerView简介: RecyclerView用以下两种方式简化了数据的展示和处理: 1. 使用LayoutManager来确定每一个item的排列方式. 2. 为增加和删除项目提供默认的动画效果,也可以自定义. RecyclerView项目结构如下: Adapter:使用RecyclerView之前,你需要一个继承自RecyclerView.Adapter的适配器

  • Android用RecyclerView实现动态添加本地图片

    本文介绍了Android用RecyclerView实现动态添加本地图片,分享给大家,具体如下: 本文所用的多图选择的library来自:https://github.com/lovetuzitong/MultiImageSelector 简单介绍一下用法: 1.跳转到图片选择页面: Intent intent = new Intent(PassengerSetActivity.this, MultiImageSelectorActivity.class); intent.putExtra(Mul

  • Android轻松实现多语言的方法示例

    本文介绍了Android轻松实现多语言的方法示例,分享给大家,具体如下: 1.创建多语言包 2.首先在onCreate方法中调用此方法查看上一次保存的是什么语言 public void setLanguage() { //根据读取到存放在sp里面的数据 进行设置 Configuration configuration = getResources().getConfiguration(); SharedPreferences sharedPreferences = getSharedPrefer

  • Android P实现静默安装的方法示例(官方Demo)

    Android9.0无法通过以下两种方式实现静默安装: 1.runtime执行shell cmd 2.PackageInstall 反射机制 但是Google已经给我们推荐了相关的APIDemos,所以建议大家多看看源码~ 在frameworks/base/core/java/android/content/pm/PackageInstaller.java有段关于该类的介绍: The ApiDemos project contains examples of using this API: <c

  • Android使用ViewStub实现布局优化方法示例

    目录 实践过程 实现方式 知识点 实践过程 Hello,大家好啊,我是小空,今天带大家了解下动态加载控件ViewStub. 在平时开发中经常会遇到复杂布局,而每一个view都是会占据内存和消耗cpu的(即使再小,累计成多,一般嵌套7级以上就有明显的卡顿了),布局优化就是我们常做的任务之一,甚至是一块心病.所以我们工作中就要留意布局优化的手段,ViewStub就是其中之一. 大家应该听过merge标签,将某个布局文件的根布局写成merge的,然后对应的布局include引用,会默认不会引入merg

  • Android编程操作手机通讯录的方法示例

    本文实例讲述了Android编程操作手机通讯录的方法.分享给大家供大家参考,具体如下: 手机通讯录的操作是经常被用到的,例如添加联系人,删除联系人或者取得联系人信息.类似的操作还有收藏夹的操作,下面就针对通讯录的操作来做个小例子.同样的这次也会使用到内容提供者的知识. 1. 要操作通信录就要得到授权,也就是读或者写通讯录的权力.这里也需要使用Junit <?xml version="1.0" encoding="utf-8"?> <manifest

  • Android实现菜单关联activity的方法示例

    本文实例讲述了Android实现菜单关联activity的方法.分享给大家供大家参考,具体如下: 简介: 有时,程序需要单击某个个菜单来实现启动ing其他的activity(或者service) 这时就要通过 setIntent()方法将其与Intent关联在一起 实现效果: 具体实现方法: public class MainActivity extends AppCompatActivity { @Override protected void onCreate(Bundle savedIns

  • Flutter调用Android和iOS原生代码的方法示例

    前言 本文主要给大家介绍了关于Flutter调用Android和iOS原生代码的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧 分3个大步骤: 1.在flutter中调用原生方法 2.在Android中实现被调用的方法 3.在iOS中实现被调用的方法 在flutter中调用原生方法 场景,这里你希望调用原生方法告诉你一个bool值,这个值的意义你可以随意定,这里表示的意义是是否是中国用户. 你可以在flutter中设计好要调用的方法名称,这里就叫 isChinese 请

  • go语言方法集为类型添加方法示例解析

    目录 1概述 2为类型添加方法 2.1基础类型作为接收者 2.2结构体作为接收者 3值语义和引用语义 4方法集 4.1类型 *T 方法集 4.2类型 T 方法集 5匿名字段 5.1方法的继承 5.2方法的重写 6方法值和方法表达式 6.1方法值 6.2方法表达式 1概述 在面向对象编程中,一个对象其实也就是一个简单的值或者一个变量,在这个对象中会包含一些函数,这种带有接收者的函数,我们称为方法(method).本质上,一个方法则是一个和特殊类型关联的函数. 一个面向对象的程序会用方法来表达其属性

随机推荐