iPhone X 適配(全)

来源:https://www.cnblogs.com/jukaiit/archive/2018/01/10/8260182.html
-Advertisement-
Play Games

背景 iPhone X 劉海機於9月13日發佈,給科技小春晚帶來一波高潮。作為開發人員卻多出來一份憂慮,iPhone X 怎麼適配?我們 App 的腦袋會不會也長一劉海出來?Tabbar 會不會被圓角?先來看一下美團 App 的表現: 圖 1.1 啟動時的 App 表現 圖 1.2 下拉刷新之後的表 ...


背景

iPhone X 劉海機於9月13日發佈,給科技小春晚帶來一波高潮。作為開發人員卻多出來一份憂慮,iPhone X 怎麼適配?我們 App 的腦袋會不會也長一劉海出來?Tabbar 會不會被圓角?先來看一下美團 App 的表現:

圖 1.1 啟動時的 App 表現           圖 1.2 下拉刷新之後的表現



圖 1.3 搜索的表現          圖 1.4 “我的Tab”表現

在圖1.1中乍一看表現還不錯,可是在圖1.2中,下拉刷新之後,我們的導航欄還是被劉海擋住了。搜索也中槍,搜索首頁沒有辦法取消,“熱門搜索區域”也多出來一塊兒空白。另外,“我的Tab”頁部分如圖1.3、圖1.4所示,導航欄回不去了,右上角的三個UIBarButtonItem也不見了。其他還有很多UI上的Bug,等著我們去一一發現並修改。

針對可能出現的問題,蘋果在 developer.apple.com 上給出了一些建議。其中一個是 HIG (Human Interface Guideline)。另外 WWDC 會議官方 App 的作者,也給出了適配時的一些經驗

我們來看看他們是怎麼說的。

HIG部分

首先看一下各個機型尺寸的變化。

圖 2.1 各版本 iPhone 的尺寸

下圖是 iPhone X 對比其他機型的變化部分。iPhone X 和 iPhone 8 的寬度一致,在垂直方向上多了145pt,這就意味著首頁可以展示更多的內容,多出來的這20%的垂直空間,也許可以掛上更高價值的運營位。

圖2.2 iPhone X 和其他設備的尺寸對比

佈局

註意圖2.2藍色部分,你會發現這些都算在了展示內容的區域。所以我們在設計的時候,要避免內容被圓角、劉海給擋住。Like this:

圖 2.3 CGRectMake(0,0,100,100)

iPhone X 的坐標系統以及能顯示內容的區域如下圖所示:

圖 2.4 iPhone X 的顯示區域

Status Bar

iPhone X 上的 StatusBar 高度比之前的 iPhone 高一些,也就是說,我們如果寫死20pt高度的 frame 佈局,都要大面積修(tu)改(xue)。在 iPhone X 上,通過列印 [[UIApplication sharedApplication] statusBarFrame] 可以看到,高度是44pt。

圖 2.5 iPhone X 的狀態欄高度

"如果你的 App 是隱藏 StatusBar 的,建議重新考慮。iPhone X 為用戶在垂直空間上提供了更多展示餘地,且狀態欄中也包含了用戶需要知道的信息,除非能通過隱藏狀態欄帶給用戶額外的價值,否則蘋果建議大家將狀態欄還給用戶。"

另外還有一點,用戶在使用 iPhone X 打電話的時候,StatusBar 的高度也不會發生變化了。

屏幕底部

因為沒有了 Home 鍵,iPhone X 的底部是預留給系統功能的一個區域 - Home Indicator,這部分的高度是34pt。

圖 2.6 iPhone X 的 Home Indicator 區域

“如果你的底部是 TabBar,那麼 Home Indicator 背景會來自於 TabBar 背景的延伸,如果我們是一個 feed 流的頁面,那麼底部會展示 feed 流的局部。”

意思是如果有 TabBar,那麼那個區域會延展你的 barTintColor;沒有的話,就顯示透明的(參照 Setting)。之所以這麼設計,是為了讓 indicator 清晰可見,告訴用戶你可以滑動這部分區域。所以蘋果不建議我們的 UI 元素過於靠近這部分區域。

圖 2.7 有 TabBar 的 Home Indicator 區

SafeArea

iOS 11 廢棄了 iOS 7 之後出現的 topLayoutGuide/bottomLayoutGuide,取而代之的是safeLayoutGuide 概念。我們的UI元素都應該佈局在這些區域之內,避免被各種 bar(NavgationBar、ToolBar、TabBar、StatusBar)遮擋。

圖2.8 iPhone 的 SafeArea

如果我們用了 AutoLayout,並且開啟了 safeAreaLayoutGuide,佈局會自動加上這些 safeLayoutGuide,你的視圖不會超出這部分 SafeArea。如2.9所示,如果你需要增加 Guide 的區域,那麼可以設置 self.additionalSafeAreaInsets 來增加區域。

