1
1
import { BUILD } from '@app-data' ;
2
- import { getHostRef , plt , supportsShadow } from '@platform' ;
3
- import { HOST_FLAGS } from '@utils/constants' ;
2
+ import { supportsShadow } from '@platform' ;
4
3
5
4
import type * as d from '../declarations' ;
6
- import { PLATFORM_FLAGS } from './runtime-constants' ;
7
5
import {
8
6
addSlotRelocateNode ,
9
7
getHostSlotChildNodes ,
@@ -12,7 +10,8 @@ import {
12
10
getSlottedChildNodes ,
13
11
updateFallbackSlotVisibility ,
14
12
} from './slot-polyfill-utils' ;
15
- import { insertBefore } from './vdom/vdom-render' ;
13
+
14
+ /// HOST ELEMENTS ///
16
15
17
16
export const patchPseudoShadowDom = ( hostElementPrototype : HTMLElement ) => {
18
17
patchCloneNode ( hostElementPrototype ) ;
@@ -22,11 +21,17 @@ export const patchPseudoShadowDom = (hostElementPrototype: HTMLElement) => {
22
21
patchSlotInsertAdjacentElement ( hostElementPrototype ) ;
23
22
patchSlotInsertAdjacentHTML ( hostElementPrototype ) ;
24
23
patchSlotInsertAdjacentText ( hostElementPrototype ) ;
24
+ patchInsertBefore ( hostElementPrototype ) ;
25
25
patchTextContent ( hostElementPrototype ) ;
26
26
patchChildSlotNodes ( hostElementPrototype ) ;
27
27
patchSlotRemoveChild ( hostElementPrototype ) ;
28
28
} ;
29
29
30
+ /**
31
+ * Patches the `cloneNode` method on a `scoped` Stencil component.
32
+ *
33
+ * @param HostElementPrototype The Stencil component to be patched
34
+ */
30
35
export const patchCloneNode = ( HostElementPrototype : HTMLElement ) => {
31
36
const orgCloneNode = HostElementPrototype . cloneNode ;
32
37
@@ -93,7 +98,14 @@ export const patchSlotAppendChild = (HostElementPrototype: any) => {
93
98
94
99
const slotChildNodes = getHostSlotChildNodes ( slotNode , slotName ) ;
95
100
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
+ }
97
109
98
110
// Check if there is fallback content that should be hidden
99
111
updateFallbackSlotVisibility ( this ) ;
@@ -150,7 +162,13 @@ export const patchSlotPrepend = (HostElementPrototype: HTMLElement) => {
150
162
addSlotRelocateNode ( newChild , slotNode , true ) ;
151
163
const slotChildNodes = getHostSlotChildNodes ( slotNode , slotName ) ;
152
164
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
+ }
154
172
}
155
173
156
174
if ( newChild . nodeType === 1 && ! ! newChild . getAttribute ( 'slot' ) ) {
@@ -223,6 +241,68 @@ export const patchSlotInsertAdjacentText = (HostElementPrototype: HTMLElement) =
223
241
} ;
224
242
} ;
225
243
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
+
226
306
/**
227
307
* Patches the `insertAdjacentElement` method for a slotted node inside a scoped component. Specifically,
228
308
* 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
253
333
} ;
254
334
255
335
/**
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
257
337
*
258
338
* @param hostElementPrototype the `Element` to be patched
259
339
*/
@@ -315,17 +395,9 @@ export const patchChildSlotNodes = (elm: HTMLElement) => {
315
395
patchHostOriginalAccessor ( 'childNodes' , elm ) ;
316
396
Object . defineProperty ( elm , 'childNodes' , {
317
397
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 ;
329
401
} ,
330
402
} ) ;
331
403
} ;
@@ -344,11 +416,12 @@ export const patchChildSlotNodes = (elm: HTMLElement) => {
344
416
*
345
417
* @param node the slotted node to be patched
346
418
*/
347
- export const patchNextPrev = ( node : Node ) => {
419
+ export const patchSlottedNode = ( node : Node ) => {
348
420
if ( ! node || ( node as any ) . __nextSibling || ! globalThis . Node ) return ;
349
421
350
422
patchNextSibling ( node ) ;
351
423
patchPreviousSibling ( node ) ;
424
+ patchParentNode ( node ) ;
352
425
353
426
if ( node . nodeType === Node . ELEMENT_NODE ) {
354
427
patchNextElementSibling ( node as Element ) ;
@@ -360,7 +433,6 @@ export const patchNextPrev = (node: Node) => {
360
433
* Patches the `nextSibling` accessor of a non-shadow slotted node
361
434
*
362
435
* @param node the slotted node to be patched
363
- * Required during during testing / mock environnement.
364
436
*/
365
437
const patchNextSibling = ( node : Node ) => {
366
438
// already been patched? return
@@ -383,7 +455,6 @@ const patchNextSibling = (node: Node) => {
383
455
* Patches the `nextElementSibling` accessor of a non-shadow slotted node
384
456
*
385
457
* @param element the slotted element node to be patched
386
- * Required during during testing / mock environnement.
387
458
*/
388
459
const patchNextElementSibling = ( element : Element ) => {
389
460
if ( ! element || ( element as any ) . __nextElementSibling ) return ;
@@ -405,7 +476,6 @@ const patchNextElementSibling = (element: Element) => {
405
476
* Patches the `previousSibling` accessor of a non-shadow slotted node
406
477
*
407
478
* @param node the slotted node to be patched
408
- * Required during during testing / mock environnement.
409
479
*/
410
480
const patchPreviousSibling = ( node : Node ) => {
411
481
if ( ! node || ( node as any ) . __previousSibling ) return ;
@@ -427,7 +497,6 @@ const patchPreviousSibling = (node: Node) => {
427
497
* Patches the `previousElementSibling` accessor of a non-shadow slotted node
428
498
*
429
499
* @param element the slotted element node to be patched
430
- * Required during during testing / mock environnement.
431
500
*/
432
501
const patchPreviousElementSibling = ( element : Element ) => {
433
502
if ( ! element || ( element as any ) . __previousElementSibling ) return ;
@@ -446,6 +515,26 @@ const patchPreviousElementSibling = (element: Element) => {
446
515
} ) ;
447
516
} ;
448
517
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
+
449
538
/// UTILS ///
450
539
451
540
const validElementPatches = [ 'children' , 'nextElementSibling' , 'previousElementSibling' ] as const ;
@@ -456,6 +545,7 @@ const validNodesPatches = [
456
545
'nextSibling' ,
457
546
'previousSibling' ,
458
547
'textContent' ,
548
+ 'parentNode' ,
459
549
] as const ;
460
550
461
551
/**
@@ -481,3 +571,19 @@ function patchHostOriginalAccessor(
481
571
}
482
572
if ( accessor ) Object . defineProperty ( node , '__' + accessorName , accessor ) ;
483
573
}
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
+ }
0 commit comments