翻译 | CSS网格(CSS Grid)布局入门

翻译 | CSS网格(CSS Grid)布局入门

banner

CSS网格布局是浏览器Flexbox布局之后最重要的布局方式。我们可以忘记过去15年经常使用的的各种“神奇数字”,hacks和一系列变通布局方案。网格布局提供了非常简单的声明布局方式,之后再也不需要借助一些常见的主流css框架,也能减少很多手动实现的布局方式

如果你以前不熟悉CSS网格布局,那么你可以开始了解它了。它是一种适用于容器元素,并能指定子元素的间距、大小和对齐方式的布局工具。

CSS网格布局赋予我们更强大的能力——大家熟悉的水平垂直居中布局,不需要增加标签就能做到。同样,这也能让我们不需要媒体查询就能根据可用空间自动适应。

学习的最低要求

首先网格布局有不少新语法需要学习,但是你只需要稍微看下就能上手。本文将会用示例带你学习CSS网格布局各种各样重要的入门概念。

浏览器兼容性

CSS网格布局从Safari 10.1, Firefox 52, Opera 44, Chrome 57开始收到支持,微软Edge在Edge 15会更新对网格布局的支持。

微软的浏览器(IE10–11和Edge 13-14)有一种比较旧的实现,所以有不少限制,我们会简单介绍新的实现方式和老的实现方式之间的区别,这样你能知道如何规避他们。

对于大多数布局,我们会使用下面的query特性来让老的浏览器对他们理解的特性也能工作:

@supports (display: grid) {
    .grid {
        display: grid;
    }
}

不支持浏览器@supports或网格的浏览器将不会生效。

为了能正确展示文中的示例,你需要使用支持网格布局的浏览器

创建带有间距(gutter)的两列(column)网格

为了演示CSS网格布局如何定义列,我们从下面的布局开始:


grid-template-columns 和 grid-gap

[使用grid-template-columns 和 grid-gap创建带间距的两列布局]

为了创建上述网格布局,我们需要使用grid-template-columnsgrid-gap
grid-template-columns表示网格中的列是如何布局的,它的值是一连串以空格分割的的值,这些值标识每列的大小,值的个数表示列的数目。

例如,四列250px宽度的网格布局可以这样表示:

grid-template-columns: 250px 250px 250px 250px;

也可以使用repeat关键字表示:

grid-template-columns: repeat(4, 250px);

定义间距

grid-gap定义了网格布局的间距大小,接收一个或两个值,如果定义两个值则表示列(column)和行(row)的间距大小。

在两列布局示例中,我们可以如下使用:

.grid {
  display: grid;
  grid-template-columns: 50vw 50vw;
  grid-gap: 1rem;
}

不幸的是,这个间距将会占用容器元素的整体宽度,计算出来就是100vw + 1rem,最终这个布局会导致出现水平滚动条。

viewport导致的水平滚动条

[通过viewport单位创建带间距网格导致的水平滚动条]

为了解决这个空间溢出问题,我们需要些不同的方法来处理,需要用分数单位或者说是FR

分数单位

分数单位标识占用可用空间的份额,如果900px是可用空间,其中的一个元素占有1份,另外的元素占有2份——那么第一个元素的宽度会是900px的1/3,另外的元素是900px的2/3。
修改后用分数代替view-port单位的新代码如下:

.grid {
  display: grid;
  grid-template-columns: 1fr 1fr;
  grid-gap: 1rem;
}

内容对齐

为了对齐示例中的内容,我们在子元素上使用grid布局,并加上对齐属性来定位他们到指定轨道(track),轨道就是一个网格的列或行的某个位置的常见的名称。网格跟Flex布局一样,有一系列对齐的属性——共有四种值——start, center, end, 和stretch,分别对应其子元素所在的轨道。stretch跟其他不太一样,它会将元素从所在轨道的头拉伸到尾。

align-items 和 justify-content

[align-items 和 justify-content]

例子中我们要将内容水平和垂直居中,可以通过在容器上设置下面这些属性:

.center-content {
    display: grid;
    align-items: center;
    justify-content: center;
}

示例地址

使用旧的网格布局实现两栏布局

