美文网首页iOS Dev
UITableView嵌套UITableView的流畅滑动(Sw

UITableView嵌套UITableView的流畅滑动(Sw

作者: 旷野独狼 | 来源:发表于2016-09-02 17:49 被阅读1081次

本次效果是参考此文(http://www.jianshu.com/p/38f0b7eef959)实现的,在此多谢此文作者分享!

思路精髓:实现手势的穿透响应(也就是不论最上层是否有对象响应滑动手势,都将这个手势往底层传递),给每个tableView 添加一个shouldScroll的实例变量,通过tableView的代理方法监听contentOffset ,根据偏移量做判断并发送相应通知,然后设置不应该滑动的那个tableView的contentOffset为CGPointZero即可。

  • ps: 我为了实现公司的需求,又多嵌套了一层UICollectionView,但万变不离其宗,思路都是一样。

先上效果图

效果动图.gif

在此,只分析两个tableView的滑动逻辑

  • 底层的UITableView 命名为 tableView
  • 嵌套在内的UITableView 命名为 innerTableView

自定义UITableView,命名为GXInnerTableView,在GXInnerTableView中遵循 UIGestureRecognizerDelegate协议,并实现如下方法 ,这是实现手势穿透的关键代码

func gestureRecognizer(gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWithGestureRecognizer otherGestureRecognizer: UIGestureRecognizer) -> Bool {
    return true
}

接下来分别实现两个tableView的ScrollViewDidScroll代理方法

  • tableView的代理方法
          /// tableView是否能够滚动
        var shouldScroll:Bool = true
            /// 底部的理论上是否应该滚动
        var bottomShouldScroll:Bool = true
            /// 顶部的理论上是否应该滚动
        var upperShouldScroll:Bool = true
        
      func scrollViewDidScroll(scrollView: UIScrollView) {
        print(scrollView.contentOffset.y)
        
        // header头的高度是150
        let standardOffsetY:CGFloat = 150
        
        let offsetY:CGFloat = scrollView.contentOffset.y
        
        bottomShouldScroll = upperShouldScroll
        
        if offsetY >= standardOffsetY{ //如果偏移量大于header头的高度,不能继续滑动,固定tableView的偏移量
            
            scrollView.contentOffset = CGPointMake(0, standardOffsetY)
            
            
            upperShouldScroll = true
            
        }else{
            
            upperShouldScroll = false
            
        }
        
        if upperShouldScroll != bottomShouldScroll {
            
            if !bottomShouldScroll && upperShouldScroll{
                
                NSNotificationCenter.defaultCenter().postNotificationName(goTopNotificationName, object: nil)
                
                shouldScroll = false
                
            }
            
            if bottomShouldScroll && !upperShouldScroll{
                
                if !shouldScroll{
                    
                    scrollView.contentOffset = CGPointMake(0, standardOffsetY)
                }
            }
        }
    }
  • innerTableView的代理方法
    func scrollViewDidScroll(scrollView: UIScrollView) {
        print(scrollView.contentOffset.y)
        
        if !shouldScroll {
            
            scrollView.setContentOffset(CGPointZero, animated: false)
            
        }
        
        if scrollView.contentOffset.y <= 0{
            
            NSNotificationCenter.defaultCenter().postNotificationName(leaveTopNotificationName, object: nil)
            
        }
    }

在代理方法中都发送了通知,自然需要对相应的通知进行监听了

  • tableView对通知的监听方法
    @objc private func changeScrollEnable(notification:NSNotification){
        
            shouldScroll = true
        
    }
  • innerTableView对通知的监听方法
    @objc private func changeScrollEnable(notification:NSNotification){
        
        
        let nameStr = notification.name
        
        if nameStr == goTopNotificationName{
            
                innerTableView.showsVerticalScrollIndicator = true
                
                shouldScroll = true
                
            
        }else if nameStr == leaveTopNotificationName{
            
            innerTableView.contentOffset = CGPointZero
            
            shouldScroll = false
            
            innerTableView.showsVerticalScrollIndicator = false
            
        }
    }

OK,关键代码就这些,不足之处望简友多多指教。

相关文章

网友评论

本文标题:UITableView嵌套UITableView的流畅滑动(Sw

本文链接:https://www.haomeiwen.com/subject/jrqkettx.html