iOS ScrollView嵌套tableView聯(lián)動(dòng)滾動(dòng)的思路與最佳實(shí)踐
前言
隨著業(yè)務(wù)的發(fā)展,頁(yè)面的復(fù)雜度越來越高,嵌套滾動(dòng)視圖的方式也越來越受設(shè)計(jì)師們的青睞,在各大電商App十分常見。如下Demo圖:
但是這樣的交互官方并不推薦,而且對(duì)開發(fā)來說確是不那么友好,需要處理滾動(dòng)手勢(shì)的沖突,頁(yè)面的多層級(jí)嵌套都給開發(fā)帶來了一定程度的麻煩。接下里我聊聊我們的實(shí)現(xiàn)思路。
思路和過程
對(duì)應(yīng)這種頁(yè)面結(jié)構(gòu)應(yīng)該毫無疑問是最底層是一個(gè)縱向滾動(dòng)的scrollView,它的頁(yè)面上面放一個(gè)固定高度的header,緊接著下面一個(gè)支持橫向滾動(dòng)切換的容器scrollView,容器上面才是各個(gè)頁(yè)面具體的tableView,如下圖:
思路一
最先想到的是,既然是滾動(dòng)視圖那么我們是否可以通過滾動(dòng)視圖的可滾動(dòng)屬性來做呢,在初始時(shí)把最上層具體業(yè)務(wù)的tableView禁止?jié)L動(dòng),那么根據(jù)事件響應(yīng)鏈,滾動(dòng)事件事件會(huì)由底層的ScrollView接收并處理,在到達(dá)最大偏移量之后,禁用底層的ScrollView滾動(dòng),同時(shí)開啟上層的tableView,使得上層可以滑動(dòng),想起來是有一定可行性的,可惜,事實(shí)現(xiàn)實(shí)是殘酷的,效果如下:
這樣會(huì)導(dǎo)致當(dāng)偏移量到達(dá)臨界值時(shí),由于設(shè)置了scrollEnable屬性和最大偏移量,此次滾動(dòng)手勢(shì)會(huì)被截?cái)?,需要再次拖拽才能繼續(xù)滾動(dòng),顯然,這樣的效果是無法接受的。
思路二
這是同事提供的思路,在做這個(gè)時(shí)和同事有過討論,他們之前有這樣的交互頁(yè)面,使用的是自定義手勢(shì),但由于UIScrollView是有彈性效果的,一般的滑動(dòng)手勢(shì)做不到這一點(diǎn)的,所以需要引入U(xiǎn)IDynamic模擬力場(chǎng),實(shí)現(xiàn)阻尼效果。想了一下,雖然有一定的可行性,但是為了一個(gè)聯(lián)動(dòng)滑動(dòng),要做這么多的事情,感覺比較繁瑣,而且自定義手勢(shì)做的模擬彈性效果可能和原生ScrollView的效果還是有一定的差距,所以選擇放棄。
思路三
回到我們思路一,除了邊界位置會(huì)阻斷聯(lián)動(dòng)滾動(dòng)外,其他效果還是可以的,那么能不能通過手段解決這個(gè)問題呢?既然能寫到了這里,那么毫無疑問,肯定是可以的。通過手勢(shì)穿透,即讓一個(gè)滑動(dòng)手勢(shì)既作用于底層的ScrollView又能作用于上層的業(yè)務(wù)tableView,同時(shí)控制他們的滾動(dòng)即可達(dá)成目的。通過讓底層的scrollView實(shí)現(xiàn)一個(gè)手勢(shì)識(shí)別的協(xié)議,同時(shí)響應(yīng)滾動(dòng)事件:
func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool { return true }
根據(jù)官方文檔描述:
Asks the delegate if two gesture recognizers should be allowed to recognize gestures simultaneously.
表達(dá)的意思是詢問委托是否允許兩個(gè)手勢(shì)識(shí)別器同時(shí)識(shí)別手勢(shì),那么我們實(shí)現(xiàn)這個(gè)協(xié)議,”穿透“手勢(shì),分別在底層容器和上層業(yè)務(wù)中實(shí)現(xiàn)滾動(dòng)視圖的代理方法func scrollViewDidScroll(_ scrollView: UIScrollView),分別控制他們的可滾動(dòng)狀態(tài)和偏移量則能實(shí)現(xiàn)目的。部分實(shí)現(xiàn)如下:
底層容器ScrollView:
func scrollViewDidScroll(_ scrollView: UIScrollView) { headerView.isHidden = scrollView.contentOffset.y >= maxOffset ? true : false if !superCanScroll { scrollView.contentOffset.y = maxOffset currentVC.childCanScroll = true } else { if scrollView.contentOffset.y >= maxOffset { scrollView.contentOffset.y = maxOffset superCanScroll = false currentVC.childCanScroll = true } } }
上層業(yè)務(wù)tableView:
func scrollViewDidScroll(_ scrollView: UIScrollView) { if !childCanScroll { scrollView.contentOffset.y = 0 } else { if scrollView.contentOffset.y <= 0 { childCanScroll = false superCanScrollBlock?(true) } } }
通過底層ScrollView是否達(dá)到最大偏移量控制header的顯示隱藏和對(duì)應(yīng)的偏移量及可滾動(dòng)狀態(tài),在業(yè)務(wù)tableView使用回調(diào)將ScrollView的可滾動(dòng)狀態(tài)回調(diào)達(dá)到狀態(tài)同步??傮w來說還是比較清晰,更多細(xì)節(jié)請(qǐng)看QFMultipleScrollView
總結(jié)
以上就是這篇文章的全部?jī)?nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,謝謝大家對(duì)我們的支持。
上一篇:iOS實(shí)現(xiàn)鎖屏頁(yè)面控制音樂
欄 目:IOS
下一篇:詳解iOS 輕松獲取當(dāng)前控制器的正確方式
本文標(biāo)題:iOS ScrollView嵌套tableView聯(lián)動(dòng)滾動(dòng)的思路與最佳實(shí)踐
本文地址:http://mengdiqiu.com.cn/a1/IOS/11856.html
您可能感興趣的文章
- 01-11iOS常用算法之兩個(gè)有序數(shù)組合并(要求時(shí)間復(fù)雜度為0(n))
- 01-11iOS 彈幕功能的實(shí)現(xiàn)思路圖解
- 01-11iOS調(diào)試Block引用對(duì)象無法被釋放的小技巧分享
- 01-11iOS動(dòng)態(tài)更換Icon的全過程記錄
- 01-11iOS實(shí)現(xiàn)文本分頁(yè)的方法示例
- 01-11iOS常見宏理解及使用方法
- 01-11iOs遷至WKWebView跨過的一些坑
- 01-11iOS模擬中獎(jiǎng)名單循環(huán)滾動(dòng)效果
- 01-11Python一鍵查找iOS項(xiàng)目中未使用的圖片、音頻、視頻資源
- 01-11iOS中如何獲取某個(gè)視圖的截圖詳析