如果使用旧的网格布局方式创建,我们需要考虑实现中的诸多限制。旧的布局方式不仅没有grid-gap,而且你需要在每一个网格元素上声明网格元素的起始位置,否则默认会设置为1,这样所有的网格都会堆在第一列。

旧版本的布局方式需要通过增加间距作为网格轨道的一部分,也需要设置每个网格从哪里开始:

.grid-legacy {
   display: -ms-grid;
   -ms-grid-columns: 1fr 1rem 1fr; // 取代 gap 间距
}
.grid-legacy:first-child {
   -ms-grid-column: 1;
}
.grid-legacy:last-child {
    -ms-grid-column: 3;
}

旧的布局方式实现对齐和全高度

旧的布局方式跟IE 11中Flexbox有一样的问题,在容器上设置最小高度(min-height)不一定会生效。这个问题通过网格布局来解决更方便。

为了实现这个效果我们在父容器的行属性上使用minmax方法,minmax指定了行或列的最大和最小值。

-ms-grid-rows: minmax(100vh, 1fr);

在子元素上我们声明一个单位为1fr的单列单行的网格:

.ms-cell {
   -ms-grid-columns: 1fr;
   -ms-grid-rows: 1fr;
}

最后,因为我们不能像Flexbox或最新网格布局那样根据父元素对齐,我们必须使用元素自身的对齐方式来对齐:

.ms-align-center {
    -ms-grid-column: 1;
    -ms-grid-column-align: center; // 新型grid布局中的 align-self
    -ms-grid-row-align: center; // 新型grid布局中的 justify-self
}

旧的两列布局示例

到此我们实现了如何创建列、实现间距、内容对齐及对旧的网格布局的支持。接下来让我们实验下如何通过grid实现内边距。

通过CSS网格实现内边距(Negative Space)

网格布局允许你通过grid-column-start属性指定列开始的位置,所以就有了可以在网格内创建内边距的可能性。

使用grid-template-columns和grid-column-start创建内边距

[使用grid-template-columns和grid-column-start创建内边距]

创建内边距的一种方式是在列的实际位置上设置一个数字,空出网格元素的原始空间, 网格元素也会被push到新的网格列。

grid-column-start push

[随着grid-column-start push 第一项]

在上面的内边距示例中,html结构中用一个div包裹另外一个div:

<div class="grid">
    <div class="child"><!-- 内容 --></div>
</div

网格像这样设置:

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

为了让子元素从右侧开始,我们设置子元素从第2列开始:

.child {
    grid-column-start: 2;
}

注意:在Firefox 52中的一个差异导致一个垂直对齐问题——基于FR单位的行不会拉伸得跟整个窗口一样高。为了解决(address)这个问题我们设置子元素为网格项,并给每一行设置一个想要的高度:

.l-grid--full-height {
    grid-template-rows: minmax(100vh, 1fr);
}

设置内边距示例

用内容死区(content dead-zones)创建空白

在四列布局中,给本来在第三列的网格项上设置grid-column-start:2;,那么会找到下一个可用的第二列来填充空间。

网格轨道会跳过某些列,直到找到下一列。我们可以利用这个方法在网格内创建空白,没有内容的网格也会被分配。
[创建空白示例]


[使用grid-template-columns 和 grid-column-start创建空白]

创建行

如果我们想分割布局为四份,我们目前所了解的关于列的布局方式对行同样有效:

.grid {
    display: grid;
    grid-template-columns: 1fr 1fr;
    grid-template-rows: 250px 250px;
}

[同时使用grid-template-columns 和 grid-template-rows创建网格布局]

理想情况下这个示例是没问题的。因为此时每个网格项的内容足够少而不会撑开每行。但随着内容的变化,一切都不一样了。当示例中的内容超出指定行的大小后,看下会发生什么:


[内容超出声明的行高]

我们创建了250px高的两行,如果内容超过每行的高度,将会打破布局并和后面的行的内容重叠。并不是一个我们想要的结果。

灵活的设置最小值

我们在该场景下需要的是设置最小尺寸的能力,但又要允许尺寸可以根据内容弹性变化。这里我们通过上面旧浏览器示例中的minmax关键字实现。

.grid {
    grid-template-rows: minmax(250px, auto) minmax(250px, auto);
}

创建有最小值的弹性行

