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:
*
* 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:
*
* 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;//一级数据的position
public 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:
*
* 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:
*
* Created by pei
* Date: 2018/5/15
*/
public class ProductGroupData extends ItemData{
private String name;
private boolean expand;//是否展开
private List 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 getProductList() {
return productList;
}
public void setProductList(List productList) {
this.productList = productList;
}
}
```
ProductGroupData最主要的是标志位expand,用于记录其二级列表是否为展开状态,然后一个 List productList 用来存放二级数据
然后是ProductData数据结构
#####2.5 ProductData代码
```
package com.android.model;
/**
* Title:产品二级数据
* Description:
*
* 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 模拟一个数据源
```
Listlist=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 ListgetItemDatas(List datas){
ListitemDatas=new ArrayList<>();
if(datas!=null&&!datas.isEmpty()){
int size=datas.size();
for(int i=0;i mData;//传入的data
protected ListmAllOrders=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)获取每项的内容
这个比较简单
```
@Override
public 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不同数据处理逻辑
```
@Override
public 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 mDatas;
private MyAdapter myAdapter;
@Override
protected int getContentViewId() {
return R.layout.activity_main;
}
@Override
protected void initView() {
}
@Override
protected void initData() {
//模拟数据
mDatas=ParseHelper.getInstance().getDatas();
//初始化RecycleView相关
ListitemDatas=new ArrayList<>();
itemDatas.addAll(ParseHelper.getInstance().getItemDatas(mDatas));
myAdapter=new MyAdapter(itemDatas,mContext);
myAdapter.setRecyclerManager(mRecyclerView);
}
@Override
protected void setListener() {
mBtn1.setOnClickListener(this);
}
@Override
public void onClick(View v){
switch (v.getId()) {
case R.id.button1:
break;
default:
break;
}
}
@Override
protected void onDestroy() {
super.onDestroy();
}
}
```
####四. 效果图和结构图
> 运行效果图
![](/contentImages/image/20180516/BTzFGlyLvSk3ckcR2LW.gif)
> 项目结构图
![](/contentImages/image/20180516/goVcqnzMB3PHQiy5fD9.png)