Home >Web Front-end >HTML Tutorial >CSS Grid布局指南_html/css_WEB-ITnose

CSS Grid布局指南_html/css_WEB-ITnose

WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWB
WBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOYWBOriginal
2016-06-24 11:21:291978browse

简介

CSS Grid布局 (又名"网格"),是一个基于二维网格布局的系统,主要目的是改变我们基于网格设计的用户接口方式。如我们所知,CSS 总是用于网页的样式设置,但它并没有起到很好的作用。刚开始的时候我们使用表格( table ),然后使用浮动( float )、 定位( position )和内联块( inline-block ),但所有这些方法本质上来讲都是hacks,存留了很多需要实现的重要功能问题(例如,垂直居中)。虽然Flexbox可以起到一定的补救作用,但是它只可以实现简单的一维布局,并不适用于复杂的二维布局(实际上 Flexbox 和 Grid 可以一起结合使用起到最佳效果)。网格是 CSS 第一次专门创建的模块,用来解决我们之前在制作网站时使用hacks处理布局问题。

这里有两件事情启发我创建本指南。第一个是 Rachel Andrew 的令人敬畏的书-- 为 CSS Grid 布局做好准备 。这本书很详尽明确的的介绍了Grid,如果你想很好的掌握Grid的基础知识,我强烈建议你去购买。另外一个很大的灵感来自于 Chris Coyier 的--Flexbox完整指南,这本书是我了解Flebox的一个很优秀的资源。这里,我还想补充一句,当你使用谷歌搜索"flexbox"时,会出现很多类似的资源,但是为什么不直接利用最好的资源呢?

我书写此指南的目的是基于目前最新版本,规范其网格概念。所以我不会再次提及过时的 IE 语法,并且随着规范的成熟,我会尽力定期更新此指南。

基础知识与浏览器支持

Grid 的入门是很容易的。你只需要定义一个容器元素并设置 display:grid ,使用 grid-template-columns 和 grid-template-rows 属性设置网格的列与 行的大小,然后使用 grid-column 和 grid-row 属性将其子元素放入网格之中。与flexbox类似,网格项的源顺序无关紧要。为了更好地使你的网格与媒体查询相结合使用,你可以在 CSS 中任意放置。想象一下你定义的整个页面布局,然后如果想要完全重新布局以适应不同的屏幕宽度,这时仅仅使用几行 CSS 代码就可以实现。Grid是曾经介绍过的最强大 CSS 模块之一。

关于 Grid 一件很重要的事情就是它现在还不适用于项目使用。目前还处于 W3C 的工作草案 之中,并且默认情况下,还不被所有的浏览器所支持。Internet Explorer 10 和 11 已经可以实现支持,但也是利用一种过时的语法实现的。现在出于示例演示,我建议你使用启用了特殊标志的 Chrome, Opera 或者 Firefox 。在 Chrome,导航到 chrome://flags 并启用" web 实验平台功能"。该方法同样适用于 Opera ( opera://flags )。在Firefox中,启用 layout.css.grid.enabled 标志。

这里有一张浏览器支持情况的表格(之后我会继续更新):

除了Microsoft,浏览器厂商似乎想要等到Grid规范成熟后再加以推广。这是一件好事,因为这意味着我们就不需要担心学习多个语法。

等待 Grid 的使用,只是时间的问题。但是现在你需要开始学习它了。

重要术语

在深入研究Grid之前,我们需要理解其相关术语概念。因为这里涉及到的术语在概念上都有点类似,如果你没有首先记住Grid规范中的相关定义,你就会很容易将其与另一个概念相混淆。但是不需要担心,这里的属性并不是很多。

网格容器(Grid Container)

当一个元素设置 display: grid 属性时,它就会成为所有网格项(Grid Items)的父元素。在下面的示例中, container 就是网格容器。

<div class="container">    <div class="item item-1"></div>    <div class="item item-2"></div>    <div class="item item-3"></div></div>

网格项(Grid Item)