现在我们已经了解了创建带有内容的行的基础方法,我们开始来创建水平和垂直交错的更复杂网格布局。


[使用grid-column-start和span关键字创建复杂网格布局Unsplash]

创建更复杂的网格

我们开始创建更复杂的网格布局。将网格中的每个网格项设置成占据多条轨道,在一列内,我们能通过grid-column-startgrid-column-end实现,或者通过如下所示更简单的写法:

grid-column: 1 / 3;

用这种实现方式的弊端是难以“模块化”,为了定位每块内容需要写很多代码。span关键字更符合模块化的思路,因为我们能放在任何地方,让网格来控制他。我们可以定义网格项的开始位置,及其占据的轨道数:

.span-column-3 {
    grid-column-start: span 3;
}

任何添加该class的网格将会从其开始位置,占据三个轨道。

[通过span实现的复杂网格]

使用span设计一个布局

我们能设计一个多轨道布局,通过将布局分解为grid布局中的最小单元。本示例中的最小单位是图中高亮的部分。


[通过最小网格单位结合span创建更大的网格]

围绕最小单位,我们能灵活的使用span来创建一些有意思的布局,因为span是可以叠加的——你可以结合列和行的轨道在网格中创建多层级。

不需要媒体查询(media queries)的弹性网格

虽然上面说到的例子能在可用空间内适应变化,但是没有一个是专门为空间变化设计的。网格有两个非常有用的特性来适应可用空间的变化。这两个属性叫‘auto-fit’和‘auto-fill’,像下面这样结合repeat functionminmax function使用:

grid-template-columns: repeat(auto-fit, minmax(250px, 1fr));

这些值代替了repeat中的数字,并计算在每条轨道上会填充多少行或列。二者之间最大不同是当一条轨道上空白的溢出时的他们的处理方式不同。

auto-fit尝试在不导致列溢出的情况下,放置该列能处理的最大数量的重复元素。当没有足够的空间来放置更多的元素时,之后的元素将会放到下一行,不能填满的空间将会被保留。

auto-fill

[示例:auto-fill. auto-fill会保留后面空间,反之auto-fit会让空白收缩为0px]

auto-fill的表现跟auto-fit类似,但是任何的空白空间都会自动收缩,同时这一行的元素也会被拉升——类似flexbox的效果,列会随着可用空间变小发生折叠。

grid-auto-fit示例

[grid-auto-fit示例]

依赖媒体查询的布局跟窗口大小关系很大,这不够模块化——系统内的组件应该能根据可用空间自适应。那么在实践中会是什么样的呢?

auto-fit

[grid auto-fit的真实示例]

[网格auto-fit示例]

这只是冰山一角

我们已经经历了快十五年的CSS浮动为主的布局方式,我们上面学习了几乎所有你能用float实现的布局——CSS网格布局是这个领域的新代表,仍然还有许多东西需要去尝试和学习。

现在最重要的步骤是开始使用它。在构建、创建更多高级布局的时候会很方便。网格布局还有不少未知领域,一旦我们更好地理解其能力并开始与其他特性结合,我们便能用更少代码创造更多有趣、灵活的布局,并能减少些框架抽象的麻烦。

如果你感兴趣并想进一步探究CSS网格,可以试下Rachel Andrew的例子,这里面通过带解释说明的实例探讨了CSS网格布局的每一个特性。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • 简介 CSS Grid布局 (又名"网格"),是一个基于二维网格布局的系统,旨在改变我们基于网格设计的用户界面方式...
    咕咚咚bells阅读 2,488评论 0 4
  • 简介CSS网格布局(又称“网格”),是一种二维网格布局系统。CSS在处理网页布局方面一直做的不是很好。一开始我们用...
    _leonlee阅读 64,976评论 25 173
  • 前言 温馨提示:本文较长,图片较多,本来是想写一篇 CSS 布局方式的,但是奈何 CSS 布局方式种类太多并且实现...
    sunshine小小倩阅读 3,124评论 0 59
  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,741评论 1 92
  • 第十四章 这一夜,二人从天界仙境,聊道凡尘人事,未曾刻意回避什么,不经意提到,聊一聊,话题自个儿就偏到十万八千里之...
    若莎Elsa阅读 7,496评论 1 52