android ListView 应用及事件监听方法

 ListView :在Android应用开发过程中属于最常用的系统组件之一,当然可能童鞋们问为什么会突然游戏开发中讲这个,呵呵,其实在游戏开发中,也会常常使用到系统组件,比如游戏排行榜,简单的游戏关卡选择等等,都可以来使用ListView来实现;

    当然关于ListView我想大家都会使用了,那么这篇文章也不是跟大家讲解ListView是如果使用的,而是如果自定义通用适配器类;

    在ListView三种适配器当中,最受大家青睐的肯定就是SimpleAdapter适配器,用过的童鞋们都很清楚,它的扩展性很强,可以将ListView中每一项都使用自定义布局,插入N多组件;但是SimpleAdapter也有弱点,那就是当ListView中每一项有Button、CheckBox等这些有事件的组件,我们想监听它们就必须自定义适配器!那么今天的重点也就是来讲解一下如何写一个自定义通用适配器类!

    SimpleAdapter 构造的时候,我们知道需要五个参数来进行映射数据到ListView中,那么我们今天的自定义通用适配器其实也就是实现系统SimpleAdapter的一个自定义版;

    OK,可能我说这么多,大家还是不太懂,其实今天要讲述的自定义通用适配器优点有两点:

    1.使用通用适配器就不需要每次使用自定义适配器的时候,都要去重新去写一个,太累。。。。

    2.构造方法与SimpleAdapter构造方法相同,五个参数也一摸一样!

    3.只需要在自定义的适配器类中,将我们需要监听的组件进行设置监听即可!别的代码不需要去改动!

例如我们需要完成下图这种ListView:

(图1)android ListView 应用及事件监听方法_第1张图片


首先我们来完成ListView中每项的布局:main.xml:


  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"  
  3.     android:orientation="horizontal"  
  4.     android:layout_width="fill_parent"  
  5.     android:layout_height="fill_parent"  
  6.     >  
  7. <ImageView   
  8.     android:layout_width="wrap_content"   
  9.     android:layout_height="wrap_content"   
  10.     android:id="@+id/iv"  
  11.     />   
  12.    <LinearLayout  
  13.         android:orientation="vertical"  
  14.         android:layout_width="wrap_content"  
  15.         android:layout_height="wrap_content"  
  16.         >  
  17.         <TextView   
  18.             android:layout_width="wrap_content"   
  19.             android:layout_height="wrap_content"   
  20.             android:textSize="20sp"  
  21.             android:id="@+id/bigtv"  
  22.             />  
  23.         <TextView   
  24.             android:layout_width="wrap_content"   
  25.             android:layout_height="wrap_content"   
  26.             android:textSize="10sp"  
  27.             android:id="@+id/smalltv"  
  28.             />  
  29.     </LinearLayout>  
  30. <Button   
  31.     android:layout_width="wrap_content"   
  32.     android:layout_height="wrap_content"   
  33.      android:text="button"  
  34.     android:id="@+id/btn"  
  35.     />  
  36. <CheckBox   
  37.    android:layout_width="wrap_content"   
  38.    android:layout_height="wrap_content"   
  39.    android:id="@+id/cb"  
  40.    />  
  41. </LinearLayout>  
复制代码
修改源码:MainActivity.java:

  1. public class MainActivity extends Activity {  
  2.     private SimpleAdapter adapter;// 声明适配器对象  
  3.     private ListView listView; // 声明列表视图对象  
  4.     private List<Map<String, Object>> list;// 声明列表容器  
  5.     public static MainActivity ma;  
  6.     @Override  
  7.     public void onCreate(Bundle savedInstanceState) {  
  8.         super.onCreate(savedInstanceState);  
  9.         ma = this;  
  10.         // 实例化列表容器  
  11.         list = new ArrayList<Map<String, Object>>();  
  12.         listView = new ListView(this);// 实例化列表视图  
  13.         // 实例一个列表数据容器  
  14.         Map<String, Object> map = new HashMap<String, Object>();  
  15.         // 往列表容器中添加数据  
  16.         map.put("item1_imageivew", R.drawable.icon);  
  17.         map.put("item1_bigtv", "BIGTV");  
  18.         map.put("item1_smalltv", "SMALLTV");  
  19.         // 将列表数据添加到列表容器中  
  20.         list.add(map);  
  21.         // --使用系统适配器,无法实现组件监听;  
  22.         // //实例适配器  
  23.         adapter = new SimpleAdapter(this, list, R.layout.main, new String[] {  
  24.                 "item1_imageivew", "item1_bigtv", "item1_smalltv" }, new int[] {  
  25.                 R.id.iv, R.id.bigtv, R.id.smalltv });  
  26.         listView.setAdapter(adapter);  
  27.         // //显示列表视图  
  28.         this.setContentView(listView);  
  29.     }  
  30. }  
复制代码

