譯者注:此文適合有一定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 實現的是一個比較老的網格標準,現在並不受支援。
網格佈局不是將零散的塊拼到一起
相信我,很快你就能掌握它的.
網格佈局只能像左邊那樣,以矩形的單元塊組合起來。並不能像右圖那樣,由一堆零散的多邊形(跟俄羅斯方塊那樣的塊)拼湊。
設計網格佈局並不是為了取代彈性盒,相反,它是彈性盒的一種補充
雖然網格佈局和彈性盒在某些方面起到相似的作用,而且你可以發現,很多人用彈性盒來實現網格佈局,但這並不是設計彈性盒的初衷。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)。
1 2 3 4 5 6 7 8 |
.grid { align-items: stretch; } .griditem { display: flex; align-items: center; } |
給grid-column-end設定負值,意想不到的有用
在小螢幕下,寫一個12列的網格,所有格子的跨度都12列。
你可以用網格這樣做:
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;
來讓你的頁面始終是從左到右貫穿的。
1 2 3 4 |
/* For small screens */ .span4, .span6, .spanAll { grid-column-end: -1; } |
在大螢幕上,你想要儘可能的接近12列,但是在移動端,一行大概是1~4列。用media來改變grid-template-columns
是非常容易的。
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,和一些大圖啥的。
對於小螢幕,我們可以這樣寫:
1 2 3 |
.wide { grid-column: 1 / 3; /* start at 1, end at 3 */ } |
不幸的是,當我們換到大屏的時候,一行12列,這些元素將僅僅佔滿前兩列,並不會佔滿12列,我們需要定義新的grid-column-end
,並且把他的值設為 13
. 這種方式比較麻煩,還有一種簡單的方式,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.
網格區域可以命名,並使用一些隱含的名字
使用grid-template-areas
和grid-line-numbers
是兩種控制行數的屬性,你也可以兩個同時用。你可以使用那些隱含的行名去設定你的網格。
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.
定義網格區域的另一種方式
就像給網格的行命名,特殊的行名能用於設定網格區域,語法是這樣的:
1 2 3 4 5 6 |
.grid { grid-template-areas: "header header header" "main main sidebar" "footer footer footer"; } 、 |
如果你的佈局設計(太多列的佈局!沒列都要起名字,可能還需要空元素)中有很多空的區域,這種寫法稍微有點麻煩。所以對於網格是有另一種寫法的,在這種寫法中,名字是什麼無所謂,只要你合理利用到[name-start]
和 [name-end]
,也能達到自己的佈局目的。下面是一個例子:
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
來確定行數的話,它會非常適合。
相等尺寸網格(equal sized box layout)使用vmin單位
雖然你可以在CSS網格中使用任意尺寸的行或列,但是如果想要相等大小的格子並是響應式的,你就需要使用vmin單位了。
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。這種方法,通過調整容器視口的百分比和內容位置,做到適配各種尺寸的螢幕。
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.
絕對定位
當我們絕對定位一個網格元素的時候,這個元素會跑到它的容器中,我們可以用grid-column 和 grid-row來定位它。正常情況下,絕對定位使元素脫離文件流,它最適合的使用場景就是想要讓元素重疊,並不打亂其他佈局元素。除非你為每個元素宣告grid-column-start
和 grid-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 item)的順序
如果你使用過彈性盒(flexbox)的order
屬性,那你已經知道一些相關的知識了。所有的網格元素都有一個預設的order值0。所以如果給一個網格元素設定 order: 1;
,這個元素將在所有元素的後面。 你可以給order
屬性設定負值,讓它跑到所有item的前面。
See the Pen Order value by CSS GRID (@cssgrid) on CodePen.
grid中 minmax()的坑
想不想要整行隨著內容的寬度而變寬,直到他們達到最大寬度,這種情況你可能想嘗試使用 minmax()
:
1 2 3 4 |
.grid { display: grid; grid-template-columns: repeat(3, minmax(1fr, 300px)); } |
不幸的是,像上面這樣看似簡單,實際上是不行的。如果max小於min的話,css會被忽略。在minmax()
中fr
不能使用。實際上實現這個需求很容易,在grid-template-columns
或 grid-template-rows
中使用auto
,這樣item就可以隨著內容增大而變大了。 See the Pen The value of auto vs fr by CSS GRID (@cssgrid) on CodePen.
我們可以設定一個 max-width
:
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.
如果你給每一個網格線命名了的話,寫佈局將容易的多
有多種辦法供你選擇,如果你就想多寫點,你可以給多行設定多個名字。
1 2 3 |
.grid { grid-template-columns: [col1-start] 1fr [col1-end col2-start] 1fr [col2-end]; } |
最簡單的命名約定使用網格自動編號。不是去寫 [col2],而是寫為col 2
1 2 3 |
.griditem1 { grid-column-start: col 2; } |
和span
關鍵字組合使用,我們就不用去寫column-start和column-end中的各種網格線數字了,這樣能直觀許多。
1 2 3 4 5 6 7 |
.grid { grid-template-columns: repeat(4, [col] 100px); } .griditem1 { grid-column: col 2 / span 2; } |
fr單位為什麼那個的重要,讓你擺脫麻煩的計算
想象一下一行上四等列這種佈局,使用百分比是多麼的容易grid-template-columns: 25% 25% 25% 25%
。
但是當想用grid-gap
屬性的時候那?如果設定grid-gap: 10px
,那麼這一行上將有三個空隙,每個10px,整體的寬度就是100% + 30px,大於100%滾動條就出來了。雖然可以通過計算來解決,但是如果使用fr,這太容易了grid-template-columns: 1fr 1fr 1fr 1fr
See the Pen fr unit vs percentage by CSS GRID (@cssgrid) on CodePen.
網格佈局中第二個我較噁心的點
沒有辦法強制自動佈局演算法留下一些行和列是空的。
grid-gap
可以讓我們設定內容間的距離。grid-row-gap
和grid-column-gap
能設定行或列之間的間隙,可是如果我想讓第一行和第二行相距10px,第二行和第三行相距50px,用現有的網格是沒法實現的,除非建個空行佔位。
你可能見到過像下面這樣寫grid-template-area
的::
1 2 3 4 5 6 |
grid-template-rows: "header header header" "main main main" " . . ." "secondary secondary secondary" "footer footer footer"; |
應該提供一個比較聰明的辦法,讓佈局演算法去做這件事。不幸的是,這樣寫也沒用。此語法簡單地表示,我們不想將第三行變成一個命名的網格區域。可是grid-template-rows將仍然在那結束。
Some design advice: You don’t necessarily need 12 columns (and columns need not be uniform in size)
一些設計上的建議: 你不一點需要12列網格 (每一列不一定大小一致)
12列網格算是web design的預設配置了。Bootstrap引導大家用12列網格,導致很多框架都是12列網格。12既能被3整除也能被4整除,能讓我們有更多種佈局擺放方式。1行12列,1行6列,1行4列,1行3列,1行2列
雖然有些人喜歡每一個專案總是使用相同的網格,但是你應該去思考你真正需要的,有時候沒有必要有更多的列,你應該建立一個網格,對針對你的內容去佈局,而不是一個12列網格到處用。
看看這個例子 Gridset. Gridset是一個製作網格非常有用的工具, 但是原生CSS的網格不需要你使用任何工具,但是可以看看它展示的一些良好的網格設計。
看看我寫的例子,CSS原生網格是多麼的自由啊:
See the Pen text layout with CSS Grid module by CSS GRID (@cssgrid) on CodePen.