页面布局是样式开发的第一步,也是 CSS 最重要的功能之一。
常用的页面布局,其实就那么几个。下面我会介绍5个经典布局,只要掌握了它们,就能应对绝大多数常规页面。
这几个布局都是自适应的,自动适配桌面设备和移动设备。代码实现很简单,核心代码只有一行,有很大的学习价值,内容也很实用。
我会用到 CSS 的 Flex 语法和 Grid 语法,不过只用到一点点,不熟悉的朋友可以先看看教程链接,熟悉一下基本概念。每一个布局都带有 CodePen 示例,也可以到这个网页统一查看。
本文是跟极客大学合作的前端学习讲座的一部分,详见文末说明。
一、空间居中布局
空间居中布局指的是,不管容器的大小,项目总是占据中心点。
CSS 代码如下(CodePen 示例)。
.container { display: grid; place-items: center; }
上面代码需要写在容器上,指定为 Grid 布局。核心代码是place-items
属性那一行,它是一个简写形式。
place-items: <align-items> <justify-items>;
align-items
属性控制垂直位置,justify-items
属性控制水平位置。这两个属性的值一致时,就可以合并写成一个值。所以,place-items: center;
等同于place-items: center center;
。
同理,左上角布局可以写成下面这样。
place-items: start;
右下角布局。
place-items: end;
二、并列式布局
并列式布局就是多个项目并列。
如果宽度不够,放不下的项目就自动折行。
它的实现也很简单。首先,容器设置成 Flex 布局,内容居中(justify-content
)可换行(flex-wrap
)。
.container { display: flex; flex-wrap: wrap; justify-content: center; }
然后,项目上面只用一行flex
属性就够了(CodePen 示例)。
.item{ flex: 0 1 150px; margin: 5px; }
flex
属性是flex-grow
、flex-shrink
、flex-basis
这三个属性的简写形式。
flex: <flex-grow> <flex-shrink> <flex-basis>;
flex-basis
:项目的初始宽度。flex-grow
:指定如果有多余宽度,项目是否可以扩大。flex-shrink
:指定如果宽度不足,项目是否可以缩小。
flex: 0 1 150px;
的意思就是,项目的初始宽度是150px,且不可以扩大,但是当容器宽度不足150px时,项目可以缩小。
如果写成flex: 1 1 150px;
,就表示项目始终会占满所有宽度。
三、两栏式布局
两栏式布局就是一个边栏,一个主栏。
下面的实现是,边栏始终存在,主栏根据设备宽度,变宽或者变窄。如果希望主栏自动换到下一行,可以参考上面的"并列式布局"。
使用 Grid,实现很容易(CodePen 示例)。
.container { display: grid; grid-template-columns: minmax(150px, 25%) 1fr; }
上面代码中,grid-template-columns
指定页面分成两列。第一列的宽度是minmax(150px, 25%)
,即最小宽度为150px
,最大宽度为总宽度的25%;第二列为1fr
,即所有剩余宽度。
四、三明治布局
三明治布局指的是,页面在垂直方向上,分成三部分:页眉、内容区、页脚。
这个布局会根据设备宽度,自动适应,并且不管内容区有多少内容,页脚始终在容器底部(粘性页脚)。也就是说,这个布局总是会占满整个页面高度。
CSS 代码如下(CodePen 示例)。
.container { display: grid; grid-template-rows: auto 1fr auto; }
上面代码写在容器上面,指定采用 Grid 布局。核心代码是grid-template-rows
那一行,指定垂直高度怎么划分,这里是从上到下分成三部分。第一部分(页眉)和第三部分(页脚)的高度都为auto
,即本来的内容高度;第二部分(内容区)的高度为1fr
,即剩余的所有高度,这可以保证页脚始终在容器的底部。
五、圣杯布局
圣杯布局是最常用的布局,所以被比喻为圣杯。它将页面分成五个部分,除了页眉和页脚,内容区分成左边栏、主栏、右边栏。
这里的实现是,不管页面宽度,内容区始终分成三栏。如果宽度太窄,主栏和右边栏会看不到。如果想将这三栏改成小屏幕自动堆叠,可以参考并列式布局。
HTML 代码如下。
<div class="container"> <header/> <div/> <main/> <div/> <footer/> </div>
CSS 代码如下(CodePen 示例)。
.container { display: grid; grid-template: auto 1fr auto / auto 1fr auto; }
上面代码要写在容器上面,指定采用 Grid 布局。核心代码是grid-template
属性那一行,它是两个属性grid-template-rows
(垂直方向)和grid-template-columns
(水平方向)的简写形式。
grid-template: <grid-template-rows> / <grid-template-columns>
grid-template-rows
和grid-template-columns
都是auto 1fr auto
,就表示页面在垂直方向和水平方向上,都分成三个部分。第一部分(页眉和左边栏)和第三部分(页脚和右边栏)都是本来的内容高度(或宽度),第二部分(内容区和主栏)占满剩余的高度(或宽度)。
六、参考链接
- Ten modern layouts in one line of CSS, Una Kravets
- Flex 布局教程
- Grid 布局教程
- grid-template 属性, MDN
前端小课
看了上面的内容,如果你还想进一步学习更多前端知识,欢迎关注 极客大学的前端小课。
前端小课除了 CSS,还讲授 JavaScript,尤其是生产中用得最多的几个框架。这次的授课老师是阿里巴巴前手淘前端团队负责人winter,他可能是目前市场上最大牌的前端讲师。
他结合自己的经验,手把手教大家,如何从零开始自己实现一个类似 React 的简单框架,教你领悟前端框架的原理。它是如何将所有功能封装在一起,暴露接口,给开发者使用,并且还能支持组件。
整个前端小课是 4 天视频课程 + 3 天实践训练,还有助教随时答疑辅导、班主任每天督促 + 配套实战作业提交,只需要 ¥9.8 !
点击这里,或者手机扫描下面的二维码就可报名。
(完)
秀妍 说:
项目正好需要第四种布局,赞!
2020年8月10日 08:44 | # | 引用
何文子 说:
哇哇,小白前天看了另一篇布局博文不太懂,看完阮老师的就豁然开朗了。
2020年8月10日 09:34 | # | 引用
Dane 说:
必须给个赞,用了第二种布局把一个组件进行了优化 简单明了
2020年8月10日 10:34 | # | 引用
littlePuppy 说:
五 圣杯布局
CSS 部分
container打错了
2020年8月10日 10:49 | # | 引用
一个陆人 说:
老师讲的通俗易懂,真的适合新手入门!
2020年8月10日 15:31 | # | 引用
李聪 说:
minmax布局现在浏览器的支持还不太行
2020年8月10日 16:29 | # | 引用
zach 说:
太爱了
2020年8月10日 23:13 | # | 引用
Moo 说:
这在多年前还无法想象,其实反思自己,那些年干吗要向那些低版本浏览器低头呢?哈哈哈哈哈 这千万不能让pm 知道。web前端大环境真的是越来越好了!给阮老师点赞???? 非常实用的整理!
2020年8月11日 04:33 | # | 引用
may 说:
不知道兼容性怎么样
2020年8月11日 08:38 | # | 引用
gloomyStar 说:
哇塞!老师太牛掰了,解释的巨清晰明了!
2020年8月12日 11:26 | # | 引用
小朗 说:
经典
2020年8月14日 16:03 | # | 引用
raotaohub 说:
请问我可以以你的文章为素材 在FCC的HTML-项目4上练习并发布在codepen上吗 我将会在页面说明此情况,请问可以吗。
2020年8月14日 19:19 | # | 引用
mirror 说:
好极了,真是简单明了实用啊!
2020年8月16日 10:58 | # | 引用
Janus 说:
"它是两个属性grid-template-rows(垂直方向)和grid-template-columns(水平方向)的简写形式。".rows是水平方向吧 columns垂直方向
2020年8月18日 11:35 | # | 引用
冬日不飘雪 说:
兼容性不好就很难受
2020年8月20日 23:29 | # | 引用
storm 说:
这个不能算一样代码吧? 按照这个说法, 不管多少代码, 都可以压缩成一行啊!
文章挺好的. 不过有点标题党
2020年8月27日 14:21 | # | 引用
qiqishibing 说:
核心样式确实只有一行
2020年9月 9日 15:08 | # | 引用
Franky 说:
阮老师的文章每次看了都有收获
2020年9月10日 11:53 | # | 引用
Zero Sharp 说:
老师讲的太好了!
2020年9月11日 18:54 | # | 引用
Zen Li 说:
阮老师的文章, 行文直白, 无晦涩而深入浅出. 重点分明, 扣主题而不陷鳞爪. 文字铺陈, 如帷幕徐开, 如行云流水, 读之意义分明, 得其义如掌上观纹, 如洞若观火, 可称道赏心悦目.
2020年10月24日 12:20 | # | 引用
lyxlovelyx 说:
三明治部分,如果只使用
{display:grid;grid-template-rows:auto 1fr auto;}页脚还是会缩在上面,而不是在底部。加上height:100%也还是不行,只有加了100vh才能正常工作。请问这个是正常的吗?因为在这个文章里没看到有这一块内容,是在范例里看到的。
2020年10月28日 08:51 | # | 引用
阿灿 说:
老师,请问要怎么做才能让收缩拉宽浏览器时网页上的照片看起来有柔性呢?
2021年9月24日 20:39 | # | 引用
吴 说:
受益匪浅
2021年10月 4日 11:02 | # | 引用
就 说:
只能说css很垃圾
2021年12月10日 20:12 | # | 引用
it法海 说:
为布局提供了另一种思路,很好。
2022年2月10日 15:11 | # | 引用
itFLicker 说:
现在可以了
2024年4月 9日 22:03 | # | 引用
托码斯 说:
经典布局,做页面布局架构必备啊
2024年8月21日 15:41 | # | 引用