网格容器的孩子(e.g. 子元素)。这里 item 元素都是网格项,但是 sub-item 不包含其中。

<div class="container">    <div class="item"></div>     <div class="item">        <p class="sub-item"></p>    </div>    <div class="item"></div></div>

网格线(Grid Line)

分界线构成了网格的结构。他们可以是垂直的("列网格线")也可以是水平的("行网格线"),并且存在于一行或一列的任一侧。下面图片中的黄线就是列网格线的一个例子。

网格轨道(Grid Track)

两个相邻网格线之间的空间。你可以把它们想像成网格的行或列。下图所示的是第二行和第三行网格线之间的网格轨道。

网格单元格(Grid Cell)

两个相邻的行和两个相邻的列之间的网格线空间。它是网格的一个"单位"。下面图片所示的是行网格线 1 和 2 与列网格线 2 和 3 之间的网格单元格。

网格区域(Grid Area)

四条网格线所包围的所有空间。网格区域可由任意数量的网格单元格组成。下面图片所示的是行网格线 1 和 3 和列网格线 1 和 3 之间的网格区域。

网格容器属性(Grid Container)

display

定义一个元素成为网格容器,并对其内容建立一个网格格式的上下文。

属性值:

  • grid : 产生一个块级的网格
  • inline-grid : 产生内联级网格
.container{    display: grid | inline-grid   }

注: column , float , clear , 和 vertical-align 元素对网格容器不起作用。

grid-template-rows

利用以空格分隔的值定义网格的列和行。值的大小代表轨道的大小,并且它们之间的空格表示网格线。

属性值:

  • : 可以是一个长度、百分比或者是网格中自由空间的一小部分(使用 fr 单位)
  • : 你选择的任意名称
  • subgrid - 如果你的网格容器本身就是一个网格项(即嵌套网格),你可以使用此属性指定行和列的大小继承于父元素而不是自身指定。
.container{    grid-template-columns: <track-size> ... | <line-name> <track-size> ... | subgrid;    grid-template-rows: <track-size> ... | <line-name> <track-size> ... | subgrid;}

示例:

当你在值之间留有空格时,网络线就会自动分配数值名称:

.container{    grid-template-columns: 40px 50px auto 50px 40px;    grid-template-rows: 25% 100px auto;}

但是你也可以显示命名,请参考下面括号语法中的名称命名方式:

.container{    grid-template-columns: [first] 40px [line2] 50px [line3] auto [col4-start] 50px [five] 40px [end];    grid-template-rows: [row1-start] 25% [row1-end] 100% [third-line] auto [last-line];}

请注意,一条网格线可以具有有多个名称。例如,这里的第二行将有两个名字: row1-end 和 row2-start :

.container{    grid-template-rows: [row1-start] 25% [row1-end row2-start] 25% [row2-end];}

如果你的定义中包含重复的部分,你可以使用 repeat() 表示法进行精简:

.container{    grid-template-columns: repeat(3, 20px [col-start]) 5%;}

等效于:

.container{    grid-template-columns: 20px [col-start] 20px [col-start] 20px [col-start] 5%;}

fr 单位允许你将一个轨道大小设置为网格容器内自由空间的一小部分。如下所示,每个网格项就会占据网格容器宽度的三分之一:

.container{    grid-template-columns: 1fr 1fr 1fr;}

这里自由空间表示除去非弹性项以后剩余的空间。在此示例中的 fr 单位的可用空间表示减去 50px 以后的空间大小:

.container{    grid-template-columns: 1fr 50px 1fr 1fr;}

grid-template-areas

使用 grid-area 属性定义网格区域名称,从而定义网格模板。网格区域重复的名称就会导致内容跨越这些单元格。句点表示一个空单元格。语法本身提供了一种可视化的网格结构。

属性值:

  • : 使用 grid-area 属性定义网格区域名称
  • . : 句点表示一个空单元格
  • none : 无网格区域被定义
