在学习前端的过程中,发现元素和文本的水平居中和垂直居中,是经常会出现的问题,在实际工作中也会经常碰到。居中的技巧有很多,但在编写代码的过程中,发现有时候技巧管用,有时候不管用,于是就将每个知道的方案都试一遍,找到合适的。这种情况究其原因是对居中的认识不够深入,只是停留在实现需求的水平上。为了更好的加深对居中的理解,搜集和阅读相关资料,发现不错的文章将其整理出来。原文参考:https://css-tricks.com/centering-css-complete-guide/
居中是什么
居中故名思意就是将物体放置在其容器的中间,在css中居中就是指元素、文本、图片等相对其容器或父元素或浏览器窗口能够居中显示。而根据显示方式的不同,又分为水平居中,垂直居中,水平垂直居中。下面就分情况对居中进行讨论。
水平居中
水平居中我们常见的一种解决方案就是设置text-align:center,然而我们会发现这种方式有时候有效,有时候却没有效果。这是因为我们在使用相关方法的时候并没有对块级元素和行内元素进行区分,不同类型的元素其居中方法是不同的:
当元素为行内元素时(如文本,链接)
当需要居中的元素为行内元素时,你可以通过在其父元素(必须是块级元素)设置如下css样式:
.parent {
text-align:center;
}
这种方法对display设置为inline、inline-block、inline-table、inline-flex的元素都有效。
块级元素居中
对于一个块级元素你可以通过设置其margin属性,来达到水平居中的效果。你可以将其margin-left和margin-right设置为auto:
.center{
margin:0 auto;
}
这样浏览器就会根据元素的宽度自动为元素左右两边设定等宽的margin,来达到水平居中的效果。这里需要注意的是元素需要设定width属性,否则元素的宽度会自动充满父元素,就不存在相对父元素水平居中一说了。
示例代码点击预览
这里需要注意的是这种方法对设置float属性的块级元素是没有效果的,浮动元素的居中这里有一个解决方案,但是该方法的布局比较混乱,子元素脱出父元素,并不推荐。
另外有一个tricks,能够实现如下图中
文字环绕图片居中的效果,具体参见这篇文章。
当有多个块级元素时
当你需要对多个块级元素进行居中显示时,如果块级元素如下垂直排列:
,那么简单的设置margin:0 auto;就可以轻松实现。
但如果如下排列:
那么就可以使用display:inline-block将多个块元素单行显示,同时由于inline的缘故,可以在父元素设置text-align:center将多个块元素居中,当然也可以将元素设置为display:flex; justify-content: center;达到居中效果。详细代码示例点击预览
垂直居中
垂直居中相比于水平居中,就复杂一些,还是分块级与行内进行讨论。
inline或inline-block元素
单行垂直居中的情况
当父元素没有设定宽度,而是根据内容自适应时,简单的设置padding就可以达到垂直居中的效果,如:
css:
.p{ border: 1px solid yellow;padding: 80px;}
.p a{
background-color: black;
color: white;
border: 2px solid red;
padding: 30px; /*相同的上下内距实现垂直居中*/
}
html:<div class="p">
<a href="#">asldkjflkadfk</a></div>
在某些情况下设置padding并不能满足需求,而你又需要将一段单行显示的文本居中,这时你可以将line-height和height属性设置为等高来达到目的:
html:<main><div>I'm a centered line.</div></main>css:
main {
background: white;
margin: 20px 0;
padding: 40px;
}
main div {
background: black;
color: white;
height: 100px; /*行高与line-height相同*/
line-height: 100px;
padding: 20px;
width: 50%;
white-space: nowrap;
}
关于line-height,有许多需要了解和注意的地方,想详细了解为什么这么这种方法可以实现垂直居中,可以参见这篇文章
多行垂直居中
设置padding同样可以使多行文本居中,但有时当文本出现在表格单元格里或其他情况也会使该方法无效。这时会用到vertical-align属性,
更多关于vertical—align的信息,可以点击这里和这篇
需要指出的是只有一个元素属于inline或是inline-block(table-cell也可以理解为inline-block水平)水平,其身上的vertical-align属性才会起作用
如果table—like不行,也许你可以试一下flex-parent, 一个flex-child可以很轻易的在flex-parent里垂直居中:
.flex-center-vertically {
display: flex;
justify-content: center;
flex-direction: column;
height: 400px;
}
需要注意的是父容器必须有一个固定的高度。
如果上述2中方法都不可行,你就需要借助伪元素,也就是在父容器里添加一个高度占满整个父元素的伪元素,文本就居中显示在里面。
.ghost-center {
position: relative;
}
.ghost-center::before {
content: " ";
display: inline-block;
height: 100%;
width: 1%;
vertical-align: middle;
}
.ghost-center p {
display: inline-block;
vertical-align: middle;
}
块级元素垂直居中
当你知道元素的高度时
如果你知道元素的高度,那么你可以这样实现垂直居中:
.parent {
position: relative;
}
.child {
position: absolute;
top: 50%;
height: 100px;
margin-top: -50px; /* account for padding and border if not using box-sizing: border-box; */ /*这里可以使用transform:translate(-50%)实现同样的效果*/ }
上述代码是通过定位的方式,来实现垂直居中,子元素参照父元素进行绝对定位,相对于父元素的上边缘向下移动50%(父元素高度的50%),然后通过margin-top元素将子元素向上拉自身的50%,达到最终的居中。
当元素的高度未知时
当元素高度未知时可以如下,本质原理与上述相似:
.parent {
position: relative;
}
.child {
position: absolute;
top: 50%;
transform: translateY(-50%);
}
使用flexbox
使用flexbox无疑是种简单有效的解决方案:
.parent {
display: flex;
flex-direction: column;
justify-content: center;
}
既水平居中又垂直居中
你可以将上述方法进行合理的组合,来达到水平垂直居中,总结一下,可分为如下三种情况:
元素是否具有固定的宽高
在通过将元素定位50%/50%后,再设置一个高度和宽度一半大小的负值给 margin,能够很好的居中,并且支持大多数浏览器:
.parent {
position: relative;
}
.child {
width: 300px;
height: 100px;
padding: 20px;
position: absolute;
top: 50%;
left: 50%;
margin: -70px 0 0 -170px;
}
当宽高未知时
当你不知道宽高时,你可以使用transform属性,设置translate(-50%),这里的50%是相对与元素本身的宽高。
.parent {
position: relative;
}
.child {
position: absolute;
top: 50%;
left: 50%;
transform: translate(-50%, -50%);
}
使用flexbox
想要水平垂直居中,你需要设置两个居中属性
链接:https://segmentfault.com/a/1190000006913661(点击尾部阅读原文前往,本文涉及到较多代码,请前往原文查看完整代码)