Android 中RecyclerView通用适配器的实现

Android 中RecyclerView通用适配器的实现

前言:

SDK的5.0版本出来已经N久了,可以说是已经经过许多人的检验了,里面的新控件不能说是非常完美,但也是非常好用了,其中最让我喜爱的就是RecyclerView了,可以完美替代ListView和GridView(除了添加headerview和footview了,网上有许多解决方式。这个下面会以一种简单的方式顺带解决,肯定为大家省心),而且可以代码动态切换这两种布局方式以及瀑布流布局。相关切换方式网上有很多,大家自行搜索,我就不贴连接了。

相信大家在之前使用listview时肯定一直很厌烦重复编写无数的adapter,当然有那么一部分机智如我的人肯定一直使用着万能适配器。然而RecyclerView要求我们必须使用ViewHolder来实现adapter。这就让许多用惯了万能适配器的人不爽了。今天我就提供一直基于listview万能适配器的实现原理来改良实现的RecyclerView的通用适配器,由于不是教学,切代码比较简单,就不分段讲解了,相信大家看注释就能看懂。

一共两个类,一个是继承了系统的Android.support.v7.widget.RecyclerView.ViewHolder所实现的RViewHolder类,通过他实现任意控件的缓存一个是继承了android.support.v7.widget.RecyclerView.Adapter所实现的RBaseAdapter类。

实现代码:

RViewHolder

import android.content.Context;
import android.graphics.Bitmap;
import android.support.v7.widget.RecyclerView.ViewHolder;
import android.util.SparseArray;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.ImageView;
import android.widget.TextView;

public class RViewHolder extends ViewHolder {

 private Context mContext;
 private View mConvertView;
 private SparseArray<View> mViews;

 public RViewHolder(View itemView) {
  super(itemView);
  mConvertView = itemView;
  this.mViews = new SparseArray<View>();
 }

 public static RViewHolder get(Context context, ViewGroup parent, int layoutId, int position) {
  View view = LayoutInflater.from(context).inflate(layoutId, parent, false);
  return new RViewHolder(view);
 }

 /**
  * 通过控件的Id获取对于的控件,如果没有则加入views
  *
  * @param viewId
  * @return
  */
 public <T extends View> T getView(int viewId) {

  View view = mViews.get(viewId);
  if (view == null) {
   view = mConvertView.findViewById(viewId);
   mViews.put(viewId, view);
  }
  return (T) view;
 }

 /**
  * 为TextView设置字符�?
  *
  * @param viewId
  * @param text
  * @return
  */
 public ViewHolder setText(int viewId, String text) {
  TextView view = getView(viewId);
  view.setText(text);
  return this;
 }

 /**
  * 为ImageView设置图片
  *
  * @param viewId
  * @param drawableId
  * @return
  */
 public ViewHolder setImageResource(int viewId, int drawableId) {
  ImageView view = getView(viewId);
  view.setImageResource(drawableId);

  return this;
 }

 /**
  * 为ImageView设置图片
  *
  * @param viewId
  * @param drawableId
  * @return
  */
 public ViewHolder setImageBitmap(int viewId, Bitmap bm) {
  ImageView view = getView(viewId);
  view.setImageBitmap(bm);
  return this;
 }

 public View getConvertView() {
  return mConvertView;
 }
}

这里是RBaseAdapter

import java.util.ArrayList;
import java.util.Collection;
import java.util.List;

import android.content.Context;
import android.support.v7.widget.RecyclerView.Adapter;
import android.view.LayoutInflater;
import android.view.View;
import android.view.ViewGroup;
import android.widget.LinearLayout;

public abstract class RBaseAdapter<T> extends Adapter<RViewHolder> {
 private Context mContext;
 private List<T> list;
 protected LayoutInflater mInflater;
 private int mItemLayoutId;

 public RBaseAdapter(Context context) {
  // TODO Auto-generated constructor stub
  this.mContext = context;
  this.mInflater = LayoutInflater.from(mContext);
  this.mItemLayoutId = new LinearLayout(mContext).getId();
  this.list = new ArrayList<T>();

 }

