Не важно, узнали ли вы о БЭМ только сейчас, или следите за ним с самого начала, вы, возможно, уже оценили столь полезную методологию. Если вы не знаете, что такое БЭМ, я рекомендую прочитать вам об этом на сайте БЭМ перед тем, как продолжить чтение этой статьи, потому что я буду использовать термины, которые предполагают базовое понимание этой CSS методологии.
image
Эта статья нацелена на людей, которые уже используют БЭМ и желают использовать его более эффективно, а также на тех, кто хочет узнать о нем больше.


Сейчас у меня нет никаких сомнений о способе называть вещи. Единственное, что отталкивало меня на протяжении длительного времени, — это уродливый синтаксис. Дизайнер внутри меня не хотел загромождать мою шикарную разметку ужасными двойными подчеркиваниями и дефисами.


Разработчик во мне смотрел на это прагматично. И, в конечном счете, модульный способ построения пользовательского интерфейса перевесил правую сторону моего мозга, которая утверждала: "Но он недостаточно хорош!". В таких случаях, я предпочитаю функциональность над внешней формой. В любом случае, хватит пустых разговоров. Представляю вашему вниманию 10 проблем, с которыми я поборолся и несколько советов, как разрешить их.


1. "Что делать с "внучатыми" селекторами (и не только)?"


Для ясности, внучатый селектор используется, когда вам нужно обратиться к элементу, вложенному на два уровня. Эти плохие мальчики — проклятье моей жизни, и я уверен, что это одна из причин, по которой у людей появляется немедленное отвращение к БЭМ. Приведу пример:


<div class="c-card">

    <div class="c-card__header">
        <!-- Появляется внук… -->
        <h2 class="c-card__header__title">Title text here</h2>
    </div>

    <div class="c-card__body">

        <img class="c-card__body__img" src="http://some-img.png" alt="description">
        <p class="c-card__body__text">Lorem ipsum dolor sit amet, consectetur</p>
        <p class="c-card__body__text">Adipiscing elit.
            <a href="/somelink.html" class="c-card__body__text__link">Pellentesque amet</a>
        </p>

    </div>
</div>

Как вы могли заметить, имена могут быстро выйти из под контроля и, чем больше вложенности, тем безобразнее становится имя класса элемента. Я использовал короткий блок с названием c-card и короткие имена body, text, link, но попробуйте представить, что получится, если первоначальный блочный элемент будет назван c-drop-down-menu.


Я считаю, что двойное нижнее подчеркивание должно появляться только однажды, в имени селектора. БЭМ расшифровывается как Блок_Элемент--Модификатор, а не Блок__Элемент__Элемент--Модификатор. Также избегайте многоуровневого наименования элементов. Если у вас получаются уровни пра-пра-пра-правнуков, то, вероятно, вам стоит пересмотреть структуру компонентов.


Именование БЭМ не привязано строго к DOM, так что не имеет значения на скольких уровнях находится вложенный элемент. Соглашение об именах в первую очередь помогает вам увидеть отношения с блоком на верхнем уровне — в нашем случае, c-card.


Как бы я рассматривал один и тот же карточный компонент:


<div class="c-card">
    <div class="c-card__header">
        <h2 class="c-card__title">Title text here</h2>
    </div>

    <div class="c-card__body">

        <img class="c-card__img" src="http://some-img.png" alt="description">
        <p class="c-card__text">Lorem ipsum dolor sit amet, consectetur</p>
        <p class="c-card__text">Adipiscing elit.
            <a href="/somelink.html" class="c-card__link">Pellentesque amet</a>
        </p>

    </div>
</div>

Это значит, что все вложенные элементы будут затронуты только карточным блоком. Также, мы бы могли переместить текст и изображения в c-card__header или даже внести новый элемент c-card__footer при этом не ломая семантической структуры.


2. "Какие наименования использовать?"


К этому времени вы возможно заметили использование c- в моих примерах. Этот префикс расшифровывается как "компонент (component)" и лежит в основе всех имен моих БЭМ классов. Идея позаимствована из техники наименований Гарри Роберта, которая улучшает читаемость кода.


Система, которую я принял, и множество префиксов, которые будут встречаться в примерах в этой статье:


