Android中RecyclerView实现多级折叠列表效果(TreeRecyclerView)

前言

首先不得不吐槽一下产品,尼玛为啥要搞这样的功能....搞个两级的不就好了嘛...自带控件,多好。三级,四级,听说还有六级的....这样丧心病狂的设计,后台也不好给数据吧。

先看看效果:

两级的效果:

三级的效果:

全部展开的效果(我只写了五级)

说说为什么写这货吧:

公司产品提出三级这个需求后,我就在网上找啊找.

找的第一个,发现实现其实是ExpandListview嵌套.

找的第二个,ExpandRecyclview,然后就用呗,发现三级展开很卡,看源码,

发现是RecyclerView套RecyclerView

就没有不嵌套的么.....

然后找到hongyang的那个博客,写个试试吧.

说说思路:

1.Treeadapter应该只需要关心List<TreeAdapterItem> datas 的内容

2.把每个item看成独立的个体,布局样式,每行所占比,bindViewHolder都由自己的来决定。

3.每一个item应该只关心自己的数据和自己的下一级的数据,不会去关心上上级,下下级

4.展开的实现,item把子数据集拿出来,然后添加到List<TreeAdapterItem> datas,变成与自己同级,因为每次展开只会展开一级数据。

5.折叠递归遍历所有子数据,递归拿到自己所有的子数据集(可以理解因为一个文件夹下所有的文件,包括子文件夹下的所有),然后从List<TreeAdapterItem> datas删除这些数据。

见代码:

/**
 * Created by Jlanglang on 2016/12/7.
*
 */
public abstract class TreeAdapterItem<D> {
 /**
 * 当前item的数据
 */
 protected D data;
 /**
 * 持有的子数据
 */
 protected List<TreeAdapterItem> childs;
 /**
 * 是否展开
 */
 protected boolean isExpand;
 /**
 * 布局资源id
 */
 protected int layoutId;
 /**
 * 在每行中所占的比例
 */
 protected int spanSize;

 ····
 get/set方法省略。。。。
 ····
 public TreeAdapterItem(D data) {
 this.data = data;
 childs = initChildsList(data);
 layoutId = initLayoutId();
 spanSize = initSpansize();
 }
 /**
 * 展开
 */
 public void onExpand() {
 isExpand = true;
 }

 /**
 * 折叠
 */
 public void onCollapse() {
 isExpand = false;
 }

 /**
 * 递归遍历所有的子数据,包括子数据的子数据
 *
 * @return List<TreeAdapterItem>
 */
 public List<TreeAdapterItem> getAllChilds() {

  ArrayList<TreeAdapterItem> treeAdapterItems = new ArrayList<>();

  for (int i = 0; i < childs.size(); i++) {

  TreeAdapterItem treeAdapterItem = childs.get(i);

  treeAdapterItems.add(treeAdapterItem);

  if (treeAdapterItem.isParent()) {

   List list = treeAdapterItem.getAllChilds();

   if (list != null && list.size() > 0) {

   treeAdapterItems.addAll(list);
   }
  }
  }
  return treeAdapterItems;
 }

 /**
 * 是否持有子数据
 *
 * @return
 */
 public boolean isParent() {
  return childs != null && childs.size() > 0;
 }
 /**
 * item在每行中的spansize
 * 默认为0,如果为0则占满一行
 * 不建议连续的两级,都设置该数值
 *
 * @return 所占值
 */
 public int initSpansize() {
  return spanSize;
 }
 /**
 * 初始化子数据
 *
 * @param data
 * @return
 */
 protected abstract List<TreeAdapterItem> initChildsList(D data);
 /**
 * 该条目的布局id
 *
 * @return 布局id
 */
 protected abstract int initLayoutId();

 /**
 * 抽象holder的绑定
 *
 * @param holder ViewHolder
 */
 public abstract void onBindViewHolder(ViewHolder holder);
}

再来看看Adapter

public class TreeRecyclerViewAdapter<T extends TreeAdapterItem> extends RecyclerView.Adapter<ViewHolder> {

 protected Context mContext;
 /**
 * 存储所有可见的Node
 */
 protected List<T> mDatas;//处理后的展示数据

 /**
 * 点击item的回调接口
 */
 private OnTreeItemClickListener onTreeItemClickListener;

 public void setOnTreeItemClickListener(OnTreeItemClickListener onTreeItemClickListener) {
 this.onTreeItemClickListener = onTreeItemClickListener;
 }

 /**
 *
 * @param context 上下文
 * @param datas 条目数据
 */
 public TreeRecyclerViewAdapter(Context context, List<T> datas) {
 mContext = context;
 mDatas = datas;
 }

