codecamp

margin系列之bug巡演(三)

margin系列之bug巡演(三)

原作者:doyoe
原文链接: http://blog.doyoe.com/2013/12/20/css/margin%E7%B3%BB%E5%88%97%E4%B9%8Bbug%E5%B7%A1%E6%BC%94%EF%BC%88%E4%B8%89%EF%BC%89/

IE8按钮margin auto居中失效Bug

你会猛然觉得,这是正解啊,因为 button 或者 input type button类型 的元素是 inline-level 的。

不对啊,button 应该是 inline 的吧?哦,可能是 inline-block

在这之前,我们似乎要先明确一些基础知识。

什么是 inline-level 元素?

要知道 inline-level 元素并不等于 inline 元素,也就是说 行内级元素行内元素 是两个不同的概念。

inline-level 元素包含 display 值为:

  • inline
  • inline-block
  • inline-table
  • inline-flex
  • other inline-*

以上情况时,元素可被称之为 inline-level 元素,但不都是 inline 元素。

什么是 block-level 元素?

block-level 指的是 display 值为 block 的元素吗?我知道不少人一直有这样的认知,不过这不完全准确。

block-level 元素包含 display 值为:

  • block
  • list-item
  • table
  • table-*
  • flex
  • 如果position既不是static也不是relative、float不是none或者元素是根元素,当display:inline-table时,display的计算值为table;当display值为 inline | inline-block | run-in | table-* 时,display的计算值为block

有如上情况时的元素均被称之为 block-level 元素。同时 block-levelblock 也不是同一个概念,所以如果你认为 display 值为 list-itemli 不是 块级元素,那就错了。

看到这里,你对 块级元素块元素行内级元素行内元素
这个4个概念,应该已经有了比较清晰的了解?

margin keyword auto只能应用在常规流中的 block-level 元素上

  • 当一个块级元素定义了 position 值为非 staticrelative 之外的值时,margin-right/left auto 的计算值为0;
  • 当一个块级元素定义了 float 值为非 none 之外的值时,margin-right/left auto 的计算值为0;
  • 非块级元素的margin-right/left auto 的计算值为0;

计算值为0,即说明其应用使用值的意图失败。所以在有如上情形的场景中,都无法使用 auto 来实现水平居中。同时也说明了,只有normal flowblock-level 才能应用 margin keyword auto。

margin可以应用于所有元素吗?

这显然不行。准确的说:margin可以应用在除某些table-*元素和某些行内元素之外的所有元素上。

和margin亲近的table-*系元素

  • table
  • inline-table
  • table-caption

除了 display 值为以上3种之外的 table系 元素,都不能应用 margin ,比如:th, td。

和margin亲近的 inline-level 元素

我之前面试的时候常会问候选人,行内元素不能设置宽高对吗?大部分人会告诉我说是;然后我又会问,那为什么 img 元素可以设置宽高呢?有人会告诉我,因为 img 是个特殊的元素?接着我又会问题,img 是如何特殊的?然后,然后就没然后了,因为没声音了。

恩,img 确实是个特殊的元素。它特殊在哪里?它的特殊就在于它是一个行内置换元素。

所有的置换元素都可以设置 margin 属性,并且可以设置宽高,这就是为什么 img 是行内元素却可以设置 widthheight

什么是置换元素(Replaced elements)?

一个元素拥有内在的二维属性,其宽高属性受外部资源影响,默认拥有CSS格式,这样的元素被称为置换元素。

意思就是说置换元素的宽高不完全由CSS决定,还受其自身内容和外部资源所影响。

举个例子来说,仍然说 img 元素吧,你会发现,如果你 src 进来不同尺寸的资源,那么在 viewport 上显示的图片宽高也是不同的,也就是说 img 元素的宽高会受外部资源影响。

再说说 input 元素,随便在页面上扔一个input,你都能发现它拥有一个默认的宽高,这就是它所具有的内在二维宽高属性,并且该类元素会受UA影响,不同UA下所呈现外观会有不同。

常见的置换元素有哪些?

img, object, button, input, textarea, select等

行内非置换元素真的不能应用margin吗?

什么是非置换元素?除了置换元素之外的元素,我想将这样的元素称之为非置换元素是没有大碍的。

那么行内非置换元素真的无法设置 margin 吗?我想在工作中你一定碰到过很多这样的场景,给一个 a 或者 span 定义间隙。这时我们写:

CSS

span{margin:5px 10px;}

结果发现 span 的水平方向上的 margin 定义生效了,但垂直方向上的 margin 定义却没被应用。

是的,这就是行内非置换元素使用 margin 时的表征,所以对各种特性的理解,在让自己的代码更有效上是大有裨益的。

回归正题

我们本来是想说IE8按钮margin auto居中失效Bug的,扯了不少题外话。
我们知道 margin keyword auto 不能应用在处于常规流中的 block-level 之外的元素上,所以我有这样的一段代码:

CSS

button{display:block;margin:auto;}

HTML

<div id="demo">
    <button>按钮</button>
</div>

恩,我们将 button 显式的转换为了 block,同时我们知道 button 作为置换元素,本身具备内在宽高,也就是说这时,我只需要加上 margin:auto ,该按钮就应该在其包含块里水平居中。
是的,所有浏览器都和预期是一样,实现了水平居中,但是却出现了奇葩的IE8,完全无效,甚至不如原始社会的IE6。来看看示例 DEMO1IE8按钮margin auto居中失效Bug

通过以上例子,你有没有突然感觉到,如果要让一个置换元素在包含块中水平居中,出乎预料的简单,只需要 display:block;margin:auto; 即可。

注意事项

令人意外的是,只有 buttoninput typebutton 相关元素的时候,在IE8中才会水平居中失效;如: input type textimg 时,margin keyword auto 运作正常。

解决方案

  • 给其显示的定义宽度
  • 不改变其display值,包含块text-align:center
  • 其它水平居中方案,如:absolute + 负margin
margin系列之bug巡演(二)
margin系列之布局篇
温馨提示
下载编程狮App,免费阅读超1000+编程语言教程
取消
确定
目录

关闭

MIP.setData({ 'pageTheme' : getCookie('pageTheme') || {'day':true, 'night':false}, 'pageFontSize' : getCookie('pageFontSize') || 20 }); MIP.watch('pageTheme', function(newValue){ setCookie('pageTheme', JSON.stringify(newValue)) }); MIP.watch('pageFontSize', function(newValue){ setCookie('pageFontSize', newValue) }); function setCookie(name, value){ var days = 1; var exp = new Date(); exp.setTime(exp.getTime() + days*24*60*60*1000); document.cookie = name + '=' + value + ';expires=' + exp.toUTCString(); } function getCookie(name){ var reg = new RegExp('(^| )' + name + '=([^;]*)(;|$)'); return document.cookie.match(reg) ? JSON.parse(document.cookie.match(reg)[2]) : null; }