android传递touch事件时,一般的路径是先按照view的layout层级从祖先到后代调用onInterceptTouchEvent
, 然后再从后代到祖先调用onTouch
(如果有OnTouchListener
, 则先调用OnTouchListener
, OnTouchListener
返回false, 才调用onTouch
)。见下图:发生在view_c区域的touch事件传递路径如下
下面分情况讨论一下:(handle touch指综合考虑OnTouchListener
和onTouch
的结果,没明确提到返回值就指返回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的OnTouchListener
或onTouch
了,所以一旦onInterceptTouchEvent
返回true后,要在父view的OnTouchListener
或onTouch
里继续处理需要拦截的touch事件,onInterceptTouchEvent
已经不会被调用了。