iOS 知识收集

ios 解决 scrollview嵌套tableview手势冲突

2020-08-25  本文已影响0人  FSDemo

1.scrollView 嵌套 tableView 类冲突

这里直接用 scrollView(后面称父视图) 嵌套 tableView(后面称子视图) 来处理下滑动时的手势冲突问题,其实苹果并不建议我们这样做,但是在实际项目中,有些需求会经常用嵌套来实现,在什么情况下滑动 tableView 不滑动 scrollView,什么情况下滑动 scrollView 不滑动 tableView,其实如果做其他的嵌套都是一样的,先看下最终效果图:


00688aVuzy7BziYJR1771&690.gif

1)首先新建一个基于 UIScrollView 的 ADABaseScrollView ,并实现 代理,ADABaseScrollView 用做主父试图来添加子试图内容

ADABaseScrollView.h

import

@interface ADABaseScrollView : UIScrollView

@end

ADABaseScrollView.m

import "ADABaseScrollView.h"

@implementation ADABaseScrollView

//是否支持多时候触发,这里返回YES

-(BOOL)gestureRecognizer:(UIGestureRecognizer *)gestureRecognizer shouldRecognizeSimultaneouslyWithGestureRecognizer:(UIGestureRecognizer *)otherGestureRecognizer{

return YES;

}

@end

2)然后新建一个基于 UITableView 的 ADATargetTableView ,并实现 代理

ADATargetTableView.h

import

@interface ADATargetTableView : UITableView

///可否滑动

@property (nonatomic,assign) BOOL canSlide;

///滑动block通知

@property (nonatomic,copy) void (^slideDragBlock)(void);

@end

ADATargetTableView.m

import "ADATargetTableView.h"

@interface ADATargetTableView ()

@property (nonatomic,assign) CGFloat currOffsetY;

@end

@implementation ADATargetTableView

}

//是否支持多时候触发,这里返回YES

}

pragma mark ========== tableView 代理 ==========

}

}

}

pragma mark ========== scrollview 代理 ==========

}

//重要

-(void)scrollViewDidScroll:(UIScrollView *)scrollView{

if (!self.canSlide) { // 如果不能滑动,保持滑动的偏移,不再滑动

    scrollView.contentOffset = CGPointMake(0, scrollView.contentOffset.y == 0 ? 0 : _currOffsetY);

}

_currOffsetY = scrollView.contentOffset.y; // 记录滑动的偏移

if (scrollView.contentOffset.y < 0 ) { // 触发自己不能滑动的效果,(当自己从上向下,滑到顶部的时候,应该不能滑了。)

    self.canSlide = NO; // 记录自己不能滑动了。偏移设置默认值,并通知外部

    scrollView.contentOffset = CGPointZero;

    //到顶通知父视图改变状态

    if (self.slideDragBlock) {

        self.slideDragBlock();

    }

}

scrollView.showsVerticalScrollIndicator = self.canSlide ? YES : NO;  // 按照能不能滑动,来显示滑动条

}

@end

import "ViewController.h"

import "SDAutoLayout.h"

import "ADABaseScrollView.h"

import "ADATargetTableView.h"

@interface ViewController ()

///容器

@property (nonatomic,strong) ADABaseScrollView *scrollView;

@property (nonatomic,strong) ADATargetTableView *tableView;

///是否可以滑动 scrollView

@property (nonatomic,assign) BOOL canSlide;

/// 记录偏移量

@property (nonatomic,assign) CGFloat lastPositionY;

///滑动临界范围值,到这个位置就不能再上滑了。

@property (nonatomic,assign) CGFloat dragCriticalY;

@end

@implementation ViewController

[self.scrollView setupAutoContentSizeWithBottomView:self.tableView bottomMargin:0];

__weak __typeof__(self) weekSelf = self;

self.tableView.slideDragBlock = ^{

    weekSelf.canSlide = YES;

    weekSelf.tableView.canSlide = NO;

};

}

//重要

-(void)scrollViewDidScroll:(UIScrollView *)scrollView{

CGFloat currentPostion = scrollView.contentOffset.y;



if (currentPostion >= self.dragCriticalY) { // 向上到临界值

    scrollView.contentOffset = CGPointMake(0, self.dragCriticalY);

    if (self.canSlide) {

        self.canSlide = NO;

        self.tableView.canSlide = YES;

    } else {

        if (_lastPositionY - currentPostion > 0){ // 上一次的滚动值大于现在的滚动值,意思是向下滚动。

            if (self.tableView.contentOffset.y > 0) { // 如果子视图向上偏移了。子视图可以继续向下

                self.tableView.canSlide = YES;

                self.canSlide = NO;

            } else {

                self.tableView.canSlide = NO; // 把子视图的位置滑完为止

                self.canSlide = YES;

            }

        }

    }

} else {

    if (!self.canSlide && scrollView.contentOffset.y ==  self.dragCriticalY ) { // 当到达临界值时,不是初始值。应该给父视图保持偏移

        scrollView.contentOffset = CGPointMake(0, self.dragCriticalY);

    } else {

        if (self.tableView.canSlide &&

            self.tableView.contentOffset.y != 0) { // 当子视图能滚动,并且子视图偏移了。应该给父视图保持偏移

            scrollView.contentOffset = CGPointMake(0, self.dragCriticalY);

        } else{

            

        }

    }

}



_lastPositionY = currentPostion;

}

}

}

@end

注:文章出自http://blog.sina.com.cn/s/blog_14ecbf39401030nh7.html

上一篇下一篇

猜你喜欢

热点阅读