.container{    grid-template-areas: "<grid-area-name> | . | none | ..."                      "..."}

示例:

.item-a{    grid-area: header;}.item-b{    grid-area: main;}.item-c{    grid-area: sidebar;}.item-d{    grid-area: footer;}.container{    grid-template-columns: 50px 50px 50px 50px;    grid-template-rows: auto;    grid-template-areas: "header header header header"                         "main main . sidebar"                         "footer footer footer footer"}

这将创建一个四列三行的网格。最上面的一行为 header 区域。中间一行由两个 main 区域,一个空单元格和一个 sidebar 区域。最后一行是 footer 区域。

你所声明的每一行都需要具有相同数目的单元格。

你可以使用任意数量的句点( . )声明单个空单元格。只要句点之间没有空格就表示一个空单元格。

注意,你只是使用此语法进行网格区域命名,而不是网格线命名。当你使用此语法时,区域两边的线就会得到自动命名。如果网格区域名称为 foo ,则其行线和列线的名称就将为 foo-start ,最后一行线及其最后一列线的名字就会为 foo-end 。这意味着一些线就可能具有多个名称,如上面示例中所示,拥有三个名称: header-start , main-start , 以及 footer-start 。

grid-column-gap 和 grid-row-gap

指定网格线的大小。你可以把它想像成在行/列之间设置间距宽度。

属性值:

  • : 一个长度值
.container{    grid-column-gap: <line-size>;    grid-row-gap: <line-size>;}

示例:

.container{    grid-template-columns: 100px 50px 100px;    grid-template-rows: 80px auto 80px;     grid-column-gap: 10px;    grid-row-gap: 15px;}

间距仅仅在列/行之间产生,而不会在边缘区。

grid-gap

grid-column-gap 和 grid-row-gap 的简写值。

属性值:

  • : 长度值
.container{    grid-gap: <grid-column-gap> <grid-row-gap>;}

示例:

.container{    grid-template-columns: 100px 50px 100px;    grid-template-rows: 80px auto 80px;     grid-gap: 10px 15px;}

如果没有指定 grid-row-gap 属性的值,默认与 grid-column-gap 属性值相同

justify-items

沿列轴对齐网格项中的内容(相反于 align-item 属性定义的沿行轴对齐)。此值适用于容器内所有的网格项。

属性值:

  • start : 内容与网格区域的左端对齐
  • end : 内容与网格区域的右端对齐
  • center : 内容处于网格区域的中间位置
  • stretch : 内容宽度占据整个网格区域空间(默认值)
.container{    justify-items: start | end | center | stretch;}

示例:

.container{    justify-items: start;}

.container{    justify-items: end;}

.container{    justify-items: center;}

.container{    justify-items: stretch;}

这也可以使用 justify-self 属性对各个网格项进行设置。

align-items

沿行轴对齐网格项中的内容(相反于 justify-item 属性定义的沿列轴对齐)。此值适用于容器内所有的网格项。

属性值:

  • start : 内容与网格区域的顶端对齐
  • end : 内容与网格区域的底部对齐
  • center : 内容处于网格区域的中间位置
  • stretch : 内容高度占据整个网格区域空间(默认值)
.container{    align-items: start | end | center | stretch;}

示例:

.container{    align-items: start;}

.container{    align-items: end;}

.container{    align-items: center;}

.container{    align-items: stretch;}

这也可以使用 align-self 属性对各个网格项进行设置。

justify-content

当你使用 px 这种非响应式的单位对你的网格项进行大小设置时,就有可能出现一种情况--你的网格大小可能小于其网格容器的大小。在这种情况下,你就可以设置网格容器内网格的对齐方式。此属性会将网格沿列轴进行对齐(相反于 align-content 属性定义的沿行轴对齐)。

