滴滴出行首頁(yè)(快車(chē)),可以上拉下拉底部菜單(tableView),但又不影響后面地圖的交互使用。
找來(lái)找去網(wǎng)上沒(méi)有Demo,就想著自己做。
一開(kāi)始思路錯(cuò)誤,想給tableview加手勢(shì)和慣性滑動(dòng),做來(lái)做去總是不完美,而且麻煩。放棄了!
先上GitHub:Demo?(1.2已修改上拉不完全問(wèn)題:tableview的高度沒(méi)有設(shè)置對(duì))
Demo演示:
滴滴圖:
首先你要懂一點(diǎn) iOS 的響應(yīng)鏈的東西,當(dāng)手指觸摸到屏幕后,這個(gè)觸摸事件是怎么派發(fā)的。
正確思路:
(1)tableview其實(shí)是全屏,設(shè)置其contentInset使其下移,并設(shè)置背景色為透明。
(2)當(dāng)手指觸碰到屏幕時(shí),其實(shí)是觸碰在tableview上的;這里要判斷手指第一下觸碰在屏幕上的坐標(biāo),是在透明區(qū)域(地圖)上還是tableview的不透明區(qū)域(cell)上。
(3)當(dāng)在透明區(qū)域時(shí),判斷手勢(shì)操作底層地圖;當(dāng)在tableview的不透明區(qū)域上,判斷手勢(shì)操作tableview。
實(shí)現(xiàn)方法:
不多說(shuō)了,直接說(shuō)做法。
1.創(chuàng)建一個(gè)繼承UITableView的子類(lèi)
2.關(guān)鍵,在UITableView子類(lèi)重寫(xiě)hitTest:withEvent:方法。
????2.1關(guān)于這個(gè)方法的理解,看這里:hitTest:withEvent:方法流程。(我也理解得不深)
-(UIView*)hitTest:(CGPoint)point withEvent:(UIEvent*)event{
? ? if(point.y<0) {
? ? ? ? return nil;
?? ?}
?? ?return? [super hitTest:point withEvent:event];
}
????2.2 方法里的手指觸碰點(diǎn)的坐標(biāo)point在我測(cè)試過(guò)后,發(fā)現(xiàn)不是以屏幕左上點(diǎn)為原點(diǎn)的,而是相對(duì)于tableview的contentOffset的。這里說(shuō)不太清楚,可以自己打印著看一下。
????tableview是不斷滑動(dòng)的,所以contentOffset也是不斷改變的。
????因此,這里只要判斷point的y值是正是負(fù)就可以確定手指第一下是觸碰在地圖上還是tableview上了。
????2.3 point.y<0時(shí),說(shuō)明手指觸碰在地圖上,返回nil,讓其不會(huì)在tableview上觸發(fā)操作。反之,則正常進(jìn)行此方法。
3. 在VC的view中加入地圖視圖,以及UITableView子類(lèi)的tableview,并設(shè)置tableview的contentInset使其下移。
注意:使tableview下移的contentInset和用來(lái)判斷point.y的contentOffset不要搞混了。
GitHub:Demo?(1.2已修改上拉不完全問(wèn)題:tableview的高度沒(méi)有設(shè)置對(duì))
參考文檔:iOS之事件穿透
? ? ? ? ? ? ? ? ??hitTest:withEvent:方法流程
? ? ? ? ? ? ? ? ??[iOS開(kāi)發(fā)]hitTest的一些使用(點(diǎn)擊穿透)
補(bǔ)充:這里是第二篇iOS 仿滴滴出行界面UI(2) - 簡(jiǎn)書(shū)