Skip to content

Commit efb40d5

Browse files
johnjenkinsJohn Jenkinschristian-bromann
authoredJan 16, 2025··
feat(slot-polyfill): patch insertBefore & slotted node parentNode (#6096)
* chore: init * feat(slot-polyfill): patch `insertBefore` & slotted `parentNode` * chore: more tests --------- Co-authored-by: John Jenkins <john.jenkins@nanoporetech.com> Co-authored-by: Christian Bromann <git@bromann.dev>
1 parent 2503dc5 commit efb40d5

File tree

13 files changed

+446
-52
lines changed

13 files changed

+446
-52
lines changed
 

‎src/declarations/stencil-private.ts

+80-2
Original file line numberDiff line numberDiff line change
@@ -1404,7 +1404,7 @@ export interface RenderNode extends HostElement {
14041404
['s-sr']?: boolean;
14051405

14061406
/**
1407-
* Slot name
1407+
* Slot name of either the slot itself or the slotted node
14081408
*/
14091409
['s-sn']?: string;
14101410

@@ -1441,7 +1441,7 @@ export interface RenderNode extends HostElement {
14411441
* This is a reference for a original location node
14421442
* back to the node that's been moved around.
14431443
*/
1444-
['s-nr']?: RenderNode;
1444+
['s-nr']?: PatchedSlotNode | RenderNode;
14451445

14461446
/**
14471447
* Original Order:
@@ -1526,6 +1526,13 @@ export interface RenderNode extends HostElement {
15261526
*/
15271527
__appendChild?: <T extends Node>(newChild: T) => T;
15281528

1529+
/**
1530+
* On a `scoped: true` component
1531+
* with `experimentalSlotFixes` flag enabled,
1532+
* gives access to the original `insertBefore` method
1533+
*/
1534+
__insertBefore?: <T extends Node>(node: T, child: Node | null) => T;
1535+
15291536
/**
15301537
* On a `scoped: true` component
15311538
* with `experimentalSlotFixes` flag enabled,
@@ -1534,6 +1541,77 @@ export interface RenderNode extends HostElement {
15341541
__removeChild?: <T extends Node>(child: T) => T;
15351542
}
15361543

1544+
export interface PatchedSlotNode extends Node {
1545+
/**
1546+
* Slot name
1547+
*/
1548+
['s-sn']?: string;
1549+
1550+
/**
1551+
* Original Location Reference:
1552+
* A reference pointing to the comment
1553+
* which represents the original location
1554+
* before it was moved to its slot.
1555+
*/
1556+
['s-ol']?: RenderNode;
1557+
1558+
/**
1559+
* Slot host tag name:
1560+
* This is the tag name of the element where this node
1561+
* has been moved to during slot relocation.
1562+
*
1563+
* This allows us to check if the node has been moved and prevent
1564+
* us from thinking a node _should_ be moved when it may already be in
1565+
* its final destination.
1566+
*
1567+
* This value is set to `undefined` whenever the node is put back into its original location.
1568+
*/
1569+
['s-sh']?: string;
1570+
1571+
/**
1572+
* Is a `slot` node when `shadow: false` (or `scoped: true`).
1573+
*
1574+
* This is a node (either empty text-node or `<slot-fb>` element)
1575+
* that represents where a `<slot>` is located in the original JSX.
1576+
*/
1577+
['s-sr']?: boolean;
1578+
1579+
/**
1580+
* On a `scoped: true` component
1581+
* with `experimentalSlotFixes` flag enabled,
1582+
* returns the actual `parentNode` of the component
1583+
*/
1584+
__parentNode?: RenderNode;
1585+
1586+
/**
1587+
* On a `scoped: true` component
1588+
* with `experimentalSlotFixes` flag enabled,
1589+
* returns the actual `nextSibling` of the component
1590+
*/
1591+
__nextSibling?: RenderNode;
1592+
1593+
/**
1594+
* On a `scoped: true` component
1595+
* with `experimentalSlotFixes` flag enabled,
1596+
* returns the actual `previousSibling` of the component
1597+
*/
1598+
__previousSibling?: RenderNode;
1599+
1600+
/**
1601+
* On a `scoped: true` component
1602+
* with `experimentalSlotFixes` flag enabled,
1603+
* returns the actual `nextElementSibling` of the component
1604+
*/
1605+
__nextElementSibling?: RenderNode;
1606+
1607+
/**
1608+
* On a `scoped: true` component
1609+
* with `experimentalSlotFixes` flag enabled,
1610+
* returns the actual `nextElementSibling` of the component
1611+
*/
1612+
__previousElementSibling?: RenderNode;
1613+
}
1614+
15371615
export type LazyBundlesRuntimeData = LazyBundleRuntimeData[];
15381616

15391617
export type LazyBundleRuntimeData = [

‎src/mock-doc/node.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -182,7 +182,7 @@ export class MockNode {
182182

183183
remove() {
184184
if (this.parentNode != null) {
185-
this.parentNode.removeChild(this);
185+
(this as any).__parentNode ? (this as any).__parentNode.removeChild(this) : this.parentNode.removeChild(this);
186186
}
187187
}
188188

‎src/runtime/client-hydrate.ts

+2-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ import { doc, plt } from '@platform';
33
import { CMP_FLAGS } from '@utils';
44

55
import type * as d from '../declarations';
6-
import { patchNextPrev } from './dom-extras';
6+
import { patchSlottedNode } from './dom-extras';
77
import { createTime } from './profile';
88
import {
99
COMMENT_NODE_ID,
@@ -195,7 +195,7 @@ export const initializeClientHydrate = (
195195

196196
if (BUILD.experimentalSlotFixes) {
197197
// patch this node for accessors like `nextSibling` (et al)
198-
patchNextPrev(slottedItem.node);
198+
patchSlottedNode(slottedItem.node);
199199
}
200200
}
201201

‎src/runtime/dom-extras.ts

+129-23
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,7 @@
11
import { BUILD } from '@app-data';
2-
import { getHostRef, plt, supportsShadow } from '@platform';
3-
import { HOST_FLAGS } from '@utils/constants';
2+
import { supportsShadow } from '@platform';
43

54
import type * as d from '../declarations';
6-
import { PLATFORM_FLAGS } from './runtime-constants';
75
import {
86
addSlotRelocateNode,
97
getHostSlotChildNodes,
@@ -12,7 +10,8 @@ import {
1210
getSlottedChildNodes,
1311
updateFallbackSlotVisibility,
1412
} from './slot-polyfill-utils';
15-
import { insertBefore } from './vdom/vdom-render';
13+
14+
/// HOST ELEMENTS ///
1615

1716
export const patchPseudoShadowDom = (hostElementPrototype: HTMLElement) => {
1817
patchCloneNode(hostElementPrototype);
@@ -22,11 +21,17 @@ export const patchPseudoShadowDom = (hostElementPrototype: HTMLElement) => {
2221
patchSlotInsertAdjacentElement(hostElementPrototype);
2322
patchSlotInsertAdjacentHTML(hostElementPrototype);
2423
patchSlotInsertAdjacentText(hostElementPrototype);
24+
patchInsertBefore(hostElementPrototype);
2525
patchTextContent(hostElementPrototype);
2626
patchChildSlotNodes(hostElementPrototype);
2727
patchSlotRemoveChild(hostElementPrototype);
2828
};
2929

30+
/**
31+
* Patches the `cloneNode` method on a `scoped` Stencil component.
32+
*
33+
* @param HostElementPrototype The Stencil component to be patched
34+
*/
3035
export const patchCloneNode = (HostElementPrototype: HTMLElement) => {
3136
const orgCloneNode = HostElementPrototype.cloneNode;
3237

@@ -93,7 +98,14 @@ export const patchSlotAppendChild = (HostElementPrototype: any) => {
9398

9499
const slotChildNodes = getHostSlotChildNodes(slotNode, slotName);
95100
const appendAfter = slotChildNodes[slotChildNodes.length - 1];
96-
const insertedNode = insertBefore(appendAfter.parentNode, newChild, appendAfter.nextSibling as d.RenderNode);
101+
102+
const parent = intrnlCall(appendAfter, 'parentNode') as d.RenderNode;
103+
let insertedNode: d.RenderNode;
104+
if (parent.__insertBefore) {
105+
insertedNode = parent.__insertBefore(newChild, appendAfter.nextSibling);
106+
} else {
107+
insertedNode = parent.insertBefore(newChild, appendAfter.nextSibling);
108+
}
97109

98110
// Check if there is fallback content that should be hidden
99111
updateFallbackSlotVisibility(this);
@@ -150,7 +162,13 @@ export const patchSlotPrepend = (HostElementPrototype: HTMLElement) => {
150162
addSlotRelocateNode(newChild, slotNode, true);
151163
const slotChildNodes = getHostSlotChildNodes(slotNode, slotName);
152164
const appendAfter = slotChildNodes[0];
153-
return insertBefore(appendAfter.parentNode, newChild, appendAfter.nextSibling as d.RenderNode);
165+
const parent = intrnlCall(appendAfter, 'parentNode') as d.RenderNode;
166+
167+
if (parent.__insertBefore) {
168+
return parent.__insertBefore(newChild, intrnlCall(appendAfter, 'nextSibling'));
169+
} else {
170+
return parent.insertBefore(newChild, intrnlCall(appendAfter, 'nextSibling'));
171+
}
154172
}
155173

156174
if (newChild.nodeType === 1 && !!newChild.getAttribute('slot')) {
@@ -223,6 +241,68 @@ export const patchSlotInsertAdjacentText = (HostElementPrototype: HTMLElement) =
223241
};
224242
};
225243

244+
/**
245+
* Patches the `insertBefore` of a non-shadow component.
246+
*
247+
* The *current* node to insert before may not be in the root of our component
248+
* (e.g. if it's 'slotted' it appears in the root, but isn't really)
249+
*
250+
* This tries to find where the *current* node lives within the component and insert the new node before it
251+
* *If* the new node is in the same slot as the *current* node. Otherwise the new node is appended to it's 'slot'
252+
*
253+
* @param HostElementPrototype the custom element prototype to patch
254+
*/
255+
const patchInsertBefore = (HostElementPrototype: HTMLElement) => {
256+
const eleProto: d.RenderNode = HostElementPrototype;
257+
if (eleProto.__insertBefore) return;
258+
259+
eleProto.__insertBefore = HostElementPrototype.insertBefore;
260+
261+
HostElementPrototype.insertBefore = function <T extends d.PatchedSlotNode>(
262+
this: d.RenderNode,
263+
newChild: T,
264+
currentChild: d.RenderNode | null,
265+
) {
266+
const slotName = (newChild['s-sn'] = getSlotName(newChild));
267+
const slotNode = getHostSlotNodes(this.__childNodes, this.tagName, slotName)[0];
268+
const slottedNodes = this.__childNodes ? this.childNodes : getSlottedChildNodes(this.childNodes);
269+
270+
if (slotNode) {
271+
let found = false;
272+
273+
slottedNodes.forEach((childNode) => {
274+
if (childNode === currentChild || currentChild === null) {
275+
// we found the node to insert before in our list of 'lightDOM' / slotted nodes
276+
found = true;
277+
278+
if (currentChild === null || slotName !== currentChild['s-sn']) {
279+
// new child is not in the same slot as 'slot before' node
280+
// so let's use the patched appendChild method. This will correctly slot the node
281+
this.appendChild(newChild);
282+
return;
283+
}
284+
285+
if (slotName === currentChild['s-sn']) {
286+
// current child ('slot before' node) is 'in' the same slot
287+
addSlotRelocateNode(newChild, slotNode);
288+
289+
const parent = intrnlCall(currentChild, 'parentNode') as d.RenderNode;
290+
if (parent.__insertBefore) {
291+
// the parent is a patched component, so we need to use the internal method
292+
parent.__insertBefore(newChild, currentChild);
293+
} else {
294+
parent.insertBefore(newChild, currentChild);
295+
}
296+
}
297+
return;
298+
}
299+
});
300+
if (found) return newChild;
301+
}
302+
return (this as d.RenderNode).__insertBefore(newChild, currentChild);
303+
};
304+
};
305+
226306
/**
227307
* Patches the `insertAdjacentElement` method for a slotted node inside a scoped component. Specifically,
228308
* we only need to patch the behavior for the specific `beforeend` and `afterbegin` positions so the element
@@ -253,7 +333,7 @@ export const patchSlotInsertAdjacentElement = (HostElementPrototype: HTMLElement
253333
};
254334

255335
/**
256-
* Patches the text content of an unnamed slotted node inside a scoped component
336+
* Patches the `textContent` of an unnamed slotted node inside a scoped component
257337
*
258338
* @param hostElementPrototype the `Element` to be patched
259339
*/
@@ -315,17 +395,9 @@ export const patchChildSlotNodes = (elm: HTMLElement) => {
315395
patchHostOriginalAccessor('childNodes', elm);
316396
Object.defineProperty(elm, 'childNodes', {
317397
get() {
318-
if (
319-
!plt.$flags$ ||
320-
!getHostRef(this)?.$flags$ ||
321-
((plt.$flags$ & PLATFORM_FLAGS.isTmpDisconnected) === 0 && getHostRef(this)?.$flags$ & HOST_FLAGS.hasRendered)
322-
) {
323-
const result = new FakeNodeList();
324-
const nodes = getSlottedChildNodes(this.__childNodes);
325-
result.push(...nodes);
326-
return result;
327-
}
328-
return FakeNodeList.from(this.__childNodes);
398+
const result = new FakeNodeList();
399+
result.push(...getSlottedChildNodes(this.__childNodes));
400+
return result;
329401
},
330402
});
331403
};
@@ -344,11 +416,12 @@ export const patchChildSlotNodes = (elm: HTMLElement) => {
344416
*
345417
* @param node the slotted node to be patched
346418
*/
347-
export const patchNextPrev = (node: Node) => {
419+
export const patchSlottedNode = (node: Node) => {
348420
if (!node || (node as any).__nextSibling || !globalThis.Node) return;
349421

350422
patchNextSibling(node);
351423
patchPreviousSibling(node);
424+
patchParentNode(node);
352425

353426
if (node.nodeType === Node.ELEMENT_NODE) {
354427
patchNextElementSibling(node as Element);
@@ -360,7 +433,6 @@ export const patchNextPrev = (node: Node) => {
360433
* Patches the `nextSibling` accessor of a non-shadow slotted node
361434
*
362435
* @param node the slotted node to be patched
363-
* Required during during testing / mock environnement.
364436
*/
365437
const patchNextSibling = (node: Node) => {
366438
// already been patched? return
@@ -383,7 +455,6 @@ const patchNextSibling = (node: Node) => {
383455
* Patches the `nextElementSibling` accessor of a non-shadow slotted node
384456
*
385457
* @param element the slotted element node to be patched
386-
* Required during during testing / mock environnement.
387458
*/
388459
const patchNextElementSibling = (element: Element) => {
389460
if (!element || (element as any).__nextElementSibling) return;
@@ -405,7 +476,6 @@ const patchNextElementSibling = (element: Element) => {
405476
* Patches the `previousSibling` accessor of a non-shadow slotted node
406477
*
407478
* @param node the slotted node to be patched
408-
* Required during during testing / mock environnement.
409479
*/
410480
const patchPreviousSibling = (node: Node) => {
411481
if (!node || (node as any).__previousSibling) return;
@@ -427,7 +497,6 @@ const patchPreviousSibling = (node: Node) => {
427497
* Patches the `previousElementSibling` accessor of a non-shadow slotted node
428498
*
429499
* @param element the slotted element node to be patched
430-
* Required during during testing / mock environnement.
431500
*/
432501
const patchPreviousElementSibling = (element: Element) => {
433502
if (!element || (element as any).__previousElementSibling) return;
@@ -446,6 +515,26 @@ const patchPreviousElementSibling = (element: Element) => {
446515
});
447516
};
448517

518+
/**
519+
* Patches the `parentNode` accessor of a non-shadow slotted node
520+
*
521+
* @param node the slotted node to be patched
522+
*/
523+
export const patchParentNode = (node: Node) => {
524+
if (!node || (node as any).__parentNode) return;
525+
526+
patchHostOriginalAccessor('parentNode', node);
527+
Object.defineProperty(node, 'parentNode', {
528+
get: function () {
529+
return this['s-ol']?.parentNode || this.__parentNode;
530+
},
531+
set: function (value) {
532+
// mock-doc sets parentNode?
533+
this.__parentNode = value;
534+
},
535+
});
536+
};
537+
449538
/// UTILS ///
450539

451540
const validElementPatches = ['children', 'nextElementSibling', 'previousElementSibling'] as const;
@@ -456,6 +545,7 @@ const validNodesPatches = [
456545
'nextSibling',
457546
'previousSibling',
458547
'textContent',
548+
'parentNode',
459549
] as const;
460550

461551
/**
@@ -481,3 +571,19 @@ function patchHostOriginalAccessor(
481571
}
482572
if (accessor) Object.defineProperty(node, '__' + accessorName, accessor);
483573
}
574+
575+
/**
576+
* Get the original / internal accessor or method of a node or element.
577+
*
578+
* @param node - the node to get the accessor from
579+
* @param method - the name of the accessor to get
580+
*
581+
* @returns the original accessor or method of the node
582+
*/
583+
function intrnlCall<T extends d.RenderNode, P extends keyof d.RenderNode>(node: T, method: P): T[P] {
584+
if ('__' + method in node) {
585+
return node[('__' + method) as keyof d.RenderNode] as T[P];
586+
} else {
587+
return node[method];
588+
}
589+
}

‎src/runtime/slot-polyfill-utils.ts

+16-8
Original file line numberDiff line numberDiff line change
@@ -45,15 +45,18 @@ export const updateFallbackSlotVisibility = (elm: d.RenderNode) => {
4545

4646
/**
4747
* Get's the child nodes of a component that are actually slotted.
48-
* This is only required until all patches are unified
48+
* It does this by using root nodes of a component; for each slotted node there is a
49+
* corresponding slot location node which points to the slotted node (via `['s-nr']`).
50+
*
51+
* This is only required until all patches are unified / switched on all the time (then we can rely on `childNodes`)
4952
* either under 'experimentalSlotFixes' or on by default
5053
* @param childNodes all 'internal' child nodes of the component
5154
* @returns An array of slotted reference nodes.
5255
*/
53-
export const getSlottedChildNodes = (childNodes: NodeListOf<d.RenderNode>) => {
56+
export const getSlottedChildNodes = (childNodes: NodeListOf<ChildNode>): d.PatchedSlotNode[] => {
5457
const result = [];
5558
for (let i = 0; i < childNodes.length; i++) {
56-
const slottedNode = childNodes[i]['s-nr'];
59+
const slottedNode = ((childNodes[i] as d.RenderNode)['s-nr'] as d.PatchedSlotNode) || undefined;
5760
if (slottedNode && slottedNode.isConnected) {
5861
result.push(slottedNode);
5962
}
@@ -68,21 +71,25 @@ export const getSlottedChildNodes = (childNodes: NodeListOf<d.RenderNode>) => {
6871
* @param slotName the name of the slot to match on.
6972
* @returns a reference to the slot node that matches the provided name, `null` otherwise
7073
*/
71-
export const getHostSlotNodes = (childNodes: NodeListOf<ChildNode>, hostName: string, slotName?: string) => {
74+
export function getHostSlotNodes(childNodes: NodeListOf<ChildNode>, hostName: string, slotName?: string) {
7275
let i = 0;
7376
let slottedNodes: d.RenderNode[] = [];
7477
let childNode: d.RenderNode;
7578

7679
for (; i < childNodes.length; i++) {
7780
childNode = childNodes[i] as any;
78-
if (childNode['s-sr'] && childNode['s-hn'] === hostName && (!slotName || childNode['s-sn'] === slotName)) {
81+
if (
82+
childNode['s-sr'] &&
83+
childNode['s-hn'] === hostName &&
84+
(slotName === undefined || childNode['s-sn'] === slotName)
85+
) {
7986
slottedNodes.push(childNode);
8087
if (typeof slotName !== 'undefined') return slottedNodes;
8188
}
8289
slottedNodes = [...slottedNodes, ...getHostSlotNodes(childNode.childNodes, hostName, slotName)];
8390
}
8491
return slottedNodes;
85-
};
92+
}
8693

8794
/**
8895
* Get slotted child nodes of a slot node
@@ -138,12 +145,13 @@ export const isNodeLocatedInSlot = (nodeToRelocate: d.RenderNode, slotName: stri
138145
* (the order of the slot location nodes determines the order of the slotted nodes in our patched accessors)
139146
*/
140147
export const addSlotRelocateNode = (
141-
newChild: d.RenderNode,
148+
newChild: d.PatchedSlotNode,
142149
slotNode: d.RenderNode,
143150
prepend?: boolean,
144151
position?: number,
145152
) => {
146153
let slottedNodeLocation: d.RenderNode;
154+
147155
// does newChild already have a slot location node?
148156
if (newChild['s-ol'] && newChild['s-ol'].isConnected) {
149157
slottedNodeLocation = newChild['s-ol'];
@@ -181,5 +189,5 @@ export const addSlotRelocateNode = (
181189
newChild['s-sh'] = slotNode['s-hn'];
182190
};
183191

184-
export const getSlotName = (node: d.RenderNode) =>
192+
export const getSlotName = (node: d.PatchedSlotNode) =>
185193
node['s-sn'] || (node.nodeType === 1 && (node as Element).getAttribute('slot')) || '';

‎src/runtime/test/dom-extras.spec.tsx

+14-4
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
11
import { Component, h, Host } from '@stencil/core';
22
import { newSpecPage, SpecPage } from '@stencil/core/testing';
33

4-
import { patchNextPrev, patchPseudoShadowDom } from '../../runtime/dom-extras';
4+
import { patchPseudoShadowDom, patchSlottedNode } from '../../runtime/dom-extras';
55

66
describe('dom-extras - patches for non-shadow dom methods and accessors', () => {
77
let specPage: SpecPage;
88

99
const nodeOrEleContent = (node: Node | Element) => {
10-
return (node as Element)?.outerHTML || node?.nodeValue.trim();
10+
return (node as Element)?.outerHTML || node?.nodeValue?.trim();
1111
};
1212

1313
beforeEach(async () => {
@@ -102,7 +102,7 @@ describe('dom-extras - patches for non-shadow dom methods and accessors', () =>
102102
});
103103

104104
it('patches nextSibling / previousSibling accessors of slotted nodes', async () => {
105-
specPage.root.childNodes.forEach((node: Node) => patchNextPrev(node));
105+
specPage.root.childNodes.forEach((node: Node) => patchSlottedNode(node));
106106
expect(nodeOrEleContent(specPage.root.firstChild)).toBe('Some default slot, slotted text');
107107
expect(nodeOrEleContent(specPage.root.firstChild.nextSibling)).toBe('<span>a default slot, slotted element</span>');
108108
expect(nodeOrEleContent(specPage.root.firstChild.nextSibling.nextSibling)).toBe(``);
@@ -122,12 +122,22 @@ describe('dom-extras - patches for non-shadow dom methods and accessors', () =>
122122
});
123123

124124
it('patches nextElementSibling / previousElementSibling accessors of slotted nodes', async () => {
125-
specPage.root.childNodes.forEach((node: Node) => patchNextPrev(node));
125+
specPage.root.childNodes.forEach((node: Node) => patchSlottedNode(node));
126126
expect(nodeOrEleContent(specPage.root.children[0].nextElementSibling)).toBe(
127127
'<div slot="second-slot"> a second slot, slotted element <span>nested element in the second slot<span></span></span></div>',
128128
);
129129
expect(nodeOrEleContent(specPage.root.children[0].nextElementSibling.previousElementSibling)).toBe(
130130
'<span>a default slot, slotted element</span>',
131131
);
132132
});
133+
134+
it('patches parentNode of slotted nodes', async () => {
135+
specPage.root.childNodes.forEach((node: Node) => patchSlottedNode(node));
136+
expect(specPage.root.children[0].parentNode.tagName).toBe('CMP-A');
137+
expect(specPage.root.children[1].parentNode.tagName).toBe('CMP-A');
138+
expect(specPage.root.childNodes[0].parentNode.tagName).toBe('CMP-A');
139+
expect(specPage.root.childNodes[1].parentNode.tagName).toBe('CMP-A');
140+
expect(specPage.root.children[0].__parentNode.tagName).toBe('DIV');
141+
expect(specPage.root.childNodes[0].__parentNode.tagName).toBe('DIV');
142+
});
133143
});

‎src/runtime/vdom/util.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ export function toVNode(node: Node): d.VNode | null {
1919
const vnode: d.VNode = newVNode(node.nodeName.toLowerCase(), null);
2020
vnode.$elm$ = node;
2121

22-
const childNodes = node.childNodes;
22+
const childNodes = (node as any).__childNodes || node.childNodes;
2323
let childVnode: d.VNode;
2424

2525
for (let i = 0, l = childNodes.length; i < l; i++) {

‎src/runtime/vdom/vdom-annotations.ts

+1-1
Original file line numberDiff line numberDiff line change
@@ -38,7 +38,7 @@ export const insertVdomAnnotations = (doc: Document, staticComponents: string[])
3838

3939
orgLocationNodes.forEach((orgLocationNode) => {
4040
if (orgLocationNode != null && orgLocationNode['s-nr']) {
41-
const nodeRef = orgLocationNode['s-nr'];
41+
const nodeRef = orgLocationNode['s-nr'] as d.RenderNode;
4242

4343
let hostId = nodeRef['s-host-id'];
4444
let nodeId = nodeRef['s-node-id'];

‎src/runtime/vdom/vdom-render.ts

+31-10
Original file line numberDiff line numberDiff line change
@@ -8,9 +8,10 @@
88
*/
99
import { BUILD } from '@app-data';
1010
import { consoleDevError, doc, plt, supportsShadow } from '@platform';
11-
import { CMP_FLAGS, HTML_NS, isDef, SVG_NS } from '@utils';
11+
import { CMP_FLAGS, HTML_NS, isDef, NODE_TYPES, SVG_NS } from '@utils';
1212

1313
import type * as d from '../../declarations';
14+
import { patchParentNode } from '../dom-extras';
1415
import { NODE_TYPE, PLATFORM_FLAGS, VNODE_FLAGS } from '../runtime-constants';
1516
import { isNodeLocatedInSlot, updateFallbackSlotVisibility } from '../slot-polyfill-utils';
1617
import { h, isHost, newVNode } from './h';
@@ -853,12 +854,29 @@ export const nullifyVNodeRefs = (vNode: d.VNode) => {
853854
* @param reference anchor element
854855
* @returns inserted node
855856
*/
856-
export const insertBefore = (parent: Node, newNode: d.RenderNode, reference?: d.RenderNode): Node => {
857+
export const insertBefore = (
858+
parent: Node,
859+
newNode: d.RenderNode,
860+
reference?: d.RenderNode | d.PatchedSlotNode,
861+
): Node => {
857862
if (BUILD.scoped && typeof newNode['s-sn'] === 'string' && !!newNode['s-sr'] && !!newNode['s-cr']) {
863+
// this is a slot node
858864
addRemoveSlotScopedClass(newNode['s-cr'], newNode, parent as d.RenderNode, newNode.parentElement);
865+
} else if (BUILD.experimentalSlotFixes && typeof newNode['s-sn'] === 'string') {
866+
// this is a slotted node.
867+
if (parent.getRootNode().nodeType !== NODE_TYPES.DOCUMENT_FRAGMENT_NODE) {
868+
// we don't need to patch this node if it's nested in a shadow root
869+
patchParentNode(newNode);
870+
}
871+
// potentially use the patched insertBefore method. This will correctly slot the new node
872+
return parent.insertBefore(newNode, reference);
873+
}
874+
875+
if (BUILD.experimentalSlotFixes && (parent as d.RenderNode).__insertBefore) {
876+
return (parent as d.RenderNode).__insertBefore(newNode, reference) as d.RenderNode;
877+
} else {
878+
return parent?.insertBefore(newNode, reference) as d.RenderNode;
859879
}
860-
const inserted = parent?.insertBefore(newNode, reference);
861-
return inserted;
862880
};
863881

864882
/**
@@ -1065,9 +1083,13 @@ render() {
10651083
) {
10661084
let orgLocationNode = nodeToRelocate['s-ol']?.previousSibling as d.RenderNode | null;
10671085
while (orgLocationNode) {
1068-
let refNode = orgLocationNode['s-nr'] ?? null;
1086+
let refNode = (orgLocationNode['s-nr'] as d.RenderNode) ?? null;
10691087

1070-
if (refNode && refNode['s-sn'] === nodeToRelocate['s-sn'] && parentNodeRef === refNode.parentNode) {
1088+
if (
1089+
refNode &&
1090+
refNode['s-sn'] === nodeToRelocate['s-sn'] &&
1091+
parentNodeRef === ((refNode as d.PatchedSlotNode).__parentNode || refNode.parentNode)
1092+
) {
10711093
refNode = refNode.nextSibling as d.RenderNode | null;
10721094

10731095
// If the refNode is the same node to be relocated or another element's slot reference, keep searching to find the
@@ -1086,10 +1108,9 @@ render() {
10861108
}
10871109
}
10881110

1089-
if (
1090-
(!insertBeforeNode && parentNodeRef !== nodeToRelocate.parentNode) ||
1091-
nodeToRelocate.nextSibling !== insertBeforeNode
1092-
) {
1111+
const parent = (nodeToRelocate as d.PatchedSlotNode).__parentNode || nodeToRelocate.parentNode;
1112+
const nextSibling = (nodeToRelocate as d.PatchedSlotNode).__nextSibling || nodeToRelocate.nextSibling;
1113+
if ((!insertBeforeNode && parentNodeRef !== parent) || nextSibling !== insertBeforeNode) {
10931114
// we've checked that it's worth while to relocate
10941115
// since that the node to relocate
10951116
// has a different next sibling or parent relocated
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
1+
import { Fragment, h } from '@stencil/core';
2+
import { render } from '@wdio/browser-runner/stencil';
3+
4+
describe('testing a `scoped="true"` component `insertBefore` method', () => {
5+
let host: HTMLScopedSlotInsertBeforeElement;
6+
let defaultSlot: HTMLDivElement;
7+
let startSlot: HTMLDivElement;
8+
let endSlot: HTMLDivElement;
9+
10+
beforeEach(async () => {
11+
render({
12+
template: () => (
13+
<>
14+
<scoped-slot-insertbefore>
15+
<p>My initial slotted content.</p>
16+
</scoped-slot-insertbefore>
17+
</>
18+
),
19+
});
20+
21+
await $('#parentDiv').waitForExist();
22+
host = document.querySelector('scoped-slot-insertbefore');
23+
startSlot = host.querySelector('#parentDiv .start-slot');
24+
endSlot = host.querySelector('#parentDiv .end-slot');
25+
defaultSlot = host.querySelector('#parentDiv .default-slot');
26+
});
27+
28+
it('slots nodes in the correct order when they have the same slot', async () => {
29+
expect(defaultSlot.children.length).toBe(2);
30+
expect(startSlot.children.length).toBe(1);
31+
expect(endSlot.children.length).toBe(1);
32+
33+
const el1 = document.createElement('p');
34+
const el2 = document.createElement('p');
35+
const el3 = document.createElement('p');
36+
el1.innerText = 'Content 1. ';
37+
el2.innerText = 'Content 2. ';
38+
el3.innerText = 'Content 3. ';
39+
40+
host.insertBefore(el1, null);
41+
host.insertBefore(el2, el1);
42+
host.insertBefore(el3, el2);
43+
44+
expect(defaultSlot.children.length).toBe(5);
45+
expect(startSlot.children.length).toBe(1);
46+
expect(endSlot.children.length).toBe(1);
47+
expect(defaultSlot.textContent).toBe(
48+
`Default slot is here:My initial slotted content.Content 3. Content 2. Content 1. `,
49+
);
50+
});
51+
52+
it('slots nodes in the correct slot despite the insertion order', async () => {
53+
expect(defaultSlot.children.length).toBe(2);
54+
expect(startSlot.children.length).toBe(1);
55+
expect(endSlot.children.length).toBe(1);
56+
57+
const el1 = document.createElement('p');
58+
const el2 = document.createElement('p');
59+
const el3 = document.createElement('p');
60+
el1.innerText = 'Content 1. ';
61+
el1.slot = 'start-slot';
62+
el2.innerText = 'Content 2. ';
63+
el2.slot = 'end-slot';
64+
el3.innerText = 'Content 3. ';
65+
66+
host.insertBefore(el1, null);
67+
host.insertBefore(el2, el1);
68+
host.insertBefore(el3, el2);
69+
70+
expect(defaultSlot.children.length).toBe(3);
71+
expect(startSlot.children.length).toBe(2);
72+
expect(endSlot.children.length).toBe(2);
73+
expect(host.textContent).toBe(`My initial slotted content.Content 1. Content 2. Content 3. `);
74+
});
75+
76+
it('can still use original `insertBefore` method', async () => {
77+
expect(defaultSlot.children.length).toBe(2);
78+
expect(startSlot.children.length).toBe(1);
79+
expect(endSlot.children.length).toBe(1);
80+
81+
const el1 = document.createElement('p');
82+
const el2 = document.createElement('p');
83+
el1.innerText = 'Content 1. ';
84+
el2.innerText = 'Content 2. ';
85+
el2.slot = 'end-slot';
86+
87+
host.__insertBefore(el1, null);
88+
host.__insertBefore(el2, host.querySelector('#parentDiv'));
89+
90+
expect(host.firstElementChild.textContent).toBe('Content 2. ');
91+
expect(host.lastElementChild.textContent).toBe('Content 1. ');
92+
93+
expect(defaultSlot.children.length).toBe(2);
94+
expect(startSlot.children.length).toBe(1);
95+
expect(endSlot.children.length).toBe(1);
96+
});
97+
});
+28
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,28 @@
1+
import { Component, h } from '@stencil/core';
2+
3+
@Component({
4+
tag: 'scoped-slot-insertbefore',
5+
scoped: true,
6+
})
7+
export class ScopedSlotInsertBefore {
8+
render() {
9+
return (
10+
<div id="parentDiv">
11+
<div class="start-slot" style={{ background: 'red' }}>
12+
<div>Start slot is here:</div>
13+
<slot name="start-slot"></slot>
14+
</div>
15+
16+
<div class="default-slot" style={{ background: 'green' }}>
17+
<div>Default slot is here:</div>
18+
<slot></slot>
19+
</div>
20+
21+
<div class="end-slot" style={{ background: 'blue' }}>
22+
<div>End slot is here:</div>
23+
<slot name="end-slot"></slot>
24+
</div>
25+
</div>
26+
);
27+
}
28+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
import { h } from '@stencil/core';
2+
import { render } from '@wdio/browser-runner/stencil';
3+
4+
describe('checks slotted node parentNode', () => {
5+
beforeEach(async () => {
6+
render({
7+
template: () => (
8+
<cmp-slotted-parentnode>
9+
A text node <div>An element</div>
10+
</cmp-slotted-parentnode>
11+
),
12+
});
13+
await $('cmp-slotted-parentnode label').waitForExist();
14+
});
15+
16+
it('slotted nodes and elements `parentNode` do not return component internals', async () => {
17+
expect((document.querySelector('cmp-slotted-parentnode').children[0].parentNode as Element).tagName).toBe(
18+
'CMP-SLOTTED-PARENTNODE',
19+
);
20+
expect((document.querySelector('cmp-slotted-parentnode').childNodes[0].parentNode as Element).tagName).toBe(
21+
'CMP-SLOTTED-PARENTNODE',
22+
);
23+
});
24+
25+
it('slotted nodes and elements `__parentNode` return component internals', async () => {
26+
expect((document.querySelector('cmp-slotted-parentnode').children[0] as any).__parentNode.tagName).toBe('LABEL');
27+
expect((document.querySelector('cmp-slotted-parentnode').childNodes[0] as any).__parentNode.tagName).toBe('LABEL');
28+
});
29+
});
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
import { Component, h, Host } from '@stencil/core';
2+
3+
@Component({
4+
tag: 'cmp-slotted-parentnode',
5+
scoped: true,
6+
})
7+
export class CmpSlottedParentnode {
8+
render() {
9+
return (
10+
<Host>
11+
<label>
12+
<slot />
13+
</label>
14+
</Host>
15+
);
16+
}
17+
}

0 commit comments

Comments
 (0)
Please sign in to comment.