浅谈为什么我的 z-index 又不生效了


Posted in HTML / CSS onJuly 15, 2022

表面看起来,z-index 似乎是一个很简单的属性,你给它设置哪个值,元素就会位于 z 轴的哪个位置。但它实际上并没有我们想象的这么简单,这个属性背后是一系列决定元素所在层级的规则。

在进行今天的介绍前,我们先列出三个问题,如果你能一眼看出它们的解决方案,那么恭喜你掌握了z-index,也就不需要阅读本文了;如果不行,那么耐心看完本文,相信能找到答案。

三个自测问题

  • 问题一:为什么z-index 数值更大,但 Content 没有在 Box 2 之上?
<div id="box1">
    Box 1
</div>

<div id="content">
    Content <br/>
    z-index: 2;
</div>

<div id="box2">
    Box 2 <br/>
    z-index: 1;
</div>
div {
    padding: 25px;
    font-size: larger;
}

#box1 {
    background-color: chocolate;
    width: 200px;
    height: 100px;
    margin-bottom: -50px;
}

#content {
    background-color: gold;
    width: 300px;
    height: 200px;
    z-index: 2;
}

#box2 {
    background-color: cyan;
    width: 200px;
    height: 100px;
    margin-top: -50px;
    z-index: 1;
}

浅谈为什么我的 z-index 又不生效了

  • 问题二:明明 z-index 数值更小,为什么 Content 这次反而在Box 2 之上了?
<div id="box1">
    Box 1
</div>

<div id="content">
    Content <br/>
    transform: rotate(90deg); <br/>
    z-index: 1;
</div>

<div id="box2">
    <br/>
    Box 2 <br/>
    z-index: 2;
</div>
div {
    padding: 25px;
    font-size: larger;
}

#box1 {
    background-color: chocolate;
    width: 200px;
    height: 100px;
    margin-bottom: -10px;
}

#content {
    background-color: gold;
    width: 250px;
    height: 200px;
    z-index: 1;
    transform: rotate(90deg);
}

#box2 {
    background-color: cyan;
    width: 200px;
    height: 100px;
    margin-top: -10px;
    z-index: 2;
}

浅谈为什么我的 z-index 又不生效了

  • 问题三:为什么明明 z-index 是最大的,但Box 2-3Content 之下?
<div id="box1">
    Box 1
</div>

<div id="content">
    Content <br/>
    z-index: 2; <br/>
    position: relative;
</div>

<div id="box2">
    <br/><br/>
    Box 2 <br/>
    z-index: 1; <br/>
    position: relative;
    <div id="box2-3">
        Box 2-3 <br/>
        z-index: 5;  <br/>
        position: absolute;
    </div>
</div>
div {
    padding: 25px;
    font-size: larger;
}

#box1 {
    background-color: chocolate;
    width: 200px;
    height: 100px;
    margin-bottom: -50px;
}

#content {
    background-color: gold;
    width: 200px;
    height: 100px;
    margin-left: 50px;
    z-index: 2;
    position: relative;
}

#box2 {
    background-color: cyan;
    width: 200px;
    height: 100px;
    margin-top: -50px;
    z-index: 1;
    position: relative;
}

#box2-3 {
    background-color: green;
    width: 200px;
    height: 100px;
    padding-left: 150px;
    left: 180px;
    top: -50px;
    z-index: 5;
    position: absolute;
}

浅谈为什么我的 z-index 又不生效了

z-index 简介

没有使用 z-index 的时候,元素的层叠关系由2个因素决定:

  • 该元素的position是否是static,如果是static,那么这个元素就称为 non-positioned ;反之,如果 position 值是 relative, absolute, fixed, 或 sticky 则称 positionedpositioned 元素享受特权,会覆盖 non-positioned 元素。而 non-positioned 元素中,有 float样式的元素覆盖没有 float 的。
  • 元素的“出场”顺序 —— 即在html中的顺序,同类型元素遵循后来者居上的原则。

z-index 属性设定了一个定位元素及其后代元素或 flex 项目的 z-order。当元素之间重叠的时候,z-index 较大的元素会覆盖较小的元素在上层进行显示。

所谓 z-index,只有在以下场景适用。分别为:

  • 首先,z-index这个属性并不是在所有的元素上都有效果。它仅仅只在 positioned 元素上有效果。
  • 要判断元素在 z轴 上的堆叠顺序,并不仅仅是直接比较两个元素的 z-index 值的大小,同时,这个堆叠顺序还由元素的层叠上下文和层叠等级共同决定。

层叠上下文

z-index 存在的一个背景是 Stacking Context ,中文常译作层叠上下文(其实数据结构中的栈的单词也是 stack,所以层叠上下文中已经蕴含了后来者居上的意思)。

