Android listView 中响应Button点击事件

1、ListVIew点击事件失效====================================================================================

在写ListView的点击事件时OnItemClickListener,onItemClick方法没有执行,导致ListView条目点击事件失效,检查发现百度上有很多不同的答案,但究其本质都是ListView的Item抢占焦点或者Item没有获取焦点甚至没有绑定上OnItemClickListener监听事件,而我所犯的错误是在ListView的Item布局中引入了一个Style,在Style中有一项<item name="android:clickable">true</item>,正是这一项导致所有Item都要抢占焦点,所以ListView的点击事件失效,在我去掉这一项之后ListView确实正常工作了。需要引以为戒的是,在androidl应用开发中,焦点没有获取或者其他组件抢占焦点的事情经常发生,我们可以在代码中,xml布局中,甚至Style中定义时候抢占焦点,在一般情况下,这个设置并不会造成什么异常,但我需要注意重要的组件在合适的时机必须拿到焦点,否则会产生意想不到的后果,比如我的ListView。一般组件获取焦点可以使用一下方法:

 

View.setFocusable(true),对应xml : android:focusable="true".                                
View.setFocusableInTouchMode(true),对应xml : android:focusableInTouchMode="true". 


注意:这两个属性要同时使用。

两者的意思是让组件可以获得焦点。不过有些区别,前者执行false条件后,在执行true,还是不能获取焦点。后者执行上述过程,还是能获取焦点。 
当你加入上述代码后,在创建activity时,调用对应view的requestFocus(),(requestFocus()需要在setContentView之后执行)这样就可以获得焦点了。当editText失去焦点了,也就不会有软键盘了

 

但针对ListView还可以使用android:descendantFocusability属性,下面我们来看一下android:descendantFocusability用法简析

以下摘自:http://www.cnblogs.com/eyu8874521/archive/2012/10/17/2727882.html

 

       开发中很常见的一个问题,项目中的listview不仅仅是简单的文字,常常需要自己定义listview,自己的Adapter去继承BaseAdapter,在adapter中按照需求进行编写,问题就出现了,可能会发生点击每一个item的时候没有反应,无法获取的焦点。原因多半是由于在你自己定义的Item中存在诸如ImageButton,Button,CheckBox等子控件(也可以说是Button或者Checkable的子类控件),此时这些子控件会将焦点获取到,所以常常当点击item时变化的是子控件,item本身的点击没有响应。

这时候就可以使用descendantFocusability来解决啦,API描述如下:

android:descendantFocusability

Defines the relationship between the ViewGroup and its descendants when looking for a View to take focus.

Must be one of the following constant values.

 

该属性是当一个为view获取焦点时,定义viewGroup和其子控件两者之间的关系。

属性的值有三种:

        beforeDescendants:viewgroup会优先其子类控件而获取到焦点

        afterDescendants:viewgroup只有当其子类控件不需要获取焦点时才获取焦点

        blocksDescendants:viewgroup会覆盖子类控件而直接获得焦点

通常我们用到的是第三种,即在Item布局的根布局加上android:descendantFocusability=”blocksDescendants”的属性就好了,至此listview点击的灵异事件告一段落。心得:遇到不会不懂的地方除了网上查询资料之外,也可以多多去尝试每种属性的作用。


2、======================================================================================================


在getView中给button添加接口
Java代码  
public class Task_list_single_item_adapter extends BaseAdapter   
{   
  
    LayoutInflater inflater;   
    Context context;   
    public Task_list_single_item_adapter(Context context){   
        this.context=context;   
        inflater=LayoutInflater.from(context);   
    }   
    @Override  
    public int getCount()   
    {   
        return 2;   
    }   
  
    @Override  
    public Object getItem(int position)   
    {   
        return null;   
    }   
  
    @Override  
    public long getItemId(int position)   
    {   
        return position;   
    }   
  
    @Override  
    public View getView(int position, View convertView, ViewGroup parent)   
    {   
        final Holder holder;   
        if(convertView!=null)   
        {   
            holder=(Holder) convertView.getTag();   
        }else {   
            holder=new Holder();   
            convertView=inflater.inflate(R.layout.download_management_item, null);   
            holder.pause=(Button) convertView.findViewById(R.id.dmi_ib1_pause);   
            holder.trys=(Button) convertView.findViewById(R.id.dmi_ib2_recover);   
            holder.install=(Button) convertView.findViewById(R.id.dmi_ib3_install);   
            convertView.setTag(holder);   
        }   
        OnClickListener listener=new OnClickListener(){   
            @Override  
            public void onClick(View v)   
            {   
                if(v==holder.pause){   
                    Toast.makeText(context, "pause", Toast.LENGTH_SHORT).show();   
                    holder.pause.setVisibility(View.GONE);   
                    holder.trys.setVisibility(View.VISIBLE);   
                    holder.install.setVisibility(View.GONE);   
                }   
                if(v==holder.trys){   
                    Toast.makeText(context, "try", Toast.LENGTH_SHORT).show();   
                    holder.pause.setVisibility(View.VISIBLE);   
                    holder.trys.setVisibility(View.GONE);   
                    holder.install.setVisibility(View.GONE);   
                }   
                if(v==holder.install)   
                    Toast.makeText(context, "install", Toast.LENGTH_SHORT).show();   
                       
            }   
        };   
        holder.pause.setOnClickListener(listener);   
        holder.trys.setOnClickListener(listener);   
        holder.install.setOnClickListener(listener);   
        return convertView;   
    }   
    class Holder{   
        public Button pause;   
        public Button trys;   
        public Button install;   
    }   
} 