圖 2.9 預設的 SafeArea 和 self.additionalSafeAreaInsets = UIEdgeInsetsMake(64, 0, 0, 0);

其他

還有其他的一些改變,比如圖片的 Aspect Ratio 在 iPhone X 上的表現也會有所不同了;

劉海兩邊的區域都能響應不同的手勢,最好不要和自己的 App 發生衝突。

來自Session 201的建議

① xib 里適配 iPhone X 的話,可以開啟 UseSafeAreaLayoutGuides(但這需要在 iOS 9 之後才能用,需要看你的 App 最低支持的版本)。


圖3.1 xib 屬性

② 如果用的系統 SearchViewController,發現沒有灰色蒙層了,可以這麼試試。

圖3.2 iOS 11 UISearchViewController適配

之所以可以這麼改,是因為 iOS 11 的 NavigationBar 和 SearchViewController 集成在一塊兒了。

③ 橫屏下的 UITableView,SenctionHeader 的背景顏色不是設置的那個顏色。

圖3.3 iOS 11 橫屏 Tableview 的作用方式

這個問題的原因是:橫屏下的 UITableView,Cell 都是和屏幕一樣寬,但是 Cell 的 ContentView 會被 inset 到 SafeArea 區域。
解決方法是:可以通過調整 Tableview 的預設行為,改變 contentView 的屬性(如上圖 inset To SafeArea)來讓 contentview 頂到邊緣,弊端是會改變整個 cell 的內容顯示,而且 contentView 的 layoutMargin 依然還是相對於 SafeArea 的。
最佳方案是:改變 UITableViewHeaderFooterView.backgroundView 的 backgroundColor。

圖3.4 iOS 11 修改前後的樣式對比

劉海打理初體驗

① 我們來看下開頭說的那個刷新之後首頁頂上去的問題怎麼處理。經過排查,這個問題屬於“狀態欄變高系列”,解決方案就是把固定的20pt高度改成 [[UIApplication sharedApplication] statusBarFrame].size.height]

② 搜索頁面輸入框的位置發生了偏移,這是因為 iOS 11 的導航欄的視圖層級結構發生了變化,和 iPhone X 的並無直接關係。iOS 11 導航欄的視圖層級關係如下:

圖4.1 iOS 11 之後的 NavigationBar     圖4.2 iOS 11 之前的 NavigationBar

適配方式是:取到這個 _UIButtonBarStackView 的位置和尺寸信息,然後更改 PFBNavigationBarContainerView 的 X 坐標。

③ “我的Tab” 頁面多出來一塊兒灰色的區域,經過排查發現這個是 Tableview 的背景色。也就是說其實是 Tableview 向下偏移了。

圖4.3 iOS11 下“我的Tab” 頁面 Tableview 發生偏移

出現這個的原因是:iOS 11 之後 scrollview 多出來一個 adjustedContentInset 區域。

圖 4.4 iOS 11下 ScrollView 的新屬性

通過列印這個值,我們發現正好和 contentoffset.y 相符合。

圖 4.5 這個新屬性在 iPhone X 上的值

那為什麼會發生偏移?這個偏移的值又是怎麼確定的?其實是當 Tableview 的 frame 超出了 safeArea 範圍之後,系統會調整內容的位置。系統通過設置 adjustedContentInset 為 safeAreaInset 的值讓 Tableview 偏移

圖 4.6 iPhone X 上 safeAreaInset 的值

註意一下這個 adjustedContentInset 是 readOnly 的屬性。我們可以通過設置 Tableview.contentInsetAdjustmentBehavior=UIScrollViewContentInsetAdjustmentNever 來糾正這個位置。當然還可以通過設置 tableview.contentOffset 來抵消這個值,但還是推薦第一種。

④ “我的Tab” 導航欄上,右邊那個按鈕全都發生了偏移,導致無法點擊。這個問題也是在新的導航欄結構視圖下會出現,原因是新的導航欄結構用了 AutoLayout 佈局,我們這個並不是用常規的 UIBarButtonItem 方式實現的,而是一個 UIBarButtonItem ,他的 customView 包含了三個 Button,這幾個 Button 都是 frame 佈局,從而導致了在 AutoLayout 下的佈局問題。

正常的解決方式是:修改成一個一個添加 UIBarButtonItem 和 UIBarButtonSystemItemFixedSpace。但是這樣引出來另外一個問題,iOS 11 之前那種設置負寬度的 fixedspace 來調整間距的 trick 方式已經失效了!詳情見https://forums.developer.apple.com/thread/80075

我們這邊的方式是:依然用那種一個 CustomView 里包含三個 CustomButton 的方式,然後分別加上約束。CustomView 只需要加上寬高,包含的 Button 加上 left、top 和 size。

