<> Chapter 3
View的事件体系
View的基础
-
view位置参数
-
View的位置主要由它的四个顶点来决定,分别对应于View的四个属性:
top
、left
、right
、bottom
,其中top是左上角纵坐标,left是左上角横坐标,right是右下角横坐标,bottom是右下角纵坐标, 这四个参数的坐标值都是View相对于父View的.
View的宽高和坐标的关系:width = right - left; height = bottom - top;
如何得到这四个参数:
Left = getLeft(); Right = getRight(); Top = getTop(); Bottom = getBottom();
-
从Android 3.0开始,view增加了
x
、y
、translationX
、translationY
四个参数,这几个参数也是相对于父容器的坐标. x和y是左上角的坐标,而translationX和translationY是view左上角相对于父容器的偏移量,默认值都是0.x = left + translationX y = top + translationY
View在平移过程中改变的是
移动前x
,y
,translationX
,translationY
这四个参数,left
和top
等是原始左上角的位置信息, 其值不会随着平移改变.
移动后setX()内部也是调用的setTranslationX()
setLeft()方法系统不建议我们人为调用, 因为left属性在layout()时系统会随时更改 -
View在滑动其内容时更改的是它的
mScrollX
mScrollY
这两个参数
mScrollX
的值总是等于View左边缘和View内容左边缘在水平方向的距离
mScrollY
的值总是等于View上边缘和View内容上边缘在垂直方向的距离
scrollTo()
和scrollBy()
内部其实就是更改这两个参数.
-
-
MotionEvent和TouchSlop
-
MotionEvent
在手指触摸屏幕后所产生的一系列事件中,典型的事件类型有:- ACTION_DOWN ----- 手指刚接触屏幕
- ACTION_MOVE ----- 手指在屏幕上移动
- ACTION_UP ----- 手机从屏幕上松开的一瞬间
正常情况下,一次手指触摸屏幕的行为会触发一系列点击事件,考虑如下几种情况:
- 点击屏幕后离开松开,事件序列为 DOWN -> UP
- 点击屏幕滑动一会再松开,事件序列为DOWN->MOVE->...->UP
通过MotionEvent对象我们可以得到点击事件发生的x和y坐标,getX/getY返回的是相对于当前View左上角的x和y坐标,getRawX和getRawY是相对于手机屏幕左上角的x和y坐标。
-
TouchSlop
TouchSlope是系统所能识别出的可以被认为是滑动的最小距离,获取方式是ViewConfiguration.get(getContext()).getScaledTouchSlope()
-
-
VelocityTracker、GestureDetector和Scroller
-
VelocityTracker
用于追踪手指在滑动过程中的速度,包括水平和垂直方向上的速度.
VelocityTracker的使用方式://初始化 VelocityTracker mVelocityTracker = VelocityTracker.obtain(); //在onTouchEvent方法中 mVelocityTracker.addMovement(event); //获取速度 mVelocityTracker.computeCurrentVelocity(1000); float xVelocity = mVelocityTracker.getXVelocity();//一般在MotionEvent.ACTION_UP的时候调用 //重置和回收 mVelocityTracker.clear(); //一般在MotionEvent.ACTION_UP的时候调用 mVelocityTracker.recycle(); //一般在onDetachedFromWindow中调用
-
GestureDetector
手势检测,用于辅助检测用户的点击、滑动、长按、双击等行为.我们通过查看源码,发现在GestureDetector类中封装了两个接口和一个内部类:
GestureDetector分别为
OnGestureListener
和OnDoubleTapListener
两种listener.
SimpleOnGestureListener
实现了上述两种listener
, 但是内部的实现方法都为null, 使用时根据个人需要来实现对应的方法.
GestureDetector使用方式:GestureDetector mGestureDetector = new GestureDetector(new SimpleOnGestureListener () { //实现需要用到的方法 }); mGestureDetector.setIsLongPressEnabled(false);//解决长按屏幕后无法拖动的现象. boolean consume = mGestureDetector.onTouchEvent(event);//一般在onTouchEvent中接管event return consume;
OnGestureListener
和OnDoubleTapListener
接口具体如下:public interface OnGestureListener { boolean onDown(MotionEvent e); //手指刚刚触碰屏幕的一瞬间, 由一个ACTION_DOWN触发 void onShowPress(MotionEvent e); //手指轻轻触碰屏幕, 尚未松开或拖动, 由一个ACTION_DOWN触,它和onDown的区别是它强调的是没有松开或者拖动的状态 boolean onSingleTapUp(MotionEvent e); //单击行为, 伴随着一个ACTION_UP而触发 boolean onScroll(MotionEvent e1, MotionEvent e2, float distanceX, float distanceY); //手指按下屏幕并拖动, 由一个ACTION_DOWN和多个ACTION_MOVE组成,这是拖动行为 void onLongPress(MotionEvent e); //用户长久的按着屏幕不放,即长按 boolean onFling(MotionEvent e1, MotionEvent e2, float velocityX, float velocityY); //快速滑动行为,由一个ACTION_DOWN,多个ACTION_MOVE,一个ACTION_UP触发 } public interface OnDoubleTapListener { boolean onSingleTapConfirmed(MotionEvent e); //严格的单击行为, 即这只可能是单击而不可能是双击中的一次单击 boolean onDoubleTap(MotionEvent e); //双击行为,它不可能和onSingleTapConfirmed共存 boolean onDoubleTapEvent(MotionEvent e); //表示发生了双击行为, 在双击期间ACTION_DOWN,ACTION_MOVE,ACTION_UP均会触发此回调 }
在日常开发中,比较常用的有:
onSingleTapUp(单击)
、onFling(快速滑动)
、onScroll(拖动)
、onLongPress(长按)
、onDoubleTap(双击)
.
建议:如果只是监听滑动相关的事件在onTouchEvent
中实现;如果要监听双击这种行为的话,那么就使用GestureDetector
。 -
Scroller
弹性滑动对象,用于实现View的弹性滑动。Scroller本身无法让View弹性滑动,它需要和View的computeScroll
方法配合使用才能共同完成这个功能。
Scroller
使用方式Scroller scroller = new Scroller(mContext); // 缓慢滚动到指定位置 private void smoothScrollTo(int destX, int destY) { int scrollX = getScrollX(); int delta = destX - scrollX; //1000ms内滑动到destX的位置 mScroller.startScroll(scrollX, 0, delta, 0, 1000); invalidate(); } @Override public void computeScroll() { if(mScroller.computeScrollOffset()) { scrollTo(mScroller.getCurrX(), mScroller.getCurrY()); postInvalidate(); } }
原理:
invalidate()
方法会触发computeScroll()
方法, 然后我们重写了computeScroll()
在里面调用scrollTo
来让View
移动到Scroller
计算过后的位置, 然后再次触发invalidate()
方法, 直到Scroller
计算完成。
-
View的滑动
-
使用scrollTo或scrollBy
scrollTo()
是基于所传参数的绝对滑动,scrollBy()
是基于目前所在位置的相对滑动.
scrollTo()
和scrollBy()
只能改变View内容的位置, 不能改变View在布局中的位置. -
使用动画
android中动画分为三种:View动画
帧动画
属性动画
.
我们通过View动画
和属性动画
都可以完成View的滑动, 使用动画主要操作的是View
的translationX
和translationY
这两个属性(因为setX()内部其实调用的时setTranslationX()).使用上我们需要注意以下两点:
-
view动画
操作的是控件的影像而不是view的位置参数(它不会移动view的本身也不会移动view的内容),尽管我们在视觉上看到了滑动的效果,但实际上view的位置却不曾发生改变。这点可以从如果我们不设置view的控件参数fillAftrer为true的时候,那么当动画完成后,View会瞬间恢复到动画前的效果就可以看得出来。而且,即便我们设置了fillAfter参数为true。也只是相当于把view投影到移动的位置,但当我们再要执行点击操作的时候,却是不能发生响应的。因为view的位置不会发生改变。它的真身仍在原始位置上。 -
view的属性动画
可以解决上面的问题, 但是它无法兼容3.0以下的版本.
-
-
通过改变布局参数
通过改变布局参数的方式来实现滑动,实际上改变的是LayoutParams参数,如果我们想要滑动某个控件,则直接通过修改LayoutParams参数来实现,这个方法最为简单暴力,但操作较为复杂,需要根据不同的情况去做不同的处理。使用方法如下(以移动一个Button为例):Button button = (Button) findViewById(R.id.btn_changeparams); MarginLayoutParams params = (MarginLayoutParams) button.getLayoutParams(); params.width += 100; params.leftMargin +=100; button.requestLayout();
-
三种滑动方式对比:
- scrollTo/scrollBy: 操作简单,适合对View内容的滑动
- 动画: 操作简单,主要适用于没有交互的View和实现复杂的动画效果
- 改变布局参数: 操作稍微复杂,适用于有交互的View
View弹性滑动
-
使用Scroller
上面已经介绍过了Scroller
的原理和使用方法 -
使用动画
采用这种方法除了能完成弹性滑动以外,还可以实现其他动画效果,我们完全可以在onAnimationUpdate
方法中加上我们想要的其他操作。 -
使用延时策略
使用延时策略来实现弹性滑动,它的核心思想是通过发送一系列延时消息从而达到一种渐进式的效果,具体来说可以使用Handler
的sendEmptyMessageDelayed(xxx)
或view
的postDelayed()
方法,也可以使用线程的sleep
方法。private Handler = new Handler(){ public void handleMwssage(Message msg){ switch(msg.what){ case MOVE_VIEW: //move view step handle.sendEmptyMessageDelayed(MOVE_VIEW,1000); break; } } };
View的事件分发机制
-
点击事件的传递规则
所谓点击事件的事件分发,其实就是对MotionEvent
的分发过程。当一个MotionEvent
产生之后,系统需要将其传递给某个具体的View
,比如Button
控件,并且被这个View
所消耗。整个事件分发过程由三个方法完成,分别是:- dispatchTouchEvent(MotionEvent event)
/** * 这个方法用来进行事件的分发,当MotionEvent事件传递到当前View时,便会触发当前View的这个方法, * 返回的结果受当前View的onTouchEvent和下级的dispatchTouchEvent方法的影响,表示是否消耗该MotionEvent。 * true表示被当前View所消耗,false则表示事件未被消耗。 */ public boolean dispatchTouchEvent(MotionEvent event);
- onInterceptTouchEvent(MotionEvent event)
/** * 这个方法在dispatchTouchEvent方法内部调用,用来判断是否拦截某个事件, * 如果当前View拦截了某个事件,那么在同一个事件序列中,此方法不会再被调用, * 返回结果表示是否拦截当前事件。 */ public boolean onInterceptTouchEvent(MotionEvent event);
- onTouchEvent(MotionEvent event)
/** * 这个方法在dispatchTouchEvent方法内部调用,用来处理点击事件, * 返回结果表示是否消耗当前事件,如果不消耗(ACTION_DOWN),则在同一事件序列中,当前View无法再次接收到该事件。 */ public boolean onTouchEvent(MotionEvent event);
以上三者的关系可以用伪代码进行表示:
public boolean dispatchTouchEvent(MotionEvent event){ boolean consume = false; if(onInterceptTouchEvent(event)){ consume = onTouchEvent(event); }else{ consume = childView.dispatchTouchEvent(event); } return consume; }
对于一个根
ViewGroup
来说,当产生点击事件后,首先会传递给它,此时调用它的dispatchTouchEvent
方法,如果dispatchTouchEvent
方法中的onInterceptTouchEvent(event)
返回true
,则表示这个ViewGroup
要消耗当前事件,于是调用ViewGroup
的OnTouchEvent(event)
方法。而如果onInterceptTouchEvent(event)
返回的是false
,则将该event
交给这个当前View
的子元素的dispatchTouchEvent
去处理。如此递归,直到事件被最终处理掉。当一个点击事件产生后,它的传递顺序如下:
Activity -> Window -> View
Activity是怎么接收到点击事件的请参考这篇文章
当顶级View接收到该事件后,就会将其按照事件分发机制去分发该事件,也即从父容器到子容器间层层传递,直到在某一个阶段事件被消耗完毕。但在这里存在另一个问题:如果最底层的子元素并没有消耗点击事件,怎么办?为解决这个问题,系统做了以下的措施:如果一个View的onTouchEvent方法返回的是false,那么该view的父容器的onTouchEvent方法也会被调用,以此类推,若该点击事件没有任何元素去消耗,那么最终仍是会由Activity进行处理关于事件传递的机制,有以下结论:
- 同一个事件序列是指从手指接触到屏幕的那一刻起,到手指离开屏幕的那一刻结束。期间以
Down
为开始,中间含有数量不等(可以为0)的MOVE
,最终则以UP
结束。 - 正常情况下,一个事件序列只能被一个View拦截且进行消耗。
- 某个
View
一旦决定拦截事件序列,那么这一个事件序列只能由它来处理(只要在这个view
进行拦截之前没有其他view
对这个事件序列进行拦截),并且它的onInterceptTouchEvent
方法也不会再被调用。 - 某个
View
一旦开始处理事件序列,如果它不消耗ACTION_DOWN
事件(OnTouchEvent
返回false
),那么同一个事件序列中的其他事件都不会由它来处理,而是直接将其交由父元素去处理。并且当前view
是无法再次接收到该事件的。 - 如果
View
不消耗除了ACTION_DOWN
之外的其他事件,那么这个点击事件就会消失,并且父元素的OnTouchEvent
方法也不会被调用,同时,当前View
可以持续收到后续的事件,最终这些消失的点击事件会交由Activity
进行处理。 -
ViewGroup
不拦截任何事件。Android
源码中ViewGroup
的onInterceptTouchEvent
方法默认返回false
。 - 在
Android
源码中,View
并没有onInterceptTouchEvent
方法,一旦有点击事件传递给它。那么它的OnTouchEvent
方法就会被调用。 -
view
的OnTouchEvent
默认会消耗该事件(默认返回true
),除非它是不可点击的(clickable
和longclickable
同时为false
)。 -
view
的enable
属性不影响onTouchEvent
的默认放回值。即便该view
是disable
状态的,但只要它的clickable
或longClickable
有一个为true
,那么它的返回值就为true
。 -
onclick
会发生的前提是当前View
是可点击的,并且它接收到了ACTION_DOWN
和ACTION_UP
事件。 - 事件传递过程是由外向内的,及事件总是先传递给父元素。然后再有父元素去分发给子元素。但通过
requestDisallowInterceptTouchEvent
方法可以在子元素中干预父元素的分发过程,但ACTION_DOWN
事件除外。
- dispatchTouchEvent(MotionEvent event)
-
从源码去看事件分发机制:
-
Activity分发
从上面我们知道,每个MotionEvent
都是最先交由Activity
进行的,那么我们来看看Activity
中的dispatchTouchEvent
方法public boolean dispatchTouchEvent(MotionEvent ev) { if (ev.getAction() == MotionEvent.ACTION_DOWN) { onUserInteraction(); } if (getWindow().superDispatchTouchEvent(ev)) { return true; } return onTouchEvent(ev); }
-
Window分发
我们可以看到Activity
其实是将点击事件交给了Window
进行下一步处理, 但是Window
类其实是一个抽象类, 它里面的superDispatchTouchEvent()
方法是一个抽象方法.
所以我们需要去它的唯一实现类PhoneWindow
中去查看superDispatchTouchEvent()
是如何实现的.//PhoneWindow中的superDispatchTouchEvent()方法 public boolean superDispatchTouchEvent(MotionEvent event){ return mDecor.superDispatchTouchEvent(event); }
这里的
mDecor
其实就是DecorView
,那么DecorView
是什么呢?我们来看private final class DecorView extends FrameLayout implements RootViewSurfaceTacker{ private DecorView mDecor; @override public final View getDecorView(){ if(mDecor == null){ installDecor(); } return mDecor; } }
我们知道,通过
(ViewGroup)getWindow().getDecorView().findViewById(android.R.id.content).getChildAt(0);
这种方式可以获取Activity
的所预设置的View
,而这个mDector
显然就是返回的对象。也就是说,这里的DecorView
是顶级View(ViewGroup)
,内部有titlebar
和contentParent
两个子元素,contentParent
的id
是content
,而我们设置的main.xml
布局则是contentParent
里面的一个子元素。那么,当事件传递到DecorView
这里的时候,因为DecorView
继承了FrameLayout
且还是父View
,所以最终的事件会传送到我们在setContentView()
所设置的顶级View
中。 -
ViewGroup分发
那么,现在事件已经传递到顶级View
(一个ViewGroup
)了,接下来又该是怎样的呢?逻辑思路如下:顶级View调用dispatchTouchEvent方法 if 顶级view需要拦截事件(onInterceptTouchEvent方法返回true) 处理点击事件 else 把事件传递给子元素进行处理
根据这个,我们先来看一下ViewGroup对点击事件的分发过程,其主要体现在dispatchTouchEvent方法中。因为这个方法比较长,分段说明,先看下面一段:
public boolean dispatchTouchEvent(MotionEvent ev) { //....省略 final boolean intercepted; if (actionMasked == MotionEvent.ACTION_DOWN || mFirstTouchTarget != null) { final boolean disallowIntercept = (mGroupFlags & FLAG_DISALLOW_INTERCEPT) != 0; if (!disallowIntercept) { intercepted = onInterceptTouchEvent(ev); ev.setAction(action); // restore action in case it was changed } else { intercepted = false; } } else { // There are no touch targets and this action is not an initial down // so this view group continues to intercept touches. intercepted = true; } //....省略 }
从上面的代码可以看出,
ViewGroup
会在两种情况下判断是否拦截当前事件:一是事件类型为ACTION_DOWN
,二则是mFirstTouchTarget != null
。在这里,mFirstTouchTarget
是什么意思呢? 可以这么理解:当事件由ViewGroup
的子元素成功处理时,mFirstTouchTarget
会被赋值并指向子元素。也就是说,当ViewGroup
不拦截事件并且把事件交给子元素处理时,则mFirstTouchTarget != null
。反之,如果ViewFroup
拦截了这个事件,则mFirstTouchTarget != null
就不成立, 所以当ACTION_MOVE
和ACTION_UP
事件到来时,actionMasked == MotionEvent.ACTION_DOWN || mFirstTouchTarget != null
为false
,ViewGroup
将会直接拦截事件而不会再次调用它自己的onInterceptTouchEvent(ev)方法,并且同一序列中的其他事件会交由它处理(前提是事件到达它之前没有被拦截)。对上面第3条结论的验证当然,事实无绝对,此处有一个特殊情况,就是
FLAG _DISALLOW _INTERCEPT
这个标志位,它是通过requestDisallowInterceptTouchEvent()
方法来设置的,一般用于子View
中。它一旦被设置,ViewGroup
则将无法拦截除了ACTION _DOWN
以外的其他点击事件。为什么是除了ACTION_DOWN
以外呢?public boolean dispatchTouchEvent(MotionEvent ev) { //省略... // Handle an initial down. if (actionMasked == MotionEvent.ACTION_DOWN) { // Throw away all previous state when starting a new touch gesture. // The framework may have dropped the up or cancel event for the previous gesture // due to an app switch, ANR, or some other state change. cancelAndClearTouchTargets(ev); resetTouchState(); } //省略... }
在这段源码中,
ViewGroup
会在ACTION_DOWN
事件到来时做重置状态的操作,而在resetTouchState
方法中会对FLAG _DISALLOW _INTERCEPT
进行重置,因此子View
调用requestDisallowInterceptTouchEvent
方法时并不能影响ViewGroup
对ACTION _DOWN
的影响。
接着我们再看当ViewGroup
不拦截事件的时候。事件会向下分发,交由它的子View
进行处理的过程:public boolean dispatchTouchEvent(MotionEvent ev) { // 省略...View的LONG_CLICKABLE属性默认为false,而CLICKABLE的属性则和具体的View有关。通过setClickable和setLongClickable方法可以修改这两个值。此外,在setOnClickListener中也会自动将CLICKABLE属性改为true,而setOnLongClickListener则将LONG _CLICKABLE设置为true。 final View[] children = mChildren; for (int i = childrenCount - 1; i >= 0; i--) { final int childIndex = customOrder ? getChildDrawingOrder(childrenCount, i) : i; final View child = (preorderedList == null) ? children[childIndex] : preorderedList.get(childIndex); // 如果一个child没有播放动画&&点击事件落在了它的区域内 if (!canViewReceivePointerEvents(child) || !isTransformedTouchPointInView(x, y, child, null)) { continue; } // 省略... if (dispatchTransformedTouchEvent(ev, false, child, idBitsToAssign)) { // 省略... // 这个child消耗了这个点击事件, 对mFirstTouchTarget赋值 newTouchTarget = addTouchTarget(child, idBitsToAssign); alreadyDispatchedToNewTouchTarget = true; break; } } // 省略... if (mFirstTouchTarget == null) { // 没有子View消耗了点击事件 handled = dispatchTransformedTouchEvent(ev, canceled, null, TouchTarget.ALL_POINTER_IDS); } }
从源码中,我们可以发现它的过程如下:首先遍历
ViewGroup
的所有子元素,然后判定子元素是否能够接收到点击事件(子元素是否在播动画或者点击事件的坐标是否落在子元素的区域内)。如果某个子元素满足这两个条件,那么事件就会交由它来处理。可以看到,dispatchTransformedTouchEvent
方法实际上调用的就是子元素的dispatchTouchEvent
方法。怎么看的呢?在这个方法的内部,有这么一段:private boolean dispatchTransformedTouchEvent(MotionEvent event, boolean cancel, View child, int desiredPointerIdBits) { // 省略... if (child == null) { handled = super.dispatchTouchEvent(event); } else { handled = child.dispatchTouchEvent(event); } // 省略... return handled; }
返回上一段源码,如果子元素的
dispatchTouchEvent(event)
方法返回true
,那么我们就不需考虑事件在子元素是怎么派发的,那么mFirstTouchTarget
就会被赋值,同时跳出for循环。从源码中抽取相关部分见下:newTouchTarget = addTouchTarget(child, idBitsToAssign); alreadyDispatchedToNewTouchTarget = true; break;
有人说,这段代码并没有对
mFirstTouchTarget
的赋值,因为它实际上出现在addTouchTarget
方法中,源码如下:private TouchTarget addTouchTarget(View child, int pointerIdBits) { TouchTarget target = TouchTarget.obtain(child, pointerIdBits); target.next = mFirstTouchTarget; mFirstTouchTarget = target; return target; }
从这个方法的内部结构可以看出,
mFirstTouchTarget
是以一种单链表结构,它的赋值与否直接影响到了ViewGroup
的拦截策略。接下来我们再次返回最初的源码中, 如果遍历所有的子元素事件后都没有被合适地处理,这包含两种情况:一是
ViewGroup
中没有子元素,二则是子元素处理了点击事件,但是在dispatchTouchEvent
方法中返回了false
。在这两种情况下,ViewGroup
会调用它自己的onTouchEvent()
处理点击事件:if (mFirstTouchTarget == null) { // 没有子View消耗了点击事件 handled = dispatchTransformedTouchEvent(ev, canceled, null, TouchTarget.ALL_POINTER_IDS); }
注意这一段源码的第三个参数
child
为null
,从前面的分析就可以知道,它会调用super.dispatchTouchEvent(event)
,很显然,这里就从ViewGroup
转到了View
的dispatchTouchEvent(event)
。
在随后我们对View
的dispatchTouchEvent(event)
分析中我们会发现,View
的dispatchTouchEvent(event)
会调用onTouchEvent()
方法.注意:在这时View的
dispatchTouchEvent()
中其实调用的是ViewGroup
中的onTouchEvent()
方法.
因此当一个ViewGroup
的ACTION_DOWN
事件没有被子View
消耗时, 这个ViewGroup
本身的onTouchEvent()
就会被调用来处理这个点击事件(对上面第4条结论的验证)
这时你们可能会奇怪, 为什么我们在View
的dispatchTouchEvent()
方法中调用ViewGroup
中的onTouchEvent()
方法.我们来看下面这段代码:
public class A { public void AA() { System.out.println("A.AA"); BB(); } public void BB() { System.out.println("A.BB"); } } public class B extends A { @Override public void AA() { System.out.println("B.AA"); } @Override public void BB() { System.out.println("B.BB"); } public void CC() { super.AA(); } }
我们定义两个类A和B, A和B中都有
AA
和BB
方法, 并且输出不同的Log, 那么此时我们执行new B().CC()
会输出什么结果呢?
答案是:A.AA B.BB
是不是猜错了?
Screenshot from 2018-03-08 17:19:40.png
为什么会是这样的结果呢, 因为我们是在B中调用的super.AA()
, 因此在A的AA()
方法中我们调用this
其实拿到的是一个B的引用, 如下图
所以在A的AA()
方法中我们会执行B的BB()
方法.现在是不是就明白了, 为什么我们在View的
dispatchTouchEvent()
中调用的是ViewGroup
中的onTouchEvent()
方法了? 因为View的dispatchTouchEvent()
是通过ViewGroup
调起来的. -
View分发
接下来我们回过头继续看View
的dispatchTouchEvent()
方法public boolean dispatchTouchEvent(MotionEvent event) { boolean result = false; // 省略... ListenerInfo li = mListenerInfo; if (li != null && li.mOnTouchListener != null && (mViewFlags & ENABLED_MASK) == ENABLED && li.mOnTouchListener.onTouch(this, event)) { result = true; } if (!result && onTouchEvent(event)) { result = true; } // 省略... return result; }
View
对点击事件的处理过程比较简单,因为View
是一个单独的元素,因此无法向下传递事件。所以它只能自己处理事件。从上面的源码可以看出View
对点击事件的处理过程:首先判断有没有设置onTouchListener
,如果OnTouchListener
中的onTouch
方法返回true
,那么onTouchEvent
就不会被调用,由此可见OnTouchListener
方法的优先级高于onTouchEvent
。接下来,分析
onTouchEvent
的实现。先看当View
处于不可用状态下点击事件的处理过程:public boolean onTouchEvent(MotionEvent event) { // 省略... if ((viewFlags & ENABLED_MASK) == DISABLED) { if (event.getAction() == MotionEvent.ACTION_UP && (mPrivateFlags & PFLAG_PRESSED) != 0) { setPressed(false); } // A disabled view that is clickable still consumes the touch // events, it just doesn't respond to them. return (((viewFlags & CLICKABLE) == CLICKABLE || (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)); } // 省略... }
很显然,不可用状态下的view照样会消耗点击事件,尽管它看起来不可用。
接着,再来看一下onTouchEvent方法中对点击事件的具体处理:
public boolean onTouchEvent(MotionEvent event) { // 省略... if (((viewFlags & CLICKABLE) == CLICKABLE || (viewFlags & LONG_CLICKABLE) == LONG_CLICKABLE)) { switch (event.getAction()) { case MotionEvent.ACTION_UP: // 省略... if (mPerformClick == null) { mPerformClick = new PerformClick(); } if (!post(mPerformClick)) { performClick(); } // 省略... break; case MotionEvent.ACTION_DOWN: // 省略... break; case MotionEvent.ACTION_CANCEL: // 省略... break; case MotionEvent.ACTION_MOVE: // 省略... break; } return true; } }
从源码来看,只要
View
的CLICKABLE
和LONG_CLICKABLE
有一个为true
,那么它就将消耗这个事件,即onTouchEvent
返回true
, 不管它是不是DISABLE
状态。
而当MOTION_UP
事件发生时,则触发performClick()
方法,如果View
设置了onClickListener
,那么performClick()
方法内部会调用它的onClick
方法View
的LONG_CLICKABLE
属性默认为false
,而CLICKABLE
的属性则和具体的View
有关。通过setClickable
和setLongClickable
方法可以修改这两个值。此外,在setOnClickListener
中也会自动将CLICKABLE
属性改为true
,而setOnLongClickListener
则将LONG_CLICKABLE
设置为true
。
-
view的滑动冲突
Android中的滑动冲突是比较常见的一个问题,只要在界面中内外两层同时滑动的时候,就会产生滑动。意即有一个占主导地位的View抢着去执行滑动操作,从而带来非常差的用户体验。常见的滑动冲突场景分为如下三种:
-
场景一:外部滑动方向与内部滑动方向不一致,主要是将ViewPager和Fragment配合使用所形成的页面滑动效果。在这个效果中,可以通过左右滑动来切换页面,而每个页面内部往往又是一个Listview。这种情况下本来是很容易发生滑动冲突的,但ViewPager内部处理了这种滑动冲突,所以如果使用ViewPager,则无需担心这个问题。但如果使用的是Scroller,则必须手动处理滑动冲突了。否则后果就是内外两层只能有一层能够滑动。
处理规则:当用户左右滑动时,需要让外部的View拦截点击事件。当用户上下滑动时,需要让内部View拦截点击事件。这个时候我们就可以根据它们的特征来解决滑动冲突。具体来说是:根据滑动的方向判断到底由什么来拦截事件。 -
场景二:外部滑动和内部滑动方向一致,比如ScrollView嵌套ListView,或者是ScrollView嵌套自己。表现在要么只能有一层能够滑动,要么两者滑动起来显得十分卡顿。
处理规则:从业务上寻找突破点,比如业务上有规定:当处于某种状态时需要外部View处理用户的操作,而处理另一种状态时则让内部View处理用户的操作。 -
场景三:上面两种情况的嵌套。
处理规则:同场景二
滑动冲突的解决方式:
针对场景一的滑动冲突,有两种处理滑动的解决方式:
- 外部拦截法:
所谓外部拦截法是指点击事件都先经过父容器的拦截处理,如果父容器需要此事件就拦截,如果不需要此事件就不拦截,这样就可以解决滑动冲突的问题,这个方法需要重写父容器的onInterceptTouchEvent()
方法。伪代码如下所示:@Override public boolean onInterceptTouchEvent(MotionEvent event) { boolean intercepted=false; int x=(int)event.getX(); int y=(int)event.getY(); switch (event.getAction()){ case MotionEvent.ACTION_DOWN: intercepted=false; break; case MotionEvent.ACTION_MOVE: if(父容器需要当前点击事件){ intercepted=true; }else { intercepted=false; } break; case MotionEvent.ACTION_UP: intercepted=false; break; default: break; } mLastXIntercept=x; mLastYIntercept=y; return intercepted; }
- 内部拦截法:
内部拦截法是指父容器不拦截任何事件,所有的事件传递给子元素,如果子元素需要此事件就直接消耗掉,如果不需要则交由父容器处理。需要配合requestDisallowInterceptTouchEvent()
方法才能正常工作。伪代码如下:
另外,为了使父容器不接收@Override public boolean dispatchTouchEvent(MotionEvent event) { int x=(int)event.getX(); int y=(int)event.getY(); switch (event.getAction()){ case MotionEvent.ACTION_DOWN: getParent().requestDisallowInterceptTouchEvent(true); break; case MotionEvent.ACTION_MOVE: int deltaX=x-mLastX; int deltaY=y-mLastY; if(父容器需要当前点击事件){ getParent().requestDisallowInterceptTouchEvent(false); } break; case MotionEvent.ACTION_UP: break; default: break; } mLastX=x; mLastY=y; return super.dispatchTouchEvent(event); }
ACTION_DOWN
事件,我们需要对父类进行一下修改:
以上两种方式,是针对场景一而得出的通用的解决方法。对于场景二和场景三而言,只需改变相关的滑动规则的逻辑即可。@Override public boolean onInterceptTouchEvent(MotionEvent event) { int action=event.getAction(); if (action==MotionEvent.ACTION_DOWN){ return false; }else{ return true; } }
注意:因为内部拦截法的操作较为复杂,因此推荐采用外部拦截法来处理常见的滑动冲突。