茗空间

深入理解table-cell

唠叨

还有一周2016就要过去了,时间过得太快,压抑的一年,好像什么都没做,闭目冥思,反思自我得失…我勒个去,文采不好,再唠叨真成大妈了。
这一年听到最多的声音就是房价涨了,房价涨了,房价上蹿上蹿,一路飙升,疯了,全国都疯了,我也疯了,房租涨了,工资没涨,不好意思,我又禁不住
啰嗦了两句,还是聊技术吧,只有技术最可靠。

最近看了几篇关于table-cell的用法,觉得旭神总结的还不错,于是乎我就抄袭,不对,是模仿,也不合适,
其实是微微扩展了以下,增加几点需要注意的地方。

支持度

从下图可以看出,所有浏览器都支持的很好嘛。

caniuse

特性(Peculiarities)

table-cell的特征就是跟table的td表现是一毛一样的;

  1. 对marigin无感:设置margin没有反应,但是可以设置宽高和padding;
  2. 并列cell等高:同容器下的并列cell的高度保持一致;
  3. 宽不会超过父容器:宽高设置成99999999px也不会超过父元素的宽度;
  4. 会生成BFC:以下的几点都会生成BFC
    1. position为absolute或fixed;
    2. display为table-cell, table-caption或者inline-block;
    3. float不为none;
    4. overflow不为visible;
  5. 天然垂直居中:注意要设置cell的vertical-align为middle,这点很奇怪,因为cell是父元素,对比下面两个实例,感受一下:
1
2
3
4
5
6
7
8
9
// demo1
<table-cell style="vertical-align: middle;">
<span></span>
</table-cell>
// demo2
<div>
<inline style="vertical-align: middle;"></inline>
</div>

用途

垂直居中

1
2
3
<div style="display: table-cell; vertical-align: middle; width: 200px; height: 150px; border: 1px solid red;">
vertical align
</div>

顺便提一下inline-height的垂直居中,当多行文本直居中时,其子元素应为inline-block,比如:

1
2
3
4
5
<div style="line-height:150px; width:200px; border:1px solid red;">
<span style="display:inline-block; line-height:1; vertical-align:middle">
vertical align vertical align vertical align vertical align vertical align vertical align vertical align
</span>
</div>

两列布局

1
2
3
4
5
6
7
8
<div style="clear">
<div style="float: left; width: 100px; height: 100px; background: red;">
cell one
</div>
<div style="display: table-cell; width: 100000px; background: blue;">
cell two
</div>
</div>

image

如果左边的cell子元素只有img,有两点需要提醒一下:

第一点、 如果未设置cell的max-width:

1
2
3
4
5
6
7
8
<div style="margin-top: 30px;">
<div style="display: table-cell;height:150px;width:150px;background:red;">
<img src="/assets/images/samples/cat1.jpg" width="200px">
</div>
<div style="display: table-cell;height:150px;width:150px;background:blue;">
blue box
</div>
</div>

image

咦?怎么cell变高了,基本上快2倍了,对于inline-block容器也存在类似情况:

1
2
3
4
5
6
<div>x
<div style="display:inline-block; height:150px; width:150px; background:red;"></div>
<div style="display:inline-block; height:150px; width:150px; background:blue;">
blue box
</div>
</div>

image

这两种情况涉及到inline-block元素的vertical-align的默认值,其默认baseline,
那么baseline是什么?有三种情况:

  • 如果inline-block元素有流内内容,那么它的baseline就是最后一行内容的baseline;
  • 如果inline-block元素有流内内容,但是它的overflow是非visible的,那么它的baseline就是magin-box的底部;
  • 如果inline-block元素没有流内内容,那它的baseline是margin-box的底边界;

看到这里应该就明白了为什么有这种垂直落差了吧,如果不明白,留言吧。

为了解决这种问题,可以设置table-cell(或inline-box的vertical-align为top)。

第二点、如果cell的max-width为100%,对比看看:

1
2
3
4
5
6
7
8
9
// img的max-width不为100%,并且第二个table-cell设置宽度很大比如100%或者9999999px
<div style="margin-top: 30px;">
<div style="display: table-cell;height:150px;width:150px;background:red;">
<img src="/assets/images/samples/cat1.jpg" width="200px">
</div>
<div style="display:table-cell; height:150px; width:150px; background:blue;">
blue box
</div>
</div>

image

1
2
3
4
5
6
7
8
9
// img的max-width为100%,并且第二个table-cell设置宽度很大比如100%或者9999999px
<div>
<div style="display: table-cell;background:red;">
<img style="max-width:100%" src="/assets/images/samples/cat1.jpg" width="100px">
</div>
<div style="display: table-cell;height:150px;width:100%;background:blue;">
blue box
</div>
</div>

image

聪明的你一眼就看到第二种情况第一个table-cell没了宽度,其主要原因是max-width的优先级高于width,所以此时的width无效了,所以img就被右侧的cell
挤压没了,顺便说一句,min-width的优先级比max-width高。

两列等高布局

其实就是利用了特性2:

1
2
3
4
5
6
7
8
9
10
<div>
<div style="display:table-cell;background:green;">
hello
</div>
<div style="display:table-cell;background:red;width: 100%">
<div>
abc def jkkl kada eix lllll ew a ee www qq ccccc zzzdfa fdadfjaklew ieiie
</div>
</div>
</div>

参考

1.我所知道的几种display:table-cell的应用

2.max-width:来自MDN的传说

3.Block formatting contexts

4.the ‘line-height’ and ‘vertical-align’ properties

5.CSS深入理解vertical-align和line-height的基友关系

6.Vertical-Align: All You Need To Know