资源摘要:有时候,使用ListView并不能满足应用程序所需要的功能。有些应用程序需要多组ListView,这时候我们就要使用一种新的控件ExpandableListView——可以扩展的ListView。它的作用就是将ListView进行分组。就好像我们使用QQ的时候,有“我的好友”,“陌生人”,“黑名单”一样,点击一下会扩展开,再点击一下又会收缩回去。ExpandableListView是一个垂直滚动显示两级列表项的视图,与ListView不同的是,它可以有两层:每一层都能够被独立的展开并显示其子项。这些子项来自于与该视图关联的ExpandableListAdapter。每一个可以扩展的列表项的旁边都有一个指示符(箭头)用来说明该列表项目前的状态(这些状态一般是已经扩展开的列表项,还没有扩展开的列表项,子列表项和最后一个子列表项)。可以使用方法:setChildIndicator(Drawable),setGroupIndicator(Drawable)(或者相应的XML文件的属性) 去设置这些指示符的样式。当然也可以使用默认的指示符。布android.R.layout.simple_expandable_list_item_1,android.R.layout.simple_expandable_list_item_2和ListView一样,ExpandableListView也是一个需要Adapter作为桥梁来取得数据的控件。一般适用于ExpandableListView的Adapter都要继承BaseExpandableListAdapter这个类,并且必须重载getGroupView和getChildView这两个最为重要的方法。BaseExpandableListAdapter的主要重载方法如下:public abstract ObjectgetChild (int groupPosition, int childPosition)取得与指定分组、指定子项目关联的数据.参数groupPosition 包含子视图的分组的位置.childPosition 指定的分组中的子视图的位置.返回与子视图关联的数据.public abstract long getChildId (int groupPosition, intchildPosition)取得给定分组中给定子视图的ID. 该组ID必须在组中是唯一的.必须不同于其他所有ID(分组及子项目的ID).参数groupPosition 包含子视图的分组的位置.childPosition 要取得ID的指定的分组中的子视图的位置.返回与子视图关联的ID.public abstract View getChildView (int groupPosition, intchildPosition, boolean isLastChild, View convertView, ViewGroup parent)取得显示给定分组给定子位置的数据用的视图.参数groupPosition 包含要取得子视图的分组位置.childPosition 分组中子视图(要返回的视图)的位置.isLastChild 该视图是否为组中的最后一个视图.convertView 如果可能,重用旧的视图对象.使用前你应该保证视图对象为非空,并且是否是合适的类型.如果该对象不能转换为可以正确显示数据的视图,该方法就创建新视图.不保证使用先前由 getChildView(int, int,boolean, View, ViewGroup)创建的视图.parent 该视图最终从属的父视图.返回指定位置相应的子视图.public abstract int getChildrenCount (int groupPosition)取得指定分组的子元素数.参数groupPosition 要取得子元素个数的分组位置.返回指定分组的子元素个数.public abstract long getCombinedChildId (long groupId, long childId)取得一览中可以唯一识别子条目的 ID(包括分组ID和子条目ID).可扩展列表要求每个条目 (分组条目和子条目)具有一个可以唯一识别列表中子条目和分组条目的ID. 该方法根据给定子条目ID和分组条目ID返回唯一识别ID.另外,如果 hasStableIds() 为真,该函数返回的ID必须是固定不变的.参数groupId 包含子条目ID的分组条目ID.childId 子条目的ID.返回可以在所有分组条目和子条目中唯一识别该子条目的ID(可能是固定不变的).public abstract long getCombinedGroupId (long groupId)取得一览中可以唯一识别子条目的 ID(包括分组ID和子条目ID).可扩展列表要求每个条目 (分组条目和子条目)具有一个可以唯一识别列表中子条目和分组条目的ID. 该方法根据给定子条目ID和分组条目ID返回唯一识别ID.另外,如果 hasStableIds() 为真,该函数返回的ID必须是固定不变的.参数groupId 分组条目ID.返回可以在所有分组条目和子条目中唯一识别该分组条目的ID(可能是固定不变的).public abstract Object getGroup (int groupPosition)取得与给定分组关联的数据.参数groupPosition 分组的位置.返回指定分组的数据.public abstract int getGroupCount ()取得分组数.返回分组数.public abstract long getGroupId (int groupPosition)取得指定分组的ID.该组ID必须在组中是唯一的.必须不同于其他所有ID(分组及子项目的ID).参数groupPosition 要取得ID的分组位置.返回与分组关联的ID.public abstract View getGroupView (int groupPosition, booleanisExpanded, View convertView, ViewGroup parent)取得用于显示给定分组的视图. 这个方法仅返回分组的视图对象, 要想获取子元素的视图对象,就需要调用 getChildView(int, int, boolean, View, ViewGroup).参数groupPosition 决定返回哪个视图的组位置 .isExpanded 该组是展开状态还是收起状态 .convertView 如果可能,重用旧的视图对象.使用前你应该保证视图对象为非空,并且是否是合适的类型.如果该对象不能转换为可以正确显示数据的视图,该方法就创建新视图.不保证使用先前由 getGroupView(int, boolean,View, ViewGroup)创建的视图.parent 该视图最终从属的父视图.返回指定位置相应的组视图.public abstract boolean hasStableIds ()是否指定分组视图及其子视图的ID对应的后台数据改变也会保持该ID.返回是否相同的ID总是指向同一个对象.public abstract boolean isChildSelectable (int groupPosition, intchildPosition)指定位置的子视图是否可选择.参数groupPosition 包含要取得子视图的分组位置.childPosition 分组中子视图的位置.返回是否子视图可选择.注意:在XML布局文件中,如果ExpandableListView上一级视图的大小没有严格定义的话,则不能对ExpandableListView的android:layout_height 属性使用wrap_content值。 (例如,如果上一级视图是ScrollView的话,则不应该指定wrap_content的值,因为它可以是任意的长度。不过,如果ExpandableListView的上一级视图有特定的大小的话,比如100像素,则可以使用wrap_content)如果由于开发的时候粗心,对ExpandableListView指定wrap_content的值,则会报一个在SetContentView处的空指针错误。