zl程序教程

您现在的位置是:首页 >  移动开发

当前栏目

iOS开发 - ScrollView滚动时怎么判断滚动停止及滚动的方向

ios开发 怎么 判断 滚动 停止 方向 ScrollView
2023-09-11 14:21:22 时间

上一篇博客渐变导航栏中,在渐变的过程中需要判断滚动的方向,滚动停止,改变方向,这篇博客,博主将对这三个问题分别进行说明。

1.判断滚动停止
先看下代码:

- (void)scrollViewDidScroll:(UIScrollView *)scrollView{
currentPostion = scrollView.contentOffset.y;
[NSObject cancelPreviousPerformRequestsWithTarget:self];              
[self performSelector:@selector(scrollViewDidEndScrollingAnimation:) withObject:nil afterDelay:0.00001];
}

- (void)scrollViewDidEndScrollingAnimation:(UIScrollView *)scrollView{
    [NSObject cancelPreviousPerformRequestsWithTarget:self];
    //有nav时一开始的偏移量为-64,所以这里加上了64,如果是自定义nav则没有这种情况
    stopPosition = currentPostion + 64;
    NSLog(@"滑动停止:%f",stopPosition);
}

这种方法判断的是scrollView停止滚动时的位置,请注意一点,是我们手动滑动,然后scrollView自然停止滚动后的位置,但是在操作中,几乎不会有人等到scrollView停止后才回去滑动界面,所以当你拿不到停止的位置,恰好又改变了方向的时候,停止的位置还是初始位置,在中部滚动时改变方向,nav就会消失,有兴趣的可以用代码尝试,为了解决这个问题,我们继续看下面。

2.判断改变方向

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

    currentPostion = scrollView.contentOffset.y;

    if(currentPostion > lastPosition)
    {
        NSLog(@"Scroll Up");
    }
    else
    {
        NSLog(@"Scroll Down");
    }


    lastPosition = currentPostion;
   }

这里判断的是滚动的上下方向,但是博主觉得这部足矣判断未停止滚动强制改变方向时的位置,所以,有了下面的小方法。

3.判断改变方向时的位置

- (void)viewDidLoad {
    [super viewDidLoad];
     //一开始一定要设定向下滚动的标示,否则是进不去改变方向的if的,一开始我们认为只能向上滚动(scrollView的弹性可以向下拉一段距离,这个因为设置了便宜量大于0,所以可忽略)
    [[NSUserDefaults standardUserDefaults]setObject:@"1" forKey:@"first"];
    [[NSUserDefaults standardUserDefaults]synchronize];
}
- (void)scrollViewDidScroll:(UIScrollView *)scrollView{

    currentPostion = scrollView.contentOffset.y;

    if (currentPostion > 0) {
        if (currentPostion - _lastPosition >= 0) {
            //仔细看的话你会发现这里的if在每次改变方向时执行一次,在这里拿到当时的位置,然后删除标示,设定改变方向的标示,改变方向后的设置和这里一样的原理。所以,这里的代码每次改变方向时只执行一次,就无需判断停止时的位置,而是在改变方向时瞬间拿到当时的停止位置。不要说这里存在误差,执行的瞬间就可以拿到位置,所以误差几乎忽略不计。
            if ([[NSUserDefaults standardUserDefaults]objectForKey:@"first"]!=nil) {
                [[NSUserDefaults standardUserDefaults]removeObjectForKey:@"first"];
                [[NSUserDefaults standardUserDefaults]synchronize];

                [[NSUserDefaults standardUserDefaults]setObject:@"1" forKey:@"second"];
                [[NSUserDefaults standardUserDefaults]synchronize];
                //加64因为系统nav初始偏移量会为-64,自定义可不加
                stopPosition = currentPostion + 64;

            }


            _lastPosition = currentPostion;
            NSLog(@"Scroll Up    current:%f    last:%f    stop:%f",currentPostion,_lastPosition,stopPosition);

            self.navigationController.navigationBar.alpha = 1 - currentPostion / 400;


        }
        else
        {
            if ([[NSUserDefaults standardUserDefaults]objectForKey:@"second"]!=nil) {
                [[NSUserDefaults standardUserDefaults]removeObjectForKey:@"second"];
                [[NSUserDefaults standardUserDefaults]synchronize];

                [[NSUserDefaults standardUserDefaults]setObject:@"1" forKey:@"first"];
                [[NSUserDefaults standardUserDefaults]synchronize];
                stopPosition = currentPostion + 64;

            }
            _lastPosition = currentPostion;
            NSLog(@"Scroll Down    current:%f   last:%f    stop:%f",currentPostion,_lastPosition,stopPosition);
                      self.navigationController.navigationBar.alpha = (stopPosition - currentPostion)/200;

        }   
    } 
}

总结:其实仔细看的话,会发现上下的透明度在改变方向时是独立计算的,没有相关联,如果比较注意细节的话会发现nav会突兀的出现,所以需要把向上下滚动的透明度结合起来,博主改进后会发出来,各位也可以自己尝试下,思路很明了,一些小技巧就可以实现。