iOS11 NavigationBar适配

part1、分析新导航的结构

首先看一下下面代码:

        let leftButton: UIButton = UIButton(type: .custom)
        leftButton.setTitle("左一", for: .normal)
        leftButton.frame = CGRect(x: 0, y: 0, width: 60, height: 30)
        leftButton.backgroundColor = .green
        let leftButton_1: UIButton = UIButton(type: .custom)
        leftButton_1.setTitle("左二", for: .normal)
        leftButton_1.frame = CGRect(x: 0, y: 0, width: 60, height: 30)
        leftButton_1.backgroundColor = .green
        
        let leftBarItem:UIBarButtonItem = UIBarButtonItem(customView: leftButton)
        let leftBarItem_1:UIBarButtonItem = UIBarButtonItem(customView: leftButton_1)
        self.navigationItem.leftBarButtonItems = [leftBarItem, leftBarItem_1]

        let rightButton: UIButton = UIButton(type: .custom)
        rightButton.setTitle("右一", for: .normal)
        rightButton.frame = CGRect(x: 0, y: 0, width: 30, height: 30)
        rightButton.backgroundColor = .green
        let rightButton_1: UIButton = UIButton(type: .custom)
        rightButton_1.setTitle("右二", for: .normal)
        rightButton_1.frame = CGRect(x: 0, y: 0, width: 50, height: 30)
        rightButton_1.backgroundColor = .green
        
        let rightBarItem:UIBarButtonItem = UIBarButtonItem(customView: rightButton)
        let rightBarItem_1:UIBarButtonItem = UIBarButtonItem(customView: rightButton_1)
        self.navigationItem.rightBarButtonItems = [rightBarItem, rightBarItem_1]
        
        let titleLabel:UILabel = UILabel(frame: CGRect(x: 0, y: 0, width: 300, height: 30))
        titleLabel.textAlignment = .center
        titleLabel.text = "titleView栏"
        self.navigationItem.titleView = titleLabel

左边两个宽度均为60的item,右边一个30一个50的item,中间一个很宽(300)的titleVIew,为什么这么设置呢?后面再说。。。
效果如下:



看起来怪怪的有没有、、、(左右item距离屏幕边距,各边items之间的边距,titleView的位置偏移)
这是为什么呢?我们看下导航的层级结构:


iOS11导航层级结构

哦呦~还真是大变样子咯!

_UIBarBackground 显然就是背景样式咯,包含一个UIImageView一个UIVisualEffectView,那就可以理解磨砂透明效果怎么来的啦。不过,我们今天先不关注这个东西。
我们看看_UINavigationBarContentView:
包含两个_UIButtonBarStackView,一个_UITAMICAdaptorView
_UIButtonBarStackView很显然就是分别装左右两边的BarItems啦,
_UITAMICAdaptorView显然就是装titleView啦。

可是为啥子会是那种显示效果呢?我们随便打印下他们的frame看看:

Printing description of $15:
<_UIButtonBarStackView: 0x7fa1b1529cc0; frame = (20 0; 128 44); layer = <CALayer: 0x600000224160>>
Printing description of $16:
<_UIButtonBarStackView: 0x7fa1b152c1e0; frame = (299 0; 95 44); layer = <CALayer: 0x600000225f40>>
Printing description of $17:
<_UITAMICAdaptorView: 0x7fa1b152d770; frame = (148 7; 151 30); autoresizesSubviews = NO; layer = <CALayer: 0x600000223480>>

那就是左边的_UIButtonBarStackView是从20起,右边_UIButtonBarStackView距离边界20,中间的_UITAMICAdaptorView填充左右两个_UIButtonBarStackView之间的部分,没有间距。

具体分析

先看左边_UIButtonBarStackView

左边_UIButtonBarStackView

很显然是两个item加上一个_UITAMICAdaptorView了,上面代码设置我们知道理论上两个item的width都是60了,打印下看看:

Printing description of $18:
<_UITAMICAdaptorView: 0x7fa1b152e8f0; frame = (0 5; 60 34); autoresizesSubviews = NO; layer = <CALayer: 0x6000002254c0>>
Printing description of $19:
<UIView: 0x7fa1b15233f0; frame = (60 22; 8 0); layer = <CALayer: 0x600000225f00>>
Printing description of $20:
<_UITAMICAdaptorView: 0x7fa1b15235d0; frame = (68 5; 60 34); autoresizesSubviews = NO; layer = <CALayer: 0x6000002251a0>>

恩、、、两个60width的item没毛病,8的space也没毛病,60+60+8=128 。老铁没毛病!美滋滋~

