写在前面,本文将同步发布于Blog、掘金、segmentfault、知乎等处,若是本文对你有帮助,记得为我获得个人我的技术博客项目给个star哦。html
许多开发者认为一个元素的包含块就是他的父元素的内容区,其实这是错误的(至少不彻底正确)!
一个元素的尺寸和位置常常受其包含块的影响。大多数状况下,包含块就是这个元素最近的祖先块元素的内容区,但也不是老是这样。
下面咱们看看盒模型:
当浏览器展现一个文档的时候,对于每个元素,它都产生了一个盒子。每个盒子都被划分为四个区域:git
包含块有分为根元素包含块和其余元素的包含块。github
根元素html的包含块是一个矩形,叫作初始化包含块(initial containing block)。
能够看到html外面还有空间,这个包含html的块就被称为初始包含块(initial containing block),它是做为元素绝对定位和固定定位的参照物。
对于连续媒体设备(continuous media),初始包含块的大小等于视口viewpor的大小,基点在画布的原点(视口左上角);对于分页媒体(paged media),初始包含块是页面区域(page area)。初始包含块的direction属性与根元素的相同。segmentfault
大多数状况下,包含块就是这个元素最近的祖先块元素的内容区,但也不是老是这样,下面就来学习如何肯定这些元素的包含块。浏览器
肯定包含块的过程彻底依赖于这个包含块的 position 属性,大体分为下列场景:学习
A transform or perspective value other than none
A will-change value of transform or perspective
A filter value other than none or a will-change value of filter (only works on Firefox).flex
元素的尺寸和位置常常受其包含块的影响。对于一个绝对定位的元素来讲(他的 position 属性被设定为 absolute 或 fixed),若是它的 width, height, padding, margin, 和 offset 这些属性的值是一个比例值(如百分比等)的话,那这些值的计算值就是由它的包含块计算而来的。
简单来讲,若是某些属性被赋予一个百分值的话,它的计算值是由这个元素的包含块计算而来的。这些属性包括盒模型属性和偏移属性:spa
下面示例公用HTML代码code
<body> <section> <p>This is a paragraph!</p> </section> </body>
CSS代码orm
body { background: beige; } section { display: block; width: 400px; height: 160px; background: lightgray; } p { width: 50%; /* == 400px * .5 = 200px */ height: 25%; /* == 160px * .25 = 40px */ margin: 5%; /* == 400px * .05 = 20px */ padding: 5%; /* == 400px * .05 = 20px */ background: cyan; }
在这里,这个P标签position为默认的static,因此它的包含块为Section标签,经过咱们的判断规则一来肯定。
CSS代码
body { background: beige; } section { display: inline; background: lightgray; } p { width: 50%; /* == half the body's width */ height: 200px; /* Note: a percentage would be 0 */ background: cyan; }
在这里,这个P标签position为默认的static且它的父标签Section的display为inline,因此P标签的包含块为body标签,经过咱们的判断规则一来肯定。
CSS代码
body { background: beige; } section { transform: rotate(0deg); width: 400px; height: 160px; background: lightgray; } p { position: absolute; left: 80px; top: 30px; width: 50%; /* == 200px */ height: 25%; /* == 40px */ margin: 5%; /* == 20px */ padding: 5%; /* == 20px */ background: cyan; }
在这里,这个P标签position为absolute且它的父标签Section的transform不为none,因此P标签的包含块为Section标签,经过咱们的判断规则四来肯定。
CSS代码
body { background: beige; } section { position: absolute; left: 30px; top: 30px; width: 400px; height: 160px; padding: 30px 20px; background: lightgray; } p { position: absolute; width: 50%; /* == (400px + 20px + 20px) * .5 = 220px */ height: 25%; /* == (160px + 30px + 30px) * .25 = 55px */ margin: 5%; /* == (400px + 20px + 20px) * .05 = 22px */ padding: 5%; /* == (400px + 20px + 20px) * .05 = 22px */ background: cyan; }
在这里,这个P标签position为absolute且它的父标签Section的position不为static,因此P标签的包含块为Section标签的padding边缘算起(前提是不能 box-sizing设置为border-box),经过咱们的判断规则二来肯定。
CSS代码
body { background: beige; } section { width: 300px; height: 300px; margin: 30px; padding: 15px; background: lightgray; } p { position: fixed; width: 50%; /* == (50vw - (width of vertical scrollbar)) */ height: 50%; /* == (50vh - (height of horizontal scrollbar)) */ margin: 5%; /* == (5vw - (width of vertical scrollbar)) */ padding: 5%; /* == (5vw - (width of vertical scrollbar)) */ background: cyan; }
在这里,这个P标签position为fixed,因此P标签的包含块为初始包含块(viewport),经过咱们的判断规则三来肯定。