代码地址如下:
http://www.demodashi.com/demo/13193.html

前言

在开发的时候,我们不免会遇到这么一种数据展示,该数据有以下特征:
1. 数据要以列表形式展示
2. 每条数据要分多行展示,如第一行展示姓名,第二行展示培训课程,第三行显示培训时间
3. 每行展示的数据样式不一样,姓名要求展示在左边,时间展示在右边
4. 更糟糕的是第二行展示的培训课程是一个可下拉列表,点击能收起和展开,在展开时能看到具体的培训内容。

今天就来讲讲这种效果的实现逻辑吧。
涉及以下内容:
1. 分析需求
2. 实现原理
3. 数据拆分整合
4. 效果图和结构图

一. 分析需求

先分析需求,第一条:“数据要以列表形式展示”
那麽這個就得用RecycleView實現
第二,三条:“每条数据要分多行展示,每行展示的数据样式不一样”,需要用到RecycleView的多布局
第四条:“展示一个可下拉列表”,这个似乎要用ExpandableListView实现?或者RecycleView嵌套?

经过一番思考和测试,发现ExpandableListView实现不了这个效果,而我用RecycleView嵌套的过程中出现错乱的问题,就没有继续下去了。但是RecycleView嵌套还是会有一个问题,那就是性能问题。所以接下来,我们会用RecycleView实现多布局的方式对原始数据进行分割展示。

二.实现原理

2.1 首先看看要展示的原始数据结构
package com.android.model;import java.io.Serializable;
import java.util.List;/*** Title:* Description:* <p>* Created by pei* Date: 2018/5/15*/
public class Data implements Serializable{private String header;private String productGroup;private List<String>product;private String footer;public String getHeader() {return header;}public void setHeader(String header) {this.header = header;}public String getProductGroup() {return productGroup;}public void setProductGroup(String productGroup) {this.productGroup = productGroup;}public List<String> getProduct() {return product;}public void setProduct(List<String> product) {this.product = product;}public String getFooter() {return footer;}public void setFooter(String footer) {this.footer = footer;}}

结合上面的效果图,我们需要将Data数据源“切割”成4部分,header,productGroup,product和footer,其中product作为一个list展示,需要满足即可展示也可隐藏的功能。
接下来,我们要实现的整体逻辑就是将一个data数据拆解成四种不同类型的数据,然后依次塞到RecycleView对应的List中平铺展示。这里,我们需要定义数据类型

2.2 定义不同的数据类型
package com.android.model;import java.io.Serializable;/*** Title:数据类型* Description:* <p>* Created by pei* Date: 2018/5/15*/
public class ItemType implements Serializable{public static final int TYPE_HEADER=0;public static final int TYPE_PRODUCT_GROUP=1;public static final int TYPE_PRODUCT=2;public static final int TYPE_FOOTER=3;
}

既然将数据分成四个不同的数据类型,又需要统一展示在RecycleView的list中,那么这四个不同的数据类型需要继承一个统一的数据类型ItemData

2.3 定义统一数据类型类ItemData
package com.android.model;import java.io.Serializable;/*** Title:总数据* Description:* <p>* Created by pei* Date: 2018/5/15*/
public class ItemData implements Serializable{public static final int DEFAULT_INDEX=-1;private int itemType=DEFAULT_INDEX;//类型private int itemId;//一级数据的positionpublic int getItemType() {return itemType;}public void setItemType(int itemType) {this.itemType = itemType;}public int getItemId() {return itemId;}public void setItemId(int itemId) {this.itemId = itemId;}
}

这里我们统一定义了一个itemType,用于给数据进行分类,然后加了一个itemId,用于给每个数据设置一个position

接下来,看看 header,productGroup,product和footer这四类数据,其中header和footer无非是显示一个字符串,这个就简单的展示下header对应的数据data吧

2.4 HeaderData数据样例
package com.android.model;/*** Title:头部数据* Description:* <p>* Created by pei* Date: 2018/5/15*/
public class HeaderData extends ItemData{private String name;public String getName() {return name;}public void setName(String name) {this.name = name;}
}

FooterData和HeaderData大同小异,此处省略,然后看productGroup和product,这两个有些特殊,类似一个二级列表,而productGroup对应的是二级列表中的Parent,product对应的是二级列表中的child。
下面看看productGroup对应的数据结构—ProductGroupData

