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);