深入理解CSS弹性盒模型flex
前面的话
CSS3 引入了一种新的布局模型——flex 布局。flex 是 flexible box 的缩写,一般称之为弹性盒模型。和 CSS3 其他属性不一样,flexbox 并不是一个属性,而是一个模块,包括多个 CSS3 属性。flex 布局提供一种更加有效的方式来进行容器内的项目布局,以适应各种类型的显示设备和各种尺寸的屏幕
版本更迭
flexbox 布局的语法规范经过几年发生了很大的变化。从 2007 年 07 月,flex 第一版本的工作草案发布,到 2012 年 09 月,flex 最新版本成为候选推荐。flex 主要经历了三个版本
【1】旧版本 display:box | inline-box;
IE 浏览器不支持,windows 下的 safari 浏览器只支持旧版本的写法且需要添加前缀,移动端可以兼容到 andriod2.1-4.3 和 ios3.2-6.1 也需要添加前缀
【2】混合版本 display:flexbox | inline-flexbox;
该版本只有 IE10 支持,且需要添加前缀 -ms-
【3】新版本 display: flex | inline-flex
该版本兼容 IE11+、firefox、safari、chrome、opera 及移动端,但移动端 ios7.1-8.4 需要添加前缀 -webkit-
display
要让一个元素变成伸缩容器,需要使用 display 属性。采用 flex 布局的元素,称为伸缩容器 (flex container),容器内的子元素称为伸缩项目 (flex item)
[注意] 浏览器会将任何直接在伸缩容器里的连续文字块包起来成为匿名伸缩项目
使用 flex 布局实现上是使元素 FFC 化 (flex formatting context 伸缩格式化上下文),FFC 是普通流的一种。而浮动流和定位流以及 CSS 其他属性对 FFC 是有影响的,主要表现在以下几点:
[1]float、clear 和 vertical-align 属性在伸缩项目上没有效果
[2] 伸缩容器的 margin 与其内容的 margin 不会重叠
[3]text-align 属性在伸缩容器上没有效果,因为其只可应用于块级 block 容器
[4] 另外,columns 属性伸缩容器上没有效果
弹性盒模型的两种容器块级伸缩容器和内联伸缩容器的区别类似于 block 和 inline-block 的区别,一个独占一行,另一个非独占一行
//弹性盒模型: 块级伸缩容器 | 内联伸缩容器 //新版本 display: flex | inline-flex; //混合版本 display: flexbox | inline-flexbox; //旧版本 display: box | inline-box;
基本概念
伸缩容器默认存在两条轴: 水平的主轴 (main axis) 和垂直的侧轴 (cross axis)
[注意] 主轴方向不一定是水平的,它主要取决于 flex-direction 属性
主轴起点叫 main start,主轴终点叫 main end;侧轴起点叫 cross start,侧轴终点叫 cross end
伸缩项目默认沿主轴排列。单个伸缩项目占据的主轴空间叫 main size ,占据的侧轴空间叫 cross size
[注意] 伸缩项目的 main size 和 cross size 主要由宽度或高度决定
伸缩容器
以下 6 个属性作用在伸缩容器上
【1】伸缩流方向: 指定主轴的方向 (即伸缩项目在伸缩容器中的排列方向)
//伸缩流方向: 水平方向 | 反向水平 | 垂直方向 | 反向垂直 //新版本同混合版本 flex-direction: row[默认] | row-reverse | column | column-reverse //旧版本 box-orient: horizontal(水平) |vertical(垂直) |inline-axis[默认](内联轴方向) |block-axis(块级轴方向) box-direction: normal(正常) | reverse(反向)
[注意] 伸缩流方向与 direction 和 writing-mode 有关系
【2】伸缩流换行: 指定伸缩项目溢出伸缩容器时是否换行
//伸缩行换行: 不换行 | 换行 | 反转换行 //新版本同混合版本 flex-wrap: nowrap[默认] | wrap | wrap-reverse //旧版本,没有浏览器支持 box-lines 属性,所以在旧版本中无法实现伸缩项目换行显示 box-lines: single[默认] | multiple | N/A
[注意] 此时,CSS 允许使用 overflow 属性来处理溢出内容的显示方式
[注意] 伸缩项目的排列顺序同样与 direction 和 wrinting-mode 有关系
【3】伸缩流: 伸缩流方向与伸缩行换行的缩写
//伸缩流: 伸缩流方向 | 伸缩行换行 //新版本同混合版本 flex-flow: <flex-direction> | <flex-wrap> [默认值] flex-flow: row nowrap //旧版本无对应属性
【4】主轴对齐: 用来设置伸缩容器当前行伸缩项目在主轴方向的对齐方式,指定如何在伸缩项目之间分布伸缩容器额外空间
当一行上的所伸缩项目不能伸缩或可伸缩已达到最大长度时,这一属性才会对伸缩容器额外空间进行分配。当伸缩项目溢出某一行时,这一属性也会在项目的对齐上施加一些控制
//主轴对齐方式: 左对齐 | 居中对齐 | 右对齐 | 两端对齐 | 扩散对齐 //新版本 justify-content: flex-start[默认] | center | flex-end | space-between | space-around //混合版本 flex-pack: start[默认] | center | end | justify | distribute //旧版本 box-pack: start[默认] | center | end | justify | N/A
[注意] 主轴对齐方式与 direction、writing-mode、flex-flow 都有关
【5】侧轴对齐: 用来设置伸缩容器当前行在侧轴方向的对齐方式
//侧轴对齐方式: 顶边对齐 | 中间对齐 | 底部对齐 | 基线对齐 | 伸缩项目拉伸填充整个伸缩容器 //新版本 align-items: flex-start | center | flex-end | baseline | stretch[默认] //混合版本 flex-align: start | center | end | baseline | stretch[默认] //旧版本 box-align: start | center | end | baseline | stretch[默认]
[注意] 如果伸缩项目有 width/height 属性将优先于侧轴对齐为拉伸的方式
[注意] 侧轴对齐方式与 direction、writing-mode、flex-flow 都有关
【6】堆栈伸缩行: 指定多个伸缩项目行在侧轴的对齐方式
//侧轴对齐方式: 顶边对齐 | 中间对齐 | 底部对齐 | 两端对齐 | 扩散对齐 | 伸缩项目拉伸填充整个伸缩容器 //新版本 align-content: flex-start | center | flex-end | space-between | space-around | stretch[默认] //混合版本 flex-line-pack: start | center | end | justify | distribute | stretch[默认] //旧版本无对应属性
[注意] 该属性只有在 flex-wrap:wrap | wrap-reverse; 且伸缩项目存在多行时才生效
[注意] 堆栈伸缩行与 direction、writing-mode、flex-flow 都有关
伸缩项目
一个伸缩项目就是伸缩容器的一个子元素。伸缩容器中的文本也被视为一个伸缩项目。以下 6 个属性设置在伸缩项目上。
【1】自身侧轴对齐方式: 单个伸缩项目在侧轴的对齐方式,该属性可以覆盖伸缩容器的侧轴对齐方式
[注意] 对于匿名伸缩项目,align-self 的值永远与其关联的伸缩容器的 align-items 的值相同
//侧轴对齐方式: 自动 | 顶边对齐 | 中间对齐 | 底部对齐 | 基线对齐 | 伸缩项目拉伸填充整个伸缩容器 //新版本 align-self: auto[默认] | flex-start | center | flex-end | baseline | stretch //混合版本 flex-item-align: auto[默认] | start | center | end | baseline | stretch //旧版本无对应属性
[注意] 如果 align-self 的值为 auto,则其计算值为伸缩项目的伸缩容器的 align-items 值
[注意] 如果伸缩项目的任一个侧轴上的外边距为 auto,则该伸缩项目在伸缩容器的剩余空间内居中对齐,且 align-self 没有效果。
【2】伸缩基准值: 伸缩项目在主轴方向上的初始大小
//新版本 flex-basis: <length> | auto[默认] //混合版本 positive-flex: <number>[默认为 1] //旧版本无对应属性
如果 flex-basis 的值为 0,表示伸缩项目在主轴方向上的初始大小为 0,分配所有空间;如果 flex-basis 的值为 auto,表示伸缩项目在主轴方向上的初始大小为设置宽度 (如果没有设置宽度,则为内容宽度),再分配剩余空间
[注意]flex-basis 的 <length> 值可以是一个数字后面跟着 px、em 等单位,也可以是一个百分数,相对于其父伸缩容器的主轴长度
【3】扩展比率: 当伸缩容器的额外空间为正值时,此伸缩项目相对伸缩容器里其他伸缩项目能扩展的空间比例
//新版本 flex-grow: <number>[默认为 0] //混合版本 positive-flex: <number>[默认为 0] //旧版本无对应属性
若 flex-grow 的值为 0 表示即使存在剩余空间也不放大;若所有项目的 flex-grow 属性都为 1,则它们将等分剩余空间 (如果有的话);若一个项目的 flex-grow 属性为 2,其他项目都为 1,则前者占据的剩余空间将比其他项多一倍
【4】收缩比率: 当伸缩容器的额外空间为负值时,此伸缩项目相对于伸缩容器里其他伸缩项目能收缩的空间比例
//新版本 flex-shrink: <number>[默认为 1] //混合版本 negative-flex: <number>[默认为 0] //旧版本无对应属性
如果所有项目的 flex-shrink 属性都为 1,当空间不足时,都将等比例缩小。如果一个项目的 flex-shrink 属性为 0,其他项目都为 1,则空间不足时,前者不缩小。
[注意] 伸缩基准值、扩展比率和收缩比率都可以为小数,但不能为负数
【5】伸缩性: 是扩展比率、收缩比率和伸缩基准值的缩写
flex: none => flex: 0 0 auto;// 表示宽度为原始宽度,不发生扩展或收缩 flex: auto => flex: 1 1 auto;// 表示除了占据原先的宽度外,还要分配剩余宽度 (包括扩展或收缩) flex: 0 => flex: 0 1 0%;// 表示收缩为最小宽度 flex: 1 => flex: 1 1 0%;// 表示分配所有宽度 (包括扩展或收缩) flex: 0 auto => flex: 0 1 auto;(默认值)// 表除了占据原先的宽度外,还要分配剩余宽度 (只收缩,不扩展) flex: 0 1 => flex: 0 1 0%;
[注意] 当 flex 为关键字 none 或存在 auto 时,flex-basis 为 auto;若 flex 只有数字值,则 flex-basis 为 0%;
//新版本 flex: none | [<flex-grow> <flex-shrink>? || <flex-basis>] //混合版本 flex: none | [<pos-flex> <neg-flex>? || <preferred-size>] //旧版本 box-flex: <number>
【6】显示顺序: 定义伸缩项目的排列顺序,数值越小,排列越靠前
[注意]伸缩容器中的伸缩项目默认显示顺序是遵循文档在源码中出现的先后顺序 (HTML 文档的 DOM 结构中的先后顺序)
//新版本 order: <number>[默认为 0] //混合版本 flex-order: <number>[默认为 0] //旧版本 box-ordinal-group: <integer>[默认为 1]
[注意]order 的属性值可以是负数,但不能是小数