如何同时滑入/滑出状态栏和导航栏?

Posted

技术标签:

【中文标题】如何同时滑入/滑出状态栏和导航栏?【英文标题】:How to slide in/out statusBar and navigationBar simultaneously? 【发布时间】:2011-03-09 03:46:06 【问题描述】:

我想使用幻灯片效果同时显示和隐藏状态栏和导航栏。

我就是这样尝试的:

[[UIApplication sharedApplication] setStatusBarHidden:hide withAnimation:UIStatusBarAnimationSlide];
[self.navigationController setNavigationBarHidden:hide animated:animated];

但是,两个动画的持续时间是不一样的。状态栏动画需要更长的时间。 我发现无法指定任一动画的持续时间。 我错过了什么明显的东西吗?

【问题讨论】:

我向 Apple 提交了错误报告:openradar.appspot.com/8548087 如果您也受到影响,请上当。 【参考方案1】:

ios-lizard 的答案几乎是我想要的,但导航栏会在旋转设备时重新出现,除非 hidden 设置正确。所以这对我有用:

隐藏动画作品/看起来不错耶!!。

显示动画没问题,(我希望我可以让状态栏与导航栏一起滑动,但至少我们看不到差距了。:D

- (void)toggleFullscreen 

    UINavigationBar *navBar = self.navigationController.navigationBar;
    CGRect statusBarFrame = [[UIApplication sharedApplication] statusBarFrame];
    float animationDuration;
    if(statusBarFrame.size.height > 20)  // in-call
        animationDuration = 0.5;
     else  // normal status bar 
        animationDuration = 0.6;
    

    _fullscreen = !_fullscreen;
    if (_fullscreen)  
        // Change to fullscreen mode
        // Hide status bar and navigation bar
        [[UIApplication sharedApplication] setStatusBarHidden:YES
                                                withAnimation:UIStatusBarAnimationSlide];
        [UIView animateWithDuration:animationDuration animations:^
            navBar.frame = CGRectMake(navBar.frame.origin.x,
                                  -navBar.frame.size.height,
                                  navBar.frame.size.width,
                                  navBar.frame.size.height);
         completion:^(BOOL finished) 
            [self.navigationController setNavigationBarHidden:YES animated:NO];
        ];

     else 
        // Change to regular mode
        // Show status bar and navigation bar
        [[UIApplication sharedApplication] setStatusBarHidden:NO
                                                withAnimation:UIStatusBarAnimationSlide];
        [UIView animateWithDuration:animationDuration animations:^
             navBar.frame = CGRectMake(navBar.frame.origin.x,
                                       statusBarFrame.size.height,
                                       navBar.frame.size.width,
                                       navBar.frame.size.height);
         completion:^(BOOL finished) 
            [self.navigationController setNavigationBarHidden:NO animated:NO];
        ];

    


【讨论】:

【参考方案2】:

这就是我为我的应用解决此问题的方法。

    CGRect statusBarFrame = [[UIApplication sharedApplication] statusBarFrame];

    // delta is the amount by which the nav bar will be moved
    delta = statusBarFrame.size.height + self.navigationController.navigationBar.frame.size.height;

    if(statusBarFrame.size.height>20)  // in-call
        animationDuration = 0.5;
    
    else  // normal status bar 
        animationDuration = 0.6;
    

    // hide status bar
    [[UIApplication sharedApplication] setStatusBarHidden:YES withAnimation:UIStatusBarAnimationSlide];

    // hide nav bar
    [UIView beginAnimations:nil context:nil];
    [UIView setAnimationDuration:animationDuration];

    self.navigationController.navigationBar.frame = CGRectOffset(self.navigationController.navigationBar.frame, 0.0, -delta);

    [UIView commitAnimations];

【讨论】:

【参考方案3】:

显然,要做到这一点并没有简单的解决方案。苹果必须fix it。

当然,一种解决方法是按照 Ephraim 的建议使用 alpha 衰减。如果您坚持滑动行为,我发现最好只为导航栏设置动画并隐藏/显示状态栏而不使用任何动画。这看起来比滑动状态栏要好得多,因为动画期间状态栏之间的间隙非常明显。

【讨论】:

【参考方案4】:

这是一个更简洁的方法,它使用系统常量来表示动画持续时间并处理传入的调用。

注意,navigationBar 是一个 outlet,statusBarHeight 是一个实例变量浮点数。

- (IBAction)toggleControls:(id)sender 
    BOOL isHidden = ! [UIApplication sharedApplication].statusBarHidden;
    if (isHidden)
        statusBarHeight = [UIApplication sharedApplication].statusBarFrame.size.height;
    [UIView animateWithDuration:[UIApplication sharedApplication].statusBarOrientationAnimationDuration animations:^
        self.navigationBar.frame = CGRectMake(self.navigationBar.frame.origin.x,
                                              isHidden ? -self.navigationBar.frame.size.height : statusBarHeight,
                                              self.navigationBar.frame.size.width,
                                              self.navigationBar.frame.size.height);
    ];
    [[UIApplication sharedApplication] setStatusBarHidden:isHidden withAnimation:UIStatusBarAnimationSlide];

【讨论】:

【参考方案5】:

nacho4d 的回答几乎是我想要的。但是,他在 navBar 可见之前改变了 navBar 的框架。所以我们看不到过渡动画。看起来navBar突然出现了。更何况,显示的时候statusBarFrame.size.height等于0。下面是他的代码:

[[UIApplication sharedApplication] setStatusBarHidden:NO
                                            withAnimation:UIStatusBarAnimationSlide];
    [UIView animateWithDuration:animationDuration animations:^
         navBar.frame = CGRectMake(navBar.frame.origin.x,
                                   statusBarFrame.size.height,
                                   navBar.frame.size.width,
                                   navBar.frame.size.height);
     completion:^(BOOL finished) 
        [self.navigationController setNavigationBarHidden:NO animated:NO];
    ];

当 Showing 时,我们希望我们可以使状态栏与导航栏一起滑动。 这是我的答案

        UINavigationBar *navBar = self.navigationController.navigationBar;
        [[UIApplication sharedApplication] setStatusBarHidden:hidden withAnimation:UIStatusBarAnimationSlide];

        [UIView animateWithDuration:0.35 delay:0 options:UIViewAnimationOptionCurveEaseOut animations:^
            // make navigationBar visual
            if (!hidden)
            
                [self.navigationController setNavigationBarHidden:hidden animated:NO];
            

            navBar.frame = CGRectMake(navBar.frame.origin.x,
                                      hidden ? -navBar.frame.size.height : 20,
                                      navBar.frame.size.width,
                                      navBar.frame.size.height);
         completion:^(BOOL finished) 
            if (hidden)
            
                [self.navigationController setNavigationBarHidden:hidden animated:NO];
            
        ];
    隐藏时,隐藏等于NO。我们应该先改变导航栏的框架,然后隐藏导航栏。 显示时,隐藏等于YES。我们先让 navBar 可视化,然后改变框架。 我们选择UIViewAnimationOptionCurveEaseOut,让它看起来更好。

【讨论】:

【参考方案6】:

您可以使用实例变量来执行此操作:

self.navigationController setNavigationBarHidden:hide animated:animated];
_shouldHideStatusBar = hide;

