什麼是 haslayout ? javascript
「Layout」是一個 IE/Win 的私有概念,它決定了一個元素如何顯示以及約束其包含的內容、如何與其餘元素交互和創建聯繫、如何響應和傳遞應用程序事件/用戶事件等,這有點相似於一個 窗體的概念。微軟的開發者們認爲盒狀元素(box-type elements)應該具備一個「屬性(property)」(這是面向對象編程中的一個概念),因而他們便使用了 layout , 也就是 hasLayout。hasLayout 其實既不是一個屬性更不是一個行爲,而是 IE 這個渲染引擎代代繼承一向擁有的一個渲染概念,在這個概念下渲染的元素將具備一種特性。實際上這種渲染特性在有些 HTML 元素中與身俱來,而在另一些元素中也能夠經過一些 CSS 屬性將其觸發爲 true ,且一旦觸發將不可逆轉。css
當咱們說一個元素「擁有layout」或「獲得layout」,或者說一個元素「has layout」 的時候,咱們的意思是指它的微軟專有屬性 hasLayout 被設爲了 true 。一個「layout元素」能夠是一個默認就擁有 layout 的元素或者是一個經過設置某些 CSS 屬性獲得 layout 的元素。而「無layout元素」,是指 hasLayout 未被觸發的元素,好比一個未設定寬高尺寸的乾淨 div 元素就能夠作爲一個 「無layout祖先」。給一個默認沒有 layout 的元素賦予 layout 的方法包括設置可觸發 hasLayout = true 的 CSS 屬性。參考默認 layout 元素以及這些屬性列表。沒有辦法設置 hasLayout = false , 除非把一開始那些觸發 hasLayout = true 的 CSS 屬性去除。IE Developer Toobar 能夠實時檢查一個元素的當前樣式;若是 hasLayout 是 true ,那麼它的值顯示爲 「-1」。 咱們能夠經過實時修改一個元素的屬性將「zoom(css)」設置爲「1」來觸發 hasLayout 以便調試。另外能夠用javascript獲取hasLayout的狀 態,<script>alert(my.currentStyle.hasLayout);</script>,可是不能夠設 置。html
另一個須要注意的是「layout」會影響javascript編程。若是一個元素沒有「layout」,那麼 clientWidth/clientHeight 老是返回0。這會讓一些腳本新手感到困惑,並且這和 Mozilla 瀏覽器的處理方式也不同。不過咱們能夠利用這一點在 IE5.0 中檢測「layout」:若是 clientWidth 是零那麼這個元素就沒有 layout。java
什麼狀況下hasLayout會出現?css3
實際上通常狀況都不會出現的,固然除了使用下面默認具備hasLayout的元素或使用特定樣式觸發hasLayout之外;它會帶來各類詭異 表現,當你發現IE6 IE7出現了一些難以想象的問題,首先要檢查的就是是不是hasLayout在搗鬼;hasLayout只出如今IE7及更早版本中,IE8以後不存在 hasLayout解析模式。編程
默認具備 haslayout 的元素(不徹底列表)
* html, body
* table, tr, th, td
* img
* hr
* input, button, file, select, textarea, fieldset
* legend
* marquee
* frameset, frame, iframe
* objects, applets, embed
對於並不是全部的元素都默認有佈局,微軟給出的主要緣由是「性能和簡潔」。若是全部的元素都默認有佈局,會對性能和內存使用上產生有害的影響。瀏覽器
大部分的 IE 顯示錯誤,均可以經過激發元素的 haslayout 屬性來修正。能夠經過設置 css 尺寸屬性(width/height)等來激發元素的 haslayout,使其「擁有佈局」。以下所示,經過設置如下 css 屬性便可。
* display: inline-block
* height: (任何值除了auto) 一般用 _height:1%;解決IE6的問題,height:1%不會改變實際高度
* float: (left 或 right)
* position: absolute
* width: (任何值除了auto)
* writing-mode: tb-rl
* zoom: (除 normal 外任意值)
IE7 還有一些額外的屬性(不徹底列表):
* min-height: (任意值)
* max-height: (除 none 外任意值)
* min-width: (任意值)
* max-width: (除 none 外任意值)
* overflow: (除 visible 外任意值)
* overflow-x: (除 visible 外任意值)
* overflow-y: (除 visible 外任意值)
* position: fixed
其中 overflow-x 和 overflow-y 是 css3 盒模型中的屬性,目前還未被瀏覽器普遍支持。
對於內聯元素(默認即爲內聯的元素,如 span,或 display:inline; 的元素),width 和 height 只在 IE5.x 下和 IE6 或更新版本的 怪異 quirks 模式下觸發 hasLayout 。而對於IE6,若是瀏覽器運行於標準兼容模式下,內聯元素會忽略 width 或 height 屬性,因此設置 width 或 height不能在此種狀況下令該元素具備 layout。
zoom 老是能夠觸發 hasLayout,可是在 IE5.0 中不支持。
具備「layout」 的元素若是同時 display: inline ,那麼它的行爲就和標準中所說的 inline-block很相似了:在段落中和普通文字同樣在水平方向連續排列,受 vertical-align 影響,而且大小能夠根據內容自適應調整。這也能夠解釋爲何單單在 IE 中內聯元素能夠包含塊級元素而少出問題,由於在別的瀏覽器中display: inline 就是內聯,不像 IE 一旦內聯元素擁有 layout 還會變成 inline-block。
haslayout 問題的調試與解決
當網頁在 IE 中有異常表現時,能夠嘗試激發 haslayout 來看看是否是問題所在。經常使用的方法是給某元素 css 設定 zoom:1。使用 zoom:1 是由於大多數狀況下,它能在不影響現有環境的條件下激發元素的 haslayout。而一旦問題消失,那基本上就能夠判斷是haslayout 的緣由。而後就能夠經過設定相應的 css 屬性來對這個問題進行修正了。建議首先要考慮的是設定元素的width/height 屬性,其次再考慮其餘屬性。
對 IE6 及更早版原本說,經常使用的方法被稱爲霍莉破解(Holly hack),即設定這個元素的高度爲 1%(height:1%;)。須要注意的是,當這個元素的 overflow 屬性被設置爲 visible 時,這個方法就失效了。或者使用 IE的條件註釋。
對 IE7 來講,最好的方法時設置元素的最小高度爲 0 (min-height:0;)。
haslayout 問題引發的常見 bug
E6 的躲躲貓(peek-a-boo) bug
bug 修復: _height:1%;佈局
補充資料:性能
position: absolute(IE5+)
float: left|right(IE5+)
display: inline-block(IE5+)
width: "auto"之外的任何值(IE5+; 對inline元素無效)
height: "auto"之外的任何值(IE5+; 對inline元素無效)
zoom: "normal"之外的任何值(IE5.5+; IE私有屬性)
writing-mode: tb-rl(IE5+; IE私有屬性)
overflow: hidden|scroll|auto(IE7; 此屬性在IE6及更早版本中不能應用在未觸發hasLayout的元素上)
overflow-x|-y: hidden|scroll|auto(IE7; 此屬性在IE6及更早版本中不觸發hasLayout; 此屬性在CSS3中才獲支持)
position: fixed(IE7)
min-width: 任何值(IE7; 即便是0)
max-width: "none"之外的任何值(IE7)
min-height: 任何值(IE7)
max-height: "none"之外的任何值 (IE7)
position: fixed(IE7)
position: static(IE5+)
float: none(IE5+)
display: "inline-block"之外的任何值(IE5+)
width|height: "auto"(IE5+; 對inline元素無效)
zoom: "normal"(IE5.5+; IE私有屬性)
writing-mode: 從'tb-rl'到'lr-tb'(IE5+; IE私有屬性)
max-width|max-height: "none"(IE7)
overflow: visible(IE7)
實際操做中
zoom:1; 是很不錯的觸發方法,不會改變原來的任何式樣,並且僅僅是IE能夠識別,可是惟一的壞處就是他不能經過W3C。
設置 display:inline-block 而後再設置回原始的 display 屬性,這樣不會移除 layout,咱們就能夠達到設置 layout 而不使用IE的條件註釋的目的。下面是例子:
[css] view plaincopyprint?
div { display: inline-block; }
div { display: block; } /* 分別在兩段 css 塊中設置 */
hasLayout都會引起什麼問題?
1. 浮動元素會被layout元素自動包含。正常狀況下,浮動元素會按照left和top的設置偏離原來文檔流中的位置,父元素是不會調整高寬去包含該浮動元 素的(也就解釋了爲何浮動元素不能撐開父容器),但在IE中,layout元素會自動調整高和寬以包含浮動元素(給父容器_height:1%;便可解 決)http://blog.csdn.net/hedong37518585/article/details/6639731
2.浮動元素旁邊的元素。當一個塊級元素緊跟在一個左浮動元素以後時,其中的文字內容應該沿着浮動元素的右邊順序排列並會滑到浮動元素下方。可是若是這個塊級元素有 layout,那麼這個元素就會表現爲一個矩形,其中文字不會滑向浮動元素下方。
[html] view plaincopyprint?
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<title>IE6 hasLayout 會影響非浮動元素中的文字是否圍繞浮動元素</title>
<style>
*{
margin:0;
padding:0;
}
.box{
width:220px;
overflow:auto;
font-size:12px;
}
.leftbox{
background:#CCC;
width:100px;
height:100px;
float:left;
*margin-right:-3px; /*針對IE6 浮動元素水平右外邊距移動-3px 便可解決*/
}
.textbox{
background:#FFCCCC;
height:1%;/*去掉後 IE6下 文字不會再圍繞浮動*/
}
</style>
</head>
<body>
<div class="box">
<div class="leftbox">浮動元素</div><p class="textbox">文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本</p>
</div>
</body>
</html>
3. IE專有的濾鏡屬性filter是隻適用於 layout 元素的,也就是說若是你給一個DIV設置透明用的是filter:alpha(opacity=80);若是你沒有讓DIV觸發hasLayout,那麼這個透明將無效
[html] view plaincopyprint?
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<title>IE6 filter透明濾鏡 須要觸發該元素的hasLayout</title>
<style>
*{
margin:0;
padding:0;
}
body{
background:#000;
font-size:12px;
}
.textbox{
background:#FFCCCC;
opacity:0.8;
filter:alpha(opacity=80);
zoom:1;
}
</style>
</head>
<body>
<div class="textbox">文本文本文本文本文本文本文本文本文本本文本文本文本文本文本文本文本文本文本文文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本文本</div>
</body>
</html>
4. hasLayout 會影響一個塊級別連接的鼠標響應區域(可點擊區域)。一般 hasLayout=false 時只有文字覆蓋區域才能響應。而 hasLayout = true 則整個塊狀區域均可響應。
[html] view plaincopyprint?
<!DOCTYPE html>
<html>
<head>
<meta charset="utf-8" />
<title>IE6 hasLayout影響塊元素鼠標響應區域</title>
<style>
*{
margin:0;
padding:0;
}
div{
width:100px;
height:50px;
background:#FF0000;
}
a{
display:block;
height:1%;/*去掉後就不具備Layout佈局 則只有文字覆蓋區域以響應*/
}
</style>
</head>
<body>
<div>
<a id="my" href="javascript:;">click me</a>
</div>
</body>
</html>
這裏有一篇hasLayout的英文介紹,貼上:http://www.satzansatz.de/cssd/onhavinglayout.html