CSS Custom Properties (CSS vars)

CSS Custom Properties is a new feature introduced to the CSS language, providing the ability to define and re-use variables across stylesheets.

CSS Custom Properties are defined using the --* property syntax, and accessed using the var(--*) CSS function.

To learn more about this language feature, check out the following resources

Stylable variables vs. CSS custom properties

Stylable variables and CSS custom properties offer different capabilities, and as such serve different use-cases.

Stylable variables exist only in your source code, and get replaced during transpilation to the final target code. They serve well for reducing code repetition, increasing readability and can benefit any static theme or styling without incurring any runtime performance cost.

CSS custom properties on the other hand do incur a small runtime cost, but offer the ability to override their values during runtime, allowing dynamic styling through Stylable.

Automatic scoping (namespacing)

Stylable automatically scopes any CSS custom property found in the stylesheet. It does so by generating a unique namespace for the stylesheet (similar to how classes are scoped), and replaces the variable with its scoped counterpart.

Example:

/* entry.st.css */
.root {
    --myVar: green;
    color: var(--myVar);
}

Transpiled output:

/* entry.st.css */
.root {
    --entry-myVar: green;
    color: var(--entry-myVar);
}

Importing CSS variables

Due to the fact Stylable provides scoping to CSS variables, it also provides the ability to import CSS variables defined in another stylesheet.

/* entry.st.css */
:import {
    -st-from: "./imported.st.css";
    -st-named: --myVar;
}

.root {
    /* value determined by the nearest property assignment up the DOM tree */
    color: var(--myVar);
}

.part {
    /* this override will match the namespace of the imported stylesheet */
    --myVar: gold;
    background-color: var(--myVar); /* gold */
}
/* imported.st.css */
.root {
    --myVar: green;
}

Overriding CSS variables during runtime

Override any variable by redefining its value using an inline style attribute.

import { classes, vars } from './entry.st.css';

<div className={classes.root}
     style={{ 
        [vars.myVar]: 'pink',  
        background: 'gold' }} 
/>

Output:

<div className="entry__root" 
    style="--entry-color: green; --entry-border-size: 5px; background: gold;" />

Using global CSS variables

In cases where you have no control over the name of the CSS variable used, use the @st-global-custom-property directive to define CSS variables that will not be scoped, and will maintain their exact given name.

This is mostly useful when working with 3rd-party libraries, where you only attempt to affect it externally.

@st-global-custom-property --color, --bg;

.root {
    --color: green;
    color: var(--color);
}
import { classes } from './entry.st.css';

<div className={classes.root}
     style={{ 
         '--color': 'red', 
         '--bg': 'yellow' }}
/>

Accessing any globally defined variable on the stylesheet will return its global name (un-scoped).