-
-
Notifications
You must be signed in to change notification settings - Fork 4.3k
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
$derived doesn't update derived values when used with a store containing an object. #14772
Comments
This isn't a bug. Instead, you should return the value you care about, or do what you're doing now and create a new value: let c = $derived($count.n); |
Interesting. I think the confusing part is that by calling the store's |
@webJose The derived does receive the update – however deriveds are memoized on the value being different, so it doesn't further pass on changes as it doesn't think anything has changed. |
I see. I think this is the behavior that has been affecting me for some weeks now. We have micro-frontends in Svelte v5. One of them has the UI for users to control the settings of the glass effect used in floating stuff like menus. The other MFE's consume this setting, which is in a $state() object that is also saved to local storage. This $state() is wrapped in a class, and said class instance (a singleton) is made available for consumption by other MFE's. So all MFE's use the same store. Yes, the MFE's that are not the creators of the singleton cannot receive updates via the signal because the signal belongs to another copy of the runtime, so we made it also implements the Enabling/disabling glass triggers reactivity in the MFE's instantaneously, but changes in blur, opacity or saturation do not. I suppose my expectation is that subscribers are executed and that a change should trigger, regardless. |
You can always be inventive here, such as wrapping the value in an object that always changes on a propagation of a change. let x = $derived({ current: $myObjectStore });
// …
x.current |
Thanks @trueadm, the memoization explanation makes sense. I guess $effect does not memoize, which would explain the different behavior? I think the "Stores" part of the svelte 5 docs should include a section that mentions 1) $derived works with stores, 2) An updated store will not necessarily trigger $derived updates if the value is === the previous, and 3) You can deal with that using the techniques you listed. |
Ha! Spreading also works. For this issue, do: let c = $derived({...$count}); This is what I did in my MFE setup and works. So does it work in your REPL. |
Why? There is fromStore which works flawlessly. |
Specifically I'm seeing: // works:
let c = $derived(Object.assign({}, $count));
// works:
let c = $derived({...$count});
// works:
let obj = fromStore(count);
let n = $derived(obj.current.n);
// doesn't work:
let {c} = $derived({c: $count});
// doesn't work:
let obj = $derived({current: $count});
let c = $derived(obj.current);
// doesn't work:
let obj = fromStore(count);
let c = $derived(obj.current); Here's a playground with these examples. |
They won't work, because you reference the store object again, which hasn't changed still. If you need to create a derived from a derived then you'll need to spread the object. |
So to make sure I understand correctly, this works let state = $state({ value: 0 });
let derived = $derived({ value: state.value * 2 });
// fires state.value changes
$effect(() => {
derived;
}); But this doesn't work let derived = $derived($store);
// doesn't fire
$effect(() => {
derived;
}); But this should work let derived = fromStore(store).current;
// fires
$effect(() => {
derived;
}); Or at least this works let derived = fromStore(store);
// fires
$effect(() => {
derived.current;
}); |
Describe the bug
When using $derived to get the value from a store containing an object, updates to the stored value don't cause the derived value to update. I suspect this is not by design, because $effect does handle those updates appropriately. If this is a known limitation, perhaps it should be mentioned in the docs.
Can work around by copying the object with Object.assign.
Reproduction
https://svelte.dev/playground/hello-world?version=5.14.4#H4sIAAAAAAAAE41Sy26DMBD8lZUVKUZBpL2SEKnqpf9QeiBm09KaNbKXJBXyv1eGhIQ-oh49s-P1zLgTVNQoUvGEWhs4GKtLkFhWjGUkYrGrNDqRPneCP5swFwARn1UPTZO4PWoO2LZw-BuuDDESO5GKtVO2aniTU85V3RjL8GhaYrSws6aGebI8nU_y-Sqn9fKiovV5frm5dRCxYDyySNm26OM_nj_dNbXwg7ttowMVBOBHI-6tsFgm7y54yFkjg4IMZiXaao-lnPWCaCTpmlQJXZj6miFYwP3AzXC3Q8VSRtmmC0DOypAzGhNtXmUuhhW5iOG0LAz56Huk25bZEBhSulIfWScjyDaDn6RtyoJRShN2gElosViBRW4tgVn5yPch9EKXQkc-hq72YcNw67-7GOOa1nANXzUwpn6wFRdbjWPwQ19Lx8b2vycnPPazfQF9R9mokh2lcOdDnNNXvvgvf8s4ARoDAAA=
Logs
No response
System Info
System: OS: macOS 15.0 CPU: (8) x64 Intel(R) Core(TM) i5-8279U CPU @ 2.40GHz Memory: 36.44 MB / 16.00 GB Shell: 5.9 - /bin/zsh Binaries: Node: 22.12.0 - /usr/local/bin/node Yarn: 1.22.19 - /usr/local/bin/yarn npm: 10.9.0 - /usr/local/bin/npm pnpm: 8.0.0 - /usr/local/bin/pnpm Browsers: Chrome: 131.0.6778.204 Safari: 18.0 npmPackages: svelte: 5.14.4 => 5.14.4
Severity
annoyance
The text was updated successfully, but these errors were encountered: