美文网首页iOS 知识收集
ios 解决 scrollview嵌套tableview手势冲突

ios 解决 scrollview嵌套tableview手势冲突

作者: FSDemo | 来源:发表于2020-08-25 23:55 被阅读0次

    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

    • (instancetype)initWithFrame:(CGRect)frame style:(UITableViewStyle)style{

      self = [super initWithFrame:frame style:style];

      if (self) {

        self.backgroundColor = [UIColor whiteColor];
      
        self.delegate = self;
      
        self.dataSource = self;
      
        self.tableFooterView = [UIView new];
      
        [self registerClass:[UITableViewCell class] forCellReuseIdentifier:@"UITableViewCell"];
      

      }

      return self;

    }

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

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

      return YES;

    }

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

    • (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{

      return 20;

    }

    • (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath{

      return 50;

    }

    • (UITableViewCell *)tableView:(UITableView *)tableView cellForRowAtIndexPath:(NSIndexPath *)indexPath{

      UITableViewCell *cell = [tableView dequeueReusableCellWithIdentifier:@"UITableViewCell"];

      cell.textLabel.text = [NSString stringWithFormat:@"%ld",indexPath.row];

      return cell;

    }

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

    • (void)scrollViewWillBeginDragging:(UIScrollView *)scrollView{

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

    }

    //重要

    -(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

    • (void)viewDidLoad {

      [super viewDidLoad];

      // Do any additional setup after loading the view, typically from a nib.

      _dragCriticalY = 200;

      [self.view addSubview:self.scrollView];

      self.scrollView.sd_layout.

      topSpaceToView(self.view, 0).

      leftSpaceToView(self.view, 0).

      rightSpaceToView(self.view, 0).

      bottomSpaceToView(self.view, 0);

    [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;
    

    }

    • (ADABaseScrollView *)scrollView{

      if (!_scrollView) {

        _scrollView = [[ADABaseScrollView alloc]init];
      
        _scrollView.showsVerticalScrollIndicator = NO;
      
      
      
        _scrollView.delegate = self;
      
        _scrollView.backgroundColor = [UIColor redColor];
      
        UIView *view = [[UIView alloc]init];
      
        view.backgroundColor = [UIColor blueColor];
      
        
      
        [_scrollView addSubview:view];
      
        view.sd_layout.
      
        topSpaceToView(_scrollView, 0).
      
        leftSpaceToView(_scrollView, 0).
      
        rightSpaceToView(_scrollView, 0).
      
        heightIs(300);
      
        
      
        [_scrollView addSubview:self.tableView];
      
        self.tableView.sd_layout.
      
        topSpaceToView(view, 0).
      
        leftSpaceToView(_scrollView, 0).
      
        rightSpaceToView(_scrollView, 0).
      
        heightIs(self.view.bounds.size.height - (300 - self.dragCriticalY));
      

      }

      return _scrollView;

    }

    • (ADATargetTableView *)tableView{

      if(!_tableView){

        _tableView = [[ADATargetTableView alloc]initWithFrame:CGRectZero style:UITableViewStylePlain];
      

      }

      return _tableView;

    }

    @end

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

    相关文章

      网友评论

        本文标题:ios 解决 scrollview嵌套tableview手势冲突

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