Android 高级UI设计笔记01:使用ExpandableListView组件(ListView的扩展)
1. ExpandableListView是一个用来显示二级节点的ListView。
比如如下效果的界面:
2. 使用ExpandableListView步骤:
(1)要给ExpandableListView设置适配器,那么必须先设置数据源;
(2)数据源,就是此处的适配器类ExpandableAdapter,此方法继承了BaseExpandableListAdapter,它是ExpandableListView的一个子类。需要重写里面的多个方法。getChildView()和getGroupView()方法设置自定义的布局;
(3)数据源设置好,直接给ExpandableListView.setAdapter()即可实现收缩功能。
3.下面就是我们就利用具体的案例来说明如何使用这个ExpandableListView组件:
(1)首先我们定义我们的布局文件activity_main.xml 如下:
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:orientation="vertical" tools:context="com.himi.expandablelistview.MainActivity" > <TextView android:layout_width="match_parent" android:layout_height="wrap_content" android:text="QQ通讯录" android:textSize="30sp" android:textColor="#55ff0000" android:gravity="center_horizontal" /> <ExpandableListView android:id="@+id/expandablelv" android:layout_width="match_parent" android:layout_height="wrap_content" > </ExpandableListView> </LinearLayout>
在这个activity_main.xml布局文件中,我们引入了组件ExpandableListView,布局效果图如下:
(2)定义我们自己的适配器MyExpandablelistviewAdapter,让它继承自BaseExpandableListAdapter,然后实现继承的方法。这里我们直接把数据源放在了我们自定义的类MyExpandablelistviewAdapter之中,然后绑定这个数据源到我们定的MyExpandablelistviewAdapter之中。
数据源我们定义为:
private String[] groups = {"家人","同学","同事"}; private String[][] childs = { {"老爸","老妈"}, {"刘德华","黎明","郭富城","张学友"}, {"马云","比尔盖茨", "巴菲特"} }; private int[][] childs_imgs = { { R.drawable.img1, R.drawable.img2 }, { R.drawable.img3, R.drawable.img4 , R.drawable.img5, R.drawable.img6}, { R.drawable.img7, R.drawable.img8 , R.drawable.img9} };
MVC模式:在Android项目中,业务逻辑,数据处理等担任了Model(模型)角色,XML界面显示等担任了View(视图)角色,Activity担任了Contronller(控制器)角色。contronller(控制器)是一个中间桥梁的作用,通过接口通信来协同 View(视图)和Model(模型)工作,起到了两者之间的通信作用。
这里的MyExpandablelistviewAdapter.java:
package com.himi.expandablelistview.adpater; import android.view.ViewGroup.LayoutParams; import android.content.Context; import android.view.LayoutInflater; import android.view.View; import android.view.ViewGroup; import android.widget.BaseExpandableListAdapter; import android.widget.ImageView; import android.widget.TextView; import com.himi.expandablelistview.R; /** * 自定义适配器MyExpandablelistviewAdapter * @author Administrator * */ public class MyExpandablelistviewAdapter extends BaseExpandableListAdapter { private Context context ; private String[] groups = {"家人","同学","同事"}; private String[][] childs = { {"老爸","老妈"}, {"刘德华","黎明","郭富城","张学友"}, {"马云","比尔盖茨", "巴菲特"} }; private int[][] childs_imgs = { { R.drawable.img1, R.drawable.img2 }, { R.drawable.img3, R.drawable.img4 , R.drawable.img5, R.drawable.img6}, { R.drawable.img7, R.drawable.img8 , R.drawable.img9} }; //引入一个字段context,方便Activity实例化MyExpandablelistviewAdapter public MyExpandablelistviewAdapter(Context context) { this.context = context; } //获取一级菜单的分组数目,比如这里就是3组:"我的好友","同学","同事" public int getGroupCount() { return groups.length; } //获取每个一节菜单中二级菜单的分组数目,比如"家人"中有2个条目("老爸","老妈") public int getChildrenCount(int groupPosition) { return childs[groupPosition].length; } //获取每个一级菜单子项对象 public String getGroup(int groupPosition) { return groups[groupPosition]; } //获取每个二级菜单子项对象 public String getChild(int groupPosition, int childPosition) { return childs[groupPosition][childPosition]; } //获取每个一级菜单子项对象Id public long getGroupId(int groupPosition) { return groupPosition; } //获取每个二级菜单子项对象Id public long getChildId(int groupPosition, int childPosition) { return childPosition; } /** * hasStableIds有关于MyExpandablelistviewAdapter适配器刷新顺序 * getGroupId和getChildId两个方法获取对象Id,获取到的Id,ExpandableListView会根据这个Id确定位置显示内容 * 然而Id是否有效稳定是由hasStableIds决定的,也就是说:这个方法就是判断item的id是否稳定, * 如果有自己的id也就是true,那就是稳定,否则不稳定,则根据item位置来确定id * */ public boolean hasStableIds() { return true; } //渲染一级菜单 public View getGroupView(int groupPosition, boolean isExpanded, View convertView, ViewGroup parent) { if(convertView == null) { /** * LayoutInflater是一个抽象类,它的inflate方法可以把一个xml文件转化为View对象 * 对于一个没有被载入或者想要动态载入的界面,都需要使用LayoutInflater.inflate()来载入 * 刚刚说明了:LayoutInflater是一个抽象类,要获取LayoutInflater的实例; * 获得 LayoutInflater 实例的三种方式: * 1.LayoutInflater inflater = getLayoutInflater(); //调用Activity的getLayoutInflater() * * 2.LayoutInflater localinflater =(LayoutInflater)context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); * * 3. LayoutInflater inflater = LayoutInflater.from(context); * 上面三种方法的本质是一样的 */ LayoutInflater minflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); convertView = minflater.inflate(R.layout.group_item,null); } TextView tv = (TextView) convertView.findViewById(R.id.textview_group); tv.setText(groups[groupPosition]); tv.setTextSize(25); tv.setPadding(15, 5, 0, 0); return convertView; } //渲染二级菜单 public View getChildView(int groupPosition, int childPosition, boolean isLastChild, View convertView, ViewGroup parent) { if(convertView == null) { LayoutInflater minflater = (LayoutInflater) context.getSystemService(Context.LAYOUT_INFLATER_SERVICE); convertView = minflater.inflate(R.layout.child_item,null); } ImageView iv = (ImageView) convertView.findViewById(R.id.imageview_child); TextView tv = (TextView) convertView.findViewById(R.id.textview_child); iv.setImageResource(childs_imgs[groupPosition][childPosition]); //导入的包为:import android.view.ViewGroup.LayoutParams; LayoutParams params = iv.getLayoutParams(); params.width = 200; params.height = 200; iv.setLayoutParams(params); tv.setText(childs[groupPosition][childPosition]); //记得return convertView return convertView; } //true:让子项可选 ; false:让子项不可选 public boolean isChildSelectable(int groupPosition, int childPosition) { return true; } }
(3)最后在MainActivity.java中绑定我们定义的MyExpandablelistviewAdapter(此时的适配器已经绑定了数据源,适配器可以控制其相应的显示),这使用我们还需要绑定适配器到模型Model,就是说绑定适配器到ExpandableListView:
这里的MainActivity.java:
package com.himi.expandablelistview; import android.app.Activity; import android.os.Bundle; import android.view.View; import android.widget.ExpandableListView; import android.widget.ExpandableListView.OnChildClickListener; import android.widget.Toast; import com.himi.expandablelistview.adpater.MyExpandablelistviewAdapter; public class MainActivity extends Activity { private ExpandableListView expan_listview; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); expan_listview = (ExpandableListView) findViewById(R.id.expandablelv); expan_listview.setAdapter(new MyExpandablelistviewAdapter(this)); expan_listview.setOnChildClickListener(new OnChildClickListener() { public boolean onChildClick(ExpandableListView parent, View v, int groupPosition, int childPosition, long id) { Toast.makeText(MainActivity.this, "你点击的是第"+(groupPosition+1)+"的菜单下的第"+(childPosition+1)+"选项", 0).show(); return false; } }); } }
这里我们添加了一个ExpandableListView的点击事件:expan_listview. setOnChildClickListener;当然这里还有其他很多监听事件,这里我们不多加详解,我们只要知道灵魂就行了。
(4)运行结果图如下: