CSS规范-BEM(2)

参考前端css、less、sass编码规范---BEM
不用ID标识符
主要考虑到样式重用性以及与页面的耦合性。ID本来就是唯一的而且权值那么大,还不如写在行内样式。

#button {
  text-decoration: underline;
}

不用大于三层嵌套选择器
嵌套选择器增加了代码耦合,使重用变得不可能,过多的嵌套会导致显示性能下降。简洁的选择器不仅可以减少css文件大小,提高页面的加载性能,让浏览器解析时也会更加高效。同时也会提高开发人员的开发效率,降低了维护成本。子代选择器不好的地方还在于,如果层次关系过长,逻辑不清晰,非常不利于维护。
css的匹配原理不是从左到右的,而是从右到左的,从右边开始匹配是为了尽早过滤掉一些无关的样式规则和元素。

.button_hovered .button__text {
  text-decoration: underline;
}

不用组合选择器
组合选择器的耦合性更强,而且可维护性更加差。

.button.button_theme_islands {
  text-decoration: underline;
}

我们通过BEM命名法写样式如下:

.block{}
.block__element{}
.block--modifier{}
.block__element--modifier{}

BEM解决了的问题

  • 页面CSS模块化,每个block就是一个模块,模块间相互独立,不会造成污染
  • 多级的class命名,避免选择器的嵌套结构
  • 减少通配符*或者类似[hidden="true"]这类选择器的使用
  • 巧妙运用scss的特性,保证了样式的可维护性

BEM将页面解析为block和element,然后根据不同的状态使用modifier来设置样式,在scss的属性帮助下,其实,写BEM并不麻烦。


一个块就是一个组件。这有点抽象,所以让我们用示例来学习。

假设您正在建立一个联系表单。在这种情况下,这个表单可以是一个块。在 BEM 中,块被写为像 class 的名字一样,如下所示:

.form { /* styles */ }

BEM 使用 .form 而不是 <form> 元素的原因是因为 类允许无限的可重用性,而即使是最基本的元素也可能改变样式。
按钮很好地阐释了可以包含不同样式的块。如果将 <button> 元素的背景颜色设置为红色,则所有 <buttons> 都将被强制继承红色背景。接下来,你必须通过覆盖你的 <button> 元素来修复代码(并且可能会在修复中“伤及无辜” )。

button { 
    background-color: red; 
} 

.something button { 
    background-color: blue;
}

如果设置了一个 .button 类的按钮,则可以在任何 <button> 元素上选择是否使用 .button 类。如果你需要一个不同的背景颜色,你所做的就是改成一个新的 class,比如说 .button--secondary

.button { 
    background-color: red; 
}

.button--secondary { 
    background-color: blue;
}

这给我们引入了 BEM 的下一部分 —— 修饰符。

修饰符
修饰符是改变某个块的外观的标志。要使用修饰符,可以将 --modifier 添加到块中。
从上面的按钮示例继续,修改的按钮将被命名为 .button--secondary
在传统的 BEM 中,当你使用修饰符时,你应该将块和修饰符添加到 HTML 中,以便在新的 .button--secondary 中不重写 .button 样式。

<button class="button">Primary button</button>
<button class="button button--secondary">Secondary button</button>

.button { 
    padding: 0.5em 0.75em; 
    background-color: red; 
} 
.button--secondary { 
    background-color: green; 
}

注意为什么没有必要在 .button--secondary 中重新声明 padding,因为它已经在 .button 中声明了。
但是,我并不喜欢在HTML中再加一个 .button,因为 .button--modifier 已经告诉我,它是一个带有 --secondary 标志的 .button 。理想情况下,我的 HTML 应该是这样的:

<button class="button">Primary button</button>
<button class="button--secondary">Secondary button</button>

这更简洁
不幸的是,如果 HTML 中没有 .button,我们必须回到非简洁的 CSS:

.button { 
    padding: 0.5em 0.75em; 
    background-color: red; 
} 
.button--secondary { 
    padding: 0.5em 0.75em; 
    background-color: green; 
}

很繁琐!
有种方法可以编写简洁的 CSS,而不需要额外的 class,那就是使用less、sass中的使用 mixin ,下面是sass的实例:
如果使用 Sass 或任何其他预处理器,则 使用mixin来封装 需要重用的 所有代码。在我们的按钮示例中,我们只需要将 padding 写入 mixin。 在这里,我在块中调用这个 mixin:

@mixin button {
    padding: 0.5em 0.75em; 
} 
.button { 
    @include button;
    background-color: red;
} 
.button--secondary { 
    @include button;
    background-color: green; 
}

