网格布局
网格布局(Grid
)是将网页划分成一个个网格单元,可任意组合不同的网格,轻松实现各种布局效果,也是目前CSS中最强大布局方案,比Flex
更强大。需要学习Flex
弹性布局的看这里。
基本概念
容器和项目
当一个 HTML 元素将 display
属性设置为 grid
或 inline-grid
(设成行内元素) 后,它就变成了一个网格容器(container)
,这个元素的所有直系子元素将成为网格元素
,称为项目(item)
。
arduino
.box {
display: grid | inline-grid;
}
注意: 设为网格布局以后,容器子元素(项目)的float
、display: inline-block
、display: table-cell
、vertical-align
等设置都将失效。
行和列
容器里面的水平区域称为"行"(row),垂直区域称为"列"(column)。
网格线
划分网格的线,称为"网格线"(grid line)。水平网格线划分出行,垂直网格线划分出列。
正常情况下,n
行有n + 1
根水平网格线,m
列有m + 1
根垂直网格线,比如三行就有四根水平网格线。
单元格
行和列的交叉区域,称为"单元格"(cell)。 正常情况下,n
行和m
列会产生n x m
个单元格。比如,3行3列会产生9个单元格。
图示中绿的背景为4个单元格,单元格各自有一个项目item。
注意:需要区分单元格和项目元素,比如一个3*3的九宫格,项目元素不一定就是9个。
容器属性
一、grid-template-columns、 grid-template-rows
grid-template-columns
属性定义每一列的列宽grid-template-rows
属性定义每一行的行高
1、固定值 px
代码:
css
.container {
display: grid;
grid-template-columns: 50px 100px 50px;
grid-template-rows: 50px 50px 50px;
}
效果:
2、百分比 %
代码:
css
.container {
display: grid;
grid-template-columns: 25% 25% 25% 25%;
grid-template-rows: 50% 50%;
}
效果:
3、repeat()
有时候,重复写同样的值非常麻烦,尤其网格很多时。这时,可以使用repeat()
函数,简化重复的值。
将上面的代码用repeat()
改写如下:
css
.container {
display: grid;
grid-template-columns: repeat(4,25%);
grid-template-rows: repeat(2,50%);
}
repeat()
接受两个参数
- 参数一:重复的次数(上例分别是列为4,行为2)
- 参数二:重复的值。
repeat()
重复某种模式也是可以的。
代码:
css
.container {
display: grid;
grid-template-columns: repeat(2, 40px 60px 80px);
grid-template-rows: repeat(2,50%);
}
上述代码是将列宽为40px、60px、80px重复2次;行高2行,等分容器的高度。
图示:
4、fr
为了方便表示比例关系,网格布局提供了fr
关键字(fraction 的缩写,意为"片段")。
代码1:
css
.container {
width: 300px
display: grid;
grid-template-columns: 1fr 2fr 1fr;
}
图示1:
同时,fr
可以与绝对长度的单位结合使用,这时会非常方便。
代码2:
css
.container {
width: 300px
display: grid;
grid-template-columns: 1fr 100px 2fr
}
上面代码表示,第二列的宽度为固定的100px,第三列的宽度是第一列的2倍。
图示2:
5、 minmax()
minmax()
函数产生一个长度范围,表示长度就在这个范围之中。它接受两个参数,分别为最小值和最大值。
css
grid-template-columns: 1fr 1fr minmax(100px, 1fr);
上面代码表示,第三列的宽度不小于100px
,不大于1fr
。
6、auto
auto
关键字表示由浏览器自己决定长度。
代码:
arduino
.container {
width: 300px
display: grid;
grid-template-columns: 50px auto 50px;
}
上面代码中,第二列的宽度为容器宽度除第一、第三列的总宽度外所占最大的宽度,除非单元格内容设置了min-width
,且这个值大于最大宽度。
图示:
7、auto-fill、auto-fit
auto-fill
与 auto-fit
直译为 自适应 与 自填充,一般用来实现自适应布局的。
代码:
css
.container {
display: grid;
grid-template-columns: repeat(auto-fill, 100px);
}
图示:
是不是有点响应式的感觉了!
除了auto-fill
,还有一个auto-fit
,我们看两者在MDN
上的解释。
auto-fill:
If the grid container has a definite or maximal size in the relevant axis, then the number of repetitions is the largest possible positive integer that does not cause the grid to overflow its grid container.
如果网格容器在相关轴上具有确定的大小或最大大小,则重复次数是最大可能的正整数,不会导致网格溢出其网格容器。
auto-fit:
Behaves the same as
auto-fill
, except that after placing the grid items any empty repeated tracks are collapsed。
行为与 auto-fill
相同,除了放置网格项目后,所有空的重复轨道都将消失。简单来说,就是如果元素数量不够放满一行,则 auto-fit
会将元素平铺,铺满一行
举个例子:
fill - 代码:
css
.container {
width: 400px;
display: grid;
grid-template-columns: repeat(auto-fill, minmax(200px , 1fr));
}
图示:
fit - 代码:
css
.container {
width: 400px;
display: grid;
grid-template-columns: repeat(auto-fit, minmax(200px , 1fr));
}
图示:
只有一个项目的情况下,容器固定宽度400px,minmax(200px , 1fr)
含义是列宽最小为200px,最大是占满整个一行。这时候,因为元素数量不够放满一行,如果再有一个或更多的元素还可以继续填充的情况下,auto-fill
就自适应为列宽200px;auto-fit
会将元素平铺填充至一行。
8、网格线名称
grid-template-columns
属性和grid-template-rows
属性里面,还可以使用方括号,指定每一根网格线的名字,方便以后的引用。
代码:
ini
.container {
display: grid;
grid-template-columns: [c1] 100px [c2] 100px [c3] 100px [c4];
grid-template-rows: [r1] 100px [r2] 100px [r3] 100px [r4];
}
上述代码表示 3*3
的九宫格,4根垂直的网格线的名称分别是c1、c2、c3、c4;4根水平的网格线名称分别是:r1、r2、r3、r4。
图示:
同时网格布局允许同一根线有多个名字,比如:
ini
grid-template-columns: [c1 c11 c111] 100px [c2 c22] 100px [c3] 100px [c4];
二、grid-template-area,grid-area(项目属性)
1、 grid-template-area
网格布局允许指定"区域"(area),一个区域由单个或多个单元格组成,用于定义网格区域名称。
代码:
css
.container {
grid-template-columns:50px 50px 50px;
grid-template-rows: 50px 50px 50px;
grid-template-areas: 'a b c'
'd e f'
'g h i';
}
以上代码定义了 3*3
的九宫格,每个单元格的区域名称分别是a~i
。
图示:
2、grid-area (项目属性)
grid-area
是项目属性,指定项目放在哪一个区域。一般搭配grid-template-area
使用。为了方便理解,这里放一起说明。
示例一: 基于上述的3*3
的网格,需求是把item1
放到中的单元格。
css
.item-1 {
grid-area: e;
}
中间区域的单元格名称是e
,直接指定项目item1
的grid-area
属性值为e
即可。
图示:
还可以通过项目指定区域,合并单元格。
示例二:3*3单元格,5个项目元素,第一行和第二行的单元格分别合并。
css
.container {
grid-template-areas: 'a a a'
'b b b'
'c d e';
}
上述代码表示有3*3
九宫格,其中第一行和第二行分别指定同一个区域。
css
.item-1 {
grid-area: a;
}
.item-2 {
grid-area: b;
}
上述代码表示将Itme1
放到区域为a
的位置,即占满第一行。将Itme2
放到区域为b
的位置,即占满第二行。
图示:
示例三:练习一个常见的布局
css
.container {
grid-template-areas: 'header header header'
'side main main'
'footer footer footer';
}
通过名称就知道有4个项目元素,再指定各个项目所在网格位置:
css
.item-1 {
grid-area: header;
}
.item-2 {
grid-area: side;
}
.item-3 {
grid-area: main;
}
.item-4 {
grid-area: footer;
}
注意 :所有指定项目区域或其他方式来合并单元格,只能合并矩形,L型或其他形状均无效。
三、column-gap、row-gap、gap
column-gap
:列与列的间隔(列间距)。row-gap
:行与行的间隔(行间距)gap
属性是grid-column-gap
和grid-row-gap
的合并简写形式。如设置一个值则 列间距 = 行间距。
四、grid-auto-flow
划分网格以后,容器的子元素item
会按照顺序,自动放置在每一个网格。grid-auto-flow
就是定义单元格排列顺序和排列方式。
1、row、column
默认值row
,即放置顺序是"先行后列",即先填满第一行,再开始放入第二行。如下图示:
css
.container {
width: 200px;
padding: 3px;
display: grid;
grid-template-columns: 1fr 1fr 1fr;
grid-template-rows: 1fr 1fr 1fr;
gap: 3px;
}
这个顺序由grid-auto-flow
属性决定,默认值是row
,即"先行后列"。也可以将它设成column
,变成"先列后行"。
css
grid-auto-flow: column;
2、row dense、column dense
代码:
css
.container {
padding: 3px;
display: grid;
grid-template-columns: 50px 50px 50px;
grid-template-rows: 50px 50px 50px;
grid-auto-columns: 50px;
grid-auto-rows: 50px;
gap: 3px;
}
.item1 , itme2 {
grid-column-start: 1;
grid-column-end: 3;
}
上述代码就是将item1
和itme2
占据2个单元格(语法后续会讲到)。在默认的grid-auto-flow: row
情况下,产生下面这样的布局:
图示中,item1
后是空白,因为itme2
的水平网格线从1开始到3结束,即占了2个单元格, 而itme3
排序默认在itme2
后面。
现将grid-auto-flow: row dense
,表示"先行后列",并且尽可能紧密填满,尽量不出现空格。效果如下图示:
五、 justify-items、align-items、place-items
justify-items
属性设置单元格内容的水平位置(左中右),align-items
属性设置单元格内容的垂直位置(上中下)。
sql
.container {
justify-items: start | end | center | stretch;
align-items: start | end | center | stretch;
}
1、justify-items
属性值 | 描述 | 图示 |
---|---|---|
start | 左对齐。 | |
end | 右对齐 | |
center | 居中对齐 | |
stretch | 拉伸,占满单元格的整个宽度(默认值)。 |
2、align-items
属性值 | 描述 | 图示 |
---|---|---|
start | 上对齐。 | |
end | 下对齐 | |
center | 居中对齐 | |
stretch | 拉伸,占满单元格的整个宽度(默认值)。 |
3、place-items
place-items
属性是align-items
属性和justify-items
属性的合并简写形式。
xml
place-items: <align-items> <justify-items>;
如果省略第二个值,则浏览器认为与第一个值相等。
容器只有一个元素下,居中样式又有一种写法了:
css
.container {
display: grid;
place-items: center;
}
六、justify-content、align-content、place-content
justify-content
属性是整个内容区域在容器里面的水平位置(左中右及空白分配),align-content
属性是整个内容区域的垂直位置(上中下及空白分配)。其效果跟Flex
布局一样。
sql
.container {
justify-content: start | end | center | stretch | space-around | space-between | space-evenly;
align-content: start | end | center | stretch | space-around | space-between | space-evenly;
}
1、justify-content
属性值 | 图示 | 属性值 | 图示 |
---|---|---|---|
start | space-around | ||
end | space-between | ||
center | space-between |
2、 align-content
属性值 | 图示 | 属性值 | 图示 |
---|---|---|---|
start | space-around | ||
end | space-between | ||
center | space-evenly |
3、place-content
place-content
属性是align-content
属性和justify-content
属性的合并简写形式。
xml
place-content: <align-content> <justify-content>
如果省略第二个值,浏览器就会假定第二个值等于第一个值。
同样的,若容器只有一个元素下,居中样式也可以这样:
css
.container {
display: grid;
place-content: center;
}
七、 grid-auto-columns、grid-auto-rows
简单来说,定义容器中多余网格的列宽、行高。比如,网格设置了2行3列,但项目元素一共有8个,如下:
代码一
css
.container {
grid-template-columns:50px 50px 50px;
grid-template-rows: 50px 50px ;
}
图示一:
也或者,改变item7
和item8
的水平网格线的起始位置来改变其位置,具体怎么改变下文会讲到。
代码二:
css
.container {
grid-template-columns:50px 50px 50px;
grid-template-rows: 50px 50px 50px;
}
.item7 {
grid-row-start: 4;
grid-column-start: 2;
}
.item8 {
grid-row-start: 5;
grid-column-start: 3;
}
图示二:
上面2个例子都能看到在超出设置的网格后,单元格变得奇怪了,行高不受控制了。这时候就可以设置grid-auto-rows
属性来固定行高:
grid-auto-columns
属性和grid-auto-rows
属性用来设置,浏览器自动创建的多余网格的列宽和行高。它们的写法与grid-template-columns
和grid-template-rows
完全相同。如果不指定这两个属性,浏览器完全根据单元格内容的大小,决定新增网格的列宽和行高。
八、 grid-template
该属性是 grid-template-columns
、grid-template-rows
、grid-template-areas
这三个属性的简写形式。
九、grid
该属性是grid-template-rows
、grid-template-columns
、grid-template-areas
、 grid-auto-rows
、grid-auto-columns
、grid-auto-flow
这六个属性的简写形式。
项目属性
一、grid-column-start、grid-column-end、grid-column
grid-row-start、grid-row-end、grid-row
项目的位置是可以指定的,除了使用上述说明过的grid-template-area
和grid-area
搭配使用来指定,也可以指定项目的四个边框,分别定位在哪根网格线来指定项目的位置和占用的单元格。
grid-column-start
:定义项目左边框所对齐的垂直网格线grid-column-end
:定义项目右边框所对齐的垂直网格线。grid-column
:是grid-column-start
和grid-column-end
的简写形式(start / end)
grid-row-start
:定义项目上边框所对齐的水平网格线。grid-row-end
:定义项目下边框所对齐的水平网格线。grid-row
:是grid-row-start
和grid-row-end
的简写形式(start / end)
示例一:基于九宫格改造下:
css
.item-1 {
grid-column-start: 2;
grid-column-end: 4;
/* 简写 */
grid-column: 2 / 4;
}
.item-2 {
grid-row-start: 1;
grid-row-end: 4;
/* 简写 */
grid-row: 1 / 4;
}
上述代码表示:简单来说,将Item1
的从第2根垂直网格线开始,到第2根垂直网格线结束,效果就是Item1
右移一个单元格,且占用2个单元格;将Item2
的从第1根水平网格线开始,到第4根水平网格线结束,效果就是Item2
上移一个单元格,且占用3个单元格。
图示:
示例二:常见的header-side-main-footer
布局
css
.item1 {
grid-column: 1 / 4;
}
.item2 {
grid-column: 1 / 2;
}
.item3 {
grid-column: 2 / 4;
}
.item4 {
grid-column: 1 / 4;
}
图示:
示例二:存在空格情况
css
.item-1 {
grid-column: 2 / 4;
}
上述将Item1
右移一个单元格,且占用2个单元格。
图示:
上图中,只指定了Item1
的左右边框,没有指定上下边框,所以会采用默认位置,即上边框是第一根水平网格线,下边框是第二根水平网格线。
除了Item1
以外,其他项目都没有指定位置,由浏览器自动布局,这时它们的位置由容器的grid-auto-flow
属性决定,这个属性的默认值是row
,因此会"先行后列"进行排列。我们可以通过设置这个属性的值为column
、row dense
或column dense
来改变布局。
比如设置 grid-auto-flow:row dense
,尽可能填满。效果如下:
span 关键字
这四个属性的值还可以使用span
关键字,表示"跨越",即左右边框(上下边框)之间跨越多少个网格。
示例:header-side-main-footer
布局
css
.item1 {
grid-column-start: span 3;
}
.item2 {
grid-column: 1 / 2;
}
.item3 {
grid-column: 2 / span 2;
}
.item4 {
grid-column-end: span 3;
}
每个项目的属性写法均有差异,可行体会。
二、grid-area
见上文
justify-self、align-self、place-self
justify-self
:定义项目在单元格区域内的水平位置(左中右)align-self
:定义项目在单元格区域内的垂直位置(上中下)place-self
:同时定义项目在单元格区域内的水平和垂直位置,是 align-self 和 justify-self 的简写。若省略第二个值,默认两个值相等。
- start:对齐单元格的起始边缘。
- end:对齐单元格的结束边缘。
- center:单元格内部居中。
- stretch:拉伸,占满单元格的整个宽度(默认值)。
justify-self
定义单个项目在单元格内的水平位置(左中右)
属性值 | 图示 | 属性值 | 图示 |
---|---|---|---|
start | center | ||
end | stretch |
align-self
定义单个项目在单元格内的垂直位置(上中下)
属性值 | 图示 | 属性值 | 图示 |
---|---|---|---|
start | center | ||
end | stretch |
CSS的Grid网格布局差不多到这里就结束了,个人的学习总结,有误的地方欢迎指正!