2.5 ProductGroupData代码
package com.android.model;import java.util.List;/*** Title:产品一级数据* Description:* <p>* Created by pei* Date: 2018/5/15*/
public class ProductGroupData extends ItemData{private String name;private boolean expand;//是否展开private List<ProductData> productList;public String getName() {return name;}public void setName(String name) {this.name = name;}public boolean isExpand() {return expand;}public void setExpand(boolean expand) {this.expand = expand;}public List<ProductData> getProductList() {return productList;}public void setProductList(List<ProductData> productList) {this.productList = productList;}
}

ProductGroupData最主要的是标志位expand,用于记录其二级列表是否为展开状态,然后一个 List productList 用来存放二级数据

然后是ProductData数据结构

2.5 ProductData代码
package com.android.model;/*** Title:产品二级数据* Description:* <p>* Created by pei* Date: 2018/5/15*/
public class ProductData extends ItemData{private int subItemId;//二级数据展示下标private String name;public int getSubItemId() {return subItemId;}public void setSubItemId(int subItemId) {this.subItemId = subItemId;}public String getName() {return name;}public void setName(String name) {this.name = name;}
}

ProductData中主要包含一个subItemId,这是用来标记二级数据展示时的childPosition

至此数据结构创建完毕

三. 数据拆分整合

3.1 模拟一个数据源
       List<Data>list=new ArrayList<>();Data data1=new Data();data1.setHeader("姓名:小明");data1.setProductGroup("培训课程");data1.setProduct(Arrays.asList("语文","数学","英语"));data1.setFooter("时间:2008-6-10");list.add(data1);Data data2=new Data();data2.setHeader("姓名:小花");data2.setProductGroup("培训课程");data2.setProduct(Arrays.asList("物理","化学"));data2.setFooter("时间:2008-6-12");list.add(data2);

Data数据结构前面已经介绍过,下面需要将数据源整合成我们展示时的数据格式

