iOS 7:用代码解决视图控制器的View整体上移问题

如果你准备将你的老的 iOS 6 app 迁移到 iOS 7 上,那么你必须注意了。当你的老的 app 在 iOS 7 设备上运行时,所有ViewController 的视图都整体上移了,因为 iOS 7 把整个屏幕高度(包括状态栏和导航栏)都作为了视图控制器的有效高度。于是你的视图上移了,并和上层的状态栏交叠在一起。

你当然可以在 Xcode 中修改每个 View,将他们下移20个像素(状态栏高度)或者64个像素(状态栏+导航栏高度)。

但是苹果显然已经考虑到这个问题,他们在 iOS 7 SDK 中为 ViewController 提供了一个 edgesForExtendedLayout 新属性。如果你将这个属性设置为UIRectEdgeNone,则 viewController 的所有子视图都会自动调整,这样在 iOS 7 下看到的效果和 iOS 6 完全一样。

为了方便,你可以为 UIViewController 扩展一个子类,并覆盖它的 viewDidLoad 方法:

@implementation DerivedViewController

- (void)viewDidLoad

{

    [superviewDidLoad];

if ([selfrespondsToSelector:@selector(edgesForExtendedLayout)])

       self.edgesForExtendedLayout = UIRectEdgeNone;

}

@end

然后你以后所有的 ViewController 都从这个 DerivedViewController 类继承。

但不幸的是,我们的程序仍然有大量 iOS<7 的用户 ,我们无法立即抛弃对 iOS 6 的支持。无论 edgesForExtendedLayout 还是UIRectEdgeNone,都只能在 iOS7 下有效。对于 iOS 6,我将以上代码修改为:

- (void)viewDidLoad

{

    [superviewDidLoad];

#if__IPHONE_OS_VERSION_MAX_ALLOWED >= 70000

    if ([selfrespondsToSelector:@selector(edgesForExtendedLayout)])

       self.edgesForExtendedLayout = UIRectEdgeNone;

#else

    float barHeight =0;

    if (!isIPad()&& ![[UIApplication sharedApplication] isStatusBarHidden]) {

       barHeight+=([[UIApplication sharedApplication]statusBarFrame]).size.height;

    }

    if(self.navigationController &&!self.navigationController.navigationBarHidden) {

       barHeight+=self.navigationController.navigationBar.frame.size.height;

    }

    for (UIView *viewin self.view.subviews) {

       

       if ([view isKindOfClass:[UIScrollView class]]) {

           view.frame = CGRectMake(view.frame.origin.x, view.frame.origin.y +barHeight, view.frame.size.width, view.frame.size.height - barHeight);

        } else {

           view.frame = CGRectMake(view.frame.origin.x, view.frame.origin.y +barHeight, view.frame.size.width, view.frame.size.height);

       }

    }

#endif

}

通过宏 __IPHONE_OS_VERSION_MAX_ALLOWED 判断 deployment target 是否 >7.0。>7.0则使用新的 edgesForExtendedLayout API,负责使用比较笨的方法逐个下移 subviews,并自动根据状态栏/导航栏的可视状态计算要移动的偏移量。

注:如果已升级至Xcode5,将导航控制器的 Top Bar 设置为一种“Opacque ...”(不透明)类型可解决此问题

  • 4
    点赞
  • 15
    收藏
    觉得还不错? 一键收藏
  • 3
    评论
iOS 中的视图控制器是非常重要的组件,它们负责管理应用程序的界面,并且需要及时释放以避免内存泄漏。下面以一个简单的例子来说明视图控制器的释放。 假设我们有一个包含两个视图控制器的应用程序,分别是 AViewController 和 BViewController。AViewController 是主界面,BViewController 是从 AViewController 跳转过来的界面。 在 AViewController 中,我们使用如下代码跳转到 BViewController: ``` BViewController *bvc = [[BViewController alloc] initWithNibName:@"BViewController" bundle:nil]; [self.navigationController pushViewController:bvc animated:YES]; ``` 在 BViewController 中,我们可以通过点击返回按钮来返回到 AViewController。这个过程中,AViewController 会被重新显示出来,但是 BViewController 已经不需要了,需要将其释放。 如果我们使用 ARC,那么只需要在 BViewController 中添加如下代码即可: ``` - (void)dealloc { NSLog(@"BViewController dealloc"); } ``` 当 BViewController 被释放时,会自动调用 dealloc 方法,我们可以在这里添加一些清理工作,例如取消网络请求、除通知等。 如果我们使用手动内存管理(MRC),那么需要在 AViewController 中添加如下代码: ``` - (void)viewWillDisappear:(BOOL)animated { [super viewWillDisappear:animated]; if (![self.navigationController.viewControllers containsObject:self]) { // BViewController 被弹出了,需要释放 [self release]; } } ``` 当 BViewController 被弹出时,会触发 AViewController 的 viewWillDisappear 方法,在这里判断当前视图控制器是否还在导航栈中,如果不在就可以释放了。注意,这里需要调用 release 方法来手动释放内存。 总之,无论是使用 ARC 还是手动内存管理,都需要及时释放不再需要的视图控制器,以避免内存泄漏。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 3
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值