Тип Префикс Примеры Описание
Компонент (Component) c- c-card, c-checklist Формируют основу приложения и содержат всю косметику для отдельных компонентов.
Модуль макета (Layout module) l- l-grid, l-container У этих модулей нет никакой косметики, используются только для позиционирования компонентов c- и для построения макета приложения.
Помощники (Helpers) h- h-show, h-hide Эти классы имеют одну функцию, часто используют !important для повышения их специфичности. (В основном используются для позиционирования или видимости.)
Состояния (States) is-, has- is-visible, has-loaded Показывают различные состояния, которые могут быть у компонентов c-. Более детальное описание можно найти ниже в 6 проблеме.
JavaScript хуки js- js-tab-switcher Они указывают на то, что поведение JavaScript привязано к компоненту. Стили с ними не ассоциируются; используются только для облегчения манипуляций со скриптами.

Я обнаружил, что подобные имена невероятно улучшили читаемость моего кода. Даже если у меня не получится подсадить вас на БЭМ, то этот прием определенно стоит запомнить.


Вы бы могли также взять и другие префиксы, такие как qa- для тестов на качество (quality-assurance), ss- для различных хуков на стороне сервера (server-side) и т.д. Но список выше — уже хорошее начало, и ввести новые имена вы можете после того, как освоитесь с этой техникой.


Вы увидите хороший пример использования этого стиля наименований в следующей проблеме.


3. "Как мне называть врапперы (wrappers)?"


Некоторые компоненты требуют родительского враппера (или контейнера), который задает макет дочерним элементам. В этих случаях, я всегда пытаюсь абстрагировать макет в модуль макета, такой как l-grid и добавлять каждый компонент как содержимое l-grid__item.


В нашем карточном примере, если бы мы хотели расположить список из четырех c-cardов, я бы использовал следующую разметку:


<ul class="l-grid">
    <li class="l-grid__item">
        <div class="c-card">
            <div class="c-card__header">
                […]
            </div>
            <div class="c-card__body">
                […]
            </div>
        </div>
    </li>
    <li class="l-grid__item">
        <div class="c-card">
            <div class="c-card__header">
                […]
            </div>
            <div class="c-card__body">
                […]
            </div>
        </div>
    </li>
    <li class="l-grid__item">
        <div class="c-card">
            <div class="c-card__header">
                […]
            </div>
            <div class="c-card__body">
                […]
            </div>
        </div>
    </li>
    <li class="l-grid__item">
        <div class="c-card">
            <div class="c-card__header">
                […]
            </div>
            <div class="c-card__body">
                […]
            </div>
        </div>
    </li>
</ul>

Теперь у вас должно быть твердое представление о том, как расположение модулей и имена компонентов должны сочетаться вместе.


Не бойтесь использовать немного более продвинутой разметки, дабы потом у вас не было головной боли. Никому не станет лучше от того, что вы сократите пару <div> тегов!


В некоторых ситуациях, это не возможно. Если, например, ваша сетка не собирается вас слушаться, или вы просто хотите какое-нибудь имя со смыслом для вашего родительского элемента, то что вам делать? Я, как правило, выбираю слово container или list, в зависимости от ситуации. Применяя это к нашему карточному примеру, я бы мог использовать <div class="l-cards-container">[…]</div> или <ul class="l-cards-list">[…]</ul>, в зависимости от используемого случая. Главное — соответствовать вашему соглашению об именовании.


4. "Кросскомпонентные… Компоненты?"


Еще одна часто встречающаяся проблема — компоненты, чьи стили или позиционирование затрагиваются родительским контейнером. Различные решения этой проблемы детально описаны у Simurai. Я вам расскажу только о том подходе, который я считаю самым эффективным.


Вкратце, давайте предположим, что мы хотим добавить c-button в наш card__body из предыдущего примера. Эта кнопка уже является его компонентом и оформлена так:


<button class="c-button c-button--primary">Click me!</button>

Если у вас нет никаких различий между обычным компонентом кнопки, то тогда проблем нет, мы просто переместим ее вот так:


<div class="c-card">
    <div class="c-card__header">
        <h2 class="c-card__title">Title text here</h3>
    </div>

    <div class="c-card__body">

        <img class="c-card__img" src="http://some-img.png">
        <p class="c-card__text">Lorem ipsum dolor sit amet, consectetur</p>
        <p class="c-card__text">Adipiscing elit. Pellentesque.</p>

        <!-- Наш вложенный компонент кнопки -->
        <button class="c-button c-button--primary">Click me!</button>

    </div>
