Derived store reruns subscribers if it's value has not changed when synced.

All invalidators of subscribers are run on a derived store when invalidated.

See https://github.com/sveltejs/svelte/pull/2955
pull/3219/head
Brian Takita 6 years ago
parent 2761da640a
commit bbeafbafab

@ -67,16 +67,15 @@ export function writable<T>(value: T, start: StartStopNotifier<T> = noop): Writa
function set(new_value: T): void { function set(new_value: T): void {
if (safe_not_equal(value, new_value)) { if (safe_not_equal(value, new_value)) {
value = new_value; value = new_value;
if (!stop) { if (stop) { // store is ready
return; // not ready
}
subscribers.forEach((s) => s[1]()); subscribers.forEach((s) => s[1]());
subscribers.forEach((s) => s[0](value)); subscribers.forEach((s) => s[0](value));
} }
} }
}
function update(fn: Updater<T>): void { function update(fn: Updater<T>): void {
set(fn(value)); return set(fn(value));
} }
function subscribe(run: Subscriber<T>, invalidate: Invalidator<T> = noop): Unsubscriber { function subscribe(run: Subscriber<T>, invalidate: Invalidator<T> = noop): Unsubscriber {
@ -129,7 +128,9 @@ export function derived<T, S extends Stores>(
const auto = fn.length < 2; const auto = fn.length < 2;
const subscribers: Array<Subscriber<T>> = [];
const invalidators: Array<Invalidator<T>> = []; const invalidators: Array<Invalidator<T>> = [];
let value: T = initial_value;
const store = readable(initial_value, (set) => { const store = readable(initial_value, (set) => {
let inited = false; let inited = false;
@ -146,6 +147,11 @@ export function derived<T, S extends Stores>(
const result = fn(single ? values[0] : values, set); const result = fn(single ? values[0] : values, set);
if (auto) { if (auto) {
set(result as T); set(result as T);
const dirty = safe_not_equal(value, result);
value = result as T;
if (!dirty) {
subscribers.forEach(s => s(value));
}
} else { } else {
cleanup = is_function(result) ? result as Unsubscriber : noop; cleanup = is_function(result) ? result as Unsubscriber : noop;
} }
@ -176,6 +182,7 @@ export function derived<T, S extends Stores>(
return { return {
subscribe(run: Subscriber<T>, invalidate: Invalidator<T> = noop): Unsubscriber { subscribe(run: Subscriber<T>, invalidate: Invalidator<T> = noop): Unsubscriber {
subscribers.push(run);
invalidators.push(invalidate); invalidators.push(invalidate);
const unsubscribe = store.subscribe(run, invalidate); const unsubscribe = store.subscribe(run, invalidate);

@ -233,6 +233,30 @@ describe('store', () => {
unsubscribe(); unsubscribe();
}); });
it('derived dependency does not update and shared ancestor updates', () => {
const root = writable({ a: 0, b:0 });
const values = [];
const a = derived(root, $root => {
return 'a' + $root.a;
});
const b = derived([a, root], ([$a, $root]) => {
return 'b' + $root.b + $a;
});
const unsubscribe = b.subscribe(v => {
values.push(v);
});
assert.deepEqual(values, ['b0a0']);
root.set({ a: 0, b: 1 });
assert.deepEqual(values, ['b0a0', 'b1a0']);
unsubscribe();
});
it('is updated with safe_not_equal logic', () => { it('is updated with safe_not_equal logic', () => {
const arr = [0]; const arr = [0];

Loading…
Cancel
Save