为了引出本文的主题,先看看这个问题,最快水平垂直居中一个元素的方法是什么?css
水平垂直居中也算是 CSS 领域最为常见的一个问题了,不一样场景下的方法也各不相同,各有优劣。嗯,下面这种应该算是最便捷的了:html
<div class="g-container">
<div class="g-box"></div>
</div>
复制代码
.g-container {
display: flex;
}
.g-box {
margin: auto;
}
复制代码
上面的
display: flex
替换成display: inline-flex | grid | inline-grid
也是能够的。前端
CodePen Demo -- 使用 margin auto 水平垂直居中元素 git
margin: auto
在垂直方向上居中元素嗯。这里其实就涉及了一个问题,如何让 margin: auto
在垂直方向上生效?github
换句话说,传统的 display: block
BFC(块格式化上下文)下,为何 margin: auto
在水平方向能够居中元素在垂直方向却不行?布局
一般咱们会使用这段代码:flex
div {
width: 200px;
height: 200px;
margin: 0 auto;
}
复制代码
让元素相对父元素水平居中。可是若是咱们想让元素相对父元素垂直居中的话,使用 margin: auto 0
是不生效的。flexbox
margin: auto
垂直方向没法居中元素的缘由查看 CSS 文档,缘由以下,在 BFC 下:spa
If both margin-left and margin-right are auto, their used values are equal, causing horizontal centring.翻译
—CSS2 Visual formatting model details: 10.3.3
If margin-top, or margin-bottom are auto, their used value is 0.
简单翻译下,在块格式化上下文中,若是 margin-left
和 margin-right
都是 auto,则它们的表达值相等,从而致使元素的水平居中。( 这里的计算值为元素剩余可用剩余空间的一半)
而若是 margin-top
和 margin-bottom
都是 auto,则他们的值都为 0,固然也就没法形成垂直方向上的居中。
margin: auto
在垂直方向上居中元素OK,这里要使单个元素使用 margin: auto
在垂直方向上可以居中元素,须要让该元素处于 FFC(flex formatting context),或者 GFC(grid formatting context) 上下文中,也就是这些取值中:
{
display: flex;
display: inline-flex;
display: grid;
display: inline-grid;
}
复制代码
margin: auto
垂直方向能够居中元素的缘由本文暂且不谈 grid 布局,咱们业务中需求中更多的多是使用 flex 布局,下文将着重围绕 flex 上下文中自动 margin 的一些表现。
嗯,也有不少前端被戏称为 flex 工程师,什么布局都 flex 一把梭。
查看 CSS 文档,缘由以下,在 dispaly: flex
下:
CSS Flexible Box Layout Module Level 1 -- 8.1. Aligning with auto margins
简单翻译一下,大意是在 flex 格式化上下文中,设置了 margin: auto
的元素,在经过 justify-content
和 align-self
进行对齐以前,任何正处于空闲的空间都会分配到该方向的自动 margin 中去
这里,很重要的一点是,margin auto 的生效不只是水平方向,垂直方向也会自动去分配这个剩余空间。
space-between | space-around
了解了上面最核心的这一句 :
justify-content
和 align-self
进行对齐以前,任何正处于空闲的空间都会分配到该维度中的自动 margin 中去以后,咱们就能够在 flex 布局下使用自动 margin 模拟实现 flex 布局下的 space-between
以及 space-around
了。
space-around
对于这样一个 flex 布局:
<ul class="g-flex">
<li>liA</li>
<li>liB</li>
<li>liC</li>
<li>liD</li>
<li>liE</li>
</ul>
复制代码
若是它的 CSS 代码是:
.g-flex {
display: flex;
justify-content: space-around;
}
li { ... }
复制代码
效果以下:
那么下面的 CSS 代码与上面的效果是彻底等同的:
.g-flex {
display: flex;
// justify-content: space-around;
}
li {
margin: auto;
}
复制代码
CodePen Demo -- margin auto 实现 flex 下的 space-around
space-between
同理,使用自动 margin,也很容易实现 flex 下的 space-between
,下面两份 CSS 代码的效果的同样的:
.g-flex {
display: flex;
justify-content: space-between;
}
li {...}
复制代码
.g-flex {
display: flex;
// justify-content: space-between;
}
li {
margin: auto;
}
li:first-child {
margin-left: 0;
}
li:last-child {
margin-right: 0;
}
复制代码
CodePen Demo -- margin auto 实现 flex 下的 space-between
固然,值得注意的是,很重要的一点:
Note: If free space is distributed to auto margins, the alignment properties will have no effect in that dimension because the margins will have stolen all the free space left over after flexing.
CSS Flexible Box Layout Module Level 1 -- 8.1. Aligning with auto margins
意思是,若是任意方向上的可用空间分配给了该方向的自动 margin ,则对齐属性(justify-content/align-self)在该维度中不起做用,由于 margin 将在排布后窃取该纬度方向剩余的全部可用空间。
也就是使用了自动 margin 的 flex 子项目,它们父元素设置的 justify-content
已经它们自己的 align-self
将再也不生效,也就是这里存在一个优先级的关系。
align-self: flex-start | flex-end | center
自动 margin 能实现水平方向的控制,也能实现垂直方向的控制,原理是同样的。
用 margin: auto
模拟 flex 下的 align-self: flex-start | flex-end | center
,能够看看下面几个 Demo:
OK,看完上面的一大段铺垫以后,大概已经初步了解了 FFC 下,自动 margin 的神奇。
不管是多个方向的自动 margin,抑或是单方向的自动 margin,都是很是有用的。
再来看几个有意思的例子:
margin-left: auto
实现不规则两端对齐布局假设咱们须要有以下布局:
DOM 结构以下:
<ul class="g-nav">
<li>导航A</li>
<li>导航B</li>
<li>导航C</li>
<li>导航D</li>
<li class="g-login">登录</li>
</ul>
复制代码
对最后一个元素使用 margin-left: auto
,能够很容易实现这个布局:
.g-nav {
display: flex;
}
.g-login {
margin-left: auto;
}
复制代码
此时, auto
的计算值就是水平方向上容器排列全部 li 以后的剩余空间。
固然,不必定是要运用在第一个或者最后一个元素之上,例如这样的布局,也是彻底同样的实现:
<ul class="g-nav">
<li>导航A</li>
<li>导航B</li>
<li>导航C</li>
<li>导航D</li>
<li class="g-login">登录</li>
<li>注册</li>
</ul>
复制代码
.g-nav {
display: flex;
}
.g-login {
margin-left: auto;
}
复制代码
Codepen Demo -- nav list by margin left auto
OK,又或者,咱们常常会有这样的需求,一大段复杂的布局中的某一块,高度或者宽度不固定,须要相对于它所在的剩余空间居中:
这里有 5 行文案,咱们须要其中的第3、第四行相对于剩余空间进行垂直居中。
这里若是使用 flex 布局,简单的 align-self
或者 align-items
好像都无法快速解决问题。
而使用自动 margin,咱们只须要在须要垂直居中的第一个元素上进行 margin-top: auto
,最后一个元素上进行 margin-bottom: auto
便可,看看代码示意:
<div class="g-container">
<p>这是第一行文案</p>
<p>这是第二行文案</p>
<p class="s-thirf">一、剩余多行文案须要垂直居中剩余空间</p>
<p class="s-forth">二、剩余多行文案须要垂直居中剩余空间</p>
<p>这是最后一行文案</p>
</div>
复制代码
.g-container {
display: flex;
flex-wrap: wrap;
flex-direction: column;
}
.s-thirf {
margin-top: auto;
}
.s-forth {
margin-bottom: auto;
}
复制代码
固然,这里将任意须要垂直居中剩余空间的元素用一个 div 包裹起来,对该 div 进行
margin: auto 0
也是能够的。
嗯,很是的好用且方便:CodePen Demo -- 自动margin快速垂直居中任意段落
margin-top: auto
实现粘性 footer 布局OK,最后再来看这样一个例子。
要求:页面存在一个 footer 页脚部分,若是整个页面的内容高度小于视窗的高度,则 footer 固定在视窗底部,若是整个页面的内容高度大于视窗的高度,则 footer 正常流排布(也就是须要滚动到底部才能看到 footer),算是粘性布局的一种。
看看效果:
嗯,这个需求若是可以使用 flex 的话,使用 justify-content: space-between
能够很好的解决,同理使用 margin-top: auto
也很是容易完成:
<div class="g-container">
<div class="g-real-box">
...
</div>
<div class="g-footer"></div>
</div>
复制代码
.g-container {
height: 100vh;
display: flex;
flex-direction: column;
}
.g-footer {
margin-top: auto;
flex-shrink: 0;
height: 30px;
background: deeppink;
}
复制代码
Codepen Demo -- sticky footer by flex margin auto
上面的例子旨在介绍更多自动 margin 的使用场景。固然,这里不使用 flex 布局也是能够实现的,下面再给出一种不借助 flex 布局的实现方式:
CodePen Demo -- sticky footer by margin/paddig
自动 margin 仍是很实用的,可使用的场景也不少,有一些上面提到的点还须要再强调下:
块格式化上下文中margin-top
和 margin-bottom
的值若是是 auto,则他们的值都为 0
flex 格式化上下文中,在经过 justify-content
和 align-self
进行对齐以前,任何正处于空闲的空间都会分配到该方向的自动 margin 中去
单个方向上的自动 margin 也很是有用,它的计算值为该方向上的剩余空间
使用了自动 margin 的 flex 子项目,它们父元素设置的 justify-content
以及它们自己的 align-self
将再也不生效
更多精彩 CSS 技术文章汇总在个人 Github -- iCSS ,持续更新,欢迎点个 star 订阅收藏。
好了,本文到此结束,但愿对你有帮助 :)
若是还有什么疑问或者建议,能够多多交流,原创文章,文笔有限,才疏学浅,文中如有不正之处,万望告知。
最后,新开通的公众号求关注,形式但愿是更短的篇幅,质量更高一些的技巧类文章,包括但不局限于 CSS: