Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

chore: alterative functional templating syntax #15599

Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
Expand Up @@ -76,7 +76,7 @@ export function transform_template(state, context, namespace, template_name, fla
/** @type {Expression[]} */
const args = [
state.is_functional_template_mode
? template_to_functions(state.template, namespace)
? template_to_functions(state.template)
: b.template([b.quasi(template_to_string(state.template), true)], [])
];

Expand Down
Original file line number Diff line number Diff line change
@@ -1,56 +1,24 @@
/**
* @import { TemplateOperations } from "../types.js"
* @import { Namespace } from "#compiler"
* @import { CallExpression, Statement } from "estree"
* @import { CallExpression, Statement, ObjectExpression, Identifier, ArrayExpression, Property, Expression, Literal } from "estree"
*/
import { NAMESPACE_SVG, NAMESPACE_MATHML } from '../../../../../constants.js';
import * as b from '../../../../utils/builders.js';
import { regex_is_valid_identifier } from '../../../patterns.js';
import fix_attribute_casing from './fix-attribute-casing.js';

class Scope {
declared = new Map();

/**
* @param {string} _name
*/
generate(_name) {
let name = _name.replace(/[^a-zA-Z0-9_$]/g, '_').replace(/^[0-9]/, '_');
if (!this.declared.has(name)) {
this.declared.set(name, 1);
return name;
}
let count = this.declared.get(name);
this.declared.set(name, count + 1);
return `${name}_${count}`;
}
}

/**
* @param {TemplateOperations} items
* @param {Namespace} namespace
*/
export function template_to_functions(items, namespace) {
let elements = [];

let body = [];

let scope = new Scope();
export function template_to_functions(items) {
let elements = b.array([]);

/**
* @type {Array<Element>}
*/
let elements_stack = [];

/**
* @type {Array<string>}
*/
let namespace_stack = [];

/**
* @type {number}
*/
let foreign_object_count = 0;

/**
* @type {Element | undefined}
*/
Expand All @@ -71,199 +39,138 @@ export function template_to_functions(items, namespace) {
// we closed one element, we remove it from the stack and eventually revert back
// the namespace to the previous one
if (instruction.kind === 'pop_element') {
const removed = elements_stack.pop();
if (removed?.namespaced) {
namespace_stack.pop();
}
if (removed?.element === 'foreignObject') {
foreign_object_count--;
}
elements_stack.pop();
continue;
}

// if the inserted node is in the svg/mathml we push the namespace to the stack because we need to
// create with createElementNS
if (instruction.metadata?.svg || instruction.metadata?.mathml) {
namespace_stack.push(instruction.metadata.svg ? NAMESPACE_SVG : NAMESPACE_MATHML);
}

// @ts-expect-error we can't be here if `swap_current_element` but TS doesn't know that
const value = map[instruction.kind](
...[
// for set prop we need to send the last element (not the one in the stack since
// it get's added to the stack only after the push_element instruction)...for all the rest
// the first prop is a the scope to generate the name of the variable
...(instruction.kind === 'set_prop' ? [last_current_element] : [scope]),
// for create element we also need to add the namespace...namespaces in the stack get's precedence over
// the "global" namespace (and if we are in a foreignObject we default to html)
...(instruction.kind === 'create_element'
? [
foreign_object_count > 0
? undefined
: namespace_stack.at(-1) ??
(namespace === 'svg'
? NAMESPACE_SVG
: namespace === 'mathml'
? NAMESPACE_MATHML
: undefined)
]
: []),
? []
: [instruction.kind === 'set_prop' ? last_current_element : elements_stack.at(-1)]),
...(instruction.args ?? [])
]
);

if (value) {
// this will compose the body of the function
body.push(value.call);
}

// with set_prop we don't need to do anything else, in all other cases we also need to
// append the element/node/anchor to the current active element or push it in the elements array
if (instruction.kind !== 'set_prop') {
if (elements_stack.length >= 1 && value) {
const { call } = map.insert(/** @type {Element} */ (elements_stack.at(-1)), value);
body.push(call);
} else if (value) {
elements.push(b.id(value.name));
if (elements_stack.length >= 1 && value !== undefined) {
map.insert(/** @type {Element} */ (elements_stack.at(-1)), value);
} else if (value !== undefined) {
elements.elements.push(value);
}
// keep track of the last created element (it will be pushed to the stack after the props are set)
if (instruction.kind === 'create_element') {
last_current_element = /** @type {Element} */ (value);
if (last_current_element.element === 'foreignObject') {
foreign_object_count++;
}
}
}
}
// every function needs to return a fragment so we create one and push all the elements there
const fragment = scope.generate('fragment');
body.push(b.var(fragment, b.call('document.createDocumentFragment')));
body.push(b.call(fragment + '.append', ...elements));
body.push(b.return(b.id(fragment)));

return b.arrow([], b.block(body));
return elements;
}

/**
* @typedef {{ call: Statement, name: string, add_is: (value: string)=>void, namespaced: boolean; element: string; }} Element
* @typedef {ObjectExpression} Element
*/

/**
* @typedef {{ call: Statement, name: string }} Anchor
* @typedef {void | null | ArrayExpression} Anchor
*/

/**
* @typedef {{ call: Statement, name: string }} Text
* @typedef {void | Literal} Text
*/

/**
* @typedef { Element | Anchor| Text } Node
*/

/**
* @param {Scope} scope
* @param {Namespace} namespace
* @param {string} element
* @returns {Element}
*/
function create_element(scope, namespace, element) {
const name = scope.generate(element);
let fn = namespace != null ? 'document.createElementNS' : 'document.createElement';
let args = [b.literal(element)];
if (namespace != null) {
args.unshift(b.literal(namespace));
}
const call = b.var(name, b.call(fn, ...args));
/**
* if there's an "is" attribute we can't just add it as a property, it needs to be
* specified on creation like this `document.createElement('button', { is: 'my-button' })`
*
* Since the props are appended after the creation we change the generated call arguments and we push
* the is attribute later on on `set_prop`
* @param {string} value
*/
function add_is(value) {
/** @type {CallExpression} */ (call.declarations[0].init).arguments.push(
b.object([b.prop('init', b.literal('is'), b.literal(value))])
);
function create_element(element) {
return b.object([b.prop('init', b.id('e'), b.literal(element))]);
}

/**
*
* @param {Element} element
* @param {string} name
* @param {Expression} init
* @returns {Property}
*/
function get_or_create_prop(element, name, init) {
let prop = element.properties.find(
(prop) => prop.type === 'Property' && /** @type {Identifier} */ (prop.key).name === name
);
if (!prop) {
prop = b.prop('init', b.id(name), init);
element.properties.push(prop);
}
return {
call,
name,
element,
add_is,
namespaced: namespace != null
};
return /** @type {Property} */ (prop);
}

/**
* @param {Scope} scope
* @param {Element} element
* @param {string} data
* @returns {Anchor}
*/
function create_anchor(scope, data = '') {
const name = scope.generate('comment');
return {
call: b.var(name, b.call('document.createComment', b.literal(data))),
name
};
function create_anchor(element, data = '') {
if (!element) return data ? b.array([b.literal(data)]) : null;
const c = get_or_create_prop(element, 'c', b.array([]));
/** @type {ArrayExpression} */ (c.value).elements.push(data ? b.array([b.literal(data)]) : null);
}

/**
* @param {Scope} scope
* @param {Element} element
* @param {string} value
* @returns {Text}
*/
function create_text(scope, value) {
const name = scope.generate('text');
return {
call: b.var(name, b.call('document.createTextNode', b.literal(value))),
name
};
function create_text(element, value) {
if (!element) return b.literal(value);
const c = get_or_create_prop(element, 'c', b.array([]));
/** @type {ArrayExpression} */ (c.value).elements.push(b.literal(value));
}

/**
*
* @param {Element} el
* @param {Element} element
* @param {string} prop
* @param {string} value
*/
function set_prop(el, prop, value) {
// see comment above about the "is" attribute
function set_prop(element, prop, value) {
const p = get_or_create_prop(element, 'p', b.object([]));

if (prop === 'is') {
el.add_is(value);
element.properties.push(b.prop('init', b.id(prop), b.literal(value)));
return;
}

const [namespace] = prop.split(':');
let fn = namespace !== prop ? '.setAttributeNS' : '.setAttribute';
let args = [b.literal(fix_attribute_casing(prop)), b.literal(value ?? '')];
const prop_correct_case = fix_attribute_casing(prop);

// attributes like `xlink:href` need to be set with the `xlink` namespace
if (namespace === 'xlink') {
args.unshift(b.literal('http://www.w3.org/1999/xlink'));
}
const is_valid_id = regex_is_valid_identifier.test(prop_correct_case);

return {
call: b.call(el.name + fn, ...args)
};
/** @type {ObjectExpression} */ (p.value).properties.push(
b.prop(
'init',
(is_valid_id ? b.id : b.literal)(prop_correct_case),
b.literal(value),
!is_valid_id
)
);
}

/**
*
* @param {Element} el
* @param {Node} child
* @param {Node} [anchor]
* @param {Element} element
* @param {Element} child
*/
function insert(el, child, anchor) {
return {
call: b.call(
// if we have a template element we need to push into it's content rather than the element itself
el.name + (el.element === 'template' ? '.content' : '') + '.insertBefore',
b.id(child.name),
b.id(anchor?.name ?? 'undefined')
)
};
function insert(element, child) {
const c = get_or_create_prop(element, 'c', b.array([]));
/** @type {ArrayExpression} */ (c.value).elements.push(child);
}

let map = {
Expand Down
41 changes: 41 additions & 0 deletions packages/svelte/src/internal/client/dom/operations.js
Original file line number Diff line number Diff line change
Expand Up @@ -204,3 +204,44 @@ export function sibling(node, count = 1, is_text = false) {
export function clear_text_content(node) {
node.textContent = '';
}

/**
*
* @param {string} tag
* @param {string} [namespace]
* @param {string} [is]
* @returns
*/
export function create_element(tag, namespace, is) {
let options = is ? { is } : undefined;
if (namespace) {
return document.createElementNS(namespace, tag, options);
}
return document.createElement(tag, options);
}

export function create_fragment() {
return document.createDocumentFragment();
}

/**
* @param {string} data
* @returns
*/
export function create_comment(data = '') {
return document.createComment(data);
}

/**
* @param {Element} element
* @param {string} key
* @param {string} value
* @returns
*/
export function set_attribute(element, key, value = '') {
if (key.startsWith('xlink:')) {
element.setAttributeNS('http://www.w3.org/1999/xlink', key, value);
return;
}
return element.setAttribute(key, value);
}
Loading