How ScrollView works ?

ScrollView 继承自 FrameLayout,是什么让 FrameLayout 变成了可以滚动,可 fling 的呢?
是因为 onInterceptTouchEvent 和 onTouchEvent 的重写。

ScrollView 只能包含一个子视图,常常是一个线性布局。ScrollView 所关心的是,子视图是否消费了 Down 事件。不关心的是,子视图内部又经历了多少次事件分发。


当子视图不消费 DOWN 事件

onInterceptTouchEvent 函数会被调用。
(先无视掉无关紧要的代码:诸如简单判断,确保滚动结束,确保父视图不拦截事件,边缘效果,多点触控,内部滚动,越界滚动,回收工作等等。)

case MotionEvent.ACTION_DOWN: {
final int y = (int) ev.getY();
mLastMotionY = y;
mActivePointerId = ev.getPointerId(0);
}
// DOWN 事件产生时,一般 mIsBeingDragged 已经被重置为 false 了
return mIsBeingDragged;

DOWN 事件不会被拦截,在子视图中寻找能够接收到事件的是视图,让它处理。
根据假设,子视图不消费 DOWN 事件,最后由 ScrollView 的 onTouchEvent 处理。

case MotionEvent.ACTION_DOWN: {
mLastMotionY = (int) ev.getY();
mActivePointerId = ev.getPointerId(0);
break;
}
return true;

ScrollView 会消费掉 Down 事件(否则将收不到后续事件),并记录位置。

MOVE 事件

由于子视图没有消费 DOWN 事件,即 touchTarget 为 null,后续事件将被拦截并由 ScrollView 的 onTouchEvent 处理。

