展会信息港展会大全

android ListView事件的研究
来源:互联网   发布日期:2015-11-26 15:35:52   浏览:1045次  

导读:1. ListView的OnItemClickListener不被触发的另外一种情况 舀上图,在一个ItemView中,只有一个TextView位于最...

1. ListView的OnItemClickListener不被触发的另外一种情况

如上图,在一个ItemView中,只有一个TextView位于最左侧,他的右侧是空白区域,没有任何控件,当点击右侧区域时,并不会触发OnItemClickListener,当点击TextView所在的区域时,就能触发这个事件。

看看这个事件的执行流程

右侧空白的部分没有View控件,也就是说虽然用手指点击了这一部分,但是没有view获取焦点,Android的事件触发是从顶层view一层层往下寻找的,如果有view获取焦点,就交给这个view处理,如果没有,就交给activity处理。

click事件与touch事件的传播方式是不同的

给ListView同时添加对touch和itemClick的监听事件,他们的触发顺序是:

actionDown--------》action up---------》onItemClick

他们的执行流程为:

先按照touch事件的处理流程进行,然后在进行click事件的处理,这就说明,当用户按下屏幕,产生了两个事件,一个是touch事件,一个是click事件,添加到主线程的队列中。

用户手指触摸屏幕,滚动ListView,看起来也进行了click操作,但是结果是,只触发了touch事件,没有触发click事件。

2. ListView 获取焦点和ItemView获取焦点之间的关系

2.1 ListView不获取焦点,ItemView能获取焦点吗?

通过设置ItemView的android:focusable="true"android:focusableInTouchMode="true"属性,可以使ItemView在Touch mode 下获取焦点,默认情况下,Touch mode下ItemView,menu等等控件都是不能获取焦点的。只有ListView获取了焦点之后,ItemView才能获取焦点。

实验一:

设置ListView的focusable属性为true,ItemView的android:focusable="true"android:focusableInTouchMode="true",自定义touch事件监听器,重写onTouch方法

public boolean onTouch(View v, MotionEvent event) {

switch (event.getAction()) {

case MotionEvent.ACTION_DOWN:

//clear();

int x = (int) event.getX();

int y = (int) event.getY();

int position = mListView.pointToPosition(x, y);

int firstVisiblePosition = mListView.getFirstVisiblePosition();

view = mListView.getChildAt(position-firstVisiblePosition);

if(view==null) return false;

if(view.isFocusable()){

Log.i(tag, "ItemView is focusable ");

}

if(view.isFocusableInTouchMode()){

Log.i(tag, "ItemView is focusable in touchMode");

}

if(view.isInTouchMode()){

Log.i(tag, "device is in touch mode.");

}

if(mListView.isFocusable()){

Log.i(tag, "mListView is focusable ");

}

if(mListView.isFocusableInTouchMode()){

Log.i(tag, "mListView isFocusableInTouchMode in touchMode");

}

if(view.isFocused()){

Log.i(tag, "ItemView have get focus ");

}

if(mListView.isFocused()){

Log.i(tag, "mListView have get focus ");

}

if(view.isPressed()){

Log.i(tag, "ItemView have get pressed ");

}

break;

case MotionEvent.ACTION_UP:

if(view==null) return false;

Log.i(tag, "OnTouchListener: up is working ");

if(view.isFocused()){

Log.i(tag, "ItemView have get focus ");

}

break;

default:

break;

}

return false;

}

打印结果为:

ItemView is focusable

ItemView is focusable in touchMode

device is in touch mode.

mListView is focusable

mListView isFocusableInTouchMode in touchMode

mListView have get focus

OnTouchListener: up is working

当用户触摸屏幕的时候,触发了touch事件,但是只有ListView获取了焦点,itemView却没有获得焦点,说明itemView在默认状态下,即时设置了能获取焦点,能在touchmode下获取焦点,实际上也是不能的

而ListView的focusable属性即时不设为true,也是能够获取焦点的,那么如何让ItemView获取焦点呢,有两个方法,一个是View类的requestFocus()方法,一个是ListView的requestChildFocus(child, focused)方法,还有就是requestFocusFromTouch()方法,该方法是View类的方法,ListView继承了该方法。