再看右边_UIButtonBarStackView

右边_UIButtonBarStackView

层级结构和左边一样的,上面代码设置我们知道理论上一个item是30width,一个是50width,再打印一下:

Printing description of $21:
<_UITAMICAdaptorView: 0x7fa1b1431440; frame = (0 5; 50 34); autoresizesSubviews = NO; layer = <CALayer: 0x608000225aa0>>
Printing description of $22:
<UIView: 0x7fa1b1432660; frame = (50 22; 8 0); layer = <CALayer: 0x608000227620>>
Printing description of $23:
<_UITAMICAdaptorView: 0x7fa1b1432840; frame = (58 5; 37 34); autoresizesSubviews = NO; layer = <CALayer: 0x608000227280>>

看到没,老铁,问题来了! 这有个家伙宽度变34了。不用怀疑代码问题,这是设置的30宽度不足以显示“右一”两个字,StackView给layout了。(这里大家需要注意一下)

两边分析完了,我们来看看中间的titleView:

也许大家在刚开始的打印中就发现问题了,titleView的width才151,我们明明设置的是300啊。好吧,看来是在contentView中先把两边的StackView安置好了之后才将剩下空间分给titleView的。所以,如果两边的items不能做到对称的话,那么titleView就不居中了。。。

我们可以通过打印下_UINavigationBarContentView的layoutConstraints来简单看下各控件的关系。

