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

[feature] Dynamic elements implementation <svelte:element> #6898

Merged
merged 73 commits into from
Apr 8, 2022
Merged
Show file tree
Hide file tree
Changes from 72 commits
Commits
Show all changes
73 commits
Select commit Hold shift + click to select a range
f001107
Implement svelte:element for dynamically setting HTML DOM type
Sep 25, 2020
4fcc681
Add add_css_class method to DynamicElement
Oct 7, 2020
54e1c23
Reuse DOM elements if possible. Add dev warnings when passing nullish…
Mar 31, 2021
392bdc6
Add documentation
Mar 31, 2021
7d308cb
Merge branch 'master' into dynamic-elements-implementation
Sep 26, 2021
bd41b85
tag -> this
Sep 26, 2021
45d3ab0
fix: css classes for dynamic elements
Oct 21, 2021
a6a8ffd
Merge remote-tracking branch 'upstream/master' into dynamic-elements-…
baseballyama Oct 23, 2021
80e93bc
replace double quote to single quote
baseballyama Oct 23, 2021
7b5247d
fix lint
baseballyama Oct 23, 2021
185332e
clean up nodes
baseballyama Oct 26, 2021
714433b
little clean up render_dom
baseballyama Oct 27, 2021
89d2733
refactor tag.ts
baseballyama Oct 28, 2021
14ca990
refactor nodes
baseballyama Oct 30, 2021
ed2eed5
refactor render_dom
baseballyama Oct 30, 2021
5d1bc8b
refactor render_dom
baseballyama Oct 31, 2021
84d9f83
refactor render_dom
baseballyama Oct 31, 2021
d65da8d
add tests
baseballyama Oct 31, 2021
7ced0ce
refactor render_ssr
baseballyama Oct 31, 2021
24a338a
refactor render_ssr
baseballyama Oct 31, 2021
ca63111
fix named slot bug
baseballyama Oct 31, 2021
85eab20
rafactor parse step
baseballyama Oct 31, 2021
e3c0091
refactor render_ssr
baseballyama Oct 31, 2021
195452e
refactor render_ssr
baseballyama Oct 31, 2021
8c9e93b
refactor parse step
baseballyama Nov 3, 2021
a86ec67
throw error if <svelte:element> uses animation
baseballyama Nov 3, 2021
89dd933
move DynamicElementWrapper folder
baseballyama Nov 3, 2021
f8cc52a
merge DynamicElement to Element
baseballyama Nov 3, 2021
e2a36a4
Reuse DOM elements if possible.
baseballyama Nov 3, 2021
75621dc
revert useless changes
baseballyama Nov 3, 2021
42e4656
refactor
baseballyama Nov 6, 2021
d223df6
stop to use dynamic_element_block
baseballyama Nov 6, 2021
cdd7180
refactor render_dom
baseballyama Nov 6, 2021
9068c90
refactor render_dom
baseballyama Nov 6, 2021
9cd3fc6
refactor render_dom
baseballyama Nov 6, 2021
c77b73c
improve transition behavior
baseballyama Nov 6, 2021
f51d0f5
refactor render_dom
baseballyama Nov 6, 2021
5160b2b
support animation
baseballyama Nov 6, 2021
da22d63
revert useless changes
baseballyama Nov 6, 2021
11d12ab
use getter
baseballyama Nov 6, 2021
25a12a7
refactor
baseballyama Nov 6, 2021
9d76a92
make it more correct of claim_element
baseballyama Nov 6, 2021
9863c5e
refactor
baseballyama Nov 6, 2021
b34b0ca
stop to use anonymous function
baseballyama Nov 7, 2021
49bef3a
improve performance of render_ssr
baseballyama Nov 7, 2021
6788b47
use child_dynamic_element
baseballyama Nov 13, 2021
28b2665
Revert "support animation"
baseballyama Nov 13, 2021
3a96f18
fix compile error
baseballyama Nov 13, 2021
2e22241
refactor
baseballyama Nov 13, 2021
7c0337a
stop reuse children of <svelte:element>
baseballyama Nov 13, 2021
7b6dbcd
revert add_transitions
baseballyama Nov 13, 2021
8560d7c
revert unnecessary changes
baseballyama Nov 13, 2021
2171c67
fix in transition when tag of <svelte:element> changed
baseballyama Nov 28, 2021
d146c29
call of action when <svelte:element> tag changed
baseballyama Nov 28, 2021
7a832d6
stop to transform to normal element even if svelte:element's value of…
baseballyama Nov 28, 2021
fa1b22a
Merge branch 'master' into dynamic-elements-implementation
baseballyama Jan 8, 2022
b437985
refactor redundant if statement
baseballyama Jan 8, 2022
5850155
prevent to create element if value of `this` attribute is null / unde…
baseballyama Jan 9, 2022
dc25e2e
Merge branch 'master' into dynamic-elements-implementation
baseballyama Feb 12, 2022
02cc14f
workaround ts error
baseballyama Feb 12, 2022
6c71959
Merge branch 'master' into dynamic-elements-implementation
Apr 1, 2022
a8d7905
refactor update block, support animations, support falsy values, adju…
Apr 1, 2022
024a748
does not always need to create a new anchor
tanhauhau Apr 6, 2022
ef8a3f9
validate tag variable in dev mode
tanhauhau Apr 6, 2022
4965010
support template literals
baseballyama Apr 6, 2022
31573fd
add test for validating tag definition
baseballyama Apr 6, 2022
c5c0dee
remove useless validation
baseballyama Apr 6, 2022
2858cf5
add test for shorthand this
baseballyama Apr 6, 2022
1e86df5
add more parser validation test
tanhauhau Apr 7, 2022
f4a8e79
improve testcases
tanhauhau Apr 7, 2022
0ef1dfc
animate while switching dynamic element
tanhauhau Apr 7, 2022
13c6c12
fix test
tanhauhau Apr 7, 2022
8057a14
add tutorial section
Apr 8, 2022
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
22 changes: 22 additions & 0 deletions site/content/docs/02-template-syntax.md
Original file line number Diff line number Diff line change
Expand Up @@ -1627,6 +1627,28 @@ If `this` is falsy, no component is rendered.
<svelte:component this={currentSelection.component} foo={bar}/>
```

### `<svelte:element>`

```sv
<svelte:element this={expression}/>
```

---

The `<svelte:element>` element lets you render an element of a dynamically specified type. This is useful for example when rich text content from a CMS. If the tag is changed, the children will be preserved unless there's a transition attached to the element. Any properties and event listeners present will be applied to the element.

The only supported binding is `bind:this`, since the element type specific bindings that Svelte does at build time (e.g. `bind:value` for input elements) does not work with a dynamic tag type.

If `this` has a nullish value, a warning will be logged in development mode.

```sv
<script>
let tag = 'div';
export let handler;
</script>

