Skip to content

Latest commit

 

History

History
481 lines (317 loc) · 12.8 KB

README.md

File metadata and controls

481 lines (317 loc) · 12.8 KB

Sass / Css Styleguide

Table of Contents

General

Sass vs Scss vs Css On this project we use indented Sass syntax that provides a more concise way of writing CSS. It uses indentation rather than brackets to indicate nesting of selectors, and newlines rather than semicolons to separate properties.

So we use this:

.block-name

  &:after
    // ...

  &_hover
    // ...
  
  &__element-name
    // ...

...instead of this:

.block-name {

  &:after {
    /* ... */
  }

  &_hover{
    /* ... */
  }
  
  &__element-name{
    /* ... */
  }
}

Some vendor libraries were imported in the Scss (Sassy CSS) format. Either syntax can import files written in the other.

Formatting

Sass and CSS:

  • Use soft tabs (2 spaces) for indentation
  • Do not use ID selectors
  • When using multiple selectors in a rule declaration, give each selector its own line.
  • In properties, put a space after, but not before, the : character.
  • Put blank lines between rule declarations

CSS:

  • Put a space before the opening brace { in rule declarations
  • Put closing braces } of rule declarations on a new line

Comments

  • Prefer line comments (// in Sass-land) to block comments.
  • Prefer comments on their own line. Avoid end-of-line comments.
  • Write detailed comments for code that isn't self-documenting:
    • Uses of z-index
    • Compatibility or browser-specific hacks
  • In markup write block start and block end comments for easy navigation
<!-- Block-name start -->
<div class="block-name">
    <div class="block-name__element-name"></div>
</div>
<!-- Block-name end -->

BEM naming

We use BEM for these reasons:

  • It helps create clear, strict relationships between CSS and HTML
  • It helps us create reusable, composable components
  • It allows for less nesting and lower specificity
  • It helps in building scalable stylesheets

BEM, or “Block-Element-Modifier”, is a naming convention for classes in HTML and CSS. It was originally developed by Yandex with large codebases and scalability in mind, and can serve as a solid set of guidelines.

  • Names of BEM entities are written using numbers and lowercase Latin characters.

  • Individual words within names are separated by a hyphen (-).

  • Element name is delimited by a double underscore (__) from a block name.

  • Modifier is separated from its name by a single underscore (_).

  • Information about the names of blocks, elements, and modifiers is stored using CSS classes.

  • Avoid nested selectors, like .block-name h2 {} or .block-name__elem-name p {}. Nested selectors increase code coupling and make reuse impossible. This is in contradiction to the BEM principles. In this case use .block-name__title {} element instead.

    Exception: nesting is appropriate if you need to change elements relative to the state of the block or the theme set: .nav_theme_islands .nav__item {}

  • Avoid combined tags and class names. Combining a tag and a class in a selector makes the CSS rules more specific.

See FAQ for more usage examples.

Block

block-name - defines a namespace for elements and modifiers.

Example

menu

lang-switcher

HTML

<div class="menu">...</div>

Sass

.menu 
  color: red

Element

An element name is delimited by a double underscore (__) from a block name. The full name of an element is created using this scheme:

block-name__elem-name

If a block has several identical elements, such as in the case of menu items, all of them will have the same name menu__item.

Important! Using elements within elements is not recommended by the BEM methodology.

Example

menu__item

lang-switcher__lang-icon

HTML

<div class="menu"> ... <span class="menu__item"></span> </div>

Sass

.menu__item
  color: red

Modifier

On the current project most of modifiers are Key-value type modifier, like block-name_mod-name_mod-val or block-name__elem-name_mod-name_mod-val

Important! Modifier can be added only to its own block or element. Mixes like this block-name other-block_mod-name_mod-val are not allowed. In this case use block-name_mod-name or wrap instead.

Block modifier The value of a modifier is separated from its name by a single underscore (_). The full name is created using the scheme: block-name_mod-name_mod-val.

Example

menu_theme_morning-forest

HTML

<div class="menu menu_state_hidden">...</div>
<div class="menu menu_theme_morning-forest">...</div>

Incorrect notation

<div class="menu_state_hidden">...</div>

Here the notation is missing the block that is affected by the modifier.

Sass

.menu_state_hidden
  display: none

.menu_theme_morning-forest
  color: green

Element modifier. The value of a modifier is separated from its name by a single underscore (_). The full name is created using the scheme: block-name__elem-name_mod-name_mod-val.

Example

menu__item_type_radio

HTML

<div class="menu"> ... <span class="menu__item menu__item_state_visible menu__item_type_radio"></span> </div>

Sass

.menu__item_type_radio
  color: blue

Example of using the naming convention

The implementation of an authorization form in HTML and CSS:

HTML

<form class="form form_role_login form_theme_forest">
    <input class="form__input">
    <input class="form__submit form__submit_state_disabled">
</form>

Sass

.form
  &_theme_forest

  &_role_login

  &__input

  &__submit

    &_state_disabled

ID Selectors

While it is possible to select elements by ID in CSS, it should generally be considered an anti-pattern. ID selectors introduce an unnecessarily high level of specificity to your rule declarations, and they are not reusable.

For more on this subject, read CSS Wizardry's article on dealing with specificity.

JavaScript hooks

Avoid binding to the same class in both your CSS and JavaScript. Conflating the two often leads to, at a minimum, time wasted during refactoring when a developer must cross-reference each class they are changing, and at its worst, developers being afraid to make changes for fear of breaking functionality.

We recommend creating JavaScript-specific classes to bind to, prefixed with .js-:

<button class="button button__role_primary js-request-to-book">Request to Book</button>

Project

Resets

Some project pages use Foundation framework, so for now we can't use general reset or normalize stylesheet. For this purpose we have a few rough sass mixins that should be added to each block and some block's elements:

// Total reset of the inheritable properties
+normalize-block()

// Box model fix
+box-model-normalizer()

// Light version of the normalize-block for text elements
+text-normalizer()

On the refactoring stage these mixins will be removed in favor of reset stylesheet

Folder structure

sass/                             * sass preprocessor styles
   ├── blocks/                    * blocks library
   |   └── block-name.sass
   ├── helpers/                   * mixins, settings, base, fonts
   ├── vendor/                    * third-party code
   └── index.sass                 * index file

Related: Architecture for a Sass Project

Index file

Contain only imports of vendor code, helpers and blocks. Place your Sass code in the appropriate file.

Blocks

blocks folder contain page components. Each block should be placed into separate files. Elements should be nested within block's class name using &.

Important Ruby-sass compiler requires direct import of mixins.sass and settings.sass into each block.

Helpers

Helpers folder includes mixins.sass, settings.sass, fonts.sass and base.sass files.

misins.sass - contains common mixins, i.e clearfix trick Mixins should be used to DRY up your code, add clarity, or abstract complexity--in much the same way as well-named functions. Mixins that accept no arguments can be useful for this, but note that if you are not compressing your payload (e.g. gzip), this may contribute to unnecessary code duplication in the resulting styles.

Use shorthand to declare =mixin-name() and include +mixin-name() mixin.

Block-related mixins, like button-hover-behavior() should be placed into block file.

@extend should be avoided because it has unintuitive and potentially dangerous behavior, especially when used with nested selectors. Even extending top-level placeholder selectors can cause problems if the order of selectors ends up changing later (e.g. if they are in other files and the order the files are loaded shifts). Gzipping should handle most of the savings you would have gained by using @extend, and you can DRY up your stylesheets nicely with mixins.

settings.sass - contains common variables, like color maps, fonts etc

base.sass contains common styles, like selection styles, html and body styles, headings defaults.

fonts.sass contains local fonts imports

Vendor libs

Third-party libs, like frameworks, plugins styles etc. Important Vendor code should not be changed. Use override within blocks instead.

Block structure

@import '../helpers/settings'
@import '../helpers/mixins'

////////////////////////////////////////
// $Block name
////////////////////////////////////////

//////////////////////////////////////// Block variables

$block-var-name: ""

//////////////////////////////////////// Block mixins and functions

=block-mixin-name()
  // ...


//////////////////////////////////////// Block styles
$block: ".block-name"

#{$block}
  // ...

  +block-mixin-name()

  // Media queries
  @media screen and (max-width: 960px)
    // ...

  &::before
    // ...

  &:hover
    // ...

    &:after
      // ...

  // Block-name STATE: active state
  &_state_active
    // ...

  // Block-name MOD: modifier name
  &_key_val
    // ...

    @media screen and (max-width: 960px)
      // ...

  // Block-name MOD: modifier name STATE: active state
  &_key_val#{&}_state_active
    // ...

    @media screen and (max-width: 960px)
      // ...


  // Block element name
  &__element
    // ...

    +block-mixin-name()

    // Block element name CONTEXT: block-name STATE: active state
    #{$block}_state_active &
      // ...

    // Block element name CONTEXT: block-name MOD: modifier name
    #{$block}_key_val &
      // ...

    @media screen and (max-width: 960px)
      // ...

      #{$block}_state_active &
        // ...

      #{$block}_key_val &
        // ...

    &::before
      // ...

    &:empty
      // ...

      &:after
        // ...

    // Block element name STATE: active state
    &_state_active
      // ...

    &_key_val
      // ...

      // Block element name MOD: modifier name CONTEXT: block-name STATE: active state
      #{$block}_state_active &
        // ...

      // Block element name MOD: modifier name CONTEXT: block-name MOD: modifier name
      #{$block}_key_val &
        // ...

      @media screen and (max-width: 960px)
        // ...

        #{$block}_state_active &
          // ...

        #{$block}_key_val &
          // ...


  // Other block element name
  &__other-element
    // ...

Helpful Links