层叠上下文,是HTML中一个三维的概念。在 CSS2.1 规范中,每个盒模型的位置是三维的,分别是平面画布上的X轴Y轴以及表示层叠的Z轴

一般情况下,元素在页面上沿 X轴Y轴 平铺,我们是察觉不到它们在Z轴上的层叠关系。而一旦元素发生堆叠,这时就能发现某个元素可能覆盖了另一个元素或者被另一个元素覆盖。

如果一个元素含有层叠上下文,(也就是说它是层叠上下文元素),我们可以理解为这个元素在Z轴上就“高人一等”,最终表现就是它离屏幕观察者更近。

浅谈为什么我的 z-index 又不生效了

构建层叠上下文和盖楼比较类似:

首先, <html> 元素是地平线或地基 —— 所有楼都是从地基开始盖的

接下来,每产生一个层叠上下文,相当于盖一座楼, z-index 的值相当于楼的高度

以下几种元素可以产生层叠上下文:

  • 元素的 position 值为 absoluterelative, 且 z-index 值不为 auto (默认值).
  • 元素的 position 值为 fixedsticky
  • 元素是 flexbox 容器的子元素, 且 z-index 值不为 auto (默认值)
  • 元素是 grid 容器的子元素, 且 z-index 值不为 auto (默认值)
  • 元素有 opacity 值且值小于 1.
  • 元素有以下任意一项的值,且值不为 none :
    • transform
    • filter
    • perspective
    • clip-path
    • mask / mask-image / mask-border
  • 元素有 isolation 值且值为 isolate.
  • 元素有 mix-blend-mode 值且值不为 normal.
  • 元素有 -webkit-overflow-scrolling 值且值为 touch.
  • 其他几种冷门的情况

第三,层叠上下文是可以嵌套的 —— 这是最容易让人误解的一块。

嵌套,顾名思义就是在一个 层叠上下文 中能创建 另一个层叠上下文

假如在地基上盖一座50米高的楼(即 z-index: 50), 是否可以在楼里再盖一栋 100米高的楼中楼呢?

当然不可能!但是你可以在这座楼里建一座 100 级阶梯高的大堂。

换句话说,在嵌套的层叠上下文中,子层叠上下文被限制在了父层叠上下文中,它们的 z-index “单位”已经不一样了(z-index 没有单位,这边只是用于理解),无论子层叠上下文的 z-index 值有多大都无法突破父层叠上下文的高度。

层叠上下文小结:

  • 元素的第一级层叠上下文
  • 特定样式的元素可以产生新的层叠上下文,且z-index的值在这些元素中才有效
  • 子层叠上下文的“高度”被限制在了父层叠上下文中
  • 在同级层叠上下文中,没有(有效) z-index 的元素依然遵循上一小节的规律;z-index 值相同的元素遵循后来者居上原则。

需要注意:层叠上下文嵌套元素嵌套 不是一一对应的关系,一个元素所处的父层叠上下文是由内向外找到的第一个能产生层叠上下文的元素所产生的层叠上下文。

看个例子便于理解:

<div id="div1" style="position: relative; z-index: 1">
  <div id="div2" style="position: relative; z-index: 1">
      所处的父层叠上下文是 div1 产生的层叠上下文
  </div>

  <div id="div3">
      <div id="div4" style="position: relative; z-index: 2">
          所处的父层叠上下文也是 div1 产生的层叠上下文
      </div>
  </div>
</div>

虽然 div4 外面还有层 div3,但是由于 div3 不能产生层叠上下文,所以 div4 所处的父层叠上下文也是 div1 (产生的层叠上下文) —— 虽然在html元素层级中 div4div2 更深了一级,但是 div4div2 在层叠上下文层面上是同级的,因此它们可以相互比较 z-index 值来决定谁在上面。

三个问题的解答

学习完上面的 z-index 相关知识点,我们来回答开头提出的三个问题。

  • 第一个问题中 z-index 不生效的原因在于这三个元素都不能产生层叠上下文,因此z-index值对它们不生效 —— 根据出场顺序决定了 Content 处在 Box 2 之下。

  • 第二个问题的 Box 2不能产生层叠上下文,因此z-index同样是无效的;Content 因为使用了 transform 属性,产生了层叠上下文,相当于盖了一座 1 米高的楼( z-index: 1 )

  • Box 2Content 在同一级层叠上下文中,且Box 2z-index 比较小, 因此 Box 2Content 之下;且 Box 2-3Box 2的层叠上下文下新建了个子层叠上下文,因此Box 2-3的高度被限制在了 Box 2 之内,因此 Box 2-3z-index 再高也没用。

