Android屏幕適配出現的緣由html
在咱們學習如何進行屏幕適配以前,咱們須要先了解下爲何Android須要進行屏幕適配。前端
因爲Android系統的開放性,任何用戶、開發者、OEM廠商、運營商均可以對Android進行定製,修改爲他們想要的樣子。java
可是這種「碎片化」到底到達什麼程度呢?android
在2012年,OpenSignalMaps(如下簡稱OSM)發佈了第一份Android碎片化報告,統計數據代表,app
2012年,支持Android的設備共有3997種。框架
2013年,支持Android的設備共有11868種。iphone
2014年,支持Android的設備共有18796種。ide
下面這張圖片所顯示的內容足以充分說明當今Android系統碎片化問題的嚴重性,由於該圖片中的每個矩形都表明着一種Android設備。佈局
而隨着支持Android系統的設備(手機、平板、電視、手錶)的增多,設備碎片化、品牌碎片化、系統碎片化、傳感器碎片化和屏幕碎片化的程度也在不斷地加深。而咱們今天要探討的,則是對咱們開發影響比較大的——屏幕的碎片化。學習
下面這張圖是Android屏幕尺寸的示意圖,在這張圖裏面,藍色矩形的大小表明不一樣尺寸,顏色深淺則表明所佔百分比的大小。
而與之相對應的,則是下面這張圖。這張圖顯示了IOS設備所須要進行適配的屏幕尺寸和佔比。
固然,這張圖片只是4,4s,5,5c,5s和平板的尺寸,如今還應該加上新推出的iphone6和plus,可是和Android的屏幕碎片化程度相比而言,仍是差的太遠。
詳細的統計數據請到這裏查看。
如今你應該很清楚爲何要對Android的屏幕進行適配了吧?屏幕尺寸這麼多,爲了讓咱們開發的程序可以比較美觀的顯示在不一樣尺寸、分辨率、像素密度(這些概念我會在下面詳細講解)的設備上,那就要在開發的過程當中進行處理,至於如何去進行處理,這就是咱們今天的主題了。
可是在開始進入主題以前,咱們再來探討一件事情,那就是Android設備的屏幕尺寸,從幾寸的智能手機,到10寸的平板電腦,再到幾十寸的數字電視,咱們應該適配哪些設備呢?
其實這個問題不該該這麼考慮,由於對於具備相同像素密度的設備來講,像素越高,尺寸就越大,因此咱們能夠換個思路,將問題從單純的尺寸大小轉換到像素大小和像素密度的角度來。
下圖是2014年初,友盟統計的佔比5%以上的6個主流分辨率,能夠看出,佔比最高的是480*800,320*480的設備居然也佔據了很大比例,可是和半年前的數據相比較,中低分辨率(320*480、480*800)的比例在減小,而中高分辨率的比例則在不斷地增長。雖然每一個分辨率所佔的比例在變化,可是總的趨勢沒變,仍是這六種,只是分辨率在不斷地提升。
因此說,咱們只要儘可能適配這幾種分辨率,就能夠在大部分的手機上正常運行了。
固然了,這只是手機的適配,對於平板設備(電視也能夠看作是平板),咱們還須要一些其餘的處理。
好了,到目前爲止,咱們已經弄清楚了Android開發爲何要進行適配,以及咱們應該適配哪些對象,接下來,終於進入咱們的正題了!
首先,咱們先要學習幾個重要的概念。
重要概念
什麼是屏幕尺寸、屏幕分辨率、屏幕像素密度?
什麼是dp、dip、dpi、sp、px?他們之間的關係是什麼?
什麼是mdpi、hdpi、xdpi、xxdpi?如何計算和區分?
在下面的內容中咱們將介紹這些概念。
屏幕尺寸
屏幕尺寸指屏幕的對角線的長度,單位是英寸,1英寸=2.54釐米
好比常見的屏幕尺寸有2.四、2.八、3.五、3.七、4.二、5.0、5.五、6.0等
屏幕分辨率
屏幕分辨率是指在橫縱向上的像素點數,單位是px,1px=1個像素點。通常以縱向像素*橫向像素,如1960*1080。
屏幕像素密度
屏幕像素密度是指每英寸上的像素點數,單位是dpi,即「dot per inch」的縮寫。屏幕像素密度與屏幕尺寸和屏幕分辨率有關,在單一變化條件下,屏幕尺寸越小、分辨率越高,像素密度越大,反之越小。
dp、dip、dpi、sp、px
px咱們應該是比較熟悉的,前面的分辨率就是用的像素爲單位,大多數狀況下,好比UI設計、Android原生API都會以px做爲統一的計量單位,像是獲取屏幕寬高等。
dip和dp是一個意思,都是Density Independent Pixels的縮寫,即密度無關像素,上面咱們說過,dpi是屏幕像素密度,假如一英寸裏面有160個像素,這個屏幕的像素密度就是160dpi,那麼在這種狀況下,dp和px如何換算呢?在Android中,規定以160dpi爲基準,1dip=1px,若是密度是320dpi,則1dip=2px,以此類推。
假如一樣都是畫一條320px的線,在480*800分辨率手機上顯示爲2/3屏幕寬度,在320*480的手機上則佔滿了全屏,若是使用dp爲單位,在這兩種分辨率下,160dp都顯示爲屏幕一半的長度。這也是爲何在Android開發中,寫佈局的時候要儘可能使用dp而不是px的緣由。
而sp,即scale-independent pixels,與dp相似,可是能夠根據文字大小首選項進行放縮,是設置字體大小的御用單位。
mdpi、hdpi、xdpi、xxdpi
其實以前還有個ldpi,可是隨着移動設備配置的不斷升級,這個像素密度的設備已經很罕見了,所在如今適配時不需考慮。
mdpi、hdpi、xdpi、xxdpi用來修飾Android中的drawable文件夾及values文件夾,用來區分不一樣像素密度下的圖片和dimen值。
那麼如何區分呢?Google官方指定按照下列標準進行區分:
在進行開發的時候,咱們須要把合適大小的圖片放在合適的文件夾裏面。下面以圖標設計爲例進行介紹。
在設計圖標時,對於五種主流的像素密度(MDPI、HDPI、XHDPI、XXHDPI 和 XXXHDPI)應按照 2:3:4:6:8 的比例進行縮放。例如,一個啓動圖標的尺寸爲48x48 dp,這表示在 MDPI 的屏幕上其實際尺寸應爲 48x48 px,在 HDPI 的屏幕上其實際大小是 MDPI 的 1.5 倍 (72x72 px),在 XDPI 的屏幕上其實際大小是 MDPI 的 2 倍 (96x96 px),依此類推。
雖然 Android 也支持低像素密度 (LDPI) 的屏幕,但無需爲此費神,系統會自動將 HDPI 尺寸的圖標縮小到 1/2 進行匹配。
下圖爲圖標的各個屏幕密度的對應尺寸:
解決方案
支持各類屏幕尺寸
使用wrap_content、match_parent、weight
要確保佈局的靈活性並適應各類尺寸的屏幕,應使用 「wrap_content」 和 「match_parent」 控制某些視圖組件的寬度和高度。
使用 「wrap_content」,系統就會將視圖的寬度或高度設置成所需的最小尺寸以適應視圖中的內容,而 「match_parent」(在低於 API 級別 8 的級別中稱爲 「fill_parent」)則會展開組件以匹配其父視圖的尺寸。
若是使用 「wrap_content」 和 「match_parent」 尺寸值而不是硬編碼的尺寸,視圖就會相應地僅使用自身所需的空間或展開以填滿可用空間。此方法可以讓佈局正確適應各類屏幕尺寸和屏幕方向。
下面是一段示例代碼
下圖是在橫縱屏切換的時候的顯示效果,咱們能夠看到這樣能夠很好的適配屏幕尺寸的變化。
weight是線性佈局的一個獨特的屬性,咱們可使用這個屬性來按照比例對界面進行分配,完成一些特殊的需求。
可是,咱們對於這個屬性的計算應該如何理解呢?
首先看下面的例子,咱們在佈局中這樣設置咱們的界面
咱們在佈局裏面設置爲線性佈局,橫向排列,而後放置兩個寬度爲0dp的按鈕,分別設置weight爲1和2,在效果圖中,咱們能夠看到兩個按鈕按照1:2的寬度比例正常排列了,這也是咱們常用到的場景,這是時候很好理解,Button1的寬度就是1/(1+2) = 1/3,Button2的寬度則是2/(1+2) = 2/3,咱們能夠很清楚的明白這種情景下的佔好比何計算。
可是假如咱們的寬度不是0dp(wrap_content和0dp的效果相同),則是match_parent呢?
下面是設置爲match_parent的效果
咱們能夠看到,在這種狀況下,佔比和上面正好相反,這是怎麼回事呢?說到這裏,咱們就不得不提一下weight的計算方法了。
android:layout_weight的真實含義是:若是View設置了該屬性而且有效,那麼該 View的寬度等於原有寬度(android:layout_width)加上剩餘空間的佔比。
從這個角度咱們來解釋一下上面的現象。在上面的代碼中,咱們設置每一個Button的寬度都是match_parent,假設屏幕寬度爲L,那麼每一個Button的寬度也應該都爲L,剩餘寬度就等於L-(L+L)= -L。
Button1的weight=1,剩餘寬度佔比爲1/(1+2)= 1/3,因此最終寬度爲L+1/3*(-L)=2/3L,Button2的計算相似,最終寬度爲L+2/3(-L)=1/3L。
這是在水平方向上的,那麼在垂直方向上也是這樣嗎?
下面是測試代碼和效果
若是是垂直方向,那麼咱們應該改變的是layout_height的屬性,下面是0dp的顯示效果
下面是match_parent的顯示效果,結論和水平是徹底同樣的
雖說咱們演示了match_parent的顯示效果,並說明了緣由,可是在真正用的時候,咱們都是設置某一個屬性爲0dp,而後按照權重計算所佔百分比。
使用相對佈局,禁用絕對佈局
在開發中,咱們大部分時候使用的都是線性佈局、相對佈局和幀佈局,絕對佈局因爲適配性極差,因此極少使用。
因爲各類佈局的特色不同,因此不能說哪一個佈局好用,到底應該使用什麼佈局只能根據實際需求來肯定。咱們可使用 LinearLayout 的嵌套實例並結合 「wrap_content」 和 「match_parent」,以便構建至關複雜的佈局。不過,咱們沒法經過 LinearLayout 精確控制子視圖的特殊關係;系統會將 LinearLayout 中的視圖直接並排列出。
若是咱們須要將子視圖排列出各類效果而不是一條直線,一般更合適的解決方法是使用 RelativeLayout,這樣就能夠根據各組件之間的特殊關係指定佈局了。例如,咱們能夠將某個子視圖對齊到屏幕左側,同時將另外一個視圖對齊到屏幕右側。
下面的代碼以官方Demo爲例說明。
在上面的代碼中咱們使用了相對佈局,而且使用alignXXX等屬性指定了子控件的位置,下面是這種佈局方式在應對屏幕變化時的表現
在小尺寸屏幕的顯示
在平板的大尺寸上的顯示效果
雖然控件的大小因爲屏幕尺寸的增長而發生了改變,可是咱們能夠看到,因爲使用了相對佈局,因此控件以前的位置關係並無發生什麼變化,這說明咱們的適配成功了。
使用限定符
使用尺寸限定符
上面所提到的靈活佈局或者是相對佈局,能夠爲咱們帶來的優點就只有這麼多了。雖然這些佈局能夠拉伸組件內外的空間以適應各類屏幕,但它們不必定能爲每種屏幕都提供最佳的用戶體驗。所以,咱們的應用不只僅只實施靈活佈局,還應該應針對各類屏幕配置提供一些備用佈局。
如何作到這一點呢?咱們能夠經過使用配置限定符,在運行時根據當前的設備配置自動選擇合適的資源了,例如根據各類屏幕尺寸選擇不一樣的佈局。
不少應用會在較大的屏幕上實施「雙面板」模式,即在一個面板上顯示項目列表,而在另外一面板上顯示對應內容。平板電腦和電視的屏幕已經大到能夠同時容納這兩個面板了,但手機屏幕就須要分別顯示。所以,咱們可使用如下文件以便實施這些佈局:
res/layout/main.xml,單面板(默認)佈局:
res/layout-large/main.xml,雙面板佈局:
請注意第二種佈局名稱目錄中的 large 限定符。系統會在屬於較大屏幕(例如 7 英寸或更大的平板電腦)的設備上選擇此佈局。系統會在較小的屏幕上選擇其餘佈局(無限定符)。
使用最小寬度限定符
在版本低於 3.2 的 Android 設備上,開發人員遇到的問題之一是「較大」屏幕的尺寸範圍,該問題會影響戴爾 Streak、早期的 Galaxy Tab 以及大部分 7 英寸平板電腦。即便這些設備的屏幕屬於「較大」的尺寸,但不少應用可能會針對此類別中的各類設備(例如 5 英寸和 7 英寸的設備)顯示不一樣的佈局。這就是 Android 3.2 版在引入其餘限定符的同時引入「最小寬度」限定符的緣由。
最小寬度限定符可以讓您經過指定某個最小寬度(以 dp 爲單位)來定位屏幕。例如,標準 7 英寸平板電腦的最小寬度爲 600 dp,所以若是您要在此類屏幕上的用戶界面中使用雙面板(但在較小的屏幕上只顯示列表),您可使用上文中所述的單面板和雙面板這兩種佈局,但您應使用 sw600dp 指明雙面板佈局僅適用於最小寬度爲 600 dp 的屏幕,而不是使用 large 尺寸限定符。
res/layout/main.xml,單面板(默認)佈局:
res/layout-sw600dp/main.xml,雙面板佈局:
也就是說,對於最小寬度大於等於 600 dp 的設備,系統會選擇 layout-sw600dp/main.xml(雙面板)佈局,不然系統就會選擇 layout/main.xml(單面板)佈局。
但 Android 版本低於 3.2 的設備不支持此技術,緣由是這些設備沒法將 sw600dp 識別爲尺寸限定符,所以咱們仍需使用 large 限定符。這樣一來,就會有一個名稱爲 res/layout-large/main.xml 的文件(與 res/layout-sw600dp/main.xml 同樣)。可是沒有太大關係,咱們將立刻學習如何避免此類佈局文件出現的重複。
使用佈局別名
最小寬度限定符僅適用於 Android 3.2 及更高版本。所以,若是咱們仍需使用與較低版本兼容的歸納尺寸範圍(小、正常、大和特大)。例如,若是要將用戶界面設計成在手機上顯示單面板,但在 7 英寸平板電腦、電視和其餘較大的設備上顯示多面板,那麼咱們就須要提供如下文件:
res/layout/main.xml: 單面板佈局
res/layout-large: 多面板佈局
res/layout-sw600dp: 多面板佈局
後兩個文件是相同的,由於其中一個用於和 Android 3.2 設備匹配,而另外一個則是爲使用較低版本 Android 的平板電腦和電視準備的。
要避免平板電腦和電視的文件出現重複(以及由此帶來的維護問題),您可使用別名文件。例如,您能夠定義如下佈局:
res/layout/main.xml,單面板佈局
res/layout/main_twopanes.xml,雙面板佈局
而後添加這兩個文件:
res/values-large/layout.xml:
res/values-sw600dp/layout.xml:
後兩個文件的內容相同,但它們並未實際定義佈局。它們只是將 main 設置成了 main_twopanes 的別名。因爲這些文件包含 large 和 sw600dp 選擇器,所以不管 Android 版本如何,系統都會將這些文件應用到平板電腦和電視上(版本低於 3.2 的平板電腦和電視會匹配 large,版本高於 3.2 的平板電腦和電視則會匹配 sw600dp)。
使用屏幕方向限定符
某些佈局會同時支持橫向模式和縱向模式,但咱們能夠經過調整優化其中大部分佈局的效果。在新聞閱讀器示例應用中,每種屏幕尺寸和屏幕方向下的佈局行爲方式以下所示:
小屏幕,縱向:單面板,帶徽標
小屏幕,橫向:單面板,帶徽標
7 英寸平板電腦,縱向:單面板,帶操做欄
7 英寸平板電腦,橫向:雙面板,寬,帶操做欄
10 英寸平板電腦,縱向:雙面板,窄,帶操做欄
10 英寸平板電腦,橫向:雙面板,寬,帶操做欄
電視,橫向:雙面板,寬,帶操做欄
所以,這些佈局中的每一種都定義在了 res/layout/ 目錄下的某個 XML 文件中。爲了繼續將每一個佈局分配給各類屏幕配置,該應用會使用佈局別名將二者相匹配:
res/layout/onepane.xml:(單面板)
res/layout/onepane_with_bar.xml:(單面板帶操做欄)
res/layout/twopanes.xml:(雙面板,寬佈局)
res/layout/twopanes_narrow.xml:(雙面板,窄佈局)
既然咱們已定義了全部可能的佈局,那就只需使用配置限定符將正確的佈局映射到各類配置便可。
如今只需使用佈局別名技術便可作到這一點:
res/values/layouts.xml:
res/values-sw600dp-land/layouts.xml:
res/values-sw600dp-port/layouts.xml:
res/values-large-land/layouts.xml:
res/values-large-port/layouts.xml:
使用自動拉伸位圖
支持各類屏幕尺寸一般意味着您的圖片資源還必須能適應各類尺寸。例如,不管要應用到什麼形狀的按鈕上,按鈕背景都必須能適應。
若是在能夠更改尺寸的組件上使用了簡單的圖片,您很快就會發現顯示效果多少有些不太理想,由於系統會在運行時平均地拉伸或收縮您的圖片。解決方法爲使用自動拉伸位圖,這是一種格式特殊的 PNG 文件,其中會指明能夠拉伸以及不能夠拉伸的區域。
.9的製做,實際上就是在原圖片上添加1px的邊界,而後按照咱們的需求,把對應的位置設置成黑色線,系統就會根據咱們的實際需求進行拉伸。
下圖是對.9圖的四邊的含義的解釋,左上邊表明拉伸區域,右下邊表明padding box,就是間隔區域,在下面,咱們給出一個例子,方便你們理解。
先看下面兩張圖,咱們理解一下這四條線的含義。
上圖和下圖的區別,就在於右下邊的黑線不同,具體的效果的區別,看右邊的效果圖。上圖效果圖中深藍色的區域,表明內容區域,咱們能夠看到是在正中央的,這是由於咱們在右下邊的是兩個點,這兩個點距離上下左右四個方向的距離就是padding的距離,因此深藍色內容區域在圖片正中央,咱們再看下圖,因爲右下邊的黑線是圖片長度,因此就沒有padding,從效果圖上的表現就是深藍色區域和圖片同樣大,所以,咱們能夠利用右下邊來控制內容與背景圖邊緣的padding。
若是你還不明白,那麼咱們看下面的效果圖,咱們分別以圖一和圖二做爲背景圖,下面是效果圖。
咱們能夠看到,使用wrap_content屬性設置長寬,圖一比圖二的效果大一圈,這是爲何呢?還記得我上面說的padding嗎?
這就是padding的效果提現,怎麼證實呢?咱們再看下面一張圖,給圖一添加padding=0,這樣背景圖設置的padding效果就沒了,是否是兩個同樣大了?
ok,我想你應該明白右下邊的黑線的含義了,下面咱們再看一下左上邊的效果。
下面咱們只設置了左上邊線,效果圖以下
上面的線沒有包住圖標,下面的線正好包住了圖標,從右邊的效果圖應該能夠看出差異,黑線所在的區域就是拉伸區域,上圖黑線所在的全是純色,因此圖標不變形,下面的拉伸區域包裹了圖標,因此在拉伸的時候就會對圖標進行拉伸,可是這樣就會致使圖標變形。注意到下面紅線區域了嘛?這是系統提示咱們的,由於這樣拉伸,不符合要求,因此會提示一下。
支持各類屏幕密度
使用非密度制約像素
因爲各類屏幕的像素密度都有所不一樣,所以相同數量的像素在不一樣設備上的實際大小也有所差別,這樣使用像素定義佈局尺寸就會產生問題。所以,請務必使用 dp 或 sp 單位指定尺寸。dp 是一種非密度制約像素,其尺寸與 160 dpi 像素的實際尺寸相同。sp 也是一種基本單位,但它可根據用戶的偏好文字大小進行調整(即尺度獨立性像素),所以咱們應將該測量單位用於定義文字大小。
例如,請使用 dp(而非 px)指定兩個視圖間的間距:
請務必使用 sp 指定文字大小:
除了介紹這些最基礎的知識以外,咱們下面再來討論一下另一個問題。
通過上面的介紹,咱們都清楚,爲了可以規避不一樣像素密度的陷阱,Google推薦使用dp來代替px做爲控件長度的度量單位,可是咱們來看下面的一個場景。
假如咱們以Nexus5做爲書寫代碼時查看效果的測試機型,Nexus5的總寬度爲360dp,咱們如今須要在水平方向上放置兩個按鈕,一個是150dp左對齊,另一個是200dp右對齊,中間留有10dp間隔,那麼在Nexus5上面的顯示效果就是下面這樣
<可是若是在Nexus S或者是Nexus One運行呢?下面是運行結果
能夠看到,兩個按鈕發生了重疊。
咱們都已經用了dp了,爲何會出現這種狀況呢?
你聽我慢慢道來。
雖說dp能夠去除不一樣像素密度的問題,使得1dp在不一樣像素密度上面的顯示效果相同,可是仍是因爲Android屏幕設備的多樣性,若是使用dp來做爲度量單位,並非全部的屏幕的寬度都是相同的dp長度,好比說,Nexus S和Nexus One屬於hdpi,屏幕寬度是320dp,而Nexus 5屬於xxhdpi,屏幕寬度是360dp,Galaxy Nexus屬於xhdpi,屏幕寬度是384dp,Nexus 6 屬於xxxhdpi,屏幕寬度是410dp。因此說,光Google本身一家的產品就已經有這麼多的標準,並且屏幕寬度和像素密度沒有任何關聯關係,即便咱們使用dp,在320dp寬度的設備和410dp的設備上,仍是會有90dp的差異。固然,咱們儘可能使用match_parent和wrap_content,儘量少的用dp來指定控件的具體長寬,再結合上權重,大部分的狀況咱們都是能夠作到適配的。
可是除了這個方法,咱們還有沒有其餘的更完全的解決方案呢?
咱們換另一個思路來思考這個問題。
下面的方案來自Android Day Day Up 一羣的【blue-深圳】,謝謝他的分享精神。
由於分辨率不同,因此不能用px;由於屏幕寬度不同,因此要當心的用dp,那麼咱們可不能夠用另一種方法來統一單位,無論分辨率是多大,屏幕寬度用一個固定的值的單位來統計呢?
答案是:固然能夠。
咱們假設手機屏幕的寬度都是320某單位,那麼咱們將一個屏幕寬度的總像素數平均分紅320份,每一份對應具體的像素就能夠了。
具體如何來實現呢?咱們看下面的代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
|
import
java.io.File;
import
java.io.FileNotFoundException;
import
java.io.FileOutputStream;
import
java.io.PrintWriter;
public
class
MakeXml {
private
final
static
String rootPath =
"C:\\Users\\Administrator\\Desktop\\layoutroot\\values-{0}x{1}\\"
;
private
final
static
float
dw = 320f;
private
final
static
float
dh = 480f;
private
final
static
String WTemplate =
"[dimen name=\"x{0}\"]{1}px[/dimen]\n"
;
private
final
static
String HTemplate =
"[dimen name=\"y{0}\"]{1}px[/dimen]\n"
;
public
static
void
main(String[] args) {
makeString(
320
,
480
);
makeString(
480
,
800
);
makeString(
480
,
854
);
makeString(
540
,
960
);
makeString(
600
,
1024
);
makeString(
720
,
1184
);
makeString(
720
,
1196
);
makeString(
720
,
1280
);
makeString(
768
,
1024
);
makeString(
800
,
1280
);
makeString(
1080
,
1812
);
makeString(
1080
,
1920
);
makeString(
1440
,
2560
);
}
public
static
void
makeString(
int
w,
int
h) {
StringBuffer sb =
new
StringBuffer();
sb.append(
"[?xml version=\"1.0\" encoding=\"utf-8\"?]\n"
);
sb.append(
"[resources]"
);
float
cellw = w / dw;
for
(
int
i =
1
; i <
320
; i++) {
sb.append(WTemplate.replace(
"{0}"
, i +
""
).replace(
"{1}"
,
change(cellw * i) +
""
));
}
sb.append(WTemplate.replace(
"{0}"
,
"320"
).replace(
"{1}"
, w +
""
));
sb.append(
"[/resources]"
);
StringBuffer sb2 =
new
StringBuffer();
sb2.append(
"[?xml version=\"1.0\" encoding=\"utf-8\"?]\n"
);
sb2.append(
"[resources]"
);
float
cellh = h / dh;
for
(
int
i =
1
; i <
480
; i++) {
sb2.append(HTemplate.replace(
"{0}"
, i +
""
).replace(
"{1}"
,
change(cellh * i) +
""
));
}
sb2.append(HTemplate.replace(
"{0}"
,
"480"
).replace(
"{1}"
, h +
""
));
sb2.append(
"[/resources]"
);
String path = rootPath.replace(
"{0}"
, h +
""
).replace(
"{1}"
, w +
""
);
File rootFile =
new
File(path);
if
(!rootFile.exists()) {
rootFile.mkdirs();
}
File layxFile =
new
File(path +
"lay_x.xml"
);
File layyFile =
new
File(path +
"lay_y.xml"
);
try
{
PrintWriter pw =
new
PrintWriter(
new
FileOutputStream(layxFile));
pw.print(sb.toString());
pw.close();
pw =
new
PrintWriter(
new
FileOutputStream(layyFile));
pw.print(sb2.toString());
pw.close();
}
catch
(FileNotFoundException e) {
e.printStackTrace();
}
}
public
static
float
change(
float
a) {
int
temp = (
int
) (a *
100
);
return
temp / 100f;
}
}
|
代碼應該很好懂,咱們將一個屏幕寬度分爲320份,高度480份,而後按照實際像素對每個單位進行復制,放在對應values-widthxheight文件夾下面的lax.xml和lay.xml裏面,這樣就能夠統一全部你想要的分辨率的單位了,下面是生成的一個320*480分辨率的文件,由於寬高分割以後總分數和像素數相同,因此x1就是1px,以此類推。
那麼1080*1960分辨率下是什麼樣子呢?咱們能夠看下,因爲1080和320是3.37倍的關係,因此x1=3.37px
不管在什麼分辨率下,x320都是表明屏幕寬度,y480都是表明屏幕高度。
那麼,咱們應該如何使用呢?
首先,咱們要把生成的全部values文件夾放到res目錄下,當設計師把UI高清設計圖給你以後,你就能夠根據設計圖上的尺寸,以某一個分辨率的機型爲基礎,找到對應像素數的單位,而後設置給控件便可。
下圖仍是兩個Button,不一樣的是,咱們把單位換成了咱們在values文件夾下dimen的值,這樣在你指定的分辨率下,無論寬度是320dp、360dp,仍是410dp,就均可以徹底適配了。
可是,仍是有個問題,爲何下面的三個沒有適配呢?
這是由於因爲在生成的values文件夾裏,沒有對應的分辨率,其實一開始是報錯的,由於默認的values沒有對應dimen,因此我只能在默認values裏面也建立對應文件,可是裏面的數據卻很差處理,由於不知道分辨率,我只好默認爲x1=1dp保證儘可能兼容。這也是這個解決方案的幾個弊端,對於沒有生成對應分辨率文件的手機,會使用默認values文件夾,若是默認文件夾沒有,就會出現問題。
因此說,這個方案雖然是一勞永逸,可是因爲實際上仍是使用的px做爲長度的度量單位,因此多少和google的要求有所背離,很差說之後會不會出現什麼不可預測的問題。其次,若是要使用這個方案,你必須儘量多的包含全部的分辨率,由於這個是使用這個方案的基礎,若是有分辨率缺乏,會形成顯示效果不好,甚至出錯的風險,而這又勢必會增長軟件包的大小和維護的難度,因此你們本身斟酌,擇優使用。
更多信息可參考鴻洋的新文章:Android 屏幕適配方案。
提供備用位圖
因爲 Android 可在具備各類屏幕密度的設備上運行,所以咱們提供的位圖資源應始終能夠知足各種廣泛密度範圍的要求:低密度、中等密度、高密度以及超高密度。這將有助於咱們的圖片在全部屏幕密度上都能獲得出色的質量和效果。
要生成這些圖片,咱們應先提取矢量格式的原始資源,而後根據如下尺寸範圍針對各密度生成相應的圖片。
xhdpi:2.0
hdpi:1.5
mdpi:1.0(最低要求)
ldpi:0.75
也就是說,若是咱們爲 xhdpi 設備生成了 200x200 px尺寸的圖片,就應該使用同一資源爲 hdpi、mdpi 和 ldpi 設備分別生成 150x150、100x100 和 75x75 尺寸的圖片。
而後,將生成的圖片文件放在 res/ 下的相應子目錄中(mdpi、hdpi、xhdpi、xxhdpi),系統就會根據運行您應用的設備的屏幕密度自動選擇合適的圖片。
這樣一來,只要咱們引用 @drawable/id,系統都能根據相應屏幕的 dpi 選取合適的位圖。
還記得咱們上面提到的圖標設計尺寸嗎?和這個實際上是一個意思。
可是還有個問題須要注意下,若是是.9圖或者是不須要多個分辨率的圖片,就放在drawable文件夾便可,對應分辨率的圖片要正確的放在合適的文件夾,不然會形成圖片拉伸等問題。
實施自適應用戶界面流程
前面咱們介紹過,如何根據設備特色顯示恰當的佈局,可是這樣作,會使得用戶界面流程可能會有所不一樣。例如,若是應用處於雙面板模式下,點擊左側面板上的項便可直接在右側面板上顯示相關內容;而若是該應用處於單面板模式下,點擊相關的內容應該跳轉到另一個Activity進行後續的處理。因此咱們應該按照下面的流程,一步步的完成自適應界面的實現。
肯定當前佈局
因爲每種佈局的實施都會稍有不一樣,所以咱們須要先肯定當前向用戶顯示的佈局。例如,咱們能夠先了解用戶所處的是「單面板」模式仍是「雙面板」模式。要作到這一點,能夠經過查詢指定視圖是否存在以及是否已顯示出來。
1
2
3
4
5
6
7
8
9
10
11
12
13
|
ublic
class
NewsReaderActivity
extends
FragmentActivity {
boolean
mIsDualPane;
@Override
public
void
onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
setContentView(R.layout.main_layout);
View articleView = findViewById(R.id.article);
mIsDualPane = articleView !=
null
&&
articleView.getVisibility() == View.VISIBLE;
}
}
|
請注意,這段代碼用於查詢「報道」面板是否可用,與針對具體佈局的硬編碼查詢相比,這段代碼的靈活性要大得多。
再舉一個適應各類組件的存在狀況的方法示例:在對這些組件執行操做前先查看它們是否可用。例如,新聞閱讀器示例應用中有一個用於打開菜單的按鈕,但只有在版本低於 3.0 的 Android 上運行該應用時,這個按鈕纔會存在,由於 API 級別 11 或更高級別中的 ActionBar 已取代了該按鈕的功能。所以,您可使用如下代碼爲此按鈕添加事件偵聽器:
1
2
3
4
5
|
Button catButton = (Button) findViewById(R.id.categorybutton);
OnClickListener listener =
/* create your listener here */
;
if
(catButton !=
null
) {
catButton.setOnClickListener(listener);
}
|
根據當前佈局作出響應
有些操做可能會因當前的具體佈局而產生不一樣的結果。例如,在新聞閱讀器示例中,若是用戶界面處於雙面板模式下,那麼點擊標題列表中的標題就會在右側面板中打開相應報道;但若是用戶界面處於單面板模式下,那麼上述操做就會啓動一個獨立活動:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
@Override
public void onHeadlineSelected(int index) {
mArtIndex = index;
if
(mIsDualPane) {
/* display article on the right pane */
mArticleFragment.displayArticle(mCurrentCat.getArticle(index));
}
else
{
/* start a separate activity */
Intent intent =
new
Intent(
this
, ArticleActivity.class);
intent.putExtra(
"catIndex"
, mCatIndex);
intent.putExtra(
"artIndex"
, index);
startActivity(intent);
}
}
|
一樣,若是該應用處於雙面板模式下,就應設置帶導航標籤的操做欄;但若是該應用處於單面板模式下,就應使用下拉菜單設置導航欄。所以咱們的代碼還應肯定哪一種狀況比較合適:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
|
final String CATEGORIES[] = {
"熱門報道"
,
"政治"
,
"經濟"
,
"Technology"
};
public void onCreate(Bundle savedInstanceState) {
....
if
(mIsDualPane) {
/* use tabs for navigation */
actionBar.setNavigationMode(android.app.ActionBar.NAVIGATION_MODE_TABS);
int i;
for
(i = 0; i < CATEGORIES.length; i++) {
actionBar.addTab(actionBar.newTab().setText(
CATEGORIES[i]).setTabListener(handler));
}
actionBar.setSelectedNavigationItem(selTab);
}
else
{
/* use list navigation (spinner) */
actionBar.setNavigationMode(android.app.ActionBar.NAVIGATION_MODE_LIST);
SpinnerAdapter adap =
new
ArrayAdapter(
this
,
R.layout.headline_item, CATEGORIES);
actionBar.setListNavigationCallbacks(adap, handler);
}
}
|
重複使用其餘活動中的片斷
多屏幕設計中的重複模式是指,對於某些屏幕配置,已實施界面的一部分會用做面板;但對於其餘配置,這部分就會以獨立活動的形式存在。例如,在新聞閱讀器示例中,對於較大的屏幕,新聞報道文本會顯示在右側面板中;但對於較小的屏幕,這些文本就會以獨立活動的形式存在。
在相似狀況下,一般能夠在多個活動中重複使用相同的 Fragment 子類以免代碼重複。例如,在雙面板佈局中使用了 ArticleFragment:
而後又在小屏幕的Activity佈局中重複使用了它 :
1
2
|
ArticleFragment frag =
new
ArticleFragment();
getSupportFragmentManager().beginTransaction().add(android.R.id.content, frag).commit();
|
固然,這與在 XML 佈局中聲明片斷的效果是同樣的,但在這種狀況下卻不必使用 XML 佈局,由於報道片斷是此活動中的惟一組件。
請務必在設計片斷時注意,不要針對具體活動建立強耦合。要作到這一點,一般能夠定義一個接口,該接口歸納了相關片斷與其主活動交互所需的所有方式,而後讓主活動實施該界面:
例如,新聞閱讀器應用的 HeadlinesFragment 會精確執行如下代碼:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
public class HeadlinesFragment extends ListFragment {
...
OnHeadlineSelectedListener mHeadlineSelectedListener =
null
;
/* Must be implemented by host activity */
public interface OnHeadlineSelectedListener {
public void onHeadlineSelected(int index);
}
...
public void setOnHeadlineSelectedListener(OnHeadlineSelectedListener listener) {
mHeadlineSelectedListener = listener;
}
}
|
而後,若是用戶選擇某個標題,相關片斷就會通知由主活動指定的偵聽器(而不是通知某個硬編碼的具體活動):
1
2
3
4
5
6
7
8
9
10
11
|
public class HeadlinesFragment extends ListFragment {
...
@Override
public void onItemClick(AdapterView parent,
View view, int position, long id) {
if
(
null
!= mHeadlineSelectedListener) {
mHeadlineSelectedListener.onHeadlineSelected(position);
}
}
...
}
|
除此以外,咱們還可使用第三方框架,好比說使用「訂閱-發佈」模式的EventBus來更多的優化組件之間的通訊,減小耦合。
處理屏幕配置變化
若是咱們使用獨立Activity實施界面的獨立部分,那麼請注意,咱們可能須要對特定配置變化(例如屏幕方向的變化)作出響應,以便保持界面的一致性。
例如,在運行 Android 3.0 或更高版本的標準 7 英寸平板電腦上,若是新聞閱讀器示例應用運行在縱向模式下,就會在使用獨立活動顯示新聞報道;但若是該應用運行在橫向模式下,就會使用雙面板佈局。
也就是說,若是用戶處於縱向模式下且屏幕上顯示的是用於閱讀報道的活動,那麼就須要在檢測到屏幕方向變化(變成橫向模式)後執行相應操做,即中止上述活動並返回主活動,以便在雙面板佈局中顯示相關內容:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
public class ArticleActivity extends FragmentActivity {
int mCatIndex, mArtIndex;
@Override
protected void onCreate(Bundle savedInstanceState) {
super
.onCreate(savedInstanceState);
mCatIndex = getIntent().getExtras().getInt(
"catIndex"
, 0);
mArtIndex = getIntent().getExtras().getInt(
"artIndex"
, 0);
// If should be in two-pane mode, finish to return to main activity
if
(getResources().getBoolean(R.bool.has_two_panes)) {
finish();
return
;
}
...
}
|
經過上面幾個步驟,咱們就徹底能夠創建一個能夠根據用戶界面配置進行自適應的App了。
最佳實踐
關於高清設計圖尺寸
Google官方給出的高清設計圖尺寸有兩種方案,一種是以mdpi設計,而後對應放大獲得更高分辨率的圖片,另一種則是以高分辨率做爲設計大小,而後按照倍數對應縮小到小分辨率的圖片。
根據經驗,我更推薦第二種方法,由於小分辨率在生成高分辨率圖片的時候,會出現像素丟失,我不知道是否是有方法能夠阻止這種狀況發生。
而分辨率能夠以1280*720或者是1960*1080做爲主要分辨率進行設計。
ImageView的ScaleType屬性
設置不一樣的ScaleType會獲得不一樣的顯示效果,通常狀況下,設置爲centerCrop能得到較好的適配效果。
動態設置
有一些狀況下,咱們須要動態的設置控件大小或者是位置,好比說popwindow的顯示位置和偏移量等,這個時候咱們能夠動態的獲取當前的屏幕屬性,而後設置合適的數值
1
2
3
4
5
6
7
8
9
10
11
|
public class ScreenSizeUtil {
public static int getScreenWidth(Activity activity) {
return
activity.getWindowManager().getDefaultDisplay().getWidth();
}
public static int getScreenHeight(Activity activity) {
return
activity.getWindowManager().getDefaultDisplay().getHeight();
}
}
|
轉自:連接