弹性盒子
网页的用户界面(User Interface 缩写 UI)包括两个部分:
- 第一部分是视觉要素,如色彩、字体和图片等。
- 第二部分是这些元素的排列和定位。
CSS3 引入了 Flexible Box,简称 flexbox(弹性盒子),它特别适合用来创建弹性的页面布局。弹性布局以一种可预见的方式排列元素,使其适用于不同尺寸的设备。虽然这是个新东西,但所有现代浏览器都已经支持 flexbox。
使用 display: flex 定位两个盒子
只要在一个元素的 CSS 中添加display: flex;
,就可以使用其他 flex 属性来构建响应式页面了。
#box-container { height: 500px; display: flex; }
#box-1 { background-color: dodgerblue; width: 50%; height: 50%; }
#box-2 { background-color: orangered; width: 50%; height: 50%; }
使用 flex-direction 属性
添加了display: flex
的元素会成为 flex 容器。只要把flex-direction
属性添加到父元素,并设置其为 row
或 column
即可轻易横或竖排列它的子元素。设为 row 可以让子元素水平排列,column 可以让子元素垂直排列。默认值为 row。
flex-direction: row-reverse; // 一行逆序排列
使用 flex-direction 创建多行,使用flex-direction
属性可以把子元素排列成一行。这个属性告诉 CSS 需要将这个元素的子元素水平排列。
使用 flex-direction 属性创建一列,#box-container { display: flex; height: 500px; flex-direction: column; }
使用 flex-direction 创建多列, header .profile-name { display: flex; flex-direction: column; margin-left: 10px; }
使用 justify-content 属性对齐元素
flex 容器里的 flex 子元素有时不能充满整个容器,所以我们需要告诉 CSS 如何以特定方案排列和调整 flex 子元素。justify-content
属性就是处理这个问题。
把 flex 容器设为一个行,它的子元素会从左到右逐个排列,把 flex 容器设为一个列,它的子元素会从上到下逐个排列。子元素排列的方向被称为 main axis(主轴)。对于行,主轴水平贯穿每一个项目;对于列,主轴垂直贯穿每一个项目。
关于 flex 子元素在主轴排列方式,可以选择以下值:其中一个很常用的是justify-content: center;
,可以让 flex 子元素排列在 flex 容器中间。其他可选值还有:
-
flex-start
:从 flex 容器的前端开始排列项目。对行来说是把项目都靠左放,对于列是把项目都靠顶部放。 -
flex-end
:从 flex 容器的后端开始排列项目。对行来说是把项目都靠右放,对于列是把项目都靠底部放。 -
space-between
:项目间保留一定间距地在主轴排列。第一个和最后一个项目会被挤到容器边沿。例如,在行中第一个项目会紧贴着容器左侧,最后一个项目会紧贴着容器右侧,然后其他项目均匀排布。 -
space-around
:与space-between相似,但头尾两个项目不会紧贴容器边缘,空间会均匀分布在所有项目两边
使用 align-items 属性对齐元素
align-items
属性与justify-content
类似。justify-content
属性使 flex 子元素沿主轴排列。行的主轴是水平线,列的主轴是垂直线。
Flex 容器中,与主轴垂直的叫做 cross axis(交叉轴)。行的交叉轴是垂直的,列的交叉轴是水平的。CSS 提供了align-items
属性,可以用于在交叉轴调整 flex 子元素。对于行,它规定了项目在容器中应该靠上还是靠下,而对于列,就是靠左或靠右。
align-items
的可选值包括:
-
flex-start
:从 flex 容器的前端开始排列项目。对行来说是把项目都靠顶部放,对于列是把项目都靠左放。 -
flex-end
:从 flex 容器的后端开始排列项目。对行来说是把项目都靠底部放,对于列是把项目都靠右放。 -
center
:把项目的位置调整到中间。对于行,垂直居中(项目上下方空间相等)。对于列,水平居中(项目左右方空间相等)。 -
stretch
:拉伸项目,填满 flex 容器。例如,排成行的项目从容器顶部拉伸到底部。 -
baseline
:基线对齐地排列。基线是字体相关的概念,可以认为字体坐落在基线上。
使用 flex-wrap 属性包裹一行或一列
CSS flexbox 有一个把 flex 子元素拆分为多行(或多列)的特性。默认情况下,flex 容器会调整项目大小,把它们都塞到一起。如果是行的话,所有项目都会在一条直线上。
使用flex-wrap
属性可以使项目换行。这意味着多出来的项目会被移到新的行或列。换行发生的断点由项目和容器的大小决定。
换行方向的可选值有这些:
-
nowrap
:默认值,不换行。 -
wrap
:行从上到下排,列从左到又排。 -
wrap-reverse
:行从下到上排,列从右到左排。
使用 flex-shrink 属性收缩项目
flex-shrink
属性使用之后,如果 flex 容器太小,该项目会自动缩小。当容器的宽度小于里面所有项目的宽度,项目就会自动压缩。
flex-shrink
属性接受 number
类型的值。数值越大,与其他项目相比会被压缩得更厉害。例如,如果一个项目的flex-shrink
为 1 ,另一个项目flex-shrink
为 3,那么 3 的那个与另一个相比会受到 3 倍压缩。
使用 flex-grow 属性扩展项目
与flex-shrink
相对的是flex-grow
。flex-shrink
会在容器太小时对元素作出调整。相应地,flex-grow
会在容器太大时对元素作出调整。
如果一个项目flex-grow
属性的值为 1,另一个flex-grow
为 3,那么 3 的会比 1 的扩大三倍。
使用 flex-basic 属性设置项目的初始大小
flex-basis
属性指定了项目在 CSS 进行flex-shrink
或flex-grow
调整前的初始大小。
flex-basis
属性的单位与其他 size 属性一致(px
、em
、%
等)。如果值为auto
,项目的大小依赖于自身内容。
使用 flex 短方法属性
flex
属性有一个简写方式。flex-grow
、flex-shrink
和flex-basis
属性可以在flex中一同设置。
例如,flex: 1 0 10px;
会把项目属性设为flex-grow: 1;
、flex-shrink: 0;
以及flex-basis: 10px;
。属性的默认设置是flex: 0 1 auto;
。
#box-1 { background-color: dodgerblue; height: 200px;
flex: 2 2 150px;
}
#box-2 { background-color: orangered; height: 200px;
flex: 1 1 150px;
}
/* 在容器大于 300px 时,会让#box-1填充倍率为#box-2的两倍;
在容器小于 300px 时,缩小倍率为#box-2的两倍。
300px 是两个盒子的flex-basis的值之和。 */
使用 order 属性重新排列项目
order
属性告诉 CSS flex 容器里项目的顺序。默认情况下,项目排列顺序与源 HTML 文件中顺序相同。这个属性接受数字作为参数,可以使用负数。
使用 align-self 属性
flex 子元素的最后一个属性是align-self
。这个属性允许你调整每个项目自己的对齐方式,而不是一次性设置全部项目。因为float
、clear
和vertical-align
等调整使用的属性都不能应用在 flex 子元素,所以这个属性显得十分有用。
align-self
的允许值与align-items
一样,并且它会覆盖align-items
的值。
CSS 网格
CSS Grid 帮助你轻松实现复杂的 Web 设计。它通过把 HTML 元素转换为具有行和列的网格容器,以便将子元素放置在所需要的位置。
创建
通过将属性display
的值设为grid
,使 HTML 元素变为网格容器。通过前面的操作,你可以对该容器使用与 CSS 网格(CSS Grid)相关的属性。display:grid;
在 CSS 网格中,父元素称为容器(container),它的子元素称为项(items)。
使用 grid-template-columns 添加多列, grid-template-rows 添加多行
在一个网格容器中使用grid-template-columns
属性可以添加一些列,示例如下:
.container { display: grid; grid-template-columns: 50px 50px; }
可以在网格容器中添加两列,宽度均为 50px。
grid-template-columns
属性值的个数表示网格的列数,而每个值表示对应列的宽度。
使用 grid-template-rows
添加多行,用法和 grid-template-columns
一样,grid-template-columns
。
使用 CSS 网格单位来更改列和行的大小
在 CSS 网格中,可以使用绝对定位和相对定位单位如px
和em
来确定行或列的大小。下面的单位也可以使用:
-
fr
:设置列或行占剩余空间的一个比例 -
auto
:设置列宽或行高自动等于它的内容的宽度或高度 -
%
:将列或行调整为它的容器宽度或高度的百分比,
生成三列的网格,每列宽度分别为:1fr,100px,和 2fr。grid-template-columns: 1fr 100px 2fr;
使用 grid-column-gap 创建多列之间的间距
如果需要在列与列之间添加一些间隙,我们可以使用grid-column-gap
:grid-column-gap: 10px;
使用 grid-row-gap 创建多行之间的间距,grid-row-gap: 5px;
使用 grid-gap 更快地添加间距
grid-gap
是grid-row-gap
和grid-column-gap
的简写,它更方便使用。如果grid-gap有一个值,行与行之间和列与列之间将添加等于该值的间隙。但是,如果有两个值,第一个值将作为行间隙的高度值,第二个值是列间隙的宽度值。
grid-gap: 10px 20px;
在行之间添加10px的间隙,在列之间添加20px的间隙。
使用 grid-column 来控制剩余部分
网格的假想水平线和垂直线被称为线(lines)。这些线在网格的左上角从 1 开始编号,垂直线向右、水平线向下累加计数。
你可以用grid-column属性定义网格项开始和结束的位置,进而控制每个网格项占用的列数。
grid-column: 1 / 3;
网格项从左侧第一条线开始到第三条线结束,占用两列。
grid-column: 2 / 4 ;
网格项占用网格的第 2 列和第 3 列。
使用 grid-row 来控制剩余部分,来确定行开始和结束的水平线。
grid-row: 2 / 4;
网格项占用网格的第 2 行和第 3 行。
使用 justify-self 水平对齐项目
在 CSS 网格中,每个网格项的内容分别位于被称为单元格(cell)的框内。你可以使用网格项的justify-self
属性,设置其内容的位置在单元格内沿行轴对齐的方式。默认情况下,这个属性的值是stretch
,这将使内容占满整个单元格的宽度。该 CSS 网格属性也可以使用其他的值:
-
start
:使内容在单元格左侧对齐, -
center
:使内容在单元格居中对齐, -
end
:使内容在单元格右侧对齐,
使用 align-self 垂直对齐项目
对网格项使用align-self
属性,设置网格项沿列轴对齐方式。对于该属性,能使用可用于justify-self
属性的任一个值。
使用 justify-items 水平对齐所有项目
对于这个属性你能使用justify-self
、align-self
中的所有值,与之不同的是,它将使网格中所有的网格项按所设置的方式对齐。
使用 align-items 垂直对齐所有项目
对网格容器使用align-items
属性可以给网格中所有的网格项设置沿列轴对齐的方式。
将网格划分为区域模板
你可以将网格中的一些网格单元格组合成一个区域(area),并为该区域指定一个自定义名称。你可以通过给容器加上grid-template-areas
来实现:
grid-template-areas: "header header header" "advert content content" "footer footer footer";
将顶部三个单元格合并成一个名为header
的区域,将底部三个单元格合并为一个名为footer
的区域,并在中间行生成两个区域————advert
和content
。
注意:在代码中,每个单词代表一个网格单元格,每对引号代表一行。除了自定义标签,你还能使用句点(.
)来表示一个空单元格。
使用 grid-area 属性将项目放置在网格区域中
在为网格容添加区域模板后,你可以通过添加你定义的名称将网格项放入自定义区域。为此,你需要对网格项使用grid-area
:
.item1 { grid-area: header; }
类名为item1
的网格项就被放到了header
区域里。这种情况下,网格项将使用整个顶行,因为这一行被名为 header 区域。
使用 grid-area 创建区域模板
如果网格中没有定义区域模板,你也可以像这样为它添加一个模板:
item1 { grid-area: 1/1/2/4; }
网格项将占用第 1 条和第 2 条水平线之间的行及第 1 条和第 4 条垂直线之间的列。
grid-area: 起始水平线 / 起始垂直线 / 末尾水平线 / 终止垂直线 ;
网格项将占用第 1 条和第 2 条水平线之间的行及第 1 条和第 4 条垂直线之间的列。
使用 repeat 函数减少重复
当使用grid-template-columns
和grid-template-rows
定义网格结构时,你需要为添加的每一行和每一列都输入一个值。
使用repeat方法指定行或列的重复次数,后面加上逗号以及需要重复的值。添加 100 行网格的例子,使每行高度均为 50px:grid-template-rows: repeat(100, 50px);
你还可以用 repeat 方法重复多个值,并在定义网格结构时与其他值一起使用。
grid-template-columns: repeat(2, 1fr 50px) 20px;
相当于:grid-template-columns: 1fr 50px 1fr 50px 20px;
使用 minmax 函数限制项目大小
内置函数minmax
也可以可用于设置grid-template-columns
和grid-template-rows
的值。它的作用是在网格容器改变大小时限制网格项的大小。为此,你需要指定网格项允许的尺寸范围。
grid-template-columns: 100px minmax(50px, 200px);
添加两列,第一列 100px 宽,第二列宽度最小值是 50px,最大值是 200px。
使用 auto-fill 创建弹性布局
重复方法带有一个名为自动填充(auto-fill)的功能。它的功能是根据容器的大小,尽可能多地放入指定大小的行或列。你可以通过结合auto-fill和minmax来更灵活地布局。
repeat(auto-fill, minmax(60px, 1fr));
列的宽度会随容器大小改变,在可以插入一个 60px 宽的列之前,当前行的所有列会一直拉伸。
注意:如果容器无法使所有网格项放在同一行,余下的网格项将移至新的一行。
使用 auto-fit 创建弹性布局
auto-fit
效果几乎和auto-fill
一样。不同点仅在于,当容器的大小大于各网格项之和时,auto-fill
将会持续地在一端放入空行或空列,这样就会使所有网格项挤到另一边;而auto-fit
则不会在一端放入空行或空列,而是会将所有网格项拉伸至合适的大小。
注意:如果容器无法使所有网格项放在同一行,余下的网格项将移至新的一行。
使用媒体查询创建响应式布局
通过使用媒体查询重新排列网格区域,更改网格尺寸以及重新排列网格项位置,CSS 网格能轻松地使网站更具响应性。
@media (min-width: 400px){
.container{
grid-template-columns: 50% 50%; // 分成两列,每列占50%
grid-template-rows: auto auto auto; // 分成三行,每行占 33.33%
grid-template-areas:
"header header"
"advert content"
"footer footer";
}
在网格中创建网格
将元素转换为网格只会影响其子代元素。因此,在把某个子代元素设置为网格后,就会得到一个嵌套的网格。
例如,设置类为item3
的元素的display
和grid-template-columns
属性,就会得到一个嵌套的网格。.item3 { display: grid; grid-template-columns: auto 1fr; }
项目实战
Build a Tribute Page
Build a Survey Form
Build a Product Landing Page
Build a Technical Documentation Page
a Technical Documentation Page