</div>

Однако, что происходит, когда присутствуют небольшие различия в стилях — например, мы хотим ее немного уменьшить, закруглить углы, но только тогда, когда она является частью компонента c-card?


Кросскомпонентный класс показался мне наиболее надежным решением:


<div class="c-card">
    <div class="c-card__header">
        <h2 class="c-card__title">Title text here</h3>
    </div>

    <div class="c-card__body">

        <img class="c-card__img" src="http://some-img.png">
        <p class="c-card__text">Lorem ipsum dolor sit amet, consectetur</p>
        <p class="c-card__text">Adipiscing elit. Pellentesque.</p>

        <!-- Мой *старый* кросскомпонентный подход -->        <button class="c-button c-card__c-button">Click me!</button>

    </div>
</div>

Это то, что известно на сайте БЭМ, как "микс (mix)". Однако, я изменил свое отношение к этому подходу, следуя некоторым замечательным комментариям от Esteban Lussich.


В примере выше, класс c-card__c-button пытается изменить одно или несколько существующих свойств c-button, но зависит от источника вызова (или даже от специфики), для успешного применения. Класс c-card__c-button будет работать только в том случае, если он определен после блока c-button, который может быстро выйти из под контроля, если вы будете строить больше таких кросскомпонентов. (Полагаться на !important, конечно, можно, но я бы не стал!).


Оформление по настоящему модульного UI элемента должно быть полностью независимо от родительского контейнера — оно должно выглядеть одинаково независимо от того, где вы его расположите. Добавление класса из другого компонента для задания стиля, как это делает способ с "миксами", нарушает открытые/закрытые принципы компонентно-ориентированного дизайна — т.е., не должно быть никакой зависимости от другого модуля для эстетики.


Лучше всего, это использовать модификатор для этих небольших косметических различий, потому что вы легко сможете использовать их снова по мере роста вашего проекта.


<button class="c-button c-button--rounded c-button--small">Click me!</button>

Даже если вы не используете эти дополнительные классы снова, по крайней мере, вы не будете привязаны к родительскому контейнеру или к исходному порядку для применения изменений.


Другой способ — пойти к вашему дизайнеру и сказать ему, что кнопки должны быть совместимы с остальными, и избегать этой проблемы вообще.


5. "Модификатор или новый компонент?"


Одна из самых больших проблем — решить, где компонент заканчивается, и начинается новый. В примере с c-card, вы бы могли создать другой компонент, названный c-panel с очень похожими атрибутами стилей, но с очень заметными различиями.


Но что определяет необходимость использования двух компонентов, c-panel и c-card, или простого модификатора для c-card, который применяет уникальные стили.


Можно легко переполнить проект модулями и сделать все в виде компонентов. Я рекомендую начать с модификаторов, если вам покажется, что управление каким-либо компонентным CSS файлом становится слишком сложным, то, возможно, наступило время разделить некоторые модификаторы. Хороший показатель, когда вам приходится сбрасывать все у блока CSS для стилизации нового модификатора — это, как по мне, самое время для создания нового компонента.


Лучше всего, если вы работаете в команде других разработчиков или дизайнеров, узнать их мнение. Соберите их на несколько минут и обсудите с ними. Несмотря на то, что этот ответ скорее выглядит как отмазка, для больших приложений очень важно понимать, какие модули доступны и что собой представляет компонент.


6. "Как управлять состояниями?"


Это довольно распространенная проблема, особенно, когда вы стилизуете компонент в активном или открытом состоянии. Давайте скажем, что наши карточки пребывают в активном состоянии; так вот, когда мы нажимаем на них, они выделяются красивой обводкой. Что насчет названия этого класса?


У вас два варианта: или использовать хук автономного состояния, или использовать БЭМ-подобное именование модификатора на уровне компонента:


<!-- хук автономного состояния -->
<div class="c-card is-active">
    […]
</div>

<!-- модификатор БЭМ -->
<div class="c-card c-card--is-active">
    […]
</div>

