Web/03-CSS进阶/04-如何让一个元素水平垂直居中?.md
2019-11-23 11:27:07 +08:00

386 lines
12 KiB
JavaScript
Raw Blame History

This file contains ambiguous Unicode characters

This file contains Unicode characters that might be confused with other characters. If you think that this is intentional, you can safely ignore this warning. Use the Escape button to reveal them.

![](http://img.smyhvae.com/20191108_2130.png)
> 本文的最新内容将在[GitHub](https://github.com/qianguyihao/Web/tree/master/03-CSS%E8%BF%9B%E9%98%B6)上实时更新。欢迎在GitHub上关注我一起入门和进阶前端。
## 前言
老板的手机收到一个红包为什么红包没居中
如何让一个子元素在父容器里**水平垂直居中**这个问题必考在实战开发中也应用得非常多
你也许能顺手写出好几种实现方法但大部分人的写法不够规范经不起千锤百炼换句话说这些人也就面试的时候夸夸其谈但真的上战场的时候他们不敢这么写也不知道怎么写最靠谱
这篇文章中我们来列出几种常见的写法最终你会明白哪种写法是最优雅的
当然我还会拿出实际应用中的真实场景来举例让你感受一下**标准垂直居中的魅力**
## 如何让一个行内元素文字图片等水平垂直居中
> 行内元素的居中问题比较简单
### 行内元素水平居中
给父容器设置
```
text-align: center;
```
### 行内元素垂直居中
**文字的行高** 等于 **盒子的高度**可以让单行文本垂直居中比如
```css
.father {
height: 20px;
line-height: 20px;
}
```
## 如何让一个块级元素水平垂直居中
> 这一段是本文的核心如何让一个块级的子元素在父容器里水平垂直居中有好几种写法我们一起来看看
### margin: auto 的问题
CSS 中对元素进行水平居中是非常简单的如果它是一个行内元素就对它的父容器应用 `text-align: center`如果它是一个块级元素就对它自身应用 `margin: auto`或者 `margin: 0 auto`
在这里`margin: auto`相当于`margin: auto auto auto auto``margin: 0 auto`相当于`margin: 0 auto 0 auto`四个值分别对应上右下左其计算值取决于**剩余空间**
但是如果要对一个元素垂直居中`margin: auto`就行不通了
比如下面这段代码
```html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
.father{
height: 500px;
background: pink;
}
.son {
width: 300px;
height: 200px;
background: red;
margin: auto;
}
</style>
</head>
<body>
<div class="father">
<div class="son"></div>
</div>
<script></script>
</body>
</html>
```
上面的代码中父元素和子元素都是定宽高的即便在这种情况下我给子元素设置 `margin: auto`子元素依然没有垂直居中
那还有没有比较好的通用的做法呢
### 方式一绝对定位 + margin需要指定子元素的宽高不推荐
```html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
* {
margin: 0;
padding: 0;
}
.father{
position: relative;
min-height: 500px;
background: pink;
}
.son {
position: absolute;
width: 200px;
height: 100px;
background: red;
top: 50%;
left: 50%;
margin-top: -50px;
margin-left: -100px;
}
</style>
</head>
<body>
<div class="father">
<div class="son">子元素的内容</div>
</div>
<script></script>
</body>
</html>
```
**代码解释**我们先让子元素的左上角居中然后向上移动宽度的一半(50px)就达到了垂直居中的效果水平居中的原理类似
**不足之处**要求指定子元素的宽高才能写出 `margin-top` `margin-left` 的属性值
但是在通常情况下对那些需要居中的元素来说其宽高往往是由其内容来决定的不建议固定宽高
### 方式二绝对定位 + translate无需指定子元素的宽高推荐
```html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
* {
margin: 0;
padding: 0;
}
.father{
position: relative;
min-height: 500px;
background: pink;
}
.son {
position: absolute;
background: red;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
}
</style>
</head>
<body>
<div class="father">
<div class="son">子元素的内容</div>
</div>
<script></script>
</body>
</html>
```
这种写法在没有指定子元素宽高的情况下也能让其在父容器中垂直居中因为 translate() 函数中使用百分比值时是以这个元素自身的宽度和高度为基准进行换算和移动的**动态计算宽高**
### 方式3flex 布局待改进
将父容器设置为 Flex 布局再给父容器加个属性`justify-content: center`这样的话子元素就能水平居中了再给父容器加个属性 `align-items: center`这样的话子元素就能垂直居中了
代码举例
```html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
* {
margin: 0;
padding: 0;
}
.father{
display: flex;
justify-content: center;
align-items: center;
min-height: 100vh;
background: pink;
}
.son {
background: red;
}
</style>
</head>
<body>
<div class="father">
<div class="son">子元素的内容</div>
</div>
<script></script>
</body>
</html>
```
上面这种写法不足之处在于给父容器设置属性`justify-content: center``align-items: center`之后导致父容器里的所有子元素们都垂直居中了如果父容器里有多个子元素的话可我明明想让指定的**某个子元素**居中要怎么改进呢
### 方式4 flex 布局 + margin: auto推荐
我们只需写两行声明即可先给父容器设置 `display: flex`再给指定的子元素设置我们再熟悉不过的 `margin: auto`即可让这个指定的子元素在**剩余空间**水平垂直居中大功告成
代码举例
```html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Document</title>
<style>
* {
margin: 0;
padding: 0;
}
.father{
display: flex;
min-height: 100vh;
background: pink;
}
.son {
margin: auto;
background: red;
}
</style>
</head>
<body>
<div class="father">
<div class="son">子元素的内容,想水平垂直居中</div>
<div class="son2">这个元素不想水平垂直居中</div>
</div>
<script></script>
</body>
</html>
```
请注意当我们给父容器使用 Flex 布局 子元素的`margin: auto`不仅能让其在水平方向上居中**垂直方向上也是居中的**
参考文章[探秘 flex 上下文中神奇的自动 margin](https://www.cnblogs.com/coco1s/p/10910588.html)
## 垂直居中的典型应用场景红包幕帘/弹窗
### 问题引入
就拿弹窗这一点来说现在大家的弹窗都是各种样式各种布局满天飞不过进公司后新人在第一次写弹窗之前都会问一个问题弹窗这么通用的东西没有一个规范吗说完之后又默默写自己的有个性的弹窗去了
建议大家在写弹窗的时候无论如何一定要**严格采用**水平居中垂直居中的写法
千万不要用 `margin-top` 这种距离屏幕顶部的距离来计算弹窗的位置太搓了不要让领导觉得你们写了这么久的前端代码连个弹窗都搞不定
### 移动端红包幕帘/弹窗 居中的规范写法非常标准
移动端场景这里提供一个 红包幕帘/弹窗 的居中写法注意是严格居中非常标准为什么是移动端你有见过PC网页端给你送红包的么
在实战开发中下面的这段代码可以直接拿去用注释详细贴心无比
```html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
<style>
/* 整个弹窗组件 */
.component_popup {
position: fixed;
top: 0;
bottom: 0;
left: 0;
right: 0;
z-index: 100;
}
/* 遮罩背景 */
.popup_mask {
position: fixed;
top: 0;
bottom: 0;
left: 0;
right: 0;
background: rgba(0, 0, 0, 0.7);
}
/* 弹窗区域(内容 + close严格居中 */
.popup_content {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
}
/* 弹窗的内容部分 */
.content_box {
width: 15.45rem;
height: 19.32rem;
background: url(http://img.smyhvae.com/20191010_1500_red-packet.png) no-repeat;
background-size: 15.45rem 19.32rem;
}
/* 弹窗的close图标 */
.content_close {
width: 1.25em;
height: 1.25em;
background: url(http://img.smyhvae.com/20191010_1500_close.png) no-repeat;
background-size: 1.25rem 1.25rem;
margin: 0 auto;
margin-top: 0.5rem;
}
</style>
</head>
<body>
<div class="content">默认文档流中的页面主体</div>
<div class="component_popup">
<div class="popup_mask"></div>
<div class="popup_content">
<div class="content_box"></div>
<div class="content_close"></div>
</div>
</div>
</body>
</html>
```
实现效果
![](http://img.smyhvae.com/20191010_1510.png)
**补充**
1如果你的页面中有很多弹窗建议将弹窗封装成一个抽象组件
2任何弹窗都需要解决滚动穿透的问题本文篇幅有限请自行查阅
## 最后一段
有些实现方式虽然简单但必须要经得起千锤百炼我们要做到**敬畏每一行代码**不能浮于表面团队开发要的不是个性而是**标准和规范**
## 参考链接
- [为什么margin:auto可以让块级元素水平居中](https://www.zhihu.com/question/21644198/answer/22392394)
- [七种方式实现垂直居中](https://jscode.me/t/topic/1936)
- [margin:auto实现绝对定位元素的水平垂直居中](http://www.zhangxinxu.com/wordpress/2013/11/margin-auto-absolute-%E7%BB%9D%E5%AF%B9%E5%AE%9A%E4%BD%8D-%E6%B0%B4%E5%B9%B3%E5%9E%82%E7%9B%B4%E5%B1%85%E4%B8%AD/)
## 我的公众号
想学习**代码之外的技能**不妨关注我的微信公众号**千古壹号**id`qianguyihao`
扫一扫你将发现另一个全新的世界而这将是一场美丽的意外
![](http://img.smyhvae.com/20190101.png)