UIScrollView及其子类的嵌套联动

效果图

层级结构
1.手势同时识别

两个纵向滚动的视图可以同时识别拖拽手势才能根据偏移量控制是否可以滑动,当滑动的时候就可以分别在各自的scrollViewDidScroll方法中获取到各自的偏移量

extension SSMainScrollView: UIGestureRecognizerDelegate
{
    //允许两个手势同时识别
    func gestureRecognizer(_ gestureRecognizer: UIGestureRecognizer, shouldRecognizeSimultaneouslyWith otherGestureRecognizer: UIGestureRecognizer) -> Bool  {
        return true
    }
}
2.联动控制

赋予两个scrollView各自属性canScroll,通过通知传递参数控制联动。
⚠️控制联动,不能通过控制scrollView.isScrollEnabled来实现,否则到临界点会卡顿不流畅,通过控制scrollView.contentOffset实现。
父控制器

extension SSInlineScrollViewController: UIScrollViewDelegate {
    
    func scrollViewDidScroll(_ scrollView: UIScrollView) {
        
        let offsetY: CGFloat = scrollView.contentOffset.y
        
        if offsetY < 0 {
            //1.下拉头图放大
            if _isEnlarged {
                let totalOffset = HEADERIMGVIEW_HEIGHT + abs(offsetY)
                let ratio = totalOffset / HEADERIMGVIEW_HEIGHT
                navigationBar.backgroundColor = UIColor(white: 1.0, alpha: 0.0)
                headerView.snp.updateConstraints({ (make) in
                    make.top.equalTo(offsetY)
                    make.left.equalTo(-SCREEN_WIDTH * (ratio - 1) * 0.5)
                    make.width.equalTo(SCREEN_WIDTH * ratio)
                    make.height.equalTo(totalOffset)
                })
            }
        }else{
            //2.导航栏颜色渐变
            var alpha: CGFloat = offsetY / (HEADERIMGVIEW_HEIGHT - NAVIGATIONBAR_HEIGHT)
            alpha = min(alpha, 1.0)
            navigationBar.backgroundColor = UIColor(white: 1.0, alpha: alpha)
            if alpha == 1.0 {
                navigationBar.backButton.setImage(UIImage(named:"personal_home_back_black_24x24_"), for: .normal)
            }else{
                navigationBar.backButton.setImage(UIImage(named:"personal_home_back_white_24x24_"), for: .normal)
            }
            
        }
        
        //3.联动控制
        if canScroll == "0" {
            mainScrollView.contentOffset = CGPoint(x: 0, y: (HEADERIMGVIEW_HEIGHT - NAVIGATIONBAR_HEIGHT))
        }else{
            if offsetY >= (HEADERIMGVIEW_HEIGHT - NAVIGATIONBAR_HEIGHT) {
                //子动 父不动
                NotificationCenter.default.post(name: NSNotification.Name(rawValue: NOTIFICATION_SSContentVC_canScroll), object: nil, userInfo: [KEY_CANSCROLL : "1"])
                mainScrollView.contentOffset = CGPoint(x: 0, y: (HEADERIMGVIEW_HEIGHT - NAVIGATIONBAR_HEIGHT))
                
            }else{
                //子不动 父动
                NotificationCenter.default.post(name: NSNotification.Name(rawValue: NOTIFICATION_SSContentVC_canScroll), object: nil, userInfo: [KEY_CANSCROLL : "0"])
            }
        }
        
    }
    
}

子控制器

extension SSContentViewController: UIScrollViewDelegate
{
    func scrollViewDidScroll(_ scrollView: UIScrollView) {
        
        if scrollView == collectionView {

            let offsetY = collectionView.contentOffset.y
            
            if canScroll == "0" {
                collectionView.contentOffset = CGPoint.zero
            }else{
                if offsetY >= 0 {
                    //子动 父不动
                    NotificationCenter.default.post(name: NSNotification.Name(rawValue: NOTIFICATION_SSInlineScrollVC_canScroll), object: nil, userInfo: [KEY_CANSCROLL : "0"])
                }else{
                    //子不动 父动
                    collectionView.contentOffset = CGPoint.zero
                    NotificationCenter.default.post(name: NSNotification.Name(rawValue: NOTIFICATION_SSInlineScrollVC_canScroll), object: nil, userInfo: [KEY_CANSCROLL : "1"])
                }
            }

        }
    }
}
3.横向纵向拖动的区分

