iOS开发-优雅的解决导航栏隐藏问题

在我们日常开发中,总有那么一两个界面需要去隐藏导航栏,这时如何去合理的处理呢?笔者这里提供了几个常用的方案和带来的问题,并在最后给个笔者认为较为优雅的方法。

方案一:使用setNavigationBarHidden:animated:方法直接处理

- (void)viewWillAppear:(BOOL)animated {
    [super viewWillAppear:animated];

    [self.navigationController setNavigationBarHidden:true animated:animated];
}

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

这个使我们解决隐藏导航栏首先会想到的方案,这种方式虽然很好的解决了,首页隐藏导航栏,push到新界面不隐藏的场景。但如下场景会有个过度动画,很难看:

    1. 首页需要隐藏,push的新界面也需要隐藏,这时就会有个隐藏--显示--隐藏的过度动画;
    1. 首页隐藏,然后在切换tabBar再回来,这时有一个导航栏向上消失的动画;

所以这种直接使用的方案,不完美,pass。

方案二:使用UINavigationControllerDelegate代理方法直接处理

@interface HomePageController () <UINavigationControllerDelegate>
@end

@implementation HomePageController 

#pragma mark - lifeCycle
- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 设置导航控制器的代理为self
    self.navigationController.delegate = self;
}

#pragma mark - < UINavigationControllerDelegate >

- (void)navigationController:(UINavigationController *)navigationController willShowViewController:(UIViewController *)viewController animated:(BOOL)animated {
    // 判断要显示的控制器是否是自己
    BOOL isShowHomePage = [viewController isKindOfClass:[self class]];
    
    [self.navigationController setNavigationBarHidden:isShowHomePage animated:YES];
}

- (void)dealloc {
    self.navigationController.delegate = nil;
}

通过当前self对象来管理导航栏的显示和隐藏,虽然能解决切换tabBar的动画问题,但是还是没有解决上面的问题1。

不完美,pass。

方案三:参考FDFullscreenPopGesture思路的实现方案,完美解决以上问题

主要的思路是使用runtimehook UIViewControllerviewWillAppear:方法和navigationControllerpushViewController:animated:setViewControllers:animated:方法实现。
相当于是在每个UIViewController控制器,在调用viewWillAppear:方法的时候,都去判断下是否需要隐藏导航栏setNavigationBarHidden:animated:方法,这样的好处是,不用再当前控制器的viewWillDisappear:中写显示方法,也就没有了过渡的动画问题,完美解决。

  • UIViewController添加一个设置隐藏导航栏的属性lsl_prefersNavigationBarHidden
// MARK: - 给UIViewController添加lsl_prefersNavigationBarHidden属性

@implementation UIViewController (HandlerNavigationBar)

- (BOOL)lsl_prefersNavigationBarHidden
{
    return [objc_getAssociatedObject(self, _cmd) boolValue];
}

