在不少方面HTML和CSS是一个强大的内容发布机制——易学、灵活和强大。但复杂的布局是他不擅长的。若是你想建立一个简单的图片与文本的布局,那么还算简单,可是制做一个复杂的多列布局,要作的众多浏览器的兼容一致那仍是很复杂的。咱们一般都是使用浮动或者其余方法来实现这个目的,而其中出现的bug和浏览器的差别性使用对布局失去兴趣。 css
为了应对这种状况,CSS3包含了许多模块,使用不一样的布局更加容易。咱们已经在其余文章中看到多栏布局和媒体生成的内容分页,如今咱们将注意力转向CSS3的 Flexbox布局模块。 html
Flexbox一般能让咱们更好的操做他的子元素布局,例如: web
听起来至关有用,不是吗?接下来让咱们更详细的探索它。 chrome
注:这篇文章使用的是Flexbox最后语法,目前支持浏览器:Opera Mobile12.1+、Opera12.5+、Firefox18+(partial)和chrome。Chrome须要添加浏览器前缀“-webkit-”,Opera支持标准语法,不用添加任何前缀。Firefox有部分支持,也须要添加前缀“-moz-”,同时须要设置一个标志(到firefox浏览器地址栏中输入:about:config,搜索“flexbox”,找到以后双击“layout.css.flexbox.enabled”,设置他的“value”值为“true”)。注意,其余浏览器除了opera自2009年支持flexbox以来,都使用旧的语法规则,真的不该该使用这些过期的语法。必定要记住阅读和使用2012年之前有关于flexbox的文章和代码时,flexbox都使用的老语法。Chris Coyie有一篇文章《“Old” Flexbox and “New” Flexbox》,这篇文章能更好的告诉你如何阅读一个过期的文章。 浏览器
在详细阅读这篇文章以前,咱们颇有必要先了解flexbox的几个经常使用术语,这样有助于你们对后文的理解。 app
下图是一个row伸缩容器中各类方向与大小术语的示意图: 布局
在制做flex例子开始,先让咱们考虑一个简单的示例来讲明flexbox布局的容易性。咱们来看一个“内容布富”的页脚,这个页脚包含了三个子元素,这种而脚的类型也是至关的典型。让咱们去接触细节,这个页脚包含了连接、联系信息和版权声明。咱们想把这三个元素显示在一个水平线上,而且但愿连接部分的宽度是其余两个内容宽度的两倍。今天以前咱们经常经过浮动子元素,设置子元素宽度,并使用不一样的margin或padding来调整对齐。在全部的维度,没有固定的宽度值,这样每每是不许确的,也让事情变得更僵化。但Flexbox在这里就能够帮助咱们实现须要的效果。 flex
我制做的flexible例子。例子很简单,除了页脚,还有一个包含三个盒子的flexible布局。 网站
注:看看我最终实现的例子(如上图所示)。你能够看到一个美丽的设计,主内容一列和页脚经过“position:fixed”固定在页面底部。布局灵活,页面的页脚子元素比例能够根据主要内容列的百分比进行调整。您还会注意到,我用了一些媒体查询来转变小屏幕宽度的布局。 this
咱们如何开始使用Flexbox呢?大多数的Flexbox属性都应用于父容器的元素上。由于Flexbox,你能够指定你想要制定的一个容器,使用一个特殊的值显示属性,就像这样:
footer { display: flex; }
接下来你可使用“flex-row”属性来指定子元素布局是在一行仍是一列显示。若是你愿意,你能够定义关键词“wrap”,来指定内容容器在新的一行(当父元素容器过小,flexbox元素想在同一行显示)。在咱们的例子中,我在footer中设置了“row wrap”
footer { display: flex; flex-flow: row wrap; }
flex-flow是用来伸缩行换行,flex-flow属性是同时设定“flex-direction(伸缩流的方向)”和“flex-wrap(伸缩行换行)”属性的缩写,两个属性决定了伸缩容器的主轴与侧轴。此属性主要适用于伸缩容器。在这篇文章的例子中,主要是“footer”元素。
flex-direction属性能够用来设定伸缩容器的主轴的方向,这也决定了用户代理配置伸缩项目的方向。主要适用于伸缩容器,主要包括如下几个值:
flex-wrap属性主要用来控制伸缩容器是单行仍是多行,也决定了侧轴方向一新的一行的堆放方向。主要适用于伸缩容器,主要包括如下几个值:
关键词“wrap”明显的变得颇有意义。
注:“flex-flow”在这里采用了缩写,他主要包括两个属性“flex-direction”(值为row、column、row-reverse和column-reverse,后面的两个属性值与row和column方向相反)和“flex-wrap”(值为:wrap、no-wrap和wrap-reverse)。
若是你要想flexbox工做正常,你有些概念必须得清楚,好比说让flexbox正常工做的主轴和侧轴,他们看上去有点像X轴和Y轴,但仍是有所差异的。主轴的方向主要是用来肯定flex的主方向,因此你子元素要么放置在一行,要么放置在一列。侧轴主要垂直于主轴运行,以下图所示:
flexbox的主轴与侧轴
Flexbox有一系列的方法来帮助你调整伸缩项目沿着主轴和侧轴的对齐。
第一个咱们一块儿看看“align-items”属性,它充许您调整伸缩项目在侧轴的对齐方式,主要包括如下几个值:
能够用来设置伸缩容器中包括匿名伸缩项目的全部项目的对齐方式。
为了更形像的理解”align-items”各个属性值对应在侧轴上的效果,能够参考下图:
这些都是术语解释,只有动手去尝试,调整不一样的值,才能知道各个属性值所表明的运行效果,你们能够看看下图所运行的效果。对于这个例子,我采用的是“stretch”属性值。
footer { display: flex; flex-flow: row wrap; align-items: stretch; }
上图是align-items各个属性值运行后的效果,从上至下依次是:flex-start、center、flex-and和stretch。
每每全部列表项的内容没法填满父元素的整个高度,特别是在不知道宽度和高度的视窗变化之下。不少状况之下为了让这些列实现等高效果,是一件多么可怕的事情,并且不少时间都浪费在一些呆反的解决方案和处理兼容上,好比说设置一个等高的效果,使用假的列?
还有一个主要属性“justify-content”使用的也比较多,这个属性主要用来设置伸缩项目沿主轴的对齐方式,从而调整伸缩项目之间的间距。设置了这个属性,在主轴方向上设置的任何margin都不会起做用。所以我特地建立了一个例子来证实这点。
在这个例子中,我为伸缩项目设置了一个百分比宽度:
#first { width: 25%; } #second { width: 40%; } #third { width: 25%; }
而后在伸缩容器设置了一个值,证实伸缩项目在主轴方向的margin不起做用:
footer { display: flex; flex-flow: row wrap; align-items: stretch; justify-content: space-around; }
这个值至关的不错,伸缩项目会平均地分布在行里,两端保留一半的空间。其余可用的值以下:
经过伸缩容器中的三个不一样颜色的项目,展现五种「justify-content」关键字的效果。
你也能够在实例中本身动手尝试一下,这几个值给伸缩项目在主轴上会带来什么样的变化,下图是示例中五种不一样属性值效果的截图:
上图是“justify-content”五种属性值的效果,从上到下依次是:flex-start、center、flex-end、space-between和space-around。
您还能够调准伸缩行在伸缩容器里的对齐方式。不过他会更改flex-wrap的行为,好比说:“wrap”。align-content和align-items类似,可是不是对齐伸缩项目,它对齐的是伸缩行,其主要包括如下几个值:
align-content各关键字对多行的伸缩容器的效果。
传统上不改变元素的结构要改变元素的布局顺序一直是一个痛苦的事情。不过在Flexbox中,你能够经过“order”属性来修改伸缩项目的布局顺序(在不调整结构前提之下)。这个属性一直接受的整数值——称为系数集——也称为排序组,会出如今伸缩项目中。拿前面的例子来讲,默认状况连接块是第二个子元素,以下图所示:
默认状况footer子元素排序是:contact、links、copyright。
默认状况之下,全部的伸缩项目的顺序组都是“0”。咱们能够很容易的给每一个伸缩项目设置不一样的顺序值。更高的值会排在后面,而原来的HTML结构并不会有任何变化。因此在个人示例中,我将连接块设置了order值为“1”
#second { order: 1; }
页脚已作出新的排序:contact、copright、links。
order属性是用来设置伸缩项的显示顺序,默认状态下,用户代理会用伸缩项目出如今源文档的次序配置这些伸缩项目。order属性透过将元素分到有序号的组以控制元素出现的顺序。在伸缩布局中,order属性控制伸缩项目在伸缩容器里的顺序。
order取值越大,越排在后面。而且order能够取负值。
Flexbox最强大的特性是可以经过“flex-flow”属性设置伸缩项目的流动方向,或者能够经过“flex”属性设置一个可用的空间。它的取值能够有三个部分(flex-grow、flex-shrink、flex-basis)。让咱们一个一个来尝试,看看他们的影响。首先添加的是“flex-grow”:
#first { flex: 1; } #second { flex: 1; } #third { flex: 1; }
这些没单位的值是做为一个比例,他们决定于伸缩容器中有多少个伸缩项目。能够决定伸缩项目在伸缩容器中的空间大小。若是每一个都设置为1,每一个伸缩项目在伸缩容器内都相等。若是你给其中一个伸缩项目设置为2,那么这个伸缩项目会占用空间是其余伸缩项目的两倍。
#first { flex: 1; } #second { flex: 2; } #third { flex: 1; }
你也能够像下面同样设置flex-basis的值:
#first { flex: 1 200px; } #second { flex: 2 300px; } #third { flex: 1 250px; }
首先flex-basis的值主要取决于伸缩项目的width或者高,同时取决于流动方向。而后,剩下的空间根据flex-grow给伸缩项目最后宽度来划分。因此伸缩项目会沿着主轴线大小为200px、300px和250px,总共750px。若是伸缩容器沿主轴方向是950px,这样就会多出一个200px空间,那么这多出的200px空间将分配给伸缩项目。第一个和第三个伸缩项目将获得50px的空间,由于他的flex-grow值是“1”,他们最终的空间是250px和300px。第二个伸缩项目将得到100px空间,由于他的flex-grow值为“2”,他的最后空间大小为400px。
flex的第三部分不多使用,但咱们能够看看他的使用方法,你也能够将flex-shrink像下面那样设置:
#first { flex: 1 1 400px; } #second { flex: 2 3 600px; } #third { flex: 1 2 400px; }
flex-shrink称为收缩比率。这个值只有伸缩项目在没主轴方向溢出伸缩容器才会发挥做用。他们充当比例值,但这回指的是溢出量,将这个溢出量按比例分配给每一个伸缩项目,用于防止伸缩容器溢出。
好比说,咱们伸缩容器沿主轴方向宽度是1100px,按照上面的示例代码计算,咱们的伸缩项目会超出300px(伸缩项目沿主轴方向总值为1400px),这个时候经过flex-shrink收缩他们:
这样flex-shrink使用伸缩项目获得一个较小的宽度。
个人例子中最终将值设置成:
#first { flex: 1 0 7rem; } #second { order: 1; flex: 2 0 8rem; } #third { flex: 1.5 0 7rem; }
flex用来决定伸缩项目的伸缩性。一个伸缩容器会等比地按照各伸缩项目的扩展比率分配剩余空间,也会按照收缩比率缩小各项目以免溢出。
flex属性能够用来指定伸缩长度的部件:扩展比率flex-grow,收缩比率flex-shrink以及伸缩基准值flex-basis。当一个元素是伸缩项目时,flex属性将代替主轴长度属性决定元素的主轴长度。若元素不是伸缩项目,则flex属性没有效果。
上图是一个显示「绝对」伸缩(以零为基准值开始)与「相对」伸缩(以项目的内容大小为基准值开始)差别的图解。这三个项目的伸缩比例分别是「1」、「1」、「2」。
在个人例子中结合一个多行flxbox(flex-flow:row wrap)和伸缩长度flex(如:flex:1 0 7rem)以及媒体查询实现了一个完美的效果。在不一样视窗宽度下,伸缩项目在伸缩容器中能够平滑的进行变化。如图所示:
flexbox对应的响应式代码:
footer { display: -webkit-flex; -webkit-flex-flow: row wrap; -webkit-align-items: stretch; display: -moz-flex; -moz-flex-flow: row wrap; -moz-align-items: stretch; display: -ms-flex; -ms-flex-flow: row wrap; -ms-align-items: stretch; display: flex; flex-flow: row wrap; align-items: stretch; } #first { -webkit-flex: 1 0 7rem; -moz-flex: 1 0 7rem; -ms-flex: 1 0 7rem; flex: 1 0 7rem; } #second { -webkit-order: 1; -webkit-flex: 2 0 8rem; -moz-order: 1; -moz-flex: 2 0 8rem; -ms-order: 1; -ms-flex: 2 0 8rem; order: 1; flex: 2 0 8rem; } #third { -webkit-flex: 1.5 0 7rem; -moz-flex: 1.5 0 7rem; -ms-flex: 1.5 0 7rem; flex: 1.5 0 7rem; } @media screen and (max-width: 1000px) { body { width: 100%; } #fixed { left: 0%; right: 0%; } } @media screen and (max-width: 520px) { #fixed { position: static; } section { padding: 1rem 2rem; } body { padding-bottom: 0; background-image: none; background-color: white; } footer { padding: 0 1rem 0; } #first { -webkit-flex: 1 0 10rem; -moz-flex: 1 0 10rem; -ms-flex: 1 0 10rem; flex: 1 0 10rem; } #second { -webkit-flex: 1 0 10rem; -moz-flex: 1 0 10rem; -ms-flex: 1 0 10rem; flex: 1 0 10rem; } #third { -webkit-flex: 1 0 10rem; -moz-flex: 1 0 10rem; -ms-flex: 1 0 10rem; flex: 1 0 10rem; } } @media screen and (max-width: 380px) { section { padding: 1rem 1rem; } footer div { right: 1rem; } }
flex还有其余的值,Common values of flex作了详细的介绍。其中“auto”和“initial”很是有用。设置“flex:auto”时,伸缩容器中的伸缩项目(至关于flex: 1 1 auto)将使用其大小根据任何width/height或者min-width/min-height设定,它将扩展占用一个比例的任何自由空间可用,但在没有额外的自由空间将缩小以适应其内容。结合min-width将可能会产生一些有趣的效果,能够看看flex auto的实例。在这个实例中,父容器设置了“flex-flow:row”,三个子元素设置了“flex:auto”并设置了一个“min-width”。所以无论是多行仍是单行,任何多余的空间都扩展到伸缩项目上。而后会收缩整齐,由于他变小后,充许子元素从新自适应。
试着将“flex:auto”修改为“flex:initial”(至关于flex:0 1 auto),你会看到,当有多余的空间时,第三个子元素安容器大小再也不增长,但仍然须要收缩。
Flexbox是一个全新的布局模块,不识别浏览器将会忽略他。这样看起来是一个破坏者,阻止你使用它。然而,它不须要。例如,你可使用浮动或者表格在桌面系统下布局你的网站,在小屏幕下选择使用flexbox,或许只是为了移动主要内容。
若是你在使用flexbox以前将页面全部元素(好比:nav、header、footer等)设置成“display:block”,在支持flexbox的浏览器会将内容自动伸缩,以适应整个设备的宽度。在不识别flexbox的浏览器中会按顺序整块的排列,但并不会影响你的阅读。
或者,表格能够把一个元素从源秩序让内容从开始到结束的排列。看到这样的例子,你可能会笑的。
我但愿这篇文章能够告诉你使用浮动和清除浮动,这些呆会的布局技术是如何的可怕,同时也明确的告诉你flexbox是如何的有用。你也能够看看媒体查询与flexbox结合实现响应式布局是多么的伟大。