像微信那样有追求的 UINavigationBar

3,630 阅读4分钟
原文链接: github.com

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

介绍

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

weixin

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

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

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

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

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

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

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

juejin

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

阴影隐与现

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

shadow

导航栏有与无

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

hidden

导航栏背景透与暗

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

gradient

导航栏背景不同

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

background

Usage

上面这些效果是三个类共同协作的结果,总共约 400 行代码

HBDNavigationBar 继承 UINavigationBar

HBDNavigationController 继承 UINavigationController, 内部使用了 HBDNavigationBar

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

@property (nonatomic, assign) UIBarStyle hbd_barStyle;   // 导航栏样式
@property (nonatomic, strong) UIColor *hbd_barTintColor; // 导航栏背景颜色
@property (nonatomic, assign) float hbd_barAlpha;        // 导航栏背景透明度
@property (nonatomic, assign) BOOL hbd_barHidden;        // 是否隐藏导航栏
@property (nonatomic, assign, readonly) float hbd_barShadowAlpha; // 只读,通过其它属性计算出来
@property (nonatomic, assign) BOOL hbd_barShadowHidden;  // 是否隐藏导航栏下面的阴影
@property (nonatomic, assign) BOOL hbd_backInteractive;  // 当前页面是否响应右滑返回,默认是 YES

实际使用起来很简单

// HBDNavigationController 只有在创建 UINavigationController 时使用到
// HBDNavigationBar 只有在使用 storyboard 时才有机会登场
DemoViewController *vc = [[DemoViewController alloc] init];
self.window.rootViewController = [[HBDNavigationController alloc] initWithRootViewController:vc];

在 viewDidLoad 中通过分类配置想要的效果即可

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

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

storyboard

注意事项以及限制

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

只支持通过 hbd_barTintColor 设置背景颜色,不支持设置背景图片,也就是说,调用 setBackgroundImage:forBarMetrics: 是无效的

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

isTranslucent 的值总是 YES,你也不应该去改变它,这意味着,controller 的 view 总是位于导航栏底下

支持通过 [UINavigationBar appearance] 设置其余导航栏相关属性

感谢

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

Requirements

iOS 8+

Installation

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

pod 'HBDNavigationBar'

License

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