<svelte:element this={tag} on:click={handler}>Foo</svelte:element>
```

### `<svelte:window>`

Expand Down
4 changes: 4 additions & 0 deletions src/compiler/compile/compiler_errors.ts
Original file line number Diff line number Diff line change
Expand Up @@ -246,6 +246,10 @@ export default {
code: 'invalid-animation',
message: 'An element that uses the animate directive must be the sole child of a keyed each block'
},
invalid_animation_dynamic_element: {
code: 'invalid-animation',
message: '<svelte:element> cannot have a animate directive'
},
invalid_directive_value: {
code: 'invalid-directive-value',
message: 'Can only bind to an identifier (e.g. `foo`) or a member expression (e.g. `foo.bar` or `foo[baz]`)'
Expand Down
18 changes: 18 additions & 0 deletions src/compiler/compile/nodes/Element.ts
Original file line number Diff line number Diff line change
Expand Up @@ -18,6 +18,9 @@ import Let from './Let';
import TemplateScope from './shared/TemplateScope';
import { INode } from './interfaces';
import Component from '../Component';
import Expression from './shared/Expression';
import { string_literal } from '../utils/stringify';
import { Literal } from 'estree';
import compiler_warnings from '../compiler_warnings';
import compiler_errors from '../compiler_errors';

Expand Down Expand Up @@ -190,11 +193,26 @@ export default class Element extends Node {
children: INode[];
namespace: string;
needs_manual_style_scoping: boolean;
tag_expr: Expression;

get is_dynamic_element() {
return this.name === 'svelte:element';
}

constructor(component: Component, parent: Node, scope: TemplateScope, info: any) {
super(component, parent, scope, info);
this.name = info.name;

if (info.name === 'svelte:element') {
if (typeof info.tag !== 'string') {
this.tag_expr = new Expression(component, this, scope, info.tag);
} else {
this.tag_expr = new Expression(component, this, scope, string_literal(info.tag) as Literal);
}
} else {
this.tag_expr = new Expression(component, this, scope, string_literal(this.name) as Literal);
}

this.namespace = get_namespace(parent as Element, this, component.namespace);

if (this.namespace !== namespaces.foreign) {
Expand Down
9 changes: 9 additions & 0 deletions src/compiler/compile/render_dom/Block.ts
Original file line number Diff line number Diff line change
Expand Up @@ -48,6 +48,7 @@ export default class Block {
hydrate: Array<Node | Node[]>;
mount: Array<Node | Node[]>;
measure: Array<Node | Node[]>;
restore_measurements: Array<Node | Node[]>;
fix: Array<Node | Node[]>;
animate: Array<Node | Node[]>;
intro: Array<Node | Node[]>;
Expand Down Expand Up @@ -96,6 +97,7 @@ export default class Block {
hydrate: [],
mount: [],
measure: [],
restore_measurements: [],
fix: [],
animate: [],
intro: [],
Expand Down Expand Up @@ -326,6 +328,12 @@ export default class Block {
${this.chunks.measure}
}`;

