By default, when you modify the value of an each
block, it will add and remove DOM nodes at the end of the block, and update any values that have changed. That might not be what you want.
It's easier to show why than to explain. The <Thing>
component sets the emoji as a constant on initialization, but the name is passed in via a prop.
Click the 'Remove first thing' button a few times, and notice what happens:
- It removes the last component.
- It then updates the
name
value in the remaining DOM nodes, but not the emoji, which is fixed when each<Thing>
is created.
Instead, we'd like to remove only the first <Thing>
component and its DOM node, and leave the others unaffected.
To do that, we specify a unique identifier (or "key") for each iteration of the each
block:
{#each things as thing (thing.id)}
<Thing name={thing.name}/>
{/each}
Here, (thing.id)
is the key, which tells Svelte how to figure out what to update when the values (name
in this example) change.
You can use any object as the key, as Svelte uses a
Map
internally — in other words you could do(thing)
instead of(thing.id)
. Using a string or number is generally safer, however, since it means identity persists without referential equality, for example when updating with fresh data from an API server.
<script>
import Thing from './Thing.svelte';
let things = [
{ id: 1, name: 'apple' },
{ id: 2, name: 'banana' },
{ id: 3, name: 'carrot' },
{ id: 4, name: 'doughnut' },
{ id: 5, name: 'egg' }
];
function handleClick() {
things = things.slice(1);
}
</script>
<button on:click={handleClick}>
Remove first thing
</button>
{#each things as thing}
<Thing name={thing.name} />
{/each}