Не смотря на то, что мне нравится идея использовать имя БЭМ для единообразия, преимущество автономного класса в том, что он позволяет использовать JavaScript для применения универсальных хуков состояния к любому компоненту. Когда вам приходится применять определенный класс состояния, основанный на модификаторе с помощью скрипта, это становится более проблематичным. Это, конечно, возможно, но требует написания дополнительного JavaScript кода.


Имеет смысл придерживаться стандартного набора хуков состояния. Крис Пирс собрал хороший список, который я рекомендую посмотреть.


7. "Когда лучше не добавлять новый класс к элементу"


Я могу понять людей, которые перегружены огромным числом классов, требуемых для построения сложной части пользовательского интерфейса, особенно, если они не присваивали класс каждому тегу.


Обычно я прикрепляю классы ко всему, что должно быть стилизовано, по разному, в контексте компонента. Я часто оставляю теги p без класса, если компонент не требует, чтобы они выглядели по-особенному в этом контексте.


Согласен, это будет обозначать то, что ваша разметка будет содержать множество классов. Но, в конечном счете, ваши компоненты будут способны жить независимо и смогут перемещаться без побочных эффектов.


Ввиду глобального характера CSS, присвоение класса ко всему дает нам полный контроль над рендером. Первоначальный дискомфорт стоит преимуществ полностью модульной системы.


8. "Как вкладывать компоненты?"


Допустим. что мы хотим отобразить чек-лист в нашем компоненте c-card. Пример того, как не стоит это делать:


<div class="c-card">
    <div class="c-card__header">
        <h2 class="c-card__title">Title text here</h3>
    </div>

    <div class="c-card__body">

        <p>I would like to buy:</p>

        <!-- Вложенный компонент -->
        <ul class="c-card__checklist">
            <li class="c-card__checklist__item">
                <input id="option_1" type="checkbox" name="checkbox" class="c-card__checklist__input">
                <label for="option_1" class="c-card__checklist__label">Apples</label>
            </li>
            <li class="c-card__checklist__item">
                <input id="option_2" type="checkbox" name="checkbox" class="c-card__checklist__input">
                <label for="option_2" class="c-card__checklist__label">Pears</label>
            </li>
        </ul>

    </div>
    <!-- .c-card__body -->
</div>
<!-- .c-card -->

У нас тут есть несколько проблем. Первая — двухуровневый селектор, о котором мы узнали в первом разделе. Второе — все стили, примененные к c-card__checklist__item, относятся только к этому конкретному случаю, из-за чего их нельзя использовать снова.


Я предпочитаю разбить список на модуль разметки, предметы чек-листа на их собственные компоненты, что позволяет использовать их самостоятельно в другом месте. Это позволяет вернуться нашему префиксу l- к игре:


<div class="c-card">
    <div class="c-card__header">
        <h2 class="c-card__title">Title text here</h3>
    </div>

    <div class="c-card__body"><div class="c-card__body">

        <p>I would like to buy:</p>

        <!-- Намного лучше - модуль разметки -->
        <ul class="l-list">
            <li class="l-list__item">

                <!-- Многоразовый вложенный компонент -->
                <div class="c-checkbox">
                    <input id="option_1" type="checkbox" name="checkbox" class="c-checkbox__input">
                    <label for="option_1" class="c-checkbox__label">Apples</label>
                </div>

            </li>
            <li class="l-list__item">

                <div class="c-checkbox">
                    <input id="option_2" type="checkbox" name="checkbox" class="c-checkbox__input">
                    <label for="option_2" class="c-checkbox__label">Pears</label>
                </div>

            </li>
        </ul>
        <!-- .l-list -->

    </div>
    <!-- .c-card__body -->
</div>
<!-- .c-card -->

Это спасает вас от необходимости повторять эти стили, а также значит, что мы можем использовать l-list и c-checkbox в других местах нашего приложения. Это требует немного больше разметки, но взамен мы получаем читаемость, инкапсуляцию и возможность повторного использования. Возможно, вы заметили, что это — основные темы!


9. "Не появятся ли у компонентов миллион классов?"


Некоторые считают, что множество классов на элемент — плохо, и, также, могут добавиться --модификаторы. Лично мне это не кажется проблемой, потому что это делает код более читаемым, и я точно знаю, что он должен делать.