case MotionEvent.ACTION_MOVE: {
final int y = (int) ev.getY(activePointerIndex);
// 相对坐标,绝对距离
int deltaY = mLastMotionY - y;
// 如果移动距离大于最小距离,则标识为正在拖拽,并消耗掉这段距离
if (!mIsBeingDragged && Math.abs(deltaY) > mTouchSlop) {  
mIsBeingDragged = true;
if (deltaY > 0) {
deltaY -= mTouchSlop;
} else {
deltaY += mTouchSlop;
}}
// 如果正在拖拽
if (mIsBeingDragged) {
final int range = getScrollRange();
if (overScrollBy(0, deltaY, 0, mScrollY, 0, range, 0, mOverscrollDistance, true)) {}
break;
// !!! ---- MOVE 和 UP 的返回值有何意义? ---- !!!
return true;
protected boolean overScrollBy(int deltaX, int deltaY, int scrollX, int scrollY, int scrollRangeX, int scrollRangeY, int maxOverScrollX, int maxOverScrollY, boolean isTouchEvent) {     
int newScrollY = scrollY + deltaY;
onOverScrolled(newScrollX, newScrollY, clampedX, clampedY);
}
protected void onOverScrolled(int scrollX, int scrollY, boolean clampedX, boolean clampedY) {
super.scrollTo(scrollX, scrollY);
}

如果移动距离小于 TouchSlop,则不做任何响应。
如果移动距离大于 TouchSlop,则确认正在拖拽,并消耗这段距离。
如果正在拖拽,则更新 mScrollY 并重画。

UP 事件
case MotionEvent.ACTION_UP:
if (mIsBeingDragged) {
// 计算滑动速度,并 fling
// 下拉向上滚,所以速度是取负值
final VelocityTracker velocityTracker = mVelocityTracker;
velocityTracker.computeCurrentVelocity(1000, mMaximumVelocity);
int initialVelocity = (int) velocityTracker.getYVelocity(mActivePointerId);
if ((Math.abs(initialVelocity) > mMinimumVelocity)) {
flingWithNestedDispatch(-initialVelocity); }}
// 重置
mActivePointerId = INVALID_POINTER;
endDrag();
}
private void flingWithNestedDispatch(int velocityY) {
if (canFling) {  fling(velocityY); }
}
public void fling(int velocityY) {
int height = getHeight() - mPaddingBottom - mPaddingTop;
int bottom = getChildAt(0).getHeight();
mScroller.fling(mScrollX, mScrollY, 0, velocityY, 0, 0, 0,Math.max(0, bottom - height), 0, height/2);
}
滑动日志

4-16 13:10:15.497 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 0
04-16 13:10:15.497 5038-5038/ivolianer.viewgroup E/result: LinearLayout dispatch event 0
04-16 13:10:15.497 5038-5038/ivolianer.viewgroup E/result: ScrollView onTouch dispatch event 0
04-16 13:10:15.532 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 2
04-16 13:10:15.532 5038-5038/ivolianer.viewgroup E/result: ScrollView onTouch dispatch event 2
04-16 13:10:15.549 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 2
04-16 13:10:15.549 5038-5038/ivolianer.viewgroup E/result: ScrollView onTouch dispatch event 2
04-16 13:10:15.560 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 2
04-16 13:10:15.561 5038-5038/ivolianer.viewgroup E/result: ScrollView onTouch dispatch event 2
04-16 13:10:15.561 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 1
04-16 13:10:15.561 5038-5038/ivolianer.viewgroup E/result: ScrollView onTouch dispatch event 1


当子视图消费 DOWN 事件(比上面的情况复杂)

interceptTouchEvent 被调用

case MotionEvent.ACTION_DOWN: {   
final int y = (int) ev.getY(); 
mLastMotionY = y;
mActivePointerId = ev.getPointerId(0);
}

不拦截 DOWN 事件,寻找能接收到事件的子视图,让它处理。
根据假设,子视图消费 DOWN 事件,ScrollView 本身的 onTouchEvent 不会被调用。

MOVE 事件

因为 target 不为 null,intereceptTouchEvent 被调用。

case MotionEvent.ACTION_MOVE: {
  final int y = (int) ev.getY(pointerIndex);
  final int yDiff = Math.abs(y - mLastMotionY);
  if (yDiff > mTouchSlop && (getNestedScrollAxes() & SCROLL_AXIS_VERTICAL) == 0){
    mIsBeingDragged = true;
    mLastMotionY = y;
  }
  break;
}
return mIsBeingDragged;
如果滑动距离始终小于 TouchSlop

事件不会被拦截。
所有 MOVE 事件都会 TouchTarget 处理。
最后的 UP 事件也会传递给 TouchTarget。

日志

04-16 13:12:03.104 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 0
04-16 13:12:03.104 5038-5038/ivolianer.viewgroup E/result: LinearLayout dispatch event 0
04-16 13:12:03.126 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 2
04-16 13:12:03.126 5038-5038/ivolianer.viewgroup E/result: LinearLayout dispatch event 2
04-16 13:12:03.176 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 2
04-16 13:12:03.176 5038-5038/ivolianer.viewgroup E/result: LinearLayout dispatch event 2
04-16 13:12:04.101 5038-5038/ivolianer.viewgroup E/result: ScrollView dispatch event 1
04-16 13:12:04.101 5038-5038/ivolianer.viewgroup E/result: LinearLayout dispatch event 1

如果滑动距离大于 TouchSlop

mIsBeingDragged = true,并拦截事件。

if ((action == MotionEvent.ACTION_MOVE) && (mIsBeingDragged)) {
  return true;
}

后续的 MOVE 事件也会被拦截。

拦截事件的实质
final boolean cancelChild = resetCancelNextUpFlag(target.child) || intercepted;
if (dispatchTransformedTouchEvent(ev, cancelChild,  target.child,target.pointerIdBits)) {
handled = true;
}

把事件传递给 TouchTarget 时。
如果不拦截,正常分发事件。
如果拦截了,事件会被替换成 CANCEL 事件。

private boolean dispatchTransformedTouchEvent(MotionEvent event, boolean cancel,  View child, int desiredPointerIdBits) {
  final boolean handled; 
  final int oldAction = event.getAction();
  if (cancel || oldAction == MotionEvent.ACTION_CANCEL){
    event.setAction(MotionEvent.ACTION_CANCEL);
    if (child == null) {
      handled = super.dispatchTouchEvent(event);
    } else {
      handled = child.dispatchTouchEvent(event);
    }
  return handled;    
}

并且会重置 TouchTarget ,这会导致后续 UP 事件也无法收到。

if (canceled || actionMasked == MotionEvent.ACTION_UP || actionMasked ==MotionEvent.ACTION_HOVER_MOVE) {
  resetTouchState();
} 

日志

04-16 13:01:47.191 17678-17678/ivolianer.viewgroup E/result: ScrollView receive event 0
04-16 13:01:47.195 17678-17678/ivolianer.viewgroup E/result: LinearLayout receive event 0
04-16 13:01:47.231 17678-17678/ivolianer.viewgroup E/result: ScrollView receive event 2
04-16 13:01:47.231 17678-17678/ivolianer.viewgroup E/result: LinearLayout receive event 2
04-16 13:01:47.247 17678-17678/ivolianer.viewgroup E/result: ScrollView receive event 2
04-16 13:01:47.247 17678-17678/ivolianer.viewgroup E/result: LinearLayout receive event 3
04-16 13:01:47.264 17678-17678/ivolianer.viewgroup E/result: ScrollView receive event 2
04-16 13:01:47.264 17678-17678/ivolianer.viewgroup E/result: ScrollView onTouch receive event 2
04-16 13:01:47.328 17678-17678/ivolianer.viewgroup E/result: ScrollView receive event 2
04-16 13:01:47.328 17678-17678/ivolianer.viewgroup E/result: ScrollView onTouch receive event 2
04-16 13:01:47.328 17678-17678/ivolianer.viewgroup E/result: ScrollView receive event 1
04-16 13:01:47.328 17678-17678/ivolianer.viewgroup E/result: ScrollView onTouch receive event 1


BB了这多,总结下:

ScrollView 给了你传递 DOWN 和 UP 事件的机会,为了让子视图有机会实现点击或长按。但当你企图传递 MOVE 事件时(滑动距离大于 TouchSlop),他会无情拦截下来,并重置 TouchTarget。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,928评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,192评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,468评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,186评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,295评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,374评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,403评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,186评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,610评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,906评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,075评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,755评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,393评论 3 320
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,079评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,313评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,934评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,963评论 2 351

推荐阅读更多精彩内容