圖 4.7 加約束修正後樣式

以下是嘗試修複這部分問題的代碼:

// offset 問題
if (@available(iOS 11.0, *)) {
        self.contentViewController.tableView.contentInsetAdjustmentBehavior = UIScrollViewContentInsetAdjustmentNever;
    }
// UIBarButtonItem 問題
if (@available(iOS 11.0, *)) {
        [messageButtonsContainerView mas_makeConstraints:^(MASConstraintMaker *make) {
            make.size.mas_equalTo(CGSizeMake(themeButton.width + settingButton.width + messageButton.width, 44));
        }];
        [themeButton mas_makeConstraints:^(MASConstraintMaker *make) {
            make.top.equalTo(messageButtonsContainerView);
            make.left.equalTo(messageButtonsContainerView);
            make.size.mas_offset(CGSizeMake(44, 44));
        }];
        [settingButton mas_makeConstraints:^(MASConstraintMaker *make) {
            make.top.equalTo(messageButtonsContainerView);
            make.left.equalTo(themeButton.mas_right);
            make.size.mas_offset(CGSizeMake(44, 44));
        }];
        [messageButton mas_makeConstraints:^(MASConstraintMaker *make) {
            make.top.equalTo(messageButtonsContainerView);
            make.left.equalTo(settingButton.mas_right).offset(-10);
            make.size.mas_offset(CGSizeMake(44, 44));
        }];
    }

    UIBarButtonItem *rightBarItem = [[UIBarButtonItem alloc] initWithCustomView:messageButtonsContainerView];
    self.navigationItem.rightBarButtonItems = @[rightBarItem];

總結

當前發現這些問題的環境是 Xcode 9 GM版本(9A235)的模擬器。歸結起來是三類問題:

  1. StatusBar 變高並且絕對佈局。
  2. 導航欄的視圖層級結構發生變化而導致 UI(titleView、UIBarButtonItem) 問題。(iPhone 6s iOS 11 上依然是舊的結構,是因為現在 AppStore 上的包依然是用 iOS 10 的 SDK 打出來的)。
  3. safeAreaInset 導致 Scrollview 偏移。

至於 Tabbar ,因為我們用的是系統的,所以目前並沒有發現什麼奇怪的地方。希望我們踩的這些坑可以讓各位在適配的過程中少走一些彎路!

 

源:https://tech.meituan.com/iPhoneX%E5%88%98%E6%B5%B7%E6%89%93%E7%90%86%E6%8C%87%E5%8C%97.html


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Box Anemometer是一個非常不錯的MySQL慢查詢可視化工具平臺,具體就不多詳述了,官網是https://github.com/box/Anemometer 最近動自己動手搭了一個,具體搭建過程,參考了http://www.cnblogs.com/xuanzhi201111/p/41288 ...
  • 概述 對於一般的資料庫系統審計可能不太會被重視,但是對於金融系統就不一樣的。金融系統對審計要求會很高,除了了記錄資料庫各種操作記錄還可能會需要開發報表來呈現這些行為數據。使用SQL Server Audit 功能,您可以對伺服器級別和資料庫級別事件組以及單個事件進行審核。 “審核”SQL Serve ...
  • System.Data.OracleClient 已經過時了。微軟不再支持它。 因此,我建議你為. NET 使用Oracle數據提供程式:ODP.Net. 你可以從以下位置下載: 版本:Release版本Release版本 Oracle Windows ( 32位 )Oracle資料庫 11g Re ...
  • 資料庫是網路應用的基礎,良好的表結構設計,對整個應用起著至關重要的作用。 資料庫設計的步驟: 1.需求分析:數據是什麼,有哪些屬性,數據和屬性的特點 2.邏輯設計:使用ER圖對資料庫進行邏輯建模 3.物理設計:選擇資料庫系統,並對邏輯設計進行轉化 4.維護優化:追加,分拆等 實例演示(電子商務網站) ...
  • 概述 擴展存儲過程是 SQL Server 實例可以動態載入和運行的 DLL。擴展存儲過程是使用 SQL Server 擴展存儲過程 API 編寫的,可直接在 SQL Server 實例的地址空間中運行。擴展存儲過程可能會產生記憶體泄漏或其他降低伺服器的性能及可靠性的問題。固定伺服器角色 sysadm ...
  • 資料庫SQL三種迴圈語句(For、While、Loop) ...
  • 學完這本書,就把精華都寫出了 ...
  • NO是常開(NORMAL OPEN),就是通常即未通電狀態下,是斷開的,通電後在電磁線圈的作用下(吸合)處於閉合狀態。NC是常閉(NORMAL CLOSE),就是通常即未通電狀態下,是閉合的,通電後在電磁線圈的作用下(吸合)處於斷開狀態。 指接觸器、繼電器等電氣開關元件輔助觸點在常態下(未通電時)的 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...