到此这篇关于浅谈为什么我的 z-index 又不生效了的文章就介绍到这了,更多相关z-index不生效内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章,希望大家以后多多支持三水点靠木!

 
HTML / CSS 相关文章推荐
CSS3轻松实现清新 Loading 效果的简单实例
Jun 06 HTML / CSS
深入理解css中vertical-align属性
Apr 18 HTML / CSS
HTML5 Canvas 起步(1) - 基本概念
May 12 HTML / CSS
HTML5边玩边学(2)基础绘图实现方法
Sep 21 HTML / CSS
css3 transform 3d 使用css3创建动态3d立方体(html5实践)
Jan 06 HTML / CSS
使用css创建三角形 使用CSS3创建3d四面体原理及代码(html5实践)
Jan 06 HTML / CSS
HTML5微信播放全屏问题的解决方法
Mar 09 HTML / CSS
详解HTML5之pushstate、popstate操作history,无刷新改变当前url
Mar 15 HTML / CSS
解决html5中video标签无法播放mp4问题的办法
May 07 HTML / CSS
html5 canvas绘制网络字体的常用方法
Aug 26 HTML / CSS
POST提交数据常见的四种方式
Jan 18 HTML / CSS
CSS 实现角标效果的完整代码
Jun 28 HTML / CSS
纯CSS实现一个简单步骤条的示例代码
Jul 15 #HTML / CSS
CSS中使用grid布局实现一套模板多种布局
Jul 15 #HTML / CSS
如何解决flex文本溢出问题小结
Jul 15 #HTML / CSS
使用HBuilder制作一个简单的HTML5网页
使用CSS定位HTML元素的实现方法
CSS实现背景图片全屏铺满自适应的3种方式
Jul 07 #HTML / CSS
HTML实现仿Windows桌面主题特效的实现
Jun 28 #HTML / CSS
You might like
PHP中exec与system用法区别分析
2014/09/22 PHP
详解阿里云视频直播PHP-SDK接入教程
2020/07/09 PHP
JavaScript定时器详解及实例
2013/08/01 Javascript
对 jQuery 中 data 方法的误解分析
2014/06/18 Javascript
JavaScript中函数(Function)的apply与call理解
2015/07/08 Javascript
完美解决UI-Grid表格元素中多个空格显示为一个空格的问题
2017/04/25 Javascript
详解Vue基于 Nuxt.js 实现服务端渲染(SSR)
2018/04/05 Javascript
vue-cli与webpack处理静态资源的方法及webpack打包的坑
2018/05/15 Javascript
bootstrap里bootstrap动态加载下拉框的实例讲解
2018/08/10 Javascript
node中的session的具体使用
2018/09/14 Javascript
vue观察模式浅析
2018/09/25 Javascript
微信小程序基于canvas渐变实现的彩虹效果示例
2019/05/03 Javascript
如何在微信小程序中实现Mixins方案
2019/06/20 Javascript
vue 源码解析之虚拟Dom-render
2019/08/26 Javascript
layui实现下拉复选功能的例子(包括数据的回显与上传)
2019/09/24 Javascript
Node Express用法详解【安装、使用、路由、中间件、模板引擎等】
2020/05/13 Javascript
你不知道的 TypeScript 高级类型(小结)
2020/08/28 Javascript
python获取android设备的GPS信息脚本分享
2015/03/06 Python
Python正则获取、过滤或者替换HTML标签的方法
2016/01/28 Python
pyQt4实现俄罗斯方块游戏
2018/06/26 Python
Django框架模型简单介绍与使用分析
2019/07/18 Python
Django 通过JS实现ajax过程详解
2019/07/30 Python
python paramiko远程服务器终端操作过程解析
2019/12/14 Python
python logging.basicConfig不生效的原因及解决
2020/02/20 Python
全面介绍python中很常用的单元测试框架unitest
2020/12/14 Python
使用 HTML5 Canvas 制作水波纹效果点击图片就会触发
2014/09/15 HTML / CSS
Bose加拿大官方网站:美国知名音响品牌
2019/03/21 全球购物
德国领先的大尺码和超大尺码男装在线零售商:Bigtex
2019/06/22 全球购物
文明教师事迹材料
2014/01/16 职场文书
班风学风建设方案
2014/05/06 职场文书
优秀的个人求职信范文
2014/05/09 职场文书
县人大领导班子四风对照检查材料思想汇报
2014/10/09 职场文书
Java基础之线程锁相关知识总结
2021/06/30 Java/Android
Nginx实现负载均衡的项目实践
2022/03/18 Servers
PostgreSQL数据库去除重复数据和运算符的基本查询操作
2022/04/12 PostgreSQL
vue 自定义的组件绑定点击事件
2022/04/21 Vue.js