Preprocessors

A preprocessor takes the source file that you write and translates it into an output file, which is a standard CSS stylesheet. A preprocessor doesn’t add features to CSS, but it makes it easier to write.

  • SCSS uses bracket syntax and the .scss extension

Installation and setup

npm init -y                         # create new npm project
npm install --save-dev sass         # install sass and add to package.json
mkdir sass build                    # create two dirs
touch sass/index.scss               # create main scss file
vim package.json                    # create build scrip
  ...
  "scripts": {
    "sass": "sass sass/index.scss build/styles.css"
  },
  ...
npm run sass                        # run build script, [over]write build/styles.css 

When you run npm run sass, it creates a styles.css.map source map file that maps code in the .css output file to the .scss source file.

Features

Variables

Variables are one of the reasons that CSS preprocessors became popular. They use the following syntax:

$primary: value;

Sass variables do not understand the DOM, and they do not use cascade or inheritance. They are block-scoped, so be sure to declare them globally unless you only want to use them within a single ruleset.

You can redefine the same variable within the same .scss file. If you change a variable value, rules defined before the change use the initial value, while rules defined after the change use the newer value.

Inline arithmetic

You can use +, -, *, /, and % in a declaration to compute values. This feature is what inspired the calc() function in CSS:

$padding-left: 3em;

.note-body {
  padding-left: $padding-left * 2;
}

/* output.css */
.note-body {
  padding-left: 6em;
}

Nested selectors

You can nest selectors inside declaration blocks to group related code. Nesting increases specificity, so be avoid extensive nesting.

The preprocessor prepends the top-level selector to the nested selector with a space between (.parent .nested) to create a descendant selector. To omit the space and create a compound selector (.parent.nested), use an &:

.site-nav {
  display: flex;

  > li {
    margin-top: 0;

    &.is-active {
      display: block;
    }
  }
}

/* output.css */
.site-nav {
  display: flex;
}
.site-nav > li {
  margin-top: 0;
}
.site-nav > li.is-active {
  display: block;
}

You can also nest @media queries. This is helpful when changing selectors–you don’t have to change the selector in the media query, too:

html {
  font-size: 1rem;
  @media (min-width: 45em) {
    font-size: 1.25rem;
  }
}
/* output.css */
html {
  font-size: 1rem;
}
@media (min-width: 45em) {
  html {
    font-size: 1.25rem;
  }
}

Partials (@use)

Partials let you split your code into separate files for better code organization. The preprocessor will concatenate partial files together so the browser only needs to request one file.

To create a partial, start the file name with an underscore. For example, _file.scss. Then, import the partial at the beginning of your main .scss file with the @use at-rule. For more information about how you can use the @use at-rule, see the Sass docs.

Here is a simple example:

/* _button.scss */
.button {
  padding: 1em 1.25em;
  background-color: #265559;
  color: #333;
}

/* index.scss */
@use "button";

html {
  font-size: 1rem;
  @media (min-width: 45em) {
    font-size: 1.25rem;
  }
}

/* output.css */
.button {
  padding: 1em 1.25em;
  background-color: #265559;
  color: #333;
}

html {
  font-size: 1rem;
}
@media (min-width: 45em) {
  html {
    font-size: 1.25rem;
  }
}

Mixins

A mixin is a small, reusable chunk of CSS. Use this if you have some commonly repeated rules, or a font style that you need to repeat in multiple places throughout the stylesheet.

Because mixins duplicate code, they compress well with gzip, which is how you should compress all network traffic.

Define a mixin with the @mixin at-rule, and use it with an @include at-rule. The @include at-rule tells the preprocessor to generate code:

@mixin box {                                  /* mixin defition */
  border-radius: 5px;
  box-shadow: 5px 5px 10px rgb(0 0 0 /0.1);
}

.main-tile {
  @include box;                               /* apply mixin */
  color: #333;
}

/* output.css */
.main-tile {
  border-radius: 5px;
  box-shadow: 5px 5px 10px rgba(0, 0, 0, 0.1);
  color: #333;
}

Mixins can also accept parameters—like functions—and return styles. Parameters begin with a $, just like variables:

@mixin alert-variant($color, $bg-color) {
  padding: 0.3em 0.5em;
  border: 1px solid $color;
  color: $color;
  background-color: $bg-color;
}

.alert-info {
  @include alert-variant(blue, lightblue);
}
.alert-danger {
  @include alert-variant(red, pink);
}

