咱们在作前端的时候,常常会遇到各类浏览器显示不太一致的状况,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等。
本文非原创,只是在学习的时候方便作笔记因此移在此处。