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
- MDN - Custom properties
- MDN - Using CSS custom properties
- Smashing Magazine - It’s Time To Start Using CSS Custom Properties
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).