属性值:

  • start : 网格与网格容器的左端对齐
  • end : 网格与网格容器的右端对齐
  • center : 网格处于网格容器的中间
  • stretch : 调整网格项的大小,使其宽度填充整个网格容器
  • space-around : 在网格项之间设置偶数个空格间隙,其最边缘间隙大小为中间空格间隙大小的一半
  • space-between : 在网格项之间设置偶数个空格间隙,其最边缘不存在空格间隙
  • space-evenly : 在网格项之间设置偶数个空格间隙,同样适用于最边缘区域
.container{    justify-content: start | end | center | stretch | space-around | space-between | space-evenly;    }

示例:

.container{    justify-content: start;}

.container{    justify-content: end; }

.container{    justify-content: center;  }

.container{    justify-content: stretch; }

.container{    justify-content: space-around;    }

.container{    justify-content: space-between;   }

.container{  justify-content: space-evenly;    }

align-content

当你使用 px 这种非响应式的单位对你的网格项进行大小设置时,就有可能出现一种情况--你的网格大小可能小于其网格容器的大小。在这种情况下,你就可以设置网格容器内网格的对齐方式。此属性会将网格沿行轴进行对齐(相反于 justify-content 属性定义的沿列轴对齐)。

属性值:

  • start : 网格与网格容器的顶端对齐
  • end : 网格与网格容器的底部对齐
  • center : 网格处于网格容器的中间
  • stretch : 调整网格项的大小,使其高度填充整个网格容器
  • space-around : 在网格项之间设置偶数个空格间隙,其最边缘间隙大小为中间空格空隙大小的一半
  • space-between : 在网格项之间设置偶数个空格间隙,其最边缘不存在空格间隙
  • space-evenly : 在网格项之间设置偶数个空格间隙,同样适用于最边缘区域
.container{    align-content: start | end | center | stretch | space-around | space-between | space-evenly;  }

示例:

.container{    align-content: start; }

.container{    align-content: end;   }

.container{    align-content: center;    }

.container{    align-content: stretch;   }

.container{    align-content: space-around;  }

.container{    align-content: space-between; }

.container{    align-content: space-evenly;  }

grid-auto-columns 和 grid-auto-rows

指定任何自动生成的网格轨道(隐式网格跟踪)的大小。当你显式定位行或列(使用 grid-template-rows / grid-template-columns 属性)时,就会产生超出定义范围内的隐式网格轨道。

属性值:

  • : 可以是长度、 百分比或网格自由空间的一小部分(使用 fr 单位)
.container{    grid-auto-columns: <track-size> ...;    grid-auto-rows: <track-size> ...;}

为了说明隐式网格轨道是如何被创造出来的,请思考如下代码:

.container{    grid-template-columns: 60px 60px;    grid-template-rows: 90px 90px}

这里创建了一个 2 x 2 的网格。

但是现在你想象你使用 grid-column 和 grid-row 来定位网格项,如下所示:

.item-a{    grid-column: 1 / 2;    grid-row: 2 / 3;}.item-b{    grid-column: 5 / 6;    grid-row: 2 / 3;}

这里我们定义 .item b 开始于列线 5 并结束于在列线 6 ,但是我们从来没有定义列线 5 或 6 。因为我们引用不存在的线,宽度为 0 的隐式轨道的就会被创建用来填补空白。我们可以使用 grid-auto-columns 和 grid-auto-rows 属性来设置这些隐式轨道的宽度:

.container{    grid-auto-columns: 60px;}

grid-auto-flow

如果你不显式的在网格中放置网格项,自动布局算法就会自动踢出此网格项。此属性用来控制自动布局算法的工作原理。

属性值:

  • row : 告诉自动布局算法填充每一行,必要时添加新行
  • column : 告诉自动布局算法填充每一列,必要时添加新列
  • dense : 告诉自动布局算法试图填补网格中之前较小的网格项留有的空白
.container{    grid-auto-flow: row | column | row dense | column dense}

注意: dense 值可能会导致更改网格项的顺序。

示例:

考虑如下HTMl代码:

