【Android游戏开发二十三】自定义ListView【通用】适配器并实现监听控件!

来源:互联网 发布:被狙击的学园 知乎 编辑:程序博客网 时间:2024/05/16 01:21

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

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

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

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

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

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

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

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

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

(图1)

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

main.xml:

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

修改源码:MainActivity.java:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
publicclass MainActivity extendsActivity {
    privateSimpleAdapter adapter;// 声明适配器对象
    privateListView listView; // 声明列表视图对象
    privateList<Map<String, Object>> list;// 声明列表容器
    publicstatic MainActivity ma;
    @Override
    publicvoid onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        ma = this;
        // 实例化列表容器
        list = newArrayList<Map<String, Object>>();
        listView = newListView(this);// 实例化列表视图
        // 实例一个列表数据容器
        Map<String, Object> map = newHashMap<String, Object>();
        // 往列表容器中添加数据
        map.put("item1_imageivew", R.drawable.icon);
        map.put("item1_bigtv","BIGTV");
        map.put("item1_smalltv","SMALLTV");
        // 将列表数据添加到列表容器中
        list.add(map);
        // --使用系统适配器,无法实现组件监听;
        // //实例适配器
        adapter = newSimpleAdapter(this, list, R.layout.main, newString[] {
                "item1_imageivew","item1_bigtv","item1_smalltv"}, newint[] {
                R.id.iv, R.id.bigtv, R.id.smalltv });
        listView.setAdapter(adapter);
        // //显示列表视图
        this.setContentView(listView);
    }
}

到此,我们之前要求完成的(图1)要求的ListView,[对ListView不太熟悉的童鞋自行百度google先学习一下基础吧]

当然这里我们只是完成了界面,如果想监听(图1)中的按钮和复选框事件,那么我们肯定需要自定义一个适配器,那么下面开始介绍如何实现通用适配器:

创建一个新类,类名:“MySimpleAdapter.java”继承BaseAdapter:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
/**
 *
 */
packagecom.himi;
importjava.util.List;
importjava.util.Map;
importandroid.app.AlertDialog;
importandroid.content.Context;
importandroid.view.LayoutInflater;
importandroid.view.View;
importandroid.view.ViewGroup;
importandroid.widget.BaseAdapter;
importandroid.widget.Button;
importandroid.widget.CheckBox;
importandroid.widget.CompoundButton;
importandroid.widget.ImageView;
importandroid.widget.TextView;
importandroid.widget.CompoundButton.OnCheckedChangeListener;
/**
 * @author Himi
 *
 */
publicclass MySimpleAdapter extendsBaseAdapter {
    privateLayoutInflater mInflater;
    privateList<Map<String, Object>> list;
    privateint layoutID;
    privateString flag[];
    privateint ItemIDs[];
    publicMySimpleAdapter(Context context, List<Map<String, Object>> list,
            intlayoutID, String flag[], intItemIDs[]) {
        this.mInflater = LayoutInflater.from(context);
        this.list = list;
        this.layoutID = layoutID;
        this.flag = flag;
        this.ItemIDs = ItemIDs;
    }
    @Override
    publicint getCount() {
        // TODO Auto-generated method stub
        returnlist.size();
    }
    @Override
    publicObject getItem(intarg0) {
        // TODO Auto-generated method stub
        return0;
    }
    @Override
    publiclong getItemId(intarg0) {
        // TODO Auto-generated method stub
        return0;
    }
    @Override
    publicView getView(intposition, View convertView, ViewGroup parent) {
        convertView = mInflater.inflate(layoutID, null);
        for(inti = 0; i < flag.length; i++) {//备注1
            if(convertView.findViewById(ItemIDs[i]) instanceofImageView) {
                ImageView iv = (ImageView) convertView.findViewById(ItemIDs[i]);
                iv.setBackgroundResource((Integer) list.get(position).get(
                        flag[i]));
            }elseif (convertView.findViewById(ItemIDs[i]) instanceofTextView) {
                TextView tv = (TextView) convertView.findViewById(ItemIDs[i]);
                tv.setText((String) list.get(position).get(flag[i]));
            }else{
                //...备注2
            }
        }
        addListener(convertView);
        returnconvertView;
    }
/**
 * 童鞋们只需要将需要设置监听事件的组件写在下面这方法里就可以啦!
 * 别的不需要修改!
 * 备注3
 */
    publicvoid addListener(View convertView) {
        ((Button)convertView.findViewById(R.id.btn)).setOnClickListener(
                newView.OnClickListener() {
                    @Override
                    publicvoid onClick(View v) {
                        newAlertDialog.Builder(MainActivity.ma)
                        .setTitle("自定义通用SimpleAdapter")
                        .setMessage("按钮成功触发监听事件!")
                        .show();
                    }
                });
        ((CheckBox)convertView.findViewById(R.id.cb)).
        setOnCheckedChangeListener(newOnCheckedChangeListener() {
            @Override
            publicvoid onCheckedChanged(CompoundButton buttonView, booleanisChecked) {
                newAlertDialog.Builder(MainActivity.ma)
                .setTitle("自定义通用SimpleAdapter")
                .setMessage("CheckBox成功触发状态改变监听事件!")
                .show();
            }
        });
    }
}

备注1:这个For循环中是对ListView中每一项中包含所有的组件进行判定每个组件的类型,从而去设置其数据!

其中 《instanceof》这个关键字可能有的童鞋不太熟习,这个是对Object 类型的判断;

这里我只是对ImageView、TextView的类型进行的数据识别,为什么我这里只写了这两种,那是因为Button、CheckBox等这些带事件响应的组件是无法通过适配器映射到ListView上的;

其实关于适配器映射的机制,这里简单说下:例如一个TextView组件,那么在ListView的每一项(List)中put()添加的时候,put()方法中第一个参数key大家知道是用于与适配器进行对应映射数据用的值,那么第二个参数其实就是put进组件的数据;其实当其数据反射在ListViw时,其实内部就是对组件进行实例化,并且对组件设置数据;

备注2 :我这里最后还有一个else{…}这里是留给童鞋们去扩展的,因为可能还有一些其他能映射的组件,所以这里留下接口,供大家扩展;

备注3:addListener(View convertView)这是我留出来的方法,童鞋们只需要将需要设置监听事件的组件写在这方法里就可以啦!

那么看一下我们使用通用监听器的效果吧:

OK,很正常!那么在来看看使用系统的SimpleAdapter 与我们自定义的MySimpleAdapter代码对比图:

怎么样!构造参数完全一样,而且我们这个比它强大,我们只要去设置下需要监听的组件监听代码就OK了。

娃哈哈,好啦,今天就到这里吧,希望此通用适配器对大家有用!

补充:大家使用自定义适配器的时候,有时候ListView每一项的焦点没有了,比如本文中是因为Button和CheckBox截获了焦点,童鞋们只要将button和checkBox的焦点设置不可见就OK啦。~

  xml中focusable是这个属性;    android:focusable=”false”

这里也提醒一下开发游戏的童鞋们,很多游戏开发者认为开发游戏不用去学习系统组件的使用,不用去沾染xml、布局啥的,其实这么想的童鞋们你们就大错特错了,Android之所以能这么火,其组件的美观占了很重的份量,这么美的组件不用岂不是很浪费!!希望童鞋们对组件不熟悉的游戏开发者都要去学习学习下组件的使用!

    源码下载:”Himi-ListView.rar”                下载地址:  http://vdisk.weibo.com/s/hq1uh

原创粉丝点击