From 2f8eda94f34f756f5f5c6ea50d202f83e0d875a8 Mon Sep 17 00:00:00 2001 From: Rich Harris Date: Tue, 3 Dec 2024 13:49:57 -0500 Subject: [PATCH] fix: ensure SvelteDate cached methods have correct reactive context (#14525) * fix: ensure SvelteDate cached methods have no reactive context * fix: ensure SvelteDate cached methods have no reactive context * fix * lint * use active reaction at time of instance creation * tweak changeset * Update packages/svelte/src/internal/client/dom/elements/bindings/shared.js Co-authored-by: Paolo Ricciuti --------- Co-authored-by: Dominic Gannaway Co-authored-by: Paolo Ricciuti --- .changeset/wicked-zebras-exist.md | 5 ++++ packages/svelte/src/reactivity/date.js | 14 +++++++--- packages/svelte/src/reactivity/date.test.ts | 30 +++++++++++++++++++++ 3 files changed, 46 insertions(+), 3 deletions(-) create mode 100644 .changeset/wicked-zebras-exist.md diff --git a/.changeset/wicked-zebras-exist.md b/.changeset/wicked-zebras-exist.md new file mode 100644 index 000000000000..f76d77bb9078 --- /dev/null +++ b/.changeset/wicked-zebras-exist.md @@ -0,0 +1,5 @@ +--- +'svelte': patch +--- + +fix: use correct reaction when lazily creating deriveds inside `SvelteDate` diff --git a/packages/svelte/src/reactivity/date.js b/packages/svelte/src/reactivity/date.js index 2d8de624bcc7..33da2e176159 100644 --- a/packages/svelte/src/reactivity/date.js +++ b/packages/svelte/src/reactivity/date.js @@ -1,8 +1,7 @@ /** @import { Source } from '#client' */ -import { DESTROYED } from '../internal/client/constants.js'; import { derived } from '../internal/client/index.js'; import { source, set } from '../internal/client/reactivity/sources.js'; -import { get } from '../internal/client/runtime.js'; +import { active_reaction, get, set_active_reaction } from '../internal/client/runtime.js'; var inited = false; @@ -12,6 +11,8 @@ export class SvelteDate extends Date { /** @type {Map>} */ #deriveds = new Map(); + #reaction = active_reaction; + /** @param {any[]} params */ constructor(...params) { // @ts-ignore @@ -43,7 +44,12 @@ export class SvelteDate extends Date { var d = this.#deriveds.get(method); - if (d === undefined || (d.f & DESTROYED) !== 0) { + if (d === undefined) { + // lazily create the derived, but as though it were being + // created at the same time as the class instance + const reaction = active_reaction; + set_active_reaction(this.#reaction); + d = derived(() => { get(this.#time); // @ts-ignore @@ -51,6 +57,8 @@ export class SvelteDate extends Date { }); this.#deriveds.set(method, d); + + set_active_reaction(reaction); } return get(d); diff --git a/packages/svelte/src/reactivity/date.test.ts b/packages/svelte/src/reactivity/date.test.ts index 87bfde41c89c..f90c5a102c52 100644 --- a/packages/svelte/src/reactivity/date.test.ts +++ b/packages/svelte/src/reactivity/date.test.ts @@ -642,3 +642,33 @@ test('Date methods invoked for the first time in a derived', () => { cleanup(); }); + +test('Date methods shared between deriveds', () => { + const date = new SvelteDate(initial_date); + const log: any = []; + + const cleanup = effect_root(() => { + const year = derived(() => { + return date.getFullYear(); + }); + const year2 = derived(() => { + return date.getTime(), date.getFullYear(); + }); + + render_effect(() => { + log.push(get(year) + '/' + get(year2).toString()); + }); + + flushSync(() => { + date.setFullYear(date.getFullYear() + 1); + }); + + flushSync(() => { + date.setFullYear(date.getFullYear() + 1); + }); + }); + + assert.deepEqual(log, ['2023/2023', '2024/2024', '2025/2025']); + + cleanup(); +});