<section class="container">    <div class="item-a">item-a</div>    <div class="item-b">item-b</div>    <div class="item-c">item-c</div>    <div class="item-d">item-d</div>    <div class="item-e">item-e</div></section>

这里定义了一个两列五行的网格,并将 grid-auto-flow 属性设置为 row (即默认值):

.container{    display: grid;    grid-template-columns: 60px 60px 60px 60px 60px;    grid-template-rows: 30px 30px;    grid-auto-flow: row;}

将网格项放置在网格中时只需要其中的两个网格项:

.item-a{    grid-column: 1;    grid-row: 1 / 3;}.item-e{    grid-column: 5;    grid-row: 1 / 3;}

因为我们将 grid-auto-flow 属性设置为了 row ,所以我们的网格看起来会像这个样子。注意我们我们没有对其进行设置的三个网格项( item-b , item-c and item-d ),会沿行轴进行布局。

如果我们将 grid-auto-flow 属性设置为 column , item-b , item-c item-d 就会沿列轴进行布局。

.container{    display: grid;    grid-template-columns: 60px 60px 60px 60px 60px;    grid-template-rows: 30px 30px;    grid-auto-flow: column;}

grid

在一行声明中设置一下所有属性的简写形式: grid-template-rows , grid-template-columns , grid-template-areas , grid-auto-rows , grid-auto-columns , 以及 grid-auto-flow 。它将 grid-column-gap 和 grid-row-gap 属性设置为初始值,即使它们不能显示的设置此属性。

属性值:

  • none : 将所有的子属性设置为初始值
  • subgrid : 将 grid-template-rows 和 grid-template-columns 属性值设置为 subgrid ,其余子属性设置为初始值
  • / : 将 grid-template-rows 和 grid-template-columns 属性值设置为指定值,其余子属性设置为初始值
  • [ [ / ] ] : grid-auto-flow , grid-auto-rows 和 grid-auto-columns 属性分别接受相同的值,如果省略了 grid-auto-columns 属性,它将设置为 grid-auto-rows 属性的值。如果两者均被忽略,那么都将被设置为初始值。
.container{    grid: none | subgrid | <grid-template-rows> / <grid-template-columns> | <grid-auto-flow> [<grid-auto-rows> [/ <grid-auto-columns>]];}

示例:

下面两个代码块是等效的:

.container{    grid: 200px auto / 1fr auto 1fr;}.container{    grid-template-rows: 200px auto;    grid-template-columns: 1fr auto 1fr;    grid-template-areas: none;}

同样,下面的两个代码块也是等效的:

.container{    grid: column 1fr / auto;}.container{    grid-auto-flow: column;    grid-auto-rows: 1fr;    grid-auto-columns: auto;}

它还接受一次性设置所有属性,更复杂但非常方便的语法。指定 grid-template-areas , grid-auto-rows 和 grid-auto-columns 属性,其他所有子属性都将设置为其初始值。你现在所做的是在其网格区域内,指定网格线名称和内联轨道大小。下面是最简单的描述:

.container{    grid: [row1-start] "header header header" 1fr [row1-end]          [row2-start] "footer footer footer" 25px [row2-end]          / auto 50px auto;}

等效于:

.container{    grid-template-areas: "header header header"                         "footer footer footer";    grid-template-rows: [row1-start] 1fr [row1-end row2-start] 25px [row2-end];    grid-template-columns: auto 50px auto;    }

网格项属性(Grid Items)

grid-column-start / grid-column-end / grid-row-start / grid-row-end

使用特定的网格线确定网格项在网格内的位置。 grid-column-start / grid-row-start 属性表示网格项的网格线的起始位置, grid-column-end / grid-row-end 属性表示网格项的网格线的终止位置。

属性值:

  • : 可以是一个数字来引用相应编号的网格线,或者使用名称引用相应命名的网格线
  • span : 网格项包含指定数量的网格轨道
  • span : 网格项包含指定名称网格项的网格线之前的网格轨道
  • auto : 表明自动定位,自动跨度或者默认跨度之一
