俄罗斯贵宾会-俄罗斯贵宾会官网
做最好的网站

(译)原生CSS网格布局学习笔记

原生CSS网格布局学习笔记

2016/12/22 · CSS · 网格布局

原文出处: css-tricks   译文出处:众成翻译   

译者注:此文适合有一定CSS原生网格布局使用经验的开发者(读前需要先去了解一下原生CSS网格的语法),原生CSS网格布局(Native CSS grid)截止目前还没有被任何正式版本的浏览器实现。

以下是来自Oliver Williams的帖子. Oliver已经学习了相当长时间的原生CSS网格,可以说是在CSS网格方面有一定的发言权。在这篇文章中,他将以非同寻常的思路分析自己的CSS网格布局学习之路。我比较赞同他的想法,就是学习一门新技术的时候,把它们拆分成比较小的单元块并配上实例,一步一步的学习。这比直接学习网格布局的所有东西要好太多了。

浏览器原生CSS网格预计会在2017年年初得到支持. 在这之前你需要在浏览器中开启这个实验性的功能 (Firefox实验版默认是开启的). Chrome Canary是当前最好的实现. 同时,火狐有一个非常好的插件叫CSS Grid Inspector, 它能显示出网格的线,它是目前唯一可以在浏览器中运行的此类工具。

在 chrome的地址栏中输入chrome://flags, 找到 ‘实验性网络平台功能’ 并开启它. IE 和 Edge 实现的是一个比较老的网格标准,现在并不受支持。

原文地址

网格布局不是将零散的块拼到一起

相信我,很快你就能掌握它的.

俄罗斯贵宾会 1

网格布局只能像左边那样,以矩形的单元块组合起来。并不能像右图那样,由一堆零散的多边形(跟俄罗斯方块那样的块)拼凑。

注:此文是我翻译的第一篇技术文章。适合有一定CSS原生网格布局使用经验的开发者(读前需要先去了解一下原生CSS网格的语法),原生CSS网格布局(Native CSS grid)截止目前还没有被任何正式版本的浏览器实现。

设计网格布局并不是为了取代弹性盒,相反,它是弹性盒的一种补充

虽然网格布局和弹性盒在某些方面起到相似的作用,而且你可以发现,很多人用弹性盒来实现网格布局,但这并不是设计弹性盒的初衷。Jake Archibald的这篇博文值得一读_Don’t use flexbox for overall page layout

这篇博文大概的意思是:

  • Flexbox(弹性盒)是为一维布局设计的(行或列)。
  • CSS网格是为二维设计的.

Rachel Andrews也 说过类似的话:

Flexbox(弹性盒)用于一维布局 – 也就是行或者列. 网格用于二维布局 – 也就是多行多列.

它们可以很好的结合,你可以往弹性容器中放入网格,也可以在网格块中加入flex元素

来看个例子吧。 我们想在一个网格元素(grid item)里垂直居中一段文字, 但我们想要让背景(图片,颜色或渐变)覆盖整个的网格区域。 我们可以使用align-items属性,并把它的值设为center,但是如果这样背景并不会填满整个网格元素的区域。align-items 默认的值是 stretch-你不改变它,始终会填满整个空间的。我们把网格元素设为align-items:center并把网格元素(grid item)设置为一个弹性容器(flex container)。

CSS

.grid { align-items: stretch; } .griditem { display: flex; align-items: center; }

1
2
3
4
5
6
7
8
.grid {
  align-items: stretch;
}
 
.griditem {
  display: flex;
  align-items: center;
}

_以下是来自Oliver Williams的帖子. Oliver已经学习了相当长时间的原生CSS网格,可以说是在CSS网格方面有一定的发言权。在这篇文章中,他将以非同寻常的思路分析自己的CSS网格布局学习之路。我比较赞同他的想法,就是学习一门新技术的时候,把它们拆分成比较小的单元块并配上实例,一步一步的学习。这比直接学习网格布局的所有东西要好太多了。

给grid-column-end设置负值,意想不到的有用

在小屏幕下,写一个12列的网格,所有格子的跨度都12列。

你可以用网格这样做:

CSS

/* For small screens */ .span4, .span6, .spanAll { grid-column-end: span 12; } /* For large screens */ @media (min-width: 650px) { .span4 { grid-column-end: span 4; } .span6 { grid-column-end: span 6; } }

1
2
3
4
5
6
7
8
9
10
11
12
13
14
/* For small screens */
.span4, .span6, .spanAll {
  grid-column-end: span 12;
}
 
