传统的 CSS 布局方式是基于盒模型(它是根据盒子与父盒子以及兄弟盒子的关系肯定大小和位置的算法),实现时依赖于 block, inline, table, position, float 这些属性,但对于一些特殊布局不易实现,好比垂直居中。css
Flexbox Layout 是一种新的布局方式,被称为弹性布局,它使得子元素(items)能够灵活的、响应式的适应父容器(flex container)的空间,即便子元素的大小未知或是动态,而且能够很容易实现元素的水平和垂直对齐。算法
盒模型是基于 block 和 inline 的流动方向进行布局,而 flex 布局则是基于弹性流方向(flex-flow directions),基本思想以下图所示。浏览器
flex 容器内的子元素会沿着 main axis(从 main-start 到 main-end )或 cross axis(从 cross-start 到 cross-end)进行布局,其中:ide
使用如下代码就能够将一个 HTML 元素指定为 flex 布局:svg
.container {
display: flex; /* or inline-flex */
}
复制代码
这个元素称为 Flex Container,内部其余 HTML 元素被称为 Flex Items。flex 容器 有 6 个属性:源码分析
肯定主轴的方向,也就是内部元素的弹性排列方向。若是不考虑换行,Flexbox 是单向布局的概念,始终将元素放置在水平行或垂直列中。布局
.container {
flex-direction: row | row-reverse | column | column-reverse;
}
复制代码
默认状况下,子元素不会换行,只会在同一行(或列)中放置,可使用此属性设置换行。flex
.container{
flex-wrap: nowrap | wrap | wrap-reverse;
}
复制代码
它是 flex-direction 和 flex-wrap 属性的缩写.ui
flex-flow: <'flex-direction'> || <'flex-wrap'>
复制代码
默认是: flex-flow: row nowrap
flexbox
定义子元素在主轴上的对齐方式。
.container {
justify-content: flex-start | flex-end | center | space-between | space-around | space-evenly;
}
复制代码
定义子元素在交叉轴上如何对齐。
.container {
align-items: stretch | flex-start | flex-end | center | baseline;
}
复制代码
定义容器内部多行元素的行对齐方式,相似于 justify-content 定义主轴内元素的对齐方式。
**注意: **当只有一行元素时,此属性不生效
.container {
align-content: flex-start | flex-end | center | space-between | space-around | stretch;
}
复制代码
Flex 容器内部的子元素也有 6 个属性,分别是:
默认状况下,元素按原始顺序排列,可是该属性能够控制元素在 Flex 容器中的显示顺序,数值越小,越靠前,默认为 0。
.item {
order: <integer>; /* default is 0 */
}
复制代码
定义元素在必要时扩大的能力s,好比容器还有剩余空间。默认为 0,有空间也不扩大。
若是全部元素都设置为 1,那么会均摊容器剩余的空间;若是其中一个元素值为 2,那么它占的剩余空间是其余元素的两倍(至少尽可能会这样作)。
**注意: **负值无效。
.item {
flex-grow: <number>; /* default 0 */
}
复制代码
定义元素在必要时缩小的能力,默认为 1,即若是空间不足,该元素将会缩小。
相对的,若是全部元素此属性值都为 1,空间不足时会等比例缩小;若是有一个元素值为 0,其余为 1,空间不足时,前者不缩小。
**注意: **负值无效。
.item {
flex-shrink: <number>; /* default 1 */
}
复制代码
定义在分配容器剩余空间以前,如何判断子元素的大小,浏览器根据此属性计算主轴剩余空间的大小。
值可设为指定长度(如 20%, 5rem)或关键字(auto)。默认为 auto,元素的大小依据 main size(主尺寸),若是 main size 为 auto,则使用元素内容的最大长度。
.item {
flex-basis: <length> | auto; /* default auto */
}
复制代码
子元素在弹性扩大时, flex-basis 的不一样可能会形成不同的结果,下图分别是**"绝对"(从0开始算起)弹性和"相对"(从元素内容长度开始算起)**弹性的差别结果。
其中三个子元素 flex-grow 的值为 1:1:2:
因为浏览器计算主轴剩余空间的大小不一样,因此致使元素弹性结果的不一样。
该属性是 flex-grow, flex-shrink 和 flex-basis 的简写,其中第二个和第三个参数( flex-shrink 和 flex-basis )是可选的.
.item {
flex: none | [ <'flex-grow'> <'flex-shrink'>? || <'flex-basis'> ]
}
复制代码
默认是 flex: 0 1 auto
, 建议优先使用此属性,它有两个简便使用的关键字: auto(1 1 auto) 和 none(0 0 auto).
容许元素覆盖默认的或是使用 align-items 指定的对齐方式。
.item {
align-self: auto | flex-start | flex-end | center | baseline | stretch;
}
复制代码
除了 auto, 其余属性值都与 align-items 一致。
**注意: **float, clear 和 vertical-align 对 flex 容器内的元素不生效。
其中:
最后回到题目的问题,经常使用的垂直居中,只需将 flex 容器的 align-items 和 justify-content 属性值设为 center 便可。
参考: A Complete Guide to Flexbox
搜索公众号「顿悟源码」获取更多源码分析和造的轮子。