.item{    grid-column-start: <number> | <name> | span <number> | span <name> | auto    grid-column-end: <number> | <name> | span <number> | span <name> | auto    grid-row-start: <number> | <name> | span <number> | span <name> | auto    grid-row-end: <number> | <name> | span <number> | span <name> | auto}

示例:

.item-a{    grid-column-start: 2;    grid-column-end: five;    grid-row-start: row1-start    grid-row-end: 3}

.item-b{    grid-column-start: 1;    grid-column-end: span col4-start;    grid-row-start: 2    grid-row-end: span 2}

如果没有声明 grid-column-end / grid-row-end 属性,默认情况下网格项的跨度为 1 。

网格项可以互相重叠。可以使用 z-index 属性控制堆叠顺序。

grid-column / grid-row

grid-column-start + grid-column-end , 和 grid-row-start + grid-row-end 属性分别的简写形式。

属性值:

  • / : 每一个属性均接收一个相同值,包括跨度。
.item{    grid-column: <start-line> / <end-line> | <start-line> / span <value>;    grid-row: <start-line> / <end-line> | <start-line> / span <value>;}

示例:

.item-c{    grid-column: 3 / span 2;    grid-row: third-line / 4;}

如果没有声明结束网格线值,默认网格轨道跨度为 1 .

grid-area

给网格项进行命名以便于模板使用 grid-template-areas 属性创建时可以加以引用。另外也可以被 grid-row-start + grid-column-start + grid-row-end + grid-column-end 属性更为简洁的加以引用。

属性值:

  • : 你所定义的名称
  • / / / : 可以为数字或者名称
.item{    grid-area: <name> | <row-start> / <column-start> / <row-end> / <column-end>;}

示例:

对网格项进行命名的一种方式:

.item-d{    grid-area: header}

grid-row-start + grid-column-start + grid-row-end + grid-column-end 属性的一种简写方式:

.item-d{    grid-area: 1 / col4-start / last-line / 6}

justify-self

沿列轴对齐网格项中的内容(相反于 align-item 属性定义的沿行轴对齐)。此值适用于单一网格项中的内容。

属性值:

  • start : 内容与网格区域的左端对齐
  • end : 内容与网格区域的右端对齐
  • center : 内容处于网格区域的中间位置
  • stretch : 内容宽度占据整个网格区域空间(默认值)
.item{    justify-self: start | end | center | stretch;}

示例

.item-a{    justify-self: start;}

.item-a{    justify-self: end;}

.item-a{    justify-self: center;}

.item-a{    justify-self: stretch;}

设置网格中所有网格项的对齐方式,可以使用网格容器上的 justify-items 属性。

align-self

沿行轴对齐网格项中的内容(相反于 justify-item 属性定义的沿列轴对齐)。此值适用于单一网格项中的内容。

属性值:

  • start : 内容与网格区域的顶端对齐
  • end : 内容与网格区域的底部对齐
  • center : 内容处于网格区域的中间位置
  • stretch : 内容高度占据整个网格区域空间(默认值)
.item{    align-self: start | end | center | stretch;}

示例:

.item-a{    align-self: start;}

.item-a{    align-self: end;}

.item-a{    align-self: center;}

.item-a{    align-self: stretch;}

使网格中所有的网格项对齐,可以使用网格容器上的 align-items 属性。

特别声明:本文来自于 Chris House写的指南 ,此份指南由Chris himself所写,并且会不断的保持更新。

本文根据 @Chris House 的《 A Complete Guide to CSS Grid Layout 》所译,整个译文带有我们自己的理解与思想,如果译得不好或有不对之处还请同行朋友指点。如需转载此译文,需注明英文出处: http://chris.house/blog/a-complete-guide-css-grid-layout/ 。

静子

在校学生,本科计算机专业。一个积极进取、爱笑的女生,热爱前端,喜欢与人交流分享。想要通过自己的努力做到心中的那个自己。微博:@静-如秋叶

Statement:
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn