Authors: [email protected], [email protected]
This is a strawperson proposal for Imperative Shadow DOM distribution API. For more context, please see the WHATWG HTML Standard issue.
One of the drawbacks of Shadow DOM v1, when compared to Shadow DOM v0, is that web developers have to specify slot= attribute for every shadow host's children (except for elements for the default slot).
<shadow-host>
<div slot=slot1></div>
<div slot=slot2></div>
</shadow-host>
Some people would see this as a kind of ugly markup.
Shadow DOM v1 can't explain how <summary>/<details>
elements can be implemented on the top of the current Web Components technology stack, given that <details>
element doesn't need slot= attribute.
Blink has a special logic for some built-in elements to control node-to-slot mapping.
<custom-tab show-panel="2">
<tab-panel></tab-panel>
<tab-panel></tab-panel>
<tab-panel></tab-panel>
</custom-tab>
The second issue is that component creators can't change the slotting behavior based on condition. In the above markup, the <custom-tab>
component can't implement which <tab-panel>
to show based on its attribute show-panel
.
The imperative slotting API allows the developer to explicitly set the assigned nodes for a slot element.
An HTMLSlotElement
gets a new API, called assign(sequence<Node> nodes)
(tentative name). Nodes are an ordered list of slot's
host light DOM children.
The following code demonstrate how to use the new API to solve Case 2 of the Motivation section above.
class CustomTab extends HTMLElement {
static get observedAttributes() {
return ['show-tab'];
}
constructor() {
super();
const shadowRoot = this.attachShadow({mode: 'open', slotAssignment: 'manual'});
shadowRoot.innerHTML = `
<div class="custom-tab">
<slot></slot>
</div>`;
}
attributeChangedCallback(name, oldValue, newValue) {
UpdateDisplayTab(this, newValue);
}
connectedCallback() {
if (!this._observed) {
const target = this;
const showTab = this.getAttribute('show-tab');
const observer = new MutationObserver(function(mutations) {
UpdateDisplayTab(target, showTab);
});
observer.observe(this, {childList: true});
this._observed = true;
}
}
}
function UpdateDisplayTab(elem, tabIdx) {
const shadow = elem.shadowRoot;
const slot = shadow.querySelector("slot");
const panels = elem.querySelectorAll('tab-panel');
if (panels.length && tabIdx && tabIdx <= panels.length ) {
slot.assign([panels[tabIdx-1]]);
} else {
slot.assign([]);
}
}
Mixing declarative API and imperative API would be troublesome and can cause confusion for web developers. We can invent complex rules. However, no one wants to remember complex rules. Also, supporting both in the same shadow tree would make a browser engine complex.
Thus, we don't allow mixing the declarative and imperative APIs in the same shadow tree. Web developers must explicitly opt-in to use an imperative API for each shadow tree. They do this with a new "slotAssignment" parameter on attachShadow():
const sr = attachShadow({ mode: 'open', slotAssignment: 'manual' });
Here, "manual" means "we support only imperative APIs for the shadow tree". If no calls to slot.assign()
are made, then assigned nodes will be left empty.
The default, "auto", means to use the existing declarative slotting algorithm. In this case, calls to slot.assign()
will throw an exception.
partial interface HTMLSlotElement {
...
void assign(sequence<Node> nodes)
}
slot.assign(sequence<Node> nodes)
runs the following steps:
- Check
slotAssignment
is "manual", throw otherwise. - Validate each node in nodes that its node.parentNode == slot.rootNode.host. Throw if not equal.
- Set the slot's
manuallyAssignedNodes
tonodes
. - Run assign slotables for a tree with slot's tree.
Step 4 is required because we have to re-calculate assigned nodes of every slot in the tree at this timing.
Note: It would be worth noting that manuallyAssignedNodes
is not used as
assigned nodes as is. You can think of slot.assign(sequence<Node> nodes)
as telling the engine a set of "candidate nodes" from which assigned nodes are constructed.
dictionary ShadowRootInit {
required ShadowRootMode mode;
boolean delegatesFocus = false;
SlotAssignmentMode slotAssignment = "auto";
};
[Exposed=Window]
interface ShadowRoot : DocumentFragment {
readonly attribute ShadowRootMode mode;
readonly attribute Element host;
attribute EventHandler onslotchange;
readonly attribute SlotAssignmentMode slotAssignment;
};
enum ShadowRootMode { "open", "closed" };
enum SlotAssignmentMode { "manual", "auto" };
To find a slot needs to be updated here. The other steps don't appear to require any updates from the standard's perspective.
To find a slot for a given slotable slotable and an optional open flag (unset unless stated otherwise), run these steps:
-
If slotable's parent is null, then return null.
-
Let shadow be slotable's parent's shadow root.
-
If shadow is null, then return null.
-
If the open flag is set and shadow's mode is not "open", then return null.
-
[New Step] If shadow's
slotAssignment
is "manual", return the associated slot in shadow's tree whosemanuallyAssignedNodes
includes slotable, if any, and null otherwise. -
Return the first slot in shadow's tree whose name is slotable's name, if any, and null otherwise. (<= No change)
Note: This change implies:
-
manuallyAssignedNodes
should be considered an implementation detail. As long as the external behavior doesn't change, UA doesn't allocatemanuallyAssignedNodes
for a slot. -
manuallyAssignedNodes
is used only when a slot is in a shadow tree whoseslotAssignment
is "manual". -
manuallyAssignedNodes
is not used when a slot is in a shadow tree whoseslotAssignment
is "auto".Web developers can not call
slot.assign(...)
for such a slot, it will throw exception to keep it consistent with when invalid nodes are passed in the "manual" case. -
If the same node is set to
manuallyAssignedNodes
when it's already an assigned node in another slot, the node will removed from the previous slot and assigned to the new slot. The "slotchange" event will be raised for both slots.
host
├──/shadowroot (slotAssignment=manual)
│ ├── slot1
│ └── slot2
├── A
└── B
assert_array_equals(slot1.assignedNodes(), []);
assert_array_equals(slot2.assignedNodes(), []);
slot2.assign([A]);
assert_array_equals(slot2.assignedNodes(), [A]);
slot1.assign([B, A]); // The order does matter.
assert_array_equals(slot1.assignedNodes(), [B, A]);
assert_array_equals(slot2.assignedNodes(), []);
slot2.assign([A, B]); // Assignment is absolute, order is preserved.
assert_array_equals(slot1.assignedNodes(), []);
assert_array_equals(slot2.assignedNodes(), [A, B]);
slot1.assign(A);
assert_array_equals(slot1.assignedNodes(), [A]); // slot1 got A.
assert_array_equals(slot2.assignedNodes(), [B]); // slot2 lost A.
slot1.assign([A, A, A]);
assert_array_equals(slot1.assignedNodes(), [A]);
slot1.assign([A, B, A]); // Last one wins
assert_array_equals(slot1.assignedNodes(), [B, A]);
slot1.assign([host]); // Exception is thrown.
assert_array_equals(slot1.assignedNodes(), [B, A]); // Existing assignment doesn't change.
assert_array_equals(slot2.assignedNodes(), []);
slot1.assign([]);
assert_array_equals(slot1.assignedNodes(), []);
assert_array_equals(slot2.assignedNodes(), []);
Example 2: Imperative slotting API doesn't have any effect in a shadow root with slotAssignment=auto.
host
├──/shadowroot (slotAssignment=auto) (default)
│ ├── slot1 name=slot1
│ └── slot2 name=slot2
├── A slot=slot1
└── B slot=slot2
assert_array_equals(slot1.assignedNodes(), [A]);
assert_array_equals(slot2.assignedNodes(), [B]);
slot1.assign([A, B]); // Throw exception, not manual mode and A or B can't be assigned to slot.
assert_array_equals(slot1.assignedNodes(), [A]);
assert_array_equals(slot2.assignedNodes(), [B]);
Example 3: Inappropriate nodes will cause an exception to be thrown, slot returns to its previous state.
host1
├──/shadowroot1 (slotAssignment=manual)
│ └── slot1
└── A
host2
├──/shadowroot2 (slotAssignment=manual)
│ └── slot2
└── B
└── C
slot2.assign([A, B, C]); // Throws exception - A and C are illegal here.
assert_array_equals(slot2.assignedNodes(), []);
slot1.assign([A]);
assert_array_equals(slot1.assignedNodes(), [A]);
shadowroot2.append(slot1); // Allows append, and appendChild, don't need to change spec for throwing exceptions for append().
assert_array_equals(slot1.assignedNodes(), []); // A is not a light dom node of shadowroot2, thus removed.
assert(slot1.getRootNode() == shadowroot2);
shadowroot1.append(slot1);
assert_array_equals(slot1.assignedNodes(), []); // Assignment is absolute. Once slotables are removed, they need to be assigned again.
Example 4: A node can be appended to a host, but it must still be imperatively assigned to slot afterward.
host
└──/shadowroot (slotAssignment=manual)
└── slot1
assert_array_equals(slot1.assignedNodes(), []);
const A = document.createElement('div');
const B = document.createElement('div');
slot1.assign([A, B]); // throw an exception
assert_array_equals(slot1.assignedNodes(), []); // Neither A nor B is slot1's shadow tree's host's child
host.append(A);
assert_array_equals(slot1.assignedNodes(), []);
slot1.assign([A]); // Assign is absolute.
assert_array_equals(slot1.assignedNodes(), [A]);
host.append(B);
assert_array_equals(slot1.assignedNodes(), [A]); // B must be manually slotted
slot1.assign([A, B])
assert_array_equals(slot1.assignedNodes(), [A, B]);
host1
├──/shadowroot1 (slotAssignment=manual)
│ ├── slot1
│ └── slot2
├── A
└── B
host3
├──/shadowroot3 (slotAssignment=manual)
│ └── slot3
assert_array_equals(slot1.assignedNodes(), []);
assert_array_equals(slot2.assignedNodes(), []);
assert_array_equals(slot3.assignedNodes(), []);
// assigned node change triggers slotchange event.
slot2.assign([A]); // slot2 dispatches slotchange event.
assert_array_equals(slot2.assignedNodes(), [A]);
// Both slot1 and slot 2 will be notified.
slot1.assign([B, A]); // slot1 dispatches slotchange event and slot2 dispatches shotchange event.
assert_array_equals(slot1.assignedNodes(), [B, A]);
assert_array_equals(slot2.assignedNodes(), []);
// Node list order change fires slot change event.
slot1.assign([A, B]); // slot1 dispatches slotchange event.
assert_array_equals(slot1.assignedNodes(), [A, B]);
// Exception doesn't trigger a slotchange event.
slot1.assign([A, A, A, host]); // Exception is thrown. No slotchange event.
assert_array_equals(slot1.assignedNodes(), [A, B]);
shadowroot3.append(slot1); // slot1 dispatches slotchange event
assert_array_equals(slot1.assignedNodes(), []);
Let manuallyAssignedNodes
be a internal property of HTMLSlotElement
where it stores an ordered list of nodes when its shadowroot's SlotAssignmentMode is manual
. Unless stated otherwise, it is empty.
Basically, slot.assign(sequence<Node> nodes)
sets the slot's manuallyAssignedNodes
to nodes
.
- manuallyAssignedNodes is an internal field. It is write-only. Users cannot read the value directly.
- manuallyAssignedNodes are different than assigned nodes. They are candidates for slot assignments. However, we check each node's slotable validity. If an invalid node is detected, we throw an exception in
slot.assign()
and clearmanuallyAssignedNodes
. If the validity check is successful, the browser recalculates assigned nodes later, based onmanuallyAssignedNodes
. The calculated assigned nodes are then observable.
The new API sets the assigned nodes for a slot element. It can't be used to track users. Since it's a new API, maybe its presence can be used in some way to finger print users. However, this would be the case for all new APIs.