前端是一個苦逼的職業,不只由於技術更新快,並且要會的東西實在太多了(若是全寫出來,那真是一籮筐),更讓人頭疼的是,還要面臨各類適配、兼容性問題。css
一直以來都是頭痛醫頭腳痛醫腳,沒有進行系統的梳理,整個思路和方向全是混亂的,因此很是想把最近整理的,CSS瀏覽器兼容性的常看法決思路和方案,分享給你們,一塊兒進步。html
還不是由於瀏覽器廠商太多了!前端
Chrome,Frirefox,Safari,Edge,IE6,IE7,IE8,IE9...360安全瀏覽器,qq瀏覽器,世界之窗,TT,搜狗,opera,maxthon(傲遊)……html5
關鍵是不一樣廠商,甚至同一廠商不一樣版本,對同一段CSS的解析效果也不一致,這就致使了頁面顯示效果不統一,也就帶來了兼容性問題。webpack
多麼但願Chrome可以一統江湖啊~~git
瀏覽器這麼多,咱們也不可能每個都要去兼容,對於用戶量通常的產品,把主流瀏覽器的適配作好,就已經很不錯啦。github
根據世界市場權威調查機構NetMarketShare公佈的2018年10月各瀏覽器市場佔有率,能夠看出Chrome的佔有率達到了66.43%,這絕對是一個振奮人心的好消息。web
而根據百度流量研究院提供的2018年11月至2019年1月的數據能夠看出,IE系仍然佔有很大比重,任重而道遠啊~chrome
今天,不想去關注太多細節問題, 好比那個css樣式須要咱們去兼容,而是想討論一下大的解決思路,主要包括4個方面,瀏覽器CSS樣式初始化、瀏覽器私有屬性,CSS hack語法和自動化插件。gulp
因爲每一個瀏覽器的css默認樣式不盡相同,因此最簡單有效的方式就是對其進行初始化,相信不少朋友都寫過這樣的代碼,在全部CSS開始前,先把marin和padding都設爲0,以防不一樣瀏覽器的顯示效果不同。
*{
margin: 0;
padding: 0;
}
複製代碼
關於瀏覽器CSS樣式初始化,經驗不豐富的話,可能也不知道該初始化什麼,這裏給你們推薦一個庫,Normalize.css,github star數量接近4萬,選取展現其中幾個樣式設置,以下
html {
line-height: 1.15; /* Correct the line height in all browsers */
-webkit-text-size-adjust: 100%; /* Prevent adjustments of font size after orientation changes in iOS. */
}
body {
margin: 0;
}
a {
background-color: transparent; /* Remove the gray background on active links in IE 10. */
}
img {
border-style: none; /* Remove the border on images inside links in IE 10. */
}
複製代碼
經過CSS樣式初始化,相信能解決很多常規的兼容性問題,接下來再看看瀏覽器的私有屬性。
咱們常常會在某個CSS的屬性前添加一些前綴,好比-webkit- ,-moz- ,-ms-,這些就是瀏覽器的私有屬性。
爲何會出現私有屬性呢?這是由於制定HTML和CSS標準的組織W3C動做是很慢的。
一般,有W3C組織成員提出一個新屬性,好比說圓角border-radius,你們都以爲好,但W3C制定標準,要走很複雜的程序,審查等。而瀏覽器商市場推廣時間緊,若是一個屬性已經夠成熟了,就會在瀏覽器中加入支持。
可是爲避免往後W3C公佈標準時有所變動,會加入一個私有前綴,好比-webkit-border-radius,經過這種方式來提早支持新屬性。等到往後W3C公佈了標準,border-radius的標準寫法確立以後,再讓新版的瀏覽器支持border-radius這種寫法。經常使用的前綴有:
對於私有屬性的順序要注意,把標準寫法放到最後,兼容性寫法放到前面
-webkit-transform:rotate(-3deg); /*爲Chrome/Safari*/
-moz-transform:rotate(-3deg); /*爲Firefox*/
-ms-transform:rotate(-3deg); /*爲IE*/
-o-transform:rotate(-3deg); /*爲Opera*/
transform:rotate(-3deg);
複製代碼
每一個CSS屬性寫這麼一堆兼容性代碼,無疑是對生命最大的浪費,後面咱們會講一下經過自動化插件來處理這塊。
有時咱們須要針對不一樣的瀏覽器或不一樣版本寫特定的CSS樣式,這種針對不一樣的瀏覽器/不一樣版本寫相應的CSS code的過程,叫作CSS hack!
CSS hack的寫法大體概括爲3種:條件hack、屬性級hack、選擇符級hack。
條件hack主要針對IE瀏覽器進行一些特殊的設置
<!--[if <keywords>? IE <version>?]>
代碼塊,能夠是html,css,js
<![endif]-->
複製代碼
keywords
if後面跟的條件共包含6種選擇方式:是否、大於、大於或等於、小於、小於或等於、非指定版本
是否:指定是否IE或IE某個版本。關鍵字:空
大於:選擇大於指定版本的IE版本。關鍵字:gt(greater than)
大於或等於:選擇大於或等於指定版本的IE版本。關鍵字:gte(greater than or equal)
小於:選擇小於指定版本的IE版本。關鍵字:lt(less than)
小於或等於:選擇小於或等於指定版本的IE版本。關鍵字:lte(less than or equal)
非指定版本:選擇除指定版本外的全部IE版本。關鍵字:!
version
IE瀏覽器版本,如六、七、8
IE10及以上版本已將條件註釋特性移除,使用時需注意。
<!--[if IE]>
<p>你在非IE中將看不到個人身影</p>
<![endif]-->
<!--[if IE]>
<style>
.test{color:red;}
</style>
<![endif]-->
<!--[if lt IE 9]>
<script src="//cdn.bootcss.com/html5shiv/3.7.2/html5shiv.min.js"></script>
<script src="//cdn.bootcss.com/respond.js/1.4.2/respond.min.js"></script>
<![endif]-->
複製代碼
屬性hack就是在CSS樣式屬性名前加上一些只有特定瀏覽器才能識別的hack前綴。
selector{<hack>?property:value<hack>?;}
複製代碼
_:選擇IE6及如下。鏈接線(中劃線)(-)亦可以使用,爲了不與某些帶中劃線的屬性混淆,因此使用下劃線(_)更爲合適。
*:選擇IE7及如下。諸如:(+)與(#)之類的都可使用,不過業界對(*)的認知度更高
\9:選擇IE6+
\0:選擇IE8+和Opera15如下的瀏覽器
如在不一樣的IE瀏覽器中設置不一樣的顏色,注意順序:低版本的兼容性寫法放到最後
.test {
color: #090\9; /* For IE8+ */
*color: #f00; /* For IE7 and earlier */
_color: #ff0; /* For IE6 and earlier */
}
複製代碼
選擇符級hack是針對一些頁面表現不一致或者須要特殊對待的瀏覽器,在CSS選擇器前加上一些只有某些特定瀏覽器才能識別的前綴進行hack。
<hack> selector{ sRules }
複製代碼
*html *前綴只對IE6生效
*+html *+前綴只對IE7生效
@media screen\9{...}只對IE6/7生效
@media \0screen {body { background: red; }}只對IE8有效
@media \0screen\,screen\9{body { background: blue; }}只對IE6/7/8有效
@media screen\0 {body { background: green; }} 只對IE8/9/10有效
@media screen and (min-width:0\0) {body { background: gray; }} 只對IE9/10有效
@media screen and (-ms-high-contrast: active), (-ms-high-contrast: none) {body { background: orange; }} 只對IE10有效
複製代碼
* html .test { color: #090; } /* For IE6 and earlier */
* + html .test { color: #ff0; } /* For IE7 */
複製代碼
看到這裏,我不得不爲前端人員自豪,這也太難了吧~~
不過花大力氣解決這些兼容性問題, 並不能給咱們技術上帶來什麼大的提高,無非是給各個瀏覽器廠商填坑罷了,隨着時間的流逝,這些技術的價值也會愈來愈小,怎麼花最小的力氣解決css兼容性問題,讓咱們把更多的時間留給美好的生活,纔是關鍵,好在有一些自動化插件能夠幫咱們從繁重的兼容性處理中解脫處理。
Autoprefixer是一款自動管理瀏覽器前綴的插件,它能夠解析CSS文件而且添加瀏覽器前綴到CSS內容裏,使用Can I Use(caniuse網站)的數據來決定哪些前綴是須要的。
把Autoprefixe添加到資源構建工具(例如Grunt)後,能夠徹底忘記有關CSS前綴的東西,只需按照最新的W3C規範來正常書寫CSS便可。若是項目須要支持舊版瀏覽器,可修改browsers參數設置 。
//咱們編寫的代碼
div {
transform: rotate(30deg);
}
//自動補全的代碼,具體補全哪些由要兼容的瀏覽器版本決定,能夠自行設置div {
-ms-transform: rotate(30deg);
-webkit-transform: rotate(30deg);
-o-transform: rotate(30deg);
-moz-transform: rotate(30deg);
transform: rotate(30deg);
}
複製代碼
目前webpack、gulp、grunt都有相應的插件,若是尚未使用,那就趕忙應用到咱們的項目中吧,別再讓CSS兼容性浪費你的時間!
限於技術水平,文中確定有不少遺漏或者錯誤,你們還有什麼好的方法能夠一塊兒討論,歡迎關注咱們的公衆號,一塊兒學習,一塊兒進步。