50

我正在尝试隐藏我的一个视图控制器的状态栏(当模态显示时)。当我呈现视图控制器时,状态栏将被隐藏,然后在关闭时返回。

我已将以下代码添加到呈现的视图控制器中

- (BOOL)prefersStatusBarHidden
{
    return YES;
}

我还将 Info.plist 文件中的键设置为以下内容:

<key>UIViewControllerBasedStatusBarAppearance</key>
<true/>

据我了解,这应该是完成这项工作所需的全部内容。

我还使用自定义动画控制器来进行符合UIViewControllerAnimatedTransitioning协议的呈现。在animateTransition:实现中,我尝试手动调用prefersStatusBarHidden,然后setNeedsStatusBarAppearanceUpdate确保正在调用,但状态栏仍然存在。

任何想法为什么会发生这种情况将不胜感激。我已经搜索过 StackOverflow,但似乎没有人遇到过这个问题,所有接受的答案都是关于调用setNeedsStatusBarAppearanceUpdate的,我已经在这样做了。

编辑-下面的代码现在似乎可以按需要工作

- (void)animateTransition:(id <UIViewControllerContextTransitioning>)transitionContext
{
    if (self.isPresenting) {
        UIView *containerView = [transitionContext containerView];

        UIViewController *fromViewController = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];
        UIViewController *toViewController = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];

        toViewController.view.frame = containerView.frame;

        [containerView addSubview:toViewController.view];

        // Ask the presented controller whether to display the status bar
        [toViewController setNeedsStatusBarAppearanceUpdate];

        [UIView animateWithDuration:1.0f delay:0.0f options:UIViewAnimationOptionCurveEaseIn animations:^{
            toViewController.view.alpha = 1.0f;
            fromViewController.view.alpha = 0.0f;
        } completion:^(BOOL finished) {
            [transitionContext completeTransition:YES];
        }];
    }
    else {
        // do the reverse
        UIViewController *fromViewController = [transitionContext viewControllerForKey:UITransitionContextFromViewControllerKey];
        UIViewController *toViewController = [transitionContext viewControllerForKey:UITransitionContextToViewControllerKey];

        [UIView animateWithDuration:1.0f delay:0.0f options:UIViewAnimationOptionCurveEaseIn animations:^{
            toViewController.view.alpha = 1.0f;
            fromViewController.view.alpha = 0.0f;
        } completion:^(BOOL finished) {
            [transitionContext completeTransition:YES];
            // Once dismissed - ask the presenting controller if the status bar should be presented
            [toViewController setNeedsStatusBarAppearanceUpdate];
        }];
    }
}

....

// PresentingController.m
- (BOOL)prefersStatusBarHidden
{
    if (self.presentedViewController) {
        return YES;
    }
    return NO;
}

// PresentedController.m
- (BOOL)prefersStatusBarHidden
{
    return YES;
}
4

4 回答 4

129

在 iOS7 中,实际上有一个 UIViewController 的新属性叫做modalPresentationCapturesStatusBarAppearance. 苹果 iOS 参考。

默认值为 NO。

当您通过调用 presentViewController:animated:completion: 方法呈现视图控制器时,仅当呈现控制器的 modalPresentationStyle 值为 UIModalPresentationFullScreen 时,状态栏外观控制才会从呈现转移到呈现的视图控制器。通过将此属性设置为 YES,您可以指定呈现的视图控制器控制状态栏的外观,即使呈现非全屏。

对于全屏显示的视图控制器,系统会忽略此属性的值。

因此,对于普通全屏以外的任何presentationStyle(例如:UIModalPresentationCustom),如果要捕获状态栏,必须设置此项。要使用,您只需将其设置为YES正在呈现的视图控制器上:

toVC.modalPresentationCapturesStatusBarAppearance = YES;
于 2014-08-15T21:31:07.910 回答
17

我会猜测(受过教育,但仍然是猜测)这是因为当您使用自定义转换执行呈现的视图控制器时,在 iOS 7 中,旧的视图控制器仍然存在。因此,它可能仍然有发言权。

你甚至可以在其中 prefersStatusBarHidden放置一个断点来查看;如果没有实施,您将不得不实施它。默认值为 NO,因此如果咨询,这将解释您的结果。

如果我是对的,您将需要实现视图控制器prefersStatusBarHidden以给出两个不同的答案,具体取决于它是否具有presentedViewController

编辑我现在已经确认了这一点。比我想象的还要糟糕;在我的测试中,第二个视图控制器prefersStatusBarHidden根本没有被调用。整个事情都在第一个视图控制器的手中。这是有道理的,因为正如我所说,第一个视图控制器永远不会消失。使用自定义演示动画,第二个视图控制器从属于第一个视图控制器,因为第二个视图可以部分悬停在第一个视图上。

因此,您将不得不完全从第一个视图控制器驱动状态栏。您可以prefersStatusBarHidden通过调用来调用它[self setNeedsStatusBarAppearanceUpdate]。您需要根据具体情况给出不同的答案。这可能有点棘手。这是一个简单的实现,但它可能无法涵盖所有​​情况:

// ViewController1:

-(void)setHide:(NSNumber*)yn {
    self->hide = [yn boolValue]; // a BOOL ivar
    [self setNeedsStatusBarAppearanceUpdate];
}
-(BOOL)prefersStatusBarHidden {
    return self->hide;
}
- (IBAction)doButton:(id)sender {
    self->hide = YES;
    [self setNeedsStatusBarAppearanceUpdate];
    [self presentViewController:[ViewController2 new] animated:YES completion:nil];
}

// ==========

// ViewController2:

- (IBAction)doButton:(id)sender {
    [self.presentingViewController setValue:NO forKey:@"hide"];
    [self.presentingViewController dismissViewControllerAnimated:YES completion:nil];
}
于 2014-05-12T19:08:43.583 回答
1

如果它不起作用并且您的 UIViewController 是 UINavigationController 中的子项,那么此代码可能是您的解决方案。

open override var prefersStatusBarHidden: Bool {
    return topViewController?.prefersStatusBarHidden ?? super.prefersStatusBarHidden
}

基本上 UINavigationController 使用它自己的prefersStatusBarHidden值,但在我的情况下,我想通过其顶部视图控制器在层次结构中的属性覆盖它。

于 2019-07-23T10:33:38.473 回答
0

您可以将此添加到 info.plist

“查看基于控制器的状态栏外观”并将值设置为“否”

于 2018-05-22T07:46:26.227 回答