一筷畦、UI設(shè)計(jì)稿尺寸
iPhone設(shè)計(jì)尺寸參考:https://uiiiuiii.com/screen/ios.htm
在說屏幕適配之前考赛,先提一下UI設(shè)計(jì)稿的尺寸問題。我們布局時(shí)的的尺寸都是通過UI設(shè)計(jì)圖獲得犹芹,那么UI設(shè)計(jì)師是以什么屏幕尺寸為標(biāo)準(zhǔn)來設(shè)計(jì)的就至關(guān)重要了熙涤。
UI設(shè)計(jì)稿為Iphone 8 plus(1242px * 2208px)
一些公司IOS和Android共用一套設(shè)計(jì)稿阁苞,假設(shè)設(shè)計(jì)師以IPhone的標(biāo)準(zhǔn)設(shè)計(jì),對(duì)IOS來說就很友好祠挫,但是對(duì)Android來說就很坑了那槽。因?yàn)镮phone的Retina屏的像素對(duì)Android設(shè)備來說不是標(biāo)準(zhǔn)的尺寸。
假設(shè)UI設(shè)計(jì)師以iPhone 8 plus的尺寸為標(biāo)準(zhǔn)來設(shè)計(jì)等舔,iPhone 8 plus的設(shè)計(jì)尺寸為1242px * 2208px(這里有個(gè)坑骚灸,iPhone8plus的分辨率為1080 * 1920,和設(shè)計(jì)稿不一樣慌植,需要注意下)甚牲,在zeplin轉(zhuǎn)化成xxhdpi為414dp*736dp。
現(xiàn)在問題來了蝶柿,Android從哪找一部手機(jī)符合這個(gè)尺寸呢丈钙?
參考某些手機(jī)尺寸信息,發(fā)現(xiàn)Nexus6p的分辨率非常接近交汤。
Nexus 6p的分辨率為1440px * 2560px雏赦,這是xxxhdpi的尺寸,按照官網(wǎng)的說法,這個(gè)屏幕的縮放因子應(yīng)該為4星岗,可是實(shí)際測(cè)試scale=3.5填大。
1440/3.5=411,2560/3.5=731俏橘。非常接近設(shè)計(jì)稿的尺寸栋盹!
但是這樣是不夠精確的,而且還要對(duì)Android其他尺寸的設(shè)備做適配敷矫,如:1080 * 1920例获,720 * 1280,因此最好是IOS和Android各出一套UI圖
UI設(shè)計(jì)稿尺寸為1080*1920
如果UI以1080 * 1920的尺寸設(shè)計(jì)曹仗,那么就很好了榨汤,因?yàn)檫@個(gè)尺寸是自帶適配屬性的。而且國內(nèi)的大部分手機(jī)都是1080的分辨率怎茫。
為什么說1080 * 1920是自帶適配屬性收壕?因?yàn)槿绻愕牟季治募胐p做單位,那么下面三種尺寸是完全適配的轨蛤,因?yàn)樗麄兊膁p尺寸完全一樣蜜宪。
screenSize = 1440 * 2560 , screenDensity = 4, dp = 360 * 640
screenSize = 1080 * 1920 , screenDensity = 3, dp = 360 * 640
screenSize = 720 * 1280 , screenDensity = 2, dp = 360 * 640
如果出現(xiàn)其他奇葩的屏幕寬度,比如1120祥山、980這種圃验,那么就需要做對(duì)應(yīng)的適配了。
二缝呕、屏幕縮放因子
屏幕尺寸計(jì)算的關(guān)鍵
Android系統(tǒng)是怎樣計(jì)算縮放因子的呢澳窑?
看看Display.getMetrics()
方法做了什么,該方法最終調(diào)用下面的方法
private void getMetricsWithSize(DisplayMetrics outMetrics, CompatibilityInfo compatInfo,
Configuration configuration, int width, int height) {
//logicalDensityDpi是屏幕固有的dpi
outMetrics.densityDpi = outMetrics.noncompatDensityDpi = logicalDensityDpi;
//假設(shè)logicalDensityDpi=480供常,那么outMetrics.density=3摊聋,這里DENSITY_DEFAULT_SCALE=1/160=0.00625
outMetrics.density = outMetrics.noncompatDensity =
logicalDensityDpi * DisplayMetrics.DENSITY_DEFAULT_SCALE;
outMetrics.scaledDensity = outMetrics.noncompatScaledDensity = outMetrics.density;
//下面的x和y方向的dpi是通過dpi的公式計(jì)算出來的
outMetrics.xdpi = outMetrics.noncompatXdpi = physicalXDpi;
outMetrics.ydpi = outMetrics.noncompatYdpi = physicalYDpi;
width = configuration != null && configuration.appBounds != null
? configuration.appBounds.width() : width;
height = configuration != null && configuration.appBounds != null
? configuration.appBounds.height() : height;
outMetrics.noncompatWidthPixels = outMetrics.widthPixels = width;
outMetrics.noncompatHeightPixels = outMetrics.heightPixels = height;
if (!compatInfo.equals(CompatibilityInfo.DEFAULT_COMPATIBILITY_INFO)) {
compatInfo.applyToDisplayMetrics(outMetrics);
}
}
屏幕的固有dpi也可以在DisplayMetrics類里面找到
/**
* The device's stable density.
* <p>
* This value is constant at run time and may not reflect the current
* display density. To obtain the current density for a specific display,
* use {@link #densityDpi}.
*/
public static final int DENSITY_DEVICE_STABLE = getDeviceDensity();
因此屏幕的縮放因子是廠商固定好的。而且可能和你根據(jù)屏幕尺寸計(jì)算的結(jié)果不一致栈暇。
真機(jī)測(cè)試
測(cè)試機(jī)Honor7X
下面是商家給出的手機(jī)參數(shù)
根據(jù)商家給出的尺寸計(jì)算dpi = Sqrt(1080^2 + 2160^2)/5.93 = 407.243
在真機(jī)上將DisplayMetrics對(duì)象打印出來可以看到
屏幕的固有dpi和實(shí)際屏幕尺寸計(jì)算的dpi差很多麻裁,但是Android設(shè)備的縮放因子計(jì)算卻是以固有dpi為準(zhǔn):
縮放因子 = 固有dpi * DisplayMetrics.DENSITY_DEFAULT_SCALE
縮放因子決定屏幕的dp尺寸,而屏幕適配用到的也是dp尺寸源祈。
三煎源、屏幕適配
這里只考慮以寬度為基準(zhǔn)適配。適配最終的效果是在不同尺寸的設(shè)備上顯示相同的UI效果新博,因此最終顯示在各個(gè)設(shè)備上的尺寸是一個(gè)百分比尺寸薪夕。
舉個(gè)例子,假設(shè)設(shè)計(jì)稿為1080px寬度(360dp)赫悄,Nexus6p手機(jī)的寬度為1440px(411dp)原献,那么在1080手機(jī)上的1dp馏慨,放到1440手機(jī)上是多少呢?很簡單
width = 1 * 411 / 360 = 1 * (1440 / 3.5) / (1080 / 3) = 1.14 dp
1.生成百分比尺寸
我們?cè)陂_發(fā)時(shí)屏幕上的尺寸一般以dp和sp為單位姑隅,Android已經(jīng)幫我們做了一部分的自適應(yīng)写隶。但是不同屏幕dp的尺寸還是有較大區(qū)別。如果以屏幕為基礎(chǔ)的百分比尺寸為單位那就肯定不會(huì)有問題了讲仰。
以生成xxdpi和xdpi文件為例慕趴。
以寬度為計(jì)算標(biāo)準(zhǔn),計(jì)算百分比尺寸需要幾個(gè)參數(shù):
1鄙陡、你現(xiàn)在使用UI設(shè)計(jì)稿的手機(jī)的屏幕寬度冕房。比如我用的Nexus 6p,那么我的寬度為1440趁矾。
2耙册、你現(xiàn)在使用UI設(shè)計(jì)稿的手機(jī)的屏幕縮放因子。Nexus 6p為3.5
3毫捣、適配手機(jī)屏幕的寬度详拙,xxhdpi取1080,xhdpi取720蔓同。
4饶辙、適配手機(jī)屏幕的縮放因子。xxhdpi取3.0斑粱,xhdpi取2.0弃揽。
xxhdpi的百分比尺寸為:
UI_SCREEN_WIDTH = 1440
UI_SCREEN_SCALE = 3.5
XXHDPI_SCREEN_WIDTH = 1080
XXHDPI_SCREEN_SCALE = 3.0
percentSize = uiSize * ( XXHDPI_SCREEN_WIDTH / XXHDPI_SCREEN_SCALE ) / ( UI_SCREEN_WIDTH / UI_SCREEN_SCALE )
xhdpi的百分比尺寸為:
UI_SCREEN_WIDTH = 1440
UI_SCREEN_SCALE = 3.5
XHDPI_SCREEN_WIDTH = 720
XHDPI_SCREEN_SCALE = 2.0
percentSize = uiSize * ( XHDPI_SCREEN_WIDTH / XHDPI_SCREEN_SCALE ) / ( UI_SCREEN_WIDTH / UI_SCREEN_SCALE)
uiSize就是你使用設(shè)計(jì)稿的dp尺寸,得出的percentSize就是相對(duì)應(yīng)屏幕的百分比dp尺寸珊佣。
上面只適配了1080和720兩種寬度蹋宦,還需要適配哪些寬度呢?
看看友盟的設(shè)備統(tǒng)計(jì):https://compass.umeng.com/#/equipment?_k=8snfbu
2.建立適配文件夾
根據(jù)自己的適配需求來建立對(duì)應(yīng)的適配文件
適配文件的創(chuàng)建可以參考:Android資源文件說明
繼續(xù)以Nexus6p為例子咒锻,設(shè)計(jì)稿的尺寸是411dp的,假設(shè)我想適配1440(360dp)守屉、1080(360dp)惑艇、720(360dp)的標(biāo)準(zhǔn)屏幕,那么我需要建立如下幾個(gè)文件:
res/values-xhdpi/dimens.xml
res/values-xxhdpi/dimens.xml
res/values-sw411dp-xxxhdpi/dimens.xml
res/values-xxxhdpi/dimens.xml
然后將算好的百分比尺寸填寫到這幾個(gè)文件中拇泛,在項(xiàng)目布局文件中直接引用就可以了滨巴。
注意,這里411dp的尺寸文件是原始的設(shè)計(jì)稿尺寸俺叭,其他xdpi恭取、xxdpi、xxxdpi都是加了百分比的尺寸熄守。
百分比尺寸文件手寫是不可能的蜈垮,因此我們借助一些其他手段自動(dòng)生成耗跛。
3.新項(xiàng)目dimens文件生成
如果我們的項(xiàng)目剛開始,那么最好先準(zhǔn)備好這些文件攒发。
生成工具:dimens文件生成腳本
下面是我生成的三種dpi的dimens.xml文件:
4.老項(xiàng)目dimens文件生成和修改
如果你像我一樣項(xiàng)目寫到一半發(fā)現(xiàn)有適配的問題调塌,那么可以用下面的解決方法。下面可以用到上一步生成的固定dp和sp尺寸文件惠猿。
沒有適配的項(xiàng)目存在一個(gè)默認(rèn)的res/value/dimens.xml文件羔砾,這里的尺寸是默認(rèn)尺寸,我的項(xiàng)目默認(rèn)尺寸是針對(duì)xxxhdpi屏幕的偶妖。還缺少xxhdpi和xhdpi屏幕的dimens.xml文件姜凄,xxxhdpi的dimens.xml文件最好也創(chuàng)建一份。
可能想到方法是將現(xiàn)有的dimens文件copy兩份趾访,然后修改里面的值為對(duì)應(yīng)的百分比屏幕尺寸檀葛。如果現(xiàn)有的dimens文件內(nèi)容非常多,那么這個(gè)工作量將會(huì)非常大腹缩。下面我們編寫腳本自動(dòng)完成這個(gè)過程:
上面三個(gè)transfer文件是根據(jù)values/dimens.xml文件生成屿聋。
自動(dòng)化腳本:
https://github.com/xionghaoo/Android-screen-adaptation/blob/master/dimen_transfer.py
dimen文件的問題解決了,我們已經(jīng)寫在布局文件中的dp和sp怎么辦藏鹊?下面給出自動(dòng)替換腳本:
自動(dòng)化腳本:
https://github.com/xionghaoo/Android-screen-adaptation/blob/master/dimen_modify.py
例如:腳本自動(dòng)將layout文件夾下面的xml文件中的10dp替換成@dimen/x10dp
注意:程序是根據(jù)字符串搜索替換润讥,如果替換出現(xiàn)問題將無法恢復(fù),建議在替換之前先做好備份
上述腳本會(huì)自動(dòng)替換已經(jīng)寫好數(shù)值的dp和sp值:
但是TextView的默認(rèn)字體大小是14sp盘寡,也是沒有經(jīng)過適配的楚殿,上述腳本也不會(huì)自動(dòng)替換。這時(shí)候我們可以在style文件的基礎(chǔ)主題里面修改竿痰。
到此為止脆粥,屏幕適配就完成了。另外還有在代碼中寫死的尺寸就需要自己手動(dòng)修改了影涉,不過我相信這種尺寸不會(huì)太多变隔。
其他
Android的適配以我目前的認(rèn)知來看并不能做到覆蓋所有機(jī)型,只能根據(jù)自身的需求適配主流機(jī)型或一些特定的機(jī)型蟹倾。
另外有一種修改系統(tǒng)固有縮放因子的方案匣缘,這種方案會(huì)修改系統(tǒng)組件的UI樣式,如果有用到系統(tǒng)控件的項(xiàng)目不推薦使用鲜棠。