元素
元素是块的子节点。为了表明某个东西是一个元素,你需要在块名后添加 __element。所以,如果你看到一个像那样的名字,比如 form__row ,你将立即知道 .form 块中有一个 row 元素。

<form class="form" action=""> 
    <div class="form__row"> 
        <!-- ... --> 
    </div> 
</form>

.form__row { 
    /* styles */ 
}

BEM 元素有两个优点 :

  1. 你可以让 CSS 的优先级保持相对扁平。
  2. 你能立即知道哪些东西是一个子元素。

为了解释以上两点,考虑使用两个单独的 class 的替代方法(许多框架这么做的)。你可能会用这样的东西:

<form class="form" action=""> 
    <div class="row"> 
        <!-- ... --> 
    </div>
</form> 

.form .row { 
    /* styles */ 
}

如果你使用 BEM 元素,则可以使用优先级为 10 而不是 20 的的选择器来为 .form__row 提供样式。此外,你可以立即分辨出(不论是在 HTML 还是 CSS 中).form__row.form__row的子节点。
有一件事你需要了解。永远不应该链式命名 BEM 元素。 如果你的 class 最终像这样 .form__row__input,你做的事情是非常错误的。

有两种方法可以绕过长长的 BEM 链式命名。 他们是:

  • 只把子子元素链接到有意义的块
  • 创建新的块来保存元素

链接孙元素到块
虽然 BEM 建议你将 BEM 元素写作 .block__element ,但它不会规定你的 HTML 应如何。所以,只要有意义的话,你可以把你的孙元素连在一起。
接下来是一个例子。在下面的代码中,你将看到 .article__header.article 的子元素。.article__titlearticle 的孙元素(或者说是 .article__header 的子元素,如果你将它们同时表示为 .article 的子元素,就没有冲突,因为这个表单同时只有他们存在。

<article class="article"> 
    <header class="article__header"> 
        <h1 class="article__title"></h1> 
    </header> 
</article>

虽然这样有效,你也会遇到无意义的链接孙元素的情况。举个例子:

<section class="comments"> 
    <h2 class="comments__title"></h2> 
    <article class="comments__comment"> 
        <h3 class="comments__comment-title"></h3> 
    </article> 
    <article class="comments__comment"> 
        <h3 class="comments__comment-title"></h3> 
    </article> 
    <!-- ... --> 
</section>

此时你需要创建新块来保存孙元素。
创建新的块来保存孙元素
在上述情况下,你可以轻松地将 .comments__comment 拆为 .comments.comment

<section class="comments"> 
    <h2 class="comments__title"></h2> 
    <article class="comment"> 
        <h3 class="comment-title"></h3> 
    </article> 
    <article class="comment"> 
        <h3 class="comment-title"></h3> 
    </article> 
    <!-- ... --> 
</section>

这更有意义,不是吗?如果你这样做,请确保将 .comments.comment块放在同一个文件中,以方便参考。

在用例中添加为 BEM 添加的 —— 命名空间。
命名空间

  • .l-: 布局(layouts)

  • .o-: 对象(objects)

  • .c-: 组件(components)

  • .js: js的钩子(JavaScript hooks)

  • .is-|.has-: 状态类(state classes)

  • .t1|.s1: 排版大小(typography sizes)

  • .u-: 实用类(utility classes)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 217,509评论 6 504
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,806评论 3 394
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 163,875评论 0 354
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,441评论 1 293
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,488评论 6 392
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,365评论 1 302
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,190评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 39,062评论 0 276
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,500评论 1 314
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,706评论 3 335
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,834评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,559评论 5 345
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,167评论 3 328
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,779评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,912评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,958评论 2 370
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,779评论 2 354

推荐阅读更多精彩内容

  • 问答题47 /72 常见浏览器兼容性问题与解决方案? 参考答案 (1)浏览器兼容问题一:不同浏览器的标签默认的外补...
    _Yfling阅读 13,751评论 1 92
  • HTML 5 HTML5概述 因特网上的信息是以网页的形式展示给用户的,因此网页是网络信息传递的载体。网页文件是用...
    阿啊阿吖丁阅读 3,887评论 0 0
  • 前端开发面试题 面试题目: 根据你的等级和职位的变化,入门级到专家级,广度和深度都会有所增加。 题目类型: 理论知...
    怡宝丶阅读 2,582评论 0 7
  • Swift1> Swift和OC的区别1.1> Swift没有地址/指针的概念1.2> 泛型1.3> 类型严谨 对...
    cosWriter阅读 11,101评论 1 32
  • 基本常识与实践 css的每一个语句包括一个场所,以及这个场所的一个属性,还要应用到这个属性一个样式,一个典型的cs...
    丁俊杰_阅读 1,037评论 0 0