 /**
 * 相应RecyclerView的点击事件 展开或关闭
 * 重要
 * @param position 触发的条目
 */
 public void expandOrCollapse(int position) {
 //获取当前点击的条目
 TreeAdapterItem treeAdapterItem = mDatas.get(position);
 //判断点击的条目有没有下一级
 if (!treeAdapterItem.isParent()) {
  return;
 }
 //判断是否展开
 boolean expand = treeAdapterItem.isExpand();
 if (expand) {
  //获取所有的子数据.
  List allChilds = treeAdapterItem.getAllChilds();
  mDatas.removeAll(allChilds);
  //告诉item,折叠
  treeAdapterItem.onCollapse();
 } else {
  //获取下一级的数据
  mDatas.addAll(position + 1, treeAdapterItem.getChilds());
  //告诉item,展开
  treeAdapterItem.onExpand();
 }
 notifyDataSetChanged();
 }
 //adapter绑定Recycleview后.
 @Override
 public void onAttachedToRecyclerView(RecyclerView recyclerView) {
 super.onAttachedToRecyclerView(recyclerView);
 //拿到布局管理器
 RecyclerView.LayoutManager layoutManager = recyclerView.getLayoutManager();
 //判断是否是GridLayoutManager,因为GridLayoutManager才能设置每个条目的行占比.
 if (layoutManager instanceof GridLayoutManager) {
  final GridLayoutManager gridLayoutManager = (GridLayoutManager) layoutManager;
  gridLayoutManager.setSpanSizeLookup(new GridLayoutManager.SpanSizeLookup() {
  @Override
  public int getSpanSize(int position) {

   TreeAdapterItem treeAdapterItem = mDatas.get(position);
   if (treeAdapterItem.getSpanSize() == 0) {
   //如果是默认的大小,则占一行
   return gridLayoutManager.getSpanCount();
   }
   //根据item的SpanSize来决定所占大小
   return treeAdapterItem.getSpanSize();
  }
  });
 }
 }

 @Override
 public ViewHolder onCreateViewHolder(ViewGroup parent, int viewType) {
 //这里,直接通过item设置的id来创建Viewholder
 return ViewHolder.createViewHolder(mContext, parent, viewType);
 }

 @Override
 public void onBindViewHolder(ViewHolder holder, final int position) {
 final TreeAdapterItem treeAdapterItem = mDatas.get(position);
 holder.itemView.setOnClickListener(new View.OnClickListener() {
  @Override
  public void onClick(View v) {
  //折叠或展开
  expandOrCollapse(position);
  if (onTreeItemClickListener != null) {
   //点击监听的回调.一般不是最后一级,不需要处理吧.
   onTreeItemClickListener.onClick(treeAdapterItem, position);
  }
  }
 });
 treeAdapterItem.onBindViewHolder(holder);
 }

 @Override
 public int getItemViewType(int position) {
 //返回item的layoutId
 return mDatas.get(position).getLayoutId();
 }

 @Override
 public int getItemCount() {
 return mDatas == null ? 0 : mDatas.size();
 }

 public interface OnTreeItemClickListener {
 void onClick(TreeAdapterItem node, int position);
 }
}

具体使用:

/**
 * Created by baozi on 2016/12/8.
 */
public class OneItem extends TreeAdapterItem<CityBean> {

 public OneItem(CityBean data) {
 super(data);
 }
 //这里数据用的是,一个三级城市列表数据。
 @Override
 protected List<TreeAdapterItem> initChildsList(CityBean data) {//这个CityBean 是一级数据
 ArrayList<TreeAdapterItem> oneChilds= new ArrayList<>();
 List<CityBean.CitysBean> citys = data.getCitys();
 if (citys == null) {//如果没有二级数据就直接返回.
  return null;
 }
 for (int i = 0; i < citys.size(); i++) {//遍历二级数据.
  TwoItem twoItem = new TwoItem(citys.get(i));//创建二级条目。
  oneChilds.add(twoItem);
 }
 return oneChilds;
 }

 @Override
 protected int initLayoutId() {//当前级数的布局
 return R.layout.itme_one;
 }

 @Override
 public void onExpand() {
 super.onExpand();

 }

 @Override
 public void onBindViewHolder(ViewHolder holder) {
 //设置当前级数的viewhodler.
 //如果需要某个view展开关闭时的动画,可以在这里保存view到成员变量。
 //然后在onExpand()方法里面操作。
 holder.setText(R.id.tv_content, data.getProvinceName());
 }
}

如果是同一级想要设置不同的布局,接着看

/**
 * Created by baozi on 2016/12/8.
 */
public class FourItem extends TreeAdapterItem<String> {
....

