CSS----层叠上下文

98 阅读4分钟

背景

你有没有在项目中遇到过这种情况,自己写的组件需要有更高的展示层级,于是尝试改变z-index的值来调整层级顺序,但是无论怎样调整z-index的值都无法达到预期效果。也就是说,虽然我们把当前组件的z-index设为了9999999+也无济于事,那么此时我们就进入了一个误区。

例如:

<!DOCTYPE html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Document</title>
    <style type="text/css">
      .parent1 {
        height: 200px;
        width: 200px;
        background-color: green;
        position: absolute;
        z-index: 1;
      }
      .child {
        height: 200px;
        width: 200px;
        background-color: rgb(132, 132, 201);
        position: absolute;
        left: 30px;
        top: 30px;
        z-index: 999;
      }
      .parent2 {
        height: 200px;
        width: 200px;
        background-color: pink;
        position: absolute;
        top: 70px;
        left: 70px;
        z-index: 2;
      }
    </style>
  </head>
  <body>
    <div class="parent1">
      <span>z-index: 1</span>
      <div class="child">z-index: 999</div>
    </div>
    <div class="parent2">z-index: 2</div>
  </body>
</html>

我们以为它是这样子显示:

image.png

实际上它是这样子

image.png

原理

"层叠上下文"是一种用于管理网页元素堆叠顺序和渲染的重要概念。它决定了元素如何在页面上叠加和显示,通常与元素的CSS属性和属性值有关。

最里层的层叠上下文即为执行堆叠的根元素,也就是产生层叠上下文的元素,如示列中的parent1、2、child,每个层叠上下文都遵循以下规则进行展示:

  1. 执行堆叠的根元素。
  2. 有设定positionz-index负数的元素和它们的子元素-1-2前面。
  3. 没有设定position的元素,一般元素。
  4. 有设定positionz-indexauto,设定opacity小于1和其他transforms等属性也在此列。
  5. 有设定positionz-index为正数。
  6. 都一样的时候就按文件中代码出现的先后顺序,后出现的出现在上面。

image.png

产生机制

每个层叠上下文中可以嵌套别的层叠上下文,每个层叠上下文中都按同一套层叠顺序规则进行展示。层叠上下文的创建有多种方式:

  1. 文档根元素,根<html></html>

  2. position 值为 absolute(绝对定位)或 relative(相对定位)且 z-index 值不为 auto 的元素

  3. position 值为 fixed(固定定位)或 sticky(粘滞定位)的元素(沾滞定位适配所有移动设备上的浏览器,但老的桌面浏览器不支持);

  4. flex (flexbox (en-US)) 容器的子元素,且 z-index 值不为 auto

  5. grid (grid) 容器的子元素,且 z-index 值不为 auto

  6. opacity 属性值小于 1 的元素(参见the specification for opacity

  7. mix-blend-mode 属性值不为 normal 的元素

  8. 以下任意属性值不为 none 的元素:

    • transform
    • filter
    • perspective
    • clip-path
    • mask/mask-image/mask-border
  9. isolation 属性值为 isolate 的元素

  10. -webkit-overflow-scrolling 属性值为 touch 的元素

  11. will-change 值设定了任一属性而该属性在 non-initial 值时会创建层叠上下文的元素(参考这篇文章);

  12. contain 属性值为 layoutpaint 或包含它们其中之一的合成值(比如 contain: strictcontain: content)的元素。

应用

产生了层叠上下文之后,就会按照层叠顺序进行展示,这个过程会应用到浏览器的分层合成机制中

浏览器分层合成机制将网页内容分成多个图层,每个图层都可以独立地渲染。这样可以减少页面重新绘制的需要,提高了性能和响应速度

浏览器会根据布局树生成图层树,为了生成图层树,需要两个条件:

  1. 拥有层叠上下文属性的元素会被提升为单独的一层

  2. 需要剪裁(clip)的地方也会被创建为图层

image.png

示例中的图层树如下所示:

image.png

总结

  • 因此,当我们的某个元素已经创建了层叠上下文环境(stacking contetxt),那么它里面的元素再怎么设置z-index也不会影响它的父元素的层叠顺序,也就是说它只能和它所在的层叠上下文环境中的元素进行比较,而他的父元素的堆叠顺序决定了他在父元素所处环境中的位置。
  • 也就是说当我们把z-index设得很大却无济于事的时候,很可能该元素的上层元素已经具备stacking context了。
  • 所以,为了达到我们在demo中想要的效果其实也很简单,只要把parent1z-index改为3即可。

省流版:元素层级很高还是被遮挡,就看他的父元素层级,以此类推(是不是很快就理解了---调皮)

参考文章: 层叠上下文、层叠层级、层叠顺序 | Jiuto's blog