Reminder that @scope and HTML style blocks are a potent combo
Reminder that @scope and HTML style blocks are a potent combo êŽë š
There are so many different tools for writing scoped CSS with very different takes on how to go about it. Sometimes itâs only a sub-feature of a tool that does other things. But itâs generally thought of as a concept the requires tooling to accomplish.
Have you ever written a React component that imported scoped styles, perhaps as a CSS module?
import styles from "./MyComponent.module.css";
Or used a Styled Component to push some styles onto a component youâre already defining?
const Button = styled.button
Maybe your Vue components used <style scoped>
blocks within them like you can do with Vue Single File Components out of the box?
<template>
<button>Submit</button>
</template>
<style scoped>
button {
border: 3px solid green;
}
</script>
Despite the seemingly widely-applicable button
selector above, the styles will actually be tightly scoped to the button you see in the template after processing.
Or maybe you use Tailwind to apply styling classes directly to elements and like it partially because you donât have to âname anythingâ.
There are a lot of solutions like this out there in the land of building websites. Iâm pretty convinced myself that scoped CSS is a good idea:
- There is little to worry about. Styles wonât leak out and have unintended consequences.
- Itâs possible to do efficient things like not load the styles for components that donât appear on a particular page at the time of loading.
- When a component retires, so do itâs styles.
- Styles are often âco-locatedâ with the component, meaning there is a logical obvious connection between markup and styles.
Iâm here to tell you: you can do all this stuff with just HTML and CSS.
And Iâm not even talking about Web Components or anything particularly controversial or limiting. Vanilla HTML and CSS.
What you do is dump a <style>
block in the HTML at the point you want the styles scoped. Just like this:
<main>
<div>
<p>Dum de dum de dum.<p>
</div>
<div>
<p>Hi ho here we go.</p>
<style>
@scope { /* Scope is the <div> above, as this is a direct child. */
:scope { /* This selects the <div> */
border: 1px solid red;
/* I can use CSS nesting in here, ensuring *everything* is safely scoped */
p { color: red; }
}
}
</style>
</div>
</main>
Hereâs an example of using it where one of these three <article>
s has a scoped styles variation:
Iâm using the scoped styles as a âvariationâ there, but the whole block of styles of that component could be used like that whether it is a variation or not. Itâs a way to apply styling only to a particular branch of the olâ DOM tree. No tooling required. Any way you produce components that end up in the DOM could be done this way, from basic HTML includes to fancy framework components.
Why isnât this being used much?
Well, itâs the Firefox support mostly I think. Firefox just straight up doesnât support it at the time of this writing. Iâd say this is a strong candidate for Interop 2025. It looked like it was tried for in 2024 but maybe it was too new or something. But maybe Interop isnât needed as it appears as if itâs being actively worked on, so maybe it wonât be long, dunno.
Once Firebox support is there, I could imagine this as being highly used as a way to accomplish scoped styles for components. It doesnât require any tooling or have any limitations on what CSS you can use. I would think that would appeal to any existing CSS scoping tool as it would require them to do much less work and work faster.
CSS @scope can do more things, but this particular feature is my favorite and likely to have the biggest impact over time.