Spectrum CSS
Spectrum CSS provides a standard CSS implementation of the Spectrum design language for internal and 3rd party use on Adobe's platforms.
Show me a demo
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
.
Where is the JavaScript?
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.
Using Spectrum CSS
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.
Installing components
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.
Using components
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.
Modifying components
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.
Importing UI icons
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/icon/dist/spectrum-css-icons.svg');
</script>
Based on which scales you'll be using, you can choose to load different files:
@spectrum-css/icon/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.
Importing workflow icons
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.
Language support
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>
Variable fallbacks
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.
Optimizing Spectrum CSS
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.
Customizing Spectrum CSS
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;
},
}),
],
};
Contributing
Check out the contributing guidelines for quick start information, and head over to the component documentation for more.
Building
Run the following commands:
yarn install
yarn start
Your dist/
folder should now have a local copy of the Spectrum CSS docs and minimal CSS files, and your browser should be open with the project's preview site. Editing any of the .css
or the .stories.js
files in components/*
will update the project documentation and live reload in your browser.
Important: Ensure you have Node.js > 14 installed or the build system will not run. Node.js > 16.x is preferred.
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 nx clean
or yarn nx run-many --target clean --all
.
To spin up the local development environment (Storybook) without first building the components, use: SKIP_BUILD=true yarn start
as yarn start
alone will start from a clean build.
Documentation site
Local documentation site
Building the project will build and launch the project documentation site in your browser automatically.
See site generation for more information.
Generating and deploying external documentation site
Checkout nextjs
branch, pull, and install dependencies.
git checkout nextjs
git pull
yarn install
Update yml
data from main
yarn importdata
Run prep
script to build the static site locally
yarn prep
Commit changes
git commit -am '<message here>'
Deploy
yarn deploy
Push changes to nextjs
branch
git push origin nextjs
CLI
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
Releasing individual components
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.
Releasing the website
After performing a release, run the following command to release the website:
yarn release:docs
Publishing prereleases
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.
Manual prerelease versioning & publishing
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.