ScrollView巢狀tableView聯動滾動最佳實踐
前言
隨著業務的發展,頁面的複雜度越來越高,巢狀滾動檢視的方式也越來越受設計師們的青睞,在各大電商App十分常見。如下Demo圖:
但是這樣的互動官方並不推薦,而且對開發來說確是不那麼友好,需要處理滾動手勢的衝突,頁面的多層級巢狀都給開發帶來了一定程度的麻煩。接下里我聊聊我們的實現思路。
思路和過程
對應這種頁面結構應該毫無疑問是最底層是一個縱向滾動的scrollView,它的頁面上面放一個固定高度的header,緊接著下面一個支援橫向滾動切換的容器scrollView,容器上面才是各個頁面具體的tableView,如下圖:
思路一
最先想到的是,既然是滾動檢視那麼我們是否可以通過滾動檢視的可滾動屬性來做呢,在初始時把最上層具體業務的tableView禁止滾動,那麼根據事件響應鏈,滾動事件事件會由底層的ScrollView接收並處理,在到達最大偏移量之後,禁用底層的ScrollView滾動,同時開啟上層的tableView,使得上層可以滑動,想起來是有一定可行性的,可惜,事實現實是殘酷的,效果如下:
這樣會導致當偏移量到達臨界值時,由於設定了scrollEnable屬性和最大偏移量,此次滾動手勢會被截斷,需要再次拖拽才能繼續滾動,顯然,這樣的效果是無法接受的。
思路二
這是同事提供的思路,在做這個時和同事有過討論,他們之前有這樣的互動頁面,使用的是自定義手勢,但由於UIScrollView是有彈性效果的,一般的滑動手勢做不到這一點的,所以需要引入UIDynamic模擬力場,實現阻尼效果。想了一下,雖然有一定的可行性,但是為了一個聯動滑動,要做這麼多的事情,感覺比較繁瑣,而且自定義手勢做的模擬彈性效果可能和原生ScrollView的效果還是有一定的差距,所以選擇放棄。
思路三
回到我們思路一,除了邊界位置會阻斷聯動滾動外,其他效果還是可以的,那麼能不能通過手段解決這個問題呢?既然能寫到了這裡,那麼毫無疑問,肯定是可以的。通過手勢穿透
func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer,shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool {
return true
}
複製程式碼
根據官方文件描述:
Asks the delegate if two gesture recognizers should be allowed to recognize gestures simultaneously.
複製程式碼
表達的意思是詢問委託是否允許兩個手勢識別器同時識別手勢,那麼我們實現這個協議,”穿透“手勢,分別在底層容器和上層業務中實現滾動檢視的代理方法func scrollViewDidScroll(_ scrollView: UIScrollView)
,分別控制他們的可滾動狀態和偏移量則能實現目的。部分實現如下:
底層容器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
}
}
}
複製程式碼
上層業務tableView:
func scrollViewDidScroll(_ scrollView: UIScrollView) {
if !childCanScroll {
scrollView.contentOffset.y = 0
} else {
if scrollView.contentOffset.y <= 0 {
childCanScroll = false
superCanScrollBlock?(true)
}
}
}
複製程式碼
通過底層ScrollView是否達到最大偏移量控制header的顯示隱藏和對應的偏移量及可滾動狀態,在業務tableView使用回撥將ScrollView的可滾動狀態回撥達到狀態同步。總體來說還是比較清晰,更多細節請看QFMultipleScrollView