美文网首页
点击状态栏回到scrollView回到最顶部

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

作者: geekAppke | 来源:发表于2017-06-17 04:58 被阅读284次

    点击状态栏滚回顶部这个功能是系统自带的,只需要设置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<UITouch *> *)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回到最顶部

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