3、================================================================================================


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

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

X7WASOJK[YLH[)A_W6$Y2%T

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

    一、在LiChildClickActivity中为LiccAdapter设置Handler
    首先,在Activity中定义Handler,用于接收点击事件返回的值:
privateHandler mHandle = newHandler() {
    @Override
    publicvoidhandleMessage(Message msg) {
        LiData data = (LiData) msg.getData().getSerializable(LiccAdapter.BUNDLE_KEY_LIDATA);
        switch(msg.what) {
            caseLiccAdapter.CLICK_INDEX_ITEM:
                onItemClicked(msg.arg1, data);
                break;
            caseLiccAdapter.CLICK_INDEX_COUNTRY:
                onCountryClicked(data);
                break;
            caseLiccAdapter.CLICK_INDEX_NAME:
                onNameClicked(data);
                break;
        }
    }
     
};
其次,为ListView设置带有mHandle的LiccAdapter:

// 获取ListView
ListView lv = (ListView) findViewById(R.id.licc_list);
// 创建数据列表       
ArrayList<LiData> list = newArrayList<LiData>();
/**
    填充列表数据代码,请查看源码 */
// 为列表设置带mHandle参数的LiccAdapter
LiccAdapter adapter = newLiccAdapter(this, mHandle, list);
lv.setAdapter(adapter);

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

    二、在LiccAdapter中设置点击事件,并通过Message将数据回传
    首先,定义一些成员变量及静态变量,保存数据,对应状态:


<span style="font-weight: normal;">privateContext context;
// 点击索引:点击列表项;点击按钮;点击名字。
protectedfinalstatic int CLICK_INDEX_ITEM = 0;
protectedfinalstatic int CLICK_INDEX_COUNTRY = 1;
protectedfinalstatic int CLICK_INDEX_NAME = 2;
// 记录数据列表
privateArrayList<LiData> list;
// 记录Activity中接受消息的Handler
privateHandler mHandle;
// 关键字
protectedfinalstatic String BUNDLE_KEY_LIDATA = "lidata";
// 构造方法
publicLiccAdapter(Context context, Handler handle, ArrayList<LiData> list) {
    this.context = context;
    mHandle = handle;
    this.list = newArrayList<LiData>();
    for(LiData data : list) {
        this.list.add(data);
    }
}</span>
其次,自定义点击事件监听器:


<span style="font-weight: normal;">privateclassOnItemChildClickListener implementsView.OnClickListener {
    // 点击类型索引,对应前面的CLICK_INDEX_xxx
    privateintclickIndex;
    // 点击列表位置
    privateintposition;
     
    publicOnItemChildClickListener(intclickIndex,intposition) {
        this.clickIndex = clickIndex;
        this.position = position;
    }
 
    @Override
    publicvoidonClick(View v) {
        // 创建Message并填充数据,通过mHandle联系Activity接收处理
        Message msg = newMessage();
        msg.what = clickIndex;
        msg.arg1 = position;
        Bundle b = newBundle();
        b.putSerializable(BUNDLE_KEY_LIDATA, list.get(position));
        msg.setData(b);
        mHandle.sendMessage(msg);
    }
}</span>
最后,在重写的getView中,为各个子元素设置监听即可:
<span style="font-weight: normal;">// 这个holder,您懂的吧。  
// 为按钮设置点击事件监听  
holder.mCountry.setOnClickListener(  
    newOnItemChildClickListener(CLICK_INDEX_COUNTRY, position));  
// 为名字设置点击事件监听  
holder.mName.setOnClickListener(  
    newOnItemChildClickListener(CLICK_INDEX_NAME, position));  
// 设置列表项的监听事件  
convertView.setOnClickListener(  
    newOnItemChildClickListener(CLICK_INDEX_ITEM, position));  </span>

 Ok,如此,点击列表项,及其设置了点击监听事件的子控件,就可以将带着数据的消息传回Activity了。
    当然,这里看不出什么优势,完全可以把LiccAdapter放到Activity里面实现。但实际应用中,有些Activity需要处理的事情还是比较多的,Adapter比较复杂的话,代码量也是有一些的。
    打个比方,一个2000行代码的Activity,包含两个各自500行代码的自定义Adapter,如果剥离出来,Activity的代码行数一下减少了接近一半,整体代码阅读起来,还是会舒服一些的吧。

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


评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值