/* output.css */
.alert-info {
  padding: 0.3em 0.5em;
  border: 1px solid blue;
  color: blue;
  background-color: lightblue;
}

.alert-danger {
  padding: 0.3em 0.5em;
  border: 1px solid red;
  color: red;
  background-color: pink;
}

To use the mixin, use the @include keyword, followed by the mixin name and arguments:


img:first-of-type { 
  @extend .image-base; 
  @include handle-img(20px 100px 10px 20px, center, left);
}

Extend

The @extend at-rule is similar to a mixin, but it doesn’t copy declarations–it groups selectors and places them at an earlier location in the stylesheet. Because @extend moves the selector to an earlier location in the stylesheet, it might affect the cascade:

%message {
  padding: 0.3em 0.5em;
  border-radius: 0.5em;
}

.message-info {
  @extend %message;
  color: blue;
  background-color: lightblue;
}

.message-danger {
  @extend %message;
  color: red;
  background-color: pink;
}

/* output.css */
.message-danger, .message-info {
  padding: 0.3em 0.5em;
  border-radius: 0.5em;
}

.message-info {
  color: blue;
  background-color: lightblue;
}

.message-danger {
  color: red;
  background-color: pink;
}

Color manipulation

The color.adjust() function lets you manipulate colors:

@use "sass:color";

.class {
  property: color.adjust($var, <property>: <val>)
}

Use $lightness, $saturation, and $hue as properties to adjust. If you want to make something darker, use a negative value for $lightness, and to desaturate a color, use a negative value with $saturation:

@use "sass:color";

$green: #63a35c;

$green-dark: color.adjust($green, $lightness: -10%);      /* darken */
$green-light: color.adjust($green, $lightness: 10%);

$green-vivid: color.adjust($green, $saturation: 20%);
$green-dull: color.adjust($green, $saturation: -20%);     /* desaturate */

$purple: color.adjust($green, $hue: 180deg);
$yellow: color.adjust($green, $hue: -70deg);

/* use directly in a property */
.class {
  background-color: color.adjust($green, $hue: -70deg);
}

Interpolation

Interpolation lets you parameterize a parameter. For example:

padding-#{$direction}: 10rem;

In the previous example, you can define a $direction variable that takes the place of the # character. This is similar to template strings in Javascript or formatted strings in Python.

Loops

You can iterate over a value to produce a slight variation. You can use the $index literally by escaping it with the #{$index} syntax. This is called interpolation:

@for $index from 2 to 5 {
  .nav-links > li:nth-child(#{$index}) {
    transition-delay: (0.1s * $index) - 0.1s;
  }
}

/* output */
.nav-links > li:nth-child(2) {
  transition-delay: 0.1s;
}

.nav-links > li:nth-child(3) {
  transition-delay: 0.2s;
}

.nav-links > li:nth-child(4) {
  transition-delay: 0.3s;
}

PostCSS

PostCSS parses a source file and outputs a processed CSS file, but it is plugin-based. These plugins run sequentially, so make sure you understand which order you want to run them through.

Here is the Webpack documentation.

Steps

  1. Install the dependencies:

    npm install postcss postcss-cli autoprefixer cssnano sass --save-dev
    

    This command installed these packages:

    • postcss: The core PostCSS processor.
    • postcss-cli: Command-line tool to run PostCSS.
    • autoprefixer: Adds vendor prefixes for better browser support.
    • cssnano: Minifies the final CSS.
    • sass: Compiles SCSS to CSS.
  2. Create a postcss.config.js file in the root of the project:

    module.exports = {
        plugins: [
            require("autoprefixer"),                   // Automatically adds vendor prefixes
            require("cssnano")({ preset: "default" })  // Minifies the CSS
        ]
    };
    
  3. Add the build scripts to your package.json file:

    ...
    "scripts": {
       "build:scss": "sass sass/index.scss build/styles.css",
       "build:css": "postcss build/styles.css --use autoprefixer --use cssnano -o build/styles.min.css",
       "build": "npm run build:scss && npm run build:css"
    },
    ...
    

Example

If you complete the setup in the previous steps, it adds vendor prefixes to your styles and minifies it in build/styles.min.css:

/* input */
.example {
  backdrop-filter: blur(5px);
  mask-image: url(star-mask.png);
}

/* styles.min.css */
.example{-webkit-backdrop-filter:blur(5px);backdrop-filter:blur(5px);-webkit-mask-image:url(star-mask.png);mask-image:url(star-mask.png)}
/*# sourceMappingURL=data:application/json;base64,... */