@@ -30,22 +30,22 @@ export const VaporTeleportImpl = {
30
30
? new TeleportFragment ( 'teleport' )
31
31
: new TeleportFragment ( )
32
32
33
- const resolvedProps = new Proxy (
34
- props ,
35
- rawPropsProxyHandlers ,
36
- ) as any as TeleportProps
37
-
38
33
let children : Block
39
-
40
34
renderEffect ( ( ) => {
41
35
frag . updateChildren (
42
36
( children = slots . default && ( slots . default as BlockFn ) ( ) ) ,
43
37
)
44
38
} )
45
39
46
40
renderEffect ( ( ) => {
47
- // access the props to trigger tracking
48
- frag . update ( extend ( { } , resolvedProps ) , children ! )
41
+ frag . update (
42
+ // access the props to trigger tracking
43
+ extend (
44
+ { } ,
45
+ new Proxy ( props , rawPropsProxyHandlers ) as any as TeleportProps ,
46
+ ) ,
47
+ children ! ,
48
+ )
49
49
} )
50
50
51
51
return frag
@@ -54,89 +54,98 @@ export const VaporTeleportImpl = {
54
54
55
55
export class TeleportFragment extends VaporFragment {
56
56
anchor : Node
57
- targetStart ?: Node | null
58
- mainAnchor ?: Node
59
- placeholder ?: Node
60
- container ?: ParentNode | null
61
- currentAnchor ?: Node | null
57
+
58
+ private targetStart ?: Node
59
+ private mainAnchor ?: Node
60
+ private placeholder ?: Node
61
+ private mountContainer ?: ParentNode | null
62
+ private mountAnchor ?: Node | null
62
63
63
64
constructor ( anchorLabel ?: string ) {
64
65
super ( [ ] )
65
66
this . anchor =
66
67
__DEV__ && anchorLabel ? createComment ( anchorLabel ) : createTextNode ( )
67
68
}
68
69
69
- updateChildren ( children : Block ) : void {
70
- const parent = this . anchor . parentNode
71
- if ( ! parent ) return
70
+ get currentParent ( ) : ParentNode {
71
+ return ( this . mountContainer || this . parent ) !
72
+ }
72
73
73
- const container = this . container || parent
74
+ get currentAnchor ( ) : Node | null {
75
+ return this . mountAnchor || this . anchor
76
+ }
74
77
75
- // teardown previous
76
- remove ( this . nodes , container )
78
+ get parent ( ) : ParentNode | null {
79
+ return this . anchor . parentNode
80
+ }
81
+
82
+ updateChildren ( children : Block ) : void {
83
+ // not mounted yet, early return
84
+ if ( ! this . parent ) return
77
85
78
- insert (
79
- ( this . nodes = children ) ,
80
- container ,
81
- this . currentAnchor || this . anchor ,
82
- )
86
+ // teardown previous children
87
+ remove ( this . nodes , this . currentParent )
88
+
89
+ // mount new
90
+ insert ( ( this . nodes = children ) , this . currentParent , this . currentAnchor )
83
91
}
84
92
85
93
update ( props : TeleportProps , children : Block ) : void {
86
- const parent = this . anchor . parentNode
87
94
this . nodes = children
88
- const disabled = isTeleportDisabled ( props )
89
95
90
96
const mount = ( parent : ParentNode , anchor : Node | null ) => {
91
97
insert (
92
98
this . nodes ,
93
- ( this . container = parent ) ,
94
- ( this . currentAnchor = anchor ) ,
99
+ ( this . mountContainer = parent ) ,
100
+ ( this . mountAnchor = anchor ) ,
95
101
)
96
102
}
97
103
98
104
const mountToTarget = ( ) => {
99
105
const target = ( this . target = resolveTarget ( props , querySelector ) )
100
106
if ( target ) {
101
107
if (
102
- // initial mount
103
- ! this . targetStart ||
108
+ // initial mount into target
109
+ ! this . targetAnchor ||
104
110
// target changed
105
- this . targetStart . parentNode !== target
111
+ this . targetAnchor . parentNode !== target
106
112
) {
107
113
; [ this . targetAnchor , this . targetStart ] = prepareAnchor ( target )
108
114
}
109
115
110
116
mount ( target , this . targetAnchor ! )
111
117
} else if ( __DEV__ ) {
112
118
warn (
113
- `Invalid Teleport target on ${ this . targetStart ? 'update' : 'mount' } :` ,
119
+ `Invalid Teleport target on ${ this . targetAnchor ? 'update' : 'mount' } :` ,
114
120
target ,
115
121
`(${ typeof target } )` ,
116
122
)
117
123
}
118
124
}
119
125
120
- if ( parent && disabled ) {
121
- if ( ! this . mainAnchor ) {
122
- this . mainAnchor = __DEV__
123
- ? createComment ( 'teleport end' )
124
- : createTextNode ( )
125
- }
126
- if ( ! this . placeholder ) {
127
- this . placeholder = __DEV__
128
- ? createComment ( 'teleport start' )
129
- : createTextNode ( )
130
- }
131
- if ( ! this . mainAnchor . isConnected ) {
132
- insert ( this . placeholder , parent , this . anchor )
133
- insert ( this . mainAnchor , parent , this . anchor )
134
- }
126
+ // mount into main container
127
+ if ( isTeleportDisabled ( props ) ) {
128
+ if ( this . parent ) {
129
+ if ( ! this . mainAnchor ) {
130
+ this . mainAnchor = __DEV__
131
+ ? createComment ( 'teleport end' )
132
+ : createTextNode ( )
133
+ }
134
+ if ( ! this . placeholder ) {
135
+ this . placeholder = __DEV__
136
+ ? createComment ( 'teleport start' )
137
+ : createTextNode ( )
138
+ }
139
+ if ( ! this . mainAnchor . isConnected ) {
140
+ insert ( this . placeholder , this . parent , this . anchor )
141
+ insert ( this . mainAnchor , this . parent , this . anchor )
142
+ }
135
143
136
- mount ( parent , this . mainAnchor )
144
+ mount ( this . parent , this . mainAnchor )
145
+ }
137
146
}
138
-
139
- if ( ! disabled ) {
147
+ // mount into target container
148
+ else {
140
149
if ( isTeleportDeferred ( props ) ) {
141
150
queuePostFlushCb ( mountToTarget )
142
151
} else {
@@ -147,13 +156,12 @@ export class TeleportFragment extends VaporFragment {
147
156
148
157
remove = ( parent : ParentNode | undefined ) : void => {
149
158
// remove nodes
150
- remove ( this . nodes , this . container || parent )
159
+ remove ( this . nodes , this . currentParent )
151
160
152
161
// remove anchors
153
162
if ( this . targetStart ) {
154
- let parentNode = this . targetStart . parentNode !
155
- remove ( this . targetStart ! , parentNode )
156
- remove ( this . targetAnchor ! , parentNode )
163
+ remove ( this . targetStart ! , this . target ! )
164
+ remove ( this . targetAnchor ! , this . target ! )
157
165
}
158
166
if ( this . placeholder ) {
159
167
remove ( this . placeholder ! , parent )
@@ -167,12 +175,11 @@ export class TeleportFragment extends VaporFragment {
167
175
}
168
176
169
177
function prepareAnchor ( target : ParentNode | null ) {
170
- const targetStart = createTextNode ( '' )
178
+ const targetStart = createTextNode ( '' ) as Text & { [ TeleportEndKey ] : Node }
171
179
const targetAnchor = createTextNode ( '' )
172
180
173
181
// attach a special property, so we can skip teleported content in
174
182
// renderer's nextSibling search
175
- // @ts -expect-error
176
183
targetStart [ TeleportEndKey ] = targetAnchor
177
184
178
185
if ( target ) {
0 commit comments