Pure-sass functions and mixins to make writing BEM a piece of cake. Keeps things simple and handles tricky cases gracefully.
- ✓ [lib]Sass ≥ 3.4
- âś— Â
[lib]Sass < 3.4
/** simple **/
@include Block(notification) { // .notification {
border: 1px solid #ccc; // border: 1px solid #ccc;
padding: 20px; // padding: 20px;
// }
@include Elem(link) { // .notification__link {
float: right; // float: right;
} // }
// .notification--error {
@include Mod(error) { // border-color: red;
border-color: red; // }
}
}
/** more involved **/
@include B(myblock) { // .myblock {
background: white; // background: white;
// }
@include E(subelem) { // .myblock__subelem {
color: blue; // color: blue;
} // }
// .myblock:hover {
&:hover { // text-decoration: underline;
text-decoration: underline; // }
// .myblock:hover .myblock__subelem {
@include E(subelem) { // color: green;
color: green; // }
} // .myblock--mymod {
} // background: green;
// }
@include M(mymod) { // .myblock--mymod .myblock__subelem {
background: green; // color: green;
// }
@include E(subelem) { // .myblock--mymod:hover {
color: green; // font-weight: bold;
} // }
// .myblock--mymod:hover .myblock__subelem {
&:hover { // color: red;
font-weight: bold; // }
@include E(subelem) {
color: red;
}
}
}
}
Right now you have 2 options:
- Copy/paste the
bemerald.scss
file (it's all one self-contained file) into your project - Install via
npm install bemerald
and configure your builder to point to thenode_modules
folder
TODO: Provide easier installation & plugins for common build tools
in order of most commonly used...
The build-in mixins allow you to write your SCSS in a nested way that feels very natural, but still get flat root-level selectors, as per BEM methodology.
Simply unrolls into a class-selector. Main purpose of using this mixin is to clearly denote the start of a BEM block.
@include Block(notification) { // .notification {
position: relative; // position: relative;
border: 1px solid #ccc; // border: 1px solid #ccc;
padding: 10px; // padding: 10px;
} // }
// Shorthand alias
@include B(notification) { ... }
Unrolls into a BEM block-modifier selector.
May be either:
- String: Simple modifier string.
mod
results in.block--mod
- Map: Modifier string with value.
(mod: value)
results in.block--mod-value
Notes:
- This mixin does not generate element-modifiers. Use
Elem(elem, $mod:mod)
. - Nesting a Mod inside a pseudo-selector is not supported, because what that should mean isn't clear.
@include Block(notification) { // .notification--error {
// ... // background: rgba(255, 0, 0, 0.15);
@include Mod(error) { // border-color: red;
background: rgba(red, 0.15); // }
border-color: red;
}
@include Mod((theme: growl)) { // .notification--theme-growl {
top: 0; // top: 0;
right: 0; // right: 0;
width: 200px; // width: 200px;
height: auto; // height: auto;
} // }
@include Mod((theme: bar)) { // .notification--theme-bar {
bottom: 0; // bottom: 0;
left: 0; // left: 0;
width: 100%; // width: 100%;
height: 200px; // height: 200px;
} // }
}
// Shorthand alias
@include M(error) { ... }
Unrolls into a proper BEM element selector, depending on the context:
- Inside just a block, yields a root-level
.block__elem
- Inside a mod or pseudo-selector, yields a nested
.block--mod .block__elem
If $mod
is included, it is appended to the block selector, like .block__elem--mod
. Multiple $mod
s are not supported.
@include Block(notification) { // .notification__dismiss-btn {
// ... // position: absolute;
@include Elem(dismiss-btn) { // right: 0;
position: absolute; // font-weight: bold;
right: 0; // }
font-weight: bold;
}
@include Elem(dismiss-btn, $mod:on-left) { // .notification__dismiss-btn--on-left {
left: 0; // left: 0;
right: auto; // right: auto;
} // }
// Works inside mods:
@include Mod(error) { // .notification--error .notification__dismiss-btn {
@include Elem(dismiss-btn) { // display: none;
display: none; // }
}
}
// ... and pseudo-selectors
&:hover { // .notification:hover .notification__dismiss-btn {
@include Elem(dismiss-btn) { // transform: scale(1.1);
transform: scale(1.1); // }
}
}
}
// Shorthand aliases (mixin and argument)
@include E(dismiss-btn, $m:on-left) { ... }
Unrolls into a full BEM selector. See the bem-selector
documentation for parameter details.
@include BEM(notification, $mod:error, $elem:dismiss-btn) {
color: red;
}
// .notification--error .notification__dismiss-btn {
// color: red;
// }
Generates a full BEM selector.
-
$block
: Required. A string block name. -
$elem
: Optional. A sub-element name. If$mod
is not present, it is joined with$block
. If$mod
is present, it is nested under$block--$mod
May be either:- String: Simple element name. Results in
.block__elem
- List: Pair like
($elem-name, $elem-mod)
Applies the element-modifier to the subelement. Multiple elem-mods are not supported. Same types as$mod
.
- String: Simple element name. Results in
-
$mod
: Optional. A block modifier. May be either:- String: Simple modifier string. Results in
.block--mod
- Map: ($modifier-name: $modifier-value). Results in
.block--mod-val
- String: Simple modifier string. Results in
-
$mods
: Optional. List. Allows generating selectors for multiple mods at once. Each element of the list should be one of the allowed type for$mod
$s: bem-selector(block); // .block
$s: bem-selector(block, $e:elem); // .block__elem
$s: bem-selector(block, $e:(elem,emod); // .block__elem--emod
$s: bem-selector(block, $m:mod); // .block--mod
$s: bem-selector(block, $m:(mod:val)); // .block--mod-val
$s: bem-selector(block, $m:mod, $e:elem); // .block--mod .block__elem
$s: bem-selector(block, $m:(mod-a,(mod-b:bval)), $e:elem);
// .block--mod-a .block__elem, .block--mod-b-bval .block__elem
You could use it like...
#{bem-selector(block, $e:elem, $m:mod)} {
// this is exactly what the BEM mixin does
}
The provided mixins come with "shorthand aliases" so you can save some typing.
@include Block(b) { ... }
@include B(b) { ... }
@include Elem(e, $mod:m) { ... }
@include E(e, $m:m) { ... }
@include Mod(m) { ... }
@include M(m) { ... }
npm install; npm install -g mocha;
npm test
Runs tests located in /test
- Sass in general (duh - sass rocks)
- libsass specifically (speed rocks)
- true (tests rock)
- SassMeister (instant feedback rocks, too)
- You (also rock)
Released under MIT license, same as Sass.
- Provide easier installation & plugins for common build tools
- test all functions. Right now at least
bem--extract-block
needs tests. - try to test mixins. Might be tricky since they use
@at-root
- automated test in various sass implementations?