Skip to content

Commit 7c7a857

Browse files
committed
feat: make all tests pass with functional templates
1 parent 8a737f1 commit 7c7a857

File tree

13 files changed

+328
-71
lines changed

13 files changed

+328
-71
lines changed

.changeset/smart-boats-accept.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -2,4 +2,4 @@
22
'svelte': minor
33
---
44

5-
feat: templateless template generation
5+
feat: functional template generation

packages/svelte/src/compiler/phases/3-transform/client/transform-template/fix-attribute-casing.js

+18
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

packages/svelte/src/compiler/phases/3-transform/client/transform-template/index.js

+16-26
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,6 @@
44
* @import { AST, Namespace } from '#compiler'
55
* @import { SourceLocation } from '#shared'
66
*/
7-
import { TEMPLATE_FRAGMENT } from '../../../../../constants.js';
87
import { dev } from '../../../../state.js';
98
import * as b from '../../../../utils/builders.js';
109
import { template_to_functions } from './to-functions.js';
@@ -18,15 +17,17 @@ import { template_to_string } from './to-string.js';
1817
*/
1918
function get_template_function(namespace, state) {
2019
const contains_script_tag = state.metadata.context.template_contains_script_tag;
21-
return namespace === 'svg'
22-
? contains_script_tag
23-
? '$.svg_template_with_script'
24-
: '$.ns_template'
25-
: namespace === 'mathml'
26-
? '$.mathml_template'
27-
: contains_script_tag
28-
? '$.template_with_script'
29-
: '$.template';
20+
return (
21+
namespace === 'svg'
22+
? contains_script_tag
23+
? '$.svg_template_with_script'
24+
: '$.ns_template'
25+
: namespace === 'mathml'
26+
? '$.mathml_template'
27+
: contains_script_tag
28+
? '$.template_with_script'
29+
: '$.template'
30+
).concat(state.prevent_template_cloning ? '_fn' : '');
3031
}
3132

3233
/**
@@ -54,21 +55,6 @@ function build_locations(locations) {
5455
* @param {number} [flags]
5556
*/
5657
export function transform_template(state, context, namespace, template_name, flags) {
57-
if (context.state.prevent_template_cloning) {
58-
context.state.hoisted.push(
59-
b.var(
60-
template_name,
61-
template_to_functions(
62-
state.template,
63-
namespace,
64-
flags != null && (flags & TEMPLATE_FRAGMENT) !== 0
65-
)
66-
)
67-
);
68-
69-
return;
70-
}
71-
7258
/**
7359
* @param {Identifier} template_name
7460
* @param {Expression[]} args
@@ -88,7 +74,11 @@ export function transform_template(state, context, namespace, template_name, fla
8874
};
8975

9076
/** @type {Expression[]} */
91-
const args = [b.template([b.quasi(template_to_string(state.template), true)], [])];
77+
const args = [
78+
state.prevent_template_cloning
79+
? template_to_functions(state.template, namespace)
80+
: b.template([b.quasi(template_to_string(state.template), true)], [])
81+
];
9282

9383
if (flags) {
9484
args.push(b.literal(flags));

packages/svelte/src/compiler/phases/3-transform/client/transform-template/to-functions.js

+85-22
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,12 @@
11
/**
22
* @import { TemplateOperations } from "../types.js"
33
* @import { Namespace } from "#compiler"
4-
* @import { Statement } from "estree"
4+
* @import { CallExpression, Statement } from "estree"
55
*/
66
import { NAMESPACE_SVG } from 'svelte/internal/client';
7-
import * as b from '../../../../utils/builders.js';
87
import { NAMESPACE_MATHML } from '../../../../../constants.js';
8+
import * as b from '../../../../utils/builders.js';
9+
import fix_attribute_casing from './fix-attribute-casing.js';
910

1011
class Scope {
1112
declared = new Map();
@@ -28,9 +29,8 @@ class Scope {
2829
/**
2930
* @param {TemplateOperations} items
3031
* @param {Namespace} namespace
31-
* @param {boolean} use_fragment
3232
*/
33-
export function template_to_functions(items, namespace, use_fragment = false) {
33+
export function template_to_functions(items, namespace) {
3434
let elements = [];
3535

3636
let body = [];
@@ -42,26 +42,61 @@ export function template_to_functions(items, namespace, use_fragment = false) {
4242
*/
4343
let elements_stack = [];
4444

45+
/**
46+
* @type {Array<string>}
47+
*/
48+
let namespace_stack = [];
49+
50+
/**
51+
* @type {number}
52+
*/
53+
let foreign_object_count = 0;
54+
4555
/**
4656
* @type {Element | undefined}
4757
*/
4858
let last_current_element;
4959

60+
if (items[0].kind === 'create_anchor') {
61+
items.unshift({ kind: 'create_anchor' });
62+
}
63+
5064
for (let instruction of items) {
5165
if (instruction.kind === 'push_element' && last_current_element) {
5266
elements_stack.push(last_current_element);
5367
continue;
5468
}
5569
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+
}
5777
continue;
5878
}
5979

80+
if (instruction.metadata?.svg || instruction.metadata?.mathml) {
81+
namespace_stack.push(instruction.metadata.svg ? NAMESPACE_SVG : NAMESPACE_MATHML);
82+
}
83+
6084
// @ts-expect-error we can't be here if `swap_current_element` but TS doesn't know that
6185
const value = map[instruction.kind](
6286
...[
6387
...(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+
: []),
65100
...(instruction.args ?? [])
66101
]
67102
);
@@ -79,23 +114,22 @@ export function template_to_functions(items, namespace, use_fragment = false) {
79114
}
80115
if (instruction.kind === 'create_element') {
81116
last_current_element = /** @type {Element} */ (value);
117+
if (last_current_element.element === 'foreignObject') {
118+
foreign_object_count++;
119+
}
82120
}
83121
}
84122
}
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)));
93127

94128
return b.arrow([], b.block(body));
95129
}
96130

