CSS 命名规范总结
CSS 命名规范总结
BEM 命名给 CSS 以及 html 提供清晰结构,命名空间提供更多信息,模块化提高代码的重用,以达到 CSS 命名语义化、可重用性高、后期维护容易、加载渲染快的要求。
首先,这是一篇仅对 CSS 命名的总结,更多 CSS 规范请参考规范文档。其次 CSS 命名并无对错之分,只是不同的命名方式和代码的优雅程度与后期维护有着密切关系。
@iamdevloper pic.twitter.com/4SRgVDDaiC
— Saeed Prez (@SaeedPrez) 2016年7月14日
CSS 很容易写,但是维护它却很难,因为 CSS 声明属性的性质不具备编程语言流程控制的特点,CSS(Cascading Style Sheets)也可以层叠生效,不同地方的 classes 相互影响。CSS 命名是规范中最重要的部分,它几乎就是 CSS 规范的全部,我们也就是依赖它完成对 CSS 的控制。
分离
曾经我们这样定义选择器:
.fl
float: left
.fr
float: right
.mr10
margin-right: 10px
.pb10
padding-bottom: 10px
.hidden
overflow: hidden
.highlight
color: $green
在 html 中添加选择器:
.fl.pb10.highlight
这就是规则分离的做法,将一些确定值的 CSS 规则单独定义选择器,这样做的目的是提高代码的重用,但是也有很多缺点:
- 可供分离的属性有限,还有很多不确定值的属性,比如 width、color,而且现实布局的多样性使 mr10 这样的分离也很少用到,因此无法做到彻底分离;
- 分离之后导致 html 中 class 特别多,而且不具有语义化;
既然还是需要添加语义化的 class 来定义一部分样式,那么分离一部分出去也没有意义,操作上缺点远远大于优点。吸取这种命名的优点,现在依然会使用它,但是只保留了少数语义化且常用的,例如:.hidden
、.highlight
。
不好的写法
html:
.container
.actions
.card_wrapper
.card
.title.light
.content
.list
i.fa.fa-cog
CSS:
body.events,
body.aaa,
body.bbb
.actions
.card_wrapper
.card
.title
.content
.list
.fa
.another
以上是我们现在常见的写法,但是是需要绝对避免的,虽然 class 有一定语义了,但是并不能真正的重用。class 命名太通用了,我们不知道还有什么地方用到 .title
.content
这样的命名,就不敢直接使用,一方面层层嵌套防止冲突,一方面想要重用时就在顶层添加 body.aaa
这样的东西,导致很多样式用不到,写的乱七八糟。
这样一直写下去,旧的 CSS 我们永远不会删,只敷衍写出效果,如此往复循环,最终就会像上面的窗帘,让人忍无可忍。每个页面都有 .content
甚至每个模块都有 .content
,.content
就毫无意义(.content 不是抽象化的全局共用模块)。深层次的嵌套也会降低渲染效率。
另一种常见的:
html:
.main-news-box
h2.news-title
ul.main-news-list
li
a
span
span
CSS:
.main-news-box
.main-news-box .news-title
.main-news-list
.main-news-list li
.main-news-list a
.main-news-list span
这种写法的问题:
.news-title
在别的地方有没有定义?能不能单独使用?其他地方定义的.news-title
会相互影响,父级更深层次的li
、a
也会覆盖现有的定义。- 这种命名也有性能问题,CSS 从右向左渲染,例如
.main-news-list a
,先渲染页面所有的 a,再渲染 .main-news-list。
BEM 命名
BEM 是一种真正消除不确定性的命名方式,它使得结构样式更加清晰,我们有足够的信心做任何修改。
- block:模块,名字单词间用
-
连接 - element:元素,模块的子元素,以
__
与 block 连接 - modifier:修饰,模块的变体,定义特殊模块,以
--
与 block 连接
例如:
.user-home-nav
.user-home-nav-item.user-home-nav-item--small
.user-home-nav-item__icon
.user-home-nav-item__text
CSS 中这样写:
.user-home-nav
&-item
&--small
&__icon
&__text
这样命名的好处:
- 语义化,此处的语义化不是指 html 标签的语义化,对 SEO 可能也没有任何意义,但是这是一种人阅读的语义化。语义化的重要意义:宁可增加 html 大小,力图使维护变得轻松。
- 减少层层嵌套,有利于渲染效率。以上 sass 解析之后就是:
.user-home-nav
.user-home-nav-item
.user-home-nav-item__icon
.user-home-nav-item__text
.user-home-nav-item--small
缺点?
- 有人说又是
__
又是--
,什么乱七八糟的符号,我并不觉得有任何问题,这种命名让结构十分清晰,只看 html 或 CSS 就能看出它对应的模块或者模块的元素或者不同类型的模块,后期维护的增删也十分确定。 - 长长的 class 名增加了 html 的字节。从加载速度上来说,当然是名字越短越好。但是这个对应的就是语义化,取舍上我更倾向于语义化。
BEM 的问题
但是 BEM 最大的问题却是这个:如何给 modifier 下的 element 定义规则呢?
.user-home-nav
&-item
&--small
.user-home-nav__icon
.user-home-nav__text
&__icon
&__text
这样做太不优雅了,太违背 sass 的优雅性质了。或许是我没有找到更好的写法?
还能产生更多的写法,例如 .user-home-nav-item--small__icon
、.user-home-nav-item__icon--small
,但是或多或少会存在问题,还是尽量保持简单。
其他问题:
- 如何确定一个 block?就像上面
.user-home-nav-item
也可以写成.user-home-nav__item
,如果 item 下有 title,那么就是.user-home-nav__title
,以避免写成.user-home-nav__item__title
,但是如果 nav 已有 title.user-home-nav__title
了,或许我们要把 item 下的 title 写成.user-home-nav__item-title
,总之为了使 BEM 也不至于复杂化,命名上也许还是要纠结一番。所以我最终写成了.user-home-nav-item
,也许可以这么写下去,没有 B 和 M,trello 貌似就是这样,例如:.attachment-thumbnail-details-options-item-text
。 - BEM 导致 CSS 规则重用性降低,如果重用尽量考虑写成通用模块.
BEM + 命名空间
命名空间定义 block 间的关系,这种方式网站上还没有大范围使用,最常见的如 .js-
表示定义 JavaScript 钩子,不用于定义样式。常见命名空间:
- o-:表示一个对象(Object),如
.o-layout
。 - c-:表示一个组件(Component),指一个具体的、特定实现的 UI。如
.c-avatar
。 - u-:表示一个通用工具(Utility),如
.u-hidden
。 - t-:表示一个主题(Theme),如
.t-light
。 - s-:表示一个上下文或作用域(Scope),如
.s-cms-content
。 - is-,has-:表示一种状态或条件样式。如
.is-active
- _:表示一个 hack,如
._important
。 - js-:表示一个 JavaScript 钩子。如
.js-modal
。 - qa-:表示测试钩子。
例如一个圆形头像组件:
.c-avatar-circle
&__img
display: block
vertical-align: top
max-width: 100%
border-radius: 100%
&__name
text-align: center
以上,BEM 产生的重用性问题还是没有解决,如何解决?
模块
模块化类似于分离的概念,也是分离进化的结果,我们从分离中提取出 .hidden {overflow: hidden}
这样的命名,进而扩展 .avatar {}
这样的命名,不再只包含一条规则。看一下 airbnb 的圆形头像写法:
html:
a.pull-right.media-photo-badge.card-profile-picture.card-profile-picture-offset.is-superhost
.media-photo.media-round
img
CSS:
.card-profile-picture
width: 60px
height: 60px
&-offset
position: relative
top: -40px
margin-bottom: -40px
&.is-superhost
position: relative
img
width: 56px
height: 56px
.media-photo
backface-visibility: hidden
position: relative
display: inline-block
vertical-align: bottom
overflow: hidden
background-color: #bbb
.media-round
border-radius: 50%
border: 2px solid #fff
这个是比较复杂,相对特例的。.media-photo
和 .media-round
是典型的抽象出来的模块,一个定义照片的基本属性,一个定义圆形属性。.card-profile-picture
是特定的命名定义特定具体的值。
总结
- 基本:以英文单词命名,避免无意义的缩写,以
-
连接。 - 我在一开始使用 BEM 的时候,也有很多疑惑,而且现在也不确定它完备的用法。
- BEM 不是万能的,但是无论如何 BEM 是应该使用的方法。
- 命名空间给 CSS 命名提供更多信息。
- 定义模块提高代码的重用。
- CSS 命名要求:语义化易于理解,可重用性高,后期维护容易,加载渲染快。
参考文章
CSS 命名规范总结的更多相关文章
- CSS命名规范
DIV+CSS规范命名大全集合 前端人员必看CSS命名规范 整理: 文件名必须由小写字母.数字.中划线组成 ).所有的命名最好都小写,一律采用小写加中划线的方式,不允许使用大写字母或 _2).属性的值 ...
- html,css命名规范 (转)
HTML+CSS命名规范总结 1.HTML部分 1.1添加必须的utf-8的字符集,并且使用HTML5的简洁 方式: <meta charset="utf-8"> 1. ...
- web前端开发CSS命名规范参考
做为一个web前端工程师,每天接触HTML.css就像吃饭一样,但是作为一名合作.优秀的web前端工程师,对DIV+CSS命名还是有一定的规范的,本文整理了一份web前端开发中DIV+CSS各种命名规 ...
- CSS 命名规范及标题供参考与学习
一.CSS 命名规范 XHTML-CSS写作建议 所有的xhtml代码小写 属性的值一定要用双引号("")括起来,且一定要有值 每个标签都要有开始和结束,且要有正确的层次 空元 ...
- Html+CSS命名规范:
Html+CSS命名规范: 1.样式命名: 2.样式文件命名:
- css命名规范: BEM 的命名法
整理自:前端早读课[第1183期]这些 CSS 命名规范,将省下你大把调试时间 试图解决 3 类问题: 仅从名字就能知道一个 CSS 选择器具体做什么 从名字能大致清楚一个选择器可以在哪里使用 从 C ...
- CSS命名规范(规则)常用的CSS命名规则
CSS命名规范(规则)常用的CSS命名规则 CSS命名规范(规则)常用的CSS命名规则 头:header 内容:content/container 尾:footer ...
- 通用CSS命名规范
一.文件命名规范 样式文件命名主要的 master.css布局,版面 layout.css专栏 columns.css文字 font.css打印样式 print.css主题 themes.css [/ ...
- CSS 命名规范将省下调试时间
我听说很多开发者厌恶 CSS.而在我的经验中,这往往是由于他们并没有花时间来学习 CSS. CSS 算不上是最优美的『语言』,但迄今二十多年来,它都是美化 web 举足轻重的工具.从这点来说,也还算不 ...
随机推荐
- jenkins双向备份;高可用部署;
如果把一个Jenkins的整个目录赋值到另一个Jenkins的目录,则需要务必保持两个Jenkins版本是相同的,不然容易出现Jenkins插件兼容性问题. 另外使用inotify+rsync备份的时 ...
- javascript学习笔记------概念相关
javascript中的函数.对象 1. 在javascript中,函数是被当成一种数据类型,它可以被存储在一个变量.数组.对象中,可以被当作参数传递到另一个函数中. 函数就像是字符串和数字这样的的数 ...
- Thinkphp学习笔记-controller与view绑定
$this->display(); 通过上面的代码则可以输出controller所对应的view
- 如何使用JW Player来播放Flash并隐藏控制按钮和自定义播放完成后执行的JS
在一个客户项目中播放的flash需要进行定制如不显示控制按钮,flash播放完成后执行特定的js等,在用过了N多的JQery插件和播放器后最终JW Player插件可以满足我的以上要求 因为JW Pl ...
- Linux统计/监控工具SAR详细介绍
转载:http://www.ctohome.com/FuWuQi/1b/688.html sysstat 工具简介 sysstat 是 Linux 系统中的常用工具包.它的主要用途是观察服务负载,比如 ...
- 安装Was liberty之步骤
安装文件下载:http://pan.baidu.com/s/1dDl8PuL 安装三大步骤:拷贝文件,安装VNC和安装WasLiberty 拷贝文件是将需要的文件InstalMgr1.6.2_LNX_ ...
- 网络地址转换相关函数使用(inet_addr,inet_ntoa,inet_addr)
aa 相关函数原型及参数类型: 函数原型:int inet_aton(const char *cp, struct in_addr *inp); in_addr_t inet_addr(const c ...
- Flutter 动画使用
旋转动画 透明度变换动画 在Android中,可以通过View.animate()对视图进行动画处理,那在Flutter中怎样才能对Widget进行处理 在Flutter中,可以通过动画库给wid ...
- 算法笔记_149:图论之桥的应用(Java)
目录 1 问题描述 2 解决方案 1 问题描述 1310 One-way traffic In a certain town there are n intersections connected ...
- TP 查询语句中如何使用 FIND_IN_SET 这样的查询方法
TP 查询语句中如何使用 FIND_IN_SET 这样的查询方法 $condition['_string'] = 'FIND_IN_SET('.$citys.',city)';