前言
1. Android 事件分發(fā)流
??:圖分為三層藻茂,從上到下依次是 Activity、ViewGroup、View
事件從左上角那個(gè)白色箭頭開始骗灶,由 Activity 的 dispatchTouchEvent 做分發(fā)
箭頭的上面字代表方法返回值(return false、return true啥么、return super.xxxx()),super 的意思是調(diào)用父類實(shí)現(xiàn)驻啤。
dispatchTouchEvent 和 onTouchEvent的框里有個(gè)【true---->消費(fèi)】的字,表示的意思是如果方法返回true道批,那么代表事件就此消費(fèi)错英,不會(huì)繼續(xù)往別的地方傳了,事件終止隆豹。
目前所有的圖說明的事件是針對(duì) ACTION_DOWN 的椭岩,對(duì)于 ACTION_MOVE 和 ACTION_UP ,當(dāng)dispatchTouchEvent在進(jìn)行事件分發(fā)的時(shí)候璃赡,只有前一個(gè)事件(如ACTION_DOWN)返回true判哥,才會(huì)收到ACTION_MOVE和ACTION_UP的事件。
仔細(xì)看整個(gè)圖碉考,我們得出事件流 走向的幾個(gè)結(jié)論:
- 如果事件不被中斷塌计,整個(gè)事件流向是一個(gè)類U型圖,我們來看下這張圖侯谁,可能更能理解U型圖的意思锌仅。
所以如果我們沒有對(duì)控件里面的方法進(jìn)行重寫或更改返回值,而直接用 super 調(diào)用父類的默認(rèn)實(shí)現(xiàn)墙贱,那么整個(gè)事件流向應(yīng)該是從Activity---->ViewGroup--->View 從上往下調(diào)用 dispatchTouchEvent 方法技扼,一直到葉子節(jié)點(diǎn)(View)的時(shí)候,再由 View--->ViewGroup--->Activity 從下往上調(diào)用 onTouchEvent 方法嫩痰。
- dispatchTouchEvent 和 onTouchEvent 一旦return true剿吻,事件就停止傳遞了(到達(dá)終點(diǎn))〈模看下圖中只要 return true 事件就沒再繼續(xù)傳下去了丽旅,對(duì)于 return true 我們經(jīng)常說事件被消費(fèi)了椰棘,消費(fèi)了的意思就是事件走到這里就是終點(diǎn),不會(huì)往下傳榄笙,沒有誰能再收到這個(gè)事件了邪狞。
- dispatchTouchEvent 和 onTouchEvent return false 的時(shí)候事件都回傳給父控件的 onTouchEvent 處理。
看上圖深藍(lán)色的線茅撞,對(duì)于返回 false 的情況帆卓,事件都是傳給父控件 onTouchEvent 處理。
??:對(duì)于 dispatchTouchEvent 返回 false 的含義應(yīng)該是:事件停止往子 View 傳遞和分發(fā)同時(shí)開始往父控件回溯(父控件的 onTouchEvent 開始從下往上回傳直到某個(gè) onTouchEvent return true)米丘,事件分發(fā)機(jī)制就像遞歸剑令,return false 的意義就是遞歸停止然后開始回溯。
對(duì)于 onTouchEvent return false 就比較簡(jiǎn)單了拄查,它就是不消費(fèi)事件吁津,并讓事件繼續(xù)往父控件的方向從下往上流動(dòng)。
- dispatchTouchEvent堕扶、onTouchEvent碍脏、onInterceptTouchEvent
ViewGroup 和 View 的這些方法的默認(rèn)實(shí)現(xiàn)就是會(huì)讓整個(gè)事件安裝 U 型完整走完,所以 return super.xxxxxx() 就會(huì)讓事件依照U型的方向的完整走完整個(gè)事件流動(dòng)路徑)稍算,中間不做任何改動(dòng)典尾,不回溯、不終止糊探,每個(gè)環(huán)節(jié)都走到急黎。
所以如果看到方法 return super.xxxxx() 那么事件的下一個(gè)流向就是走 U 型下一個(gè)目標(biāo),稍微記住上面這張圖侧到,你就能很快判斷出下一個(gè)走向是哪個(gè)控件的哪個(gè)函數(shù)。
- onInterceptTouchEvent 的作用
Intercept 的意思就攔截淤击,每個(gè) ViewGroup 每次在做分發(fā)的時(shí)候匠抗,問一問攔截器要不要攔截(也就是問問自己這個(gè)事件要不要自己來處理)如果要自己處理那就在 onInterceptTouchEvent 方法中 return true 就會(huì)交給自己的 onTouchEvent 的處理,如果不攔截就是繼續(xù)往子控件往下傳污抬。默認(rèn)是不會(huì)去攔截的汞贸,因?yàn)樽?View 也需要這個(gè)事件,所以 onInterceptTouchEvent 攔截器 return super.onInterceptTouchEvent() 和 return false 是一樣的印机,是不會(huì)攔截的矢腻,事件會(huì)繼續(xù)往子 View 的 dispatchTouchEvent 傳遞。
- ViewGroup 和 View 的 dispatchTouchEvent 方法返回 super.dispatchTouchEvent() 的時(shí)候事件流走向射赛。
首先看下 ViewGroup 的 dispatchTouchEvent多柑,之前說的 return true 是終結(jié)傳遞。return false 是回溯到父 View 的 onTouchEvent楣责,然后 ViewGroup 怎樣通過 dispatchTouchEvent 方法能把事件分發(fā)到自己的onTouchEvent處理呢竣灌?
return true 和 false 都不行聂沙,那么只能通過 Interceptor 把事件攔截下來給自己的 onTouchEvent,所以 ViewGroup dispatchTouchEvent 方法的 super 默認(rèn)實(shí)現(xiàn)就是去調(diào)用 onInterceptTouchEvent初嘹,記住這一點(diǎn)及汉。
那么對(duì)于 View 的 dispatchTouchEvent return super.dispatchTouchEvent() 的時(shí)候呢事件會(huì)傳到哪里呢?
很遺憾 View 沒有攔截器屯烦。但是同樣的道理 return true 是終結(jié)坷随。return false 是回溯會(huì)父類的 onTouchEvent,怎樣把事件分發(fā)給自己的 onTouchEvent 處理呢驻龟,那只能return super.dispatchTouchEvent温眉,View 類的 dispatchTouchEvent() 方法默認(rèn)實(shí)現(xiàn)就是能幫你調(diào)用 View 自己的 onTouchEvent 方法的。
說了這么多迅脐,不知道有說清楚沒有芍殖,我這邊最后總結(jié)一下:
對(duì)于 dispatchTouchEvent,onTouchEvent谴蔑,return true 是終結(jié)事件傳遞豌骏。return false 是回溯到父 View 的 onTouchEvent 方法。
ViewGroup 想把自己分發(fā)給自己的 onTouchEvent隐锭,需要攔截器 onInterceptTouchEvent 方法 return true 把事件攔截下來窃躲。
ViewGroup 的攔截器 onInterceptTouchEvent 默認(rèn)是不攔截的,所以 return super.onInterceptTouchEvent() = return false钦睡;
View 沒有攔截器蒂窒,為了讓 View 可以把事件分發(fā)給自己的 onTouchEvent,View的dispatchTouchEvent 默認(rèn)實(shí)現(xiàn)(super)就是把事件分發(fā)給自己的 onTouchEvent荞怒。
ViewGroup 和 View 的 dispatchTouchEvent 是做事件分發(fā)洒琢,那么這個(gè)事件可能分發(fā)出去的四個(gè)目標(biāo):
- 自己消費(fèi),終結(jié)傳遞褐桌。------->return true 衰抑;
- 給自己的 onTouchEvent 處理-------> 調(diào)用 super.dispatchTouchEvent() 系統(tǒng)默認(rèn)會(huì)去調(diào)用 onInterceptTouchEvent,在 onInterceptTouchEvent return true 就會(huì)去把事件分給自己的 onTouchEvent 處理荧嵌。
- 傳給子 View------> 調(diào)用 super.dispatchTouchEvent() 默認(rèn)實(shí)現(xiàn)會(huì)去調(diào)用 onInterceptTouchEvent 在 onInterceptTouchEvent return false呛踊,就會(huì)把事件傳給子類。
- 不傳給子 View啦撮,事件終止往下傳遞谭网,事件開始回溯,從父 View 的 onTouchEvent 開始事件從下到上回歸執(zhí)行每個(gè)控件的 onTouchEvent------->return false赃春;
?? 注: 由于 View 沒有子 View 所以不需要 onInterceptTouchEvent 來控件是否把事件傳遞給子 View 還是攔截愉择,所以 View 的事件分發(fā)調(diào)用 super.dispatchTouchEvent() 的時(shí)候默認(rèn)把事件傳給自己的 onTouchEvent 處理(相當(dāng)于攔截),對(duì)比 ViewGroup 的 dispatchTouchEvent 事件分發(fā),View 的事件分發(fā)沒有上面提到的 4 個(gè)目標(biāo)的第 3 點(diǎn)薄辅。
申明:開始的圖片來源網(wǎng)絡(luò)要拂,侵刪