介紹
Flexbox(伸縮布局盒) 是 CSS3 中一個新的布局模式,為了現(xiàn)代網(wǎng)絡(luò)中更為復(fù)雜的網(wǎng)頁需求而設(shè)計。本文將介紹 Flexbox 語法的技術(shù)細節(jié)。瀏覽器的支持越來越快,所以當 Flexbox 被廣泛支持并應(yīng)用時你將會快人一步。如果你想知道它是什么并是如何工作的,不妨仔細了解下吧!
為什么需要伸縮布局盒(Flexbox)?
作者長期以來使用表格、浮動、行內(nèi)塊元素和其他 CSS 屬性來布局網(wǎng)站內(nèi)容。然而,這些并不是為復(fù)雜的頁面和網(wǎng)頁應(yīng)用而設(shè)計的。不管是簡單的垂直居中,還是靈活的網(wǎng)格布局都很難靠一己之力輕易實現(xiàn),因此成就了 CSS 網(wǎng)格框架。但是,如果真的需要那么多項目來實現(xiàn)做這些事,為什么不讓它來的更簡單些呢?Flexbox 的目的就是改變這一切。
規(guī)范狀態(tài)和瀏覽器支持情況
Flexbox 規(guī)范的相關(guān)工作已經(jīng)進展了3年。不同的瀏覽器也實現(xiàn)了不同的實驗版本。在2012年9月,Flexbox 語法的第三個主要修訂版本進入到候選推薦階段。這意味著 W3C 認為當前的語法是穩(wěn)定的,并鼓勵瀏覽器開發(fā)商去實現(xiàn)它。
Flexbox 規(guī)范時間表:
- 2009年7月 工作草案 (display: box;)
- 2011年3月 工作草案 (display: flexbox;)
- 2011年11月 工作草案 (display: flexbox;)
- 2012年3月 工作草案 (display: flexbox;)
- 2012年6月 工作草案 (display: flex;)
- 2012年9月 候選推薦 (display: flex;)
Flexbox 已經(jīng)被瀏覽器快速支持。Chrome 22+, Opera 12.1+, 和 Opera Mobile 12.1+ 已經(jīng)支持了本文中所描述的 Flexbox。Firefox 18 和 Blackberry 10 也很快就會實現(xiàn)。我推薦大家使用已經(jīng)支持的瀏覽器來閱讀本文和查看例子。
概念和術(shù)語
雖然現(xiàn)在我們可以使用 Flexbox 輕松創(chuàng)建布局,而不會像以前那樣難以理解,但我們?nèi)匀恍枰ㄒ恍r間去熟悉到底如何使用 Flexbox。新的術(shù)語和概念可能會是我們使用 Flexbox 時的一個障礙,所以讓我們先來了解以下它們。
Flexbox 由 伸縮容器 和 伸縮項目 組成。通過設(shè)置元素的 display 屬性為flex
或inline-flex
可以得到一個伸縮容器。設(shè)置為flex
的容器被渲染為一個塊級元素,而設(shè)置為inline-flex
的容器則渲染為一個行內(nèi)元素。
這里的示例創(chuàng)建了一個伸縮容器。
1234 |
|
1234 |
|
本文中所有的示例都會帶有相應(yīng)的瀏覽器廠商前綴。
伸縮容器中的每一個子元素都是一個伸縮項目。伸縮項目可以是任意數(shù)量的。伸縮容器外和伸縮項目內(nèi)的一切元素都不受影響。簡單地說,F(xiàn)lexbox 定義了伸縮容器內(nèi)伸縮項目該如何布局。
Flex Lines 伸縮行
伸縮項目沿著伸縮容器內(nèi)的一個 伸縮行 定位。通常每個伸縮容器只有一個伸縮行。
這個示例展示了2個項目在默認情況下的定位:沿著一個水平伸縮行從左至右顯示。
Writing Modes 書寫模式
在你設(shè)計 Flexbox 時的有一個重要的部分是更改伸縮行的方向。默認情況下,伸縮行和文本方向一致:從左至右,從上往下。
這是 W3C 關(guān)于一個名為書寫模式的新特性工作草稿。書寫模式是一個新的方法,讓你可以從右往左寫,甚至豎著寫,就像你知道的某些語言一樣。
書寫模式是一個正在進行的計劃,但是 Chrome 已經(jīng)率先支持了direction
CSS 屬性。如果我們在上一個例子中設(shè)置方向為rtl
(從右往左) 那么不僅僅文字會從右往左書寫,而且 伸縮行也改變了方向,并更改了頁面的布局。
這也許就是 Flexbox 為什么如此抽象難懂的地方。當你正在制作一個語言不確定的頁面時你不能簡單的只是說“上”、“下”、“左”、“右”。
The Main Axis and the Cross Axis 主軸和側(cè)軸
為了描述抽象的書寫模式,F(xiàn)lexbox 使用 主軸 和 側(cè)軸的概念。伸縮行跟隨主軸。側(cè)軸則垂直于主軸。
起點、終點和各軸的方向的名稱如下:
- 主軸起點 Main Start
- 主軸終點 Main End
- 主軸方向 Main Direction (有時候也成為伸縮流方向 Flow Direction)
- 側(cè)軸起點 Cross Start
- 側(cè)軸終點 Cross End
- 側(cè)軸方向 Cross Direction
在繼續(xù)了解之前明白主軸和側(cè)軸是至關(guān)重要的。Flexbox 中的一切都和這些軸有關(guān)。在我們所有的例子中,書寫模式都是從左至右,從上到下,但是你需要記住并不是所有的 Flexbox 都是這樣的。
伸縮容器的屬性
flex-direction 伸縮流的方向
flex-direction
允許你更改伸縮容器的主軸方向。flex-direction
的默認值是row
。該值表示伸縮項目根據(jù)書寫模式
的方向布局。再次提醒,默認是從左至右,從上到下。其他的值如下:
- row-reverse: 主軸起點和主軸終點交換。如果書寫模式是從左至右,伸縮項目則是從右往左顯示。
- column: 主軸和側(cè)軸交換。如果書寫系統(tǒng)是垂直的,那么伸縮項目也是垂直顯示的。
- column-reverse: 和 column 一樣,但是方向相反。
讓我們把前一個示例中的flex-direction
改為column
。
現(xiàn)在我們的伸縮項目就是垂直顯示的了。
justify-content 主軸對齊
伸縮容器的justify-content
屬性用于調(diào)整主軸上伸縮項目的位置。可能的值為:
- flex-start (默認)
- flex-end
- center
- space-between
- space-around
這里我們設(shè)置justify-content
為center
讓伸縮項目在主軸上居中對齊:
flex-start
,flex-end
, 和center
一看就懂。space-between
和space-around
則是分配伸縮項目之間空白空間的不同方法。這張規(guī)范中的圖示很好的解釋了一切:
align-items 側(cè)軸對齊
align-items
是一個和justify-content
相呼應(yīng)的屬性。align-items
調(diào)整伸縮項目在側(cè)軸上的定位方式。可能的值有:
- flex-start (默認)
- flex-end
- center
- baseline
- stretch
這里我們設(shè)置align-items
為center
讓伸縮項目在側(cè)軸上居中對齊:
和之前一樣,flex-start
,flex-end
, 和center
的意義顯而易見。stretch
也很簡單:它會將伸縮項目從側(cè)軸起點拉伸到側(cè)軸終點。baseline
則是讓伸縮項目與它們的基線對齊。基線根據(jù)伸縮項目的內(nèi)容計算得到。下面這張來自W3C標準的圖例很好的解釋了這些屬性:
flex-wrap 伸縮行換行
目前為止,每個伸縮容器都有且只有一個伸縮行。使用flex-wrap
你可以為伸縮容器創(chuàng)建多個伸縮行。這個屬性接受以下值:
- nowrap (默認)
- wrap
- wrap-reverse
如果flex-wrap
設(shè)置為wrap
,在一個伸縮行容不下所有伸縮項目時,伸縮項目會換行到一條新增的伸縮行上。新增的伸縮行根據(jù)側(cè)軸的方向添加。
我們使用flex-wrap
來看個例子:
wrap-reverse
和 wrap 一樣,只是新的伸縮行會被添加到側(cè)軸的反方向上。
align-content 堆棧伸縮行
align-content
會更改flex-wrap
的行為。它和align-items
相似,但是不是對齊伸縮項目,它對齊的是伸縮行。可能你已經(jīng)想到了,它接受的值也很相似:
- stretch (默認)
- flex-start
- flex-end
- center
- space-between
- space-around
這些值與justify-content
和align-items
中的值一樣。
在這個例子中,我們設(shè)置align-content
為center
:
flex-flow 伸縮方向與換行
flex-flow
是flex-direction
和flex-wrap
的縮寫。
flex-flow: [flex-direction] [flex-wrap]
舉個例子:
1234 |
|
1234 |
|
伸縮項目的屬性
一個伸縮項目是一個伸縮容器的子元素。伸縮容器中的文本也被視為一個伸縮項目。
伸縮項目中內(nèi)容與普通流一樣。舉例來說,當一個伸縮項目被設(shè)置為浮動,你依然可以在這個伸縮項目中放置一個浮動元素。
伸縮項目都有一個 主軸長度(Main Size) 和一個 側(cè)軸長度(Cross Size)。主軸長度是伸縮項目在主軸上的尺寸。側(cè)軸長度是伸縮項目在側(cè)軸上的尺寸。或者說,一個伸縮項目的寬或高取決于伸縮容器的軸,可能就是它的主軸長度或側(cè)軸長度。
下面的屬性可以調(diào)整伸縮項目的行為:
order 顯示順序
order
是最簡單明了的屬性。設(shè)置伸縮項目的 order 可以調(diào)整它們渲染時的順序。在這個例子中,我們設(shè)置其中一個伸縮項目的order
為 -1,于是它被提前到了其他伸縮項目的最前面。
如果需要文檔順序和顯示順序不同時,這就是個很有用的功能了。
margin 外邊距
你應(yīng)該對margin: auto;
的這種用法很熟悉。在伸縮盒中,它也能做同樣的事情,但是更加強大。一個 "auto" 的 margin 會合并剩余的空間。它可以用來把伸縮項目擠到其他位置。
這里我們在第一個伸縮項目上聲明了margin-right: auto;
,導(dǎo)致了所有的剩余空間被合并到那個元素的右邊去了:
這里我們使用margin: auto;
來重現(xiàn)經(jīng)典CSS布局中的圣杯:真·垂直居中:
align-self 側(cè)軸對齊
伸縮項目的align-self
屬性會覆蓋該項目的伸縮容器的align-items
屬性。它的值和align-items
一樣:
- stretch (默認)
- flex-start
- flex-end
- center
- baseline
在這個例子中我們?yōu)槊總€伸縮項目應(yīng)用了不同的align-self
值:
我在例子中包含了2個基線對齊的伸縮項目,因為它們的對齊需要互相作用。
flex 伸縮性
現(xiàn)在我們終于要開始設(shè)置伸縮盒的伸縮性了。flex
指定了一個伸縮項目該如何分配主軸上的剩余空間。
讓我們一次把所有的常見值都看一遍吧。
flex: [number]
這個語法指定了一個數(shù)字,代表了這個伸縮項目該占用的剩余空間比例。
在這個例子中,第一個伸縮項目占用了 2/4 的剩余空間,而另外兩個各占用了 1/4 的剩余空間。
如果把每個伸縮項目都設(shè)置為 1 的話,那么剩余空間就會被平均分配了。
flex: initial
一個flex
屬性值被設(shè)為initial
的伸縮項目,在有剩余空間的情況下不會有任何變化,但是在必要的情況下會被收縮。
flex: auto
一個flex
屬性值被設(shè)為auto
的伸縮項目,會根據(jù)主軸自動伸縮以占用所有剩余空間。
auto
目前僅在 Opera 12.11 尚有效,在 Chrome 23.0.1271.95 上無效。你可以通過使用flex: 1;
來達到一樣的效果。
flex: none
一個flex
屬性值被設(shè)為none
的伸縮項目,在任何情況都不會發(fā)生伸縮。
flex 縮寫
flex
也可以把flex-grow
,flex-shrink
, 和flex-basis
這3個縮寫為1個聲明:
flex: [flex-grow] [flex-shrink] [flex-basis]
大多數(shù)情況下沒必要使用這種語法。另外,它需要一個更容易理解的伸縮算法。如果你覺得自己挺厲害的,到規(guī)范里看一下吧。
當然你也可以將flex-grow
,flex-shrink
, 和flex-basis
作為單個屬性分開來設(shè)置。但我強烈反對這種方式:當使用flex
縮寫時,即使沒有某些值沒有設(shè)置也能獲得更合理的默認值。
visibility 疊加項目
當該值生效時,應(yīng)用visibility: collapse;
和visibility: hidden;
與display: none;
的效果是不一樣的。如果是collapse
,該元素會影響伸縮容器的側(cè)軸長度,但不會被現(xiàn)實或占用主軸的空間。如果你想動態(tài)添加或移除伸縮項目又不會影響伸縮容器的側(cè)軸長度,這將會非常有用。
目前為止,visibility: collapse;
還沒有被讓任何瀏覽器正確的實現(xiàn)。現(xiàn)在visibility: collapse;
還和visibility: hidden;
實現(xiàn)著一樣的效果。我希望能盡快得到改觀。
你可以在 這里 看到collapse
應(yīng)該是如何工作的。
總結(jié)
如你所見,伸縮布局盒(Flexbox) 是一個強大的新型布局模式,將會給網(wǎng)站帶來革命性的布局方法,但它也需要一種全新的思考方式。希望這篇文章能為你使用伸縮布局盒構(gòu)建網(wǎng)站帶來幫助。我不知道你怎么想,但是在我看來未來是美好的。
原文:Dive into Flexbox (http://weblog.bocoup.com/dive-into-flexbox)
參考資料:Css3-flexbox (http://www.w3.org/html/ig/zh/wiki/Css3-flexbox)
轉(zhuǎn)自:[譯]深入了解 Flexbox 伸縮盒模型:http://c7sky.com/dive-into-flexbox.html