全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

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, "我是第三种五级");
 }
 }
}

更新及详解:

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

下面附上Demo下载地址:

github传送门:TreeRecyclerView

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

总结

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


# recyclerview多级列表  # recyclerview二级列表  # recyclerview折叠效果  # 递归  # 我是  # 自己的  # 所占  # 遍历  # 五级  # 这篇文章  # 绑定  # 回调  # 两级  # 的是  # 判断是否  # 我就  # 在这里  # 本地下载  # 第一个  # 下载地址  # 不需要  # 我只  # 只会 


相关文章: 头像制作网站在线观看,除了站酷,还有哪些比较好的设计网站?  子杰智能建站系统|零代码开发与AI生成SEO优化指南  香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南  音乐网站服务器如何优化API响应速度?  如何在新浪SAE免费搭建个人博客?  官网自助建站平台指南:在线制作、快速建站与模板选择全解析  大连网站设计制作招聘信息,大连投诉网站有哪些?  Swift中switch语句区间和元组模式匹配  北京营销型网站制作公司,可以用python做一个营销推广网站吗?  如何在IIS中新建站点并解决端口绑定冲突?  深圳网站制作费用多少钱,读秀,深圳文献港这样的网站很多只提供网上试读,但有些人只要提供试读的文章就能全篇下载,这个是怎么弄的?  ,怎么用自己头像做动态表情包?  建站主机服务器选购指南:轻量应用与VPS配置解析  深圳网站制作的公司有哪些,dido官方网站?  网站制作外包价格怎么算,招聘网站上写的“外包”是什么意思?  如何在万网自助建站平台快速创建网站?  h5在线制作网站电脑版下载,h5网页制作软件?  头像制作网站在线制作软件,dw网页背景图像怎么设置?  如何在阿里云高效完成企业建站全流程?  如何通过西部数码建站助手快速创建专业网站?  如何自定义建站之星网站的导航菜单样式?  制作门户网站的参考文献在哪,小说网站怎么建立?  香港代理服务器配置指南:高匿IP选择、跨境加速与SEO优化技巧  建站之星安装后界面空白如何解决?  如何注册花生壳免费域名并搭建个人网站?  香港服务器选型指南:免备案配置与高效建站方案解析  电商网站制作公司有哪些,1688网是什么意思?  网站建设制作需要多少钱费用,自己做一个网站要多少钱,模板一般多少钱?  建站之星安装失败:服务器环境不兼容?  建站之星如何取消后台验证码生成?  如何在Golang中处理模块冲突_解决依赖版本不兼容问题  教学论文网站制作软件有哪些,写论文用什么软件 ?  如何在服务器上三步完成建站并提升流量?  制作网站外包平台,自动化接单网站有哪些?  Python文件管理规范_工程实践说明【指导】  5种Android数据存储方式汇总  建站VPS推荐:2025年高性能服务器配置指南  如何打造高效商业网站?建站目的决定转化率  较简单的网站制作软件有哪些,手机版网页制作用什么软件?  如何在建站之星网店版论坛获取技术支持?  香港服务器网站测试全流程:性能评估、SEO加载与移动适配优化  桂林网站制作公司有哪些,桂林马拉松怎么报名?  保定网站制作方案定制,保定招聘的渠道有哪些?找工作的人一般都去哪里看招聘信息?  建站之星代理费用多少?最新价格详情介绍  ,巨量百应是干嘛的?  制作表格网站有哪些,线上表格怎么弄?  如何访问已购建站主机并解决登录问题?  建站上市公司网站建设方案与SEO优化服务定制指南  建站主机如何安装配置?新手必看操作指南  建站IDE高效指南:快速搭建+SEO优化+自适应模板全解析 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。