if (this.chunks.restore_measurements.length) {
properties.restore_measurements = x`function #restore_measurements(#measurement) {
${this.chunks.restore_measurements}
}`;
}

properties.fix = x`function #fix() {
${this.chunks.fix}
}`;
Expand Down Expand Up @@ -379,6 +387,7 @@ export default class Block {
m: ${properties.mount},
p: ${properties.update},
r: ${properties.measure},
s: ${properties.restore_measurements},
f: ${properties.fix},
a: ${properties.animate},
i: ${properties.intro},
Expand Down
168 changes: 158 additions & 10 deletions src/compiler/compile/render_dom/wrappers/Element/index.ts
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,7 @@ import Action from '../../../nodes/Action';
import MustacheTagWrapper from '../MustacheTag';
import RawMustacheTagWrapper from '../RawMustacheTag';
import is_dynamic from '../shared/is_dynamic';
import create_debugging_comment from '../shared/create_debugging_comment';
import { push_array } from '../../../../utils/push_array';

interface BindingGroup {
Expand Down Expand Up @@ -134,6 +135,8 @@ const events = [
}
];

const CHILD_DYNAMIC_ELEMENT_BLOCK = 'child_dynamic_element';

export default class ElementWrapper extends Wrapper {
node: Element;
fragment: FragmentWrapper;
Expand All @@ -147,6 +150,9 @@ export default class ElementWrapper extends Wrapper {
var: any;
void: boolean;

child_dynamic_element_block?: Block = null;
child_dynamic_element?: ElementWrapper = null;

constructor(
renderer: Renderer,
block: Block,
Expand All @@ -156,6 +162,24 @@ export default class ElementWrapper extends Wrapper {
next_sibling: Wrapper
) {
super(renderer, block, parent, node);

if (node.is_dynamic_element && block.type !== CHILD_DYNAMIC_ELEMENT_BLOCK) {
this.child_dynamic_element_block = block.child({
comment: create_debugging_comment(node, renderer.component),
name: renderer.component.get_unique_name('create_dynamic_element'),
type: CHILD_DYNAMIC_ELEMENT_BLOCK
});
renderer.blocks.push(this.child_dynamic_element_block);
this.child_dynamic_element = new ElementWrapper(
renderer,
this.child_dynamic_element_block,
parent,
node,
strip_whitespace,
next_sibling
);
}

this.var = {
type: 'Identifier',
name: node.name.replace(/[^a-zA-Z0-9_$]/g, '_')
Expand Down Expand Up @@ -199,6 +223,8 @@ export default class ElementWrapper extends Wrapper {
block.add_animation();
}

block.add_dependencies(node.tag_expr.dependencies);

// add directive and handler dependencies
[node.animation, node.outro, ...node.actions, ...node.classes, ...node.styles].forEach(directive => {
if (directive && directive.expression) {
Expand All @@ -221,6 +247,7 @@ export default class ElementWrapper extends Wrapper {
node.handlers.length > 0 ||
node.styles.length > 0 ||
this.node.name === 'option' ||
node.tag_expr.dynamic_dependencies().length ||
renderer.options.dev
) {
this.parent.cannot_use_innerhtml(); // need to use add_location
Expand All @@ -232,6 +259,110 @@ export default class ElementWrapper extends Wrapper {
}

render(block: Block, parent_node: Identifier, parent_nodes: Identifier) {
if (this.child_dynamic_element) {
this.render_dynamic_element(block, parent_node, parent_nodes);
} else {
this.render_element(block, parent_node, parent_nodes);
}
}

render_dynamic_element(block: Block, parent_node: Identifier, parent_nodes: Identifier) {
this.child_dynamic_element.render(
this.child_dynamic_element_block,
null,
(x`#nodes` as unknown) as Identifier
);

const previous_tag = block.get_unique_name('previous_tag');
const tag = this.node.tag_expr.manipulate(block);
block.add_variable(previous_tag, tag);

block.chunks.init.push(b`
${this.renderer.options.dev && b`@validate_dynamic_element(${tag});`}
let ${this.var} = ${tag} && ${this.child_dynamic_element_block.name}(#ctx);
`);

block.chunks.create.push(b`
if (${this.var}) ${this.var}.c();
`);

if (this.renderer.options.hydratable) {
block.chunks.claim.push(b`
if (${this.var}) ${this.var}.l(${parent_nodes});
`);
}

block.chunks.mount.push(b`
if (${this.var}) ${this.var}.m(${parent_node || '#target'}, ${parent_node ? 'null' : '#anchor'});
`);

const anchor = this.get_or_create_anchor(block, parent_node, parent_nodes);
const has_transitions = !!(this.node.intro || this.node.outro);
const not_equal = this.renderer.component.component_options.immutable ? x`@not_equal` : x`@safe_not_equal`;

block.chunks.update.push(b`
if (${tag}) {
if (!${previous_tag}) {
${this.var} = ${this.child_dynamic_element_block.name}(#ctx);
${this.var}.c();
${has_transitions && b`@transition_in(${this.var})`}
${this.var}.m(${this.get_update_mount_node(anchor)}, ${anchor});
} else if (${not_equal}(${previous_tag}, ${tag})) {
${this.var}.d(1);
${this.renderer.options.dev && b`@validate_dynamic_element(${tag});`}
${this.var} = ${this.child_dynamic_element_block.name}(#ctx);
${this.var}.c();
${this.var}.m(${this.get_update_mount_node(anchor)}, ${anchor});
} else {
${this.var}.p(#ctx, #dirty);
}
} else if (${previous_tag}) {
${
has_transitions
? b`
@group_outros();
@transition_out(${this.var}, 1, 1, () => {
${this.var} = null;
});
@check_outros();
`
: b`
${this.var}.d(1);
${this.var} = null;
`
}
}
${previous_tag} = ${tag};
`);

if (this.child_dynamic_element_block.has_intros) {
block.chunks.intro.push(b`@transition_in(${this.var});`);
}

if (this.child_dynamic_element_block.has_outros) {
block.chunks.outro.push(b`@transition_out(${this.var});`);
}

block.chunks.destroy.push(b`if (${this.var}) ${this.var}.d(detaching)`);

if (this.node.animation) {
const measurements = block.get_unique_name('measurements');
block.add_variable(measurements);
block.chunks.measure.push(b`${measurements} = ${this.var}.r()`);
block.chunks.fix.push(b`${this.var}.f();`);
block.chunks.animate.push(b`
${this.var}.s(${measurements});
${this.var}.a()
`);
}
}

is_dom_node() {
return super.is_dom_node() && !this.child_dynamic_element;
}

render_element(block: Block, parent_node: Identifier, parent_nodes: Identifier) {

const { renderer } = this;

if (this.node.name === 'noscript') return;
Expand All @@ -249,7 +380,7 @@ export default class ElementWrapper extends Wrapper {
if (renderer.options.hydratable) {
if (parent_nodes) {
block.chunks.claim.push(b`
${node} = ${this.get_claim_statement(parent_nodes)};
${node} = ${this.get_claim_statement(block, parent_nodes)};
`);

if (!this.void && this.node.children.length > 0) {
Expand Down Expand Up @@ -357,14 +488,16 @@ export default class ElementWrapper extends Wrapper {
b`@add_location(${this.var}, ${renderer.file_var}, ${loc.line - 1}, ${loc.column}, ${this.node.start});`
);
}

block.renderer.dirty(this.node.tag_expr.dynamic_dependencies());
}

can_use_textcontent() {
return this.is_static_content && this.fragment.nodes.every(node => node.node.type === 'Text' || node.node.type === 'MustacheTag');
}

get_render_statement(block: Block) {
const { name, namespace } = this.node;
const { name, namespace, tag_expr } = this.node;

if (namespace === namespaces.svg) {
return x`@svg_element("${name}")`;
Expand All @@ -379,22 +512,32 @@ export default class ElementWrapper extends Wrapper {
return x`@element_is("${name}", ${is.render_chunks(block).reduce((lhs, rhs) => x`${lhs} + ${rhs}`)})`;
}

return x`@element("${name}")`;
const reference = tag_expr.manipulate(block);
return x`@element(${reference})`;
}

get_claim_statement(nodes: Identifier) {
get_claim_statement(block: Block, nodes: Identifier) {
const attributes = this.attributes
.filter((attr) => !(attr instanceof SpreadAttributeWrapper) && !attr.property_name)
.map((attr) => p`${(attr as StyleAttributeWrapper | AttributeWrapper).name}: true`);

const name = this.node.namespace
? this.node.name
: this.node.name.toUpperCase();
let reference;
if (this.node.tag_expr.node.type === 'Literal') {
if (this.node.namespace) {
reference = `"${this.node.tag_expr.node.value}"`;
} else {
reference = `"${(this.node.tag_expr.node.value as String || '').toUpperCase()}"`;
}
} else if (this.node.namespace) {
reference = x`${this.node.tag_expr.manipulate(block)}`;
} else {
reference = x`(${this.node.tag_expr.manipulate(block)} || 'null').toUpperCase()`;
}

if (this.node.namespace === namespaces.svg) {
return x`@claim_svg_element(${nodes}, "${name}", { ${attributes} })`;
return x`@claim_svg_element(${nodes}, ${reference}, { ${attributes} })`;
} else {
return x`@claim_element(${nodes}, "${name}", { ${attributes} })`;
return x`@claim_element(${nodes}, ${reference}, { ${attributes} })`;
}
}

Expand Down Expand Up @@ -847,6 +990,11 @@ export default class ElementWrapper extends Wrapper {
${rect} = ${this.var}.getBoundingClientRect();
`);

if (block.type === CHILD_DYNAMIC_ELEMENT_BLOCK) {
block.chunks.measure.push(b`return ${rect}`);
block.chunks.restore_measurements.push(b`${rect} = #measurement;`);
}

block.chunks.fix.push(b`
@fix_position(${this.var});
${stop_animation}();
Expand Down Expand Up @@ -940,7 +1088,7 @@ export default class ElementWrapper extends Wrapper {
if (should_cache) {
block.chunks.update.push(b`
if (${block.renderer.dirty(dependencies)} && (${cached_snippet} !== (${cached_snippet} = ${snippet}))) {
${updater}
${updater}
}
`);
} else {
Expand Down
Loading