97131
/**
98-
* @typedef {{ call: Statement, name: string }} Element
132+
* @typedef {{ call: Statement, name: string, add_is: (value: string)=>void, namespaced: boolean; element: string; }} Element
99133
*/
100134

101135
/**
@@ -118,14 +152,26 @@ export function template_to_functions(items, namespace, use_fragment = false) {
118152
*/
119153
function create_element(scope, namespace, element) {
120154
const name = scope.generate(element);
121-
let fn = namespace !== 'html' ? 'document.createElementNS' : 'document.createElement';
155+
let fn = namespace != null ? 'document.createElementNS' : 'document.createElement';
122156
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+
);
125168
}
126169
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
129175
};
130176
}
131177

@@ -162,8 +208,21 @@ function create_text(scope, value) {
162208
* @param {string} value
163209
*/
164210
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+
165224
return {
166-
call: b.call(el.name + '.setAttribute', b.literal(prop), b.literal(value))
225+
call: b.call(el.name + fn, ...args)
167226
};
168227
}
169228

@@ -175,7 +234,11 @@ function set_prop(el, prop, value) {
175234
*/
176235
function insert(el, child, anchor) {
177236
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+
)
179242
};
180243
}
181244

packages/svelte/src/compiler/phases/3-transform/client/types.d.ts

+4
Original file line numberDiff line numberDiff line change
@@ -50,6 +50,10 @@ type TemplateOperationsKind =
5050
type TemplateOperations = Array<{
5151
kind: TemplateOperationsKind;
5252
args?: Array<string>;
53+
metadata?: {
54+
svg: boolean;
55+
mathml: boolean;
56+
};
5357
}>;
5458
export interface ComponentClientTransformState extends ClientTransformState {
5559
readonly analysis: ComponentAnalysis;

packages/svelte/src/compiler/phases/3-transform/client/visitors/Fragment.js

+26-7
Original file line numberDiff line numberDiff line change
@@ -36,7 +36,8 @@ export function Fragment(node, context) {
3636
namespace,
3737
context.state,
3838
context.state.preserve_whitespace,
39-
context.state.options.preserveComments
39+
context.state.options.preserveComments,
40+
context.state.prevent_template_cloning
4041
);
4142

4243
if (hoisted.length === 0 && trimmed.length === 0) {
@@ -124,22 +125,40 @@ export function Fragment(node, context) {
124125
// special case — we can use `$.text` instead of creating a unique template
125126
const id = b.id(context.state.scope.generate('text'));
126127

127-
process_children(trimmed, () => id, false, {
128-
...context,
129-
state
130-
});
128+
process_children(
129+
trimmed,
130+
() => id,
131+
false,
132+
{
133+
...context,
134+
state
135+
},
136+
context.state.prevent_template_cloning
137+
);
131138

132139
body.push(b.var(id, b.call('$.text')));
133140
close = b.stmt(b.call('$.append', b.id('$$anchor'), id));
134141
} else {
135142
if (is_standalone) {
136143
// no need to create a template, we can just use the existing block's anchor
137-
process_children(trimmed, () => b.id('$$anchor'), false, { ...context, state });
144+
process_children(
145+
trimmed,
146+
() => b.id('$$anchor'),
147+
false,
148+
{ ...context, state },
149+
context.state.prevent_template_cloning
150+
);
138151
} else {
139152
/** @type {(is_text: boolean) => Expression} */
140153
const expression = (is_text) => b.call('$.first_child', id, is_text && b.true);
141154

142-
process_children(trimmed, expression, false, { ...context, state });
155+
process_children(
156+
trimmed,
157+
expression,
158+
false,
159+
{ ...context, state },
160+
context.state.prevent_template_cloning
161+
);
143162

144163
let flags = TEMPLATE_FRAGMENT;
145164

0 commit comments

Comments
 (0)