- (void)setLsl_prefersNavigationBarHidden:(BOOL)hidden
{
    objc_setAssociatedObject(self, @selector(lsl_prefersNavigationBarHidden), @(hidden), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

@end
  • hook UIViewControllerviewWillAppear:方法,并在此方法中执行已经存好的代码块:
typedef void(^_LSLViewControllerWillAppearInjectBlock)(UIViewController *viewController, BOOL animated);

@interface UIViewController (HandlerNavigationBarPrivate)

@property(nonatomic, copy) _LSLViewControllerWillAppearInjectBlock lsl_willAppearInjectBlock;

@end

// MARK: - 替换UIViewController的viewWillAppear方法,在此方法中,执行设置导航栏隐藏和显示的代码块。
@implementation UIViewController (HandlerNavigationBarPrivate)

+ (void)load
{
    Method orginalMethod = class_getInstanceMethod(self, @selector(viewWillAppear:));
    Method swizzledMethod = class_getInstanceMethod(self, @selector(lsl_viewWillAppear:));
    method_exchangeImplementations(orginalMethod, swizzledMethod);
}

- (void)lsl_viewWillAppear:(BOOL)animated
{
    [self lsl_viewWillAppear:animated];

    if (self.lsl_willAppearInjectBlock) {
        self.lsl_willAppearInjectBlock(self, animated);
    }
}

- (_LSLViewControllerWillAppearInjectBlock)lsl_willAppearInjectBlock
{
    return objc_getAssociatedObject(self, _cmd);
}

- (void)setLsl_willAppearInjectBlock:(_LSLViewControllerWillAppearInjectBlock)block
{
    objc_setAssociatedObject(self, @selector(lsl_willAppearInjectBlock), block, OBJC_ASSOCIATION_COPY_NONATOMIC);
}

@end
  • hooknavigationControllerpushViewController:animated:setViewControllers:animated:方法,当控制器被压入栈中的时候,预存设置隐藏和显示导航栏的代码块到即将显示的控制器中,备控制器调用:
// MARK: - 替换UINavigationController的pushViewController:animated:方法,在此方法中去设置导航栏的隐藏和显示
@implementation UINavigationController (NavigationBar)

+ (void)load
{
    Method originMethod = class_getInstanceMethod(self, @selector(pushViewController:animated:));
    Method swizzedMethod = class_getInstanceMethod(self, @selector(lsl_pushViewController:animated:));
    method_exchangeImplementations(originMethod, swizzedMethod);

    Method originSetViewControllersMethod = class_getInstanceMethod(self, @selector(setViewControllers:animated:));
    Method swizzedSetViewControllersMethod = class_getInstanceMethod(self, @selector(lsl_setViewControllers:animated:));
    method_exchangeImplementations(originSetViewControllersMethod, swizzedSetViewControllersMethod);
}

- (void)lsl_pushViewController:(UIViewController *)viewController animated:(BOOL)animated
{
    // Handle perferred navigation bar appearance.
    [self lsl_setupViewControllerBasedNavigationBarAppearanceIfNeeded:viewController];

    // Forward to primary implementation.
    [self lsl_pushViewController:viewController animated:animated];
}

- (void)lsl_setViewControllers:(NSArray<UIViewController *> *)viewControllers animated:(BOOL)animated
{
    // Handle perferred navigation bar appearance.
    for (UIViewController *viewController in viewControllers) {
        [self lsl_setupViewControllerBasedNavigationBarAppearanceIfNeeded:viewController];
    }

    // Forward to primary implementation.
    [self lsl_setViewControllers:viewControllers animated:animated];
}

- (void)lsl_setupViewControllerBasedNavigationBarAppearanceIfNeeded:(UIViewController *)appearingViewController
{
    if (!self.lsl_viewControllerBasedNavigationBarAppearanceEnabled) {
        return;
    }

    // 即将被调用的代码块
    __weak typeof(self) weakSelf = self;
    _LSLViewControllerWillAppearInjectBlock block = ^(UIViewController *viewController, BOOL animated){
        __strong typeof(weakSelf) strongSelf = weakSelf;
        if (strongSelf) {
            [strongSelf setNavigationBarHidden:viewController.lsl_prefersNavigationBarHidden animated:animated];
        }
    };

    // 给即将显示的控制器,注入代码块
    appearingViewController.lsl_willAppearInjectBlock = block;

    // 因为不是所有的都是通过push的方式,把控制器压入stack中,也可能是"-setViewControllers:"的方式,所以需要对栈顶控制器做下判断并赋值。
    UIViewController *disappearingViewController = self.viewControllers.lastObject;
    if (disappearingViewController && !disappearingViewController.lsl_willAppearInjectBlock) {
        disappearingViewController.lsl_willAppearInjectBlock = block;
    }
}

- (BOOL)lsl_viewControllerBasedNavigationBarAppearanceEnabled
{
    NSNumber *number = objc_getAssociatedObject(self, _cmd);
    if (number) {
        return number.boolValue;
    }
    self.lsl_viewControllerBasedNavigationBarAppearanceEnabled = YES;
    return YES;
}

- (void)setLsl_viewControllerBasedNavigationBarAppearanceEnabled:(BOOL)enabled
{
    SEL key = @selector(lsl_viewControllerBasedNavigationBarAppearanceEnabled);
    objc_setAssociatedObject(self, key, @(enabled), OBJC_ASSOCIATION_RETAIN_NONATOMIC);
}

@end

方案三可以完美解决以上所遇见的问题,demo中有相应的案例和完整代码。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 159,117评论 4 362
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,328评论 1 293
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 108,839评论 0 243
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,007评论 0 206
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,384评论 3 287
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,629评论 1 219
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,880评论 2 313
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,593评论 0 198
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,313评论 1 243
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,575评论 2 246
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,066评论 1 260
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,392评论 2 253
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,052评论 3 236
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,082评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,844评论 0 195
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,662评论 2 274
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,575评论 2 270

推荐阅读更多精彩内容