Пример из четырех классов, с помощью которых стилизуется кнопка:


<button class="c-button c-button--primary c-button--huge  is-active">Click me!</button>

Я понимаю, что этот синтаксис немного уродлив, но он понятный.


Однако, если от этого у вас начинает болеть голова, вы можете взглянуть на технику Сергея Зароуски. Проще говоря, нам нужно использовать .className [class^="className"], [class*=" className"] в таблице стилей для имитации дополнительной функциональности. Если этот синтаксис кажется вам знакомым, то это потому, что Icomoon использует похожий способ для управления селекторами иконок.


С этой техникой, ваш код может выглядеть примерно так:


<button class="c-button--primary-huge  is-active">Click me!</button>

Я не знаю, намного ли снижение производительности при использовании class^= и class*= больше, чем при использовании индивидуальных классов, но в теории, это классная альтернатива. Для себя мне хватает варианта с несколькими классами, но мне кажется, что этот способ определенно заслуживает упоминания для тех, кто предпочитает альтернативу.


10. "Можем ли мы изменить реакцию типа компонента?"


Эту проблему поставил мне Arie Thulank, и к которой я пытался найти 100% решение.


Примером может быть выпадающее меню, которое превращается в набор вкладок в заданный момент, или закадровая боковая навигация, которая превращается в панель меню.


По сути, у одного компонента могут быть два разных состояния, продиктованных медиа-запросом.


Для этих двух примеров я склоняюсь к тому, чтобы просто сделать компонент c-navigation, так как изменение в заданный момент — это то, что он делает. Но это заставило меня задуматься, что на счет списков изображений, которые превращаются в карусель на больших экранах? Это проблемный случай для меня, и, так как он хорошо документирован и прокомментирован, я думаю, что в идеале стоит создать отдельный одноразовый компонент для этого типа интерфейса, с понятным названием (таким как c-image-list-to-carousel).


Гарри Робертс упоминал об отзывчивых префиксах, одном из способов управлять этим. Его подход предназначен больше для изменений в макете и написания стилей, а не сдвигая все компоненты, но не вижу причин, по которым мы не можем применить эту технику здесь. Таким образом, вы можете назвать классы так:


<ul class="c-image-list@small-screen c-carousel@large-screen">

В дальнейшем это будет жить в ваших медиа-запросах для соответствующих размеров экрана.
Совет: Вам придется отключать @ с помощью обратной косой черты, как тут:


.c-image-list\@small-screen {
    /* стили тут */
}

У меня не было причин для создания таких компонентов, но этот способ выглядит очень дружелюбным по отношению к разработчику. Другому человеку должно быть достаточно просто понять ваши намерения. Но я не пропагандирую такие имена, как small-screen и large-screen — они использованы только для улучшения читаемости.


Заключение


БЭМ оказался спасением для меня в моем стремлении к созданию модульных приложений компонентным способом. Я использую его уже почти 3 года, и проблемы, перечисленные выше, были камнями преткновения на моем пути. Я надеюсь, что эта статья покажется вам полезной, и, если вы еще не пользуетесь БЭМ, я очень сильно рекомендую вам начать это делать.


Обо всех ошибках (грамматических, лексических и т.д.) пишите в комментарии, буду рад их исправить. Спасибо за внимание

Поделиться с друзьями
-->