到此,我们之前要求完成的 (图1)要求的ListView,[对ListView不太熟悉的童鞋自行百度google先学习一下基础吧]当然这里我们只是完成了界面,如果想监听 (图1)中的按钮和复选框事件,那么我们肯定需要自定义一个适配器,那么下面开始介绍如何实现通用适配器:创建一个新类,类名:“MySimpleAdapter.java”继承BaseAdapter:
  1. package com.himi;  
  2. import java.util.List;  
  3. import java.util.Map;  
  4. import android.app.AlertDialog;  
  5. import android.content.Context;  
  6. import android.view.LayoutInflater;  
  7. import android.view.View;  
  8. import android.view.ViewGroup;  
  9. import android.widget.BaseAdapter;  
  10. import android.widget.Button;  
  11. import android.widget.CheckBox;  
  12. import android.widget.CompoundButton;  
  13. import android.widget.ImageView;  
  14. import android.widget.TextView;  
  15. import android.widget.CompoundButton.OnCheckedChangeListener;  
  16. /**
  17. * @author Himi
  18. *  
  19. */  
  20. public class MySimpleAdapter extends BaseAdapter {  
  21.     private LayoutInflater mInflater;  
  22.     private List<Map<String, Object>> list;  
  23.     private int layoutID;  
  24.     private String flag[];  
  25.     private int ItemIDs[];  
  26.     public MySimpleAdapter(Context context, List<Map<String, Object>> list,  
  27.             int layoutID, String flag[], int ItemIDs[]) {  
  28.         this.mInflater = LayoutInflater.from(context);  
  29.         this.list = list;  
  30.         this.layoutID = layoutID;  
  31.         this.flag = flag;  
  32.         this.ItemIDs = ItemIDs;  
  33.     }  
  34.     @Override  
  35.     public int getCount() {  
  36.         // TODO Auto-generated method stub  
  37.         return list.size();  
  38.     }  
  39.     @Override  
  40.     public Object getItem(int arg0) {  
  41.         // TODO Auto-generated method stub  
  42.         return 0;  
  43.     }  
  44.     @Override  
  45.     public long getItemId(int arg0) {  
  46.         // TODO Auto-generated method stub  
  47.         return 0;  
  48.     }  
  49.     @Override  
  50.     public View getView(int position, View convertView, ViewGroup parent) {  
  51.         convertView = mInflater.inflate(layoutID, null);  
  52.         convertView = mInflater.inflate(layoutID, null);  
  53.         for (int i = 0; i < flag.length; i++) {//备注1  
  54.             if (convertView.findViewById(ItemIDs[i]) instanceof ImageView) {  
  55.                 ImageView iv = (ImageView) convertView.findViewById(ItemIDs[i]);  
  56.                 iv.setBackgroundResource((Integer) list.get(position).get(  
  57.                         flag[i]));  
  58.             } else if (convertView.findViewById(ItemIDs[i]) instanceof TextView) {  
  59.                 TextView tv = (TextView) convertView.findViewById(ItemIDs[i]);  
  60.                 tv.setText((String) list.get(position).get(flag[i]));  
  61.             }else{  
  62.                 //...备注2  
  63.             }  
  64.         }  
  65.         addListener(convertView);  
  66.         return convertView;  
  67.     }  
  68. /**
  69. * 童鞋们只需要将需要设置监听事件的组件写在下面这方法里就可以啦!
  70. * 别的不需要修改!
  71. * 备注3
  72. */  
  73.     public void addListener(View convertView) {  
  74.         ((Button)convertView.findViewById(R.id.btn)).setOnClickListener(  
  75.                 new View.OnClickListener() {  
  76.                     @Override  
  77.                     public void onClick(View v) {  
  78.                         new AlertDialog.Builder(MainActivity.ma)  
  79.                         .setTitle("自定义通用SimpleAdapter")  
  80.                         .setMessage("按钮成功触发监听事件!")  
  81.                         .show();  
  82.                     }  
  83.                 });  
  84.         ((CheckBox)convertView.findViewById(R.id.cb)).  
  85.         setOnCheckedChangeListener(new OnCheckedChangeListener() {  
  86.             @Override  
  87.             public void onCheckedChanged(CompoundButton buttonView, boolean isChecked) {  
  88.                 new AlertDialog.Builder(MainActivity.ma)  
  89.                 .setTitle("自定义通用SimpleAdapter")  
  90.                 .setMessage("CheckBox成功触发状态改变监听事件!")  
  91.                 .show();  
  92.             }  
  93.         });  
  94.     }  
  95. }  
复制代码android ListView 应用及事件监听方法_第2张图片

备注1:这个For循环中是对ListView中每一项中包含所有的组件进行判定每个组件的类型,从而去设置其数据!其中 《instanceof》这个关键字可能有的童鞋不太熟习,这个是对Object 类型的判断;这里我只是对ImageView、TextView的类型进行的数据识别, 为什么我这里只写了这两种,那是因为Button、CheckBox等这些带事件响应的组件是无法通过适配器映射到ListView上的;其实关于适配器映射的机制,这里简单说下:例如一个TextView组件,那么在ListView的每一项(List)中put()添加的时候,put()方法中第一个参数key大家知道是用于与适配器进行对应映射数据用的值,那么第二个参数其实就是put进组件的数据,所以备注1这里,当判定是一个TextView的时候,其实就是对其进行进行setText,而如果是ImageView,其实就是设置其背景图片属性setBackgroundResource! 备注2 :我这里最后还有一个else{...}这里是留给童鞋们去扩展的,因为可能还有一些其他能映射的组件,所以这里留下接口,供大家扩展; 备注3:addListener(View convertView)这是我留出来的方法,童鞋们只需要将需要设置监听事件的组件写在这方法里就可以啦!那么看一下我们使用通用监听器的效果吧:
OK,很正常!那么在来看看使用系统的SimpleAdapter 与我们自定义的MySimpleAdapter代码对比图:


怎么样!构造参数完全一样,而且我们这个比它强大,我们只要去设置下需要监听的组件监听代码就OK了。娃哈哈,好啦,今天就到这里吧,希望此通用适配器对大家有用!
这里也提醒一下开发游戏的童鞋们,很多游戏开发者认为开发游戏不用去学习系统组件的使用,不用去沾染xml、布局啥的,其实这么想的童鞋们你们就打错特错了,Android之所以能这么火,其组件的美观占了很重的份量,这么美的组件不岂不是很浪费!!希望童鞋们对组件不熟悉的游戏开发者都要去学习学习下组件的使用!

你可能感兴趣的:(游戏,android,ListView,object,layout,button)