Mainmatter started the Ember Initiative and made Vite support our team's first goal. After one month of work, new Ember apps from version 3.28
to the latest can be built with Vite. One blocker developers may encounter though relates to the addons they use. If your app depends on a classic addon that is incompatible with Vite, you will be stuck in the classic-build world. This is not the way the Ember community usually operates, we don't like to leave anyone behind, and we - the Ember Initiative team - are aligned with this philosophy. To prevent developers from being blocked by classic addons, we audited all the top 100 addons on Ember Observer and started to sort them out: Which are already v2 and fully compatible? Which classic ones should be compatible with Vite? Which ones need a bit of rework? And last but not least, which ones rely too much on classic-build semantics and should be abandoned in favor of a different solution?
ember-css-modules is a widely used addon that brings CSS modules to classic Ember apps. However, we identified it as an addon that should be replaced with a different solution before moving to @embroider/vite
. The problem is that until a week ago, there was no out-of-the-box solution to migrate away from ember-css-modules. You would have to figure out a way all by yourself, but CSS management is so challenging, especially in bigger apps. Fear no more, we have paved the way for you: it goes through ember-scoped-css. In this blog post, we will explain everything about the migration strategy we recommend and the type of refactoring you should expect.
anchorAbout ember-scoped-css
Just like ember-css-modules, ember-scoped-css is an Ember addon that brings CSS modules to your Ember apps. It is a v2 addon that can be used in conjunction with a v1 addon - ember-scoped-css-compat - that brings the same functionality to classic apps.
anchorIs ember-scoped-css the best choice?
If you want to start using CSS modules in an Ember Vite app, ember-scoped-css is not the only option. You could also follow Vite documentation directly or look into some other solutions the Ember community came up with. The advantage of ember-scoped-css is that it's quite similar to ember-css-modules in the way the implementation is structured, so it's a migration path we recommend to unblock your upgrade to Vite without drastically changing all your CSS.
The similarities between ember-css-modules and ember-scoped-css enable the possibility to get them to work at the same time. Let's assume your Ember app contains lots of components and you are using a continuous deployment process: it could be very challenging for you to upgrade to Vite if you need to drastically refactor the CSS as part of the upgrade in a single Pull Request. The path we suggest in this blog post allows you to enable ember-scoped-css file by file, so you can control the pace of the migration until ember-css-modules is no longer used.
anchorCSS isolation by class rename
ember-css-modules and ember-scoped-css both allow you to isolate your component styles by creating a CSS file alongside your component templates. They are different, but they both work following the same general idea: "CSS isolation by class rename". To understand this approach, you can use two resources:
- The document CSS Isolation from ember-scoped-css repository.
- The article Cookbook: migrate an existing Ember app to CSS modules is a walkthrough to install ember-css-modules in an Ember application that used initially one global CSS file. It presents the introduction of CSS modules with a more "practical" angle that allows you to see clearly the "without / with" CSS modules.
To sum it up, ember-css-modules and ember-scoped-css both provide ways to have your custom-class
renamed to custom-class+sha
following an interpolation pattern. This way, this class becomes unique and applies only to this specific component.
ember-css-modules renames the classes to ._custom-class_sha
, and ember-scoped-css renames the classes to .custom-class_sha
. It's a minor difference that turns to be important in the migration process (because both renaming methods conflict). There are also a few major differences that will force you to rework your code while migrating from one to the other.
anchorIntroducing the resource repository
To help you with the migration, we have created a GitHub repository that serves as resource material. The demo consists of a 6.2
app inspired by ember-welcome-page. A <WelcomePageCopy />
component has been implemented directly in the application and has been reworked to illustrate ember-css-modules and ember-scoped-css features.
This repository contains several branches that can be compared as a before/after diff views. These branches and views illustrate the different steps of the migration:
- Installing ember-scoped-css for a file by file migration: diff
- From ember-css-modules to ember-scoped-css: diff
- From Classic to Vite once ember-scoped-css is used: diff
Additionally, refer to the How to use this repo section of the README.md
, and especially the Try it yourself for additional information about the available resources for comparing and experimenting.
anchorInstalling ember-scoped-css for a file by file migration
✨ Click here to see the diff ✨
To use ember-scoped-css in a classic Ember app, you need to install ember-scoped-css
and ember-scoped-css-compat
. As soon as you do this, ember-scoped-css class rename starts to apply to your .css
files and conflicts with ember-css-modules class rename. As a result, your styles are broken. To solve this problem, the idea is to split the CSS files into two categories: those handled by ember-css-modules, and those handled by ember-scoped-css. To create these categories, we are going to use the extension
option of ember-css-modules, and a few other things.
anchor1. Change the modules extension for ember-css-modules
The extension
option of ember-css-modules defines what's the file extension of CSS modules (.css
by default). This option is not properly documented as an individual feature, it's only mentioned in integration with other features like using sass preprocessor, but it actually works as an individual feature.
👉 Change the extension of all the CSS files of your app to .module.css
, including app.css
to app.module.css
.
👉 Add the following configuration to your ember-cli-build.js
:
const app = new EmberApp(defaults, {
+ cssModules: {
+ extension: 'module.css',
+ },
});
ember-css-modules should now work exactly as before.
anchor2. Install and configure ember-scoped-css
👉 Install latest ember-scoped-css
and ember-scoped-css-compat
.
👉 Add the following configuration to your ember-cli-build.js
:
const app = new EmberApp(defaults, {
cssModules: {
extension: 'module.css',
+ intermediateOutputPath: 'css-modules.css',
},
+ 'ember-scoped-css': {
+ passthrough: ['css-modules.css'],
+ passthroughDestination: 'assets',
+ },
});
👉 Create a file app/styles/app.css
with the following content:
@import "css-modules";
Here, we tell ember-css-modules to emit the modules resulting CSS in a file called css-modules.css
, then we import this file in the regular app.css
consumed by ember-scoped-css. The passthrough
option will tell ember-scoped-css about the existence of this file so it's correctly included in the build. All the files .module.css
are processed by ember-css-modules, all the files .css
are processed by ember-scoped-css, and everything gets included in the build.
At this point, your app styles should be fixed and everything should show exactly as before.
anchor3. Do the file by file migration
The migration consists in having more and more CSS modules processed by ember-scoped-css, until you no longer have any module processed by ember-css-modules. To do so, rename a file .module.css
to .css
, and eventually refactor ember-css-modules specific features with a different approach using the next section of this guide.
⚠️ Be aware that ember-css-modules allows you to import styles from another CSS module (see composes
, @value
, and {{local-class}}
later in this guide). If you use this feature, figure out the best migration order for all your files. For instance, you may want to refactor your current CSS to migrate away from the composes
pattern even while you are still using ember-css-modules.
anchorFrom ember-css-modules to ember-scoped-css
✨ Click here to see the diff ✨
anchor1. local-class
versus class
ember-css-modules introduces the attribute local-class
as a marker that identifies scoped classes. In a template, it's your responsibility as a developer to assign to local-class
the scoped classes that you defined in the corresponding CSS module, and to assign to class
the global classes.
In ember-scoped-css, you only use the attribute class
. If the class name is defined in the CSS module, it will be transformed, else it just stays what it is and the global styles apply.
anchor2. Tag selectors: global versus local
ember-css-modules is designed to rename class selectors. It doesn't rename tag selectors like div
, p
, img
, a
... even when they are defined in the CSS module of a component. These selectors are bundled as is in the final style and therefore the style applies globally. In that way, using tag selectors in components when using ember-css-modules requires caution.
ember-scoped-css is a bit more intuitive on that front. The tag selectors are no longer global but scoped. When these selectors are used in the component's CSS, a class named after the sha
is added to the corresponding DOM elements, and the syntax selector.sha
is used CSS-side to scope the style only to the component. This way, tag selectors behave just like any class selector, which can make the CSS module clearer.
<!-- Final DOM -->
<ul class="e9accf110">
<li class="e9accf110"></li>
</ul>
/* Built CSS */
ul.e9accf110 > li.e9accf110 {
padding-bottom: 0.5em;
font-size: 1.1em;
}
anchor3. Id selectors: weird versus global
If you use ember-css-modules, you may know already that its behavior with ids is a bit buggy. It doesn't rename element ids in the DOM, but it transforms them in the CSS. As a result, you can't use HTML ids to style the component, the styles wouldn't be applied:
<!-- Final DOM -->
<main id="ember-welcome-page-id-selector"></main>
/* Built CSS */
#_ember-welcome-page-id-selector_f42zxn {
/* Style for _ember-welcome-page-id-selector_f42zxn which doesn't exist in the DOM */
}
ember-scoped-css, on the other hand, tries to be a bit smarter on that front. It considers the following HTML principle: the HTML id should be unique in the document. Therefore, ember-scoped-css doesn't transform ids at all and the style remains global.
anchor4. Each CSS file is a CSS module versus each component has its CSS module
In ember-css-modules, each CSS file in your app/styles/
folder is considered a CSS module, and all the classes in there are scoped. This approach allows you to scope the CSS of route controllers. For instance, the local CSS of your application.hbs
is expected to be defined in app/styles/application.css
. The class names transform also applies to app/styles/app.css
(this questionable behaviour is the purpose of an issue, but let's consider the addon as it is here). Each time you want to apply a style globally using a class name, you need to use :global
pseudo-selector.
ember-scoped-css is more... "scoped". The idea of this addon is to scope your components CSS by creating a component.css
alongside a component.hbs
. On the other hand, app/styles/app.css
contains global CSS, and if you have other global CSS files they should be explicitly imported.
anchor5. Importing styles and properties versus parent passes in to child
anchori. ember-css-modules specific features
ember-css-modules has three specific features:
{{local-class}}
helper imports a local class from another CSS module directly in the template.
<p class="{{local-class 'postscript' from='css-modules-to-scoped-css/components/welcome-page-copy.css'}}">
composes
gets a local class inherited from another local class that could be located in another CSS module.
.guide {
composes: guide from "css-modules-to-scoped-css/components/welcome-page-copy.css";
font-weight: bold;
}
@value
imports a CSS variable that could be located in another CSS module.
/* app/styles/app.css */
@value ember-orange: rgb(255, 92, 68);
/* app/components/welcome-page-copy.css */
@value ember-orange from 'css-modules-to-scoped-css/styles/app';
All of these features are based on the same idea: given the path to a CSS module, the content of this CSS module can be imported in another CSS module. Phrased this way, the approach sounds inspired by importing js modules in our code files, but applied to CSS with a very specific semantic.
anchorii. Alternatives in ember-scoped-css
{{scoped-class}}
helper can be used to pass a local class from the parent component to the child component, which results in something similar to the former {{local-class}}
. The approach is still very different: the scoped class is passed in from the component that defines it (rather than any component can retrieve the scoped class from anywhere given the path), and since only components have their CSS scoped, we can't use that helper from outside a component (because outside a component there is no CSS module with scoped classes to pass in).
Instead of relying on @values
, we can try to rework the styles with CSS custom properties (also called CSS variables) directly.
anchor6. Classic cascading versus providing layers
ember-css-modules transforms the class names but doesn't take any initiative such as emitting layers.
By default, ember-scoped-css emits the components CSS in a CSS layer components
. The name of the layer is configurable.
⚠️ If you were not using layers until now, you can choose disable this behavior with layerName: false
, but... We recommend to start using them. Vite manages the CSS a bit differently in dev mode and build mode. Without layers to state the CSS order clearly, you may end up with a different order between what you see in development and your production build.
As long as your app is still a classic app though, ember-scoped-css
is not very intuitive when it comes to layers. You need to state the layers order first, but ember-scoped-css
won't let you do this in app.css
, it will define your component styles first. To work around this issue, you can add a <style>
tag in your index.html
before any CSS is imported:
+ <style>
+ @layer utilities, components;
+ </style>
<link integrity="" rel="stylesheet" href="assets/vendor.css">
<link integrity="" rel="stylesheet" href="assets/css-modules-to-scoped-css.css">
anchorFrom Classic to Vite once ember-scoped-css is used
✨ Click here to see the diff ✨
Once your classic app uses only ember-scoped-css without style regressions, your app is one step closer to Vite. If you don't have any other blocker to handle, then you can start building with Vite. ember-vite-codemod is here to help!
👉 After running the codemod, remove ember-scoped-css-compat
from your dependencies. It was here only to ensure compatibility with the classic app.
👉 Adapt ember-scoped-css configuration. You can also have a look at the corresponding PR on this repository to see additional comments.
anchorConclusion
With the introduction of 3.28 support, even your oldest Ember apps are closer to Vite than they have ever been. Managing all the prerequisites to use Vite is also a very good way to pave a smoother upgrade path for the future of your app and adopt all the modern practices at your own pace. As long as the Ember Initiative goes on, we will continue to work hard to keep Ember easy to use and up-to-date with the latest web standards.
Do you rely on v1 addons that don't belong to the top 100? Do you need guidance to make them compatible with Vite or find a migration path like the one presented in this blog post? If you'd like to help us help you, and improve Ember for the entire web, support the Ember Initiative, spread the word, and follow our progress on this blog.