切换分页,在横向拖动的过程中又进行了纵向拖动,此时我们希望视图不会上下跑,只是左右滚动

//指定一个手势需要另一个手势执行失败才会执行,同时触发多个手势使用其中一个手势的解决办法
//有时手势是相关联的,如单机和双击,点击和长按,点下去瞬间可能只会识别到单击无法识别其他,该方法可以指定某一个 手势,即便自己已经满足条件了,也不会立刻触发,会等到该指定的手势确定失败之后才触发
mainScrollView.panGestureRecognizer.require(toFail: containerView.containerGestureRecongizer)
4.索引指示器

索引指示器可以用UIScrollView实现,也可以用UICollectionView实现,根据个人喜好,建议数据量多的话最好用UICollectionView了。
如果用UICollectionView实现,当手动滚动索引指示器,当把选中的索引滚出屏幕显示范围,再点选别的,此时可能会产生由于复用引发的崩溃和选中状态错乱,要进行判空处理,swift中可以设置cell为Optional可选型类型。

private var _selectedIndex: Int! = 0
var selectedIndex: Int? {
    didSet {
            
        //获取上一个选中的cell
        let lastCell = segCollectionView.cellForItem(at: IndexPath.init(item: _selectedIndex, section: 0)) as? SSSegmentCell
        lastCell?.isSelected = false
            
        //赋新值
        _selectedIndex = selectedIndex
            
        //获取当前选中的cell
        let currentCell = segCollectionView.cellForItem(at: IndexPath.init(item: _selectedIndex, section: 0)) as? SSSegmentCell
        currentCell?.isSelected = true
            
        //指示器动画
        UIView.animate(withDuration: 0.25, animations: {
            self.indicatorView.snp.updateConstraints { (make) in
                make.left.equalTo(CGFloat(self._selectedIndex) * self.segCollectionItem_Width)
            }
            self.layoutIfNeeded()
        }) { (_) in
                
        }
            
        //滚动到可显示范围
        segCollectionView.scrollToItem(at: IndexPath.init(item: _selectedIndex, section: 0), at: .centeredHorizontally, animated: true)
            
        //判断不在屏幕内的cell 更新选中状态
        if lastCell == nil || currentCell == nil {
            segCollectionView.reloadData()
         }
    }
}
5.控制整体上下滚动的底层控制器

该控制器内的滚动视图可以是UIScrollView,也可以是UITableView,就可以将整个containerView部分添加到cell.contentView上,一样可以实现效果。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,372评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,368评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,415评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,157评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,171评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,125评论 1 297
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,028评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,887评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,310评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,533评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,690评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,411评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,004评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,659评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,812评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,693评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,577评论 2 353

推荐阅读更多精彩内容

  • 1、通过CocoaPods安装项目名称项目信息 AFNetworking网络请求组件 FMDB本地数据库组件 SD...
    阳明先生_X自主阅读 15,979评论 3 119
  • 老琢磨着春天该捂一捂,就像秋天的末尾跟人比谁穿的少一样,每到这个时候总是对自己的耐热程度倍加注意,厚厚的衣服舍不得...
    小小二萌阅读 203评论 0 1
  • 文/卷毛佟 第53篇原创文章 记得在去年的时候,我定了一个阅读计划,希望一年“读50本书”。促使我定下这个目标的原...
    卷毛佟阅读 1,208评论 1 14
  • 一 昨晚凌晨四点突然失眠,闲来无事想找个人聊天,便在同学群里发了句“还有夜猫要出来嗨的吗?” 没过一分钟尚彦回了句...
    一路向北去你的季节阅读 475评论 3 5
  • 《赠别》 往昔四载同未名, 今朝一逢似熟知。 梅朋远来多重喜, 更因兼容自由思。 网上加油无馅饼, 线下干杯有袖子...
    赛德传播阅读 389评论 0 0