What is shadow DOM?
Shadow DOM allows authors to attach DOM trees to elements, providing encapsulation for markup, styles and even JavaScript.
Note: This article assumes a basic understanding of HTML, CSS and JavaScript and specifically the (regular) Document Object Model.
Introduction
Most HTML elements are "boxes" for their children. They perhaps have some semantic differences, but ultimately we can make a <section> behave like a <b> element through CSS if we wanted to; there's no fundamental difference between them. Some HTML elements, however, have special behavior; for example, a <video> element, an <iframe>, <input> or even a <canvas>. These elements have behaviors that are not achieveable by styles alone.
Web components allow authors to create elements like those. Elements that are unaffected by what's happening around them. The key to making this possible is shadow DOM.
Overview
A shadow DOM essentially changes the way an element renders. Once it has a shadow DOM, it no longer renders its children like "normal" HTML elements do. Instead, it renders the markup inside the shadow DOM.
Now, let's say we want a toggle switch. The native <input type="checkbox"> is not cutting it; it renders a checkbox, and we can't do much about that in terms of styling. That's when shadow DOM comes in handy; we can create the HTML and CSS necessary to render a toggle switch, insert that into the shadow DOM of an element (usually a custom element), and voilà! We've got the toggle switch we wanted.
The difference between this, and just inserting that HTML and CSS into a regular HTML element, is that shadow DOM provides encapsulation. The elements outside of the shadow DOM cannot "see" the elements in the shadow, and neither can styles. Styles outside of the shadow DOM do not affect elements inside it and vice versa. This is great, because it means neither relies on implementation details of the other. If we have a .thumb class in our shadow DOM, and we happen to have an image gallery with .thumb elements outside of the shadow as well, then the styles for either do not affect the other. This encapsulation is perfect for reusability.
Creating a shadow
To create a shadow, we call the .attachShadow() method on an element, or use the declarative syntax. Let's choose the latter for now. Then, let us attach a shadow to a regular <div> element:
<div id=shadow-host>
<template shadowrootmode="closed">
<p>I live in the shadow of my ancestor.</p>
</template>
<p>I am a regular child!</p>
</div>
The <div> element has now gotten a shadow. We'll get to what the "mode" is a bit later; let's first get some terminology out of the way.
The shadow host is the element that the shadow is attached to. In this case, the <div> is the shadow host.
The shadow tree is the markup inside the shadow. In this example, that consists of a paragraph element with some text inside it.
The shadow root is the root node of the shadow DOM. It is not an Element, but an object inheriting from DocumentFragment. This node is not part of the same DOM tree as the shadow host, but is nevertheless attached to (associated with) the shadow host. This allows the shadow to render arbitrary content in complete isolation.
So, in the example above, the shadow tree is entirely represented by the contents of the <template>. The shadow root is automatically created and attached to the <div>, which is the shadow host.
Instead, we can programmatically attach a shadow using the .attachShadow() method. Keep in mind that not all elements may have a shadow attached; in general, shadows are attached to custom elements. We are only using <div> elements as an example. To attach a shadow in JavaScript, first, we obtain a reference to the div; then, we call the method with an options object as first argument.
const div = document.querySelector('#shadow-host');
div.attachShadow({ mode: 'closed' });
Here, we see the equivalent of the shadowrootmode attribute we used on the <template> earlier, when using declarative shadow DOM. The mode essentially determines whether or not the shadow host exposes a reference to the shadow root in its .shadowRoot property. When the mode is 'open', then one may retrieve a reference to the shadow root by accessing div.shadowRoot; when the mode is 'closed', then the .shadowRoot property evaluates to null instead.
Using open shadow roots on custom elements is generally not needed; closed shadows are recommended for complete encapsulation. Specifically, JavaScript should not need access to the markup structure in the shadow DOM, since that creates a tight coupling between implementation details of either side. Instead, use methods and properties to interface with a custom element, so that the shadow tree's markup structure is not relied upon.
Slots
Now, let's say we want to create a custom element that renders a button which toggles a dropdown. Naturally, we want to be able to insert arbitrary content into both the dropdown and the button. We can do this using native <slot> elements. In short, (direct) children are slotted into a <slot> element. We are allowed multiple slots, and we may name them with a name attribute. Then, we can slot the children of the shadow host into certain named slots by giving the children a matching slot attribute. Elements without a slot attribute are slotted in the <slot> element that has no name attribute (in a way, this is the "default" slot).
Let's look at an example implementation for the dropdown component. For simplicity, let's use the declarative way of attaching a shadow.
<custom-dropdown>
<template shadowrootmode="closed">
<button>
<slot name="label"></slot>
</button>
<div id="dropdown">
<slot></slot>
</div>
</template>
<span slot="label">Click me!</span>
<p>I am inside the dropdown</p>
<div>I am also inside the dropdown</div>
</custom-dropdown>
The span with slot="label" is slotted into the <slot> with name="label" (since those names match). The p and div do not have a slot attribute and are thus slotted into the unnamed <slot>.
It might be a bit unclear where the slotted children live. Are the inside the shadow DOM, which they are slotted into, or are they children of the shadow host? It is the latter; the slots signify where to render the children of the shadow host, but they are not part of the shadow tree. This is an important distinction both for styling and scripting.
Encapsulation
The primary focus of shadow DOM is encapsulation. So, let's go through how exactly shadow DOM is protected from outside JavaScript and CSS.
Script encapsulation
First, let's have a look at how shadow DOM is protected from outside JavaScript.
Since shadow DOM is not part of the same DOM tree as its host, we cannot find elements inside a shadow DOM using document.querySelector(). Instead, if we need to find a reference to an element in the shadow tree, we need to call .querySelector() on the shadow root instead. This is where 'open' and 'closed' shadows come in; if the shadow is 'open', then document-level JavaScript can obtain a reference to the shadow root through the .shadowRoot property, and can subsequently query the shadow tree. With 'closed' shadows, this is not possible; document-level JavaScript can no longer obtain a reference to the shadow root or an element inside the shadow tree if not explicitly given by the host element.
The encapsulation really just comes from scoping. In the case of closed shadows, the references to the shadow root are not exposed to code that is not creating it. One reference is returned by the .attachShadow() call; it is then wise to keep a reference to it around, usually in the form of a private property when writing vanilla web components.
The beauty of this encapsulation is that web component authors are free to change internal implementation details as long as the interface remains the same. If the methods and properties don't behave differently, then code relying on those will by definition still work. This promotes more stable, reusable code.
Style encapsulation
Applying styles to a shadow is as simple as including a <style> element inside the shadow tree. And, while shadow DOM provides some form of encapsulation, it is a bit more nuanced than saying "styles don't bleed into shadows". There is a very important distinction to make.
Note: Adding a <style> element to the shadow tree is the most straight-forward way of styling it, but not necessarily the cleanest. If the element itself is not desired, an alternative is creating a CSSStyleSheet object and using its .replace() method to set its contents. Then, it can be added to the shadow by adding it to the .adoptedStyleSheets array on the shadow root itself.
First: selectors outside a shadow DOM do not select (and therefore do not affect) markup inside the shadow, since they are separate DOM trees. Similarly, styles inside a shadow DOM cannot select elements outside of the shadow.
Inherited styles do bleed into the shadow tree; while this may sound undesired at first, it is generally not. For example, text-related properties such as font-family or letter-spacing are inherited properties, and are therefore applied to elements inside shadow trees as well (so we don't have to repeat those declarations in every shadow). Similarly, CSS custom properties are also inherited by default, which allows us to create styling APIs for our components.
There are additional, more specific methods for styling something on the other side of the shadow boundary. Specifically, when used inside a shadow DOM:
- :host selects the host element;
- :host(selector) selects the host element if it also matches selector. For example :host([hidden]) selects the host if it has the hidden attribute.
- ::slotted(selector) selects an element slotted into a certain <slot> element, specifically only direct children that match selector. To target a named slot, apply the pseudo-element directly to the slot, like slot[name=foo]::slotted(…). Note that it is not possible to select "deep" elements this way; only direct, slotted children can be selected.
This gives us control from the inside of the shadow, but what about the other way around? After all, if we're authoring a web component, we also want to provide some amount of flexibility to those using our components. For that, we use parts.
In short, parts are like classes; they are names for specific elements inside a certain shadow, and they are exposed by that very name to the outer DOM.
To name a part, use the part attribute on the element(s) to expose to CSS outside the shadow. This attribute is, like class, a space-separated list of tokens. We may then select the element in question by any of the names in the list through ::part(name). In a way, this behavior is similar to some built-in pseudo-elements that allow styling of certain native elements, such as using ::-webkit-slider-thumb to style the knob on a <input type="range">. It is also possible to select a part using multiple different part names by space-separating them inside the ::part(…) selector like ::part(name1 name2).
Shadows in Yozo
In Yozo, the boilerplate logic behind attaching and and managing a shadow is abstracted away. It stays close to the native API to make the transition from vanilla components easier, but takes away the repetitive nature of writing web components with shadows. Let's look at an example:
<title>in-shadow</title>
<template mode="closed">
<div>I am in the shadow, and I am red!</div>
</template>
<script>
const div = query('div');
// …
</script>
<style>
div { color: red; }
</style>
When the <template>
element has a mode attribute, a shadow is created with the given mode. Additional options to the underlying .attachShadow(…) call may be added as additional attributes, such as setting delegates-focus="true" to pass the delegatesFocus: true option to .attachShadow(…).
Then, querying the template becomes a bit less verbose with query()
. Additionally, this helper function works very similarly for elements without a shadow, making it easier to redesign a component if necessary.
The <style>
block is automatically applied to the shadow, but not appended as <style> element. This keeps the shadow tree clean.
This gives authors complete control over how they want to set up their web components. Boilerplate is hidden, but still completely in the author's control. And, even if one would want to manually attach the shadow, that is possible, too:
<title>in-shadow</title>
<script>
const shadow = this.attachShadow({ mode: 'closed' });
shadow.innerHTML = `…`;
const div = shadow.querySelector('div');
// …
</script>
See also