詳解CSS中的z-index屬性在層疊布局中的用法

z-index的重要性
在我看來,z-index 給了我們?nèi)粘9ぷ髦幸詷O大的幫助,我們用它來定義元素的層疊級別(stack level)。受益于它,你能做Popup, DropDown, Tips, 圖文替換等等。
在開始本篇之前,或許我們要先了解一下關(guān)于z-index的基本信息。
W3C這樣描述
每個元素都具有三維空間位置,除了水平和垂直位置外,還能在 "Z軸" 上層層相疊、排列。元素在 "Z軸" 方向上的呈現(xiàn)順序,由層疊上下文和層疊級別決定。
在文檔中,每個元素僅屬于一個層疊上下文。元素的層疊級別為整型,它描述了在相同層疊上下文中元素在 "Z軸" 上的呈現(xiàn)順序。
同一層疊上下文中,層疊級別大的顯示在上,層疊級別小的顯示在下,相同層疊級別時,遵循后來居上的原則,即其在HTML文檔中的順序。
不同層疊上下文中,元素呈現(xiàn)順序以父級層疊上下文的層疊級別來決定呈現(xiàn)的先后順序,與自身的層疊級別無關(guān)。
順序規(guī)則
如果不對節(jié)點設(shè)定 position 屬性, 位于文檔流后面的節(jié)點會遮蓋前面的節(jié)點.
- <div id="a">A</div>
- <div id="b">B</div>
定位規(guī)則
如果將 position 設(shè)為 static, 位于文檔流后面的節(jié)點依然會遮蓋前面的節(jié)點浮動, 所以 position:static 不會影響節(jié)點的遮蓋關(guān)系.
- <div id="a" style="position:static;">A</div>
- <div id="b">B</div>
如果將 position 設(shè)為 relative (相對定位), absolute (絕對定位) 或者 fixed (固定定位), 這樣的節(jié)點會覆蓋沒有設(shè)置 position 屬性或者屬性值為 static 的節(jié)點, 說明前者比后者的默認層級高.
- <div id="a" style="position:relative;">A</div>
- <div id="b">B</div>
在沒有 z-index 屬性干擾的情況下, 根據(jù)這順序規(guī)則和定位規(guī)則, 我們可以做出更加復雜的結(jié)構(gòu). 這里我們對 A 和 B 都不設(shè)定 position, 但對 A 的子節(jié)點 A-1 設(shè)定 position:relative. 根據(jù)順序規(guī)則, B 會覆蓋 A, 又根據(jù)定位規(guī)則 A' 會覆蓋 B.
- <div id="a">
- <div id="a-1" style="position:relative;">A-1</div>
- </div>
- <div id="b">B</div>
上面互相覆蓋在什么時候用到這樣的實現(xiàn)? 看起來偏門, 其實很常用, 比如說, 電子商務(wù)網(wǎng)站側(cè)欄的類目展示列表就可以用這個技巧來實現(xiàn).
下圖是某網(wǎng)站的類目展示區(qū)域, 二級類目的懸浮層覆蓋一級類目列表外框, 而一級類目的節(jié)點覆蓋二級類目的懸浮層. 如果使用 CSS 實現(xiàn)展示效果, 一級類目的外框相當于上面例子中的 A, 一級類目的節(jié)點相當于 A-1, 二級類目的懸浮層相當于 B.
參與規(guī)則
我們嘗試不用 position 屬性, 但為節(jié)點加上 z-index 屬性. 發(fā)現(xiàn) z-index 對節(jié)點沒起作用.
- <div id="a" style="z-index:2;">A</div>
- <div id="b" style="z-index:1;">B</div>
- <div id="c" style="z-index:0;">C</div>
W3C 對 z-index 屬性的描述中提到 在 z-index 屬性僅在節(jié)點的 position 屬性為 relative, absolute 或者 fixed 時生效.
The z-index property specifies the stack order of an element. Only works on positioned elements(position: absolute;, position: relative; or position: fixed;).
- <div id="a" style="z-index:2;">A</div>
- <div id="b" style="position:relative;z-index:1;">B</div>
- <div id="c" style="position:relative;z-index:0;">C</div>
默認值規(guī)則
如果所有節(jié)點都定義了 position:relative. z-index 為 0 的節(jié)點與沒有定義 z-index 在同一層級內(nèi)沒有高低之分; 但 z-index 大于等于 1 的節(jié)點會遮蓋沒有定義 z-index 的節(jié)點; z-index 的值為負數(shù)的節(jié)點將被沒有定義 z-index 的節(jié)點覆蓋.
- <div id="a" style="position:relative;z-index:1;">A</div>
- <div id="b" style="position:relative;z-index:0;">B</div>
- <div id="c" style="position:relative;">C</div>
- <div id="d" style="position:relative;z-index:0;">D</div>
通過檢查我們還發(fā)現(xiàn), 當 position 設(shè)為 relative, absolute 或者 fixed, 而沒有設(shè)置 z-index 時, IE8 以上和 W3C 瀏覽器 (下文我們統(tǒng)稱為 W3C 瀏覽器) 的 z-index 默認值是 auto, 但 IE6 和 IE7 是 0.
從父規(guī)則
如果 A, B 節(jié)點都定義了 position:relative, A 節(jié)點的 z-index 比 B 節(jié)點大, 那么 A 的子節(jié)點必定覆蓋在 B 的子節(jié)點前面.
- <div id="a" style="position:relative;z-index:1;">
- <div id="a-1">A-1</div>
- </div>
- <div id="b" style="position:relative;z-index:0;">
- <div id="b-1">B-1</div>
- </div>
如果所有節(jié)點都定義了 position:relative, A 節(jié)點的 z-index 和 B 節(jié)點一樣大, 但因為順序規(guī)則, B 節(jié)點覆蓋在 A 節(jié)點前面. 就算 A 的子節(jié)點 z-index 值比 B 的子節(jié)點大, B 的子節(jié)點還是會覆蓋在 A 的子節(jié)點前面.
- <div id="a" style="position:relative;z-index:0;">
- <div id="a-1" style="position:relative;z-index:2;">A-1</div>
- </div>
- <div id="b" style="position:relative;z-index:0;">
- <div id="b-1" style="position:relative;z-index:1;">B-1</div>
- </div>
很多人將 z-index 設(shè)得很大, 9999 什么的都出來了, 如果不考慮父節(jié)點的影響, 設(shè)得再大也沒用, 那是無法逾越的層級.
層級樹規(guī)則
可能你會覺得在 DOM 結(jié)構(gòu)中的兄弟節(jié)點會拎出來進行比較并確定層級, 其實不然.
- <div id="a" style="position:relative;z-index:2;">
- <div id="a-1" style="position:relative;z-index:0;">A-1</div>
- </div>
- <div id="b">
- <div id="b-1" style="position:relative;z-index:1;">B-1</div>
- </div>
我們認為同時將 position 設(shè)為 relative, absolute 或者 fixed, 并且 z-index 經(jīng)過整數(shù)賦值的節(jié)點, 會被放置到一個與 DOM 不一樣的層級樹里面, 并且在層級樹中通過對比 z-index 決定顯示的層級. 上面的例子如果用層級樹來表示的話, 應(yīng)該如下圖所示.
圖中雖然 A-1 (z-index:0) 的值比 B-1 (z-index:1) 小, 但因為在層級樹里 A (z-index:2) 和 B-1 在一個層級, 而 A 的值比 B-1 大, 根據(jù)從父規(guī)則, A-1 顯示在 B-1 前面.
參與規(guī)則 2
前面提到的參與規(guī)則認為只要節(jié)點的 position 屬性為 relative, absolute 或者 fixed, 即可參與層級比較, 其實不準確. 如果所有節(jié)點都定義了 position:relative, 并且將 z-index 設(shè)為整數(shù)值, 根據(jù)從父規(guī)則, 父節(jié)點的層級決定了子節(jié)點所在層級.
- <div id="a" style="position:relative;z-index:0;">
- <div id="a-1" style="position:relative;z-index:100;">A-1</div>
- </div>
- <div id="b">
- <div id="b-1" style="position:relative;z-index:0;">
- <div id="b-1-1" style="position:relative;z-index:10;">B-1-1</div>
- </div>
- </div>
- <div id="c" style="position:relative;z-index:0;">
- <div id="c-1">
- <div id="c-1-1">
- <div id="c-1-1-1" style="position:relative;z-index:1;">C-1-1-1</div>
- </div>
- </div>
- </div>
例子中 A, B-1, C-1-1 作為父節(jié)點, z-index 的值相同, 根據(jù)順序規(guī)則, C-1-1 在 B-1 之前, B-1 在 A 之前; 又根據(jù)從父規(guī)則, 無論子節(jié)點的 z-index 值是什么, C-1-1-1 在 B-1-1 之前, B-1-1 在 A-1 之前.
如果我們將所有父節(jié)點的 z-index 屬性去除, 詭異的事情發(fā)生了. IE6 和 IE7 瀏覽器顯示效果不變, 而 W3C 瀏覽器的子節(jié)點不再從父, 而是根據(jù)自身的 z-index 確定層級.
- <div id="a" style="position:relative;">
- <div id="a-1" style="position:relative;z-index:100;">A-1</div>
- </div>
- <div id="b">
- <div id="b-1" style="position:relative;">
- <div id="b-1-1" style="position:relative;z-index:10;">B-1-1</div>
- </div>
- </div>
- <div id="c" style="position:relative;">
- <div id="c-1">
- <div id="c-1-1">
- <div id="c-1-1-1" style="position:relative;z-index:1;">C-1-1-1</div>
- </div>
- </div>
- </div>
根據(jù)默認值規(guī)則, IE6 / IE7 和 W3C 瀏覽器上的元素存在 z-index 默認值的區(qū)別. 我們相信, 僅當 position 設(shè)為 relative, absolute 或者 fixed, 并且 z-index 賦整數(shù)值時, 節(jié)點被放置到層級樹; 而 z-index 為默認值時, 只在 document 兄弟節(jié)點間比較層級. 在 W3C 瀏覽器中, A, B-1 和 C-1-1 的 z-index 均為 auto, 不參與層級比較.
而在 IE6 和 IE7 中, 因為 z-index 的默認值是 0, 所以也參與了層級比較.
設(shè)置了 position 而沒有 z-index 的節(jié)點雖然不參與層級樹的比較, 但還會在 DOM 中與兄弟節(jié)點進行層級比較.
- <div id="a" style="position:relative;">
- <div id="a-1" style="position:relative;z-index:100;">A-1</div>
- </div>
- <div id="b">
- <div id="b-1">
- <div id="b-1-1" style="position:relative;z-index:10;">B-1-1</div>
- </div>
- </div>
- <div id="c" style="position:relative;">
- <div id="c-1">
- <div id="c-1-1">
- <div id="c-1-1-1" style="position:relative;z-index:1;">C-1-1-1</div>
- </div>
- </div>
- </div>
我們對上個例子改造一下, 將 B-1 的 position 屬性刪除后, W3C 瀏覽器顯示如下圖. 根據(jù)定位規(guī)則, A 和 C-1-1 會顯示在 B-1 的前面; 而根據(jù)順序規(guī)則, C-1-1 又顯示在 A 前面.
在 IE6 和 IE7 中, 因為 A 和 C-1-1 設(shè)置了 position:relative, 而且 z-index 的默認值為 0, 所以也參與層級樹比較, 所以有如下效果.
總結(jié)
瀏覽器節(jié)點顯示層級是一個費力的活, 今天你覺得 A 區(qū)塊會永遠置頂, 但明天因為需求變動, 突然出現(xiàn) B 元素需要置頂. 一層一層往上堆砌, 某天回頭一看, 發(fā)現(xiàn)很多區(qū)塊交錯在一起, 而且他們的值一個比一個大, 根本搞不清頭緒. 我覺得在操刀干活之前, 最好先將 position, z-index 和層級的關(guān)系搞搞清楚, 以免后患無窮.
另外, 非情非得已, 切勿用 JavaScript 計算 z-index, 并將某個節(jié)點的 z-index 設(shè)置成所有節(jié)點中層級最高.
因為篇幅太長, 本文僅從節(jié)點屬性角度進行討論, 沒有涉及 select 和 iframe 等特殊頁面節(jié)點考慮, 如果有機會下次再為大家分享.
相關(guān)文章
- 你對CSS層疊和繼承的概念和用法你是否熟悉,這里和大家分享一下,文檔中的一個元素可能同時被多個CSS選擇器選中,每個選擇器都有一些CSS規(guī)則,這就是層疊,而所謂繼承,就2011-01-27
- 我們知道文檔中的一個元素可能同時被多個css選擇器選中,每個選擇器都有一些css規(guī)則,這就是層疊。所謂繼承,就是父元素的規(guī)則也會適用于子元素。2010-03-14
- 下面小編就為大家?guī)硪黄狢SS默認可繼承樣式詳解。小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2016-05-06
- 這篇文章主要介紹了CSS中的繼承,是CSS入門學習中的基礎(chǔ)知識,需要的朋友可以參考下2015-07-16
CSS教程:學習CSS的繼承性-CSS教程-網(wǎng)頁制作-網(wǎng)頁教學網(wǎng)
所謂CSS的繼承是指被包在內(nèi)部的標簽將擁有外部標簽的樣式性質(zhì)。繼承特性最典型的應(yīng)用通常發(fā)揮在整個網(wǎng)頁的樣式預設(shè),需要指定為其它樣式的部份設(shè)定在個別元素里即可。這項2008-10-17CSS樣式表中繼承關(guān)系的空格與不空格-CSS教程-網(wǎng)頁制作-網(wǎng)頁教學網(wǎng)
這可能是一個非常簡單的問題,但是今天花一點點時間把這個簡單的問題在說清晰一點,相信大家對CSS的學習和認識會很有幫助,強化一些概念的東西,對以后的工作效率的2008-10-17CSS教程:水平對齊(text-align)-CSS教程-網(wǎng)頁制作-網(wǎng)頁教學網(wǎng)
水平對齊(text-align),用以設(shè)定元素內(nèi)文本的水平對齊方式。 1.語法 text-align具體參數(shù)如下: 語法:text-align:left|right|center|justify 說明:2008-10-17CSS教程:line-height屬性的繼承-CSS教程-網(wǎng)頁制作-網(wǎng)頁教學網(wǎng)
原文:http://www.planabc.net/2008/07/25/unitless-line-heights/ 淘寶商城的detail 頁面“產(chǎn)品詳情”部分是商家自定義區(qū)塊,曾出現(xiàn)這樣一個問題: &2008-10-17CSS教程:行高line-height屬性(2)-CSS教程-網(wǎng)頁制作-網(wǎng)頁教學網(wǎng)
相關(guān)文章:CSS教程:行高line-height屬性(1) 7.3.3行高的計算與繼承 以em、ex和百分比為單位的行高,其基數(shù)是元素本身的字體尺寸。例如有代碼如下: <pstyle=&quo2008-10-17CSS教程:理解繼承屬性及應(yīng)用-CSS教程-網(wǎng)頁制作-網(wǎng)頁教學網(wǎng)
4.5繼承 在本書的[3.3.2繼承與層疊]一節(jié)曾經(jīng)簡要介紹過繼承:(X)HTML元素可以從其父元素那里繼承部分CSS屬性,即使當前元素并沒有定義該屬性。 4.5.1值的繼承 繼承也是2008-10-17