一、UINavigationBar的结构

导航栏几乎是每个页面都会碰到的问题,一般两种处理方式:1.隐藏掉不显示 2.自定义

1. 添加导航栏

TestViewController * mainVC = [[TestViewController alloc] init];
UINavigationController * nav = [[UINavigationController alloc] initWithRootViewController:mainVC];
self.window.rootViewController = nav;

2. 隐藏导航栏

在的TestViewController.m文件中,使用以下代码:

- (void)viewDidLoad {
[super viewDidLoad]; [self.navigationController setNavigationBarHidden:YES];
//self.navigationController.navigationBar.hidden = YES;
}

说明:通过属性直接设置之所以能成功,是因为虽然navigationBar是readonly,但是hidden是默认的(readwrite)。建议使用第一个,通过发送消息来设置。

3. 修改导航栏背景色

由于系统自带的导航栏已经不能满足用户的审美需求,因此开发中导航栏或多或少都被自定义了;即便只是修改了背景色或是字体颜色。

  • 方式一:通过代理来设置

    [[UINavigationBar appearance] setBarTintColor:[UIColor redColor]];

    效果图

说明:为什么能修改所有导航栏的背景色?

  1. 查看UINavigationBar发送的消息appearance,是一个名为UIAppearance协议。根据官网解释:UIAppearance是一个外观协议(装饰模式)。要想改变外观需要实现此协议,因此UINavigationBar内部肯定实现了此协议。
  2. 查看头文件UIAppearance协议里面有4个方法,但是根据官网文档,iOS9以后有两个方法被废弃。因此只有下面两个可用,都返回的是类的对象,因此可以给此类发送修改外观的消息.比如:setBarTintColor。其实这里拿到对象后可以任意修改。
+ (instancetype)appearance;
+ (instancetype)appearanceForTraitCollection:(UITraitCollection *)trait
  • 方式二、直接通过属性设置,或发送消息来设置

在的TestViewController.m文件中,使用以下代码:

- (void)viewDidLoad {
[super viewDidLoad];
// 无效果
self.navigationController.navigationBar.backgroundColor = [UIColor redColor];
// 可行
[self.navigationController.navigationBar setBackgroundImage:[UIImage imageNamed:@"nav.png"] forBarMetrics:UIBarMetricsDefault];
}

说明:

  1. 下面代码为什么无效果,其实看一下结构图就明白了。
self.navigationController.navigationBar.backgroundColor = [UIColor redColor];

UINavgationBar结构图

没错:这样修改的是navigationBar的背景色,而navigationBar里面添加了UIView和UIImageView,UILabel等控件,覆盖了navigationBar。

  1. 那么如何来修改?
[self.navigationController.navigationBar setBackgroundImage:[UIImage imageNamed:@"yao_1"] forBarMetrics:UIBarMetricsDefault];

通过添加背景图,使用默认的模式。
注意:这里会出现状态栏背景色也变化的问题,其实这是由于在navigationBar添加的控件自动延伸到边界造成的。如果只是修改导航栏的背景色,完全没必要再给navigationBar添加控件了,给navigationBar堆太多的无用控件,不能忍。

// 设置透明
[self.navigationController.navigationBar setBackgroundImage:[UIImage imageNamed:@"yao_1"] forBarMetrics:UIBarMetricsCompact];
// 修改navigationBar的背景色
self.navigationController.navigationBar.backgroundColor = [UIColor redColor];

这样修改navigationBar背景色,才有种很舒畅的感觉,有木有。因为我就只是要修改navigationBar的背景色而已么。效果图:

UINavgationBar效果图

为什么要这样

[[UINavigationBar appearance] setBarTintColor:[UIColor redColor]];

或者为什么要这样

[self.navigationController.navigationBar setBackgroundImage:[UIImage imageNamed:@"yao_1"] forBarMetrics:UIBarMetricsDefault];

UINavigationBar和UINavigationController、UIViewController关系

1.导航栏类名:UINavigationBar,是导航控制器(UINavgationController)下得一个属性,一般给一个视图控制器(TestViewController)添加导航栏,也就是把它添加为导航控制器(UINavgationController)的根视图控制器。

2.在此视图控制器中为什么可以直接操作导航控制器,因为TestViewController类是自定义的继承自UIViewController类的。而UIViewController类中有一个类别,声明了UINavgationController类的一个属性。在第一步的时候,把这个属性设置为上面导航控制器的对象。