我们让itemView调用requestFocus()方法或者requestFocusFromTouch()方法,这时itemView获取了焦点,但是ListView没有获取焦点,这说明一个视图中只能有一个view获取焦点。

去掉itemView的android:focusableInTouchMode="true"属性,调用requestFocusFromTouch()方法,可以强制使itemViw获取焦点。ListView依旧没有获取焦点。

总结:

1. 当使用导航键上下左右滚动时,android框架会自动让view获取焦点(获取焦点后,就会高亮显示),然而当用户用手触摸屏幕的时候,就不需要让view自动获取焦点了,也就是说,当用户点击了屏幕上的某个控件时,该控件就没有必要自动获取焦点了,因为用户知道自己操作的是哪个控件,当用户触摸手机屏幕,就会进入touch mode模式。

2. 在touch mode 模式下,有些控件是不会自动获取焦点的,但是还有些控件会,通过isFocusableInTouchMode()可以知道该控件在该touch模式下能否获得焦点,TextView是默认不能获得焦点的,ListView默认能够获得焦点,EditText等文字编辑类控件都可以。通过设置android:focusableInTouchMode="true"貌似可以使view控件获取焦点,但是实际上并不会是veiw控件获取焦点,还需要手动调用requestFocusFromTouch()方法或者requestFocus()方法才能真正获取焦点。这里推荐使用requestFocusFromTouch()方法,即时不用设置android:focusableInTouchMode="true",也能强制使控件获取焦点。

3. 获取ListView中有那个控件获取了焦点的方法

mListView.findFocus();

mListView.getFocusedChild();

4. 通过ListView的setItemsCanFocus(true)方法并不可以使ItemView在touch mode下可以获取焦点,他只是表明在由ListAdapter创建的视图中,可包含能获得焦点的项目。

滚动事件发生在touch事件的后面,这种说法是不对的,通过实验可以得到onScroll方法的执行是在touch事件之前的,并且每一次触摸屏幕,先触发这个方法,然后才触发touch事件,此外,当我们第一次进入列表界面时,onScroll方法也多次被调用,第一次是在执行onCreate方法时被调用,这事还没有生成界面,所以visibleItemCount参数为0,然后的几次调用就有了界面了,visibleItemCount也被赋予界面上显示的item的个数,显示不全的也算。差不多有两次,这两次的调用也是不一样的,看看这三次调用这个方法的不同

当用手触摸屏幕上的某一项时,也会首先触发这个方法,然后才是touch事件

通过重写onTouch方法,可以实现当用户触摸屏幕的时候,itemView获取焦点,并且变色,但是出现了两个问题:

1. 如果ListView可以显示多页,可以看到每页上又有一个ItemViw获取了焦点,不知道为什么?

推测:Adapter中getView方法的第二个蚕食convertView是复用的,估计是生成新的页面的时候复用了获取焦点的veiw,这里还要看源码

2. 当滚动到非第一页的时候,触发屏幕,并不能使触发点所在的ItemView获取焦点?

ListView的getChildAt(int index)方法的参数index,与Adapter的getView方法的第一个参数position是不一样的,每个界面显示几个item,就建立从0到界面显示个数的索引,比如一个屏幕上显示8条记录,那么索引就是

0---7,翻页了,仍然会建立类似的索引,因此应该计算出触发点的item在屏幕上的索引。

int x = (int) event.getX();

int y = (int) event.getY();

int position = mListView.pointToPosition(x, y);

int firstVisiblePosition = mListView.getFirstVisiblePosition();

view = mListView.getChildAt(position-firstVisiblePosition);

赞助本站

人工智能实验室

相关热词: android开发 教程

相关内容
AiLab云推荐
推荐内容
展开

热门栏目HotCates

Copyright © 2010-2024 AiLab Team. 人工智能实验室 版权所有    关于我们 | 联系我们 | 广告服务 | 公司动态 | 免责声明 | 隐私条款 | 工作机会 | 展会港