碼動未來教案篇
隨著移動設(shè)備的普及,移動web在前端工程師們的工作中占有越來越重要的位置蝌麸。移動設(shè)備更新速度頻繁点寥,手機(jī)廠商繁多艾疟,導(dǎo)致的問題是每一臺機(jī)器的屏幕寬度和分辨率不一樣。這給我們在編寫前端界面時增加了困難,適配問題在當(dāng)下顯得越來越突出蔽莱。記得剛剛開始開發(fā)移動端產(chǎn)品的時候向設(shè)計(jì)MM要了不同屏幕的設(shè)計(jì)圖弟疆,結(jié)果可想而知。本篇博文分享一些鹵煮處理多屏幕自適應(yīng)的經(jīng)驗(yàn)盗冷,希望有益于諸君怠苔。
特別說明:在開始這一切之前,請開發(fā)移動界面的工程師們在頭部加上下面這條meta:
<meta?name="viewport" content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=0">
簡單事情簡單做-寬度自適應(yīng)
所謂寬度自適應(yīng)嚴(yán)格來說是一種PC端的自適應(yīng)布局方式在移動端的延伸仪糖。在處理PC端的前端界面時候需要用到全屏布局時采用的就是此種布局方式柑司。它的實(shí)現(xiàn)方式也比較簡單,將外層容器元素按照百分比鋪滿地方式锅劝,里面的子元素固定或者左右浮動攒驰。?
.div {??width:100%;?height:100px;}.child {??float:?left;?}.child {??float:right;}
由于父級元素采用百分比的布局方式,隨著屏幕的拉伸故爵,它的寬度會無限的拉伸玻粪。而子元素由于采用了浮動,那么它們的位置也會固定在兩端诬垂。該寬度自適應(yīng)在新的時代有了新的方法劲室,隨著彈性布局的普及,它經(jīng)常被flex或者box這樣的伸縮性布局方式替代结窘,變得越來越“彈性”十足很洋。需要了解彈性布局,請前往Flex布局教程和鹵煮box布局教程比較晦鞋。
大小之辨-完全自適應(yīng)
“完全自適應(yīng)式”是小編對越此方案的叫法蹲缠,由于小編現(xiàn)在找不到官方名稱,所以暫時就這樣叫它悠垛。這種解決方案相對前一種來說進(jìn)步不少线定,不僅僅寬度實(shí)現(xiàn)了自適應(yīng),而且界面所有的元素大小和高度都會根據(jù)不同分辨率和屏幕寬度的設(shè)備來調(diào)整元素确买、字體斤讥、圖片、高度等屬性的值湾趾。簡單來說就是在不同的屏幕下芭商,你看到的字體和元素高寬度的大小是不一樣的。在這里搀缠,有人就會說利用的是媒體查詢屬性铛楣,根據(jù)不同的屏幕寬度,調(diào)整樣式艺普。小編之前也是這樣想的簸州,但是你需要考慮到界面上的許多元素需要設(shè)置字體鉴竭,如果用media query為每個元素在不同的設(shè)備下都設(shè)置不同的屬性的話,那么有多少種屏幕我們的css就會增加多少倍岸浑。實(shí)際上在這里搏存,我們采用的是js和css屬性rem來解決這個問題的。
REM屬性指的是相對于根元素設(shè)置某個元素的字體大小矢洲。它同時也可以用作為設(shè)置高度等一系列可以用px來標(biāo)注的單位璧眠。
html {
?font-size: 10px;
}
div {
?font-size: 1rem;
?height: 2rem;
?width: 3rem;
?border: .1rem solid #000;
}
采用以上寫法,div繼承到了html節(jié)點(diǎn)的font-size读虏,為本身定義了一系列樣式屬性责静,此時1em計(jì)算為10px,即根節(jié)點(diǎn)的font-size值盖桥。所以泰演,這時div的高度就是20px,寬度是30px葱轩,邊框是1px睦焕,字體大小則是10px;一旦有了這樣的方法靴拱,我們自然可以根據(jù)不同的屏幕寬度設(shè)置不同的根節(jié)點(diǎn)字體大小垃喊。假設(shè)我們現(xiàn)在設(shè)計(jì)的標(biāo)準(zhǔn)是iphone5s,iphone5系列的屏幕分辨率是640袜炕。為了統(tǒng)一規(guī)范本谜,我們將iphone5 分辨率下的根元素font-size設(shè)置為100px;
<!--iphone5-->html {?font-size:?100px;}
那么以此為基準(zhǔn),可以計(jì)算出一個比例值6.4偎窘。我們可以得知其他手機(jī)分辨率的設(shè)備下根元素字體大小:
/*數(shù)據(jù)計(jì)算公式
640/100 = device-width / x? 可以設(shè)置其他設(shè)備根元素字體大小
iphone5: 640? : 100
iphone6: 750 : 117
iphone6s: 1240 : 194*/
var?deviceWidth = window.documentElement.clientWidth;document.documentElement.style.fontSize = (deviceWidth / 6.4) +?'px';
在head中乌助,我們將以上代碼加入,動態(tài)地改變根節(jié)點(diǎn)的font-size值陌知,得到如下結(jié)果:
接下來我們可以根據(jù)根元素的字體大小用rem設(shè)置各種屬性的相對值他托。當(dāng)然,如果是移動設(shè)備仆葡,屏幕會有一個上下限制赏参,我們可以控制分辨率在某個范圍內(nèi),超過了該范圍沿盅,我們就不再增加根元素的字體大小了:
var?deviceWidth = document.documentElement.clientWidth > 1300 ? 1300 : document.documentElement.clientWidth;
document.documentElement.style.fontSize = (deviceWidth / 6.4) +?'px';
一般的情況下把篓,你是不需要考慮屏幕動態(tài)地拉伸和收縮。當(dāng)然腰涧,假如用戶開啟了轉(zhuǎn)屏設(shè)置韧掩,在網(wǎng)頁加載之后改變了屏幕的寬度,那么我們就要考慮這個問題了窖铡。解決此問題也很簡單疗锐,監(jiān)聽屏幕的變化就可以做到動態(tài)切換元素樣式:
window.onresize =?function(){
??????var?deviceWidth = document.documentElement.clientWidth > 1300 ? 1300 : document.documentElement.clientWidth;
??????document.documentElement.style.fontSize = (deviceWidth / 6.4) +?'px';
?};
為了提高性能郎楼,讓代碼開起來更加完美,可以為它加上節(jié)流閥函數(shù):
window.onresize = _.debounce(function() {
??????var?deviceWidth = document.documentElement.clientWidth > 1300 ? 1300 : document.documentElement.clientWidth;
??????document.documentElement.style.fontSize = (deviceWidth / 6.4) +?'px';
}, 50);
順帶解決高保真標(biāo)注與實(shí)際開發(fā)值比例問題
如果你們設(shè)計(jì)稿標(biāo)準(zhǔn)是iphone5窒悔,那么拿到設(shè)計(jì)稿的時候一定會發(fā)現(xiàn),完全不能按照高保真上的標(biāo)注來寫css敌买,而是將各個值取半简珠,這是因?yàn)橐苿釉O(shè)備分辨率不一樣。設(shè)計(jì)師們是在真實(shí)的iphone5機(jī)器上做的標(biāo)注虹钮,而iphone5系列的分辨率是640聋庵,實(shí)際上我們在開發(fā)只需要按照320的標(biāo)準(zhǔn)來。為了節(jié)省時間芙粱,不至于每次都需要將標(biāo)注取半祭玉,我們可以將整個網(wǎng)頁縮放比例,模擬提高分辨率春畔。這個做法很簡單脱货,為不同的設(shè)備設(shè)置不同的meta即可:
var scale = 1 / devicePixelRatio;
document.querySelector('meta[name="viewport"]').setAttribute('content', 'initial-scale=' + scale + ', maximum-scale=' + scale + ', minimum-scale=' + scale + ', user-scalable=no');
這樣設(shè)置同樣可以解決在安卓機(jī)器下1px像素看起來過粗的問題,因?yàn)樵谙袼貫?px的安卓下機(jī)器下律姨,邊框的1px被壓縮成了0.5px了振峻。總之是一勞永逸择份!淘寶和網(wǎng)易新聞的手機(jī)web端就是采用以上這種方式扣孟,自適應(yīng)各種設(shè)備屏幕的,大家有興趣可以去參考參考荣赶。下面是完整的代碼:
<!DOCTYPE?html>
<html>
<head> ?
<title>測試</title>?
<meta?name="viewport" content="width=device-width,user-scalable=no,maximum-scale=1" />??
<script?type="text/javascript">
(function() {??/
/ deicePixelRatio :設(shè)備像素??
var scale = 1 / devicePixelRatio;??//設(shè)置meta 壓縮界面 模擬設(shè)備的高分辨率?
?document.querySelector('meta[name="viewport"]').setAttribute('content', 'initial-scale=' + scale + ', maximum-scale=' + scale + ', minimum-scale=' + scale + ', user-scalable=no');??//debounce 為節(jié)流函數(shù)凤价,自己實(shí)現(xiàn)“未矗或者引入underscoure即可利诺。?
?var reSize = _.debounce(function() {??????
????var deviceWidth = document.documentElement.clientWidth > 1300 ? 1300 : document.documentElement.clientWidth;??????//按照640像素下字體為100px的標(biāo)準(zhǔn)來,得到一個字體縮放比例值 6.4??????
????document.documentElement.style.fontSize = (deviceWidth / 6.4) + 'px';??}, 50);???window.onresize = reSize;})();??
</script>??
<style?type="text/css">????
html {??????height: 100%;??????width: 100%;??????overflow: hidden;??????font-size: 16px;????}?????div {??????height: 0.5rem;??????widows: 0.5rem;??????border: 0.01rem solid #19a39e;????}?????........??
</style>??<body>????<div>????</div>??</body></html>
讓元素飛起來-媒體查詢
運(yùn)用css新屬性media query 特性也可以實(shí)現(xiàn)我們上說到過的布局樣式剩燥。為尺寸設(shè)置根元素字體大小:
@media screen and (device-width: 640px) {?/*iphone4/iphon5*/
??????html {
????????font-size: 100px;
??????}
????}
@media screen and (device-width: 750px) {?/*iphone6*/
??????html {
????????font-size: 117.188px;
??????}
????}
@media screen and (device-width: 1240px) {?/*iphone6s*/??????html {
????????font-size: 194.063px;
??????}
????}
這種方式也是可行的立轧,缺點(diǎn)是靈活性不高,取每個設(shè)備的精確值需要自己去計(jì)算躏吊,所以只能取范圍值氛改。考慮設(shè)備屏幕眾多比伏,分辨率也參差不齊胜卤,把每一種機(jī)型的css代碼寫出來是不太可能的。但是它也有優(yōu)點(diǎn)赁项,就是無需監(jiān)聽瀏覽器的窗口變化葛躏,它會跟隨屏幕動態(tài)變化澈段。媒體查詢的用法當(dāng)然不僅僅像在此處這么簡單,相對于第二種自適應(yīng)來說有很多地方是前者所遠(yuǎn)遠(yuǎn)不及的舰攒。最明顯的就是它可以根據(jù)不同設(shè)備顯示不同的布局樣式败富!請注意,這里已經(jīng)不是改變字體和高度那么簡單了摩窃,它直接改變的是布局樣式兽叮!
@media screen and (min-width: 320px) and (max-width: 650px) { /*手機(jī)*/
? .class {
? ? float: left;
? }
}
@media screen and (min-width: 650px) and (max-width: 980px) { /*pad*/
? .class {
? ? float: right;
? }
}
@media screen and (min-width: 980px) ?and (max-width: 1240px) { /*pc*/
? .class {
? ? float: clear;
? }
}
此種自適應(yīng)布局一般常用在兼容PC和手機(jī)設(shè)備,由于屏幕跨度很大猾愿,界面的元素以及遠(yuǎn)遠(yuǎn)不是改改大小所能滿足的鹦聪。這時候需要重新設(shè)計(jì)整界面的布局和排版了:
如果屏幕寬度大于1300像素
如果屏幕寬度在600像素到1300像素之間,則6張圖片分成兩行蒂秘。
許多css框架經(jīng)常用到這樣的多端解決方案泽本,著名的bootstrap就是采用此種方式進(jìn)行柵格布局的。
不管哪一種自適應(yīng)方式姻僧,我們的目的是使得開發(fā)網(wǎng)頁在各種屏幕下變得好看:如果你的項(xiàng)目定位的用戶群僅僅是使用某種機(jī)型的人规丽,那么可以采用第一種自適應(yīng)方式。如果你的客戶主要是移動端撇贺,但是客戶的設(shè)備類型龐雜嘁捷,建議采用第二種方式。如果你雄心勃勃地需要建立一套兼容PC显熏、PAD雄嚣、mobile多端的一體化web應(yīng)用,那么第三種選擇顯然是最適合你的喘蟆。每種方式都有自己的利弊缓升,根據(jù)需求權(quán)衡利害,合理地實(shí)現(xiàn)自適應(yīng)布局蕴轨,需要不停的實(shí)踐和摸索港谊。碼動未來歡迎各位莘莘學(xué)子的光臨,更多內(nèi)容請?zhí)砑游⑿牛簃adongweikai橙弱,或掃描下方二維碼添加