CSS 盒模型、BFC、IFC、GFC、FFC

基本慨念

内容区域(content)是包含元素真实内容的区域。它通常包含背景、颜色或者图片等,位于内容边界的内部,它的大小为内容宽度或 content-box 宽及内容高度或 content-box 高。

如果 box-sizing 为默认值,width、min-width、max-width、height、min-height、max-height控制内容大小。

内边距区域(padding)延伸到包围 padding 的边框。如果内容区域设置了背景、颜色或者图片,这些样式将会延伸到 padding 上。它位于内边距边界内部,它的大小为 padding-box 宽与 padding-box 高。

内边距与内容边界之间的空间可以由 padding-top、padding-right、padding-bottom、padding-left 和简写属性 padding 控制。

边框区域(border)是包含边框的区域,扩展了内边距区域。它位于边框边界内部,大小为 border-box 宽和 border-box 高。

由 border-width 及简写属性 border 控制。

外边距区域(margin)用空白区域扩展边框区域,以分开相邻元素。它的大小为 margin-box 的高宽。

外边距区域大小由 margin-top、margin-right、margin-bottom、margin-left  及简写属性 margin 控制。

注意:对于非替换行内元素来说,尽管内容周围存在内边距与边框,但其占用空间(行高)由 line-height 属性决定。

标准盒模型:width = content

IE 盒模型:width = content + 2 * padding + 2 * border

CSS 如何设置这两种模型

用到了 CSS3 的属性 box-sizing

/* 标准模型 */ 

box-sizing: content-box; 

/*IE模型*/

 box-sizing: border-box;

JS 获取宽高

通过 JS 获取盒模型对应的宽和高,有一下几种方式,以下用 dom 来表示获取的 HTML节点:

1. dom.style.width/height

    这种方式只能取到 dom 元素内联样式所设置的宽高,也就是说如果该节点的样式是在 style 标签中或外联的 CSS 文件中设置的话,通过这种方式是获取不到 dom 的宽高的。

2. dom.currentStyle.width/height

    这种方式获取的是在页面渲染完成后的结果,就是说不管是哪种方式设置的样式,都能获取到。到这种方式只有 IE 支持

3. window.getComputedStyle(dom).width/height

    这种方式的原理和 2 一样,这个可以兼容更多的浏览器,通用性更好。

4. dom.getBoundingClientRect().width/height

    这种方式是根据元素在视窗中的绝对位置来获取宽高的。

5. dom.offsetWidth/offsetHeight

    包括高度(宽度)、内边距和边框,不包括外边距。最常用,兼容性最好。

边距重叠与BFC

边距重叠

边距重叠是指两个或多个盒子(可能相邻也可能嵌套)的相邻边界重合在一起而形成一个单一边界。

两个或多个块级盒子的垂直相邻边界会重合,它们的边界宽度是相邻边界宽度的最大值。注意水平边界是不会重合的。

边距重叠例子

父子元素的边界重叠

<style type="text/css">

    .parent {

        background: #E7A1C5;

        height: 200px; width: 200px;

    }

    .child {

        background: #C8CDF5;

        height: 100px;

        width: 100px;

        margin-top: 10px;

    }

</style>

<section class="parent">

    <article class="child"></article>

</section>

期待的效果:

实际效果:

这里父元素的高度不是110px,而是100px,在这里发生了高度坍塌。原因是如果块元素的 margin-top 与它的第一个子元素的 margin-top 之间没有 border、padding等来分隔,或者块元素的 margin-bottom与它最后一个子元素的 margin-bottom 之间没有 border、padding、height、min-height、max-height 分隔,那么外边距会塌陷。子元素多余的外边距会被父元素的外边距截断。

兄弟元素的边界重叠

<style>

    #margin {

        background: #E7A1C5;

        overflow: hidden;

        width: 300px;

    }

    #margin>p {

        background: #C8CDF5;

        margin: 20px auto 30px;

    }

</style>

<section id="margin">

    <p>1</p>

    <p>2</p>

    <p>3</p>

</section>

可以看到 1 和 2 之间、2 和 3 之间的间距不是 50px ,发生了边距重叠是取它们之间的最大值 30px.

空元素的边界重叠

假设有一个空元素,它有外边距,但是没有边框或填充。在这种情况下,上外边距与下外边距就碰到了一起,它们会发生合并:

BFC 原理

解决上述问题的其中一个办法就是创建 BFC。BFC 的全称为 Block Formatting Content,即块级格式化上下文。一个 BFC 有如下特性:

1. 处于同一个 BFC 中的元素相互影响,可能会发生 margin collapse
2. BFC 在页面上是一个独立的容器,容器里面的子元素不会影响到外面的元素,反之亦然
3. 计算 BFC 的高度时,考虑 BFC 所包含的所有元素,包括浮动元素也参与计算
4. 浮动盒的区域不会叠加到 BFC 上

创建 BFC

创建 BFC 的方法如下:

1. 浮动( float 的值不为 none )
2. 绝对定位元素( position 的值为 absolute 或 fixed )
3. 行内块( display 为 inline-block )
4. 表格单元( display 为 table、table-cell、table-caption 等 HTML 表格相关的属性 )
5. 弹性盒( display 为 flex 或 inline-flex )
6. overflow 不为 visible

BFC 使用场景

1. 防止垂直 margin 重叠

父子元素的边界重叠得解决方案:

在父元素上加上 overflow: hidden; 使其成为 BFC。

.parent {

    background: #E7A1C5;

    overflow: hidden;

}

兄弟元素的边界重叠,在第二个子元素创建一个 BFC 上下文:

