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方法中,这个想必大家都比较了解了,不赘述,请参见源代码,代码目录结构如下:
功能的实现比较简单,主要是一种方式,主要两部分:
一、在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;">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>
<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的代码行数一下减少了接近一半,整体代码阅读起来,还是会舒服一些的吧。
<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>
转载请保留原文地址“http://my.oschina.net/gluoyer/blog/182322”,谢谢!