iOS旋屏开发实践

最近一段时间的开发都以iPhone为主,iPad相关的涉及不多。iPhone开发很少会有横屏的需求,因此关于旋屏的问题很久不接触了。最近有开发同学来问关于特定界面支持旋屏的问题,一查资料发现,随着iOS版本的升级先关API的接口发生了很多变化,而且网上的资料很多都是老版本的解决方案,并不能很好地帮助解决问题。经过几天断断续续地梳理,基本上把旋屏的大部分使用逻辑理清楚了,这里作一个统一的梳理。本文主要针对iOS7以后的版本,很多老版本的旋屏问题,不在本文的讨论范围。

一、如何不支持旋屏

以iPhone应用为例,如何保证一个app只支持竖屏,不支持旋屏呢。其实非常简单,只需要在下面工程设置中,只勾选Portrait即可。


iOS旋屏开发实践_第1张图片

对应的Info.plist配置文件也会改变,如下:



只要照着以上配置好,那么coding部分就不用关注任何旋屏的问题了。

二、常见的旋屏需求解决

首先要支持旋屏,以iPhone为例,一般都会支持上上、左、右三个方向,那么先进行下面的配置。

iOS旋屏开发实践_第2张图片

进行相关配置后,那么默认情况,所有的viewcontroller都支持横竖屏的切换。但是很多时候我们得到的需求是,多数的viewcontroller支持竖屏,只有个别的viewcontroller要求支持旋屏。
针对上述的需求,我们只需要弄清楚下面几个相关的回调或API使用即可:

1.是否支持旋屏
- (BOOL)shouldAutorotate
{
    return YES;
}

返回YES,表示该viewcontroller支持旋屏,否则不支持。

2.支持哪些方向
#if __IPHONE_OS_VERSION_MAX_ALLOWED > __IPHONE_8_4
- (UIInterfaceOrientationMask)supportedInterfaceOrientations
#else
- (NSUInteger)supportedInterfaceOrientations
#endif
{
    return UIInterfaceOrientationMaskAllButUpsideDown;
}

可通过该API进一步定制该viewcontroller支持哪些方向。

3.旋屏后,是否显示状态栏
- (BOOL)prefersStatusBarHidden
{
    return NO;
}

iOS8以后,系统的默认设置是竖屏显示状态栏,横屏不显示。如果希望横屏也显示状态栏,那么增加上面的回调,并且返回NO即可。

Important
特别需要说明的是,上述三个回调默认都是没有实现的,只有系统默认设置无法满足我们需求的时候,我们再考虑定制化这些回调的使用。

三、实际使用过程中可能会遇到的问题

其实上述三个API的使用,都非常简单。但是使用过程中,却会遇到很多莫名其妙的问题。这里把一些常见的问题总结一下。

1.shouldAutorotate回调没有被执行
- (BOOL)shouldAutorotate
{
    return YES;
}

造成这种情况的原因有很多种,但是共性的原因都是该回调被上级的UI链给劫持了。这里列举一个常见的情况:NavigationController。

如果入口的根viewcontroller这么写:

- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {
    // Override point for customization after application launch.
    self.window = [[UIWindow alloc] initWithFrame:[[UIScreen mainScreen] bounds]];
    self.mainViewController = [[ViewController alloc] init];
    BaseNavigationViewController *rootNavgationController = [[BaseNavigationViewController alloc] initWithRootViewController:self.mainViewController];
    rootNavgationController.navigationBarHidden = YES;
    self.window.rootViewController = rootNavgationController;
    self.window.backgroundColor = [UIColor whiteColor];
 
    [self.window makeKeyAndVisible];
    
    return YES;
}

那么viewcontroller的整个shouldAutorotate回调都会被NavigationController给劫持。通常的解决方案是实现一个NavigationController的子类,并增加下面的代码:

-(BOOL)shouldAutorotate {
    return self.topViewController.shouldAutorotate;
}

#if __IPHONE_OS_VERSION_MAX_ALLOWED > __IPHONE_8_4
- (UIInterfaceOrientationMask)supportedInterfaceOrientations
#else
- (NSUInteger)supportedInterfaceOrientations
#endif
{
    return self.topViewController.supportedInterfaceOrientations;
}

增加断点,可以发现每次旋屏时,上述的回调都会被执行。

注意:self.topViewController返回的对象时整个导航器UI链,最后被push出来的那个viewcontroller。

2.prefersStatusBarHidden回调没有被执行
- (BOOL)prefersStatusBarHidden
{
    return NO;
}

如果在Info.plist里面增加了下面的配置,并且值设置为NO,那么上面的回调就不会被执行。


参数"View controller-based status bar appearance"的作用:
iOS8以后,系统默认横屏时状态栏是隐藏的。
上述参数默认为YES,意味着用户可以通过在回调- (BOOL)prefersStatusBarHidden中进行配置,自定义状态栏旋屏后的隐藏状态。
如果设置为NO,prefersStatusBarHidden回调将不会被执行。意味着禁止用户自定义状态栏旋屏后的隐藏状态,使用系统的默认设置。

3.偶然会碰到的坑:主viewcontroller禁止旋屏,但是状态栏依旧会旋转。

像下面的截图:


iOS旋屏开发实践_第3张图片

看下Info.plist里面的配置,是否有Main storyboard file相关的配置,如果多余的话,去掉就好了。

四、demo示例

一个简单的测试demo
https://github.com/kmplayer/pracRotate

你可能感兴趣的:(iOS旋屏开发实践)