 public RBaseAdapter(Context context, List<T> list) {
  // TODO Auto-generated constructor stub
  this.mContext = context;
  this.mInflater = LayoutInflater.from(mContext);
  this.mItemLayoutId = new LinearLayout(mContext).getId();
  this.list = list;

 }

 public RBaseAdapter(Context context, List<T> list, int itemLayoutId) {
  this.mContext = context;
  this.mInflater = LayoutInflater.from(mContext);
  this.mItemLayoutId = itemLayoutId;
  this.list = list;

 }

 public RBaseAdapter(Context context, int itemLayoutId) {
  this.mContext = context;
  this.mInflater = LayoutInflater.from(mContext);
  this.mItemLayoutId = itemLayoutId;
  this.list = new ArrayList<T>();

 }

 public void setitemLayoutId(int itemLayoutId) {
  this.mItemLayoutId = itemLayoutId;
 }

 public List<T> getList() {
  return this.list;
 }

 public void appendList(List<T> list) {
  // TODO Auto-generated method stub
  this.list = list;
  notifyDataSetChanged();
 }

 public void addList(List<T> list2) {
  // TODO Auto-generated method stub
  this.list.addAll((Collection<? extends T>) list2);
  notifyDataSetChanged();
 }

 @Override
 public int getItemCount() {
  return list.size();

 }

 boolean hasHeader = false;
 boolean hasFooter = false;
 View headerView;
 View footerView;

 public void setHeaderView(View headerView) {
  hasHeader=true;
  this.headerView = headerView;
 }

 public void setFooterView(View footerView) {
   hasFooter = true;
  this.footerView = footerView;
 }

 public View getHeaderView() {
  return headerView;
 }

 public View getFooterView() {
  return footerView;
 }

 @Override
 public void onBindViewHolder(RViewHolder holder, int position) {
  if (hasHeader && position == 0) {
   return;
  } else if (hasFooter && position == (list.size() + (hasHeader ? 1 : 0))) {
   return;
  } else
   convert(holder, (T) list.get(position));
 }

 @Override
 public RViewHolder onCreateViewHolder(ViewGroup parent, int position) {
  if (hasHeader && position == 0) {
   return new RViewHolder(headerView);
  } else if (hasFooter && position == (list.size() + (hasHeader ? 1 : 0))) {
   return new RViewHolder(footerView);
  } else
   return RViewHolder.get(mContext, parent, mItemLayoutId, position);

 }
//这里定义抽象方法,我们在匿名内部类实现的时候实现此方法来调用控件
 public abstract void convert(RViewHolder holder, T item);
}

对于RBaseAdapter稍微讲解下,首先是泛型,这样任何对象类型都可以使用,再来就是前面提到的headerview和footerview的解决,可以看到RBaseAdapter里面定义了几个方法,通过position的不同来加载不同的布局的思想来添加headerview和footerview。

Activity里面调用

recyclerView.setAdapter(new RBaseAdapter<VirtualWinsBean>(mContext, R.layout.virtual_win_users_list_item) {

   @Override
   public void convert(RViewHolder holder, VirtualWinsBean item) {
    if (Util.checkNULL(item.getNick_name())) {
     holder.setText(R.id.name, Util.HidePhone(item.getPhone() + ""));
    } else {
     holder.setText(R.id.name, item.getNick_name());
    }
    RoundImageView networkImageView = holder.getView(R.id.photo);
    networkImageView.setLoadingImage(R.drawable.header_def);
    networkImageView.setDefultImage(R.drawable.header_def);
    networkImageView.LoadUrl(U.g(item.getFile_url()));
   }
  });

感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!

(0)

