如何组织CSS-分层
应用 css 的能力分两部分:一部分是css的API,重点是如何用css控制页面内元素的样式;另一部分是css框架,重点是如何对 css 进行组织。如何组织 css 可以有多种角度,例如按功能划分,或者按区块划分。这里讲一下 base.css + common.css + page.css 的组织方法。将网站内的所有样式,按照职能分成三大类:base、common、page,这三者是层叠结构。
1、base 层-精简通用
位于三者的最底层,提供 css reset 功能和粒度最小的通用类——原子类。这一层会被所有页面引用,是页面样式所需依赖的最底层,不同风格的网站可以使用同一个 base 层,所以,应具有高度可移植性,力求精简和通用。该层相对稳定,基本不需要维护,可以简单地放在一个文件里,如 base.css。
css reset 即一开始就将浏览器的默认样式全部去掉,就是通过重新定义标签的样式,“覆盖”掉浏览器提供的默认样式。可以将常用的标签显式地罗列出来,避免使用“*”,如来自于 YUI 的部分 css reset 的代码:
/*CSS reset*/body, div, dl, dt, dd, ul, ol, li, h1, h2, h3, h4, h5, h6, pre, form, fieldset, input, textarea, p, blockquote, th, tdmargin:0;padding:0;tableborder-collapse:collapse;border-spacing:0;fieldset, imgborder:0;address, caption, cite, code, dfn, em, strong, th, varfont-style:normal;font-weight:normal;ol, ullist-style:none;caption, thtext-align:left;h1, h2, h3, h4, h5, h6font-size:100%;font-weight:normal;q:before, q:aftercontent:”;abbr, acronymborder:0;
通用原子类是一系列常用的基本类,包括:文字、定位、长度和边距,由于其原子性,除少数特殊类,大部分都只包含一句 css,如:.f12font-size:12px;,由于其通用性,在保证命名有语义前提下,命名应尽量简短,如 paddingtop20 可以写成 pt20。通用原子类里有几个类较特殊,特别说明一下:
1) .fl类和 .fr类:除了设置 float: left和 float: right之外,还应设置 display: inline,可以解决 IE6的双外边距 Bug。 .flfloat:left;display:inline;.frfloat:right;display:inline;2) .bc类:为使块级元素居中,还要设定宽度,可以把它和 .w100等类同时使用,如: .bcmargin-left:auto;margin-right:auto;.w100width:100;3) .clearfix类:用于在父容器直接清除子元素浮动。 4) .zoom类:设置样式是 zoom:1,它是 IE的专有属性,用以触发 hasLayout。
2、common 层-组件级,模块化,重用
位于中间,提供组件级的 css 类。可以将页面内的元素拆分成一小块功能和样式相对独立的小“模块”,将大量重复地“模块”视为一个组件,放在 common 层里(“模块化”可以从样式和行为两个层面来考虑,与 common 层相关的是样式的模块化)。common 层相当于 MVC 模式中的 M(Model,模型),需尽可能将内部实现封装。
(web前端学习交流群:328058344 禁止闲聊,非喜勿进!)
common 层是网站级的,不同的网站有不同的 common 层,同一个网站只有一个 common 层,可以放在一个文件里,也可按功能划分放在多个文件里。在团队合作中,common 层最好由一个人负责,统一管理。
3、page 层
网站中非高度重用的模块,可以放在 page 层里。page 层位于最高层,提供页面级的样式。如果网站规模不过于庞大,可以将所有 page 层放在一个文件里,根据页面配上注释,分块书写,便于维护。page 层 css 文件应越简越好,能用 base 层和 common 层的 css 解决的,尽量不要用到 page 层。
避免滥用子标签
低权重原则。
能用组合,不用继承。
驼峰命名用于区别不同单词,划线用于表明从属关系。
CSS sprite
技术是针对作为背景的图片,对与html文档设置的图片,不能合并到CSS Sprite大图,否则图片会影响页面可读性。
对于横向和纵向都平铺的图片,也不能使用CSS sprite;如果是横向平铺的,只能将所有横向平铺的图合并成一张大图,只能竖直排列,不能水平排列;如果是纵向平铺的,只能将所有纵向平铺的图合并成一张大图,只能水平排列,不能竖直排列。
CSS sprite的图片定位可以使用bg2css小工具快速定位background-positon坐标,有利于提高开发速度。
网站是否使用CSS Sprite技术,主要取决于网站流量。
流量不大的网站代价很大:降低开发效率,增大维护难度(后台管理系统一般不适合使用)
CSS hack
1.IE条件注释法
该方法安全性、兼容行好,也是微软推荐的hack方法,但是不利于开发维护,需要维护多份css文件。比如涉及到针对不同版本IE的css。
lt;!--[if IE]gt;lt;![endif]--gt;只在IE下有效 lt;!--[if IE 6]gt;lt;![endif]--gt;只在IE6有效 lt;!--[if gt IE 6]gt;lt;![endif]--gt;只在IE6以上版本有效 注意:结合lte(小于等于)、lt(小于)、gte(大于等于)、gt(大于)、!(非)关键字使用。
2.选择符前缀法
“* html” 前缀只对 IE6生效 "*+ html"前缀只对 IE7生效 .test width:80px; /*IE 6 7 8*/* html .test width:70px; /*IE6*/*+ html .test width:60px; /*IE7*/缺点:不能保证 IE9,10不识别* html,*+ html,有向后兼容风险。
3.样式属性前缀法:
如“_”只在IE6下生效,“*”在IE6和IE7下生效。同样有向后兼容隐患。 .testwidth:80px;*width:70px;_width:60px; 可用于内联样式: lt;divgt;lt;/divgt;由于IE条件注释法不利于开发维护,实际中常用的hack方法常常是后两者。
display:inline-block 和 hasLayout
块级元素、行内元素
块级元素会独占一行,默认宽度自动填满其父元素宽度,可以设置width、height、margin、padding属性;
行内元素一行排满才会换行,宽度随元素内容多少变化,设置width、height属性无效,只有水平方向的margin、padding边距有效果。
常见的块级元素有div、p、table、fieldset、form、ul、ol、dl、h1~h6、hr、pre、address、blockquote、center、dir、menu、noframes、no。
常见内联元素有input span strong em a abbr acronym br img select textarea等等。
可以通过修改display属性值转行块元素和行内元素。
display的值除了block和inline,还有其他值,例如list-item、teble-cell等,但因为IE6和IE7浏览器支持的display类型很少,所以为兼容IE,真正能用的display类型只有block、inline和none三种。
IE6、IE7支持但不完全支持display:inline-block属性,但IE8+和FF等标准浏览器支持。
display:inline-block,行内的块级元素,它拥有块级元素的特点,可以设置长宽,可以设置margin和padding值,但它却不是独占一行,它的宽度并不占满父元素,而是和行内元素一样,可以和其他行内元素排在同一里。它集块级元素和行内元素的特点于一身,是个非常有用的display类型(web前端学习交流群:328058344 禁止闲聊,非喜勿进!)
让IE6、IE7支持display:inline-block
利用hasLayout可以再不支持display:inline-blcok的IE6和IE7下模拟出display:inline-block的效果,实现IE6、IE7、IE8+和Firefox都兼容的display:inline-block的应用。但是也有一些问题需要特别注意:
下表列出一些CSS属性及其值,以下属性一旦设置,将触发元素的layout:
有两种方法:
1、先使用 display:inline-block 属性触发块元素,然后再定义 display:inline,让块元素呈递为内联对象(两个display 要先后放在两个 CSS 声明中才有效果,这是 IE 的一个经典 bug ,如果先定义了 display:inline-block,然后再将 display 设回 inline 或 block,layout 不会消失)。代码如下:
divdisplay:inline-block;divdisplay:inline;
说明:在IE下,display: inline-block只是触发了元素的layout。比如将display: inline-block给到div上,只能保证这个div拥有块元素的特征(可以设置宽度,高度等),但是还是行布局(产生换行)。接下来要设置display: inline,更改这个div的布局为内联布局(不产生换行)。
2、直接让块元素设置为内联对象呈递(设置属性 display:inline),然后触发块元素的 layout(可使用zoom:1 等)。代码如下:
divdisplay:inline; zoom:1;
hasLayout
hasLayout设计的初衷是用于辅助块级元素的盒子模型的,它是用于块级元素的。如果用于行内元素,会引发一些特殊效果。(结合上面红色字体理解)
设置宽高属性width、height值都可以触发hasLayout,但是有时候带来副作用,现在常用zoom:1;来触发,极少数非常复杂的css设置情况zoom无效的时候,需要借助更为强大的“position:relative”来触发hasLayout。
haslayout
是Windows Internet Explorer渲染引擎的一个内部组成部分。在InternetExplorer中,一个元素要么自己对自身的内容进行计算大小和组织,要么依赖于父元素来计算尺寸和组织内容。为了调节这两个不同的概念,渲染引擎采用了 hasLayout 的属性,属性值可以为true或false。当一个元素的 hasLayout属性值为true时,我们说这个元素有一个布局(layout)。