下面的代码隐藏并显示了导航栏。它在第一个视图加载时被隐藏,然后在“子”被调用时被隐藏。问题是,当他们回到根视图....时,我找不到触发它再次隐藏的事件/动作

我在根页面上有一个“测试”按钮,手动执行操作,但它并不漂亮,我希望它是自动的。

-(void)hideBar 
{
    self.navController.navigationBarHidden = YES;
}
-(void)showBar 
{       
    self.navController.navigationBarHidden = NO;
}

当前回答

我发现的最好的解决方案是在第一个视图控制器中执行以下操作。

objective - c

- (void)viewWillAppear:(BOOL)animated {
    [self.navigationController setNavigationBarHidden:YES animated:animated];
    [super viewWillAppear:animated];
}

- (void)viewWillDisappear:(BOOL)animated {
    [self.navigationController setNavigationBarHidden:NO animated:animated];
    [super viewWillDisappear:animated];
}

斯威夫特

override func viewWillAppear(_ animated: Bool) {
    self.navigationController?.setNavigationBarHidden(true, animated: animated)
    super.viewWillAppear(animated)
}

override func viewWillDisappear(_ animated: Bool) {
    self.navigationController?.setNavigationBarHidden(false, animated: animated)
    super.viewWillDisappear(animated)
} 

这将导致当你在堆栈上推下一个UIViewController时,导航栏从左边(连同下一个视图)动画,当你按下UINavigationBar上的后退按钮时,导航栏动画移到左边(连同旧的视图)。

请注意这些不是委托方法,你重写了UIViewController对这些方法的实现,根据文档,你必须在你的实现中调用super的实现。

其他回答

如果你想把导航栏完全隐藏在控制器中,一个更干净的解决方案是,在根控制器中,有这样的东西:

@implementation MainViewController
- (void)viewDidLoad {
    self.navigationController.navigationBarHidden=YES;
    //...extra code on view load  
}

当你在控制器中推送子视图时,导航栏将保持隐藏状态;如果你想只在子对象中显示它,你将在viewWillAppear回调中添加用于显示它的代码(self.navigationController.navigationBarHidden=NO;),类似地,在viewWillDisappear中隐藏它的代码

最简单的实现可能是让每个视图控制器在viewWillAppear:animated:方法中指定它的导航栏是否隐藏。同样的方法也适用于隐藏/显示工具栏:

- (void)viewWillAppear:(BOOL)animated {
    [self.navigationController setToolbarHidden:YES/NO animated:animated];
    [super viewWillAppear:animated];
}

我相信@chad-m的回答。

以下是Swift版本:

创建一个新文件MyNavigationController.swift


import UIKit

class MyNavigationController: UINavigationController, UINavigationControllerDelegate {

    override func viewDidLoad() {
        super.viewDidLoad()

        // Do any additional setup after loading the view.
        self.delegate = self
    }

    func navigationController(_ navigationController: UINavigationController, willShow viewController: UIViewController, animated: Bool) {
        if viewController == self.viewControllers.first {
            self.setNavigationBarHidden(true, animated: animated)
        } else {
            self.setNavigationBarHidden(false, animated: animated)
        }
    }

}

在StoryBoard中将UINavigationController的类设置为MyNavigationController 就是这样!

chad-m的回答和我的不同之处在于:

继承自UINavigationController,所以你不会污染你的rootViewController。 使用self.viewControllers.first而不是homeViewController,这样你就不会为你的100个UINavigationControllers在一个StoryBoard中做100次。

只在第一页隐藏导航栏也可以通过故事板实现。在故事板上,进入导航控制器场景->导航栏。从属性检查器中选择“Hidden”属性。这将从第一个视图控制器开始隐藏导航条,直到所需的视图控制器可见为止。

导航栏可以在ViewController的ViewWillAppear回调中设置为可见。

-(void)viewWillAppear:(BOOL)animated {

    [self.navigationController setNavigationBarHidden:YES animated:animated];
    [super viewWillAppear:animated];                                                  
}

我不得不对其他答案做出的一个轻微调整是,只有在viewWillDisappear中取消隐藏条,如果它消失的原因是由于导航项被推到它上面。这是因为视图可能因其他原因而消失。

所以我只在这个视图不再是最顶层视图时才取消隐藏条:

- (void) viewWillDisappear:(BOOL)animated
{
    if (self.navigationController.topViewController != self)
    {
        [self.navigationController setNavigationBarHidden:NO animated:animated];
    }

    [super viewWillDisappear:animated];
}