1
1
/**
2
2
* @import { TemplateOperations } from "../types.js"
3
3
* @import { Namespace } from "#compiler"
4
- * @import { Statement } from "estree"
4
+ * @import { CallExpression, Statement } from "estree"
5
5
*/
6
6
import { NAMESPACE_SVG } from 'svelte/internal/client' ;
7
- import * as b from '../../../../utils/builders.js' ;
8
7
import { NAMESPACE_MATHML } from '../../../../../constants.js' ;
8
+ import * as b from '../../../../utils/builders.js' ;
9
+ import fix_attribute_casing from './fix-attribute-casing.js' ;
9
10
10
11
class Scope {
11
12
declared = new Map ( ) ;
@@ -28,9 +29,8 @@ class Scope {
28
29
/**
29
30
* @param {TemplateOperations } items
30
31
* @param {Namespace } namespace
31
- * @param {boolean } use_fragment
32
32
*/
33
- export function template_to_functions ( items , namespace , use_fragment = false ) {
33
+ export function template_to_functions ( items , namespace ) {
34
34
let elements = [ ] ;
35
35
36
36
let body = [ ] ;
@@ -42,26 +42,61 @@ export function template_to_functions(items, namespace, use_fragment = false) {
42
42
*/
43
43
let elements_stack = [ ] ;
44
44
45
+ /**
46
+ * @type {Array<string> }
47
+ */
48
+ let namespace_stack = [ ] ;
49
+
50
+ /**
51
+ * @type {number }
52
+ */
53
+ let foreign_object_count = 0 ;
54
+
45
55
/**
46
56
* @type {Element | undefined }
47
57
*/
48
58
let last_current_element ;
49
59
60
+ if ( items [ 0 ] . kind === 'create_anchor' ) {
61
+ items . unshift ( { kind : 'create_anchor' } ) ;
62
+ }
63
+
50
64
for ( let instruction of items ) {
51
65
if ( instruction . kind === 'push_element' && last_current_element ) {
52
66
elements_stack . push ( last_current_element ) ;
53
67
continue ;
54
68
}
55
69
if ( instruction . kind === 'pop_element' ) {
56
- elements_stack . pop ( ) ;
70
+ const removed = elements_stack . pop ( ) ;
71
+ if ( removed ?. namespaced ) {
72
+ namespace_stack . pop ( ) ;
73
+ }
74
+ if ( removed ?. element === 'foreignObject' ) {
75
+ foreign_object_count -- ;
76
+ }
57
77
continue ;
58
78
}
59
79
80
+ if ( instruction . metadata ?. svg || instruction . metadata ?. mathml ) {
81
+ namespace_stack . push ( instruction . metadata . svg ? NAMESPACE_SVG : NAMESPACE_MATHML ) ;
82
+ }
83
+
60
84
// @ts -expect-error we can't be here if `swap_current_element` but TS doesn't know that
61
85
const value = map [ instruction . kind ] (
62
86
...[
63
87
...( instruction . kind === 'set_prop' ? [ last_current_element ] : [ scope ] ) ,
64
- ...( instruction . kind === 'create_element' ? [ namespace ] : [ ] ) ,
88
+ ...( instruction . kind === 'create_element'
89
+ ? [
90
+ foreign_object_count > 0
91
+ ? undefined
92
+ : namespace_stack . at ( - 1 ) ??
93
+ ( namespace === 'svg'
94
+ ? NAMESPACE_SVG
95
+ : namespace === 'mathml'
96
+ ? NAMESPACE_MATHML
97
+ : undefined )
98
+ ]
99
+ : [ ] ) ,
65
100
...( instruction . args ?? [ ] )
66
101
]
67
102
) ;
@@ -79,23 +114,22 @@ export function template_to_functions(items, namespace, use_fragment = false) {
79
114
}
80
115
if ( instruction . kind === 'create_element' ) {
81
116
last_current_element = /** @type {Element } */ ( value ) ;
117
+ if ( last_current_element . element === 'foreignObject' ) {
118
+ foreign_object_count ++ ;
119
+ }
82
120
}
83
121
}
84
122
}
85
- if ( elements . length > 1 || use_fragment ) {
86
- const fragment = scope . generate ( 'fragment' ) ;
87
- body . push ( b . var ( fragment , b . call ( 'document.createDocumentFragment' ) ) ) ;
88
- body . push ( b . call ( fragment + '.append' , ...elements ) ) ;
89
- body . push ( b . return ( b . id ( fragment ) ) ) ;
90
- } else {
91
- body . push ( b . return ( elements [ 0 ] ) ) ;
92
- }
123
+ const fragment = scope . generate ( 'fragment' ) ;
124
+ body . push ( b . var ( fragment , b . call ( 'document.createDocumentFragment' ) ) ) ;
125
+ body . push ( b . call ( fragment + '.append' , ...elements ) ) ;
126
+ body . push ( b . return ( b . id ( fragment ) ) ) ;
93
127
94
128
return b . arrow ( [ ] , b . block ( body ) ) ;
95
129
}
96
130
97
131
/**
98
- * @typedef {{ call: Statement, name: string } } Element
132
+ * @typedef {{ call: Statement, name: string, add_is: (value: string)=>void, namespaced: boolean; element: string; } } Element
99
133
*/
100
134
101
135
/**
@@ -118,14 +152,26 @@ export function template_to_functions(items, namespace, use_fragment = false) {
118
152
*/
119
153
function create_element ( scope , namespace , element ) {
120
154
const name = scope . generate ( element ) ;
121
- let fn = namespace !== 'html' ? 'document.createElementNS' : 'document.createElement' ;
155
+ let fn = namespace != null ? 'document.createElementNS' : 'document.createElement' ;
122
156
let args = [ b . literal ( element ) ] ;
123
- if ( namespace !== 'html' ) {
124
- args . unshift ( namespace === 'svg' ? b . literal ( NAMESPACE_SVG ) : b . literal ( NAMESPACE_MATHML ) ) ;
157
+ if ( namespace != null ) {
158
+ args . unshift ( b . literal ( namespace ) ) ;
159
+ }
160
+ const call = b . var ( name , b . call ( fn , ...args ) ) ;
161
+ /**
162
+ * @param {string } value
163
+ */
164
+ function add_is ( value ) {
165
+ /** @type {CallExpression } */ ( call . declarations [ 0 ] . init ) . arguments . push (
166
+ b . object ( [ b . prop ( 'init' , b . literal ( 'is' ) , b . literal ( value ) ) ] )
167
+ ) ;
125
168
}
126
169
return {
127
- call : b . var ( name , b . call ( fn , ...args ) ) ,
128
- name
170
+ call,
171
+ name,
172
+ element,
173
+ add_is,
174
+ namespaced : namespace != null
129
175
} ;
130
176
}
131
177
@@ -162,8 +208,21 @@ function create_text(scope, value) {
162
208
* @param {string } value
163
209
*/
164
210
function set_prop ( el , prop , value ) {
211
+ if ( prop === 'is' ) {
212
+ el . add_is ( value ) ;
213
+ return ;
214
+ }
215
+
216
+ const [ namespace ] = prop . split ( ':' ) ;
217
+ let fn = namespace !== prop ? '.setAttributeNS' : '.setAttribute' ;
218
+ let args = [ b . literal ( fix_attribute_casing ( prop ) ) , b . literal ( value ?? '' ) ] ;
219
+
220
+ if ( namespace === 'xlink' ) {
221
+ args . unshift ( b . literal ( 'http://www.w3.org/1999/xlink' ) ) ;
222
+ }
223
+
165
224
return {
166
- call : b . call ( el . name + '.setAttribute' , b . literal ( prop ) , b . literal ( value ) )
225
+ call : b . call ( el . name + fn , ... args )
167
226
} ;
168
227
}
169
228
@@ -175,7 +234,11 @@ function set_prop(el, prop, value) {
175
234
*/
176
235
function insert ( el , child , anchor ) {
177
236
return {
178
- call : b . call ( el . name + '.insertBefore' , b . id ( child . name ) , b . id ( anchor ?. name ?? 'undefined' ) )
237
+ call : b . call (
238
+ el . name + ( el . element === 'template' ? '.content' : '' ) + '.insertBefore' ,
239
+ b . id ( child . name ) ,
240
+ b . id ( anchor ?. name ?? 'undefined' )
241
+ )
179
242
} ;
180
243
}
181
244
0 commit comments