3.2 整个整合流程代码
   public List<ItemData>getItemDatas(List<Data> datas){List<ItemData>itemDatas=new ArrayList<>();if(datas!=null&&!datas.isEmpty()){int size=datas.size();for(int i=0;i<size;i++){Data data=datas.get(i);//头部HeaderData headerData=getHeaderData(data,i);itemDatas.add(headerData);//中部ProductGroupData productGroup = getProductGroupData(data, i);if (productGroup != null) {itemDatas.add(productGroup);}else{LogUtil.i("========productGroup为null======i="+i);}//底部FooterData footerData=getFooterData(data,i);itemDatas.add(footerData);}}return itemDatas;}

然后具体的大家要对HeaderData,ProductGroupData和FooterData进行处理,其中ProductGroupData里面包含ProductData的列表数据,这些数据有一个共同点,就是需要设置itemType和itemId,itemType用于设置数据类型,itemId用来存放数据下标(即position),当然大家更需要处理好 ProductGroupData与ProductData中的数据关联。

ok,数据整合完毕后,就是adapter的处理了,所有数据放到RecyclerView.Adapter中展示,因为所有数据(包括HeaderData,ProductGroupData,FooterData以及ProductData的list)均会平铺展示在RecycleView中,于是我们需要重写以下方法:

getItemCount  ----- 重新计算data的所有数目
getItemViewType(int position)  ---- 获取每项的内容
onCreateViewHolder(ViewGroup parent, int viewType)  ---- 每项根据类型定义不同ui布局
onBindViewHolder(RecyclerView.ViewHolder holder, int position) --- 不同数据类型的对应逻辑处理

在adapter声明里,我们会定义两个数据list

   protected List<ItemData> mData;//传入的dataprotected List<ItemData>mAllOrders=new ArrayList<>();//展示的data

mData用于从activity中传入数据,此时的数据结构为

   ----HeaderData
   ----ProductGroupData
        ----ProductData1
        ----ProductData2
        ----........
   ----FooterData

mAllOrders是最终展示数据,其数据结构为

   ----HeaderData
   ----ProductGroupData
   ----ProductData1
   ----ProductData2
   ----........
   ----FooterData

mData与mAllOrders最大区别在于mData中包含有二级数据结构,而mAllOrders中的数据就一层,统一平铺展示。

getItemCount既承载着将mData数据结构转化成mAllOrders的数据结构,也需要计算所有数据的个数

3.2 getItemViewType(int position)获取每项的内容

这个比较简单

  @Overridepublic int getItemViewType(int position) {return mAllOrders.get(position).getItemType();}

然后是onCreateViewHolder(ViewGroup parent, int viewType)

3.3 onCreateViewHolder示例代码
switch (viewType) {case ItemType.TYPE_HEADER:View v = mInflater.inflate(R.layout.item_header, parent, false);holder = new HeaderHolder(v);break;case ItemType.TYPE_PRODUCT_GROUP:View v1 = mInflater.inflate(R.layout.item_product_group, parent, false);holder = new ProductGroupHolder(v1);break;case ItemType.TYPE_PRODUCT:View v2 = mInflater.inflate(R.layout.item_product, parent, false);holder = new ProductHolder(v2);break;case ItemType.TYPE_FOOTER:View v3 = mInflater.inflate(R.layout.item_footer, parent, false);holder = new FooterHolder(v3);break;default:break;}
3.4 onBindViewHolder不同数据处理逻辑
  @Overridepublic void onBindViewHolder(RecyclerView.ViewHolder holder, int position) {Object obj = mAllOrders.get(position);if (holder instanceof HeaderHolder) {bindHeader(holder, obj,position);}else if(holder instanceof ProductGroupHolder){bindProductGroup(holder,obj,position);}else if(holder instanceof ProductHolder){bindProduct(holder, obj,position);}else if(holder instanceof FooterHolder){bindFooter(holder,obj, position);}}

最后在bindHeader,bindProductGroup,bindProduct,bindFooter中对各数据的展示和逻辑做处理。
这里需要提醒的是bindProductGroup(holder,obj,position);方法,因为当中涉及到点击展开ProductData集合,再点击收起ProductData集合的操作,需要用到RecycleView的两个更新方法:

//展开时调用
notifyItemRangeInserted(int positionStart, int itemCount)
//收起时调用
notifyItemRangeRemoved(int positionStart, int itemCount)
3.5 点击展开,点击收起ProductData列表的逻辑

处理此逻辑示例代码如下:

 //点击事件productGroupHolder.mTvName.setOnClickListener(new View.OnClickListener() {@Override public void onClick(View v) {//更新数据if(productGroupData.isExpand()){//收起notifyItemRangeRemoved(productGroupHolder.getAdapterPosition() + 1, productGroupData.getProductList().size());}else{//展开notifyItemRangeInserted(productGroupHolder.getAdapterPosition() + 1, productGroupData.getProductList().size());}productGroupData.setExpand(!productGroupData.isExpand());notifyItemChanged(productGroupHolder.getAdapterPosition());}});

一切就绪以后,在MainActivity中调用

3.6 MainActivity中示例代码
package com.android.testdemo;import android.support.v7.widget.RecyclerView;
import android.view.View;
import android.widget.Button;import com.android.adapter.MyAdapter;
import com.android.base.BaseActivity;
import com.android.model.Data;
import com.android.model.ItemData;import java.util.ArrayList;
import java.util.List;import butterknife.BindView;public class MainActivity extends BaseActivity {@BindView(R.id.button1)Button mBtn1;@BindView(R.id.rv)RecyclerView mRecyclerView;private List<Data> mDatas;private MyAdapter myAdapter;@Overrideprotected int getContentViewId() {return R.layout.activity_main;}@Overrideprotected void initView() {}@Overrideprotected void initData() {//模拟数据mDatas=ParseHelper.getInstance().getDatas();//初始化RecycleView相关List<ItemData>itemDatas=new ArrayList<>();itemDatas.addAll(ParseHelper.getInstance().getItemDatas(mDatas));myAdapter=new MyAdapter(itemDatas,mContext);myAdapter.setRecyclerManager(mRecyclerView);}@Overrideprotected void setListener() {mBtn1.setOnClickListener(this);}@Overridepublic void onClick(View v){switch (v.getId()) {case R.id.button1:break;default:break;}}@Overrideprotected void onDestroy() {super.onDestroy();}
}

四. 效果图和结构图

运行效果图

项目结构图

RecycleView实现多布局可展开列表

代码地址如下:
http://www.demodashi.com/demo/13193.html

注:本文著作权归作者,由demo大师发表,拒绝转载,转载需要作者授权

RecycleView实现多布局可展开列表相关推荐

  1. Android RecycleView切换条目布局visibility导致列表滑动

    在项目开发过程遇到一个问题,切换布局中某个子view的visibility为gone后,RecyclerView 自己莫名其妙的滚动了一点.通过网上查找发现,这是RecyclerView 抢占焦点导致 ...

  2. 安卓学习笔记22:常用控件 - 可展开列表视图

    文章目录 零.学习目标 一.可展开列表视图概述 二.可展开列表视图继承关系图 三.教学案例 - 选择四大名著人物 (一)运行效果 (二)涉及知识点 (三)实现步骤 1.创建安卓应用[SelectCha ...

  3. Android精通:View与ViewGroup,LinearLayout线性布局,RelativeLayout相对布局,ListView列表组件...

    UI的描述 对于Android应用程序中,所有用户界面元素都是由View和ViewGroup对象构建的.View是绘制在屏幕上能与用户进行交互的一个对象.而对于ViewGroup来说,则是一个用于存放 ...

  4. 精通android布局,Android精通:View与ViewGroup,LinearLayout线性布局,RelativeLayout相对布局,ListView列表组件...

    标题图 UI的描述 对于Android应用程序中,所有用户界面元素都是由View和ViewGroup对象构建的.View是绘制在屏幕上能与用户进行交互的一个对象.而对于ViewGroup来说,则是一个 ...

  5. Android之如何实现阿拉伯版本(RTL)的recycleView的网格布局

    1 问题 比如正常的recycleView的网格布局效果如下 1 2 34 5 67 8 现在需要变成这样的效果 3 2 16 5 48 7 2 思考过程和尝试解决方法 1)从recycleView上 ...

  6. [转]Shared——RN如何实现一个ExpandableList(可展开列表)组件

    作者:小石头若海 原文地址:https://segmentfault.com/a/1190000011754908 RN如何实现一个ExpandableList(可展开列表)组件 讨论与分析 首先,我 ...

  7. Flutter折叠展开列表的使用

      flutter中官方其实已经封装好了折叠展开列表:ExpansionPanel,但是官方的可定制性太差,我今天主要说下如何简单实现一个自己的折叠展开列表. 状态控制   折叠展开列表,每一个父列表 ...

  8. Android中RecyclerView点击item展开列表详细内容(超简单实现)

    请注明出处: http://blog.csdn.net/qq_23179075/article/details/79230457 Android中RecyclerView点击item展开列表详细内容( ...

  9. android 固定底部 布局_Android系统列表控件

    在android系统控件中,有多个控件可以展示列表数据. 一.ListView 该组件是android中最常用的一个UI组件,用于实现在屏幕上显示多个内容,以便于我们用手指进行滑动. ListView ...

最新文章

  1. DevOps的工程化
  2. redis在容器里连接不上_Redis服务器被劫持风波,服务器相关知识共享学习
  3. 应用域名改造-https证书部分
  4. Hibernatediscriminator-value用法
  5. Log4net 日志使用介绍
  6. 怎么利用linux来操作手机,Linux_在Linux操作系统下操作蓝牙手机的方法,所谓操作,到现在只是通过蓝 - phpStudy...
  7. 立即表达式的多种写法与注意点以及in操作符的作用
  8. 全新 Veeam Availability Suite 9.5 成为率先全面集成 Windows Server 2016和Hyper-V技术的 可用性解决方案之一...
  9. php在windows安装,php在windows环境下的安装
  10. 概率论中两个独立连续随机变量X,Y,变量Z=X+Y的密度函数为X,Y的卷积与特征函数原理
  11. 悲剧四个月python培训班,需要踩完坑犯过错,这些免费的编程资源,值得一生推
  12. 如何在html中插入乘积函数,excel乘法怎么保留两位小数
  13. WIN10安装cad2006提示无权限安装的解决办法
  14. MircoPython 的组件扩展方法
  15. 美光证实:DDR5 内存供应不足与 PMIC 和 VRM 短缺有关,2022 年改善的可能性很小
  16. NB-IoT、LoRa、eMTC、Zigbee、Sigfox、WiFi、蓝牙,谁能称霸物联网时代
  17. Deepin 微信版本太低无法登录
  18. 一体化伺服电机一圈多少脉冲
  19. 直播预告 | 哈工大HIT-SCIR实验室专场二
  20. 微软Project项目管理软件简介

热门文章

  1. oracle的sid相同如何解决,oracle数据库的SID重复有关问题
  2. feign post 传递空值_http中post和get的区别和联系
  3. 计算机职称考试知识点,职称计算机考试复习知识点
  4. 数字信号处理——时频分析(短时傅里叶变换)
  5. Java学习日报—2021/11/18
  6. python的csv标准库,Python标准库: csv模块——CSV文件的读写
  7. Spring容器的底层实现
  8. 【LeetCode】剑指 Offer 63. 股票的最大利润
  9. System.gc()和Runtime.gc()的区别?
  10. Python学习day07 - Python进阶(1) 内置方法