可以知道的是,iphone设备在通话或录音状态中,状态栏下移了20个像素,这时controller的view的frame的高度就减少了20个像素,所以我们需要根据view的frame去布局子view。
看了一下uiviewcontroller的api,发现在ios5.0以后多增加了两个函数:
- (void)viewwilllayoutsubviews
- (void)viewdidlayoutsubviews
试了一下,发现如果一个controller是用系统方式压栈进去的,但系统状态栏改变的时候,会回调这个两个函数,所以我们可以在这两个函数里重新获取self.view.frame进行子view的布局。(已经测试过,在iphone4s,当状态栏下移的时候,self.view.frame的高度变成440,还原的时候复原为460)
但是通常系统的压栈方式是不符合我们的需求的,如果我们自定义一个containerviewcontroller,例如自己的tabbarcontroller,通过addsubview的方式显示controller,就不会回调这两个函数。这时,我们就可以监听状态栏变化的通知,对view进行重布局。代码如下:
//在init的时候监听状态栏改变的通知uiapplicationdidchangestatusbarframenotification
-(id)initwithnibname:(nsstring *)nibnameornil bundle:(nsbundle *)nibbundleornil
{
self = [super initwithnibname:nibnameornil bundle:nibbundleornil];
if(self){
[[nsnotificationcenter defaultcenter] addobserver:selfselector:@selector(layoutcontrollersubviews) name:uiapplicationdidchangestatusbarframenotification object:nil];
}
return self;
}
看了一下uiviewcontroller的api,发现在ios5.0以后多增加了两个函数:
- (void)viewwilllayoutsubviews
- (void)viewdidlayoutsubviews
试了一下,发现如果一个controller是用系统方式压栈进去的,但系统状态栏改变的时候,会回调这个两个函数,所以我们可以在这两个函数里重新获取self.view.frame进行子view的布局。(已经测试过,在iphone4s,当状态栏下移的时候,self.view.frame的高度变成440,还原的时候复原为460)
但是通常系统的压栈方式是不符合我们的需求的,如果我们自定义一个containerviewcontroller,例如自己的tabbarcontroller,通过addsubview的方式显示controller,就不会回调这两个函数。这时,我们就可以监听状态栏变化的通知,对view进行重布局。代码如下:
//在init的时候监听状态栏改变的通知uiapplicationdidchangestatusbarframenotification
-(id)initwithnibname:(nsstring *)nibnameornil bundle:(nsbundle *)nibbundleornil
{
self = [super initwithnibname:nibnameornil bundle:nibbundleornil];
if(self){
[[nsnotificationcenter defaultcenter] addobserver:selfselector:@selector(layoutcontrollersubviews) name:uiapplicationdidchangestatusbarframenotification object:nil];
}
return self;
}
这时,我们就可以在layoutcontrollersubviews对controller的子view进行重布局。
最主要的还是在项目一开始就要考虑到这个问题,在一些UI元素是据底部0的时候考虑会有这种情况的发生