相关推荐

  • Android中RecyclerView 滑动时图片加载的优化

    RecyclerView 滑动时的优化处理,在滑动时停止加载图片,在滑动停止时开始加载图片,这里用了Glide.pause 和Glide.resume.这里为了避免重复设置增加开销,设置了一个标志变量来做判断. mRecyclerView.addOnScrollListener(new RecyclerView.OnScrollListener() { @Override public void onScrollStateChanged(RecyclerView recyclerView, in

  • 浅谈Android为RecyclerView增加监听以及数据混乱的小坑

    为 RecyclerView增加监听 1.在实现好的MyAdapter中写内部接口: public void setOnItemLongClickListener(OnItemLongClickListener onItemLongClickListener) { this.onItemLongClickListener = onItemLongClickListener; } public void setOnItemClickListener(OnItemClickListener onIt

  • Android 滑动监听RecyclerView线性流+左右划删除+上下移动

    废话不多说了,直接给大家贴代码了.具体代码如下所示: <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_wid

  • Android RecyclerView添加上拉加载更多效果

    先看一下效果 刷新的时候是这样的: 没有更多的时候是这样的: 既然有刷新的时候有两种状态就要定义两个状态 //普通布局的type static final int TYPE_ITEM = 0; //脚布局 static final int TYPE_FOOTER = 1; 在特定的时候去显示: @Override public int getItemViewType(int position) { //如果position加1正好等于所有item的总和,说明是最后一个item,将它设置为脚布局

  • Android 给RecyclerView添加分割线的具体步骤(分享)

    [吐槽]RecyclerView没有提供分割线的方法,想要加个线还要自己画,点击事件的监听都要自己实现,不过真的好用. 给RecyclerView添加分割线的步骤 1.新建类继承于RecyclerView.ItemDecoration,此为是抽象类: public static abstract class ItemDecoration { public void onDraw(Canvas c, RecyclerView parent, State state) { onDraw(c, par

  • android给RecyclerView加上折叠的效果示例

    RecyclerView有很高的自由度,可以说只有想不到没有做不到,真是越用越喜欢.这次用超简单的方法,让RecyclerView带上折叠的效果. 效果是这样的. 总结一下这个列表的特点,就是以下三点: 1. 重叠效果: 2. 层次感: 3. 首项的差动效果. 下面我们来一个个解决. 我们新建一个ParallaxRecyclerView,让它继承RecyclerView,并使用LinearLayoutManager作为布局管理器. 重叠效果 其实就是每一项都搭一部分在它前面那项而已.我们知道,R

  • Android中RecyclerView实现滑动删除与拖拽功能

    前言 从Android 5.0开始,谷歌推出了新的控件RecyclerView,相对于早它之前的ListView,优点多多,功能强大,也给我们的开发着提供了极大的便利,今天自己学习一下RecyclerView轻松实现滑动删除及拖拽的效果. 如下图. 相信研究过RecyclerView的同学,应该很清楚该怎么实现这样的效果,若是用ListView,这样的效果实现起来可能就有点麻烦,但是在强大的RecyclerView面前这样的的效果只需很少的代码,因为谷歌给我们提供了强大的工具类ItemTouch

  • Android RecyclerView的卡顿问题的解决方法

    RecyclerView为什么会卡 RecyclerView作为v7包的新控件,自从推出就广受Android Developer们欢迎,实际上它已经取代了ListView和GridView两位老前辈的地位.然而不少亲们想必也已经发现了:没有优化过的Recycler性能很poor.上一篇博主使用的item也仅仅是一个图两串字而已,结果一滑动就卡的要命,不能忍! 那么why?回想在用ListView和GridView的adapter时,我们是用一种叫ViewHolder的自定义类(容器)来实现优化的

  • Android recyclerview实现拖拽排序和侧滑删除

    Recyclerview现在基本已经替代Listview了,RecyclerView也越来越好用了  当我们有实现条目的拖拽排序和侧滑删除时  可以直接时候Recyclerview提供的API就可以直接实现了 先贴上主要代码 private void initveiw() { ArrayList<String> items = new ArrayList<>(Arrays.asList("itme1", "item2", "itme

  • Android 中RecyclerView通用适配器的实现

    Android 中RecyclerView通用适配器的实现 前言: SDK的5.0版本出来已经N久了,可以说是已经经过许多人的检验了,里面的新控件不能说是非常完美,但也是非常好用了,其中最让我喜爱的就是RecyclerView了,可以完美替代ListView和GridView(除了添加headerview和footview了,网上有许多解决方式.这个下面会以一种简单的方式顺带解决,肯定为大家省心),而且可以代码动态切换这两种布局方式以及瀑布流布局.相关切换方式网上有很多,大家自行搜索,我就不贴连

  • Android中RecyclerView拖拽、侧删功能的实现代码

    废话不多说,下面展示一下效果. 这是GridView主文件实现. public class GridViewActivity extends AppCompatActivity { RecyclerView mRecyclerView; List<String> mStringList; RecyclerAdapter mRecyAdapter; @Override protected void onCreate(@Nullable Bundle savedInstanceState) { s

  • Android中RecyclerView实现横向滑动代码

    RecyclerView 是Android L版本中新添加的一个用来取代ListView的SDK,它的灵活性与可替代性比listview更好.本文给大家介绍Android中RecyclerView实现横向滑动代码,一起看看吧. android.support.v7.widget.RecyclerView 功能:RecyclerView横向滑动 控件:<android.support.v7.widget.RecyclerView /> Java类:RecyclerView.GalleryAdap

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

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

  • Android中RecyclerView实现商品分类功能

    本文实例为大家分享了Android中RecyclerView实现商品分类功能的具体代码,供大家参考,具体内容如下 三个个RecyclerView实现 //左边的布局  <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"     android:layout_width="match_parent"     android:layout_height="50d

  • Android中RecyclerView上拉下拉,分割线,多条目的实例代码

    //activity的xml <?xml version="1.0" encoding="utf-8"?> <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:id="@+id/activity

  • Android 中RecyclerView顶部刷新实现详解

    Android 中RecyclerView顶部刷新实现详解 1. RecyclerView顶部刷新的原理 RecyclerView顶部刷新的实现通常都是在RecyclerView外部再包裹一层布局.在这个外层布局中,还包含一个自定义的View,作为顶部刷新时的指示View.也就是说,外层布局中包含两个child,一个顶部刷新View,一个RecyclerView,顶部刷新View默认是隐藏不可见的.在外层布局中对滑动事件进行处理,当RecyclerView滑动到顶部并继续下滑的时候,根据滑动的距

  • Android 中RecyclerView多种item布局的写法(头布局+脚布局)

    RecyclerView多个item布局的写法(头布局+脚布局) 上图 github 下载源码 Initial commit第一次提交的代码,为本文内容 以下的为主要代码,看注释即可,比较简单 MainActivity 含上拉加载更多 package com.anew.recyclerviewall; import android.os.Bundle; import android.os.Handler; import android.support.v7.app.AppCompatActivi

  • Android中RecyclerView点击Item设置事件

    在上一篇Android RecylerView入门教程中提到,RecyclerView不再负责Item视图的布局及显示,所以RecyclerView也没有为Item开放OnItemClick等点击事件,这就需要开发者自己实现.博客最下面有Demo程序运行动画. 奉上Demo的Github链接. 在调研过程中,发现有同学修改RecyclerView源码来实现Item的点击监听,但认为这不是一个优雅的解决方案,最终决定在RecyclerView.ViewHolder上做文章. 思路是:因为ViewH

  • Android中RecyclerView的item宽高问题详解

    前言 本文主要给大家介绍了关于Android中RecyclerView的item宽高问题的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧. 在创建viewholder传入的View时,如果不指定其viewgroup,就会出现宽高只包裹显示内容的问题. View view = LayoutInflater.from(context).inflate(R.layout.test_test,null); 上面的做法就会出问题 改成这样就可以正常显示设置的宽高 View vie

随机推荐