Styling components
Styling components using CSS, CSS pre and post-processors as well as working with web components.
Aurelia 2 simplifies the process of styling components, supporting a variety of CSS flavors and encapsulation methods. Whether you prefer raw CSS, PostCSS, SASS, or Stylus, Aurelia 2 streamlines their integration into your components. Ultimately, all styles compile into standard CSS that browsers can interpret.
Style Conventions
Aurelia 2 automatically imports styles for custom elements based on file naming conventions. For instance, if you have a custom element named my-component
, Aurelia 2 looks for a corresponding stylesheet named my-component.css
.
Example: Automatic Style Import
Consider the following file structure:
my-component.ts
: The TypeScript file defining theMyComponent
class.my-component.css
: The stylesheet containing styles forMyComponent
.my-component.html
: The HTML template forMyComponent
.
Aurelia 2's convention-based approach eliminates the need to import the CSS file explicitly. When you run your application, Aurelia 2 automatically detects and applies the styles.
When MyComponent
is used in the application, the associated styles are automatically applied, giving the paragraph text a blue color and a light grey background with padding and rounded corners.
Shadow DOM
The Shadow DOM API, part of the Web Components standard, provides encapsulation by hiding the component's internal DOM and styles from the rest of the application. Aurelia 2 offers several options for working with Shadow DOM:
Global Shadow DOM: By default, encapsulates all components in your application.
Configured Shadow DOM: Use the
useShadowDOM
decorator to opt-in per component.Global opt-out Shadow DOM: Enable Shadow DOM globally but disable it for specific components.
If your application uses CSS frameworks like Bootstrap, which rely on global styles, consider how Shadow DOM encapsulation may affect their behavior. The following sections guide managing global and shared styles.
Enabling Shadow DOM
To enable Shadow DOM after the initial setup, configure it in the main.ts
file:
The StyleConfiguration
class from Aurelia allows you to specify how styles are applied, including options for Shadow DOM.
Webpack Configuration for Shadow DOM
When using Webpack, ensure the following rule is included in the Webpack configuration:
This rule ensures that HTML files within your src
directory are processed correctly to work with Shadow DOM.
This only applies if you choose Webpack as you bundler. By default, the Aurelia skeleton using makes
will choose Vite.
Vite Configuration for Shadow DOM
By default, Vite is the bundler that makes
uses in Aurelia 2 apps. If you choose Shadow DOM during the configuration, it will add the following into your vite.config.ts
file.
Global Shared Styles
In the Shadow DOM, styles are scoped to their components and don't leak to the global scope. To apply global styles across all components, use the sharedStyles
property in the Shadow DOM configuration.
Example: Using Bootstrap Globally
The sharedStyles
property accepts an array, allowing you to include multiple shared stylesheets.
Using sharedStyles in Vite
When using `sharedStyles` with Vite, you need to make a few changes to work with Vite.
Opting Into Shadow DOM with useShadowDOM
useShadowDOM
The useShadowDOM
decorator, imported from Aurelia, lets you enable Shadow DOM on a per-component basis. Without configuration options, it defaults to open
mode.
Example: Enabling Shadow DOM on a Component
You can specify the Shadow DOM mode (open
or closed
) as a configuration option. The open
mode allows JavaScript to access the component's DOM through the shadowRoot
property, while closed
mode restricts this access.
Example: Specifying Shadow DOM Mode
Disabling Shadow DOM
The useShadowDOM
decorator also allows disabling Shadow DOM for a specific component by passing false
.
Example: Disabling Shadow DOM for a Component
Shadow DOM Special Selectors
Shadow DOM introduces special selectors that offer additional styling capabilities. These selectors are part of the CSS Scoping Module specification.
:host
The :host
selector targets the custom element itself, not its children. You can use the :host()
function to apply styles based on the host element's classes or attributes.
Example: Styling the Host Element
In this example, all app-header
elements have a solid border, and those with the active
class have a grey background.
:host-context
The :host-context
selector styles the custom element based on its context within the document.
Example: Styling Based on Context
Here, app-header
elements will have white text on a dark background inside an element with the dark-mode
class.
CSS Modules
When Shadow DOM does not meet your needs, CSS Modules balance style encapsulation and global accessibility. CSS Modules transform class names into unique identifiers, preventing style collisions.
Vite Configuration for CSS Modules
Vite supports CSS Modules without the need for any configuration. To use CSS Modules, you use the naming convention .module.css
— so for the default skeleton CSS your styles would be my-app.module.css
which is what you will see if you choose CSS Modules during the makes configuration process.
Webpack Configuration for CSS Modules
To use CSS Modules, include the following loader configuration in your Webpack setup:
This rule processes CSS files, enabling CSS module functionality.
Example: Using CSS Modules in Components
Define your styles, and reference the class names in your HTML templates. Webpack will handle the conversion to unique class names.
After processing, the title
class may be transformed to a unique identifier like title_1a2b3c
.
CSS Modules' Special Selectors
CSS Modules support the :global
selector for styling global elements without transformation.
Example: Using Global Selectors
This style will apply globally to elements with the button-primary
class, maintaining the class name without transformation.
Advanced Shadow DOM Usage
Shadow DOM encapsulates a component's styles, preventing them from leaking into the global scope. Aurelia 2 offers granular control over Shadow DOM usage, including global and per-component configuration.
Example: Styling Slotted Content
When using Shadow DOM, you can style content passed into slots using the ::slotted()
pseudo-element.
In this example, content assigned to the tab-content
slot will receive padding and a border. At the same time, the encapsulation ensures that these styles don't affect other elements outside the tab-panel
component.
Example: Theming with CSS Variables
CSS variables can be used within Shadow DOM to create themeable components:
Users of the button-group
component can then define these variables at a higher level to theme the buttons consistently across the application.
Example: Scoped Animations in Shadow DOM
Animations can be defined within Shadow DOM to ensure they are scoped to the component:
The slide-in
animation is encapsulated within the animated-banner
component, preventing it from conflicting with any other animations defined in the global scope.
CSS Modules: Advanced Techniques
CSS Modules provide a powerful way to locally scope class names, avoiding global conflicts. With Aurelia 2, you can leverage CSS Modules to create maintainable, conflict-free styles.
Example: Composing CSS Module Classes
CSS Modules support composing classes from other modules, promoting reusability.
In this example, primaryButton
composes the baseButton
styles, adding its own background and text color. CSS Modules ensures that these class names are unique to avoid styling conflicts.
Example: Theming with CSS Modules
CSS Modules can also facilitate theming by exporting and importing variables.
By defining and exporting theme variables in theme.css
, they can be imported and used in other CSS Modules to ensure consistent theming across the application.
Additional Resources
Please look at the CSS classes and styling section for more guidance on class and style bindings in Aurelia applications. This section covers strategies for dynamically working with classes and inline styles.
Last updated