点击状态栏回到scrollView回到最顶部

点击状态栏滚回顶部这个功能是系统自带的,只需要设置self.scrollView.scrollsToTop = YES即可,但是这个属性的前提是窗口下必须只有一个可滚动的View才有效果,这时候就需要自定义创建一个窗口来完成这个功能

  • 界面含有多个scrolView时,点击状态栏回到顶部失效
  • iOS9以后创建的window要给添加根控制器,否则会报错。可以通过dispatch_after来给添加窗口一个延时就可以不设置根控制器
  • 状态栏颜色交给控制器管理
  • 窗口是有级别的windowLevel,级别越高就越显示在顶部,如果级别一样,后添加的创建显示在顶部。级别分为三种,UIWindowLevelAlert > UIWindowLevelStatusBar > UIWindowLevelNormal

- (void)applicationDidBecomeActive:(UIApplication *)application {
    // 添加一个window,屏幕上的scrollView滚动到最顶部
    [MGTopWindow show];
}
@implementation MGTopWindow
// 全局对象
static UIWindow *topWindow_;

+ (void)initialize {

    topWindow_ = [[UIWindow alloc] init];
    topWindow_.frame = [UIApplication sharedApplication].statusBarFrame;
    topWindow_.windowLevel = UIWindowLevelAlert;
    topWindow_.backgroundColor = [UIColor clearColor];

    MGTopViewController *rootVc = [MGTopViewController shareInstance];
    topWindow_.rootViewController = rootVc;
}
@end

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

+ (void)hide
{
    topWindow_.hidden = YES;
}

点击状态栏,scrollView滚动最前面去

// MGTopViewController.m
- (void)touchesBegan:(NSSet *)touches withEvent:(UIEvent *)event
{
//    NSLog(@"%s", __func__);
    UIWindow *window = [UIApplication sharedApplication].keyWindow;
    [self searchScrollViewInView:window];
}

// 递归搜索所有view查找当前位置合适的scrollView
- (void)searchScrollViewInView:(UIView *)view
{
    for (UIScrollView *subView in view.subviews) {
        if ([subView isKindOfClass:[UIScrollView class]] && [self isShowingInKeyWindow:subView]) {
            //开始进行滚动
            CGPoint offset = subView.contentOffset;
            offset.y = -subView.contentInset.top;
            [subView setContentOffset:offset animated:YES];
        }
        //寻找子视图的子视图
        [self searchScrollViewInView:subView];
    }
}

// 根据位置判断是否合适
- (BOOL)isShowingInKeyWindow:(UIView *)view
{
    UIWindow *keyWindow = [UIApplication sharedApplication].keyWindow;
    CGRect currentFrame = [keyWindow convertRect:view.frame fromView:view.superview];
    CGRect winBounds = keyWindow.bounds;
    BOOL intersects = CGRectIntersectsRect(currentFrame, winBounds);
    return !view.isHidden && view.alpha > 0.01 && view.window == keyWindow && intersects;
}
说明
  • 点击状态栏会将所有的scrollView全部滚回顶部,这里需要判断当前view是否与窗口有重叠,重叠才滚回顶部
  • 判断不是同一坐标系的2个View是否重叠,并且如果anotherView为空的话,就返回窗口

window的控制器决定状态栏的显示隐藏和样式

// MGTopViewController.m
#pragma mark - 状态栏控制
- (BOOL)prefersStatusBarHidden
{
    return self.statusBarHidden;
}

#pragma mark - 重写setter方法
- (void)setStatusBarHidden:(BOOL)statusBarHidden
{
    _statusBarHidden = statusBarHidden;
    [self setNeedsStatusBarAppearanceUpdate];
}

创建了一个控件,就是看不见

  1. 当前控件没有添加到父控件中
  2. 当前控件的hidden = YES
  3. 当前控件的alpha <= 0.01
  4. 没有设置尺寸(frame.size、bounds.size)
  5. 位置不对(当前控件显示到窗口以外的区域)
  6. 背景色是clearColor
  7. 当前控件被其他可见的控件挡住了
  8. 当前控件是个显示图片的控件(没有设置图片\图片不存在,比如UIImageView)
  9. 当前控件是个显示文字的控件(没有设置文字\文字颜色跟后面的背景色一样,比如UILabel、UIButton)
  10. 检查父控件的前9种情况

一个控件能看见,但是点击后没有任何反应:

  1. 当前控件的userInteractionEnabled = NO
  2. 当前控件的enabled = NO
  3. 当前控件不在父控件的边框范围内
  4. 当前控件被一个背景色是clearColor的控件挡住了
  5. 检查父控件的前4种情况

参考资料

  • 解决点击状态栏时ScrollView自动滚动到初始位置失效办法
  • 如何让页面中有多个UIScrollView时支持statusbar点击回顶部的功能
  • ios-自定义点击状态栏滚回顶部

你可能感兴趣的:(点击状态栏回到scrollView回到最顶部)