iOS之事件穿透

前言

小伙伴們?cè)陂_發(fā)中是否遇到過這樣的需求呢敌土,一個(gè)控件的某個(gè)部分被另外一個(gè)控件遮擋住瞳遍,當(dāng)點(diǎn)擊這個(gè)重疊部分時(shí),需要響應(yīng)被遮蓋控件的點(diǎn)擊事件睦优,就如下圖所示


當(dāng)我們點(diǎn)擊區(qū)域3時(shí)渗常,響應(yīng)藍(lán)色按鈕的點(diǎn)擊事件,點(diǎn)擊區(qū)域1和2時(shí)汗盘,響應(yīng)紅色按鈕的點(diǎn)擊事件皱碘,對(duì)于區(qū)域1和3沒什么好說的,那如何讓紅色按鈕響應(yīng)區(qū)域2的點(diǎn)擊呢隐孽?這就是筆者今天要講的內(nèi)容癌椿。

事件傳遞

大家應(yīng)該都知道健蕊,事件從應(yīng)用程序開始,按照從上到下的順序(UIApplication -> UIWindow -> rootViewController -> ...)一級(jí)一級(jí)傳遞踢俄,并且系統(tǒng)在尋找最適合處理事件的控件時(shí)缩功,是從后往前遍歷子控件的(網(wǎng)上資料太多,不做詳細(xì)闡述褪贵,請(qǐng)自行百度)

上圖中藍(lán)色按鈕在紅色按鈕之后添加掂之,當(dāng)系統(tǒng)尋找最適合的控件時(shí),藍(lán)色按鈕在紅色按鈕之前被找到脆丁,系統(tǒng)發(fā)現(xiàn)藍(lán)色按鈕很適合處理事件世舰,所以方法便返回了,紅色按鈕就沒有了處理事件的機(jī)會(huì)槽卫。

系統(tǒng)如何尋找最適合控件

  • 判斷自己能否接受觸摸事件跟压,如果不能,返回nil
  • 判斷觸摸點(diǎn)是否在自己身上歼培,如果不能震蒋,返回nil
  • 從后往前遍歷子控件,重復(fù)上面的步驟躲庄,如果沒有適合的子控件查剖,返回自己

我們來看看系統(tǒng)內(nèi)部是如何實(shí)現(xiàn)的,筆者這里自定義了一個(gè)UIWindow噪窘,讓它成為主窗口笋庄,并重寫它的hitTest方法,運(yùn)行之后倔监,其事件處理功能直砂,與系統(tǒng)的類似,所以系統(tǒng)內(nèi)部大概就是這樣實(shí)現(xiàn)的


當(dāng)一個(gè)控件的透明度小于某個(gè)值時(shí)浩习,就不再響應(yīng)事件静暂,上圖中0.01僅僅是為了測(cè)試,并非準(zhǔn)確的值谱秽,要注意的就是洽蛀,對(duì)于繼承自UIControl的控件,還需要判斷enable的值

事件穿透

既然系統(tǒng)尋找最合適控件的方法滿足不了我們疟赊,那我們就重寫系統(tǒng)的方法

思路
  • 點(diǎn)擊藍(lán)色按鈕的區(qū)域2辱士,紅色按鈕響應(yīng)事件,那肯定要重寫藍(lán)色按鈕的hitTest方法
  • 在hitTest方法中听绳,將觸摸點(diǎn)的坐標(biāo)系從藍(lán)色按鈕轉(zhuǎn)換到紅色按鈕上,即以紅色按鈕左上角為原點(diǎn)
  • 坐標(biāo)系轉(zhuǎn)換后异赫,判斷觸摸點(diǎn)是否在紅色按鈕上椅挣,如果是头岔,直接返回紅色按鈕(嚴(yán)謹(jǐn)一點(diǎn)的做法是調(diào)用紅色按鈕的hitTest方法),如果不是鼠证,那就調(diào)用系統(tǒng)的方法峡竣,讓系統(tǒng)去處理

有了思路,那萬事具備只欠東風(fēng)了量九,接下來上東風(fēng)

新建一個(gè)類适掰,繼承自UIButton,筆者這里直接命名為BlueButton荠列,修改sb\xib中藍(lán)色按鈕的類型為BlueButton

將紅色按鈕連線到BlueButton.m文件中类浪,不用試了,直接連是連不了的肌似,我們可以先在BlueButton.m中定義一個(gè)屬性费就,前面加上IBOutlet,然后單擊圖中的空心圓川队,拖到紅色按鈕上就OK了

最后力细,在BlueButton.m中重寫藍(lán)色按鈕的hitTest方法,代碼如下

- (UIView *)hitTest:(CGPoint)point withEvent:(UIEvent *)event {
    CGPoint redBtnPoint = [self convertPoint:point toView:_redButton];
    if ([_redButton pointInside:redBtnPoint withEvent:event]) {
        return _redButton;
    }
    //如果希望嚴(yán)謹(jǐn)一點(diǎn)固额,可以將上面if語句及里面代碼替換成如下代碼
    //UIView *view = [_redButton hitTest: redBtnPoint withEvent: event];
    //if (view) return view;
    return [super hitTest:point withEvent:event];
}

來看運(yùn)行結(jié)果眠蚂,點(diǎn)擊區(qū)域2時(shí),紅色按鈕高亮并響應(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)自:codingZero原文鏈接:http://www.reibang.com/p/0bece5f27650 前...
    sea777777閱讀 567評(píng)論 1 2
  • 前言:按照時(shí)間順序秸侣,事件的生命周期是這樣的:事件的產(chǎn)生和傳遞(事件如何從父控件傳遞到子控件并尋找到最合適的view...
    貝勒老爺閱讀 998評(píng)論 1 9
  • 本文來自:http://ios.jobbole.com/84081/ 前言: 按照時(shí)間順序存筏,事件的生命周期是這樣的...
    HackerOnce閱讀 2,831評(píng)論 1 10
  • 好奇觸摸事件是如何從屏幕轉(zhuǎn)移到APP內(nèi)的?困惑于Cell怎么突然不能點(diǎn)擊了味榛?糾結(jié)于如何實(shí)現(xiàn)這個(gè)奇葩響應(yīng)需求椭坚?亦或是...
    Lotheve閱讀 56,661評(píng)論 51 597
  • 每個(gè)星期天對(duì)于我來說 睡覺 然后就是消磨時(shí)光了。曾經(jīng) 想過 好多種 豐富的星期天生活 跑步 吃早飯 去圖書館看書...
    莫小井閱讀 266評(píng)論 0 0