能够让不定宽高元素水平和垂直居中的方法

在开发中经常遇到这个问题,即让某个元素的内容在水平和垂直方向上都居中,内容不仅限于文字,可能是图片或其他元素。而且我们希望不要涉及宽度和高度,也就是说,我们不知道父元素的宽高,也不知道内容元素的宽高。这篇文章就来总结一下都有哪些方法可以实现水平和垂直都居中。

不适合的方案

text-align和line-height

显然,使用text-align和line-height的方式更适合单排文字,而不适合本文的需求。特别是line-height,无法保证在不知道高度的情况下还能垂直居中。而且就算是文字,我们也不知道文字有多少行。

position:absolute、50%和margin:-px

绝大多数情况下,我们可以考虑这种方案,让想要居中的元素通过定位和margin为负值进行偏移的方法让它在垂直方向上居中。这种方案不要求父元素的高度,也就是即使父元素的高度变化了,仍然可以保持在父元素的垂直居中位置,水平方向上是一样的操作。但是这里有一个问题,就是我们的需求往往是内部的这个元素的宽度高度也不确定,比如是一段文字,你无法保证这段文字的字数多少,所以通过margin为负值来偏移在这种情况下行不通。

position:fixed、0和margin:auto

当我们要制作一个modal dialog弹出框时,比如弹出居中于屏幕的广告或登录框。这个时候可以考虑一些相对于窗口或网页居中的方案。

<style> 
.container {
   position:fixed;
   top:0;
   right:0;
   bottom:0;
   left:0;
   margin:auto; 
}
</style>
<div class="container" style="width: 300px; height: 200px; background: #f1f1f1"> this is a box fixed in center of screen</div>

这里面最重要的是margin: auto;,对于块级元素而言,确定了自己的宽度之后,margin:auto可以帮助它居中,即使在position:fixed时。不过必须规定要居中的元素的宽高度,无法满足我们的需求。

position:absolute、0和margin:auto

上面的fixed方案只适合在整个窗口实现居中。fixed会使元素脱离网页,因此在内容流中还是不适用。在内容流中也想实现居中,可以如下:

<style>
  .container {
    position: relative;
  }
  .inner-wrapper {
    position: absolute;
    top: 0;
    left: 0;
    width: 100%;
    height: 100%;
  }
  .inner {
    position:absolute;
    top:0;
    right:0;
    bottom:0;
    left:0;
    margin:auto;
}
</style>
<div class="container">
    <p>This is a p</p>
    <br><br><br><br><br><br><br><br><br><br><br><br><br><br><br><br><br><br>
    <div class="inner-wrapper">
        <div class="inner" style="width: 300px;height: 200px;background: #f1f1f1">
        this is a box fixed in center of screen
        </div>
    </div>
</div>

首先是仿造上面一个方法,使用margin:auto,只不过使用absolute。使用absolute定位的话,父级元素必须也具有position(不为static)。所以把.inner放在一个有position的父级元素.inner-wrapper。这样.inner相对于.inner-wrapper就是居中的(前提还是.inner要有宽高)。同理因为要设定宽高,所以不满足我们的需求。

正确的方案

1、display:table和diaplay:table-cell

这个方案是理解上最容易的,因为table具备垂直居中的属性,所以很容易通过属性就能实现。

<style>
.container {
  display: table;
}
.inner {
  display: table-cell;
  vertical-align:middle;
  text-align:center;
}
</style>
<div class="container">
  <div class="inner">
  you own content
  </div>
</div>

这种情况下,我们可以通过随意改变.inner的宽高,当内部的内容仍然保持居中状态。

2、position:absolute、50%和translate

在css3里面提供了translate函数,它的主要作用是位移,传给transform属性。

<style>
.container {
  position: relative;
}
.inner {
  position: absolute;
  top: 50%;
  left: 50%;
  transform: translate(-50%, -50%);
}
</style>
<div class="container">
  <div class="inner">
    your own content
  </div>
</div>

html代码和上面一样。translate(-50%, -50%)将会将元素位移自己宽度和高度的-50%。这种方法其实和最上面被否定掉的margin负值用法一样,可以说是margin负值的替代方案。这样你就非常容易理解了。这个方法最厉害的地方是不需要确定.inner的宽高,而.container的宽高也不需要手动设置,如果它自己本身就被撑大的话。这里只是为了演示方便,才特意给它设置了宽高。

