touch事件分发详解

android传递touch事件时,一般的路径是先按照view的layout层级从祖先到后代调用onInterceptTouchEvent, 然后再从后代到祖先调用onTouch(如果有OnTouchListener, 则先调用OnTouchListener, OnTouchListener返回false, 才调用onTouch)。见下图:发生在view_c区域的touch事件传递路径如下

touch事件分发详解_第1张图片
layout.png
touch事件分发详解_第2张图片
touch_event.png

下面分情况讨论一下:(handle touch指综合考虑OnTouchListeneronTouch的结果,没明确提到返回值就指返回false)

  • 假设layout层级中的所有view的onInterceptTouchEvent都返回false, 某一个view的handle touch down event返回true, 其余view的handle touch down event返回false,则这次touch up/cancel之前(包括up/cancel),所有的touch event都是经由这个view的祖先的onInterceptTouchEvent再传递给这个view的handle touch截止,无论handle touch返回什么,都不会向上传递了。比如设view_b的handle touch down event返回true,其余touch event都返回false, 则touch down之后的touch event传递路径是view_a.onInterceptTouchEvent->view_b.handle_touch
  • 假设layout层级中的某个view的onInterceptTouchEvent针对touch down返回true, 这个view的handle touch down event也返回true, 则这次touch up/cancel之前(包括up/cancel),所有的touch event都是经由这个view的祖先的onInterceptTouchEvent再传递给这个view的handle touch截止,无论handle touch返回什么,都不会向上传递了。比如设view_b的handle touch down event返回true,无论其余handle touch返回什么, touch down之后的touch event传递路径都是是view_a.onInterceptTouchEvent->view_b handle_touch。而touch down的传递路径是view_a.onInterceptTouchEvent->view_b.onInterceptTouchEvent->view_b.handle_touch
  • 如果layout层级中的所有view的handle touch down event都返回false, 则touch up/cancel前(包括up/cancel)所有的后续touch event都不会再传递给这个layout层级中的view了,即使某个view的onInterceptTouchEvent针对touch down event返回true也无济于事。
  • 如果layout层级中某个view的handle touch down event返回true, 这个view的某个祖先的onInterceptTouchEvent对于touch down也返回true, 但这个祖先的handle touch down event返回false, 则touch up/cancel前(包括up/cancel)所有的后续touch event都不会再传递给这个layout层级中的view了。比如设view_c的handle touch down event返回true,其余所有view的handle touch down event都返回false,view_b.onInterceptTouchEvent返回true,则touch up/cancel前(包括up/cancel)所有的后续touch event都不会再传递给view_a,view_b,view_c中的任何一个。而touch down的传递路径是view_a.onInterceptTouchEvent->view_b.onInterceptTouchEvent->view_b.handle_touch->view_a.handle_touch
  • 如果layout层级中某个view的handle touch down event返回true, 这个view的某个祖先的onInterceptTouchEvent对于touch down以外的其他touch event也返回true, 无论这个祖先的handle touch都返回什么, 比如设view_c的handle touch down event返回true,其余所有view的handle touch down event都返回false,view_b.onInterceptTouchEvent针对非touch down event返回true,则touch down的传递路径是view_a.onInterceptTouchEvent->view_b.onInterceptTouchEvent->->view_c.onInterceptTouchEvent->view_c.handle_touch。而之后view_c会立刻收到一个touch cancel事件,所有的后续touch event都不会再传递给view_c,在touch down之后紧随的touch event是沿着view_a.onInterceptTouchEvent->view_b.onInterceptTouchEvent->view_b.handle_touch的路径传递,再之后的touch event则是沿着view_a.onInterceptTouchEvent->view_b.handle_touch的路径传递。
  • 如果layout层级中某个view的onInterceptTouchEvent总是返回true, handle touch down event也是返回true, 其他handle touch的返回值对touch传递的路径没有影响。比如view_b.onInterceptTouchEvent总是返回true, view_b.handle_touch针对touch down返回true, 则touch down的路径是view_a.onInterceptTouchEvent->view_b.onInterceptTouchEvent->view_b.handle_touch,其他touch event的路径是view_a.onInterceptTouchEvent->view_b.handle_touch

综上,如果想要在父view里拦截某些特殊的手势,一般不建议在父view的onInterceptTouchEvent拦截touch down事件,否则子类无法响应这些特殊手势之外的其他手势。而且一旦父view的onInterceptTouchEvent返回true了,则后续的touch事件是传递给父view的OnTouchListeneronTouch了,所以一旦onInterceptTouchEvent返回true后,要在父view的OnTouchListeneronTouch里继续处理需要拦截的touch事件,onInterceptTouchEvent已经不会被调用了。

你可能感兴趣的:(touch事件分发详解)