关于flex 上下文中自动 margin的问题(完整例子)


Posted in HTML / CSS onMay 20, 2021

为了引出本文的主题,先看看这个问题,最快水平垂直居中一个元素的方法是什么?

水平垂直居中也算是 CSS 领域最为常见的一个问题了,不同场景下的方法也各不相同,各有优劣。嗯,下面这种应该算是最便捷的了:

<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 水平垂直居中元素

如何让margin: auto在垂直方向上居中元素

嗯。这里其实就涉及了一个问题,如何让margin: auto在垂直方向上生效?

换句话说,传统的display: blockBFC(块格式化上下文)下,为什么margin: auto在水平方向可以居中元素在垂直方向却不行?

通常我们会使用这段代码:

div {
    width: 200px;
    height: 200px;
    margin: 0 auto;
}

让元素相对父元素水平居中。但是如果我们想让元素相对父元素垂直居中的话,使用margin: auto 0是不生效的。

BFC 下margin: auto垂直方向无法居中元素的原因

查看 CSS 文档,原因如下,在 BFC 下:

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.

CSS2 Visual formatting model details: 10.6.3

简单翻译下,在块格式化上下文中,如果margin-leftmargin-right都是 auto,则它们的表达值相等,从而导致元素的水平居中。( 这里的计算值为元素剩余可用剩余空间的一半)

而如果margin-topmargin-bottom都是 auto,则他们的值都为 0,当然也就无法造成垂直方向上的居中。

使用 FFC/GFC 使margin: auto在垂直方向上居中元素

OK,这里要使单个元素使用margin: auto在垂直方向上能够居中元素,需要让该元素处于 FFC(flex formatting context),或者 GFC(grid formatting context) 上下文中,也就是这些取值中:

{
    display: flex;
    display: inline-flex;
    display: grid;
    display: inline-grid;
}

FFC 下margin: auto垂直方向可以居中元素的原因

本文暂且不谈 grid 布局,我们业务中需求中更多的可能是使用 flex 布局,下文将着重围绕 flex 上下文中自动 margin 的一些表现。

嗯,也有很多前端被戏称为 flex 工程师,什么布局都 flex 一把梭。

查看 CSS 文档,原因如下,在dispaly: flex下:

Prior to alignment via justify-content and align-self, any positive free space is distributed to auto margins in that dimension.

CSS Flexible Box Layout Module Level 1 -- 8.1. Aligning with auto margins

简单翻译一下,大意是在flex 格式化上下文中,设置了margin: auto的元素,在通过justify-contentalign-self进行对齐之前,任何正处于空闲的空间都会分配到该方向的自动 margin 中去

这里,很重要的一点是,margin auto 的生效不仅是水平方向,垂直方向也会自动去分配这个剩余空间。

使用自动 margin 实现 flex 布局下的space-between | space-around

了解了上面最核心的这一句 :

在通过justify-contentalign-self进行对齐之前,任何正处于空闲的空间都会分配到该维度中的自动 margin 中去

之后,我们就可以在 flex 布局下使用自动 margin 模拟实现 flex 布局下的space-between以及space-around了。

自动 margin 实现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 { ... }

效果如下:

关于flex 上下文中自动 margin的问题(完整例子)

那么下面的 CSS 代码与上面的效果是完全等同的:

.g-flex {
    display: flex;
    // justify-content: space-around;
}
 
li {
    margin: auto;
}

CodePen Demo -- margin auto 实现 flex 下的 space-around

自动 margin 实现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将不再生效,也就是这里存在一个优先级的关系。

使用自动 margin 实现 flex 下的align-self: flex-start | flex-end | center

自动 margin 能实现水平方向的控制,也能实现垂直方向的控制,原理是一样的。

margin: auto模拟 flex 下的align-self: flex-start | flex-end | center,可以看看下面几个 Demo:

CodePen Demo -- margin auto 实现 flex 下的 align-self: center

CodePen Demo -- margin auto 实现 flex 下的 align-self: flex-end

不同方向上的自动 margin

OK,看完上面的一大段铺垫之后,大概已经初步了解了 FFC 下,自动 margin 的神奇。

无论是多个方向的自动 margin,抑或是单方向的自动 margin,都是非常有用的。

再来看几个有意思的例子:

使用margin-left: auto实现不规则两端对齐布局

假设我们需要有如下布局:

关于flex 上下文中自动 margin的问题(完整例子)

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 之后的剩余空间。

当然,不一定是要运用在第一个或者最后一个元素之上,例如这样的布局,也是完全一样的实现:

关于flex 上下文中自动 margin的问题(完整例子)

<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;
}

关于flex 上下文中自动 margin的问题(完整例子)

Codepen Demo -- nav list by margin left auto

垂直方向上的多行居中

OK,又或者,我们经常会有这样的需求,一大段复杂的布局中的某一块,高度或者宽度不固定,需要相对于它所在的剩余空间居中:

关于flex 上下文中自动 margin的问题(完整例子)

这里有 5 行文案,我们需要其中的第三、第四行相对于剩余空间进行垂直居中。

这里如果使用 flex 布局,简单的align-self或者align-items好像都没法快速解决问题。

而使用自动 margin,我们只需要在需要垂直居中的第一个元素上进行margin-top: auto,最后一个元素上进行margin-bottom: auto即可,看看代码示意:

