iOS中的觸摸事件 事件傳遞及響應(yīng)鏈

上一篇文章簡(jiǎn)單介紹了UIView的構(gòu)成肥荔,其實(shí)ios中的其他ui控件较曼,如UIWindow酱床,UIButton弥奸,UIlabel等都是繼承自UIView,也就是說,他們的響應(yīng)事件也是如UIView一樣,由UIResponder負(fù)責(zé),比如顶吮,單指點(diǎn)擊,滑動(dòng)粪薛,縮放等悴了,而其中則是由The Responder Chain負(fù)責(zé)各種交互在不同層次的傳遞與判斷由誰響應(yīng)。

在官方文檔中介紹了如何尋找hit-test view 和 first responder object(第一個(gè)響應(yīng)對(duì)象)
這里做簡(jiǎn)單介紹:

尋找hit-test view

iOS使用hit-testing尋找觸摸的view汗菜。 Hit-Testing通過檢查觸摸點(diǎn)是否在關(guān)聯(lián)的view邊界內(nèi)让禀,如果在,則遞歸地檢查該view的所有子view陨界。在層級(jí)上處于lowest(就是離用戶最近的view)且邊界范圍包含觸摸點(diǎn)的view成為hit-test view巡揍。確定hit-test view后,它傳遞觸摸事件給該view菌瘪。

舉例說明腮敌,假設(shè)用戶觸摸了圖中的view E。iOS通過如下順序查找hit-test view俏扩。


Figure 2-1 Hit-testing returns the subview that was touched
  1. 觸摸點(diǎn)在view A中糜工,所以要檢查子view B和C。
  2. 觸摸點(diǎn)不在view B中录淡,但在C中捌木,所以檢查C的子view D和E。
  3. 觸摸點(diǎn)不在D中嫉戚,但在E中刨裆。

View E是這個(gè)層級(jí)上處于lowest的view的邊界范圍包含觸摸點(diǎn)澈圈,所以它成為了hit-test view。
hitTest:withEvent:方法通過傳遞進(jìn)來CGPoint和UIEvent返回hit test view帆啃。該方法調(diào)用pointInside:withEvent:方法瞬女,如果傳入hitTest:withEvent:的point在view的邊界范圍內(nèi),則pointInside:withEvent:返回YES努潘。然后诽偷,這個(gè)方法會(huì)在view的所有子view中遞歸的調(diào)用hitTest:withEvent:。
如果傳入hitTest:withEvent:的point在view的邊界范圍內(nèi)疯坤,則pointInside:withEvent:返回NO报慕。這個(gè)point會(huì)被忽略,hitTest:withEvent:返回nil压怠。如果一個(gè)子view返回NO卖子,則它所在的view的層級(jí)上的分支的子view都會(huì)被忽略。
Hit-test view是處理觸摸事件的第一選擇刑峡,如果hit-test view不能處理事件,該事件將從事件響應(yīng)鏈中尋找響應(yīng)器玄柠,直到系統(tǒng)找到一個(gè)處理事件的對(duì)象突梦。具體見“The Responder Chain Is Made Up of Responder Objects”

響應(yīng)器鏈的傳遞方式

響應(yīng)鏈?zhǔn)鞘裁磿r(shí)候創(chuàng)建的:
當(dāng)一個(gè)view被add到superView上的時(shí)候羽利,他的nextResponder屬性就會(huì)被指向它的superView宫患,當(dāng)controller被初始化的時(shí)候,self.view(topmost view)的nextResponder會(huì)被指向所在的controller这弧,而controller的nextResponder會(huì)被指向self.view的superView娃闲,這樣,整個(gè)app就通過nextResponder串成了一條鏈匾浪,也就是我們所說的響應(yīng)鏈皇帮。所以響應(yīng)鏈就是一條虛擬的鏈,并沒有一個(gè)對(duì)象來專門存儲(chǔ)這樣的一條鏈蛋辈,而是通過UIResponder的屬性串連起來的

