前端是一個(gè)讓人又愛又恨的職業(yè),愛,是因?yàn)榧夹g(shù)更新快,發(fā)展道路寬闊;恨的是需要學(xué)習(xí)、掌握的東西太多太多,永遠(yuǎn)學(xué)不完,正如俗話說:活到老,學(xué)到老;但更讓人頭疼的是,還要面臨各種適配、兼容性問題。
網(wǎng)上對(duì)適配、兼容性問題都有相關(guān)解答,但一直以來都是頭痛醫(yī)頭腳痛醫(yī)腳,沒有進(jìn)行系統(tǒng)的梳理,整個(gè)思路和方向全是混亂的,所以最近整理了一份css瀏覽器兼容性的常見解決思路和方案,分享給大家,一起進(jìn)步。
瀏覽器兼容性問題原因
其實(shí)就是一句話,瀏覽器廠商太多。關(guān)鍵是不同廠商,甚至同一廠商不同版本,對(duì)同一段CSS的解析效果也不一致,這就導(dǎo)致了頁(yè)面顯示效果不統(tǒng)一,也就帶來了兼容性問題。
各瀏覽器市場(chǎng)表現(xiàn)
瀏覽器種類這么多,不可能每一個(gè)都要去兼容,所以對(duì)于用戶量一般的產(chǎn)品,把主流瀏覽器的適配做好,就已經(jīng)很不錯(cuò)啦。
根據(jù)世界市場(chǎng)權(quán)威調(diào)查機(jī)構(gòu)NetMarketShare公布的2018年10月各瀏覽器市場(chǎng)占有率,可以看出Chrome的占有率達(dá)到了66.43%。
但根據(jù)百度流量研究院提供的2018年11月至2019年1月的數(shù)據(jù)可以看出,IE系仍然占有很大比重,兼容處理工作還需繼續(xù)。
解決思路和方案(重點(diǎn))
這里我們不會(huì)去關(guān)注太多細(xì)節(jié)問題, 比如哪個(gè)css樣式需要我們?nèi)ゼ嫒莸?,主要?個(gè)方面討論一下大的解決思路,分別是:瀏覽器CSS樣式初始化、瀏覽器私有屬性,CSS hack語(yǔ)法和自動(dòng)化插件。
- CSS初始化
前端的小伙伴一定遇到過因?yàn)槟J(rèn)樣式導(dǎo)致的錯(cuò)亂問題,而且每個(gè)瀏覽器的css默認(rèn)樣式不盡相同,所以最簡(jiǎn)單有效的方式就是對(duì)其進(jìn)行初始化(覆蓋默認(rèn)樣式)。相信很多朋友也都寫過這樣的代碼,在所有CSS開始前,先把marin和padding都設(shè)為0。
*{ margin: 0; padding: 0; }
關(guān)于瀏覽器CSS樣式初始化,經(jīng)驗(yàn)不豐富的話,尤其對(duì)于剛?cè)腴T的小白,可能也不知道該初始化什么,這里推薦一個(gè)庫(kù)給大家,Normalize.css,github star數(shù)量接近4萬(wàn),自行選取展示其中幾個(gè)樣式設(shè)置,如下:
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樣式初始化,已經(jīng)解決了一大部分因?yàn)闉g覽器默認(rèn)樣式導(dǎo)致的常規(guī)兼容性問題。接下來再看看瀏覽器的私有屬性。
- 瀏覽器私有屬性
-webkit- ,-moz- ,-ms-等,這是我們經(jīng)常在某個(gè)CSS屬性前添加的一些前綴,這些就是瀏覽器的私有屬性。
說到私有屬性的出現(xiàn)也是因?yàn)橹贫℉TML和CSS標(biāo)準(zhǔn)的組織W3C動(dòng)作很慢。
通常,有W3C組織成員提出一個(gè)新屬性,比如圓角border-radius,大家都覺得好,但W3C制定標(biāo)準(zhǔn),要走很復(fù)雜的程序。而瀏覽器商市場(chǎng)推廣時(shí)間緊,如果一個(gè)屬性已經(jīng)夠成熟了,就會(huì)在瀏覽器中加入支持。為避免日后W3C公布標(biāo)準(zhǔn)時(shí)有所變化,所以加入一個(gè)私有前綴,比如-webkit-border-radius,常用的前綴有:
- -moz代表firefox瀏覽器私有屬性
- -ms代表IE瀏覽器私有屬性
- -webkit代表chrome、safari私有屬性
- -o代表opera私有屬性
對(duì)于書寫順序一定要注意,兼容性寫法放到前面,把標(biāo)準(zhǔn)寫法放到最后
-webkit-transform:rotate(-3deg); /*為Chrome/Safari*/ -moz-transform:rotate(-3deg); /*為Firefox*/ -ms-transform:rotate(-3deg); /*為IE*/ -o-transform:rotate(-3deg); /*為Opera*/ transform:rotate(-3deg);
大家想一下,如果每個(gè)CSS屬性寫這么一堆兼容性代碼,那無(wú)疑是對(duì)生命折磨,到后面就會(huì)講如何通過自動(dòng)化插件來處理。
- CSS hack
除了以上的默認(rèn)樣式覆蓋及私有屬性添加,有時(shí)我們還需要針對(duì)不同的瀏覽器甚至不同版本編寫特定的CSS樣式,這一過程就叫做CSS hack!
CSS hack的寫法大致可以歸納為以下幾種:條件hack、屬性級(jí)hack、選擇符級(jí)hack。
條件hack:主要針對(duì)IE瀏覽器進(jìn)行一些特殊的設(shè)置
<!--[if <keywords>? IE <version>?]> 代碼塊,可以是html,css,js <![endif]-->
- 取值
關(guān)鍵詞
if后面跟的條件共包含6種選擇方式:是否、大于、大于或等于、小于、小于或等于、非指定版本
是否:指定是否IE或IE某個(gè)版本。關(guān)鍵字:空
大于:選擇大于指定版本的IE版本。關(guān)鍵字:gt(greater than)
大于或等于:選擇大于或等于指定版本的IE版本。關(guān)鍵字:gte(greater than or equal)
小于:選擇小于指定版本的IE版本。關(guān)鍵字:lt(less than)
小于或等于:選擇小于或等于指定版本的IE版本。關(guān)鍵字:lte(less than or equal)
非指定版本:選擇除指定版本外的所有IE版本。關(guān)鍵字:!
版本
IE瀏覽器版本,如6、7、8,但I(xiàn)E10及以上版本已將條件注釋特性移除,使用時(shí)需注意。
- 舉個(gè)例子
<!--[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樣式屬性名前加上一些只有特定瀏覽器才能識(shí)別的hack前綴。
selector{<hack>?property:value<hack>?;}
- 取值:
_:選擇IE6及以下。連接線(中劃線)(-)亦可使用,為了避免與某些帶中劃線的屬性混淆,所以使用下劃線(_)更為合適
*:選擇IE7及以下。諸如:(+)與(#)之類的均可使用,不過業(yè)界對(duì)(*)的認(rèn)知度更高
9:選擇IE6+
:選擇IE8+和Opera15以下的瀏覽器
- 舉個(gè)例子
注意順序:低版本的兼容性寫法放到最后
.test { color: #0909; /* For IE8+ */ *color: #f00; /* For IE7 and earlier */ _color: #ff0; /* For IE6 and earlier */ }
選擇符級(jí)hack:是針對(duì)一些頁(yè)面表現(xiàn)不一致或者需要特殊對(duì)待的瀏覽器,在CSS選擇器前加上一些只有某些特定瀏覽器才能識(shí)別的前綴進(jìn)行hack。
<hack> selector{ sRules }
- 取值: 常見的選擇符級(jí)hack有
*html *前綴只對(duì)IE6生效 *+html *+前綴只對(duì)IE7生效 @media screen9{...}只對(duì)IE6/7生效 @media screen {body { background: red; }}只對(duì)IE8有效 @media screen,screen9{body { background: blue; }}只對(duì)IE6/7/8有效 @media screen {body { background: green; }} 只對(duì)IE8/9/10有效 @media screen and (min-width:0) {body { background: gray; }} 只對(duì)IE9/10有效 @media screen and (-ms-high-contrast: active), (-ms-high-contrast: none) {body { background: orange; }} 只對(duì)IE10有效
- 舉個(gè)例子:
* html .test { color: #090; } /* For IE6 and earlier */ * + html .test { color: #ff0; } /* For IE7 */
寫到這里,不得不說,前端開發(fā)這也太難了吧~~
花大力氣解決這些兼容性問題,并不能給我們技術(shù)上帶來什么大的提升,無(wú)非是給各個(gè)瀏覽器廠商填坑罷了。隨著時(shí)間的流逝,技術(shù)更新,到現(xiàn)在我們就該想怎么花最小的力氣解決css兼容性問題。接下來,自動(dòng)化插件可以上場(chǎng)了,我們終于可以從繁重的兼容性處理中解脫出來。
- 自動(dòng)化插件
Autoprefixer是一款自動(dòng)管理瀏覽器前綴的插件,它可以解析CSS文件并且添加瀏覽器前綴到CSS內(nèi)容里。
把Autoprefixe添加到資源構(gòu)建工具(如webpack)后,可以完全忘記前面的東西,只需按照最新的W3C規(guī)范來正常書寫CSS,剩下的工作交給插件來處理。另外,如果項(xiàng)目需要支持舊版瀏覽器,可修改browsers參數(shù)設(shè)置。
//我們編寫的代碼 div { transform: rotate(30deg); } //自動(dòng)補(bǔ)全的代碼,具體補(bǔ)全哪些由要兼容的瀏覽器版本決定,可以自行設(shè)置div { -ms-transform: rotate(30deg); -webkit-transform: rotate(30deg); -o-transform: rotate(30deg); -moz-transform: rotate(30deg); transform: rotate(30deg); }
目前webpack、gulp、grunt都有相應(yīng)的插件,趕快行動(dòng)起來,別再讓CSS兼容性浪費(fèi)你的時(shí)間!
受限于技術(shù)能力,如有問題可以在下方留言討論。想要了解更多前端技術(shù)、精彩熱文可關(guān)注同名公眾號(hào)“一郭鮮”。別著急,慢慢來,小郭與你一起成長(zhǎng)