A mostly reasonable approach to CSS
Selectors that contain elements tightly coupled the CSS to specific markup. It is not a safe assumption that the semantics of the content will never change so authors should prefer classes which exist independent of markup and create more flexible CSS.
Source: SMACSS on modules
A location based selector is a selector that changes a modules appearance based on its location (content area, side bar, header etc). Where a module has different appearances it is better to use a module subclass. If the appearance and/or content is very different it would be better to use a different module
Source: OOCSS, slide 20
It is never safe to assume there will only ever be one of something on a page so do not use ID's for CSS. Id's are much better used as javascript hooks so use them for this instead.
Source: SMACSS on layout
Inline styles added by javascript are harder to update and maintain, prefer to add classes using javascript. CSS3 transitions can then handle any animations and if CSS3 transitions are not supported the state will still be updated.
Source: SMACSS on state
Long complex selector chains spawn specificity wars and are slower for the browser to evaluate. Its unlikely you need to use a selector more than 4 levels deep. If it is, consider adding additional classes to the markup to enable a shorter selector length.
Utilise a preprocessors features such as mixins, variables, nested rules and functions to make code easier to maintain but always double check compiled stylesheets to avoid code bloat. Preprocessors, when used inefficiently, can cause code bloat.
Source: SMACSS on preprocessors (members only)
HTTP Requests are expensive and slow the rendering of a page, so minimise them. Ideally a HTML page should contain just one CSS file with extra CSS being loaded as the page changes.
There are big advantages for vector icons (SVG or Icon Fonts): resizable up and down without losing quality, extra sharp on retina displays, and small file size among them.
The value of 0
works without specifying units in all situations where numbers with units or percentages are allowed. There is no difference between 0px
, 0em
, 0%
, or any other zero-value. The units aren't important because the value is still zero. CSS allows you to omit the units for zero values and still remain valid CSS.
Source: CSSLint
Writing selectors such as li.active
are unnecessary unless the element name causes the class to behave differently. In most cases, it's safe to remove the element name from the selector, both reducing the size of the CSS as well as improving the selector performance (doesn't have to match the element anymore). Removing the element name also loosens the coupling between your CSS and your HTML, allowing you to change the element on which the class is used without also needing to update the CSS.
Source: CSSLint
The !important
annotation is used to artificially increase the specificity of a given property value in a rule. This is usually an indication that the specificity of the entire CSS has gotten a bit out of control and needs to be refactored. The more frequently !important
is found in CSS, the more likely it is that developers are having trouble styling parts of a page effectively.
Source: CSSLint
Minifying code can save up to 60% on a files size. Use a tool like YUI Compressor or Gulp CSS Min together with build tools such as Gulp or Grunt to manage this and it's easy-peasy.
Source: Google Page Speed on Minifying CSS
Using a CSS Linter will make obvious whenever a rule is not matching the coding standard and best practices. It will make developers know something is wrong even before looking at it in the browser. When using Sublime Text a great solution is to use SublimeLinter, there are other options for other IDEs and it can also be used online via http://csslint.net/
CSS Architecture is more difficult to do well than it is difficult to understand CSS Properties. Reading long lines of CSS code makes it difficult to understand the properties of the rule. This, together with the correct ordering of the CSS properties make it really easy to understand a rule as first sight.
Source: SMACSS on formatting
CSS properties do not use underscores or camel case, they use dashes. Do the same with classes.
Mobile First technique increase the performance of the site. Usually desktop and large screen devices have more hardware powerful and better internet connections than mobile ones. So instead of declaring large screen rules first only to override them for smaller screens, is better define the mobile rules (which are usually less) and then add more content for larger screens.
.element {
display: none;
}
@media screen and (min-width: 480px) {
.element {
background: url(small-image.png);
display: inline;
width: 100%;
}
}
@media screen and (min-width: 768px) {
.element {
background: url(large-image.png);
width: 50%;
}
}
Define a global font-size
in pixels:
body {
font-size: 16px;
}
... and then use rem
for the rest of declarations:
h1 {
font-size: 1.5rem; /* 24px / 16px = 1.5rem */
}
h2 {
font-size: 1.25rem; /* 20px / 16px = 1.25rem */
}
... eventually you'll want to use em
for dynamically changing text size, with a definition in rem
for the container:
.list-container {
font-size: 1.25rem;
...
}
@media screen and (min-width: 480px) {
.list-container {
font-size: 1.5rem;
}
}
@media screen and (min-width: 768px) {
.list-container {
font-size: 1.75rem;
}
}
.list-container .list-title {
font-size: 2em;
...
}
.list-container .list-subtitle {
font-size: 1.5em;
...
}
Declare media queries in the same component definition instead of a general media query. This practice improves the maintainability
(Sass code) Bad example:
// modules/_lists.scss
.list-title {
...
}
.list-image {
...
}
@media screen and (min-width: 468px) {
.list-title {
...
}
.list-image {
...
}
}
@media screen and (min-width: 768px) {
.list-title {
...
}
.list-image {
...
}
}
(Sass code) Good example:
// modules/_lists.scss
.list-title {
...
@media screen and (min-width: 468px) {
...
}
@media screen and (min-width: 768px) {
...
}
}
.list-image {
...
@media screen and (min-width: 468px) {
...
}
@media screen and (min-width: 768px) {
...
}
}
Prefer dash-cased variable names (e.g. $my-variable
) over camelCased or snake_cased variable names. It is acceptable to prefix variable names that are intended to be used only within the same file with an underscore (e.g. $_my-variable
).
Use variables instead of hardcoded values whenever possible. You might also like to use a configuration file for this:
// base/_config.scss
// Fonts
$font-family-primary : "Lato", sans-serif, Helvetica, Arial;
$font-family-secondary : "Open Sans", sans-serif, Helvetica, Arial;
$font-family-icon : "icomoon";
$font-size-primary : 16px;
// Color Pallete
$brand-color : #70CFD9;
$brand-color-2 : #FF3643;
$brand-color-3 : #2B3444;
$bg-color-primary : #F5EE9B;
// Transitions
$transition-primary : all 0.3s ease;
$transition-secondary : all 0.4s cubic-bezier(0.2, 1, 0.3, 1);
// Responsive
$phone-small : 320px;
$phone-medium : 480px;
$phone-large : 512px;
$phone-xlarge : 640px;
$tablet : 768px;
$desktop-small : 1024px;
$desktop-medium : 1280px;
$desktop-large : 1440px;
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 a media queries mixin within the pre-setted variables of breakpoints to improves the maintainability
.list-item {
...
@include breakpoint($phone-medium) {
...
}
@include breakpoint($tablet) {
...
}
@include breakpoint($desktop-medium) {
...
}
}
Avoid nesting more than 4 levels to prevent large file sizes (in generated css files)
// Bad example
.wrapper {
...
.main-content {
...
.feature-list {
...
.feature-list-item {
...
.default-button {
...
}
}
}
}
}
// Good example
.wrapper {
...
}
.main-content {
...
}
.feature-list {
...
}
.feature-list-item {
...
.default-button {
...
}
}
- 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
- Use indentation of 4 spaces to improve readability.
- Put spaces after
:
in property declarations. - Put spaces before
{
in rule declarations. - Use hex color codes #000 unless using rgba.
Syntax example:
.styleguide-format {
background: rgba(0, 0, 0, 0.5);
border: 1px solid #0F0;
color: #000;
}
/* Bad */
.my-class, .our-class, .your-class {
border: 1px solid #0F0;
}
/* Good */
.my-class,
.our-class,
.your-class {
border: 1px solid #0F0;
}
/* Bad */
.my-class { border: 1px solid #0F0; color: #F00; margin: 20px; }
/* Good */
.my-class {
border: 1px solid #0F0;
color: #F00;
margin: 20px;
}
/* Bad */
.demoImage,
.demo_image {
...
}
/* Good */
.demo-image {
...
}
/* Bad */
.container {
width: 100%;
color: #000;
background: #FFF;
margin: 20px;
}
/* Good */
.container {
background: #FFF;
color: #000;
margin: 20px;
width: 100%;
}
/* Bad */
.featured-text {
background-color: #CCC;
background-image: url("img/bg.jpg");
background-position: top right;
background-repeat: no-repeat;
margin-bottom: 10px;
margin-left: 5px;
margin-right: 15px;
margin-top: 20px;
}
/* Good */
.featured-text {
background: #CCC url("img/bg.jpg") no-repeat top right;
margin: 20px 15px 10px 5px;
}
/* Not recommended */
.list-item {
float: left;
...
}
/* Better */
.list-item {
display: inline-block;
...
}
/* Bad */
.heart-icon {
background: url(sprite.png) -16px 0 no-repeat;
}
.task-icon {
background: url(sprite.png) -32px 0 no-repeat;
}
/* Good */
.icons {
background: url(sprite.png) no-repeat;
}
.heart-icon {
background-position: -16px 0;
}
.task-icon {
background-position: -32px 0;
}
/* Bad */
.my-class {
border: 1px solid #0F0;
color: #F00
}
/* Good */
.my-class {
border: 1px solid #0F0;
color: #F00;
}
Splitting CSS into multiple files in a constant way makes it easier to find the CSS you are looking for.
It sucks to have to search through 1000+ lines of code, so avoid it. This rule may mean having to split the CSS module file into multiple files. When doing this make sure to only serve one file to the user.