UIController分类

3.UINavigationBar对照图如下:

UINavigationBar对照图

UINavgationBar结构图说明:

  • 从图中的对照关系可以看到,UINavigationBar包含一个:UINavigationBarBackground控件(已经延伸到边界,覆盖掉了导航栏),另一个UINavigationBarBackIndicatorView(其实就是那个蓝色的返回按钮,当push进来一个ViewController才会显示)
  • UINavigationBarBackground控件里面包含:UIBackdropView(UIBackdropEffectView标识阴影视图),UIImageView

总结:

可以看到,UINavigationBarBackIndicatorView(返回按钮只占了左边一部分区域,并且在push新的视图控制器时才显示。因此,显示在最上面的是UIImageView)。所以,这才是通过代码

self.navigationController.navigationBar.backgroundColor = [UIColor redColor];

*修改背景色,老是被遮住的根本原因。解决方法,因此掉UIImageView控件,或者干脆给UIImageView设置一张你想要的背景色的图片。*

二、UINavigationBar的自定义

1. 自定义

一般自定义分三种情况:左侧、右侧和中间视图

  • 自定义左侧按钮
- (void)viewDidLoad {
[super viewDidLoad];
// 自定义导航栏左侧按钮
UIButton * leftBtn = [UIButton buttonWithType:UIButtonTypeRoundedRect];
leftBtn.frame = CGRectMake(0, 7, 83, 30);
leftBtn.backgroundColor = [UIColor orangeColor];
[leftBtn addTarget:self action:@selector(onTap) forControlEvents:UIControlEventTouchUpInside];
UIBarButtonItem * leftItem = [[UIBarButtonItem alloc] initWithCustomView:leftBtn];
self.navigationItem.leftBarButtonItem = leftItem;
} // 点击事件处理
- (void)onTap {
NSLog(@"点击了导航栏左侧按钮");
}
  • 自定义右侧按钮

    和自定义左侧按钮方法类似,差别如下:

    self.navigationItem.rightBarButtonItem = rightItem;
  • 自定义中间视图

    UIView * centerView = [[UIView alloc] initWithFrame:CGRectMake(0, 7, 110, 30)];
    centerView.backgroundColor = [UIColor greenColor];
    self.navigationItem.titleView = centerView;

    2. 结构分析

从自定义的代码看到,自定义都是在修改navigationItem里面的东西。查看头文件可知,TestViewController之所以可以操作navigationItem,和上面UINavigationController实现方式一样。

UINavigationBar类中有一个items集合,存储navigationItem和其他相关内容。UINavigationItem类里面存储leftBarButtonItems、rightBarButtonItems等集合。同时也会有leftBarButtonItem、rightBarButtonItem控件,根据集合是否为空来决定是否创建和显示这些控件。

因此,当不设置内容时,默认集合为空,所有控件也就不显示。

3. 手写代码技巧

对于导航栏的自定义,大部分app都是左侧为"返回",右侧为"地图",“扫码”等按钮,中间除显示标题外,可能会自定义个分段按钮,选择菜单等。而对于这些自定义的按钮和视图,尺寸都是根据图片尺寸和导航栏的宽高凭感觉设置的(总之,我以前是这样>.<)。
今天在测试iOS7的布局使用了SB,发现控件拖上去它会自适应系统给定的尺寸。然后我绑定控件获取了这些控件的尺寸。测试了iPhone4s和iPhone6,获取到的左侧和右侧按钮尺寸:frame = {0,7,83,30},中间的视图尺寸:frame = {0,7,110,30}。
这样,以后导航栏按钮没有什么特殊要求,就可以都按这个尺寸设置了。

原文链接:http://www.jianshu.com/p/b7818eba288c

