ie相容---haslayout,ie---haslayout
要想更好的理解 css, 尤其是 IE 下對 css 的渲染,haslayout 是一個非常有必要徹底弄清除的概念。大多IE下的顯示錯誤,就是源於 haslayout。
什麼是 haslayout ?
haslayout 是Windows Internet Explorer渲染引擎的一個內部組成部分。在InternetExplorer中,一個元素要麼自己對自身的內容進行計算大小和組織,要麼依賴於父元素來計算尺寸和組織內容。為了調節這兩個不同的概念,渲染引擎採用了 hasLayout 的屬性,屬性值可以為true或false。當一個元素的 hasLayout屬性值為true時,我們說這個元素有一個布局(layout)
當一個元素有一個布局時,它負責對自己和可能的子孫元素進行尺寸計算和定位。簡單來說,這意味著這個元素需要花更多的代價來維護自身和裡面的內容,而不是依賴於祖先元素來完成這些工作。因此,一些元素預設會有一個布局。當我們說一個元素“擁有layout”或“得到layout”,或者說一個元素“has layout” 的時候,我們的意思是指它的微軟專有屬性 hasLayout 被設為了 true。一個“layout元素”可以是一個預設就擁有 layout 的元素或者是一個通過設定某些 CSS 屬性得到 layout的元素。如果某個HTML元素擁有 haslayout 屬性,那麼這個元素的 haslayout 的值一定只有 true,haslayout為唯讀屬性 一旦被觸發,就無法復原轉。通過 IE Developer Toolbar 可以查看 IE 下 HTML元素是否擁有haslayout,在 IE Developer Toolbar 下,擁有 haslayout的元素,通常顯示為“haslayout = -1”。
負責組織自身內容的元素將預設有一個布局,主要包括以下元素(不完全列表):
* body and html
* table, tr, th, td
* img
* hr
* input, button, file, select, textarea, fieldset
* marquee
* frameset, frame, iframe
* objects, applets, embed
所以這些屬性有的雖然是內嵌元素,但是可以設定自己的寬與高。
對於並非所有的元素都預設有布局,微軟給出的主要原因是“效能和簡潔”。如果所有的元素都預設有布局,會對效能和記憶體使用量上產生有害的影響。
如何激發 haslayout?
大部分的 IE 顯示錯誤,都可以通過激發元素的 haslayout 屬性來修正。可以通過設定 css 尺寸屬性(width/height)等來激發元素的 haslayout,使其“擁有布局”。如下所示,通過設定以下 css 屬性即可。
* display: inline-block (適用於所有主流瀏覽器)
* height: (任何值除了auto)
* float: (left 或 right)
* position: absolute
* width: (任何值除了auto)
* writing-mode: tb-rl
* zoom: (除 normal 外任意值)
Internet Explorer 7 還有一些額外的屬性(不完全列表):
* 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 盒模型中的屬性,目前還未被瀏覽器廣泛支援。
zoom 總是可以觸發 hasLayout,但是在 IE5.0 中不支援。
具有“layout” 的元素如果同時 display: inline ,那麼它的行為就和標準中所說的 inline-block很類似了:在段落中和普通文字一樣連續排列,受 vertical-align影響,並且大小可以根據內容自適應調整。這也可以解釋為什麼單單在 IE/Win 中內嵌元素可以包含區塊層級元素而少出問題,因為在別的瀏覽器中display: inline 就是內聯,不像 IE/Win 一旦內嵌元素擁有 layout 還會變成 inline-block。
舉個經典的例子:
無浮動的外層div並沒有被裡面的浮動元素的高度撐開,外層無浮動的div其高度並不會自動計算。我們下面再給這個無浮動的div加上個zoom:1;來觸發其hasLayout屬性,這時外層div被內層浮動的div撐開了,當然這隻適用於ie6與ie7,而對於遵守標準的firefox等標準瀏覽器可以加上overflow: hidden;禍水display:table,或是運用虛擬元素:after來解決。
詳見:http://xiaogai1010.blog.163.com/blog/static/13717030820122242616468/
hasLayout對於內嵌元素也可以有效果,當內嵌元素的hasLayout為true的時候,可以給這個內嵌元素設定高度和寬度並得到期望的效果。
(1)可以用zoom:1給內嵌元素觸發haslayout屬性,從而計算其高度,寬度,但這隻適用於ie6與ie7。
(2)display:inline-block,觸發內嵌元素的haslayout屬性,適用於所有主流的瀏覽器。
haslayout 問題的調試與解決
(1)當 網頁在 IE 中有異常表現時,可以嘗試激發 haslayout 來看看是不是問題所在。常用的方法是給某元素 css 設定 zoom:1。使用 zoom:1 是因為大多數情況下,它能在不影響現有環境的條件下激發元素的 haslayout。而一旦問題消失,那基本上就可以判斷是haslayout 的原因。然後就可以通過設定相應的 css 屬性來對這個問題進行修正了。建議首先要考慮的是設定元素的width/height 屬性,其次再考慮其他屬性。
(2)有時是因為觸發了 haslayout屬性,而使元素的布局在ie6與ie7下出問題,這需要用IE外掛程式的DeveloperToolbar看看該元素是否觸發了 haslayout屬性,若是觸發了,可以去掉其寬高,float,position等相關屬性,從而去掉 haslayout屬性,再在頁面中瀏覽下網頁,找出問題所在。