针对项目中出现的问题,一一列出解决。

iphoneX适配官方指南:https://developer.apple.com/ios/human-interface-guidelines/overview/iphone-x/


1.iPhoneX基本属性

启动图尺寸:1125px × 2436px(即 375pt × 812pt @3x))

iphoneX 屏幕高:812.0个点

导航栏高度+状态栏高度:88.0个点(导航栏高度仍是44个点,状态栏高度增高为44个点,所以刘海的高度并不是状态栏的高度。状态栏和导航栏平分了头部总的高度)

tabbar高度:83.0个点(原是固定49个点,增高了34个点。所以)

iOS 获取导航栏和状态栏高度 iphone11状态栏导航栏高度_状态栏高度


2.MJRefresh错位等 使用UIScrollview的控件的页面造成的页面错位

iOS11废弃了automaticallyAdjustsScrollViewInsets属性,需要使用scrollview的contentInsetAdjustmentBehavior属性。

if (@available(iOS 11.0, *)) {
        self.tableView.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever;
    } else {
        // Fallback on earlier versions
    }



3.替换写死的状态栏高度20 导航栏44  和64。安全起见替换写死的tabbar高度49

- (CGFloat)tabbarHeight{
    return self.tabBarController.tabBar.frame.size.height;
}

- (CGFloat)navigationBarHeight{
    return self.navigationController.navigationBar.frame.size.height;
}

- (CGFloat)naviBarAndStatusBarHeight{
    CGFloat height = self.navigationController.navigationBar.frame.size.height + [UIApplication sharedApplication].statusBarFrame.size.height;
    return height;
}




4.UITableView中的UITableViewStyleGrouped类型设置heightForHeaderInSection失效

在iOS11上,原来使用UITableViewStyleGrouped设置的sectionheader高度可能失效。本人发现两种解决办法

方法一:.在iOS11上tableview的style使用UITableViewStylePlain。

UITableViewStyle style = UITableViewStyleGrouped;
    if (@available(iOS 11.0,*)) {
        style = UITableViewStylePlain;
    }
    self.tableView = [[UITableView alloc] initWithFrame:CGRectMake(0, 0, SCREEN_WIDTH, SCREEN_HEIGHT) style:style];

然后在代理中照常设置就好了

- (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section{
    return yourHeight;
}

//注意,设置heightForHeaderInSection的同时也要实现这个代理。 如果不实现这个代理,仍然会出现间隙不均的情况。
- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section{
    UIView *header = [[UIView alloc] init];
    header.frame = CGRectMake(0, 0, SCREEN_WIDTH, yourHeight);
    return header;
}

方法二。所有系统下都使用UITableViewStyleGrouped,然后同时实现以下四个代理方法(在iOS11下,对应的sectionHeaderHeight和sectionHeaderView要成对存在,之前只设置sectionHeaderHeight也可以,所以规范起见:统一成双入对)

- (NSInteger)tableView:(UITableView *)tableView numberOfRowsInSection:(NSInteger)section{
    return 1;
}
- (CGFloat)tableView:(UITableView *)tableView heightForHeaderInSection:(NSInteger)section{
    return CountHeight(30);
}
- (UIView *)tableView:(UITableView *)tableView viewForHeaderInSection:(NSInteger)section{
    UIView *header = [[UIView alloc] init];
    header.frame = CGRectMake(0, 0, SCREEN_WIDTH, CountHeight(30));
    header.backgroundColor = SHBackgroundColor;
    return header;
}
- (CGFloat)tableView:(UITableView *)tableView heightForFooterInSection:(NSInteger)section{
    return 0.01;//注意这里!如果你不需要footerView,这里不能返回0,否则间距还是有问题
}
- (UIView *)tableView:(UITableView *)tableView viewForFooterInSection:(NSInteger)section{
    return [[UIView alloc] initWithFrame:CGRectZero];
}





5.contentInset失效。iOS11下引入了safeArea的概念。需要注意这几个属性:

1.UIScrollView的属性contentInsetAdjustmentBehavior
2.UIScrollView的只读属性adjustedContentInset
3.UIView的只读属性safeAreaInsets

iOS11下当你没有设置contentInsetAdjustmentBehavior时,系统会自动为你将试图添加到安全区域。

如下图所示区域

iOS 获取导航栏和状态栏高度 iphone11状态栏导航栏高度_iOS_02

所以,此时你如果仍然设置了contentInset,比如下面的代码

self.tableView.contentInset = UIEdgeInsetsMake(0, 0, 49, 0);

就会出现下方多了49个点。


解决办法仍然是两个:

方法一:

if (@available(iOS 11.0, *)) {
            self.tableView.contentInset = UIEdgeInsetsMake(headerH, 0, 0, 0);
        }else{//11系统以下,如果需要还是要加的
            self.tableView.contentInset = UIEdgeInsetsMake(headerH, 0, 49, 0);
        }



方法二:

if (@available(iOS 11.0, *)) {
            self.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever;
        }else{
        }
        self.contentInset = UIEdgeInsetsMake(headerH, 0, 49, 0);



6.代码块调用失效

Xcode9中原来设置的代码块可能会失效。这个问题比较蛋疼。你去多打开几个 多点一下edit,多换一下completion scopes可能就恢复了。

iOS 获取导航栏和状态栏高度 iphone11状态栏导航栏高度_导航栏_03

iOS 获取导航栏和状态栏高度 iphone11状态栏导航栏高度_导航栏_04



7.contentInsetAdjustmentBehavior详解

以下引自腾讯bugly博文。(简单地说就是UIScrollViewContentInsetAdjustmentNever相当于以前的automaticallyAdjustsScrollViewInsets=NO,此时你可屏蔽系统自动调整,布局由你自己支配。其他三个属性基本是系统帮你将视图添加到safeArea里面)

/* Configure the behavior of adjustedContentInset.
Default is UIScrollViewContentInsetAdjustmentAutomatic.
*/@property(nonatomic) UIScrollViewContentInsetAdjustmentBehavior contentInsetAdjustmentBehavior

adjustContentInset表示contentView.frame.origin偏移了scrollview.frame.origin多少;是系统计算得来的,计算方式由contentInsetAdjustmentBehavior决定。有以下几种计算方式:

1.UIScrollViewContentInsetAdjustmentAutomatic:如果scrollview在一个automaticallyAdjustsScrollViewContentInset = YES的controller上,并且这个Controller包含在一个navigation controller中,这种情况下会设置在top & bottom上 adjustedContentInset = safeAreaInset + contentInset不管是否滚动。其他情况下与UIScrollViewContentInsetAdjustmentScrollableAxes相同

2.UIScrollViewContentInsetAdjustmentScrollableAxes: 在可滚动方向上adjustedContentInset = safeAreaInset + contentInset,在不可滚动方向上adjustedContentInset = contentInset;依赖于scrollEnabled和alwaysBounceHorizontal / vertical = YES,scrollEnabled默认为yes,所以大多数情况下,计算方式还是adjustedContentInset = safeAreaInset + contentInset

3.UIScrollViewContentInsetAdjustmentNever: adjustedContentInset = contentInset

4.UIScrollViewContentInsetAdjustmentAlways: adjustedContentInset = safeAreaInset + contentInset

当contentInsetAdjustmentBehavior设置为UIScrollViewContentInsetAdjustmentNever的时候,adjustContentInset值不受SafeAreaInset值的影响。