閱讀排行
- 1C語(yǔ)言 while語(yǔ)句的用法詳解
- 2java 實(shí)現(xiàn)簡(jiǎn)單圣誕樹的示例代碼(圣誕
- 3利用C語(yǔ)言實(shí)現(xiàn)“百馬百擔(dān)”問題方法
- 4C語(yǔ)言中計(jì)算正弦的相關(guān)函數(shù)總結(jié)
- 5c語(yǔ)言計(jì)算三角形面積代碼
- 6什么是 WSH(腳本宿主)的詳細(xì)解釋
- 7C++ 中隨機(jī)函數(shù)random函數(shù)的使用方法
- 8正則表達(dá)式匹配各種特殊字符
- 9C語(yǔ)言十進(jìn)制轉(zhuǎn)二進(jìn)制代碼實(shí)例
- 10C語(yǔ)言查找數(shù)組里數(shù)字重復(fù)次數(shù)的方法
本欄相關(guān)
- 01-11UILabel顯示定時(shí)器文本跳動(dòng)問題的解決
- 01-11iOS常用算法之兩個(gè)有序數(shù)組合并(要
- 01-11iOS 彈幕功能的實(shí)現(xiàn)思路圖解
- 01-11詳解MacOs免密登錄CentOs操作步驟
- 01-11iOS動(dòng)態(tài)更換Icon的全過程記錄
- 01-11iOS調(diào)試Block引用對(duì)象無法被釋放的小技
- 01-11iOS常見宏理解及使用方法
- 01-11iOS實(shí)現(xiàn)文本分頁(yè)的方法示例
- 01-11iOs遷至WKWebView跨過的一些坑
- 01-11iOS模擬中獎(jiǎng)名單循環(huán)滾動(dòng)效果
隨機(jī)閱讀
- 08-05DEDE織夢(mèng)data目錄下的sessions文件夾有什
- 01-11ajax實(shí)現(xiàn)頁(yè)面的局部加載
- 01-10delphi制作wav文件的方法
- 01-10SublimeText編譯C開發(fā)環(huán)境設(shè)置
- 04-02jquery與jsp,用jquery
- 01-10C#中split用法實(shí)例總結(jié)
- 08-05dedecms(織夢(mèng))副欄目數(shù)量限制代碼修改
- 01-10使用C語(yǔ)言求解撲克牌的順子及n個(gè)骰子
- 08-05織夢(mèng)dedecms什么時(shí)候用欄目交叉功能?
- 01-11Mac OSX 打開原生自帶讀寫NTFS功能(圖文