点击状态栏滚回顶部这个功能是系统自带的,只需要设置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];
}
创建了一个控件,就是看不见
- 当前控件没有添加到父控件中
- 当前控件的hidden = YES
- 当前控件的alpha <= 0.01
- 没有设置尺寸(frame.size、bounds.size)
- 位置不对(当前控件显示到窗口以外的区域)
- 背景色是clearColor
- 当前控件被其他可见的控件挡住了
- 当前控件是个显示图片的控件(没有设置图片\图片不存在,比如UIImageView)
- 当前控件是个显示文字的控件(没有设置文字\文字颜色跟后面的背景色一样,比如UILabel、UIButton)
- 检查父控件的前9种情况
一个控件能看见,但是点击后没有任何反应:
- 当前控件的userInteractionEnabled = NO
- 当前控件的enabled = NO
- 当前控件不在父控件的边框范围内
- 当前控件被一个背景色是clearColor的控件挡住了
- 检查父控件的前4种情况