Android屏幕適配方案(UI稿為iPhone8 plus)

一筷畦、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ù)


image.png

根據(jù)商家給出的尺寸計(jì)算dpi = Sqrt(1080^2 + 2160^2)/5.93 = 407.243

在真機(jī)上將DisplayMetrics對(duì)象打印出來可以看到


image.png

屏幕的固有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文件:


image.png

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è)過程:


image.png

上面三個(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)替換腳本:

image.png

自動(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值:


image.png

但是TextView的默認(rèn)字體大小是14sp盘寡,也是沒有經(jīng)過適配的楚殿,上述腳本也不會(huì)自動(dòng)替換。這時(shí)候我們可以在style文件的基礎(chǔ)主題里面修改竿痰。


image.png

到此為止脆粥,屏幕適配就完成了。另外還有在代碼中寫死的尺寸就需要自己手動(dòng)修改了影涉,不過我相信這種尺寸不會(huì)太多变隔。
寫死的尺寸.png

替換后的適配尺寸.png

其他

Android的適配以我目前的認(rèn)知來看并不能做到覆蓋所有機(jī)型,只能根據(jù)自身的需求適配主流機(jī)型或一些特定的機(jī)型蟹倾。

另外有一種修改系統(tǒng)固有縮放因子的方案匣缘,這種方案會(huì)修改系統(tǒng)組件的UI樣式,如果有用到系統(tǒng)控件的項(xiàng)目不推薦使用鲜棠。

屏幕適配github

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末肌厨,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子豁陆,更是在濱河造成了極大的恐慌柑爸,老刑警劉巖,帶你破解...
    沈念sama閱讀 211,817評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件盒音,死亡現(xiàn)場離奇詭異表鳍,居然都是意外死亡馅而,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,329評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門进胯,熙熙樓的掌柜王于貴愁眉苦臉地迎上來用爪,“玉大人,你說我怎么就攤上這事胁镐≠搜” “怎么了?”我有些...
    開封第一講書人閱讀 157,354評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵盯漂,是天一觀的道長颇玷。 經(jīng)常有香客問我,道長就缆,這世上最難降的妖魔是什么帖渠? 我笑而不...
    開封第一講書人閱讀 56,498評(píng)論 1 284
  • 正文 為了忘掉前任,我火速辦了婚禮竭宰,結(jié)果婚禮上空郊,老公的妹妹穿的比我還像新娘。我一直安慰自己切揭,他們只是感情好狞甚,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,600評(píng)論 6 386
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著廓旬,像睡著了一般哼审。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上孕豹,一...
    開封第一講書人閱讀 49,829評(píng)論 1 290
  • 那天涩盾,我揣著相機(jī)與錄音,去河邊找鬼励背。 笑死春霍,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的椅野。 我是一名探鬼主播终畅,決...
    沈念sama閱讀 38,979評(píng)論 3 408
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢(mèng)啊……” “哼竟闪!你這毒婦竟也來了?” 一聲冷哼從身側(cè)響起杖狼,我...
    開封第一講書人閱讀 37,722評(píng)論 0 266
  • 序言:老撾萬榮一對(duì)情侶失蹤,失蹤者是張志新(化名)和其女友劉穎蝶涩,沒想到半個(gè)月后理朋,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 44,189評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡兽愤,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,519評(píng)論 2 327
  • 正文 我和宋清朗相戀三年彼念,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片浅萧。...
    茶點(diǎn)故事閱讀 38,654評(píng)論 1 340
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡逐沙,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出洼畅,到底是詐尸還是另有隱情吩案,我是刑警寧澤,帶...
    沈念sama閱讀 34,329評(píng)論 4 330
  • 正文 年R本政府宣布帝簇,位于F島的核電站徘郭,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏丧肴。R本人自食惡果不足惜残揉,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 39,940評(píng)論 3 313
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望闪湾。 院中可真熱鬧冲甘,春花似錦、人聲如沸途样。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,762評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽何暇。三九已至陶夜,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間裆站,已是汗流浹背条辟。 一陣腳步聲響...
    開封第一講書人閱讀 31,993評(píng)論 1 266
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留宏胯,地道東北人羽嫡。 一個(gè)月前我還...
    沈念sama閱讀 46,382評(píng)論 2 360
  • 正文 我出身青樓,卻偏偏與公主長得像肩袍,于是被迫代替她去往敵國和親杭棵。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,543評(píng)論 2 349

推薦閱讀更多精彩內(nèi)容