響應(yīng)鏈的傳遞機(jī)制:
如果初始化對(duì)象(initial object)—— 即hit-test view或者first responder —— 不處理事件属拾,UIKit會(huì)將事件傳遞給responder chain的下一個(gè)responder。每個(gè)responder決定它是傳遞事件還是通過nextResponder方法傳遞給它的下一個(gè)responder冷溶。這個(gè)操作繼續(xù)直到一個(gè)responder處理event或者沒有responder了渐白。

Responder chain 序列在iOS確定一個(gè)事件并將它傳遞給initial object(通常是view)時(shí)開始。所以initial view有處理事件的第一個(gè)機(jī)會(huì)逞频。下圖描述了兩個(gè)不同的事件傳遞路徑(因?yàn)椴煌腶pp 設(shè)置)纯衍。一個(gè)App的事件傳遞路徑由app特殊的構(gòu)成決定,但事件傳遞路徑會(huì)遵守相同的規(guī)則苗胀。
以下圖片很能說明響應(yīng)鏈?zhǔn)侨绾蝹鬟f的


Figure 2-2 The responder chain on iOS

總結(jié):

事件的傳遞和響應(yīng)分兩個(gè)鏈:

傳遞鏈:由系統(tǒng)向離用戶最近的view傳遞襟诸。UIKit –> active app’s event queue –> window –> root view –>……–>lowest view
響應(yīng)鏈:由離用戶最近的view向系統(tǒng)傳遞瓦堵。initial view –> super view –> …..–> view controller –> window –> Application

事件處理的整個(gè)流程總結(jié):

1.觸摸屏幕產(chǎn)生觸摸事件后,觸摸事件會(huì)被添加到由UIApplication管理的事件隊(duì)列中
2.UIApplication會(huì)從事件隊(duì)列中取出最前面的事件励堡,把事件傳遞給應(yīng)用程序的主窗口谷丸,這時(shí)候執(zhí)行事件傳遞流程 找到一個(gè)最合適的視圖來處理觸摸事件。(這時(shí)候如果某一個(gè)view上添加了手勢(shì)应结,且該手勢(shì)能響應(yīng)對(duì)應(yīng)事件刨疼,則走手勢(shì)的響應(yīng),根據(jù)手勢(shì)的設(shè)置來決定是否阻斷下面的步驟鹅龄,但是事件傳遞過程依舊揩慕。如沒有或者不能響應(yīng)則繼續(xù)走下面步驟)
3.最合適的view會(huì)調(diào)用自己的touches方法處理事件 如果需要的話可以把事件順著響應(yīng)者鏈條向上拋。

