HBDNavigationBar

A custom UINavigationBar for smooth switching between various states, including bar style, bar tint color, background image, background alpha, bar hidden, title text attributes, tint color, shadow hidden...

1653
215
Objective-C

HBDNavigationBar

English Document

UINavigationBar 是每一个 iOS 工程师都会遇到的坎,它令人抓狂的地方在于是否能随着页面切换而平滑地过渡到目的状态。想要把这件事情做好,不需要高深的算法,不需要深刻的底层原理,只需要一颗执着的内心。

介绍

我们来看微信是如何平滑切换导航栏的状态的

weixin

页面和 收藏 页面的导航栏具有不同的 barStyle 以及背景色

当从 收藏 右滑返回 页面时,NavigationBar 的背景被分成黑白两段,并且 bar 上的元素平滑切换,就和只有一个背景色时一样。

仔细观察, 页面和 收藏 页面的导航栏背景颜色不一样,但是都有毛玻璃效果

收藏 页面往上滑动到一定程度时,导航栏会出现 shadowImage,此时如果右滑返回,导航栏依然在 收藏 页保留 shadowImage,然而 页面却没有这根线

不得不说,细节处理得真好

下面,我们来看一个反例,这是掘金 app 收藏页面的效果,当右滑返回上一个页面时,导航栏那反应实在是突兀,尖锐。

特别说明,本人举掘金这个例子,纯粹是因为掘金是本人常用 app 之一

juejin

导航栏的平滑过渡,可以划分为以下情况

阴影显示与隐藏

以下展示了平滑切换 shadowImage 的隐与现

shadow

导航栏有与无

以下展示导航栏有与无之间的平滑切换,和调用 setNavigationBarHidden:animated: 的效果不一样哦

hidden

导航栏背景透明度随 UIScrollView 滚动变化

这种效果是不是比掘金好多了

gradient

控制器拥有不同的导航栏背景

看下面效果,导航栏背景的表现是不是和微信一样

background

Usage

上面这些效果是三个类共同协作的结果:

HBDNavigationBar 继承 UINavigationBar

HBDNavigationController 继承 UINavigationController, 内部使用了 HBDNavigationBar

UIViewController(HBD) 是个分类,里面有一些可配置属性

@property (nonatomic, assign) UIBarStyle hbd_barStyle;   // 导航栏样式
@property (nonatomic, strong) UIColor *hbd_barTintColor; // 导航栏背景颜色
@property (nonatomic, strong) UIImage *hbd_barImage;     // 导航栏背景图片
@property (nonatomic, strong) UIColor *hbd_tintColor;    // 导航栏按钮颜色
@property (nonatomic, strong) NSDictionary *hbd_titleTextAttributes; // 导航栏标题属性
@property (nonatomic, assign) float hbd_barAlpha;        // 导航栏背景透明度
@property (nonatomic, assign) BOOL hbd_barHidden;        // 是否隐藏导航栏
@property (nonatomic, assign) BOOL hbd_barShadowHidden;  // 是否隐藏导航栏下面的阴影
@property (nonatomic, assign) BOOL hbd_backInteractive;  // 当前页面是否响应右滑返回,以及通过 `UINavigationBar` 的返回按钮返回,默认是 YES
@property (nonatomic, assign) BOOL hbd_swipeBackEnabled; // 当前页面是否可以右滑返回,默认是 YES

实际使用起来很简单

和使用普通的 UINavigationBar 一样,定义全局样式:

// AppDelegate.m
- (BOOL)application:(UIApplication *)application didFinishLaunchingWithOptions:(NSDictionary *)launchOptions {

    [[UINavigationBar appearance] setTintColor:UIColor.blackColor];
    // ...

    return YES;
}

使用 HBDNavigationController 代替 UINavigationController

DemoViewController *vc = [[DemoViewController alloc] init];
self.window.rootViewController = [[HBDNavigationController alloc] initWithRootViewController:vc];

如果某个控制器的导航栏样式和全局样式有差异,可以使用 UIViewController(HBD) 中的属性,在 viewDidLoad 这个生命周期函数里进行微调。这是声明式 API,只需要设置有差异的样式即可,也不需要清理。

@implementation DemoViewController
- (void)viewDidLoad {
    [super viewDidLoad];
    // 隐藏导航栏,就这样,不需要调用 setNavigationBarHidden:animated:
    // 也不需要担心其它页面会受到影响
    self.hbd_barHidden = YES;
}
@end