iOS 自定义导航栏笔记的更多相关文章

  1. iOS 自定义导航栏 和状态栏

    一.更改状态栏颜色 (StatusBar) 就是比如导航栏是红色的状态栏是绿色的. 要实现这样的效果其实很简单,就是添加一个背景view. 简单的实现过程如下: 1 // 设置导航颜色 可用 2 [s ...

  2. ios 自定义导航栏,开启侧滑返回手势

    自定义一个常用ListViewController .h文件 #import <UIKit/UIKit.h> @interface ListViewController : UIViewC ...

  3. IOS 自定义导航栏背景

    //- (void)setBackgroundImage:(UIImage *)backgroundImage forBarMetrics:(UIBarMetrics)barMetrics NS_AV ...

  4. iOS 自定义导航栏

    参考链接:https://blog.csdn.net/huanglinxiao/article/details/100537988 demo下载地址:https://github.com/huangx ...

  5. iOS个人中心渐变动画、微信对话框、标签选择器、自定义导航栏、短信验证输入框等源码

    iOS精选源码 简单的个人中心页面-自定义导航栏并予以渐变动画 程序员取悦女票的正确姿势---Tip1(iOS美容篇) iOS 前台重启应用和清除角标的问题 微信原生提醒对话框3.0 JHLikeBu ...

  6. iOS:自定义导航栏,随着tableView滚动显示和隐藏

    自定义导航栏,随着tableView滚动显示和隐藏 一.介绍 自定义导航栏是APP中很常用的一个功能,通过自定义可以灵活的实现动画隐藏和显示效果.虽然处理系统的导航栏也可以实现,但是这个是有弊端的,因 ...

  7. ios7以上自定义导航栏标题的字体大小及颜色的方法

    自定义导航栏的字体和颜色,只需要自定义一个lable,然后将lable添加到导航栏的titleview中就可以了 代码如下 UILabel *label = [[UILabel alloc] init ...

  8. iOS 11 导航栏 item 偏移问题 和 Swift 下 UIButton 设置 title、image 显示问题

    html,body,div,span,applet,object,iframe,h1,h2,h3,h4,h5,h6,p,blockquote,pre,a,abbr,acronym,address,bi ...

  9. 解决 ios7.0 以后自定义导航栏左边按钮靠右的问题

    解决 ios7.0 以后自定义导航栏左边按钮靠右的问题 www.111cn.net 编辑:edit02_lz 来源:转载 最近开发了一个ios的app,在ios7.0+出现自定义导航栏左边按钮出现靠右 ...

随机推荐

  1. PQ分区魔术师v9.0 中文版

    软件名称: pqmagic 硬盘分区大师9.0中文绿色版 软件大小:5.80MB 软件语言:简体中文 软件类别:磁盘工具 软件授权:免费软件 更新时间:2013-10-082013-10-08 09: ...

  2. 【linux-shell】rsync

    https://en.wikipedia.org/wiki/Rsync https://en.wikipedia.org/wiki/Andrew_Tridgell 安德鲁.锤子,发布了rsync的第一 ...

  3. kubernetes port nodePort targetPort 理解

    port The port that the service is exposed on the service's cluster ip (virsual ip). Port is the serv ...

  4. 第一篇,jos

    关于jos环,使用递推公式简化问题和代码,关键在于找到正确的递推公式,可使用一个例子来寻找. (数学能力较差,只好打个表找规律了) 为方便取余运算,将编号1---n的下标表示为0--(n-1)     ...

  5. javascript静态页面传值的三种方法分享

    一:JavaScript静态页面值传递之URL篇能过URL进行传值.把要传递的信息接在URL上.Post.htm 复制代码 代码如下: <input type="text" ...

  6. 为什么 把单一元素的数组放在一个struct的尾端问题

    问题摘自<深度探究c++对象模型>: struct mumble { /* stuff */ char pc[ 1 ];};[sizeof(mumble)是一个字节 .pc则代表的是指向这 ...

  7. Lucene全文检索技术

    Lucene全文检索技术 今日大纲 ●    搜索的概念.搜索引擎原理.倒排索引 ●    全文索引的概念 ●    使用Lucene对索引进行CRUD操作 ●    Lucene常用API详解 ●  ...

  8. centsOS下安装vsftp的配置

    1. 添加用户组 # groupadd www 2. 修改配置 # vi /etc/vsftpd/vsftpd.conf 查找: #chroot_list_enable=YES #chroot_lis ...

  9. bootstrip 折叠菜单

    body { font-family: "宋体"; font-size: 12px } .menu { height: 30px; background: #e6e6e6; mar ...

  10. NDK常见错误

    1.错误1: android mk文件没有定义 $ ndk-buildAndroid NDK: Your APP_BUILD_SCRIPT points to an unknown file: /cy ...