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

docs: Improve explanation of 05-keyed-each-blocks in tutorial #1252

Open
wants to merge 1 commit into
base: main
Choose a base branch
from
Open
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 @@ -2,7 +2,7 @@
title: Keyed each blocks
---

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.
By default, when you modify the iterable the `each` block displays, it will add or remove DOM nodes at the _end_ of the block, and update all nodes' values to reflect the modification. That might not be what you want.

It's easier to show why than to explain. Inside `Thing.svelte`, `name` is a dynamic prop but `emoji` is a constant.

Expand All @@ -11,6 +11,8 @@ Click the 'Remove first thing' button a few times, and notice what happens:
1. It removes the last component.
2. It then updates the `name` value in the remaining DOM nodes, but not the emoji.

To make it concrete, on the first iteration, Svelte removes the last DOM node ('Egg') and updates the remaining nodes to reflect the new data. 'Egg' takes the place of 'Doughnut', 'Doughnut' replaces 'Carrot, and so on. Because the emoji for each `<Thing>`'s DOM node was initialized at the beginning, with no dependence on state, each node's emoji remains unchanged.

> [!NOTE] If you're coming from React, this might seem strange, because you're used to the entire component re-rendering when state changes. Svelte works differently: the component 'runs' once, and subsequent updates are 'fine-grained'. This makes things faster and gives you more control.

One way to fix it would be to make `emoji` a [`$derived`](derived-state) value. But it makes more sense to remove the first `<Thing>` component altogether rather than remove the _last_ one and update all the others.
Expand Down