以上

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末扮休,一起剝皮案震驚了整個(gè)濱河市迎卤,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌玷坠,老刑警劉巖蜗搔,帶你破解...
    沈念sama閱讀 206,214評(píng)論 6 481
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異八堡,居然都是意外死亡樟凄,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 88,307評(píng)論 2 382
  • 文/潘曉璐 我一進(jìn)店門兄渺,熙熙樓的掌柜王于貴愁眉苦臉地迎上來缝龄,“玉大人,你說我怎么就攤上這事挂谍∈迦溃” “怎么了?”我有些...
    開封第一講書人閱讀 152,543評(píng)論 0 341
  • 文/不壞的土叔 我叫張陵口叙,是天一觀的道長(zhǎng)炼绘。 經(jīng)常有香客問我,道長(zhǎng)妄田,這世上最難降的妖魔是什么饭望? 我笑而不...
    開封第一講書人閱讀 55,221評(píng)論 1 279
  • 正文 為了忘掉前任,我火速辦了婚禮形庭,結(jié)果婚禮上铅辞,老公的妹妹穿的比我還像新娘。我一直安慰自己萨醒,他們只是感情好斟珊,可當(dāng)我...
    茶點(diǎn)故事閱讀 64,224評(píng)論 5 371
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著富纸,像睡著了一般囤踩。 火紅的嫁衣襯著肌膚如雪旨椒。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 49,007評(píng)論 1 284
  • 那天堵漱,我揣著相機(jī)與錄音综慎,去河邊找鬼。 笑死勤庐,一個(gè)胖子當(dāng)著我的面吹牛示惊,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播愉镰,決...
    沈念sama閱讀 38,313評(píng)論 3 399
  • 文/蒼蘭香墨 我猛地睜開眼米罚,長(zhǎng)吁一口氣:“原來是場(chǎng)噩夢(mèng)啊……” “哼!你這毒婦竟也來了丈探?” 一聲冷哼從身側(cè)響起录择,我...
    開封第一講書人閱讀 36,956評(píng)論 0 259
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎碗降,沒想到半個(gè)月后隘竭,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 43,441評(píng)論 1 300
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡讼渊,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 35,925評(píng)論 2 323
  • 正文 我和宋清朗相戀三年货裹,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片精偿。...
    茶點(diǎn)故事閱讀 38,018評(píng)論 1 333
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖赋兵,靈堂內(nèi)的尸體忽然破棺而出笔咽,到底是詐尸還是另有隱情,我是刑警寧澤霹期,帶...
    沈念sama閱讀 33,685評(píng)論 4 322
  • 正文 年R本政府宣布叶组,位于F島的核電站,受9級(jí)特大地震影響历造,放射性物質(zhì)發(fā)生泄漏甩十。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,234評(píng)論 3 307
  • 文/蒙蒙 一吭产、第九天 我趴在偏房一處隱蔽的房頂上張望侣监。 院中可真熱鬧,春花似錦臣淤、人聲如沸橄霉。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,240評(píng)論 0 19
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽姓蜂。三九已至按厘,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間钱慢,已是汗流浹背逮京。 一陣腳步聲響...
    開封第一講書人閱讀 31,464評(píng)論 1 261
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留束莫,地道東北人懒棉。 一個(gè)月前我還...
    沈念sama閱讀 45,467評(píng)論 2 352
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像麦箍,于是被迫代替她去往敵國和親漓藕。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 42,762評(píng)論 2 345

推薦閱讀更多精彩內(nèi)容

  • 好奇觸摸事件是如何從屏幕轉(zhuǎn)移到APP內(nèi)的挟裂?困惑于Cell怎么突然不能點(diǎn)擊了享钞?糾結(jié)于如何實(shí)現(xiàn)這個(gè)奇葩響應(yīng)需求?亦或是...
    Lotheve閱讀 56,661評(píng)論 51 597
  • 在iOS開發(fā)中經(jīng)常會(huì)涉及到觸摸事件诀蓉。本想自己總結(jié)一下栗竖,但是遇到了這篇文章,感覺總結(jié)的已經(jīng)很到位渠啤,特此轉(zhuǎn)載狐肢。作者:L...
    WQ_UESTC閱讀 5,989評(píng)論 4 26
  • 用戶以多種方式操縱他們的iOS設(shè)備,例如觸摸屏幕或搖動(dòng)設(shè)備沥曹。 iOS會(huì)解釋用戶何時(shí)以及如何操作硬件并將此信息傳遞到...
    坤坤同學(xué)閱讀 3,978評(píng)論 7 19
  • 一. Hit-Testing 什么是Hit-Testing?對(duì)于觸摸事件, window首先會(huì)嘗試將事件交給事件觸...
    面糊閱讀 815評(píng)論 0 50
  • 事件傳遞:響應(yīng)者鏈 當(dāng)你設(shè)計(jì)一個(gè)app的時(shí)候份名,你很可能需要你的app能夠動(dòng)態(tài)響應(yīng)某些事件。比如妓美,觸摸可以發(fā)生在屏幕...
    hjfrun閱讀 1,021評(píng)論 1 5