Комментарии (22)


  1. mdss
    13.07.2016 09:07
    +4

    Я считаю, что двойное нижнее подчеркивание должно появляться только однажды, в имени селектора.

    Это очень важное правило, о чем написано в методологии. Т.е. делать элемент элемента вообще нельзя, а не просто не рекомендуется.


  1. TwistedAndy
    13.07.2016 10:31
    +4

    Если для повышения специфичности приходится использовать !important, то с кодом по определению что-то не так. По-хорошему за такое нужно давать по пальцам. Поддерживать код с кучей !important крайне сложно.

    По поводу методологии БЭМ как таковой. Да, действительно, она позволяет решать проблемы с модульностью и переносимостью кода на больших и средних проектах.

    В то же время она обладает и недостатками. Самый основной из них — это многочисленные классы с длинными названиями. Хорошо, если у класса сравнительно короткое имя c-card, как в примерах в статье, но если это будет какой-то c-front-page-slider или c-contact-information, то список классов у элементов может банально перестать вмещаться в окне редактора (80-120 символов чаще всего). Работать с таким HTML-кодом крайне проблематично, да и выглядит он не очень красиво.


    1. drakmail
      13.07.2016 10:44

      Чем мне нравится SMACSS – если его правильно использовать, то он сохраняет все плюсы BEM, но при этом позволяет обходиться без очень длинных классов


      1. Radiocity
        13.07.2016 12:38
        +1

        Да, но когда state классов в SMACSS становится слишком много, описание стилей увеличивается на порядок по сравнению с БЭМ. Последний позволяет хранить любые правила без дополнительной вложенности. Он как внедорожник — не слишком комфортный, но преодолеет бездорожье при правильном применении.


    1. kirillbykov2
      13.07.2016 10:55
      +1

      В 9 пункте описано довольно неплохое решение для этой проблемы, можно использовать class^= и class*= и тогда это:


      <div class="container container--shopping-cart">
        <div class="container__content container__content--special-offer">
          <h2 class="container__title">Title</h2>
        </div>
      </div>

      можно превратить в это:


      <div class="container--shopping-cart">
        <div class="container__content--special-offer">
          <h2 class="container__title">Title</h2>
        </div>
      </div>

      используя вот это:


      .container, [class^="container--"], [class*=" container--"] {
        border: 1px solid grey;
        padding: 10px;
        font-family: Arial, sans-serif;
      }
      .container--shopping-cart {
        border: 4px solid orange;
        background-color: #eee;
      }


      1. drupa
        13.07.2016 11:51

        Вот как раз так по методологии БЭМ писать и не надо.


        1. kirillbykov2
          13.07.2016 12:35

          Я согласен, но, как способ избежать слишком большого числа классов, может пригодиться


          1. drupa
            13.07.2016 12:40
            +1

            Не как просто писать много классов. В первом варианте хотя-бы явно понятно где начинается блок, какой у блока модификатор. Где начинается элемент блока. Во втором варианте что вы написали это не явно, и не понятно.


            1. zorro1211
              13.07.2016 15:31

              Мне кажется это дело привычки, можно без труда понять какой модификатор, где элемент — подход к именованию классов практически не изменился. Уверен что на практике найти блок будет совсем легко.


      1. alexey-m-ukolov
        13.07.2016 12:10
        +2

        Стоит это добавить прямо в статью, потому что я, когда читал этот пункт, вообще не понял о чём речь и пришёл в комментарии об этом написать.


        1. kirillbykov2
          13.07.2016 12:38

          Немного изменил этот пункт.
          Добавлять в статью не вижу смысла, так как цель пункта только в том, чтобы сказать о том, что такой способ существует, а для более подробного чтения есть ссылка на статью об этом способе.


      1. TwistedAndy
        13.07.2016 14:10
        +1

        Это решение на самом деле чревато негативными побочными эффектами, которые сводят на нет самое основное преимущество БЭМ — модульность и хорошую изолированность стилей для разных блоков.

        Представим, что вы написали блок стилей с селектором в виде .container, [class^=«container--»], [class*=" container--"]. По мере роста проекта какому-то из разработчиков захотелось использовать в селекторе конструкцию container--. И вот тут его ожидает сюрприз: к его стилям стали подмешиваться совершенно чужие правила. Если таких селекторов с классами будет много, то разработка и поддержка такого проекта быстро превратиться в пробежку по минному полю.


        1. zorro1211
          13.07.2016 15:24

          Привет, давай попробуем разобраться. Если в проекте все таки используется БЭМ повсемесно то .container, [class^=«container--»], [class*=" container--"] применится только к классам которые начинаются на container--, независимо от расположения этого класса в атрибуте class. В БЭМ классы с модификаторами вседа используются совместно с классами блока или элемента, и никогда сами по себе, а значит container-- должен применяться совместно с .container и раз так, то поведение .container, [class^=«container--»], [class*=" container--"] как раз играет на руку тем кто использует БЭМ. Кроме того эту конструкцию нужно использовать только тогда когда есть необходимость в модификаторах, так что лишний раз можно не писать.


          1. antdigo
            13.07.2016 19:16
            +1

            Просто вы рассматриваете БЭМ только как набор правил по именованию селекторов, поэтому на этом уровне фактически нет разницы в применении стилей к .container*, [class^=«container--»] или [class*=" container--"]. Но не нужно забывать, что БЭМ — это методология, и у нее есть инструменты для работы и на уровне JS (фреймворк i-bem.js). В нем, как раз, js-поведение DOM-элементу назначается на основе названия блока. Именно поэтому отдельный .container и необходим.


            1. zorro1211
              13.07.2016 20:04

              Все верно. Спасибо за комментарий.


    1. dima117
      13.07.2016 13:48
      +1

      Еще хороший вариант — формировать интерфейс в bemhtml (т.е. не работать напрямую с html) — это устраняет все недостатки, связанные с количеством и названиями классов.


  1. fetis26
    13.07.2016 18:29
    +1

    Очень хорошая статья. И во многих подходах я пришел примерно к таким же решениям. Пара замечаний


    №1
    Тоже немного колбасит, когда классы не совсем следуют структуре вложенности. Но я для себя решил что структура элементов должна быть плоской. Иногда по коду получается что block__elem2 вложен в block_elem1, но с этим можно жить и достаточно просто разобраться, потому что компоненты изолированы


    №2
    Не использую c-, l- префиксы. На мой взгляд это избыточно и немного мусорно. И последнее время все меньше использую хелперы, предпочитаю вставлять миксины.


    №3
    Тут речь почему-то идет не про настоящие врапперы, а просто про вложенность компонентов. Иногда реально нужна обертка, потому что дивов не хватает застайлить. Тогда название блока выношу наверх, а дальше использую &__inner.


    №4
    Тоже предпочитаю модификаторы, но имхо не стоит бояться называть по используемому месту, например, так вполне лучше .button--card-view, нежели абстрактные .button--small .button--round.


    №6
    Я больше склоняюсь к .block--active, по крайней мере обычно начинаю с него. Если действительно предстоит много манипулировать состояниями через JS в подобных компонентах, то .block.is-active будет проще.


    №10
    На мой взгляд не стоит бояться и делать различные ветки кода в зависимости от медиа-выражений. Т.е. у меня это выглядит примерно так


    .block {
      @respond-to('mobile') {
       // стили мообилок
      }
    
      @respond-to('tablet-desktop') {
       // стили планшета и десктопа
      }
    }

    respond-to это миксин, который генерит медиавыражения в зависимости от ключевого слова.


    1. fetis26
      13.07.2016 18:33

      З.Ы.


      №9
      На мой взгляд это надуманная проблема. Там не так уж много классов выходит. Гляньте как Ангуляр, к примеру, инпуты обвешивает, вот это уже реально перебор.


  1. vintage
    13.07.2016 21:46
    +1

    1. "Не появятся ли у компонентов миллион классов?"


    <button class="c-button--primary-huge  is-active">Click me!</button>

    Слишком хрупко. Не понимаю, чего все верстальщики так за классы держатся. С атрибутами тут получается куда гибче, проще и наглядней:


    <button my_button="primary huge" my_button_active="true">Click me!</button>

    [my_button] { border: 1px solid steelblue; background: none; padding: .25rem }
    [my_button*="huge"] { zoom: 2 }
    [my_button*="primary"] { background: steelblue; color: snow }
    [my_button_active="false"] { opacity: .5 }

    my_button — блок
    primary, huge — статические модификаторы
    my_button_active — динамический модификатор


    1. anotherpit
      14.07.2016 02:02
      +1

      Вот тут ребята то же самое придумали: http://amcss.github.io/


  1. leonik
    14.07.2016 12:13
    +1

    Вот скажите, почему все используют в качестве разделителя для модификатора "--". Ведь в официальном соглашении по именованию используется "_" ru.bem.info/methodology/naming-convention/#Имя-модификатора. Я понимаю, что это альтернативная схема именования, но зачем отходить от официальной Яндекса? Чтобы было больше путаницы?


    1. kirillbykov2
      14.07.2016 12:23

      Этот стиль именования очень сильно популярен в англоязычной среде, думаю, что его популярность связана с тем, что автор этого стиля, Гарри Робертс, одним из первых заговорил о БЭМ в англоязычной среде. Советую прочитать вам оригинальную статью, там он многое объясняет.