Skip to content

Grid 详解

Grid 概述

Grid 布局即网格布局,较擅长将一个页面划分为几个主要区域,以及定义这些区域的大小、位置、层次等关系,号称是最强大的的 CSS 布局方案,是目前唯一一种 CSS 二维布局。利用 Grid 布局,我们可以轻松实现类似下面的布局:

Grid VS Flex

讲到布局,我们就会想到 Flex 布局,甚至有人认为既然有 Flex 布局了,似乎没有必要去了解 Grid 布局。但 Flex 布局和 Grid 布局有实质的区别,那就是 Flex 布局是一维布局,Grid 布局是二维布局。Flex 布局一次只能处理一个维度上的元素布局,一行或者一列。Grid 布局是将容器划分成了“行”和“列”,产生了一个个的网格,我们可以将网格元素放在与这些行和列相关的位置上,从而达到我们布局的目的。

Grid 布局远比 Flex 布局强大!

Grid 基础概念

  1. 容器和项目:我们通过在元素上声明 display:griddisplay:inline-grid 来创建一个网格容器。一旦我们这样做,这个元素的所有直系子元素将成为网格项目。比如上面 .wrapper 所在的元素为一个网格容器,其直系子元素将成为网格项目。
  2. 网格轨道:grid-template-columnsgrid-template-rows 属性来定义网格中的行和列。容器内部的水平区域称为行,垂直区域称为列。下图中 OneTwoThree 组成了一行,OneFour 则是一列网格单元:一个网格单元是在一个网格元素中最小的单位, 从概念上来讲其实它和表格的一个单元格很像。下图中 OneTwoThreeFour...都是一个个的网格单元
  3. 网格线:划分网格的线,称为"网格线"。应该注意的是,当我们定义网格时,我们定义的是网格轨道,而不是网格线。Grid 会为我们创建编号的网格线来让我们来定位每一个网格元素。m 列有 m + 1 根垂直的网格线,n 行有 n + 1 跟水平网格线。比如下图示例中就有 4 根垂直网格线。一般而言,是从左到右,从上到下,1,2,3 进行编号排序。当然也可以从右到左,从下到上,按照 -1,-2,-3...顺序进行编号排序 网格线
  4. 示例&代码

容器属性

display

我们通过在元素上声明 display:griddisplay:inline-grid 来创建一个网格容器。声明 display:grid 则该容器是一个块级元素,设置成 display: inline-grid 则容器元素为行内元素。

css
.wrapper {
  display: grid;
}
.inline-wrapper {
  display: inline-grid;
}

grid-template-columns 和 grid-template-rows

grid-template-columns 属性设置列宽,grid-template-rows 属性设置行高。

  1. 固定列宽和行高
css
.wrapper {
  display: grid;
  /*  声明了三列,宽度分别为 200px 100px 200px */
  grid-template-columns: 200px 100px 200px;
  grid-gap: 10px;
  /*  声明了两行,行高分别为 50px 50px  */
  grid-template-rows: 50px 50px;
}
  1. repeat() 函数

repeat() 函数可以简化重复的值。该函数接受两个参数,第一个参数是重复的次数,第二个参数是所要重复的值。比如上面行高都是一样的,我们可以这么去实现,实际效果是一模一样的。

css
.wrapper {
  display: grid;
  grid-template-columns: 200px 100px 200px;
  grid-gap: 10px;
  /*  2行,而且行高都为 50px  */
  grid-template-rows: repeat(2, 50);
}

auto-fill 关键字:表示自动填充,让一行(或者一列)中尽可能的容纳更多的单元格。grid-template-columns: repeat(auto-fill, 200px) 表示列宽是 200 px,但列的数量是不固定的,只要浏览器能够容纳得下,就可以放置元素,代码以及效果如下所示【拖动缩放浏览器宽度查看效果】:

css
.wrapper {
  display: grid;
  grid-template-columns: repeat(auto-fill, 200px);
  grid-gap: 5px;
  grid-auto-rows: 50px;
}
  1. fr 关键字:Grid 布局还引入了一个另外的长度单位来帮助我们创建灵活的网格轨道。fr 单位代表网格容器中可用空间的一等份。grid-template-columns: 200px 1fr 2fr 表示第一个列宽设置为 200px,后面剩余的宽度分为两部分,宽度分别为剩余宽度的 1/3 和 2/3。代码以及效果如下所示【拖动缩放浏览器宽度查看效果】:
css
.wrapper {
  display: grid;
  grid-template-columns: 200px 1fr 2fr;
  grid-gap: 5px;
  grid-auto-rows: 50px;
}
  1. minmax() 函数:我们有时候想给网格元素一个最小和最大的尺寸,minmax() 函数产生一个长度范围,表示长度就在这个范围之中都可以应用到网格项目中。它接受两个参数,分别为最小值和最大值。grid-template-columns: 1fr 1fr minmax(300px, 2fr) 的意思是,第三个列宽最少也是要 300px,但是最大不能大于第一第二列宽的两倍。代码以及效果如下所示【拖动缩放浏览器宽度查看效果】:
css
.wrapper {
  display: grid;
  grid-template-columns: 1fr 1fr minmax(300px, 2fr);
  grid-gap: 5px;
  grid-auto-rows: 50px;
}
  1. auto 由浏览器决定长度。通过 auto 关键字,我们可以轻易实现三列或者两列布局。grid-template-columns: 100px auto 100px 表示第一第三列为 100px,中间由浏览器决定长度,代码以及效果如下所示【拖动缩放浏览器宽度查看效果】:
