转载请注明出处:葡萄城官网,葡萄城为开发者提供专业的开发工具、解决方案和服务,赋能开发者。
【年末促销】葡萄城 2018 岁末福利火热放送中
原文出处:https://www.sitepoint.com/flexbox-css-flexible-box-layout/
近几年,CSS领域出现了一些复杂的专用布局工具,用以代替原有的诸如使用表格、浮动和绝对定位之类的各种变通方案。Flexbox,或者说是弹性盒子布局模块(Flexible Box Layout Module)是这些新布局工具中的第一个,接着是CSS网格布局模块(CSS Grid Layout Module)。我们会在本文给出一个易于理解的flexbox入门介绍。
随着CSS网格布局的引入,你可能会问flexbox布局是否真的还有必要。虽然它们所能做的事情有一些重叠,但其各自在CSS布局中有着非常特别的目的。一般来说,flexbox在一维场景(比如,一串类似的元素)下有最佳应用,而网格是二维场景下理想的布局方案(例如整个页面的元素)。
即便如此,flexbox仍可以用于整个页面的布局,这样它能为那些还不支持网格布局的浏览器提供合适的兼容处理。(必须承认,网格布局正在大多数现代浏览器中快速得到支持,不过对flexbox的支持仍然更为广泛,所以如果你想让你的布局在稍微老旧的浏览器中也生效,使用flexbox作为网格布局的降级方案是很容易的)。
使用Flexbox的好处
flexbox的一些好处是:
- 页面元素能被任意方向地放置(靠左、靠右、从上往下甚至从下往上)
- 布局内容的可视顺序能够被反转或重排
- 元素大小能“弹性”适应可用空间,并根据容器或者兄弟元素进行相应地对齐
- 能轻松实现等列宽布局(与每一列里面的内容无关)
为了阐述其多样的属性和可能性,让我们假设下面有这样的布局用例:
header content here
main content here
首先,是把元素一起放进.main
里,比如,<nav>
和<aside>
。如果没有flexbox,我们可能会把三个元素全部进行浮动,但想让它按理想的方式工作显得并不直观。而且,按传统的方式做这件事会出现一个众所周知的问题:每一列仅仅和它的内容一样高。因此,你可能需要把三个元素都设置为统一的高度,或者使用某种黑科技。
让flexbox来救场吧。
让我们Flex
flexbox的要点是出现在display
属性上的flex
值,它需要被设置在容器元素上。如此设置会让它的子元素变成“弹性项目(flex item)”。这些弹性项目拥有一些易于使用的默认属性。比如,它们被紧挨着放置,那些没有特别指明宽度的元素自动占满了剩余的空间。
因此,如果你给.main
设置了display:flex
,它的子元素.content
就被自动挤在<nav>
和<aside>
之间。不需要再多余的计算,多么方便是吧?作为附加奖赏,所有三个元素神奇地拥有了相同的高度。
.main {
display: flex;
}
请查看下面的例子,包含了所有的细节:flexbox-demo-1。
项的顺序:Flebox的order
属性
另外一个flexbox的能力,是能够轻松改变元素的显示顺序。让我们假设你为一个客户制作了上面的布局,而她现在想要.content
出现在<nav>
之前。
通常,你需要深入到HTML源码中去,在那里改变元素的顺序。而有了Flexbox,你可以完全使用CSS完成这项任务。只需把.content
的order
属性设置为-1
,那么这一列就会出现在前面,这本例就是最左边。
.main {
display: flex;
}
.content {
order: -1;
}
本例中,你不需要改变其他列的order
。例子在flexbox-demo-2。
如果你倾向于显式地为每一列指定order
,你可以将.content
的order
设为1
,把<nav>
的order
设为2
,把<aside>
的设为3
。
HTML源码独立于CSS的Flexbox样式
但你的客户并不满足。她想让<footer>
成为页面的第一个元素,显示在<header>
之前。那好,同样的,flexbox是你的朋友(虽然像在此例中,可能你得跟你的客户好好谈谈,而不是跟随指示)。因为你不仅要重排列内部元素,还要重排外部的,display:flex
规则将被设置在<div class="example">
之上。注意这里是如何在页面中嵌套使用flex容器来达到你想要的效果的。
因为<header>
,<main class="main">
和<footer>
相互堆叠着,你需要首先设置一个垂直上下文,它能够通过设置flex-direction:column
来快速完成。还有,<footer>
的order
被设置为-1
,如此一来它就出现在页面的最上头。就这么简单。
.example {
display: flex;
flex-direction: column;
}
footer {
order: -1;
}
所以,如果你想把一行元素修改为一列,或者相反,你可以使用flex-direction
属性,并设置它相应地为column
或row
(row
是默认值)。
完整的例子在flexbox-demo-3。
然而,强大的能力也到来了更多的责任:谨记,一些用户可能会使用键盘来导航你的基于flexbox的网站,如果你HTML源码中元素的顺序和屏幕上显示的有所出入,那么无障碍访问的能力就成为需要认真对待的问题。如果想了解得更多,请不要错过HTML源码顺序 vs CSS显示顺序,网站无障碍访问和易用性的专家Adrian Roselli针对这个问题给出了深入讨论。
如何在Flexbox中对齐子项
Flexbox能非常直观地处理子项的水平对齐和垂直对齐。
你可以使用align-items
对flex容器中的所有子项设置统一的对齐。如果你想给个别元素设置不同的对齐方式,使用align-self
。元素的对齐方式跟它所在父容器的flex-direction
有关。如果它的值是row
(意味着元素水平排列),对齐方式是指在垂直轴上。如果flex-direction
被设置为column
(意味着元素垂直排列),对齐方式就是指在水平轴上。
例如,你让一些元素在容器中分别有不同的对齐方式,你需要:
- 设置每个元素的
align-self
属性为合适的值。可能的值有:center
,stretch
(元素撑满它的容器),flex-start
,flex-end
和baseline
(元素被放置在父容器的baseline上) - 把容器元素设置为
display:flex
- 最后,注意父容器的
flex-direction
属性,因为它关系到子元素的对齐方式。
.example {
display: flex;
flex-direction: column;
}
.red {
align-self: center;
}
.blue {
align-self: flex-start;
}
.pink {
align-self: flex-end;
}
试试在下面的例子中,把父容器的flex-direction
在row
和column
之间切换,看看它们引起的实时变化。
例子flexbox-demo-4.html。
如果想要容器中所有的元素有统一的对齐方式,你可以在容器上使用align-items
。可以的值有center
,flex-start
,flex-end
,stretch
(默认值:子项被拉伸以适应它们的容器)和baseline
(子项被放置在父容器的baseline上)。
.example {
display: flex;
align-items: center;
}
像往常一样,试着把父容器的flex-direction
在row
和column
之间切换,看看它们如何影响着你设置align-items
值时所发生的作用。
例子flexbox-demo-5.html
在Flexbox里两端对齐
另一个控制对齐的属性是justify-content
,当你想让多个元素等分空间时非常有用。
可接受的值有:center
,flex-start
,flex-end
, space-between
(元素利用主轴之间的空间而排布)和space-around
(元素利用主轴之前、之间和之后的空间而排布)。
例如,在之前你一直使用的简单HTML模板里,你可以在<main>
里找到三个元素:<nav>
,.content
和<aside>
。之前,它们都被挤在页面的左边。如果你想让它们之间有一些空间,但是不让第一个元素的左边有空间,也不想让最后一个元素的右边有空间,你可以把.main
(即它们的父容器)里的justify-content
设置为space-between
。
.main {
display: flex;
justify-content: space-between;
}
也试一下设置为space-around
,观察不同的结果。例子在flexbox-demo-6。
在上面的例子中,我同样把<header>
中的文字水平和垂直对齐了,分别是把justify-content
(水平居中)和align-items
(垂直居中)都设置为center
。
header {
height: 100vh;
display: flex;
justify-content: center;
align-items: center;
}
Flexbox中弹性子项的大小
使用flex
属性,你能够对照flex容器中其他元素来控制弹性子项的大小。
这个属性是以下独立属性的简写:
flex-grow
:一个数字,指明元素如何相对其他flex项来拉伸flex-shrink
:一个数字,指明元素如何相对其他flex项来收缩flex-basis
:元素的长度。可接受的值有:auto
,inherit
或者一个数字后面紧跟着%
,px
,em
或其他长度单位。
例如,想得到三个等宽的列,只需给每一列设置flex:1
,其他什么都不用做:
nav, aside, .content {
flex: 1;
}
如果你需要.content
占据<nav>
和<aside>
的两倍宽,那么就把.content
设为flex:2
,让其他两个为1
。
例子在flexbox-demo-7.html。
那仅仅是对flex
属性最简单的应用。同样可以设置flex-grow
,flex-shrink
和flex-basis
这些值,不过那超出本文的话题范围了。
进一步的资源
如果你准备好继续前进,并想学着精通flexbox的更多东西,请查看下面的资源:
- Flexbox SitePoint上Guy Routledge制作的一个付费课程
- Building Mega Menus with Flexbox
- How 3 Modern Tools are Using Flexbox Grids
- Make Forms Fun with Flexbox.
总结
如你所见,如果我们想控制元素在网页中的布局,flexbox可以让我们的生活更加轻松。它非常稳固和可靠,让以前那些我们每天使用的诸如使
让容器坍缩之类的奇技淫巧,成为了过去。
像我们说的,如今,在针对整个页面进行布局时,CSS网格是更好的方案,但我们仍然值得去了解flexbox能做的那些事情。flexbox的最佳应用场景,体现在对元素的一维排列上,但如果有需要,它也能在稍老旧的浏览器中,为CSS网格布局提供方便的替代方案。