/* For large screens */
@media (min-width: 650px) {
  .span4 {
    grid-column-end: span 4;
  }
  .span6 {
    grid-column-end: span 6;
  }
}

这样的显示效果是没什么错误的,当使用CSS网格,重新定义列数非常简单。并且你可以通过设置grid-column-end: -1;来让你的页面始终是从左到右贯穿的。

CSS

/* For small screens */ .span4, .span6, .spanAll { grid-column-end: -1; }

1
2
3
4
/* For small screens */
.span4, .span6, .spanAll {
  grid-column-end: -1;
}

在大屏幕上,你想要尽可能的接近12列,但是在移动端,一行大概是1~4列。用media来改变grid-template-columns是非常容易的。

CSS

.grid { grid-template-columns: 1fr 1fr; } @media (min-width: 700px) { .grid { grid-template-columns: repeat(12, 1fr); } }

1
2
3
4
5
6
7
8
9
.grid {
  grid-template-columns: 1fr 1fr;
}
 
@media (min-width: 700px) {
  .grid {
    grid-template-columns: repeat(12, 1fr);
  }
}

有一些元素,我们想让它贯穿整个视口,比如像 header, footer,和一些大图啥的。

对于小屏幕,我们可以这样写:

CSS

.wide { grid-column: 1 / 3; /* start at 1, end at 3 */ }

1
2
3
.wide {
  grid-column: 1 / 3; /* start at 1, end at 3 */
}

不幸的是,当我们换到大屏的时候,一行12列,这些元素将仅仅占满前两列,并不会占满12列,我们需要定义新的grid-column-end,并且把他的值设为 13. 这种方式比较麻烦,还有一种简单的方式,grid-column: 1 / -1;,这样不论在什么屏幕尺寸下,它们都是占满整行的了。就像下面这样:

CSS

.wide, .hero, .header, .footer { grid-column: 1 / -1; }

1
2
3
.wide, .hero, .header, .footer {
  grid-column: 1 / -1;
}

See the Pen Easier media queries with -1 by CSS GRID (@cssgrid) on CodePen.

浏览器原生CSS网格预计会在2017年年初得到支持. 在这之前你需要在浏览器中开启这个实验性的功能 (Firefox实验版默认是开启的). Chrome Canary是当前最好的实现. 同时,火狐有一个非常好的插件叫CSS Grid Inspector, 它能显示出网格的线,它是目前唯一可以在浏览器中运行的此类工具。

网格区域可以命名,并使用一些隐含的名字

使用grid-template-areasgrid-line-numbers是两种控制行数的属性,你也可以两个同时用。你可以使用那些隐含的行名去设置你的网格。

CSS

.grid { grid-template-areas: "main main sidebar sidebar"; }

1
2
3
.grid {
  grid-template-areas: "main main sidebar sidebar";
}

这段代码,我们能得到四个隐含名字,main-start, main-end, sidebar-start, 和 sidebar-end.

这可能很有用,如果你想重叠内容,无论是在几个网格区域或在一个特定分段的网格区域。

See the Pen implicit line names with grid areas by CSS GRID (@cssgrid) on 俄罗斯贵宾会,CodePen.

在 chrome的地址栏中输入chrome://flags, 找到 ‘实验性网络平台功能’ 并开启它. IE 和 Edge 实现的是一个比较老的网格标准,现在并不受支持。

定义网格区域的另一种方式

就像给网格的行命名,特殊的行名能用于设置网格区域,语法是这样的:

CSS

.grid { grid-template-areas: "header header header" "main main sidebar" "footer footer footer"; } 、

1
2
3
4
5
6
.grid {
  grid-template-areas:
    "header header header"
    "main main sidebar"
    "footer footer footer";
} 、

如果你的布局设计(太多列的布局!没列都要起名字,可能还需要空元素)中有很多空的区域,这种写法稍微有点麻烦。所以对于网格是有另一种写法的,在这种写法中,名字是什么无所谓,只要你合理利用到[name-start][name-end],也能达到自己的布局目的。下面是一个例子:

CSS

.grid { display: grid; grid-template-columns: 20px 100px [main-start] 1fr [main-end] 100px 20px; grid-template-rows: 100px [main-start] 100px [main-end] 100px; } .griditem1 { background-color: red; grid-area: main; }

1
2
3
4
5
6
7
8
9
10
.grid {
  display: grid;
  grid-template-columns: 20px 100px [main-start] 1fr [main-end] 100px 20px;
  grid-template-rows: 100px [main-start] 100px [main-end] 100px;
}
 
.griditem1 {
  background-color: red;
  grid-area: main;
}

See the Pen Another way of defining grid-areas by CSS GRID (@cssgrid) on CodePen.

