ListView事件的研究
如上图,在一个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);
ListView事件的研究的更多相关文章
- 解决ScrollView与ListView事件冲突
1,在最近做项目的时候使用ScrollView嵌套ListView的时候发现ListView的滑动效果失效,简单的网上搜索了一下,也就有了下面的解决方法,在ListView中设置事件的监听listvi ...
- 通过布局文件来显示ListView内容并注册 ListView事件
1:layout/vlist.xml是我们的布局文件,在这里一定要对首节点加上 android:descendantFocusability="blocksDescendants" ...
- listview复用机制研究
Listview在第一次的时候会先把屏幕上绘制的item都new出来,为了讲解方便我把new出来的item都用红色背景,复用的则用绿色背景. 可以看到这个list种有三种item.在第一次展示的时候, ...
- RecyclerView的滚动事件OnScrollListener研究
(1)滚动事件分类 列表的滚动一般分为两种: 1.手指按下 -> 手指拖拽列表移动 -> 手指停止拖拽 -> 抬起手指 2.手指按下 -> 手指快速拖拽后抬起手指 -> ...
- 利用RTTI实现Delphi的多播事件代理研究
我们知道Delphi的每个对象可以包含多个Property,Property中可以是方法,例如TButton.OnClick属性.Delphi提供的仅仅是 一对一的设置,无法直接让TButton.On ...
- 对于移动端浏览器touch事件的研究总结(4)判断手指滑动方向
最近有一些微信的项目,虽然页面很简单,但配合手势后的效果却是很不错的.最基本的效果就是手指向上滑,页面配合css3出现一个展开效果,手指向下滑将展开的内容按原路径收起.其实就是一个简单的判断手指滑动方 ...
- 移动端浏览器touch事件的研究总结
$("body").on("touchstart", function(e) { e.preventDefault(); startX = e. ...
- ListView介绍
原文:http://blog.csdn.net/qingye_love/article/details/13772391?utm_source=tuicool&utm_medium=refer ...
- listview选中滑动时背景变黑
喵的今天调这个一直以为是背景色的问题,花了好多时间 下面才是解决方法:转自:http://daijun74.iteye.com/blog/1175143 手指在ListView上下滚动时,ListVi ...
随机推荐
- input text文本框内部最后面放一个按钮
.ContSpan { border: 1px solid #; display: inline-block; } .ContSpan span { cursor: pointer; backgrou ...
- php hash算法
任意长度的输入, 固定长度的输出 ,该输出就是hash值,这种转换就是一种压缩映射,也就是hash值的空间远远小于输入的空间, 不同的输入可能散列成相同的输出,而不能从hash值来唯一的确定输入值. ...
- php中in_array使用注意
可能会导致长耗时: http://www.jb51.net/article/41446.htm
- 20155305乔磊2016-2017-2《Java程序设计》第五周学习总结
20155305乔磊2016-2017-2<Java程序设计>第五周学习总结 教材学习内容总结 try.catch 1.求平均数程序示例 import java.util.Scanner; ...
- RPC和RabbitMQ
在单台机器或者单个进程中,如果要调用某个函数,只需要通过函数指针,传入相关参数,即可调用成功并获得结果.但如果是在分布式系统中,某个进程想要调用远程机器上的其它进程提供的方法(服务),就需要采用RPC ...
- js 变量 作用域及内存
由于Javascript是松散型的,所以其变量只是在特定时间用于保存特定值的一个名字而已,并不存在某个变量必须保存某种类型的值的规则,变量的值以及其数据类型都可以在脚本的声明周期内改变 一.基本类型与 ...
- Linux下Oracle常用命令
1. 备份表 exp database_user/pass tables='(table1,table2)' file=filename.dmp(例如:exp ismrenbao/iflytek ta ...
- nginx做http向https的自动跳转
在访问百度时,在浏览器输入www.baidu.com会自动跳转到https://www.baidu.com不用人工干预,nginx也可以做这样的自动跳转! 首先让nginx服务器监听两个端口,分别是8 ...
- DBCP、c3p0、Druid三大连接池区别
DBCP.c3p0.Druid三大连接池区别 一.连接池优势 如果一个项目中如果需要多个连接,如果一直获取连接,断开连接,这样比较浪费资源: 如果创建一个池,用池来管理Connection,这样就可以 ...
- Eclipse编码规范——Code Templates设置
Eclipse编码规范——Code Templates设置 Eclipse编码规范主要包括三个方面:设置Code Templates.Eclipse formatter.Checkstyle, 本篇主 ...