咱們在作前端的時候,常常會遇到各類瀏覽器顯示不太一致的狀況,css hack就是讓你的css代碼兼容不一樣的瀏覽器。javascript
css hack大部分針對IE不一樣版本之間的表現差別而引入的,下面有幾個demo能夠把IE6-IE10和其餘的瀏覽器作一個兼容。css
一、那麼什麼是css hack呢?html
因爲各大瀏覽器來自不一樣的廠商(如Internet Explorer/Safari/ Mozilla Firefox/Chrome等),對css的支持、解釋都不太同樣,致使頁面效果在不一樣的瀏覽器環境中顯示不一致。前端
因此咱們就要針對不一樣的瀏覽器/不一樣版本寫出一些特定的css樣式啦,要否則寫出頁面顯示雜七雜八的,本身都會以爲不舒服。java
咱們把這個過程叫作css hack。css3
二、接下來咱們來看一下css hack的原理吧web
不一樣的瀏覽器和不一樣的瀏覽器版本對css的支持和解析不同,以及css優先級對瀏覽器展示效果的影響,咱們能夠根據這個對不一樣的瀏覽器情景來使用不一樣的css。瀏覽器
三、css hack分類:(3中表現形式)ide
(1)css屬性前綴法:(即內部hack)學習
(2)選擇器前綴法:(即選擇器hack)
(3)IE條件註釋法(即HTML頭部引用 if IE)(即HTML條件註釋hack):
針對除了IE10+之外全部的IE:<!--[if IE]>IE瀏覽器顯示的內容<![endif]-->;
針對IE6及如下的版本:<!--[if It IE 6]>只在IE6-顯示的內容<![endif]-->;
這類hack不只對css生效,對寫在判斷句裏面的全部代碼都會生效。
四、css hack 書寫順序
通常是將使用範圍廣、被識別能力強的css定義在後面。
五、css hack 方式一:條件註釋法
這種方式是專門歸IE瀏覽器全部的hack方式。
下面是幾個例子:
六、css hack 方式二:類內屬性前綴法
屬性前綴法:在css樣式樣式屬性名前加上一些只有特定瀏覽器才能識別的hack前綴。
IE瀏覽器各版本 CSS hack 對照表:
hack | 寫法 | 實例 | IE6(S) | IE6(Q) | IE7(S) | IE7(Q) | IE8(S) | IE8(Q) | IE9(S) | IE9(Q) | IE10(S) | IE10(Q) |
* | *color | 青色 | Y | Y | Y | Y | N | Y | N | Y | N | Y |
+ | +color | 綠色 | Y | Y | Y | Y | N | Y | N | Y | N | Y |
- | -color | 黃色 | Y | Y | N | N | N | N | N | N | N | N |
_ | _color | 藍色 | Y | Y | N | Y | N | Y | N | Y | N | N |
# | #color | 紫色 | Y | Y | Y | Y | N | Y | N | Y | N | Y |
\0 | color:red\0 | 紅色 | N | N | N | N | Y | N | Y | N | Y | N |
\9\0 | color:red\9\0 | 粉色 | N | N | N | N | N | N | Y | N | Y | N |
!important | color:blue !important;color:green; | 棕色 | N | N | Y | N | Y | N | Y | N | Y | Y |
說明:Q是Quirks mode 怪異模式或混雜模式,S是Standards mode 標準模式。
在標準模式中:
demo以下:
<script type="text/javascript">
//alert(document.compatMode); //document.compatMode用來判斷當前瀏覽器的渲染方式
</script>
<style type="text/css">
body:nth-of-type(1) .iehack{
color: #F00;/* 對Windows IE9/Firefox 7+/Opera 10+/全部Chrome/Safari的CSS hack ,選擇器也適用幾乎所有Mobile/Linux/Mac browser*/
}
.demo1,.demo2,.demo3,.demo4{
width:100px;
height:100px;
}
.hack{
/*demo1 */
/*demo1 注意順序,不然IE6/7下可能沒法正確顯示,致使結果顯示爲白色背景*/
background-color:red; /* 全部瀏覽器 */
background-color:blue !important;/* 除了IE6之外的瀏覽器*/
*background-color:black; /* IE6, IE7 */
+background-color:yellow;/* IE6, IE7*/
background-color:gray\9; /* IE6, IE7, IE8, IE9, IE10 */
background-color:purple\0; /* IE8, IE9, IE10 */
background-color:orange\9\0;/*IE9, IE10*/
_background-color:green; /*只在 IE6 */
*+background-color:pink; /* IE7*/
}
/*能夠經過javascript檢測IE10,而後給IE10的<html>標籤加上class=」ie10″ 這個類 */
.ie10 #hack{
color:red; /* 只在IE10 */
}
/*demo2*/
.iehack{
/*該實例是用於區分標準模式下ie6~ie9和Firefox/Chrome的hack,注意順序
IE6顯示爲:綠色,
IE7顯示爲:黑色,
IE8顯示爲:紅色,
IE9顯示爲:藍色,
Firefox/Chrome顯示爲:橘色,
(本例IE10效果同IE9,Opera最新版效果同IE8)
*/
background-color:orange; /*Firefox/Chrome */
background-color:red\0; /* IE 8/9/10/Opera - IE8/IE10/Opera */
background-color:blue\9\0; /* ie 9/10 - for ie9/10 */
*background-color:black; /* ie 6/7 - for ie7 */
_background-color:green; /* ie 6 - for ie6 */
}
/*demo3
實例是用於區分標準模式下ie6~ie9和Firefox/Chrome的hack,注意順序
IE6顯示爲:紅色,
IE7顯示爲:藍色,
IE8顯示爲:綠色,
IE9顯示爲:粉色,
Firefox/Chrome顯示爲:橘色,
(本例IE10效果同IE9,Opera最新版效果也同IE9爲粉色)
*/
.element {
background-color:orange; /* 全部的IE/Firefox/Chrome/Oprea*/
}
.element {
*background-color: blue; /* IE6/7,不在 IE8/9/10*/
}
.element {
_background-color: red; /* IE6 */
}
.element {
background-color: green\0; /* IE8/9/10 */
}
:root .element { background-color:pink\0; } /* IE9/10 */
/*demo4*/
/*
該實例是用於區分標準模式下ie6~ie10和Opera/Firefox/Chrome的hack,本例特別要注意順序
IE6顯示爲:橘色,
IE7顯示爲:粉色,
IE8顯示爲:黃色,
IE9顯示爲:紫色,
IE10顯示爲:綠色,
Firefox顯示爲:藍色,
Opera顯示爲:黑色,
Safari/Chrome顯示爲:灰色,
*/
.hacktest{
background-color:blue; /* 都識別,此處針對firefox */
background-color:red\9; /*全部的IE*/
background-color:yellow\0; /*for IE8/IE9/10 最新版opera也認識*/
+background-color:pink; /*ie6/7*/
_background-color:orange; /*ie6*/
}
@media screen and (min-width:0){
.hacktest {background-color:black\0;} /*opera*/
}
@media screen and (min-width:0) {
.hacktest { background-color:purple\9; }/* IE9/IE10 ,:國外有些習慣常寫做\0,根本沒考慮Opera也認識\0的實際 */
}
@media screen and (-ms-high-contrast: active), (-ms-high-contrast: none) {
.hacktest { background-color:green; } /* for IE10+ 此寫法能夠適配到高對比度和默認模式,故可覆蓋全部ie10的模式 */
}
@media screen and (-webkit-min-device-pixel-ratio:0){ .hacktest {background-color:gray;} } /*for Chrome/Safari*/
/* #963棕色 :root is for IE9/IE10, 優先級高於@media, 慎用!若是兩者合用,必要時在@media樣式加入 !important 才能區分IE9和IE10 */
/*
:root .hacktest { background-color:#963\9; }
*/
</style>
demo1是測試不一樣IE瀏覽器下hack 的顯示效果
IE6顯示爲:粉色,
IE7顯示爲:粉色,
IE8顯示爲:藍色,
IE9顯示爲:藍色,
Firefox/Chrome/Opera顯示爲:藍色,
若去掉其中的!important屬性定義,則IE6/7仍然是粉色,IE8是紫色,IE9/10爲橙色,Firefox/Chrome變爲紅色,Opera是紫色。是否是有些奇怪:除了IE6之外,其餘全部的表現都符合咱們的期待。那爲什麼IE6表現的顏色不是_的綠色而是*+background-color:pink的粉色呢?實際上是最後一句所謂的IE7私有hack惹的禍?不是說*+是IE7的專有hack嗎???錯,你可能太粗心了!咱們常說的IE7專有*+hack的格式是*+html selector,而不是上面的直接在屬性上加*+前綴。若是是爲IE7定製特殊樣式,應該這樣使用:
*+html #ie7test { /* IE7 only*/ color:green; }
通過測試,屬性前綴*+只有IE6和IE7認識。而*+html selector只有IE7認識。因此咱們在使用時候必定要特別注意。
demo2實例是用於區分標準模式下ie6~ie9和Firefox/Chrome的hack,注意順序
IE6顯示爲:綠色,
IE7顯示爲:黑色,
IE8顯示爲:紅色,
IE9顯示爲:藍色,
Firefox/Chrome顯示爲:橘色,
(本例IE10效果同IE9,Opera最新版效果同IE8)
demo3實例也是用於區分標準模式下ie6~ie9和Firefox/Chrome的hack,注意順序
IE6顯示爲:紅色,
IE7顯示爲:藍色,
IE8顯示爲:綠色,
IE9顯示爲:粉色,
Firefox/Chrome顯示爲:橘色,
(本例IE10效果同IE9,Opera最新版效果也同IE9爲粉色)
demo4實例是用於區分標準模式下ie6~ie10和Opera/Firefox/Chrome的hack,本例特別要注意順序
IE6顯示爲:橘色,
IE7顯示爲:粉色,
IE8顯示爲:黃色,
IE9顯示爲:紫色,
IE10顯示爲:綠色,
Firefox顯示爲:藍色,
Opera顯示爲:黑色,
Safari/Chrome顯示爲:灰色,
七、css hack 方式三:選擇器前綴法
選擇器前綴法是針對一些頁面表現不一致或者須要特殊對待的瀏覽器,在CSS選擇器前加上一些只有某些特定瀏覽器才能識別的前綴進行hack。
目前最多見的是
*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有效 等等
結合CSS3的一些選擇器,如html:first-child,body:nth-of-type(1),衍生出更多的hack方式,具體的能夠參考下表:
八、css3選擇器結合JavaScript的hack
咱們用IE10進行舉例:
因爲IE10用戶代理字符串(UserAgent)爲:Mozilla/5.0 (compatible; MSIE 10.0; Windows NT 6.2; Trident/6.0),因此咱們可使用javascript將此屬性添加到文檔標籤中,再運用CSS3基本選擇器匹配。
JavaScript代碼:
var htmlObj = document.documentElement; htmlObj.setAttribute('data-useragent',navigator.userAgent); htmlObj.setAttribute('data-platform', navigator.platform );
CSS3匹配代碼:
html[data-useragent*='MSIE 10.0'] #id { color: #F00; }
九、css hack利弊
通常狀況下,咱們儘可能避免使用CSS hack,可是有些狀況爲了顧及用戶體驗實現向下兼容,不得已才使用hack。好比因爲IE8及如下版本不支持CSS3,而咱們的項目頁面使用了大量CSS3新屬性在IE9/Firefox/Chrome下正常渲染,這種狀況下若是不使用css3pie或htc或條件註釋等方法時,可能就得讓IE8-的專屬hack出馬了。使用hack雖然對頁面表現的一致性有好處,但過多的濫用會形成html文檔混亂不堪,增長管理和維護的負擔。相信只要你們一塊兒努力,少用、慎用hack,將來必定會促使瀏覽器廠商的標準愈來愈趨於統一,順利過渡到標準瀏覽器的主流時代。拋棄那些陳舊的IE hack,必將減輕咱們編碼的複雜度,少作無用功。
最後補上一張引自國外某大牛總結的CSS hack表,這時一張6年前的舊知識彙總表了,放在這裏僅供須要時候方便參考。
說明:本文測試環境爲IE6~IE10,Chrome 29.0.1547.66 m,Firefox 20.0.1 ,Opera 12.02等。
本文非原創,只是在學習的時候方便作筆記因此移在此處。