woocommerce/plugins/woocommerce-blocks/docs/contributors/coding-guidelines.md

7.2 KiB
Raw Blame History

Coding Guidelines

This living document serves to prescribe coding guidelines specific to the WooCommerce Blocks project. Base coding guidelines follow the WordPress Coding Standards and Gutenberg coding standards.

The following sections outline additional patterns and conventions used in the Blocks project.

CSS

CSS Class Names

To avoid class name collisions, class names must adhere to the following guidelines, based on the BEM methodology and Gutenberg coding standards.

Prefixing

As a WordPress plugin, Blocks has to play nicely with other plugins and themes, and WordPress itself. To minimize potential conflicts, all classes should be prefixed with .wc-block-.

Naming

All class names assigned to an element must be prefixed with the following, each joined by a dash (-):

  • The wc-block plugin prefix.
  • The name of the sub-package (where applicable, e.g. if there was a distributed sub-package called components living within the blocks plugin, the prefix would be wc-block-components-).
  • The name of the directory in which the component resides.

Any descendant of the component's root element must append a dash-delimited descriptor, separated from the base by two consecutive underscores __.

Example, `assets/base/components/checkbox-list` uses the class name: `wc-block-checkbox-list`.

A root element (or Block in BEM notation) is a standalone entity that is meaningful on its own. Whilst they can be nested and interact with each other, semantically they remain equal; there is no precedence or hierarchy.

Example: `wc-block-package-directory`

A child element (or Element in BEM notation) has no standalone meaning and is semantically tied to its block.

Example: `wc-block-package-directory__descriptor-foo-bar`

Finally, A modifier is a flag on an element which can be used to change appearance, behavior or state.

Example: `wc-block-package-directory__descriptor-foo-bar--state`

The root element is considered to be the highest ancestor element returned by the default export in the index.js. Notably, if your folder contains multiple files, each with their own default exported component, only the element rendered by that of index.js can be considered the root. All others should be treated as descendants.

Naming is not strictly tied to the DOM so it doesnt matter how many nested levels deep a descendant element is. The naming convention is there to help you identify relationships with the root element.

Nesting Example:

  • wc-block-dropdown-selector (Root Element/BEM Block)
  • ├── wc-block-dropdown-selector__input (Child Element/BEM Element)
  • ├── wc-block-dropdown-selector__input--hidden (Modifier)
  • └── wc-block-dropdown-selector__placeholder (Child Element/BEM Element)

RTL Styles

Blocks uses the webpack-rtl-plugin package to generate styles for Right-to-Left languages. These are generated automatically.

To make adjustments to the generated RTL styles, for example, excluding certain rules from the RTL stylesheets, you should use the control directives here.

For example, you can exclude individual lines:

.code {
	/*rtl:ignore*/
	direction: ltr;
	/*rtl:ignore*/
	text-align: left;
}

Or exclude blocks of CSS:

.code {
	/*rtl:begin:ignore*/
	direction: ltr;
	text-align: left;
	/*rtl:end:ignore*/
}

SCSS File Naming Conventions for Blocks

The build process will split SCSS from within the blocks library directory into two separate CSS files when Webpack runs.

Styles placed in a style.scss file will be built into build/style.css, to load on the front end theme as well as in the editor. If you need additional styles specific to the block's display in the editor, add them to an editor.scss.

Accessible font sizes

We want our font sizes to be declared with rem or em units instead of hardcoded px. That's important for accessibility because it allows users to make the text bigger and easier to read.

We have a mixin named font-size() that given a number of the font size in px, it converts it to rem. It accepts a second parameter which is the line height, it will be divided by the font-size and the result will be the relative units for the line-height CSS property.

In parallel to that, consider whether other size/distance units in your CSS need to be rem/em instead of px. In general, rem/em should be preferred if it doesn't break the design with big font sizes. There is another mixin named rem() that helps converting px units to rem (given a px size and optionally a base size).

CSS specificity wars with 3rd party themes

We want our blocks to look good with as many themes as possible out of the box. Sometimes our styles will conflict with theme styles that have higher specificity. In these cases it may be tempting to increase the specificity of selectors, but increasing them too much makes it harder for other themes to style our blocks.

The following guidelines should help you decide when to increase specificity, if at all. They are not hard rules so feel free to apply your best judgement on a case-by-case basis.

Imagine we are styling the radio control input but our styles are conflicting with some themes. For example, two themes that have the following styles:

Theme A:

input[type="radio"] { // specificity 0, 1, 1
	background: red;
}

Theme B:

input[type="radio"]:checked { // specificity 0, 2, 1
	background: blue;
}

And these are the styles of the block:

.wc-block-radio-control__input { // specificity 0, 1, 0
	background: #fff;
}

As you can see, the styles coming from the themes have higher specificity, so our styles would be overriden. In order to solve this:

  1. Never use !important rules in CSS to engage in a specificity war with a theme.
  2. Never use ID selectors.
  3. Try wrapping the entire component/block CSS with the root class name of that component: For example:
.wc-block-radio-control {
	.wc-block-radio-control__input { // specificity 0, 2, 0, we win theme A!
		background: #fff;
	}
}
  1. Try adding an extra css class (or tag selector) to increase specificity. When doing so, add a comment explaining it.
.wc-block-radio-control {
	// Extra class for specificity.
	.wc-block-radio-control__option .wc-block-radio-control__input { // specificity 0, 3, 0, we win theme B!
		background: #fff;
	}
}
  1. If these steps weren't enough, consider not increasing specificity at all. If it's just a minor visual issue, consider ignoring it and assume the theme will update its conflicting styles at some point. If it's completely breaking the block or component in that theme, consider sending feedback to theme authors so they can fix it on their side.

Notice in the worst case scenario we would have increased selector specificity by 2 classes (0, 2, 0). That shouldn't make it too difficult for other themes to write styles on top of ours.