并实现如下功能:

- (BOOL)prefersStatusBarHidden
    return _shouldHideStatusBar;

setNavigationBarHidden:animated函数会自动调用prefersStatusBarHidden函数。如果没有,您可以使用以下 UIViewController 的方法调用它:

[self setNeedsStatusBarAppearanceUpdate];

当然你可以选择你的状态栏隐藏动画样式:

- (UIStatusBarAnimation) preferredStatusBarUpdateAnimation 
    return UIStatusBarAnimationSlide;

【讨论】:

【参考方案7】:

这不是一个很好的答案,但它确实有效。所以我做的是:

// This method gets called by whatever action you want

- (void) toggleShowStatusNavBars:(id)sender 

    // Assuming you have a ivar called barsHidden

    [UIView beginAnimations:nil context:nil];
    [UIView setAnimationDuration:0.4]; // This is IMPORTANT, 0.4s

    self.navigationController.navigationBar.alpha = (barsHidden?1.0:0.0);  

    barsHidden = !barsHidden; 

    [UIView setAnimationDelegate:self];
    [UIView setAnimationWillStartSelector:@selector(setStatusBarHidden)];

    [UIView commitAnimations];


- (void) setStatusBarHidden 
    [[UIApplication sharedApplication] setStatusBarHidden:barsHidden animated:YES];

这将基本上同步动画的开始(因为您在导航栏动画开始时调用setStatusBarHidden。关键部分是状态栏动画似乎需要0.4秒。

这对我有用,但如果您找到更好的方法,请在此处发布。

【讨论】:

您正在使用 Alpha 渐变隐藏导航栏。这更容易,因为在这种情况下,您可以直接操作导航栏。在滑动案例中,我感兴趣的是这不是一种可行的方法,因为 navigationController 还操纵内容视图框架。

以上是关于如何同时滑入/滑出状态栏和导航栏?的主要内容,如果未能解决你的问题,请参考以下文章

点击设备时如何隐藏状态栏和导航栏

iOS 8 - 动画显示导航和状态栏无法正常工作

Android 显示、隐藏状态栏和导航栏

iOS 状态栏和导航栏颜色相同

如何添加覆盖所有其他视图(包括导航栏和状态栏)的暗屏?

如何使导航栏和状态栏模糊(UIBlurEffect)? iOS,斯威夫特 3