Bem is a library that allow you to define BEM style class attribute values to use in your markup.
Let for example you want to include a class attribute value into your html tag. You can do it by using the library.
<div class="<?php echo $bem->value() ?>">
<!-- your content here -->
</div>
The library works with WordPress but also as a standalone library to use in your projects.
If you use it into your WordPress project you can take advantage of the bem
filter allowing you
to manipulate the bem string depending on your needs.
It's is possible to retrieve the entire class value such as the block, element and modifiers singularly.
Since Version 1.0.0 the string values are no longer sanitized, so you have to use a function like sanitize_html_class
.
The classes now only check against the value passed in construction phase.
PHP >= 7.1.33
BEM — Block Element Modifier is a methodology that helps you to create reusable components and code sharing in front-end development.
- Block
- Encapsulates a standalone entity that is meaningful on its own. While blocks can be nested and interact with each other, semantically they remain equal; there is no precedence or hierarchy. Holistic entities without DOM representation (such as controllers or models) can be blocks as well.
- Element
- Parts of a block and have no standalone meaning. Any element is semantically tied to its block.
- Modifier
- Flags on blocks or elements. Use them to change appearance, behavior or state.
Example
.block {}
.block__element {}
.block block--modifier {}
For more information have a look at: getbem
To use the bem value you must create a instance of a class that implements Valuable
interface such as Standard
then
call value
method.
Since version 1.0.0 isn't possible to retrieve the bem's components separately, this is because the
responsibility to hold data has been moved under a different class that implements Bem
interface.
Also, the modifiers
are no longer an array but an instance of a class that implements Modifiable
.
The class used to hold modifiers is BlockModifier
.
$bem = new Data('block');
$standard = new Standard($bem);
$standard->value(); // will print 'block'
$bem = new Data('block', 'element');
$standard = new Standard($bem);
$standard->value(); // will print 'block__element'
$modifiers = new BlockModifiers(['modifier'], 'block');
$bem = new Data('block', 'element', $modifiers);
$standard = new Standard($bem);
$standard->value(); // will print 'block block--modifier'
To create a bem as you seen above isn't much complex but isn't simple as doing it in one line of code
(if you want to keep your code readable). For that reason you make use of a simple Factory
class
that allow you to create all of the Bem classes you want just in one line of code by passing only
the necessary data.
$standardBem = Factory::createStandard('block', 'element', ['modifier']);
Then you can use the object as usual, by calling the method value
.
Note:
Even though it's possible to pass all of the parameters to the class that implements Bem
interface,
when both block and modifiers are passed the element is ignored.
This is right and in line with the BEM requirements. Infact isn't possible to have a BEM string like block block--modifier__element
.
Creating one object for every changes you need to make when you are building a component or a block
for you project can be quite prolix and useless because most of the time what you want to do is to have
a block
and than change the element
or the modifier
parts.
For this reason Bem
offer a configurable service with which is possible to set only the block
part of the bem object so you can inject it into your classes and change only the parts needed during
the component building.
$service = Factory::createServiceForStandard('block');
Then you can make use of forElement
and withModifiers
methods to update the bem object.
class MyBlockClass
{
private $bemService;
public function __construct(Service $bemService)
{
$this->bemService = $bemService;
}
public function buildComponentElement()
{
$element = $this->bemService->forElement('element');
$subElement = $this->bemService->forElement('sub-element');
?>
<div class="<?= $element->value() ?>">
<div class="<?= $subelement->value() ?>">
</div>
</div>
<?php
}
public function buildBlockVariant()
{
$blockVariant = $this->bemService->withModifiers(['modifier']);
?>
<div class="<?= $blockVariant->value() ?>">
</div>
<?php
}
}
It's possible to pass more than one modifier to BlockModifier
infact the __construct
method
get an array of strings.
So for example, the following code will output block block--modifier-one block--modifier-two
.
$modifiers = new BlockModifiers(['modifier', 'modifier-two'], 'block');
$bem = new Data('block', 'element', $modifiers);
$standard = new Standard($bem);
$standard->value(); // will print 'block block--modifier block--modifier-two'
Since version 1.0.0 there are additional checks to the block, element and modifier values to ensure
a valid string is passed during construction. The check is made against the pattern [^a-zA-Z0-9\-\_]
.
Since version 1.0.0 it's no longer possible to retrieve the bem components from the Valuable
instance,
you have to use Data
instance if you want to get them.
To report a bug simply open an issue on github project
This library is license under MIT
For more info read the LICENSE
file