1
1
import { BUILD } from '@app-data' ;
2
2
3
3
import type * as d from '../declarations' ;
4
+ import { internalCall } from './dom-extras' ;
4
5
import { NODE_TYPE } from './runtime-constants' ;
5
6
6
7
/**
7
8
* Adjust the `.hidden` property as-needed on any nodes in a DOM subtree which
8
- * are slot fallbacks nodes - `<slot-fb>...</slot-fb>`
9
+ * are slot fallback nodes - `<slot-fb>...</slot-fb>`
9
10
*
10
11
* A slot fallback node should be visible by default. Then, it should be
11
12
* conditionally hidden if:
@@ -17,15 +18,15 @@ import { NODE_TYPE } from './runtime-constants';
17
18
* @param elm the element of interest
18
19
*/
19
20
export const updateFallbackSlotVisibility = ( elm : d . RenderNode ) => {
20
- const childNodes : d . RenderNode [ ] = elm . __childNodes || ( elm . childNodes as any ) ;
21
+ const childNodes = internalCall ( elm , 'childNodes' ) ;
21
22
22
23
// is this is a stencil component?
23
24
if ( elm . tagName && elm . tagName . includes ( '-' ) && elm [ 's-cr' ] && elm . tagName !== 'SLOT-FB' ) {
24
25
// stencil component - try to find any slot fallback nodes
25
26
getHostSlotNodes ( childNodes as any , ( elm as HTMLElement ) . tagName ) . forEach ( ( slotNode ) => {
26
27
if ( slotNode . nodeType === NODE_TYPE . ElementNode && slotNode . tagName === 'SLOT-FB' ) {
27
28
// this is a slot fallback node
28
- if ( getHostSlotChildNodes ( slotNode , slotNode [ 's-sn' ] , false ) ?. length ) {
29
+ if ( getSlotChildSiblings ( slotNode , getSlotName ( slotNode ) , false ) ?. length ) {
29
30
// has slotted nodes, hide fallback
30
31
slotNode . hidden = true ;
31
32
} else {
@@ -35,8 +36,11 @@ export const updateFallbackSlotVisibility = (elm: d.RenderNode) => {
35
36
}
36
37
} ) ;
37
38
}
38
- for ( const childNode of childNodes ) {
39
- if ( childNode . nodeType === NODE_TYPE . ElementNode && ( childNode . __childNodes || childNode . childNodes ) . length ) {
39
+
40
+ let i = 0 ;
41
+ for ( i = 0 ; i < childNodes . length ; i ++ ) {
42
+ const childNode = childNodes [ i ] as d . RenderNode ;
43
+ if ( childNode . nodeType === NODE_TYPE . ElementNode && internalCall ( childNode , 'childNodes' ) . length ) {
40
44
// keep drilling down
41
45
updateFallbackSlotVisibility ( childNode ) ;
42
46
}
@@ -54,7 +58,7 @@ export const updateFallbackSlotVisibility = (elm: d.RenderNode) => {
54
58
* @returns An array of slotted reference nodes.
55
59
*/
56
60
export const getSlottedChildNodes = ( childNodes : NodeListOf < ChildNode > ) : d . PatchedSlotNode [ ] => {
57
- const result = [ ] ;
61
+ const result : d . PatchedSlotNode [ ] = [ ] ;
58
62
for ( let i = 0 ; i < childNodes . length ; i ++ ) {
59
63
const slottedNode = ( ( childNodes [ i ] as d . RenderNode ) [ 's-nr' ] as d . PatchedSlotNode ) || undefined ;
60
64
if ( slottedNode && slottedNode . isConnected ) {
@@ -71,7 +75,7 @@ export const getSlottedChildNodes = (childNodes: NodeListOf<ChildNode>): d.Patch
71
75
* @param slotName the name of the slot to match on.
72
76
* @returns a reference to the slot node that matches the provided name, `null` otherwise
73
77
*/
74
- export function getHostSlotNodes ( childNodes : NodeListOf < ChildNode > , hostName : string , slotName ?: string ) {
78
+ export function getHostSlotNodes ( childNodes : NodeListOf < ChildNode > , hostName ? : string , slotName ?: string ) {
75
79
let i = 0 ;
76
80
let slottedNodes : d . RenderNode [ ] = [ ] ;
77
81
let childNode : d . RenderNode ;
@@ -80,8 +84,8 @@ export function getHostSlotNodes(childNodes: NodeListOf<ChildNode>, hostName: st
80
84
childNode = childNodes [ i ] as any ;
81
85
if (
82
86
childNode [ 's-sr' ] &&
83
- childNode [ 's-hn' ] === hostName &&
84
- ( slotName === undefined || childNode [ 's-sn' ] === slotName )
87
+ ( ! hostName || childNode [ 's-hn' ] === hostName ) &&
88
+ ( slotName === undefined || getSlotName ( childNode ) === slotName )
85
89
) {
86
90
slottedNodes . push ( childNode ) ;
87
91
if ( typeof slotName !== 'undefined' ) return slottedNodes ;
@@ -92,18 +96,19 @@ export function getHostSlotNodes(childNodes: NodeListOf<ChildNode>, hostName: st
92
96
}
93
97
94
98
/**
95
- * Get slotted child nodes of a slot node
96
- * @param node - the slot node to get the child nodes from
99
+ * Get all ' child' sibling nodes of a slot node
100
+ * @param slot - the slot node to get the child nodes from
97
101
* @param slotName - the name of the slot to match on
98
102
* @param includeSlot - whether to include the slot node in the result
99
- * @returns slotted child nodes of the slot node
103
+ * @returns child nodes of the slot node
100
104
*/
101
- export const getHostSlotChildNodes = ( node : d . RenderNode , slotName : string , includeSlot = true ) => {
105
+ export const getSlotChildSiblings = ( slot : d . RenderNode , slotName : string , includeSlot = true ) => {
102
106
const childNodes : d . RenderNode [ ] = [ ] ;
103
- if ( ( includeSlot && node [ 's-sr' ] ) || ! node [ 's-sr' ] ) childNodes . push ( node as any ) ;
107
+ if ( ( includeSlot && slot [ 's-sr' ] ) || ! slot [ 's-sr' ] ) childNodes . push ( slot as any ) ;
108
+ let node = slot ;
104
109
105
- while ( ( node = node . nextSibling as any ) && ( node as d . RenderNode ) [ 's-sn' ] === slotName ) {
106
- childNodes . push ( node as any ) ;
110
+ while ( ( node = node . nextSibling as any ) ) {
111
+ if ( getSlotName ( node ) === slotName ) childNodes . push ( node as any ) ;
107
112
}
108
113
return childNodes ;
109
114
} ;
@@ -150,37 +155,34 @@ export const addSlotRelocateNode = (
150
155
prepend ?: boolean ,
151
156
position ?: number ,
152
157
) => {
153
- let slottedNodeLocation : d . RenderNode ;
154
-
155
- // does newChild already have a slot location node?
156
158
if ( newChild [ 's-ol' ] && newChild [ 's-ol' ] . isConnected ) {
157
- slottedNodeLocation = newChild [ 's-ol' ] ;
158
- } else {
159
- slottedNodeLocation = document . createTextNode ( '' ) as any ;
160
- slottedNodeLocation [ 's-nr' ] = newChild ;
159
+ // newChild already has a slot location node
160
+ return ;
161
161
}
162
162
163
+ const slottedNodeLocation = document . createTextNode ( '' ) as any ;
164
+ slottedNodeLocation [ 's-nr' ] = newChild ;
165
+
166
+ // if there's no content reference node, or parentNode we can't do anything
163
167
if ( ! slotNode [ 's-cr' ] || ! slotNode [ 's-cr' ] . parentNode ) return ;
164
168
165
169
const parent = slotNode [ 's-cr' ] . parentNode as any ;
166
- const appendMethod = prepend ? parent . __prepend || parent . prepend : parent . __appendChild || parent . appendChild ;
167
-
168
- if ( typeof position !== 'undefined' ) {
169
- if ( BUILD . hydrateClientSide ) {
170
- slottedNodeLocation [ 's-oo' ] = position ;
171
- const childNodes = ( parent . __childNodes || parent . childNodes ) as NodeListOf < d . RenderNode > ;
172
- const slotRelocateNodes : d . RenderNode [ ] = [ slottedNodeLocation ] ;
173
- childNodes . forEach ( ( n ) => {
174
- if ( n [ 's-nr' ] ) slotRelocateNodes . push ( n ) ;
175
- } ) ;
170
+ const appendMethod = prepend ? internalCall ( parent , 'prepend' ) : internalCall ( parent , 'appendChild' ) ;
176
171
177
- slotRelocateNodes . sort ( ( a , b ) => {
178
- if ( ! a [ 's-oo' ] || a [ 's-oo' ] < b [ 's-oo' ] ) return - 1 ;
179
- else if ( ! b [ 's-oo' ] || b [ 's-oo' ] < a [ 's-oo' ] ) return 1 ;
180
- return 0 ;
181
- } ) ;
182
- slotRelocateNodes . forEach ( ( n ) => appendMethod . call ( parent , n ) ) ;
183
- }
172
+ if ( BUILD . hydrateClientSide && typeof position !== 'undefined' ) {
173
+ slottedNodeLocation [ 's-oo' ] = position ;
174
+ const childNodes = internalCall ( parent , 'childNodes' ) as NodeListOf < d . RenderNode > ;
175
+ const slotRelocateNodes : d . RenderNode [ ] = [ slottedNodeLocation ] ;
176
+ childNodes . forEach ( ( n ) => {
177
+ if ( n [ 's-nr' ] ) slotRelocateNodes . push ( n ) ;
178
+ } ) ;
179
+
180
+ slotRelocateNodes . sort ( ( a , b ) => {
181
+ if ( ! a [ 's-oo' ] || a [ 's-oo' ] < ( b [ 's-oo' ] || 0 ) ) return - 1 ;
182
+ else if ( ! b [ 's-oo' ] || b [ 's-oo' ] < a [ 's-oo' ] ) return 1 ;
183
+ return 0 ;
184
+ } ) ;
185
+ slotRelocateNodes . forEach ( ( n ) => appendMethod . call ( parent , n ) ) ;
184
186
} else {
185
187
appendMethod . call ( parent , slottedNodeLocation ) ;
186
188
}
@@ -190,4 +192,75 @@ export const addSlotRelocateNode = (
190
192
} ;
191
193
192
194
export const getSlotName = ( node : d . PatchedSlotNode ) =>
193
- node [ 's-sn' ] || ( node . nodeType === 1 && ( node as Element ) . getAttribute ( 'slot' ) ) || '' ;
195
+ typeof node [ 's-sn' ] === 'string'
196
+ ? node [ 's-sn' ]
197
+ : ( node . nodeType === 1 && ( node as Element ) . getAttribute ( 'slot' ) ) || undefined ;
198
+
199
+ /**
200
+ * Add `assignedElements` and `assignedNodes` methods on a fake slot node
201
+ *
202
+ * @param node - slot node to patch
203
+ */
204
+ export function patchSlotNode ( node : d . RenderNode ) {
205
+ if ( ( node as any ) . assignedElements || ( node as any ) . assignedNodes || ! node [ 's-sr' ] ) return ;
206
+
207
+ const assignedFactory = ( elementsOnly : boolean ) =>
208
+ function ( opts ?: { flatten : boolean } ) {
209
+ const toReturn : d . RenderNode [ ] = [ ] ;
210
+ const slotName = this [ 's-sn' ] ;
211
+
212
+ if ( opts ?. flatten ) {
213
+ console . error ( `
214
+ Flattening is not supported for Stencil non-shadow slots.
215
+ You can use \`.childNodes\` to nested slot fallback content.
216
+ If you have a particular use case, please open an issue on the Stencil repo.
217
+ ` ) ;
218
+ }
219
+
220
+ const parent = this [ 's-cr' ] . parentElement as d . RenderNode ;
221
+ // get all light dom nodes
222
+ const slottedNodes = parent . __childNodes ? parent . childNodes : getSlottedChildNodes ( parent . childNodes ) ;
223
+
224
+ ( slottedNodes as d . RenderNode [ ] ) . forEach ( ( n ) => {
225
+ // find all the nodes assigned to slots we care about
226
+ if ( slotName === getSlotName ( n ) ) {
227
+ toReturn . push ( n ) ;
228
+ }
229
+ } ) ;
230
+
231
+ if ( elementsOnly ) {
232
+ return toReturn . filter ( ( n ) => n . nodeType === NODE_TYPE . ElementNode ) ;
233
+ }
234
+ return toReturn ;
235
+ } . bind ( node ) ;
236
+
237
+ ( node as any ) . assignedElements = assignedFactory ( true ) ;
238
+ ( node as any ) . assignedNodes = assignedFactory ( false ) ;
239
+ }
240
+
241
+ /**
242
+ * Dispatches a `slotchange` event on a fake `<slot />` node.
243
+ *
244
+ * @param elm the slot node to dispatch the event from
245
+ */
246
+ export function dispatchSlotChangeEvent ( elm : d . RenderNode ) {
247
+ elm . dispatchEvent ( new CustomEvent ( 'slotchange' , { bubbles : false , cancelable : false , composed : false } ) ) ;
248
+ }
249
+
250
+ /**
251
+ * Find the slot node that a slotted node belongs to
252
+ *
253
+ * @param slottedNode - the slotted node to find the slot for
254
+ * @param parentHost - the parent host element of the slotted node
255
+ * @returns the slot node and slot name
256
+ */
257
+ export function findSlotFromSlottedNode ( slottedNode : d . PatchedSlotNode , parentHost ?: HTMLElement ) {
258
+ parentHost = parentHost || slottedNode [ 's-ol' ] ?. parentElement ;
259
+
260
+ if ( ! parentHost ) return { slotNode : null , slotName : '' } ;
261
+
262
+ const slotName = ( slottedNode [ 's-sn' ] = getSlotName ( slottedNode ) || '' ) ;
263
+ const childNodes = internalCall ( parentHost , 'childNodes' ) ;
264
+ const slotNode = getHostSlotNodes ( childNodes , parentHost . tagName , slotName ) [ 0 ] ;
265
+ return { slotNode, slotName } ;
266
+ }
0 commit comments