上一篇博客渐变导航栏中,在渐变的过程中需要判断滚动的方向,滚动停止,改变方向,这篇博客,博主将对这三个问题分别进行说明。
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会突兀的出现,所以需要把向上下滚动的透明度结合起来,博主改进后会发出来,各位也可以自己尝试下,思路很明了,一些小技巧就可以实现。