3、vw vh和translate

vh和vw是两个比较偏的单位,是指“viewport的height和width的1%”,比如说50vh就是当前视口(窗口的高度,实验中包含了滚动条)高度的50%。也就是说vw将获得和1%差不多的window宽度。因此用在fixed的时候更加适合

<style>
.inner {
   position:fixed;
   top: 50vh;
   left: 50vw;
   transform: translate(-50%, -50%); 
}
</style>
<div class="inner">
  this is a box fixed in center of screen
</div>

其实和使用50%没有太大的差别,因为这时top、left取的50%是相对于父元素的,和margin、padding不一样。如果非得要margin的话,就可以从这里衍生出变体:

 .inner2 {
   position:fixed;
   top: 0;
   left: 0;
   margin: 50vh 0 0 50vw;
   transform: translate(-50%, -50%); 
}

vh vw只能从窗口的大小去考虑,不适合正常的文本流。不过有的时候可以非常有用,特别是在做全屏应用的时候,比如full page。

4、:before和display:inline-block

这也是一种处理方式,通过伪类:before在元素内增加新元素后在用display:inline-block,通过高度的处理得到想要的效果。

<style>
.container{
    text-align: center;
}
.container:before {
    content: '';
    display: inline-block;
    height: 100%;
    vertical-align: middle;
}
.inner {
    display: inline-block;
}
</style>
<div class="container">
    <div class="inner">
        this is a box fixed in center of screen<br>The second line
    </div>
</div>

这个方案是比较特别一些,不是很好理解。首先,.container水平居中没问题。接着,给.container伪类:before设定为height:100%,这样可以用一个伪元素在.container获得与父元素等高的空间。然后用inline-block和vertical-align: middle改变对齐的基线,关于这一点,我也不甚懂,这里有一篇文章可以参考。通过:before之后,.container内的行级元素的对齐基线就跑到居中的位置,也就实现了垂直居中对齐。这个时候,如果里面仅一排文字,其实可以不用.inner,但是上面的例子里面有一个<br>,这就不一样了。如果直接把文字放在.container里面,
之前的文字会基于:before基线,会保持垂直对齐的状态。但是
之后的文字会另起一行,这一行将起始于:before的下一行,所以会在:before的100%高度下面,导致被顶出.container。但是如果把文字放在.inner里面,再让.inner为inline-block,就可以使.inner和:before处于同一基线,这样就让整个.inner处于垂直居中的状态。

5、css3 flex

css3新增了布局相关的属性,其中flex布局可以非常简单地帮我们实现我们想要的效果。

<style>
.container {
    display: flex;
    align-items: center;
    justify-content: center;
}
</style>
<div class="container">
    <div class="inner">
        this is a box fixed in center of screen<br>The second line
    </div>
</div>

简单解释一下,当display: flex时,表示该容器内部的元素将按照flex进行布局。align-items: center表示这些元素将相对于本容器水平居中,justify-content: center也是同样的道理垂直居中。对.container赋予了这些样式之后,作为它的内部元素.inner自己自觉的居中了。而且这里你会发现,由于没有使用text-align: center,.inner里面的文字是不会居中的,也就是说仅仅.inner这个容器居中而已。

总结

从上面的几种可行的方案,大致可以分为两类:display对齐方案、translate位移方案。display方案是充分发挥css的布局特性,利用布局和UI引擎的特性来控制布局中的对齐方式。而translate方案则是利用位移,通过先50%的位移,可以是通过position,也可以是通过margin vw vh,但是完成之后,在通过translate把元素拉回去,之所以用translate而不是margin是因为translate是相对于元素本身,而margin不是。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 218,525评论 6 507
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 93,203评论 3 395
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 164,862评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,728评论 1 294
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,743评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,590评论 1 305
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,330评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,244评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,693评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,885评论 3 336
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 40,001评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,723评论 5 346
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,343评论 3 330
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,919评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 33,042评论 1 270
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 48,191评论 3 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,955评论 2 355

推荐阅读更多精彩内容