如果你使用 storyboard, 除了设置 HBDNavigationController, 也别忘了设置 HBDNavigationBar

storyboard

注意事项以及限制

hbd_barHidden

hbd_barHidden 并不真正隐藏导航栏,只是把它变透明了,当然事件是可以穿透的,也正因为并不真正隐藏导航栏,才可以在导航栏有无之间平滑而优雅地切换

Background algorithm

一旦通过 hbd_barImage 设置背景图片,hbd_barTintColor 就会失效

背景的计算规则如下:

  1. hbd_barImage 是否有值,如果有,将其设置为背景,否则下一步
  2. hbd_barTintColor 是否有值,如果有,将其设置为背景,否则下一步
  3. [[UINavigationBar appearance] backgroundImageForBarMetrics:UIBarMetricsDefault] 是否有返回值,如果有,将其设置为背景,否则下一步
  4. [UINavigationBar appearance].barTintColor 是否有值,如果有,将其设置为背景,否则下一步
  5. 根据 barStyle 计算出默认的背景颜色,并将其设置为背景

如果使用图片来设置背景,并且希望带有透明度,使用带有透明度的图片即可。

如果需要毛玻璃效果,那么设置给 hbd_barTintColor 的值应该带有透明度,具体数值根据色值的不同而不同。不要通过 hbd_barAlpha 来调整毛玻璃效果,它是用来动态控制导航栏背景的透与暗的,就像掘金收藏页面那个效果一样。

图片是没有毛玻璃效果的

Aways translucent

本库重写了 UINavigationBar 的 translucent 属性,使得它的值总是 YES。

本库根据导航栏的背景是否含有透明度,自动调整 UIViewController#edgesForExtendedLayout 这个属性。

如果导航栏一开始是不透明的,由于后续操作而变透明,需要设置 UIViewController#extendedLayoutIncludesOpaqueBars 的值为 YES

- (void)viewDidLoad {
    [super viewDidLoad];
    // 一开始导航栏为不透明
    self.hbd_barTintColor = UIColor.whiteColor;
    self.extendedLayoutIncludesOpaqueBars = YES;
}

- (void)handleScroll {
    // 由于用户操作而变透明
    self.hbd_barAlpha = 0.5;
    [self hbd_setNeedsUpdateNavigationBar];
}

基本原则就是如果我们设置的背景是含有透明度的,那么页面就应该位于 NavigationBar 底下(under),否则位于 NavigationBar 下面(below).

如果我们的 NavigationBar 一开始是不透明的,但有可能因为用户操作而变透明,那么设置 extendedLayoutIncludesOpaqueBars 的值为 YES

拦截返回事件

有时,我们需要在用户点击返回按钮或者侧滑返回时提醒用户,此时,可以重写以下方法,返回 NO


- (BOOL)hbd_backInteractive {
    // show alert
    return NO;
}

隐藏状态栏

如果你需要隐藏状态栏,请配合 HBDStatusBar 一起使用

全屏返回

创建一个继承于 HBDNavigationController 的子类,具体参考 FSPNavigationController

// FSPNavigationController.m
@implementation FSPNavigationController

- (void)viewDidLoad {
    [super viewDidLoad];
    
    // 获取系统自带滑动手势的target对象
    id target = self.interactivePopGestureRecognizer.delegate;
    // 创建全屏滑动手势,调用系统自带滑动手势的 target 的 action 方法
    UIPanGestureRecognizer *pan = [[UIPanGestureRecognizer alloc] initWithTarget:target action:@selector(handleNavigationTransition:)];
    // 设置手势代理,拦截手势触发
    pan.delegate = self.interactivePopGestureRecognizer.delegate;
    // 给导航控制器的view添加全屏滑动手势
    [self.view addGestureRecognizer:pan];
    // 禁止使用系统自带的滑动手势
    self.interactivePopGestureRecognizer.enabled = NO;
}

@end

感谢

在完善导航栏相关功能时,查看了 GitHub 上十多个相关项目,其中给我帮助最大的是 YPNavigationBarTransition,它为我解决不同背景之间如何平滑切换提供了非常有价值的参考。

Requirements

iOS 9+

Installation

HBDNavigationBar is available through CocoaPods. To install
it, simply add the following line to your Podfile:

pod 'HBDNavigationBar', '~> 1.9.8'

License

HBDNavigationBar is available under the MIT license. See the LICENSE file for more info.