css
.wrapper {
  display: grid;
  grid-template-columns: 100px auto 100px;
  grid-gap: 5px;
  grid-auto-rows: 50px;
}

grid-row-gap 、grid-column-gap 以及 grid-gap

grid-row-gap 属性、grid-column-gap 属性分别设置行间距和列间距。 grid-gap 属性是两者的简写形式。

警告

grid-row-gapgrid-column-gapgrid-gap都已被废弃,现在改成row-gapcolumn-gapgap,当然之前的还能用【并且 grid-*更加清晰】。

css
.wrapper {
  display: grid;
  grid-template-columns: 200px 100px 100px;
  grid-auto-rows: 50px;
  gap: 10px 20px;
}

等同于

css
.wrapper {
  display: grid;
  grid-template-columns: 200px 100px 100px;
  grid-auto-rows: 50px;
  row-gap: 10px;
  column-gap: 20px;
}

grid-template-areas

grid-template-areas 属性用于定义区域,一个区域由一个或者多个单元格组成,一般这个属性跟网格元素的 grid-area 一起使用,指定项目放在哪一个区域。

上面代码表示划分出 6 个单元格,其中值得注意的是 . 符号代表空的单元格,也就是没有用到该单元格。并将类 .sidebar .content .header所在的元素放在上面 grid-template-areas 中定义的 sidebar content header 区域中。

grid-auto-flow

grid-auto-flow 属性控制着自动布局算法怎样运作,精确指定在网格中被自动布局的元素怎样排列。默认的放置顺序是"先行后列",即先填满第一行,再开始放入第二行,即下图英文数字的顺序 one,two,three...。这个顺序由 grid-auto-flow 属性决定,默认值是 row

css
.wrapper {
  grid-auto-flow: row | column | dense | row dense | column dense;
}

justify-items 、align-items 以及 place-items

justify-items 属性设置单元格内容的水平位置(左中右),align-items 属性设置单元格的垂直位置(上中下)。默认值 stretch

css
.wrapper {
  justify-items: start | end | center | stretch;
  align-items: start | end | center | stretch;
}

justify-content 、align-content 以及 place-content

justify-content 属性是整个内容区域在容器里面的水平位置(左中右),align-content 属性是整个内容区域的垂直位置(上中下)。它们都有如下的属性值。

css
.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;
}

grid-auto-columns 和 grid-auto-rows

概念解读

在讲 grid-auto-columns 属性和 grid-auto-rows 属性之前,先来看看隐式和显示网格的概念。

隐式和显示网格:显式网格包含了你在 grid-template-columnsgrid-template-rows 属性中定义的行和列。如果你在网格定义之外又放了一些东西,或者因为内容的数量而需要的更多网格轨道的时候,网格将会在隐式网格中创建行和列。

假如有多余的网格(也就是上面提到的隐式网格),那么它的行高和列宽可以根据 grid-auto-columns 属性和 grid-auto-rows 属性设置。它们的写法和 grid-template-columnsgrid-template-rows 完全相同。如果不指定这两个属性,浏览器完全根据单元格内容的大小,决定新增网格的列宽和行高。

项目属性

grid-column-start 、grid-column-end 、grid-row-start 以及 grid-row-end

可以指定网格项目所在的四个边框,分别定位在哪根网格线,从而指定项目的位置

  • grid-column-start 属性:左边框所在的垂直网格线
  • grid-column-end 属性:右边框所在的垂直网格线
  • grid-row-start 属性:上边框所在的水平网格线
  • grid-row-end 属性:下边框所在的水平网格线

grid-area

grid-area 属性指定项目放在哪一个区域,在上面介绍 grid-template-areas 时有讲。

justify-self 、align-self 以及 place-self

  • justify-self 属性设置单元格内容的水平位置(左中右),跟 justify-items 属性的用法完全一致,但只作用于单个项目。
  • align-self 属性设置单元格内容的垂直位置(上中下),跟 align-items 属性的用法完全一致,也是只作用于单个项目。
css
.item {
  justify-self: start | end | center | stretch;
  align-self: start | end | center | stretch;
}

实战示例

等分响应式

解析:fr 等分单位,可以将容器的可用空间分成想要的多个等分空间。利用这个特性,我们能够轻易实现一个等分响应式。grid-template-columns: 1fr 1fr 1fr 表示容器分为三等分。代码以及效果如下所示【拖动缩放浏览器宽度查看效果】

固定列宽,列数量变化

解析:网格能够固定列宽,并根据容器的宽度来改变列的数量。这个时候,我们可以用到上面提到 repeat() 函数以及 auto-fit 关键字。grid-template-columns: repeat(auto-fit, 200px) 表示固定列宽为 200px,数量是自适应的,只要容纳得下,就会往上排列,代码以及效果如下所示【拖动缩放浏览器宽度查看效果】

最小宽度,去掉右侧空白

提示

上例中,右侧通常会留下空白,这是我们不希望看到的。

如果列的宽度也能在某个范围内自适应就好了。minmax() 函数就帮助我们做到了这点。将 grid-template-columns: repeat(auto-fit, 200px) 改成 grid-template-columns: repeat(auto-fit, minmax(100px, 1fr)) 表示列宽至少 100px,如果还有空余则一起等分,代码以及效果如下所示【拖动缩放浏览器宽度查看效果】

解决空缺问题

实现爱摸鱼主页,让不同宽高的App,自动填充

css
/* 关键代码 */
.wrapper {
  grid-auto-flow: row dense;
}

Grid 兼容性

caniuse

参考资料