ListView的Item点击事件(消息传递)

声明:博文为原创,文章内容为,效果展示,思路阐述,及代码片段。

转载请保留原文出处“http://my.oschina.net/gluoyer/blog”,谢谢!

您可以到博客的“友情链接”中,“程序猿媛(最新下载)*.*”下载最新版本,持续更新!当前版本,也可直接点击“当前1.5版本”下载。

    引子:自定义ListView的Adapter,给Item中的子控件(按钮、图标、文字等)添加点击事件,来进行点击后的不同处理,是经常要做的事情。但有些需要在处理事件中,对Activity中其他控件进行更新的话,通常把Adapter在Activity中实现,处理比较多的话,在Activity中增加几百行代码还是比较普遍的情况。 我的习惯,还是把Adapter独立出来。那么,其他控件的更新如何处理呢?

    当然,每个人的习惯不同,我这里只是阐述一种思路,如果您觉得有用,可以作为参考;其他就麻烦绕道吧。

首先,还是看下实现效果吧:


e01
 e02 e03 e04 e05

本文介绍:

    主要是通过Handler管理Message的方式,在Adapter处理点击事件时,将所需数据传送回所属Activity。在Activity中处理点击事件的后续操作的同时,又将Adapter独立了出来,代码结构就比较清晰,便于阅读和管理了。

    Adapter的重写,主要处理集中在getView方法中,这个想必大家都比较了解了,不赘述,请参见源代码,代码目录结构如下:

X7WASOJK[YLH[)A_W6$Y2%T

    功能的实现比较简单,主要是一种方式,主要两部分:

    一、在LiChildClickActivity中为LiccAdapter设置Handler

    首先,在Activity中定义Handler,用于接收点击事件返回的值:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
private Handler mHandle = new Handler() {
    @Override
    public void handleMessage(Message msg) {
        LiData data = (LiData) msg.getData().getSerializable(LiccAdapter.BUNDLE_KEY_LIDATA);
        switch(msg.what) {
            case LiccAdapter.CLICK_INDEX_ITEM:
                onItemClicked(msg.arg1, data);
                break;
            case LiccAdapter.CLICK_INDEX_COUNTRY:
                onCountryClicked(data);
                break;
            case LiccAdapter.CLICK_INDEX_NAME:
                onNameClicked(data);
                break;
        }
    }
     
};
    其次,为ListView设置带有mHandle的LiccAdapter:
1
2
3
4
5
6
7
8
9
// 获取ListView
ListView lv = (ListView) findViewById(R.id.licc_list);
// 创建数据列表      
ArrayList<LiData> list = new ArrayList<LiData>();
/**
    填充列表数据代码,请查看源码 */
// 为列表设置带mHandle参数的LiccAdapter
LiccAdapter adapter = new LiccAdapter(this, mHandle, list);
lv.setAdapter(adapter);

    这样,在LiccAdapter中,处理点击事件时发送的消息,就可以在handleMessage中接收,并处理了。而且,onNameClicked等方法中,涉及到Activity中包含的一些成员,或布局中的其他控件时,就不用担心,在LiccAdapter中无法调用了。

    二、在LiccAdapter中设置点击事件,并通过Message将数据回传

    首先,定义一些成员变量及静态变量,保存数据,对应状态:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
private Context context;
// 点击索引:点击列表项;点击按钮;点击名字。
protected final static int CLICK_INDEX_ITEM = 0;
protected final static int CLICK_INDEX_COUNTRY = 1;
protected final static int CLICK_INDEX_NAME = 2;
// 记录数据列表
private ArrayList<LiData> list;
// 记录Activity中接受消息的Handler
private Handler mHandle;
// 关键字
protected final static String BUNDLE_KEY_LIDATA = "lidata";
// 构造方法
public LiccAdapter(Context context, Handler handle, ArrayList<LiData> list) {
    this.context = context;
    mHandle = handle;
    this.list = new ArrayList<LiData>();
    for(LiData data : list) {
        this.list.add(data);
    }
}
    其次,自定义点击事件监听器:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
private class OnItemChildClickListener implements View.OnClickListener {
    // 点击类型索引,对应前面的CLICK_INDEX_xxx
    private int clickIndex;
    // 点击列表位置
    private int position;
     
    public OnItemChildClickListener(int clickIndex, int position) {
        this.clickIndex = clickIndex;
        this.position = position;
    }
 
    @Override
    public void onClick(View v) {
        // 创建Message并填充数据,通过mHandle联系Activity接收处理
        Message msg = new Message();
        msg.what = clickIndex;
        msg.arg1 = position;
        Bundle b = new Bundle();
        b.putSerializable(BUNDLE_KEY_LIDATA, list.get(position));
        msg.setData(b);
        mHandle.sendMessage(msg);
    }
     
}
    最后,在重写的getView中,为各个子元素设置监听即可:
1
2
3
4
5
6
7
8
9
10
// 这个holder,您懂的吧。
// 为按钮设置点击事件监听
holder.mCountry.setOnClickListener(
    new OnItemChildClickListener(CLICK_INDEX_COUNTRY, position));
// 为名字设置点击事件监听
holder.mName.setOnClickListener(
    new OnItemChildClickListener(CLICK_INDEX_NAME, position));
// 设置列表项的监听事件
convertView.setOnClickListener(
    new OnItemChildClickListener(CLICK_INDEX_ITEM, position));

    Ok,如此,点击列表项,及其设置了点击监听事件的子控件,就可以将带着数据的消息传回Activity了。

    当然,这里看不出什么优势,完全可以把LiccAdapter放到Activity里面实现。但实际应用中,有些Activity需要处理的事情还是比较多的,Adapter比较复杂的话,代码量也是有一些的。

    打个比方,一个2000行代码的Activity,包含两个各自500行代码的自定义Adapter,如果剥离出来,Activity的代码行数一下减少了接近一半,整体代码阅读起来,还是会舒服一些的吧。

原文地址:https://www.cnblogs.com/manmanlu/p/4365205.html