 @Override
 protected List<TreeAdapterItem> initChildsList(String data) {
 ArrayList<TreeAdapterItem> treeAdapterItems = new ArrayList<>();
 for (int i = 0; i < 10; i++) {
  FiveItem threeItem = new FiveItem("我是五级");
  //在遍历的时候,通过条件,重设孩子的布局id.和所占比
  if (i % 4 == 0) {//偷个懒,不多写布局了.
  threeItem.setLayoutId(R.layout.itme_one);
  threeItem.setSpanSize(0);
  } else if (i % 3 == 0) {
  threeItem.setLayoutId(R.layout.item_two);
  threeItem.setSpanSize(2);
  }
  treeAdapterItems.add(threeItem);
 }
 return treeAdapterItems;
 }
....
}
/**
 * Created by baozi on 2016/12/8.
 */
public class FiveItem extends TreeAdapterItem<String> {
 .......
//设置默认的布局
 @Override
 protected int initLayoutId() {
 return R.layout.item_five;
 }
//设置默认的占比
 @Override
 public int initSpansize() {
 return 2;
 }
 //根据layoutId来判断viewhodler并设置
 @Override
 public void onBindViewHolder(ViewHolder holder) {
 if (layoutId == R.layout.itme_one) {
  holder.setText(R.id.tv_content, "我是第一种五级");
 } else if (layoutId == R.layout.item_five) {
  holder.setText(R.id.tv_content, "我是第二种五级");
 }else if (layoutId == R.layout.item_two) {
  holder.setText(R.id.tv_content, "我是第三种五级");
 }
 }
}

更新及详解:

更深入的介绍可以查看这篇文章:http://www.jb51.net/article/113516.htm

下面附上Demo下载地址:

github传送门:TreeRecyclerView

本地下载:http://xiazai.jb51.net/201705/yuanma/TreeRecyclerView(jb51.net).rar

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作能带来一定的帮助,如果有疑问大家可以留言交流,谢谢大家对我们的支持。

(0)