<div class="g-container">
    <p>这是第一行文案</p>
    <p>这是第二行文案</p>
    <p class="s-thirf">1、剩余多行文案需要垂直居中剩余空间</p>
    <p class="s-forth">2、剩余多行文案需要垂直居中剩余空间</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 上下文中自动 margin的问题(完整例子)

嗯,这个需求如果能够使用 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-topmargin-bottom的值如果是 auto,则他们的值都为 0
  • flex 格式化上下文中,在通过justify-contentalign-self进行对齐之前,任何正处于空闲的空间都会分配到该方向的自动 margin 中去
  • 单个方向上的自动 margin 也非常有用,它的计算值为该方向上的剩余空间
  • 使用了自动 margin 的 flex 子项目,它们父元素设置的justify-content以及它们本身的align-self将不再生效

最后

更多精彩 CSS 技术文章汇总在我的Github -- iCSS,持续更新,欢迎点个 star 订阅收藏。

到此这篇关于flex 上下文中自动 margin(完整例子)的文章就介绍到这了,更多相关flex 自动 margin内容请搜索三水点靠木以前的文章或继续浏览下面的相关文章,希望大家以后多多支持三水点靠木!

 
HTML / CSS 相关文章推荐
35款精致的 CSS3 和 HTML5 网页模板 推荐
Aug 03 HTML / CSS
css3实现动画的三种方式
Aug 24 HTML / CSS
HTML5 Plus 实现手机APP拍照或相册选择图片上传功能
Jul 13 HTML / CSS
HTML5边玩边学(3)像素和颜色
Sep 21 HTML / CSS
利用HTML5中的Canvas绘制一张笑脸的教程
May 07 HTML / CSS
使用HTML5的Notification API制作web通知的教程
May 08 HTML / CSS
利用HTML5 Canvas API绘制矩形的超级攻略
Mar 21 HTML / CSS
基于HTML5实现类似微信手机摇一摇功能(计算摇动次数)
Jul 24 HTML / CSS
HTML5触摸事件实现移动端简易进度条的实现方法
May 04 HTML / CSS
HTML5 manifest离线缓存的示例代码
Aug 08 HTML / CSS
利用Node实现HTML5离线存储的方法
Oct 16 HTML / CSS
CSS布局之浮动(float)和定位(position)属性的区别
Sep 25 HTML / CSS
CSS的class与id常用的命名规则
CSS 制作波浪效果的思路
HTML通过表单实现酒店筛选功能
面试必问:圣杯布局和双飞翼布局的区别
May 13 #HTML / CSS
HTML5简单实现添加背景音乐的几种方法
May 12 #HTML / CSS
CSS 实现多彩、智能的阴影效果
解析CSS 提取图片主题色功能(小技巧)
You might like
一个php作的文本留言本的例子(三)
2006/10/09 PHP
Zend Framework教程之Bootstrap类用法概述
2016/03/14 PHP
javascript-简单的日历实现及Date对象语法介绍(附图)
2013/05/30 Javascript
上传图片js判断图片尺寸和格式兼容IE
2014/09/01 Javascript
angularjs客户端实现压缩图片文件并上传实例
2015/07/06 Javascript
jQuery获取页面及个元素高度、宽度的总结——超实用
2015/07/28 Javascript
JavaScript动态设置div的样式的方法
2015/12/26 Javascript
基于jQuery实现Tabs选项卡自定义插件
2016/11/21 Javascript
利用node.js实现自动生成前端项目组件的方法详解
2017/07/12 Javascript
vue深入解析之render function code详解
2017/07/18 Javascript
JS实现简单的选择题测评系统代码思路详解(demo)
2017/09/03 Javascript
浅谈AngularJS中使用$resource(已更新)
2017/09/14 Javascript
Vue.js中的computed工作原理
2018/03/22 Javascript
详解Vue中数组和对象更改后视图不刷新的问题
2018/09/21 Javascript
基于JS实现数字动态变化显示效果附源码
2019/07/18 Javascript
Vue 通过公共字段,拼接两个对象数组的实例
2019/11/07 Javascript
浅谈Vue SSR中的Bundle的具有使用
2019/11/21 Javascript
JS Thunk 函数的含义和用法实例总结
2020/04/08 Javascript
用jQuery实现抽奖程序
2020/04/12 jQuery
Pycharm学习教程(5) Python快捷键相关设置
2017/05/03 Python
python+pillow绘制矩阵盖尔圆简单实例
2018/01/16 Python
VSCode下好用的Python插件及配置
2018/04/06 Python
Sanic框架路由用法实例分析
2018/07/16 Python
python数据结构之线性表的顺序存储结构
2018/09/28 Python
Python爬虫实现使用beautifulSoup4爬取名言网功能案例
2019/09/15 Python
Python WebSocket长连接心跳与短连接的示例
2020/11/24 Python
法国票务网站:Ticketmaster法国
2018/07/09 全球购物
德国家用电器购物网站:Premiumshop24
2019/08/22 全球购物
澳大利亚最大的在线美发和美容零售商之一:My Hair Care & Beauty
2019/08/24 全球购物
乌克兰数字设备、配件和智能技术的连锁商店:KTC
2020/08/18 全球购物
艺术系应届生的自我评价
2013/10/19 职场文书
客服专员岗位职责
2014/02/28 职场文书
2015年班主任德育工作总结
2015/05/21 职场文书
centos8安装nginx1.9.1的详细过程
2021/08/02 Servers
Kubernetes控制节点的部署
2022/04/01 Servers
spring 项目实现限流方法示例
2022/07/15 Java/Android