IOS - 響應者鏈條

weixin_34236869發表於2017-02-07

簡單來說就是 :一級一級的找到響應的檢視,如果沒有就傳給UIWindow例項和UIApplication例項,要是他們也處理不了,就丟棄這次事件...
對於iOS裝置使用者來說,他們操作裝置的方式主要有三種:觸控螢幕、晃動裝置、通過遙控設施控制裝置。對應的事件型別有以下三種:
1、觸屏事件(Touch Event)
2、運動事件(Motion Event)
3、遠端控制事件(Remote-Control Event)
響應者鏈條概念: iOS系統檢測到手指觸控(Touch)操作時會將其打包成一個UIEvent物件,並放入當前活動Application的事件佇列,單例的UIApplication會從事件佇列中取出觸控事件並傳遞給單例的UIWindow來處理,UIWindow物件首先會使用hitTest:withEvent:方法尋找此次Touch操作初始點所在的檢視(View),即需要將觸控事件傳遞給其處理的檢視,這個過程稱之為hit-test view。
響應者物件(Responder Object) 指的是 有響應和處理事件能力的物件。 響應者鏈就是由一系列的響應者物件 構成的一個層次結構。
UIResponder 是所有響應物件的基類,在UIResponder類中定義了處理上述各種事件的介面。我們熟悉的 UIApplication、 UIViewController、 UIWindow 和所有繼承自UIView的UIKit類都直接或間接的繼承自UIResponder,所以它們的例項都是可以構成響應者鏈的響應者物件。
UIWindow例項物件會首先在它的內容檢視上呼叫hitTest:withEvent:,此方法會在其檢視層級結構中的每個檢視上呼叫pointInside:withEvent:(該方法用來判斷點選事件發生的位置是否處於當前檢視範圍內,以確定使用者是不是點選了當前檢視),如果pointInside:withEvent:返回YES,則繼續逐級呼叫,直到找到touch操作發生的位置,這個檢視也就是要找的hit-test view。
hitTest:withEvent:方法的處理流程如下: 首先呼叫當前檢視的pointInside:withEvent:方法判斷觸控點是否在當前檢視內; 若返回NO,則hitTest:withEvent:返回nil; 若返回YES,則向當前檢視的所有子檢視(subviews)傳送hitTest:withEvent:訊息,所有子檢視的遍歷順序是從最頂層檢視一直到到最底層檢視,即從subviews陣列的末尾向前遍歷,直到有子檢視返回非空物件或者全部子檢視遍歷完畢; 若第一次有子檢視返回非空物件,則hitTest:withEvent:方法返回此物件,處理結束; 如所有子檢視都返回非,則hitTest:withEvent:方法返回自身(self)。

2278500-743eb3ad7aeea22e.png
Paste_Image.png

假如使用者點選了View E,下面結合圖二介紹hit-test view的流程:

1、A是UIWindow的根檢視,因此,UIWindow物件會首相對A進行hit-test;

2、顯然使用者點選的範圍是在A的範圍內,因此, pointInside:withEvent:返回了YES,這時會繼續檢查A的子檢視;

3、這時候會有兩個分支,B和C:

點選的範圍不再B內,因此B分支的 pointInside:withEvent:返回NO,對應的hitTest:withEvent:返回nil;

點選的範圍在C內,即C的 pointInside:withEvent:返回YES;

4、這時候有D和E兩個分支:

點選的範圍不再D內,因此D 的 pointInside:withEvent:返回NO,對應的hitTest:withEvent:返回nil;

點選的範圍在E內,即E的 pointInside:withEvent:返回YES,由於E沒有子檢視(也可以理解成對E的子檢視進行hit-test時返回了nil),因此,E的 hitTest:withEvent:會將E返回,再往回回溯,就是C的 hitTest:withEvent:返回E--->>A的hitTest:withEvent:返回E。
至此,本次點選事件的第一響應者就通過響應者鏈的事件分發邏輯成功的找到了。

不難看出,這個處理流程有點類似二分搜尋的思想,這樣能以最快的速度,最精確地定位出能響應觸控事件的UIView。

上面找到了事件的第一響應者,接下來就該沿著尋找第一響應者的相反順序來處理這個事件,如果UIWindow單例和UIApplication都無法處理這一事件,則該事件會被丟棄。

說明:

1、如果最終 hit-test沒有找到第一響應者,或者第一響應者沒有處理該事件,則該事件會沿著響應者鏈向上回溯,如果UIWindow例項和UIApplication例項都不能處理該事件,則該事件會被丟棄;

2、hitTest:withEvent:方法將會忽略隱藏(hidden=YES)的檢視,禁止使用者操作(userInteractionEnabled=YES)的檢視,以及alpha級別小於0.01(alpha<0.01)的檢視。如果一個子檢視的區域超過父檢視的bound區域(父檢視的clipsToBounds 屬性為NO,這樣超過父檢視bound區域的子檢視內容也會顯示),那麼正常情況下對子檢視在父檢視之外區域的觸控操作不會被識別,因為父檢視的pointInside:withEvent:方法會返回NO,這樣就不會繼續向下遍歷子檢視了。當然,也可以重寫pointInside:withEvent:方法來處理這種情況。

相關文章