po view.superview.constraints
<__NSArrayI 0x7f8e51728790>(
<NSLayoutConstraint:0x60c00008a190 H:|-(0)-[UILayoutGuide:0x60c0001b67a0'BackButtonGuide(0x7f8e515088e0)']   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>,
<NSLayoutConstraint:0x60c00008a2d0 UILayoutGuide:0x60c0001b67a0'BackButtonGuide(0x7f8e515088e0)'.trailing >= _UINavigationBarContentView:0x7f8e51508210.leading   (active)>,
<NSLayoutConstraint:0x60c00008a230 H:[UILayoutGuide:0x60c0001b67a0'BackButtonGuide(0x7f8e515088e0)']-(0)-[UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)']   (active)>,
<NSLayoutConstraint:0x60c00008a140 H:[UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)']-(0)-[UILayoutGuide:0x60c0001b6960'TitleView(0x7f8e515088e0)']   (active)>,
<NSLayoutConstraint:0x60c00008a0f0 H:[UILayoutGuide:0x60c0001b6960'TitleView(0x7f8e515088e0)']-(0)-[UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)']   (active)>,
<NSLayoutConstraint:0x60c00008a0a0 UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)'.trailing == _UINavigationBarContentView:0x7f8e51508210.trailing priority:999   (active)>,
<NSLayoutConstraint:0x60c00008bf90 UILayoutGuide:0x60c0001b67a0'BackButtonGuide(0x7f8e515088e0)'.width == 0 priority:50   (active)>,
<NSLayoutConstraint:0x60c00008c030 UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)'.width == 0 priority:50   (active)>,
<NSLayoutConstraint:0x60c00008bfe0 UILayoutGuide:0x60c0001b6960'TitleView(0x7f8e515088e0)'.width == UILayoutGuide:0x60c0001b6b20'UIViewLayoutMarginsGuide'.width priority:50   (active)>,
<NSLayoutConstraint:0x60c00008c2b0 UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)'.width == 0 priority:50   (active)>,
<NSLayoutConstraint:0x600000089740 V:|-(0)-[UILayoutGuide:0x60c0001b67a0'BackButtonGuide(0x7f8e515088e0)']   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>,
<NSLayoutConstraint:0x600000089650 V:|-(0)-[UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)']   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>,
<NSLayoutConstraint:0x60000008a6e0 V:|-(0)-[UILayoutGuide:0x60c0001b6960'TitleView(0x7f8e515088e0)']   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>,
<NSLayoutConstraint:0x60000008a910 V:|-(0)-[UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)']   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>,
<NSLayoutConstraint:0x60000008afa0 UILayoutGuide:0x60c0001b67a0'BackButtonGuide(0x7f8e515088e0)'.height == 44   (active)>,
<NSLayoutConstraint:0x60000008b1d0 UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)'.height == 44   (active)>,
<NSLayoutConstraint:0x60000008b3b0 UILayoutGuide:0x60c0001b6960'TitleView(0x7f8e515088e0)'.height == 44   (active)>,
<NSLayoutConstraint:0x60000008b220 UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)'.height == 44   (active)>,
<NSLayoutConstraint:0x60c00008c6c0 _UIButtonBarStackView:0x7f8e51502e80.leading == UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)'.leading   (active)>,
<NSLayoutConstraint:0x60c00008c710 _UIButtonBarStackView:0x7f8e51502e80.trailing == UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)'.trailing   (active)>,
<NSLayoutConstraint:0x60c00008c760 _UIButtonBarStackView:0x7f8e51502e80.top == UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)'.top   (active)>,
<NSLayoutConstraint:0x60c00008c7b0 _UIButtonBarStackView:0x7f8e51502e80.bottom == UILayoutGuide:0x60c0001b6880'LeadingBarGuide(0x7f8e515088e0)'.bottom   (active)>,
<NSLayoutConstraint:0x60c00008ce90 _UIButtonBarStackView:0x7f8e5150bbe0.leading == UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)'.leading   (active)>,
<NSLayoutConstraint:0x60c00008cee0 _UIButtonBarStackView:0x7f8e5150bbe0.trailing == UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)'.trailing   (active)>,
<NSLayoutConstraint:0x60c00008cf30 _UIButtonBarStackView:0x7f8e5150bbe0.top == UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)'.top   (active)>,
<NSLayoutConstraint:0x60c00008cf80 _UIButtonBarStackView:0x7f8e5150bbe0.bottom == UILayoutGuide:0x60c0001b6a40'TrailingBarGuide(0x7f8e515088e0)'.bottom   (active)>,
<NSLayoutConstraint:0x60c00008c210 'UIView-bottomMargin-guide-constraint' V:[UILayoutGuide:0x60c0001b6b20'UIViewLayoutMarginsGuide']-(0)-|   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>,
<NSLayoutConstraint:0x6080000922a0 'UIView-Encapsulated-Layout-Height' _UINavigationBarContentView:0x7f8e51508210.height == 44   (active)>,
<NSAutoresizingMaskLayoutConstraint:0x608000092340 h=--& v=--& 'UIView-Encapsulated-Layout-Left' _UINavigationBarContentView:0x7f8e51508210.minX == 0   (active, names: '|':UINavigationBar:0x7f8e51505bc0 )>,
<NSAutoresizingMaskLayoutConstraint:0x608000092480 h=--& v=--& 'UIView-Encapsulated-Layout-Top' _UINavigationBarContentView:0x7f8e51508210.minY == 0   (active, names: '|':UINavigationBar:0x7f8e51505bc0 )>,
<NSLayoutConstraint:0x608000092250 'UIView-Encapsulated-Layout-Width' _UINavigationBarContentView:0x7f8e51508210.width == 414   (active)>,
<NSLayoutConstraint:0x60c00008c1c0 'UIView-leftMargin-guide-constraint' H:|-(0)-[UILayoutGuide:0x60c0001b6b20'UIViewLayoutMarginsGuide'](LTR)   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>,
<NSLayoutConstraint:0x60c00008c260 'UIView-rightMargin-guide-constraint' H:[UILayoutGuide:0x60c0001b6b20'UIViewLayoutMarginsGuide']-(0)-|(LTR)   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>,
<NSLayoutConstraint:0x60c00008c080 'UIView-topMargin-guide-constraint' V:|-(0)-[UILayoutGuide:0x60c0001b6b20'UIViewLayoutMarginsGuide']   (active, names: '|':_UINavigationBarContentView:0x7f8e51508210 )>
)

简单分析可大概得到下面的关系:
LeadingBarGuide-(0)-TitleView-(0)-TrailingBarGuide

part2、解决方案

我们这里主要解决左右_UIButtonBarStackView距离屏幕边距,和其中各个items间距的问题。至于titleView的问题,相信看过之后大家也有相应的思路了。
主要有两种方式:
1.修改LeadingBarGuide,TrailingBarGuide的layout,遍历_UIButtonBarStackView.subViews得到中间的填充View修改width。
2.直接修改左右_UIButtonBarStackView的layout,遍历_UIButtonBarStackView.subViews得到中间的填充View修改width。
针对两种方式也有不同的实现:
1.利用Runtime替换UIView的layout方法。
2.利用Runtime替换UINavigationItem的setLeftBarButtonItem,setLeftBarButtonItems,setRightBarButtonItem,setRightBarButtonItems方法。
这两种方法的具体实现代码大家可以去下载demo,默认使用方法1的实现。
大家使用时候只需要将UIViewAdaptor或者NavigationItemAdaptor文件夹下的文件导入工程,并且修改对应的space和margin参数即可。

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

推荐阅读更多精彩内容