<section id="margin">

    <p>1</p>

    <div style="overflow:hidden;">

        <p>2</p>

    </div>

    <p>3</p>

</section>

2. 清除内部浮动

<style>

    #float {

        background: #FEC68B;

    }

    #float .float {

        float: left;

    }

</style>

<section id="float">

    <div class="float">我是浮动元素</div>

</section>

父元素 #float 的高度为 0,解决方案为:为父元素创建 BFC,这样浮动子元素的高度也会参与到父元素的高度计算:

#float {

    background: #FEC68B;

    overflow: hidden; /*这里也可以用float:left*/

}

3. 自适应两栏布局

<section id="layout">

    <style>

        #layout {

            background: red;

        }

        #layout .left {

            float: left;

            width: 100px;

            height: 100px;

            background: pink;

        }

        #layout .right {

            height: 110px;

            background: #ccc;

        }

</style>

<!--左边宽度固定,右边自适应-->

<div class="left">左</div>

<div class="right">右</div>

</section>

在这里设置右边的高度高于左边,可以看到左边超出的部分跑到右边去了,这是由于浮动框不在文档的普通流中,所以文档的普通流中的块框表现得就像浮动框不存在一样导致的。

解决方案为:给右侧元素创建一个 BFC,原理就是 BFC 不会与 float 元素发生重叠。

#layout .right {

    height: 110px;

    background: #ccc;

    overflow: auto;

}

IFC 原理

全称 Inline Formatting Contexts,也就是"内联格式化上下文"。

创建 IFC 

块级元素中仅包含内联级别元素。
当 IFC 中有块级元素插入时,会产生两个匿名块将父元素分割开来,产生两个 IFC。

IFC 的布局规则:

1. 子元素水平方向排列,并且垂直方向起点为元素顶部
2. 子元素只会计算水平方向元素空间(margin padding border),不会计算垂直方向
3. 在垂直方向上,子元素会以不同形式来对齐(vertical-align)
4. 能把一行上的框全部包含进去的一个矩形区域,被称为该行的行框(line box)。行框的宽度由包含块(containing box)和与其中的浮动元素来决定
5. IFC 中的 "line box" 一般紧贴其包含块,但 float 元素会优先排列
6. IFC 中的 "line box" 高度由 CSS 行高计算规则来确定,同个 IFC 下的多个 "line box" 高度可能会不同
7. 当内联元素的总宽度小于包含它们的  "line box" 时,其水平渲染规则由 "text-align" 属性值来决定
8. 当一个  "line box" 超过父元素的宽度时,它会被分割成多个 box,这些 box 分布在多个 "line box" 中。如果子元素未设置强制换行的情况下,"inline box" 将不可被分割,将会溢出父元素

IFC 使用场景:

1. 水平居中:当一个块要在环境中水平居中时,设置其为inline-block则会在外层产生IFC,通过text-align则可以使其水平居中。
2. 垂直居中:创建一个IFC,用其中一个元素撑开父元素的高度,然后设置其vertical-align:middle,其他行内元素则可以在此父元素下垂直居中。

GFC 原理

GFC(GridLayout Formatting Contexts)直译为"网格布局格式化上下文",当为一个元素设置display值为grid的时候,此元素将会获得一个独立的渲染区域,我们可以通过在网格容器(grid container)上定义网格定义行(grid definition rows)和网格定义列(grid definition columns)属性各在网格项目(grid item)上定义网格行(grid row)和网格列(grid columns)为每一个网格项目(grid item)定义位置和空间。 

和 table 的区别:同样是一个二维表格,gridLayout 会以更加丰富的属性来控制行列、控制对齐以及更精细的控制语义

FFC 原理

FFC(Flex Formatting Contexts)直译为"自适应格式化上下文",display 值为flex 或者 inline-flex 的元素将会生成自适应容器(flex container)。

Flex Box 由伸缩容器和伸缩项目组成。通过设置元素的 display 属性为 flex 或 inline-flex 可以得到一个伸缩容器。设置为 flex 的容器被渲染为一个块级元素,而设置为 inline-flex 的容器则渲染为一个行内元素。

伸缩容器中的每一个子元素都是一个伸缩项目。伸缩项目可以是任意数量的。伸缩容器外和伸缩项目内的一切元素都不受影响。简单地说,Flexbox 定义了伸缩容器内伸缩项目该如何布局。

参考:盒子模型 - CSS:层叠样式表 | MDN

            深入理解CSS盒模型 - 程序猿的程 - 博客园

            边距重叠与BFC - 个人文章 - SegmentFault 思否

            CSS之IFC - 前端大宝剑 - SegmentFault 思否

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容

  • 食货星球上有这么一个让人又爱又恨的团体,每天潜伏在食货好奇官群里“为非作歹”。美食素材库像个黑洞,一言不合就发吃,...
    食货君阅读 207评论 0 0
  • 不要把我对你的容忍 当作你不要脸的资本
    感谢经历258阅读 144评论 0 0
  • 今天我们去北京,坐了火车和地铁。我们去了天安门广场,天安门广场很多人挨挨挤挤的,广场中央有一个花坛,上面有各...
    栀子花开1234阅读 197评论 0 0
  • 1 写作群里抛出一个问题,平时气氛沉默的群里炸开了锅,也许是因为这个问题,可以引起大家的共鸣。宝宝们七嘴八舌的回复...
    苏青梵阅读 501评论 4 3
  • 2017.12.10 周日 高畅 孔子回到鲁国后,自六十八岁到七十三岁卒,共生活了五年。在这五年中,他跨越了自...
    畅_759c阅读 269评论 0 0