你可能并不想整个页面都用这种方式布局,但是如果你想要结合 grid-area来确定行数的话,它会非常适合。

网格布局不是将零散的块拼到一起

相信我,很快你就能掌握它的.

俄罗斯贵宾会 2

网格布局只能像左边那样,以矩形的单元块组合起来。并不能像右图那样,由一堆零散的多边形(跟俄罗斯方块那样的块)拼凑。

相等尺寸网格(equal sized box layout)使用vmin单位

虽然你可以在CSS网格中使用任意尺寸的行或列,但是如果想要相等大小的格子并是响应式的,你就需要使用vmin单位了。

CSS

.grid { grid-template-columns: repeat(5, 20vw); grid-template-rows: repeat(5, 20vh); }

1
2
3
4
.grid {
  grid-template-columns: repeat(5, 20vw);
  grid-template-rows: repeat(5, 20vh);
}

这种布局在台式电脑和笔记本上基本都可以完美显示,但是在手机上,高度大于宽,内容将会溢出,产生出一个横向的滚动条。Dudley Storey写了篇blog说这件事the usefulness of a lesser-known css unit: vmin。这种方法,通过调整容器视口的百分比和内容位置,做到适配各种尺寸的屏幕。

CSS

.gridcontainer { display: grid; width: 100vw; height: 100vh; justify-content: center; align-content: center; grid-template-columns: repeat(5, 20vmin); grid-template-rows: repeat(5, 20vmin); }

1
2
3
4
5
6
7
8
9
.gridcontainer {
  display: grid;
  width: 100vw;
  height: 100vh;
  justify-content: center;
  align-content: center;
  grid-template-columns: repeat(5, 20vmin);
  grid-template-rows: repeat(5, 20vmin);
}

See the Pen Boxy Layout with CSS Grid and vmin by CSS GRID (@cssgrid) on CodePen.

设计网格布局并不是为了取代弹性盒,相反,它是弹性盒的一种补充

虽然网格布局和弹性盒在某些方面起到相似的作用,而且你可以发现,很多人用弹性盒来实现网格布局,但这并不是设计弹性盒的初衷。Jake Archibald的这篇博文值得一读_Don't use flexbox for overall page layout

这篇博文大概的意思是:

  • Flexbox(弹性盒)是为一维布局设计的(行或列)。

  • CSS网格是为二维设计的.

Rachel Andrews也 说过类似的话:

Flexbox(弹性盒)用于一维布局 – 也就是行或者列. 网格用于二维布局 – 也就是多行多列.

它们可以很好的结合,你可以往弹性容器中放入网格,也可以在网格块中加入flex元素

来看个例子吧。 我们想在一个网格元素(grid item)里垂直居中一段文字, 但我们想要让背景(图片,颜色或渐变)覆盖整个的网格区域。 我们可以使用align-items属性,并把它的值设为center,但是如果这样背景并不会填满整个网格元素的区域。align-items 默认的值是 stretch-你不改变它,始终会填满整个空间的。我们把网格元素设为align-items:center并把网格元素(grid item)设置为一个弹性容器(flex container)。

.grid {
  align-items: stretch;
}

.griditem {
  display: flex;
  align-items: center;
}

绝对定位

当我们绝对定位一个网格元素的时候,这个元素会跑到它的容器中,我们可以用grid-column 和 grid-row来定位它。正常情况下,绝对定位使元素脱离文档流,它最适合的使用场景就是想要让元素重叠,并不打乱其他布局元素。除非你为每个元素声明grid-column-startgrid-row-start,要不然即使使用了绝对定位,元素也是不会重叠的。

尝试删除这个例子中div的position: absolute;,思考grid-column 和 grid-row的值,也可以试试修改它们,你就明白是什么意思了。

See the Pen preserving auto-placement with position: absolute by CSS GRID (@cssgrid) on CodePen.

给grid-column-end设置负值,意想不到的有用

在小屏幕下,写一个12列的网格,所有格子的跨度都12列。

你可以用网格这样做:

/* For small screens */
.span4, .span6, .spanAll {
  grid-column-end: span 12;
}

/* For large screens */
@media (min-width: 650px) {
  .span4 {
    grid-column-end: span 4;
  }
  .span6 {
    grid-column-end: span 6;
  }
}

这样的显示效果是没什么错误的,当使用CSS网格,重新定义列数非常简单。并且你可以通过设置grid-column-end: -1;来让你的页面始终是从左到右贯穿的。

/* For small screens */
.span4, .span6, .spanAll {
  grid-column-end: -1;
}

