增加的空格应当可以使得在使用多个 class 时更易阅读与定位。
JavaScript 钩子
千万不要把 CSS 样式用作 JavaScript 钩子。把 JS 行为与样式混在一起将无法对其分别处理。
如果你要把 JS 和某些标记绑定起来的话,写一个 JS 专用的 class。简单地说就是划定一个前缀 .js-的命名空间,例如 .js-toggle, .js-drag-and-drop。这意味着我们可以通过 class 同时绑定 JS 和 CSS 而不会因为冲突而引发麻烦。
|
上面的这个标记有两个 class,你可以用其中一个来给这个可排序的表格栏添加样式,用另一个添加排序功能。
I18n
虽然我(该 CSS Guideline 文档原作者 Harry Roberts)是个英国人,而且我一向拼写 colour 而非 color,但是为了追求统一,我认为在 CSS 中用美式拼法更佳。CSS 以及其它多数语言都是以美式拼法编写,所以如果在 .colour-picker{}中写 color:red就缺乏统一性。我以前主张同时用两种拼法,例如:
.color-picker,.colour-picker{}
但是我最近参与了一份规模庞大的 Sass 项目,这个项目中有许多的颜色变量(例如 $brand-color, $highlight-color等等),每个变量要维护两种拼法实在辛苦,要查找并替换时也需要两倍的工作量。
所以为了统一,把所有的 class 与变量都以你参与的项目的惯用拼法命名即可。
注释
我使用行宽不超过 80 字节的块状注释:
/** * 这是一个文档块(DocBlock)风格的注释。 * * 这里开始是描述更详细、篇幅更长的注释正文。当然,我们要把行宽控制在 80 字以内。 * * 我们可以在注释中嵌入 HTML 标记,而且这也是个不错的办法: * * * 如果是注释内嵌的标记的话,在它前面不加星号,否则会被复制进去。 */
在注释中应当尽量详细描述代码,因为对你来说清晰易懂的内容对其他人可能并非如此。每写一部分代码就要专门写注释以详解。
注释的拓展用法
注释有许多很先进的用法,例如:
准修饰选择器
你应当避免过分修饰选择器,例如如果你能写 .nav{}就尽量不要写 ul.nav{}。过分修饰选择器将影响性能,影响 class 复用性,增加选择器私有度。这些都是你应当竭力避免的。
不过有时你可能希望告诉其他开发者 class 的使用范围。以 .product-page为例,这个 class 看起来像是一个根容器,可能是 html或者 body元素,但是仅凭 .product-page则无法判断。
我们可以在选择器前加上准修饰(即将前面的类型选择器注释掉)来描述我们规划的 class 作用范围:
/*html*/.product-page{}
这样我们就能准确获知该 class 的作用范围而不会影响复用性。
其它例子如:
/*ol*/.breadcrumb{}/*p*/.intro{}/*ul*/.image-thumbs{}
这样我们就能在不影响代码私有度的前提下获知 class 作用范围。
代码标签
如果你写了一个新规则的话,可以在它上面加上标签,例如:
/** * ^navigation ^lists */.nav{}/** * ^grids ^lists ^tables */.matrix{}
这些标签可以使得其他开发者快速找到相关代码。如果一个开发者需要查找和列表相关的部分,他只要搜索 ^lists就能快速定位到 .nav, .matrix以及其它相关部分。
继承标记
将面向对象的思路用于 CSS 编写的话,你经常能找到两部分 CSS 密切相关(其一为基础,其一为拓展)却分列两处。我们可以用继承标记来在原元素和继承元素之间建立紧密联系。这些在注释中的写法如下:
在元素的基本样式中:
/** * Extend `.foo` in theme.css */ .foo{}
在元素的拓展样式中:
/** * Extends `.foo` in base.css */ .bar{}
这样一来我们就能在两块相隔很远的代码间建立紧密联系。
编写 CSS
之前的章节主要探讨如何规划 CSS,这些都是易于量化的规则。本章将探讨更理论化的东西,也将探讨我们的态度与方法。
编写新组件
编写新组件时,要在着手处理 CSS 之前写好 HTML 部分。这可以令你准确判断哪些 CSS 属性可以继承,避免重复浪费。
先写标记的话,你就可以关注数据、内容与语义,在这之后再添加需要的 class 和 CSS 样式。
面向对象 CSS
我以面向对象 CSS 的方式写代码。我把组件分成结构(对象)与外观(拓展)。正如以下分析:
.room{}.room--kitchen{}.room--bedroom{}.room--bathroom{}
我们在屋子里有许多房间,它们都有共同的部分:它们都包含地板、天花板、墙壁和门。这些共享的部分我们可以放到一个抽象的 .room{}class 中。不过我们还有其它与众不同的房间:一个厨房可能有地砖,卧室可能有地毯,洗手间可能没有窗户但是卧室会有,每个房间的墙壁颜色也许也会不一样。面向对象 CSS 的思路使得我们把相同部分抽象出来组成结构部分,然后用更具体的 class 来拓展这些特征并添加特殊的处理方法。
所以比起编写大量的特殊模块,应当努力找出这些模块中重复的设计模式并将其抽象出来,写成一个可以复用的 class,将其用作基础然后编写其它拓展模块的特殊情形。
当你要编写一个新组件时,将其拆分成结构和外观。编写结构部分时用最通用 class 以保证复用性,编写外观时用更具体的 class 来添加设计方法。
布局
所有组件都不要声明宽度,而由其亲元素或格栅系统来决定。
坚决不要声明高度。高度应当仅仅用于尺寸已经固定的东西,例如图片和 CSS Sprite。在 p, ul, div等元素上不应当声明高度。如果需要的话可以写 line-height,这个更加灵活。
格栅系统应当当作书架来理解。是它们容纳内容,而不是把它们本身当成内容装起来,正如你先搭起书架再把东西放进去。比起声明它们的尺寸,把格栅系统和元素的其它属性分来开处理更有助于布局,也使得我们的前端工作更高效。
你在格栅系统上不应当添加任何样式,他们仅仅是为布局而用。在格栅系统内部再添加样式。在格栅系统中任何情况下都不要添加盒模型相关属性。
UI 尺寸
我用很多方法设定 UI 尺寸,包括百分比, px, em, rem以及干脆什么都不用。
理想情况下,格栅系统应当用百分比设定。如上所述,因为我用格栅系统来固定栏宽和页宽,所以我可以不用理会元素的尺寸。
我用 rem 定义字号,并且辅以 px 以兼容旧浏览器。这可以兼具 em 和 px 的优势。下面是一个非常漂亮的 Sass Mixin,假设你在别处声明了基本字号(base-font-size)的话,用它就可以生成 rem 以及兼容旧浏览器的 px。
@mixin font-size($font-size){ font-size:$font-size +px; font-size:$font-size / $base-font-size +rem;}
我只在已经固定尺寸的元素上使用 px,包括图片以及尺寸已经用 px 固定的 CSS Sprite。
字号
我会定义一些与格栅系统原理类似的 class 来声明字号。这些 class 可以用于双重标题分级,关于这点请阅读 Pragmatic, practical font-sizing in CSS。
简写
CSS 简写应当谨慎使用。
编写像 background:red;这样的属性的确很省事,但是你这么写的意思其实是同时声明 background-image:none; background-position:top left; background-repeat: repeat; background-color:red;。虽然大多数时候这样不会出什么问题,但是哪怕只出一次问题就值得考虑要不要放弃简写了。这里应当改为 background-color:red;。
类似的,像 margin:0;这样的声明的确简洁清爽,但是还是应当 尽量写清楚。如果你只是想修改底边边距,就要具体一些,写成 margin-bottom:0;。
反过来,你需要声明的属性也要写清楚,不要因为简写而波及其它属性。例如如果你只想改掉底部的 margin,那就不要用会把其它边距也清零的 margin:0。
简写虽然是好东西,但是注意切勿滥用。
ID
在我们开始处理选择器之前,牢记这句话:
在 CSS 里坚决不要用 ID。
在 HTML 里 ID 可以用于 JS 以及锚点定位,但是在 CSS 里只要用 class,一个 ID 也不要用。
Class 的优势在于复用性,而且私有度也并不高。私有度非常容易导致问题,所以将其降低就尤为重要。ID 的私有度是 class 的 255倍,所以在 CSS 中坚决不要使用。
选择器
务必保持选择器简短高效。
通过页面元素位置而定位的选择器并不理想。例如 .sidebar h3 span{}这样的选择器就是定位过于依赖相对位置,所以把 span 移到 h3 和 sidebar 外面时就很难保持其样式。
结构复杂的选择器将会影响性能。选择器结构越复杂(如 .sidebar h3 span为三层, .content ul p a是四层),浏览器的消耗就越大。
尽量使得样式不依赖于其定位,尽量保持选择器简洁清晰。
作为一个整体,选择器应当尽量简短(例如只有一层结构),但是 class 名则不应当过于简略,例如 .user-avatar就远比 .usr-avt好。
牢记:class 无所谓是否语义化;应当关注它们是否合理。不要强调 class 名要符合语义,而要注重使用合理且不会过时的名称。
过度修饰的选择器
由前文所述,过度修饰的选择器并不理想。
过度修饰的选择器是指像 div.promo这样的。很可能你只用 .promo也能得到相同的效果。当然你可能偶尔会需要用元素类型来修饰 class(例如你写了一个 .error而且想让它在不同的元素类型中显示效果不一样,例如 .error{ color:red; } div.error{ padding:14px;}),但是大多数时候还是应当尽量避免。
再举一个修饰过度的选择器例子, ul.nav li a{}。如前文所说,我们马上就可以删掉 ul因为我们知道 .nav是个列表,然后我们就可以发现 a一定在 li中,所以我们就能将这个选择器改写成 .nav a{}。
选择器性能
虽然浏览器性能日渐提升,渲染 CSS 速度越来越快,但是你还是应当关注效率。使用间断、没有嵌套的选择器,不把全局选择器( *{})用作核心选择器,避免使用日渐复杂的 CSS3 新选择器可以避免这样的问题。
译注,核心选择器:浏览器解析选择器为从右向左的顺序,最右端的元素是样式生效的元素,是为核心选择器。
使用 CSS 选择器的目的
比起努力运用选择器定位到某元素,更好的办法是直接给你想要添加样式的元素直接添加一个 class。我们以 .header ul{}这样一个选择器为例。
假定这个 ul就是这个网站的全站导航,它位于 header 中,而且目前为止是 header 中唯一的 ul元素。 .header ul{}的确可以生效,但是这样并不是好方法,它很容易过时,而且非常晦涩。如果我们在 header 中再添加一个 ul的话,它就会套用我们给这个导航部分写的样式,哪怕我们设想的不是这个效果。这意味着我们要么要重构许多代码,要么给后面的 ul新写许多样式来抵消之前的影响。
你的选择器必须符合你要给这个元素添加样式的原因。思考一下, 「我定位到这个元素,是因为它是 .header 下的 ul ,还是因为它是我的网站导航?」这将决定你应当如何使用选择器。
确保你的核心选择器不是类型选择器,也不是高级对象或抽象选择器。例如你在我们的 CSS 中肯定找不到诸如 .sidebar ul{}或者 .footer .media{}这样的选择器。
表达清晰:直接找到你要添加样式的元素,而非其亲元素。不要想当然地认为 HTML 不会改变。 用 CSS 直接命中你需要的元素,而非投机取巧。
完整内容请参考我的文章 Shoot to kill; CSS selector intent
!important
只在起辅助作用的 class 上用 !important。用 !important提升优先级也可以,例如如果你要让某条规则 一直生效的话,可以用 .error{ color:red!important; }。
避免主动使用 !important。例如 CSS 写得很复杂时不要用它来取巧,要好好整理并重构之前的部分,保持选择器简短并且避免用 ID 将效果拔群。
魔数与绝对定位
魔数(Magic Number)是指那些「凑巧有效果」的数字,这东西非常不好,因为它们只是治标不治本而且缺乏拓展性。
例如 .dropdown-nav li:hover ul{ top:37px; }把下拉菜单移动下来远非良策,因为这里的 37px 就是个魔数。37px 会生效的原因是因为这时 .dropbox-nav碰巧高 37px 而已。
这时你应该用 .dropdown-nav li:hover ul{ top:100%; },也即无论 .dropbox-down多高,这个下拉菜单都会往下移动 100%。
每当你要在代码中放入数字的时候,请三思而行。如果你能用一个关键字(例如 top:100%意即「从上面拉到最下面」)替换之,或者有更好的解决方法的话,就尽量避免直接出现数字。
你在 CSS 中留下的每一个数字,都是你许下而不愿遵守的承诺。
条件判断
专门为 IE 写的样式基本上都是可以避免的,唯一需要为 IE 专门处理的是为了处理 IE 不支持的内容(例如 PNG)。
简而言之,如果你重构 CSS 的话,所有的布局和盒模型都不用额外兼容 IE。也就是说你基本上不用 或者类似的兼容 IE 的写法。
Debugging
如果你要解决 CSS 问题的话, 先把旧代码拿掉再写新的。如果旧的 CSS 中有问题的话,写新代码是解决不了的。
把 CSS 代码和 HTML 部分删掉,直到没有 BUG 为止,然后你就知道问题出在哪里了。
有时候写上一个 overflow:hidden或者其它能把问题藏起来的代码的确效果立竿见影,但是 overflow 方面可能根本就没问题。所以 要治本,而不是单纯治标。
CSS 预处理器
我用 Sass。使用时应当 灵活运用。用 Sass 可以令你的 CSS 更强大,但是不要嵌套得太复杂。在 Vanilla CSS 中,只在必要的地方用嵌套即可,例如:
.header{}.header .site-nav{}.header .site-nav li{}.header .site-nav li a{}
这样的写法在普通 CSS 里完全用不到。以下为 不好的Sass 写法:
.header{ .site-nav{ li{ a{} } }}
如果你用 Sass 的话,尽量这么写:
.header{}.site-nav{ li{} a{}}
Update 最新版本 英文
-
原文 CSS Guildelines
翻译 chadluo