像微信那样有追求的 UINavigationBar

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

介绍

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

ios-navbar-2021-10-15-01-32-16

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

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

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

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

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

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

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

ios-navbar-2021-10-15-01-32-47

笔者查看了十几个开源的解决方案,并没有特别满意的,决定自己写一个。念念不忘,必有回响。笔者折腾出了 HBDNavigationBaropen in new window 这个项目,下面来看看可以达成哪些效果:

阴影显示与隐藏

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

ios-navbar-2021-10-15-01-33-03

导航栏有与无

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

ios-navbar-2021-10-15-01-33-19

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

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

ios-navbar-2021-10-15-01-33-36

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

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

ios-navbar-2021-10-15-01-33-53

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

ios-navbar-2021-10-15-01-34-10

Installation

HBDNavigationBar is available through CocoaPodsopen in new window. To install it, simply add the following line to your Podfile:

pod 'HBDNavigationBar', '~> 1.7.7'

注意事项以及限制

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;
}

隐藏状态栏

如果你需要隐藏状态栏,请配合 HBDStatusBaropen in new window 一起使用

全屏返回

创建一个继承于 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

源码open in new window

上次更新: 6/30/2022, 6:52:25 AM