在大屏幕上,你想要尽可能的接近12列,但是在移动端,一行大概是1~4列。用media来改变grid-template-columns是非常容易的。

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

@media (min-width: 700px) {
  .grid {
    grid-template-columns: repeat(12, 1fr);
  }
}

有一些元素,我们想让它贯穿整个视口,比如像 header, footer,和一些大图啥的。

对于小屏幕,我们可以这样写:

.wide {
  grid-column: 1 / 3; /* start at 1, end at 3 */
}

不幸的是,当我们换到大屏的时候,一行12列,这些元素将仅仅占满前两列,并不会占满12列,我们需要定义新的grid-column-end,并且把他的值设为 13. 这种方式比较麻烦,还有一种简单的方式,grid-column: 1 / -1;,这样不论在什么屏幕尺寸下,它们都是占满整行的了。就像下面这样:

.wide, .hero, .header, .footer {
  grid-column: 1 / -1;
}

See the Pen Easier media queries with -1 by CSS GRID (@cssgrid) on CodePen.

改变网格元素(grid item)的顺序

如果你使用过弹性盒(flexbox)的order 属性,那你已经知道一些相关的知识了。所有的网格元素都有一个默认的order值0。所以如果给一个网格元素设置 order: 1;,这个元素将在所有元素的后面。 你可以给order属性设置负值,让它跑到所有item的前面。

See the Pen Order value by CSS GRID (@cssgrid) on CodePen.

网格区域可以命名,并使用一些隐含的名字

使用grid-template-areasgrid-line-numbers是两种控制行数的属性,你也可以两个同时用。你可以使用那些隐含的行名去设置你的网格。

.grid {
  grid-template-areas: "main main sidebar sidebar";
}

这段代码,我们能得到四个隐含名字,main-start, main-end, sidebar-start, 和 sidebar-end.

这可能很有用,如果你想重叠内容,无论是在几个网格区域或在一个特定分段的网格区域。

See the Pen implicit line names with grid areas by CSS GRID (@cssgrid) on CodePen.

grid中 minmax()的坑

想不想要整行随着内容的宽度而变宽,直到他们达到最大宽度,这种情况你可能想尝试使用 minmax()

CSS

.grid { display: grid; grid-template-columns: repeat(3, minmax(1fr, 300px)); }

1
2
3
4
.grid {
  display: grid;
  grid-template-columns: repeat(3, minmax(1fr, 300px));
}

不幸的是,像上面这样看似简单,实际上是不行的。如果max小于min的话,css会被忽略。在minmax()fr不能使用。实际上实现这个需求很容易,在grid-template-columnsgrid-template-rows中使用auto,这样item就可以随着内容增大而变大了。 See the Pen The value of auto vs fr by CSS GRID (@cssgrid) on CodePen.

我们可以设置一个 max-width:

CSS

.grid { display: grid; grid-template-columns: repeat(3, auto); } .item { max-width: 300px; }

1
2
3
4
5
6
7
8
.grid {
  display: grid;
  grid-template-columns: repeat(3, auto);
}
 
.item {
  max-width: 300px;
}

See the Pen The limits of minmax by CSS GRID (@cssgrid) on CodePen.

minmax()的运行方式和使用我还没有完全想出来,虽然如此,我还是写了一篇文章(译者注:Medium entitled是什么我没有理解清楚,原文:I wrote an entire post on Medium entitled) The One Thing I Hate About Grid.

定义网格区域的另一种方式

就像给网格的行命名,特殊的行名能用于设置网格区域,语法是这样的:

.grid {
  grid-template-areas:
    "header header header"
    "main main sidebar"
    "footer footer footer";
} 、

如果你的布局设计(太多列的布局!没列都要起名字,可能还需要空元素)中有很多空的区域,这种写法稍微有点麻烦。所以对于网格是有另一种写法的,在这种写法中,名字是什么无所谓,只要你合理利用到[name-start][name-end],也能达到自己的布局目的。下面是一个例子:

.grid {
  display: grid;
  grid-template-columns: 20px 100px [main-start] 1fr [main-end] 100px 20px;
  grid-template-rows: 100px [main-start] 100px [main-end] 100px;
}

.griditem1 {
  background-color: red;
  grid-area: main;
}

See the Pen Another way of defining grid-areas by CSS GRID (@cssgrid) on CodePen.

你可能并不想整个页面都用这种方式布局,但是如果你想要结合 grid-area来确定行数的话,它会非常适合。

本文由俄罗斯贵宾会发布于Web前端,转载请注明出处:(译)原生CSS网格布局学习笔记

您可能还会对下面的文章感兴趣: