全部奇技淫巧都只在方寸之間。css
幾乎從咱們踏入前端開發這個領域開始,就不停地接觸不一樣的佈局技術。從常見的浮動到表格佈局,再到現在大行其道的flex佈局,css佈局技術一直在不斷地推陳出新。其中網格佈局(grid)做爲css3的產物,它更加貼近網頁設計師所使用的佈局策略,學習並利用好它可讓咱們免受不少佈局困擾。html
雖然網格佈局好處有不少,但學習起來並不簡單,緣由是用來設置佈局的屬性實在太多,其中光是做用於父容器的屬性就有17種,再加上子元素屬性有10種,另外還有這些屬性值的不一樣取值方式。這些對於記憶來講絕對是個不小的負擔。那麼這麼多屬性以及用法,要如何在短期內消化掉呢?在接下來這篇文章裏,我將針對這27種屬性以及它們各自的用法,分享我獨家的學習策略,但願對你們的學習有所幫助。前端
CSS做爲一種網頁排版設計語言,其核心的設計思想必然要遵照相關的領域知識。網格佈局是一種二維佈局結構,它是由縱橫相交的兩組網格線造成的框架性佈局結構。網頁設計者能夠利用這些由行(row)和列(column)造成的框架性結構來佈局設計元素。 在定義一種網格佈局結構的時候,咱們須要在父容器上描述要佈局的主體框架結構。爲了描述這一框架結構,咱們就須要給它的基本構成元素命名。一個網格佈局的構成元素能夠歸納爲如下幾種概念:css3
牢記上述這些概念是以後熟練掌握和應用網格佈局的基礎。算法
要熟練掌握一門技術,核心是找到最基本的套路,而後不斷練習從而能夠在以後的實踐過程當中減小決策的時間。因此,這一部分主要就是介紹網格佈局構建過程當中的一些經常使用套路。 這裏咱們要解決的問題是,如何利用最基本的規則來構建出理想的佈局模型。在佈局過程當中,歸根結底須要處理的就兩種頁面元素:父容器和子元素。前者主要用來設置基礎的佈局框架,至關於建築中的設計藍圖,然後者就是用來進行個性化的佈局調整。所以我我的概括了在使用網格佈局過程當中的套路是:針對父容器元素進行設置須要三個步驟:定框架、設間隔和找對齊,對子元素來講有兩個步驟:擺位置和找對齊。我把它們統稱爲**"32構建之法"**。bash
在這一小節中,我將把重心主要放在網格佈局中全部用到的27個屬性名的講解上,而取值邏輯將在最後一部分進行統一介紹。微信
設置父容器的網格佈局的第一步就是將父容器的盒模型設置爲grid
,這樣就能觸發渲染引擎的網格佈局算法。框架
.parent {
display: grid;
}
複製代碼
接着咱們要開始準備**"畫線"**,即設置所需行和列的基礎線。這些線條將構成咱們接下來進行佈局排布的基礎模板(template)。在畫線過程當中,咱們須要分別根據行(row)和列(column)兩個維度進行設置。你須要畫幾條線,就設置幾個值(不包括邊框),其取值是軌道(track)的大小。這裏我先畫出一個3x3的網格框架,代碼以下:wordpress
.parent {
display: grid;
grid-template-rows: 100px 100px 100px;
grid-template-columns: 100px 100px 100px;
}
複製代碼
在這裏你也能夠選擇使用縮寫形式同時爲行和列設置值,採用/
分隔開:函數
.parent {
display: grid;
grid-template: 100px 100px 100px / 100px 100px 100px;
}
複製代碼
畫完線後,下一步咱們能夠選擇爲這些線條和線條之間造成的網格區域(area)進行命名,這樣在後續使用的時候就能直接使用這些名字,便於子元素的定位。
.parent {
display: grid;
grid-template-areas: "a a b"
"c d e"
"c d ."
}
複製代碼
上面這兩步畫線和命名一樣能夠採用縮寫形式進行設置,代碼以下:
.parent {
display: grid;
grid-template: "a a b" 101px
"c d e" 102px
"c d ." 103px / 104px 105px 105px
}
複製代碼
由於使用grid-template
同時設置行列和區域名的寫法比較複雜,爲了講解方便,我把值設置成規律的遞增數字。其中(101, 102,103)
設置的是grid-template-rows
的值,而(104,105,106)
設置的則是grid-template-columns
的值。
到這一步,咱們能夠說已經完成所需工做的一大半了。
間距(gap)的設置在實際開發中是可選的,主要根據網頁設計的需求而定。若是你須要給網格線之間設置間距,咱們能夠在行列兩個維度上分別進行設置, 下面這段代碼將給每一個行和列分別設置10px的間隔:
.parent {
display: grid;
grid-template: 100px 100px 100px / 100px 100px 100px;
grid-row-gap: 10px;
grid-column-gap: 10px;
}
複製代碼
若是採用縮寫形式,上述代碼又能夠簡化成:
.parent {
display: grid;
grid-template: 100px 100px 100px / 100px 100px 100px;
grid-gap: 10px 10px;
}
複製代碼
設置後的效果以下:
有了前面兩個步驟,咱們的網格佈局框架基本上算是搭建得差很少了。每一個子元素都會默認佔據一個網格區域。而在父容器這裏咱們若是有須要,就要進行最後一個步驟:找對齊。所謂對齊方式,能夠分爲內部和外部兩種(前者是針對每一個網格區域的子元素而言,然後者是相對於網格區域自己)。另外在行和列(更專業的術語是main axis和cross axis)上又各自有兩個維度,這就構成了4種設置對齊的方式。
先來處理一下每一個子元素相對網格區域內部的對齊方式:
.parent {
display: grid;
grid-template: 100px 100px 100px / 100px 100px 100px;
grid-gap: 10px 10px;
justify-items: center;
align-items: center;
}
複製代碼
在上面的代碼中我分別在行和列方向上都設置了居中對齊,這樣每一個網格區域中的子元素相對於各自的區域行爲是一致的,都能均勻排布。能夠看到效果以下圖所示:
再來看一下另外一種狀況:
.parent {
display: grid;
width: 500px;
height: 500px;
grid-template: 100px 100px 100px / 100px 100px 100px;
grid-gap: 10px 10px;
justify-content: space-between;
align-content: center;
}
複製代碼
有時候咱們設置的網格不足以覆蓋整個父容器的大小時,好比在上述的例子中整個父容器有500px*500px
的大小,而咱們只設置了300px*300px
的網格區域,這時候就須要指定多出來空間的處理規則。justify-content
和align-content
就是分別在行和列兩個方向上用來解決這個問題的屬性。它們將針對每一個網格區域去設置其在父容器中的對齊方式。
justify
和align
這兩個單詞在方向上比較容易搞混,因此我在記憶上採起的方式是記住justifyrow
和aligncolumn
這兩個合併詞,它們長度差很少。若是你有更好的記憶方式,請留言告訴我。
咱們經過在父容器上搭建好了基礎的框架後,對於大部分子元素來講,就已經可以很好地知足佈局要求了。針對部分子元素,能夠根據需求進行微調。若是要在子元素上進行佈局微調,一般須要如下兩個步驟:擺位置和找對齊。
像下棋同樣,針對子元素的排布,咱們須要給它們指定要擺放的具體位置。要肯定具體位置,能夠利用以前在父容器中所指定的線名和區域名來定位。一種方式是直接經過設置起始行,結束行和起始列,結束列來給子元素劃定它所要擺放的區域,另一種方式是指定要擺放的區域名。
/* 指定起始行,結束行,起始列,結束列 */
.child:first-child {
grid-row-start: 1;
grid-row-end: 2;
grid-column-start: 1;
grid-column-end: 3;
background: red;
}
/* 使用縮寫形式 */
.child:nth-child(2) {
grid-row: 2/3;
grid-column: 2/4;
background: yellow;
}
/* 直接指定區域名 */
.child:nth-child(3) {
grid-area: i;
background: green;
}
複製代碼
這段代碼的效果以下:
還有一種更加靈活的設置位置方式,是指定跨越的行數和列數,關鍵字span
用來控制一次跨越的行數或列數, 如上面第三個子元素能夠改寫成:
.child-nth-child(3) {
grid-row: 2/3;
grid-column: span 2;
}
複製代碼
和父容器中所設置的對齊方式相似,針對個別子元素的對齊處理,咱們能夠按照行列兩組屬性進行分別處理:
/* 列對齊 */
.child:nth-child(1) {
align-self: end;
}
/* 行對齊 */
.child:nth-child(2) {
justify-self: end;
}
/* 採用縮寫形式 */
.child:nth-child(3) {
place-self: center center;
}
複製代碼
靈活性是網格佈局的一大優點,除了採用上述那種手動指定框架結構的方式,網格佈局還有一套自動化佈局的機制,這套機制稱爲**「隱式網格佈局」**。當咱們在網格定義的區域外放置子元素時,或因子元素數量過多而須要更多的網格線時,佈局算法就會自動生成隱式網格。默認狀況下這些隱式網格的大小也會隨着內容尺寸不一樣而變化,而咱們能夠利用屬性grid-auto-rows
和grid-auto-columns
來控制隱式網格的大小。 考慮下面這個例子:
<div class="parent">
<div class="child" style="background: red"></div>
<div class="child" style="background: yellow"></div>
<div class="child" style="background: green"></div>
</div>
複製代碼
.parent {
display: grid;
grid-auto-rows: 100px;
grid-auto-columns: 100px;
}
複製代碼
經過手動在父容器中設置隱式網格大小爲100x100的大小後,效果以下:
若是子元素引用了不存在的行號和列號,父容器會自動生成隱式網格以容納全部子元素:.child:first-child {
grid-row-start: 1;
grid-row-end: 2;
grid-column-start: 1;
grid-column-end: 3;
background: red;
}
複製代碼
有了網格大小的控制,咱們還須要位置的控制。默認狀況下,子元素都是先將行填充滿,容器大小不夠的時候纔會生成新的隱式行。若是要改變這一默認行爲,咱們須要使用grid-auto-flow
屬性來控制:
.parent {
display: grid;
grid-auto-rows: 100px;
grid-auto-columns: 100px;
grid-template-areas: "a b c" "d e f" "g h i";
grid-auto-flow: column;
}
複製代碼
介紹完全部的網格佈局屬性後,咱們再來講一下各類屬性的取值策略。
在CSS中,咱們一般使用px,em等取值單位進行屬性大小的設置,對於靈活的佈局需求來講,百分比也是經常使用的取值單位。這些單位在日常工做中彷佛已經足夠用了。不過,爲了讓佈局可以更加靈活,網格佈局中引入了一種新單位fr
,它是fraction這個單詞的縮寫,意思是容器內剩餘空間的分數比。考慮下面這個例子:
.parent {
height: 100px;
display: grid;
grid-template-columns: 100px 1fr 100px;
}
複製代碼
咱們經過設置100px 1fr 100px
的佈局框架,從而很輕鬆地就實現了兩邊寬度固定,中間自適應的效果。
若是要實現有比例關係的佈局結構,還可使用多個fr
的取值:
.parent {
width: 400px;
height: 100px;
display: grid;
grid-template-rows: 100px 1fr 100px;
grid-template-columns: 1fr 1fr 2fr;
grid-template-areas: "a b c"
}
複製代碼
能夠看到區域a,b,c之間的比例關係就是"1:1:2"的關係。
除了上述這個支持自適應的單位外,網格佈局中還可以使用max-content
和min-content
這組關鍵字來達到自適應的目的。要理解這兩個關鍵字,首先須要理解內在尺寸(intrinsic size)和外部尺寸(extrinsic size)這兩個概念。先說一下extrinsic size
,它的相對值計算是相對於父容器對應的屬性值。咱們知道,width
若是使用百分比單位,其計算值是相對於該元素所在的容器寬度的,好比父容器寬度100px, 子元素設置width: 20%
,那麼它的寬度就是100px * 20% = 20px
。在css3中引入了intrinsic size
則是相對於元素自身尺寸進行計算。max-content
和min-content
就是相對於元素自身內容塊進行計算的屬性值。
min-content
顧名思義是根據元素內容來設置的最小寬度大小,在英文句子中,一般是最長單詞的那個長度,而中文中則是一個字的長度。好比下面這個例子:
.parent {
display: grid;
grid-template-columns: auto min-content auto;
}
複製代碼
能夠看到,中間那個網格的那個寬度就等於scq000
這個單詞的長度。
與min-content
相對應,max-content
會將尺寸設置成內容尺寸能達到的最大寬度。咱們把代碼改爲下面這樣:
.parent {
display: grid;
grid-template-columns: auto max-content auto;
}
複製代碼
有了這兩個屬性值,咱們能夠很容易地讓佈局區域根據內容進行自適應。
函數是用來避免重複性工做的一種有效工具,在網格佈局中提供了一些經常使用CSS函數來方便咱們的工做。
第一個要介紹的是minmax
這個函數。在設置網格框架的過程當中,對於自適應的網格區域,咱們都會設置一個最小值和最大值,這個函數就是用來實現這個目的的。
.parent {
display: grid;
grid-template-columns: 100px minmax(100px, 200px) 100px;
}
/* 最經常使用的狀況是隻設置最小,不設置最大值 */
.parent {
display: grid;
grid-template-columns: 100px minmax(100px, auto) 100px;
}
複製代碼
利用這個函數設置的網格佈局能夠作到很好的自適應,在頁面伸縮過程當中也能保證佈局的穩定性。
另外一個頗有用的函數是fit-content
,它其實是min(maximum size, max(minimum size, argument))
的簡寫,表示將元素寬度收縮到內容寬度。說得通俗點就是,使用這個函數後會儘可能不佔用多餘的空間。若是內容的寬度小於fit-content
中設置的長度,那麼實際子元素寬度是內容寬度。若是內容寬度超出了fit-content
中設置的長度,那麼實際子元素寬度就是設置的那個長度。下面看兩個例子:
.parent {
display: grid;
grid-template-rows: auto fit-content(200px) auto;
}
複製代碼
第一個句子中的長度超出了200px,那麼此時中間網格的寬度是200px。而第二個例子中內容寬度不足200px,此時中間網格的寬度是句子實際佔用的寬度。
最後要介紹的是repeat
函數,它主要用來批量設置框架的間距,這個函數接受兩個參數,第一個參數控制循環次數,第二個參數控制間距大小。讓咱們用這個函數改寫一下上述例子:
.parent {
display: grid;
grid-template-rows: 100px 100px 100px;
grid-template-columns: 100px 100px 100px;
}
/* 利用repeat函數改寫 */
.parent {
display: grid;
grid-template-rows: repeat(3, 100px);
grid-template-columns: repeat(3, 100px);
}
複製代碼
另外,第一個參數除了可使用數字顯示設置網格數量外,還能使用auto-fit
和auto-fill
兩個關鍵字自動分配空間。一般狀況下,這兩個關鍵字的使用效果都差很少,惟一的差異是空餘空間的分配規則。搭配minmax
函數能夠看出區別,以下面這兩個例子:
.parent {
display: grid;
width: 500px;
height: 100px;
grid-gap: 10px;
grid-template-columns: repeat(auto-fit, minmax(100px, 1fr));
}
.parent {
display: grid;
width: 500px;
height: 100px;
grid-gap: 10px;
grid-template-columns: repeat(auto-fill, minmax(100px, 1fr));
}
複製代碼
在單行佈局的時候,若是有空餘空間auto-fit
會將它們平均分配到全部子元素中,而auto-fill
會自動建立空白的列。
咱們在建立網格佈局框架的時候,經過"畫線"來指定基本佈局結構。默認狀況下,網格佈局會給每一條網格線進行命名,命名順序同書寫順序一致:從左到右,由上至下按數字命名。假設咱們指定的是3x3的網格佈局結構,那麼包含邊框線,就會生成4+4=8條線。
除了採用默認命名方式,咱們還可以自定義網格線的名稱以便於後續在子元素定位中使用。.parent {
display: grid;
grid-template-rows: [row-a] 100px [row-b] 100px [row-c] 100px [row-d];
}
複製代碼
對齊是佈局過程當中一個不可缺乏的步驟,它的取值是經過已有的關鍵字來指定的。其中用於網格佈局中對齊的關鍵字有start
,center
, end
和stretch
四個。 咱們先從默認值stretch
看起,這個關鍵字是伸展的意思,因此在默認狀況下網格中的子元素會盡量地填充滿網格區域,因爲是默認值因此日常寫代碼的時候就不太會能夠去用這個關鍵字進行聲明。 接着再來看一下經常使用的對齊取值策略,只須要記住一點就能夠了:用屬性justify-*
,align-*
來分別控制橫軸和縱軸兩個方向,屬性值控制其對齊位置。 start
,center
和end
三個屬性值就分別對應了前中後
三個位置。
.parent {
display: grid;
}
.child:first-child {
justify-self: start;
}
.child:first-child {
justify-self: center;
}
.child:first-child {
justify-self: end;
}
複製代碼
讓咱們從新回到隱式網格那部分,隱式的排列規則是經過指定grid-auto-flow
這個屬性來設置的。它的取值只有三個row
,column
和dense
。上面屬性部分已經介紹過row
和column
兩個屬性值,前者是按行優先來擺放子元素,後者是按列優先來擺放子元素。這裏主要介紹一下dense
這個屬性值。一般狀況下,排列子元素會按照順序填充行或列,若是空間不足的時候,會換行或換列。而使用了dense
屬性後,會盡可能使用空餘空間,考慮下面這段代碼:
.parent {
display: grid;
grid-template-columns: repeat(3, 100px);
}
複製代碼
<div class="parent">
<div class="child">1</div>
<div class="child">2</div>
<div class="child">3</div>
<div class="child">4</div>
<div class="child">5</div>
</div>
複製代碼
默認狀況下,顯示效果是這樣的:
當咱們使用dense
值進行排列的時候,就至關於開啓「緊湊」模式,會盡量利用空餘空間。
.parent {
display: grid;
grid-auto-flow: row dense;
/* 因爲默認爲按行排列,可省略爲dense */
grid-auto-flow: dense;
}
複製代碼
因此,顯示效果以下:
在這篇文章裏,我經過拆解分類全部網格佈局相關的知識點,但願可以爲你們提供一個比較系統的運用網格佈局的指導方法。在實際應用過程當中,沿着"32構建之法」的這個套路來走,能夠節約不少思考和決策時間。另外,因爲這篇文章信息密度可能比較大,但願你們可以多多複習,並跟着例子實際操練幾遍,這樣在實際工做運用中才能如魚得水。最後,送上一張思惟導圖,幫助你們可以一覽本文全部的重點。
www.zhangxinxu.com/wordpress/2…
畫圖調試工具: www.mozilla.org/en-US/firef…
——本文首發於我的公衆號,轉載請註明出處———
最後,歡迎你們關注個人公衆號,一塊兒學習交流。