前言
最近在iPhone 12 系列机型上开发项目时,发现使用项目提供的获取状态栏、导航栏高度方法获取到的高度是错误的,随后跟踪排查最终解决这个问题,所以自己想简单的总结一下问题原因和解决办法。
本文主要介绍问题原因和解决办法,最终提供一个能准确获取iPhone 状态栏、导航栏、TabBar高度的方法。
问题原因
出现问题的原因是,我们大多开发在使用获取状态栏、导航栏高度方法都是以下方法来获取的:
// 状态栏高度,iPhoneX 是判断是否为刘海屏
#define StatusBar_Height (isIPhoneX ? 44.0f : 20.0f)
在iOS 14系统以前,使用这种方法获取是没有问题的,但是在之后,刘海屏的手机状态栏高度就不在是统一的44px咯,根据下表我们看一看目前iOS 15系统上的各机型的状态栏高度:
机型 | 状态栏高度 |
iPhone XR/11 | 48px |
iPhone X/11 Pro/ 11 Pro Max/12 mini | 44px |
iPhone 12/12 Pro/Pro Max | 47px |
使用系统方法获取状态栏高度
可以看出使用上面方法已经不在满足我们的开发需求了,于是我就想到使用系统提供的方法来获取,通过 UIApplication
单例中的 statusBarFrame
属性获取状态状态栏改度:
[UIApplication sharedApplication].statusBarFrame.size.height;
我发现 iOS 13.0系统 之后,UIApplication
单例中的 statusBarFrame
属性被废弃了。不建议使用了,系统希望我们使用 UIStatusBarManager
类中的 statusBarFrame
来进行获取:
if (@available(iOS 13.0, *)) {
NSSet *set = [UIApplication sharedApplication].connectedScenes;
UIWindowScene *windowScene = [set anyObject];
UIStatusBarManager *statusBarManager = windowScene.statusBarManager;
return statusBarManager.statusBarFrame.size.height;
}
代码模块
为了统一和方便快速获取系统顶部和底部安全区、顶部状态栏和导航栏、底部 tabBar,我将新建一个分类来实现这些方法,方便项目全局引用:
在 UIDevice+S
tateHeight
.h
@interface UIDevice ()
/** 顶部安全区高度 **/
+ (CGFloat)dev_safeDistanceTop;
/** 底部安全区高度 **/
+ (CGFloat)dev_safeDistanceBottom;
/** 顶部状态栏高度(包括安全区) **/
+ (CGFloat)dev_statusBarHeight;
/** 导航栏高度 **/
+ (CGFloat)dev_navigationBarHeight;
/** 状态栏+导航栏的高度 **/
+ (CGFloat)dev_navigationFullHeight;
/** 底部导航栏高度 **/
+ (CGFloat)dev_tabBarHeight;
/** 底部导航栏高度(包括安全区) **/
+ (CGFloat)dev_tabBarFullHeight;
@end
在 UIDevice+S
tateHeight
.m
#import "UIDevice+StateHeight.h"
@implementation UIDevice ()
// 顶部安全区高度
+ (CGFloat)dev_safeDistanceTop {
if (@available(iOS 13.0, *)) {
NSSet *set = [UIApplication sharedApplication].connectedScenes;
UIWindowScene *windowScene = [set anyObject];
UIWindow *window = windowScene.windows.firstObject;
return window.safeAreaInsets.top;
} else if (@available(iOS 11.0, *)) {
UIWindow *window = [UIApplication sharedApplication].windows.firstObject;
return window.safeAreaInsets.top;
}
return 0;
}
// 底部安全区高度
+ (CGFloat)dev_safeDistanceBottom {
if (@available(iOS 13.0, *)) {
NSSet *set = [UIApplication sharedApplication].connectedScenes;
UIWindowScene *windowScene = [set anyObject];
UIWindow *window = windowScene.windows.firstObject;
return window.safeAreaInsets.bottom;
} else if (@available(iOS 11.0, *)) {
UIWindow *window = [UIApplication sharedApplication].windows.firstObject;
return window.safeAreaInsets.bottom;
}
return 0;
}
//顶部状态栏高度(包括安全区)
+ (CGFloat)dev_statusBarHeight {
if (@available(iOS 13.0, *)) {
NSSet *set = [UIApplication sharedApplication].connectedScenes;
UIWindowScene *windowScene = [set anyObject];
UIStatusBarManager *statusBarManager = windowScene.statusBarManager;
return statusBarManager.statusBarFrame.size.height;
} else {
return [UIApplication sharedApplication].statusBarFrame.size.height;
}
}
// 导航栏高度
+ (CGFloat)dev_navigationBarHeight {
return 44.0f;
}
// 状态栏+导航栏的高度
+ (CGFloat)dev_navigationFullHeight {
return [UIDevice vg_statusBarHeight] + [UIDevice vg_navigationBarHeight];
}
// 底部导航栏高度
+ (CGFloat)dev_tabBarHeight {
return 49.0f;
}
// 底部导航栏高度(包括安全区)
+ (CGFloat)dev_tabBarFullHeight {
return [UIDevice vg_statusBarHeight] + [UIDevice vg_safeDistanceBottom];
}
@end