博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
谈谈一些有趣的CSS题目(三)-- 层叠顺序与堆栈上下文知多少
阅读量:4679 次
发布时间:2019-06-09

本文共 3748 字,大约阅读时间需要 12 分钟。

开本系列,讨论一些有趣的 CSS 题目,抛开实用性而言,一些题目为了拓宽一下解决问题的思路,此外,涉及一些容易忽视的 CSS 细节。

解题不考虑兼容性,题目天马行空,想到什么说什么,如果解题中有你感觉到生僻的 CSS 属性,赶紧去补习一下吧。

不断更新,不断更新,不断更新,重要的事情说三遍。

所有题目汇总在我的  。

 

3、层叠顺序(stacking level)与堆栈上下文(stacking context)知多少?

z-index 看上去其实很简单,根据 z-index 的高低决定层叠的优先级,实则深入进去,会发现内有乾坤。

看看下面这题,定义两个 div A 和 B,被包括在同一个父 div 标签下。HTML结构如下:

1
2
3
4
<
div 
class="container">
    
<
div 
class="inline-block">#divA display:inline-block</
div
>
    
<
div 
class="float"> #divB float:left</
div
>
</
div
>

它们的 CSS 定义如下:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
.container{
    
position
:
relative
;
    
background
:
#ddd
;
}
.container > div{
    
width
:
200px
;
    
height
:
200px
;
}
.float{
    
float
:
left
;
    
background-color
:deeppink;
}
.inline-
block
{
    
display
:inline-
block
;
    
background-color
:yellowgreen;
    
margin-left
:
-100px
;
}

大概描述起来,意思就是拥有共同父容器的两个 DIV 重叠在一起,是 display:inline-block 叠在上面,还是float:left 叠在上面?

注意这里 DOM 的顺序,是先生成 display:inline-block ,再生成 float:left 。当然也可以把两个的 DOM 顺序调转如下:

1
2
3
4
<
div 
class="container">
    
<
div 
class="float"> #divB float:left</
div
>
    
<
div 
class="inline-block">#divA display:inline-block</
div
>
</
div
>

会发现,无论顺序如何,始终是 display:inline-block 的 div 叠在上方。

这里其实是涉及了所谓的层叠水平(stacking level),有一张图可以很好的诠释:

运用上图的逻辑,上面的题目就迎刃而解,inline-blcok 的 stacking level 比之 float 要高,所以无论 DOM 的先后顺序都堆叠在上面。

不过上面图示的说法有一些不准确,按照  的说法,准确的 7 层为:

  1. the background and borders of the element forming the stacking context.

  2. the child stacking contexts with negative stack levels (most negative first).

  3. the in-flow, non-inline-level, non-positioned descendants.

  4. the non-positioned floats.

  5. the in-flow, inline-level, non-positioned descendants, including inline tables and inline blocks.

  6. the child stacking contexts with stack level 0 and the positioned descendants with stack level 0.

  7. the child stacking contexts with positive stack levels (least positive first).

稍微翻译一下:

  1. 形成堆叠上下文环境的元素的背景与边框

  2. 拥有负 z-index 的子堆叠上下文元素 (负的越高越堆叠层级越低)

  3. 正常流式布局,非 inline-block,无 position 定位(static除外)的子元素

  4. 无 position 定位(static除外)的 float 浮动元素

  5. 正常流式布局, inline-block元素,无 position 定位(static除外)的子元素(包括 display:table 和 display:inline )

  6. 拥有 z-index:0 的子堆叠上下文元素

  7. 拥有正 z-index: 的子堆叠上下文元素(正的越低越堆叠层级越低)

所以我们的两个 div 的比较是基于上面所列出来的 4 和 5 。5 的 stacking level 更高,所以叠得更高。

