iOS自定义视图系列(二)-状态栏(点击状态栏返回到顶部效果)

iOS有一个特有的功能--点击状态栏会使UIScollView自动滚动到顶部,这个效果其实是系统API里UIScrollView的一个属性

 @property(nonatomic) BOOL scrollsToTop

这个属性默认是YES,也就是支持点击状态栏会使UIScollView自动滚动到顶部的效果(如果不想要这个效果设置为NO就可以了).

 但是有一个局限就是如果在当前的屏幕上有
 "多于1个"UIScollView(包括所有继承自UIScollView)视图的"scrollsTop属性是YES"的时候,
 这个效果会失效.

本篇文章就是要解决点击状态栏会使UIScollView自动滚动到顶部失效的问题.

本人最终的解决思路是:

在点击状态栏的手势方法里把所有的scrollView的contentOffSize设置为CGPointZero

当然在此之前需要做一些判断的逻辑,针对我的项目我做了如下判断:

把不需要滚动到顶部的scrollView筛选掉
if ([scrollView isKindOfClass:[TuiJianCollectionView class]])return;
if ([scrollView isKindOfClass:[TopView class]]) return;
if (scrollView.width < ScreenWidth-100)return;
//然后就可以滚动了
[scrollView setContentOffset:CGPointZero animated:YES];

好了,核心的东西说完了,上代码
LQTopStatusWindow.h
#import
@interface LQTopStatusWindow : NSObject
/*
在需要用到自定义状态栏的页面显示(具体用法见注2)
/
+ (void)show;
/

在不需要的页面隐藏(具体用法见注2)
*/
+(void)hidden;
@end
LQTopStatusWindow.m

#import "LQTopStatusWindow.h"
@implementation LQTopStatusWindow

static UIWindow *_topWindow = nil;
/**
 * 显示顶部窗口
 */
+ (void)initialize
{
    _topWindow = [[UIWindow alloc] init];
    _topWindow.windowLevel = UIWindowLevelAlert;
    CGRect frame = [UIApplication sharedApplication].statusBarFrame;
   //下边设置frame为什么�需要加50,原因见注1
    _topWindow.frame = CGRectMake(frame.origin.x + 50, frame.origin.y, frame.size.width, frame.size.height);
    _topWindow.backgroundColor = [UIColor clearColor];
    _topWindow.hidden = NO;
    [_topWindow addGestureRecognizer:[[UITapGestureRecognizer alloc] initWithTarget:self action:@selector(topWindowClick)]];

}

+ (void)show{
    _topWindow.hidden = NO;
}
+ (void)hidden{
    _topWindow.hidden = YES;
}

/**
 * 监听顶部窗口点击
 */
+ (void)topWindowClick
{
    UIWindow *keyWindow = [UIApplication sharedApplication].keyWindow;
    [self searchAllScrollViewsInView:keyWindow];
}

/**
 * 找到参数view中所有的UIScrollView
 */
+ (void)searchAllScrollViewsInView:(UIView *)view
{
    // 递归遍历所有的子控件
    for (UIView *subview in view.subviews) {
        [self searchAllScrollViewsInView:subview];
    }
    // 判断子控件类型(如果不是UIScrollView,直接返回)
    if (![view isKindOfClass:[UIScrollView class]]) return;

    // 找到了UIScrollView
    //过滤掉不需要滚动到顶部的scrollview
    UIScrollView *scrollView = (UIScrollView *)view;
    if ([scrollView isKindOfClass:[TopView class]]) return;
    if (scrollView.width < ScreenWidth-100) return;
    //让所有需要滚动到顶部的scrollView滚动到顶部
    [scrollView setContentOffset:CGPointZero animated:YES];
}

好了,代码贴完,具体用法是: 不用写别的代码.

 注1:此处没有让frame直接与frame(系统的状态栏的frame)完全相同,是因为,iOS应用间跳转的时候左
    上角有"返回XX"按钮,为了不影响它的点击这里让自定义的_topWindow向右移动了50(这个是试出来的)
    _topWindow.frame = CGRectMake(frame.origin.x + 50, frame.origin.y, frame.size.width, frame.size.height);

 注2:当你有需要在某些页面

如果您只是为了解决当前问题,请关闭本文;如果您有一颗程序猿的进取之心,想了解我解决此问题的过程,请继续往下读:

解决过程一:

首先我知道了"点击状态栏会使UIScollView自动滚动到顶部"这个效果是由下边这个属性决定的
   @property(nonatomic) BOOL scrollsToTop
   而此效果失效的原因是当前的屏幕上有
 "多于1个"UIScollView(包括所有继承自UIScollView)视图的"scrollsTop属性是YES"的时候,

我产生了两个解决办法:
一 减少UIScollView的数量,让当前屏幕中只有一个UIScollView,但是我的项目中界面比较多,如果每个界面都重构,替换scrollView的工作量会很大,而且时间不允许------PASS.
二 把其余不需要响应点击状态栏效果的scrollView的scrollsToTop设置为NO,由于办法一的原因-----PASS.

解决过程二:

求助度娘:果然有先贤已经处理过这个问题,下边附上我当时参考的两篇""上的文章
一 http://www.jianshu.com/p/e18356fccbc9
借鉴此文章,我自定义好了视图,并添加好了手势,但是有一个问题,这篇文章用了一个"bs_intersectsWithAnotherView:"方法,也没有写实现,我就暂时不知道该怎么办了,此路不通.
二 http://www.jianshu.com/p/68632cbd6263
后来我发现这篇文章中有"bs_intersectsWithAnotherView:"这个方法(不知道是不是一个作者,看名字好像不是),具体是

- (BOOL)bs_intersectsWithAnotherView:(UIView *)anotherView{ 
   if (anotherView == nil){
   anotherView = [UIApplication sharedApplication].keyWindow
   }; 
  // 判断self和anotherView是否重叠 
  CGRect selfRect = [self convertRect:self.bounds toView:nil];    
  CGRect anotherRect = [anotherView convertRect:anotherView.bounds toView:nil]; 
  return CGRectIntersectsRect(selfRect, anotherRect);
}

这个方法只用的判断两个矩形框是不是重叠的方法CGRectIntersectsRect()但是这个方法不适用于我的项目,因为与窗口重叠的scrollView也不是一个,此路不通.

解决过程三:

我观察了一下我需要此效果的scrollView的frame是一样的,而且只有一个,所以我在判断要不要滚动到顶部的手势方法里加了一个frame的判断,只有特定frame的scrollView需要滚动到顶部,然而因为不明原因(可能是判断太严格了)还是无法实现,此路不通.
解决过程三:
最后我无奈了,我发了一个大招,在点击状态栏的手势方法里把所有的scrollView的contentOffSize设置为CGPointZero.这下好了,果然在点击状态栏的时候,所有scrollView都滚动到顶部了.我以为这样就可以了,但是,但是,我发现我所有界面的scrollView包括大的,小的,继承自它的控件都滚动到顶部了.我最后加了几个判断,解决问题,详见上文中最终解决方案.

总结:我觉得我解决这个问题的方法不是最好的解决办法,希望您能给我提出宝贵的意见或建议,欢迎各位指点,留言!

你可能感兴趣的:(iOS自定义视图系列(二)-状态栏(点击状态栏返回到顶部效果))