-
-
Notifications
You must be signed in to change notification settings - Fork 4.5k
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
Dynamic elements implementation <svelte:element>
#5481
Dynamic elements implementation <svelte:element>
#5481
Conversation
346eb46
to
fe4158a
Compare
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I've quickly look through the implementation, overall it's great.
I have 2 questions that I'm curious about:
-
how does bindings, and event listeners works on the
<svelte:element>
?
Currently, some of the events or bindings works only for a specific elements, for example:bind:checked
andbind:group
only works forinput[type="checkbox"]
andinput[type="radio"]
,bind:value
only works forinput
& etc. -
I notice that if the tag variable changed in
<svelte:element {tag}>
, you'll remove the entire elements + all the child elements and recreate them. I wonder is it possible to reuse the child elements, by swapping the elements only?
also i feel that having the following would be great:
- test cases that involved dynamic elements + bindings / dynamic elements + event listener?
- test case that involved using store value as tag?
<svelte:element tag={$tag} >
though I imagine it should work fine - dev warnings + test case for passing non string value or undefined / null to the tag variable.
Been testing this and think it's an awesome feature! this isn't a major problem however, as I expect most people using this feature to be using CSS-in-JS / stylesheets for the page-builders, arbitrary content renderers that this feature might be used for |
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
Instead, what if Svelte just came with a set of importable values that could be passed into What if instead the solution was: <script>
import { elements } from "svelte";
import CustomComponent from "./CustomComponent.svelte";
let element = elements.h1;
</script>
<main>
<svelte:component this={elements.h1}>Hello World!</svelte:component>
<button on:click={() => element = element === elements.h1 ? CustomComponent : elements.h1}>Toggle Element</button>
</main> This solution is currently possible by creating your own wrappers for each element you want to use this way. But I think implementing it in Svelte would be more convenient and flexible than the |
@DanielPower What's wrong with |
As you already noted this is possible to do in userland. The problem is that it’s not flexible enough. A solution that requires you to know ahead of time what elements you will be using does not work for arbitrary elements like rendering content from a third party (like a CMS) or use in a library where the library author does not have control of the usage. |
@AradAral The only issue I have with
Versus
The latter would require importing an object of elements. But otherwise behaves exactly the same, and also allows using the same feature for both elements and components. @jonatansberg Objects are also indexable by string. So |
@DanielPower Yes, bundle size. (And support for custom elements.) |
Here is an example of my proposal: https://svelte.dev/repl/926ffe1260ef444f8dfa1f63f985a1a8?version=3.35.0 I hadn't considered custom elements though. My proposal was based on the assumption that Svelte could just introduce a helper object with each HTML element. But custom elements throws a wrench in that. |
This unnecessarily complicates the dynamic elements. Let's just stick with What you are suggesting already exists in userland, and it still has the exact same issues which prompted the addition of the I'm not going to get into it, but what you're suggesting won't solve anything |
Ignore my question about styles/classes not being picked up by the dynamic elements, it's not critical. I'm currently using this fork on a page-builder project i'm working on. The feature works flawlessly, and i'm sure the class issue could be resolved at a later stage. I'd rather this get pulled sooner rather than later |
… values as tag. Throw an error when trying to bind other than this. Add more tests.
2f6a886
to
54e1c23
Compare
Sorry for keeping you waiting. I have now updated the PR with some added functionality (retaining children when tag type has changed), better warnings, and have added documentation. |
@alfredringstad Thanks a lot for this! Personally, I've also encountered certain cases where I want to render a parent element conditionally, that is, when a condition is I'm not sure if this would also be possible in this new feature. But I'd propose that when the value of the What do you think about that? |
@AradAral You could solve this using a conditional and slots, like this: <script>
export let condition = false;
export let tag = 'a';
</script>
{#if condition && tag}
<svelte:element {tag} {...$$restProps}><slot /></svelte:element>
{:else}
<slot />
{/if} |
@jonatansberg Update: Sorry I was confused for a second, I hadn't looked at the REPL you linked to. Good idea. Thanks! |
I'm not sure when I'll have time to get back to this. Renaming tag to this shouldn't take too much time, but extracting a common ancestor class is a bit more tricky. Feel free to continue building on this if you have time. |
This is a great PR, still waiting for the final changes to be made. |
@alfredringstad I've got surely no right to be pushy, but as this is a crucial enhancement to be able to build quality component-libraries on top of svelte, I wanted to repeat the question about the eta? Or if this could be handed over to someone else with the needed knowledge of the svelte-compiler and a bit more time on his or her hands? Would be very much appreciated! |
Hi guys, it's been some time since the last update on this. Is anyone able to step in and get this PR merged? As far as I know there's no issues with it. I'm hating having to use a svelte fork to have this feature in my app. I'm sure many others are urgently waiting for this release. |
@AlbertMarashi |
There is no |
This PR would need to be rebased before it can be merged |
Personally, I liked Nonetheless, I'd like to continue with this approach provided that I can still use bind:this in conjunction with this={tag} as it's already been a while since this was initially proposed |
Anything I can do to help? @alfredringstad @benmccann Anything that needs to be fixed? |
I think this needs some more cleanup work. I discovered that applying css classes was not working yet for dynamic elements (which I did a quick-fix for), which put me down the path of discovering how the rest is built out, and I feel like there are some quick-fixes in that code which hurt maintainability in the long run. Essentially, both |
Fine with leaving the |
Hi. And I started to clean up this PR but I need 1-2-3 weeks due to working resources. And I found 1 bug that event handlers and attributes are lost when updating <script>
let tag = 'h1';
let num = 1;
let style = "color: red";
function click() {
num += 1;
tag = `h${num}`;
style = num % 2 === 0 ? "color: red" : "color: blue";
}
</script>
<svelte:element this={tag} on:click={click} style={style}>{tag}</svelte:element> And I would like to confirm that is there a remaining discussion point about this feature? |
Don't think so, you can ignore all of my comments, happy with whatever the svelte team wants to call the "this" attribute. I think the community is keen to start using it ASAP |
I mean I just would like to confirm that is there any remaining discussion point regarding And regarding |
@baseballyama don't think so, I think everyone's happy with moving forward. |
I created a new PR. (Because I don't have permission to push to this branch) I'd appreciate it if you check it! |
Close in favor of #6898 |
Someone wanted I HAVE AN IDEA We can make it so that What do you guys think? |
that makes no sense lukaz. Fragments are not for this |
@AlbertMarashi No problem, the name can be whatever you want. You guys are a million times better at coming up with names. |
Just use a |
Ok, I was just proposing the name. Someone else at some place asked for such a function, only proposed the syntax |
The latest PR removes (and prevents to create) an element if value of |
@baseballyama Ok. I have no problem with it. I know they're in no hurry for this feature, and they'll come up with a name. :D |
This PR is based of #2324 and implements a new tag in Svelte called
<svelte:element>
. This is used to dynamically render a HTML tag where you don't know the type upfront.Use cases
When rendering data from a CMS you may end up with a structure looking like this:
The way to render this today is by creating a Svelte component that renders the corresponding HTML tag by defining all of them. However, this adds a lot of boilerplate and unnecessary weight to the compiled bundle. The implemented solution in this PR solves this problem.
Usage
The
<svelte:element>
takes a special property calledtag
, which is the type of DOM element you want to render (e.g. div, h1 etc). All other props/attributes are passed through to the DOM element itself, meaning you can use it just like you would use any HTML element.Example: Passing tag as a string
Example: Passing tag as a variable
Implementation details
The implementation is based on the works of erzr (see PR #3928) as well as the new implementation of the {#key} block. It works with ssr and hydration.
This PR introduces a new Node type called DynamicElement that wraps the regular Element implementation. The Element node has also been extended with a prop called
dynamic_tag
, which is used instead of the name to render the element, if this is present.When the tag is changed the element is destroyed and a new one is created. If other props change except the tag, the element is re-used.
The server side implementation of DynamicElement is a copy of the Element implementation, with one important difference. Since we don't know upfront what kind of HTML element this is, we cannot do many of the optimizations/special cases (e.g. handling events in different ways depending on the element type or having textarea content be set both via value and as children). I think these tradeoffs are okay, since most of the time this should be used with more or less "static" content (if you want to have full control of it, you should probably create a Svelte component for this anyway).
Remaining