相关推荐

  • Android TableLayout数据列表的回显清空实现思路及代码

    复制代码 代码如下: //数据列表的回显 public void shujuList(){ List<Customer> customerList = dao.findALL(); TableLayout tl = (TableLayout) findViewById(R.id.tlLayout); Log.i(">>>", String.valueOf(tl.getChildCount())); int j = tl.getChildCount(); i

  • Android中RecyclerView实现多级折叠列表效果(二)

    前言 在本文开始之前请大家先看一下这篇文章:http://www.jb51.net/article/113510.htm 上面的这篇文章是之前写的,里面发现有很多不好用地方,也学到些新姿势,改动了许多地方.下面来看看详细的介绍: 要点: 1.可以通过后台控制Item的展示. 2.TreeRecyclerAdapter,可以展开,折叠.多级展示 3.adapter可以使用装饰者模式进行扩展.支持EmptyAdapter.可以添加headview和footview 4.item的样式可以编写文档,t

  • Android RecyclerView实现下拉列表功能

    现在市面上的很多的应用,都带有下拉列表的功能,将所有选项都放在下拉列表中,当用户点击选择的时候,弹出所有的选项,用户选择一项后,下拉列表自动隐藏,很多下拉列表都是用ListView + PopupWindow来实现的,由于Google推出了替代ListView的RecyclerView,所以简单实现一下: MainActivity.java package com.jackie.countdowntimer; import android.graphics.drawable.BitmapDraw

  • Android 模仿iPhone列表数据View刷新动画详解

    因为我本人很喜欢在不同的页面之间跳转时加点好玩的动画,今天无意间看到一个动画效果感觉不错,几种效果图如下:既然好玩就写在博客中,直接说就是:该效果类似于iPhone中View的切换动画效果,今天就只介绍上面展示的效果. 废话不多说,先上效果,再看代码!! 效果一: 效果二: 效果三: 效果四:(犯错的效果): 效果五(回旋效果一): 效果六(回旋效果二): 效果看完了,就来看下上面效果实现的具体代码吧, 中间会把我自己试验的.犯的错误都以注释的形式写下来的, 大家使用的时候别出错就行了!先来看下

  • Android编程实现在Activity中操作刷新另外一个Activity数据列表的方法

    本文实例讲述了Android编程实现在Activity中操作刷新另外一个Activity数据列表的方法.分享给大家供大家参考,具体如下: 做Android项目中遇到这样一个问题:有两个acticity,一个显示好友列表,另外一个显示会话列表,现在问题是在会话界面增加一个添加好友功能,添加好友后要求实时的刷新好友列表. 想了想,找了两种方式 方法一:通过声明要刷新的activity(FriendActivity)为静态变量,然后在一个activity(SessionActivity)中直接调用刷新

  • Android基于RecyclerView实现高亮搜索列表

    话不多说先看今天的实现的效果: 相信这种效果很多项目都会用到,今天就讲讲利用RecycleView来实现他,博主把此篇文章定位初级篇,可能因为这确实很简单,所以我要更要讲的详细一点让新手也可以能看的懂. 饭要开始做了,我们要准备哪些食材呢. 1.一个RecyclerView或是listview或是其他可以显示多item的控件(主要的干货) 2.搞清楚EditText的实时监听 3.让一个textview出现不同的颜色 4.如何穿过Adpter找出textview中key值(也就是高亮字符串) 当

  • Android使用RecyclerView实现自定义列表、点击事件以及下拉刷新

    Android使用RecyclerView 1. 什么是RecyclerView RecyclerView 是 Android-support-v7-21 版本中新增的一个 Widgets,官方对于它的介绍则是:RecyclerView 是 ListView 的升级版本,更加先进和灵活. 简单来说就是:RecyclerView是一种新的视图组,目标是为任何基于适配器的视图提供相似的渲染方式.它被作为ListView和GridView控件的继承者,在最新的support-V7版本中提供支持. 2.

  • Android编程使用ListView实现数据列表显示的方法

    本文实例讲述了Android编程使用ListView实现数据列表显示的方法.分享给大家供大家参考,具体如下: 要将数据库中的数据列表显示在屏幕上,我们要使用ListView这个控件,当用户从数据库中取出数据时,要将数据绑定到显示控件上,如何绑定呢,我们需要创建适配器进行绑定,创建适配器有两种方式: 第一种是用SimpleAdapter创建(要求绑定的数据是List<HashMap<String, Object>>数据类型) 第二种是用SimpleCursorAdapter创建(要求

  • Android使用RecyclerView实现列表数据选择操作

    这些时间做安卓盒子项目,因为安卓电视的显示器比较大,所以一个界面显示 很多数据 ,最多的时候,一个Actvity中用到了好几个RecyclerView. 在RecyclerView中实现Item选中处理时,发现用CheckBox的OnCheckedChangeListener监听事件时,会达不到预期,所以用了OnClickListener来实现. 主界面代码: public class CheckRecyclerViewActivity extends AppCompatActivity imp

  • Android中RecyclerView实现多级折叠列表效果(TreeRecyclerView)

    前言 首先不得不吐槽一下产品,尼玛为啥要搞这样的功能....搞个两级的不就好了嘛...自带控件,多好.三级,四级,听说还有六级的....这样丧心病狂的设计,后台也不好给数据吧. 先看看效果: 两级的效果: 三级的效果: 全部展开的效果(我只写了五级) 说说为什么写这货吧: 公司产品提出三级这个需求后,我就在网上找啊找. 找的第一个,发现实现其实是ExpandListview嵌套. 找的第二个,ExpandRecyclview,然后就用呗,发现三级展开很卡,看源码, 发现是RecyclerView

  • 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实现Item添加和删除的代码示例

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

  • Android中利用viewflipper动画切换屏幕效果

    整个项目的 package com.example.viewflipper; import android.R.integer; import android.app.Activity; import android.os.Bundle; import android.util.Log; import android.view.GestureDetector.OnDoubleTapListener; import android.view.Menu; import android.view.Me

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

  • Android中recyclerView底部添加透明渐变效果

    前言 最近实现一个recyclerView透明渐变的效果,遇到了一些坑,尝试了一些方法,这里记录一下. 效果图 图片在上面显示2列,文字在下面显示1列:底部要有个透明渐变的效果,直到完全看不到. gridLayoutManager动态设置列数 大概是分两类,一类以图片为item 一行2个,一类以文字为item 一行一个. 这个第一反应是用viewType去区分图片类型,但是由于起初不知道gridLayout可以动态列数.就在上面两列,下面一列上为难起来了. 如果统一用一列吧,那就把两个image

  • Android中使用RecylerView实现聊天框效果

    从Android 5.0开始,谷歌公司推出了一个用于大量数据展示的新控件RecylerView,可以用来代替传统的ListView,更加强大和灵活.在上篇文章给大家介绍了Android RecylerView入门教程,大家可以点击查看详情. 效果图如下:(其中,聊天框背景图用9-patch图,可以内容自适应调节.利用AndroidStudio自带的功能制作就行了,图片->右键->create 9-patch file. 其中要注意的是: 1.将9-patch图保存到drawable目录下才管用

  • Android中使用itemdecoration实现时间线效果

    代码如下: // 时间线装饰器 public class TimeLineDecoration extends RecyclerView.ItemDecoration { private Paint mPaint; public TimeLineDecoration() { mPaint = new Paint(); mPaint.setStyle(Paint.Style.FILL); mPaint.setColor(Color.BLUE); mPaint.setStrokeWidth(5);

随机推荐