Spectrum CSS provides a standard CSS implementation of the Spectrum design language for internal and 3rd party use on Adobe's platforms.
Check out the component viewer for a demo of every component included in Spectrum CSS or spin it up locally with: yarn start
. This preview is driven by Storybook, a development tool that enables rapid prototyping and test suite integration.
Alternatively, you can preview the documentation site or spin up the site locally with: yarn dev
.
Spectrum CSS is CSS-only, implementing only the interactivity that can be done with pure CSS. Thus, Spectrum CSS should only be used by implementations of Spectrum, or very simple applications that only need things like typography, checkboxes, text fields, etc.
Adobe maintains separate JavaScript libraries written with web components and React that use Spectrum CSS to create fully interactive Spectrum components.
The preferred method of using Spectrum CSS relies on custom properties to swap out variables for different themes and colorstops. This has the lowest bundle size and the simplest usage, but is incompatible with < IE 11.
To use Spectrum CSS with IE 11, see the legacy usage documentation.
Each component is released on npm as a separate, individually versioned package inside of the @spectrum-css
org.
To get started, install the following components:
yarn install @spectrum-css/vars @spectrum-css/typography @spectrum-css/page @spectrum-css/icon @spectrum-css/button
Installed components will be available in the node_modules/@spectrum-css/
folder.
Spectrum CSS components have build output that uses CSS custom properties to change themes and scales. Start by including the base set of variables:
<!-- Include global variables first -->
<link
rel="stylesheet"
href="node_modules/@spectrum-css/vars/dist/spectrum-global.css"
/>
<!-- Include only the scales your application needs -->
<link
rel="stylesheet"
href="node_modules/@spectrum-css/vars/dist/spectrum-medium.css"
/>
<link
rel="stylesheet"
href="node_modules/@spectrum-css/vars/dist/spectrum-large.css"
/>
<!-- Include only the colorstops your application needs -->
<link
rel="stylesheet"
href="node_modules/@spectrum-css/vars/dist/spectrum-light.css"
/>
<link
rel="stylesheet"
href="node_modules/@spectrum-css/vars/dist/spectrum-dark.css"
/>
<link
rel="stylesheet"
href="node_modules/@spectrum-css/vars/dist/spectrum-darkest.css"
/>
<!-- Include tokens -->
<link
rel="stylesheet"
href="node_modules/@spectrum-css/tokens/dist/index.css"
/>
<!-- Include index-vars.css for all components you need -->
<link
rel="stylesheet"
href="node_modules/@spectrum-css/page/dist/index-vars.css"
/>
<link
rel="stylesheet"
href="node_modules/@spectrum-css/typography/dist/index-vars.css"
/>
<link
rel="stylesheet"
href="node_modules/@spectrum-css/icon/dist/index-vars.css"
/>
<link
rel="stylesheet"
href="node_modules/@spectrum-css/button/dist/index-vars.css"
/>
Make sure you've included the relevant classes to choose which scale and colorstop you want:
<html class="spectrum spectrum--medium spectrum--light"></html>
To switch to Express, load vars from @spectrum-css/expressvars
instead of @spectrum-css/vars
and add the .spectrum--express
class to the <html>
element:
<html class="spectrum spectrum--medium spectrum--light spectrum--express">
<head>
<!-- Include only the scales your application needs -->
<link
rel="stylesheet"
href="node_modules/@spectrum-css/expressvars/dist/spectrum-medium.css"
/>
<link
rel="stylesheet"
href="node_modules/@spectrum-css/expressvars/dist/spectrum-large.css"
/>
<!-- Include only the colorstops your application needs -->
<link
rel="stylesheet"
href="node_modules/@spectrum-css/expressvars/dist/spectrum-light.css"
/>
<link
rel="stylesheet"
href="node_modules/@spectrum-css/expressvars/dist/spectrum-dark.css"
/>
</head>
</html>
Components can then be added by leveraging the code from the documentation.
Because CSS custom properties honor the cascading nature of CSS, you can infinitely nest different colorstops and themes, as illustrated here.
You can override variables and modify Spectrum CSS' look and feel by re-defining the custom properties in context. Look for the Custom Property API section in each component to determine which custom properties you can override. See ActionButton for a complete example.
Some components require certain "UI icons" to render. To get these icons, you'll need to use loadicons
.
For most use cases, you'll want to use spectrum-css-icons.svg
so you have support for both scales:
<script src="node_modules/loadicons/index.js"></script>
<script>
loadIcons('node_modules/@spectrum-css/ui-icons/dist/spectrum-css-icons.svg');
</script>
Based on which scales you'll be using, you can choose to load different files:
@spectrum-css/ui-icons/dist/spectrum-css-icons.svg
- Both medium and large icons for responsive UIs that support both.spectrum--medium
and.spectrum--large
@spectrum-css/icon/dist/spectrum-css-icons-medium.svg
- Medium icons only, supports.spectrum--medium
only@spectrum-css/icon/dist/spectrum-css-icons-large.svg
- Large icons only, supports.spectrum--large
only
Note: If you're using spectrum-css-icons.svg
, be sure to add .spectrum--medium
or .spectrum--large
to the <html>
element, or you'll see both medium and large icons at once.
If your app has any level of complexity, you'll need "workflow" icons to indicate actions. These icons are not required to render the base components, and instead are used within buttons or menu items for actions like share, play, justify, save, etc.
These icons are released within the @adobe/spectrum-css-workflow-icons
package. After installing the package, you can import the entire set of icons in all sizes as follows:
loadIcons(
"node_modules/@adobe/spectrum-css-workflow-icons/dist/spectrum-icons.svg"
);
You can then use the icons in your app. Visit the Spectrum workflow icon list and click on any icon to get the markup.
To take advantage of locale specific changes such as placeholders not italicizing Japanese, your application should specify a Content-Language
response header or set the lang
attribute.
In addition, you must set the dir
attribute for components to render correctly.
For English, a left-to-right language:
<html lang="en" dir="ltr"></html>
For Arabic, a right-to-left language:
<html lang="ar" dir="rtl"></html>
Each component has a dist/vars.css
file that contains declarations for each component-level variable used by the component. The CSS in dist/index-vars.css
references these variables, but has fallbacks for the global variables or hardcoded value that that the component-level variables resolve to.
As such, you do not need to include dist/vars.css
unless:
- You wish to reference the component-level variables used by a component in external CSS (i.e.
--spectrum-divider-medium-height
) - You have upgraded
@spectrum-css/vars
, but have not updated a component (such as@spectrum-css/divider
) and do not want to update the component-level variables used by that component
When this file is imported, if in updated version of @spectrum-css/vars
changed global variables (such as a global color, --spectrum-global-color-gray-300
), you will get those updates. However, if the updated version of @spectrum-css/vars
changed component-level variables (such as the height of a medium divider, --spectrum-divider-medium-height
), you will not get those updates. As such, this file can be used to lock-in the basic visual style of a component while still allowing for system-level updates.
In most cases, this file will not be required, so you can safely ignore it. If you see unexpected visual changes creeping into components that you have not updated, dist/vars.css
may correct them.
Spectrum CSS is designed to be as flexible as possible, and as such, leaves room for optimization. Let's assume you've included a few components as dependencies in your package.json
:
{
"name": "my-project",
"devDependencies": {
"@spectrum-css/button": "^3.0.0",
"@spectrum-css/page": "^3.0.0",
"@spectrum-css/vars": "^3.0.0"
}
}
You've created an index.css
that imports a few components, a scale, and a color-theme:
@import "node_modules/@spectrum-css/vars/dist/spectrum-global.css";
@import "node_modules/@spectrum-css/vars/dist/spectrum-medium.css";
@import "node_modules/@spectrum-css/vars/dist/spectrum-light.css";
@import "node_modules/@spectrum-css/page/dist/index-vars.css";
@import "node_modules/@spectrum-css/button/dist/index-vars.css";
To build an more optimized bundle, you can employ a few simple PostCSS plugins. First, install them:
yarn add -D postcss-import postcss-varfallback postcss-dropunusedvars cssnano
Next, create a postcss.config.js
:
module.exports = {
plugins: [
require("postcss-import"),
require("postcss-varfallback"),
require("postcss-dropunusedvars"),
require("cssnano"),
],
};
Finally, include PostCSS in your build process, or run it from the command line:
postcss -o dist/index.min.css index.css
dist/index.min.css
file will contain a much slimmer version of Spectrum CSS customized to only include the variables used by the components that you imported. If you've referenced any of variables Spectrum CSS defines in your CSS, they will be perserved as well.
If you need an even smaller bundle, you can employ a tool such as PurifyCSS to strip unused CSS classes from the output.
You can employ postcss-transformselectors
to change the classnames Spectrum CSS uses. For instance, you may want to use bare h1
/h2
/h3
instead of .spectrum-Heading.spectrum-Heading--size*
.
To do this, first install the plugin:
npm i postcss-transformselectors --save-dev
Then, add something like this to your postcss.config.js
:
module.exports = {
plugins: [
require("postcss-transformselectors")({
replace: [
{ search: ".spectrum-Heading--sizeXXL", replace: "h1" },
{ search: ".spectrum-Heading--sizeXL", replace: "h2" },
{ search: ".spectrum-Heading--sizeL", replace: "h3" },
],
transform: (selector) => {
if (selector.startsWith(".spectrum-Heading")) {
// Operate on each selector in a selector list
return selector
.split(",")
.map((selectorPart) => {
// Create separate selectors for each reference to .spectrum-Heading
return ["h1", "h2", "h3"]
.map((h) => {
return selectorPart.replace(".spectrum-Heading", h);
})
.join(",");
})
.join(",");
}
// Don't mess with things that don't have .spectrum-Heading in them
return selector;
},
}),
],
};
A very special thank you to all of our contributors without whom this project would not be possible.
Want to join the team? Check out the contributing guidelines for quick start information.
Run the following commands:
yarn install
Important: Requires at least Node 18.x.
yarn start
This will spin up the local development environment (Storybook). Editing any of the .css
or the .stories.js
files in components/*
will live reload in your browser.
This project is leveraging caching from Nx to speed up the build process. If you are seeing unexpected results, you can clear the cache by running yarn cache:clean
.
This project includes several small scripts to help with common tasks.
yarn compare
: This compares the current version of components with the previous versions published to NPM and output a list of all the changes that have been made. This is useful for reviewing changes before a release. The information is provided in the command-line output as well as in a simple web page that is opened in your default browser upon completion. The web page includes links to the visual diffs for each component when the file sizes have changed. Components with no changes are not included in the output.yarn refresh:env
: This copies values for the project's.env
file (an asset never committed to the repo as it contains login secrets) by using the.env.example
file as a template. This script is useful when you need to update the.env
file with new values from the.env.example
file or when you checkout or clean the repo and need to restore the.env
file.yarn refresh:directory
: This will remove any deprecated package folders that are no longer in use. The goal is to make migrating to a new project architecture easier for the most number of users.yarn lint:components
: Provides helpful updates and warnings for a component's package.json file. This helps keep all components in alignment.
The following tasks are available:
yarn build:all
- Performs a build of all components, documentation site, and storybookyarn build
- Performs a build of all componentsyarn dev
- Performs a component build, runs storybook, and serves the documentation on the default port (3000), then starts watching components and website filesyarn clean
- Cleans all output files for the project and all componentsyarn watch
- Assuming a build has already been performed, re-starts starts watching components and website files. Presumes a browser is already open to your locally served docs
Releasing individual components is handled by Lerna. When any component or its dependencies change, Lerna will queue that component (and all of its dependents) up for a release.
To release everything that has changed, simply run:
yarn release
Version numbers are automatically determined, changelogs generated, and packages published.
After performing a release, run the following command to release the website:
yarn release:docs
Occasionally, it can be helpful for our subscribers to test CSS changes before they're considered ready to be part of a stable release. To facilitate this, we can publish prerelease versions.
To publish prerelease versions:
- First, be sure that you're working on a branch other than
main
. - Once your change(s) are ready to be committed, be aware of the severity of the change(s), and be sure to author your commit message so that Lerna understands how to increase the version number(s) of the affected components.
- Once your changes are committed, you must build the affected package(s) locally before publishing them to npm. An npm task for cleaning, building, and beta publishing is available, and it can be run via the following command:
yarn release:beta-from-package
. This command will perform a fullclean
(via theclean
task), a fullbuild
(via thebuild
task), and will attempt to bump the version numbers in the affected package(s) (vialerna publish --conventional-prerelease --preid beta --pre-dist-tag beta --no-private
). - Depending on the severity of your change(s), and before publishing to npm, Lerna should show a preview of the affected package version numbers that look something like:
@spectrum-css/tag: 3.3.8 => 3.3.9-beta.0
. Additionally, at this time, Lerna will ask if you would like to continue with publishing the changes or cancel. - Selecting
y
to publish will publish the affected package(s) to npm.
Occasionally, you may want to run a prerelease for an individual package and skip a version bump for consuming packages. It's possible to manually change a package's version number to achieve this.
- For the package that you want to prerelease, manually alter the version number in the package's
package.json
file.- For example, let's say you'd like to release a
beta
version of the Switch component. In the Switch'spackage.json
, manually change theversion
number from its current number ("version": "1.0.23"
) to the next appropriate semver version number ("version": "2.0.0-beta.0"
).
- For example, let's say you'd like to release a
- Save your changes, and commit them with the appropriate conventional commit-style commit message:
chore(switch): manual version bump for beta release
or something similar. - You must run a build before continuing with the prerelease. An npm task for cleaning, building, and beta publishing is available, and it can be run via the following command:
yarn release:beta-from-package
. This command will perform a fullclean
(via theclean
task), a fullbuild
(via thebuild
task), and will attempt to publish the package (vialerna publish --conventional-prerelease --preid beta --pre-dist-tag beta --no-private
). - Depending on the severity of your change(s), and before publishing to npm, Lerna should show a preview of the affected package version number that looks something like:
@spectrum-css/switch: 1.0.23 => 2.0.0-beta.0
. Additionally, at this time, Lerna will ask if you would like to continue with publishing the changes or cancel. - Selecting
y
to publish will publish the affected package(s) to npm.