不过!不过!不过!重点来了,请注意,上面的比较是基于两个 div 都没有形成 堆叠上下文 这个为基础的。下面我们修改一下题目,给两个 div ,增加一个 opacity:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
.container{
    
position
:
relative
;
    
background
:
#ddd
;
}
.container > div{
    
width
:
200px
;
    
height
:
200px
;
    
opacity:
0.9
; // 注意这里,增加一个 opacity
}
.float{
    
float
:
left
;
    
background-color
:deeppink;
}
.inline-
block
{
    
display
:inline-
block
;
    
background-color
:yellowgreen;
    
margin-left
:
-100px
;
}

会看到,inline-block 的 div 不再一定叠在 float 的 div 之上,而是和 HTML 代码中 DOM 的堆放顺序有关,后添加的 div 会 叠在先添加的 div 之上。

这里的关键点在于,添加的 opacity:0.9 这个让两个 div 都生成了 stacking context(堆叠上下文) 的概念。此时,要对两者进行层叠排列,就需要 z-index ,z-index 越高的层叠层级越高。

堆叠上下文是HTML元素的三维概念,这些HTML元素在一条假想的相对于面向(电脑屏幕的)视窗或者网页的用户的 z 轴上延伸,HTML 元素依据其自身属性按照优先级顺序占用层叠上下文的空间。

那么,如何触发一个元素形成 堆叠上下文 ?方法如下,摘自 :

  • 根元素 (HTML),
  • z-index 值不为 "auto"的 绝对/相对定位,
  • 一个 z-index 值不为 "auto"的 flex 项目 (flex item),即:父元素 display: flex|inline-flex,
  • opacity 属性值小于 1 的元素(参考 the specification for opacity),
  • transform 属性值不为 "none"的元素,
  • mix-blend-mode 属性值不为 "normal"的元素,
  • filter值不为“none”的元素,
  • perspective值不为“none”的元素,
  • isolation 属性被设置为 "isolate"的元素,
  • position: fixed
  • 在 will-change 中指定了任意 CSS 属性,即便你没有直接指定这些属性的值
  • -webkit-overflow-scrolling 属性被设置 "touch"的元素

所以,上面我们给两个 div 添加 opacity 属性的目的就是为了形成 stacking context。也就是说添加 opacity 替换成上面列出来这些属性都是可以达到同样的效果。

在层叠上下文中,其子元素同样也按照上面解释的规则进行层叠。 特别值得一提的是,其子元素的 z-index 值只在父级层叠上下文中有意义。意思就是父元素的 z-index 低于父元素另一个同级元素,子元素 z-index再高也没用。

理解上面的 stacking-level 与 stacking-context 是理解 CSS 的层叠顺序的关键。

 

所有题目汇总在我的  ,发到博客希望得到更多的交流。

到此本文结束,如果还有什么疑问或者建议,可以多多交流,原创文章,文笔有限,才疏学浅,文中若有不正之处,万望告知。

转载于:https://www.cnblogs.com/libin-1/p/5899842.html

你可能感兴趣的文章
【VS开发】使用VS2010创建MFC ActiveX工程项目
查看>>
【算法集中营】计算两个日期之间的天数
查看>>
react diff
查看>>
std::bind()图解
查看>>
shell格式化字符串
查看>>
浙江金华 图论整理
查看>>
大数运算——加法减法
查看>>
centos swap
查看>>
从实践认识修饰符
查看>>
网络程序服务端维持10000个对象
查看>>
QTableWidget嵌入QpushButton后定位QpushButton
查看>>
自己动手实现java数据结构(二) 链表
查看>>
Virtualbox安装Windows 8.1遇到0x000000C4错误解决办法 - 转
查看>>
一个简单的管理Web站点文件的页面程序(修改版)
查看>>
蛋疼的mysql_ping()以及MYSQL_OPT_RECONNECT
查看>>
POI读写Word docx文件
查看>>
Maven项目的SSM框架搭建
查看>>
live555 解析
查看>>
sqlserver varchar和Nvarchar区别
查看>>
业务系统设计
查看>>