本文討論的問題:
- 自定義導航欄,導航欄高度;
- items 與導航欄間的間距荚恶;
- 導航欄的層次結(jié)構(gòu)撩穿;
- 自定義導航欄的一大好處!
正文:
有個項目中自定義導航欄谒撼,當然是繼承自 UINavigationBar 食寡,但是顯示效果與期望值不符,上移了一段距離(20px)廓潜。如下圖:
首先想到的是調(diào)整frame的高度冻河,運行之后仍沒有效果。
于是想到查看一下視圖的層次結(jié)構(gòu)茉帅。如下圖:
如圖所示的層次結(jié)構(gòu)叨叙,navbar 下面有兩個直接子視圖,background 和 contentView堪澎。
而navBar的高度時正確的擂错,只是background和contentView上移了。而且這倆家伙的高度也與期望的不相符樱蛤。
(background)
(contentView)
通過打印的內(nèi)容可以看出钮呀,background 和 contentView 的高度都是44,比預期的64減小了20昨凡,正好是一個狀態(tài)欄的高度爽醋。
有個思路是將它們下移 20 。
根據(jù)視圖層次便脊,我們知道background是負責渲染背景相關的蚂四,contentView負責顯示我們設置的控件、title等內(nèi)容。
所以遂赠,為了使導航欄與狀態(tài)欄的風格一致久妆,
決定將 background 的高度從 44 變?yōu)?64,與導航欄一致跷睦;
將 contentView 下移 20 筷弦。
怎么才能拿到這兩個視圖呢?方法是遍歷導航欄navBar的子視圖 subViews抑诸。
那么在什么時機遍歷子視圖-修改frame合適呢烂琴?
還是在 自定義navBar 的 layoutSubviews 方法中最合適。
注意在 自定義navBar 中重寫 layoutSubViews 方法時蜕乡,要先 super.layoutSubviews() 一下监右,否則會導致自定義導航欄中沒有東西,因為系統(tǒng)沒給layout异希。
解決思路:
- 在自定義navBar中重寫 layoutSubviews 方法健盒;
- 調(diào)用父類 super.layoutSubviews() 讓系統(tǒng)布局完畢;
- 遍歷 navBar 的 subViews 找到 視圖background 和 視圖contentView称簿;
- 分別設置它們的frame扣癣;
- 當然有個前提,那就是要先判斷當前系統(tǒng)版本憨降,如果 versionCode > 11.0 才需要自己處理父虑。
- 還要注意一個問題,那就是 iPhone X 設備授药,需要為此單獨設置frame士嚎。
為了清晰明了,直接上代碼:
設置完畢悔叽,運行莱衩,效果如下:
有沒有發(fā)現(xiàn)上面這張圖的 Left 按鈕跟第一張圖的 Left 按鈕的區(qū)別?
是的娇澎,這就是要繼續(xù)討論的【2. items 與導航欄間的間距 】這個問題笨蚁。
我們都有過這樣的感覺,設置完 leftBarButtonItem 或者 rightBarButtonItem 的時候趟庄,總感覺它距離屏幕的邊距有點遠括细。
在iOS10之前,要使邊距近一點戚啥,是在它倆之間添加一個 "占位" 奋单,并設置占位item的寬度width = -15; 對是負的,如下:
【
UIBarButtonItem *leftFixed = [[UIBarButtonItem alloc] initWithBarButtonSystemItem:UIBarButtonSystemItemFixedSpace target:nil action:nil];
leftFixed.width = -15;
】
而在 iOS11 之后猫十,這樣設置不起作用了览濒。
根據(jù)視圖層次結(jié)構(gòu)呆盖,之后設置 導航欄的 barContentView 的內(nèi)邊距小一點,于是就有了這行代碼:
【 view.layoutMargins = UIEdgeInsetsMake(0, 2, 0, 2) 】
OK匾七,雖然顯示在導航欄里的東西設置完了絮短,也顯示正常了江兢,但是還有一個問題沒有討論昨忆,那就是自定義導航欄的一大好處!
先來看效果:
這個效果避免了與大多數(shù)App那樣前后兩個頁面的導航欄融合在一起混亂的感覺杉允,這樣子多么清爽邑贴!
So,自定義導航欄的一大作用就是在push或者pop時前后兩個導航欄相互獨立叔磷,看起來清爽宜人拢驾!
附:導航欄結(jié)構(gòu)類名
UINavigationBar
** _UIBarBackground
****** UIImageView
****** UIImageView
** _UINavigationBarContentView
****** _UIButtonBarStackView(left)
********** _UITAMICAdaptorView(左側(cè)按鈕)
********** UIView(多個按鈕之間的間距)
********** _UIButtonBarButton
****** _UIButtonBarStackView(right)
********** _UITAMICAdaptorView
************** UIButton(右側(cè)按鈕)
********** UIView(多個按鈕之間的間距)
********** _UITAMICAdaptorView
********** UIView(多個按鈕之間的間距)