Skip to content

Commit bd950a0

Browse files
authored
fix: more efficient and correct reactive set (#11967)
* fix: more efficient and correct reactive set * Update .changeset/thin-spoons-float.md
1 parent 4f12846 commit bd950a0

File tree

4 files changed

+57
-79
lines changed

4 files changed

+57
-79
lines changed

.changeset/thin-spoons-float.md

Lines changed: 5 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
---
2+
'svelte': patch
3+
---
4+
5+
fix: update reactive set when deleting initial values

packages/svelte/src/reactivity/set.js

Lines changed: 24 additions & 26 deletions
Original file line numberDiff line numberDiff line change
@@ -66,76 +66,74 @@ export class ReactiveSet extends Set {
6666

6767
/** @param {T} value */
6868
has(value) {
69+
var has = super.has(value);
6970
var sources = this.#sources;
7071
var s = sources.get(value);
7172

7273
if (s === undefined) {
73-
var ret = super.has(value);
74-
if (ret) {
75-
s = source(true);
76-
sources.set(value, s);
77-
} else {
78-
// We should always track the version in case
79-
// the Set ever gets this value in the future.
74+
if (!has) {
75+
// If the value doesn't exist, track the version in case it's added later
76+
// but don't create sources willy-nilly to track all possible values
8077
get(this.#version);
8178
return false;
8279
}
80+
81+
s = source(true);
82+
sources.set(value, s);
8383
}
8484

8585
get(s);
86-
return super.has(value);
86+
return has;
8787
}
8888

8989
/** @param {T} value */
9090
add(value) {
91-
var sources = this.#sources;
92-
var res = super.add(value);
93-
var s = sources.get(value);
94-
95-
if (s === undefined) {
96-
sources.set(value, source(true));
91+
if (!super.has(value)) {
92+
super.add(value);
9793
set(this.#size, super.size);
9894
increment(this.#version);
99-
} else {
100-
set(s, true);
10195
}
10296

103-
return res;
97+
return this;
10498
}
10599

106100
/** @param {T} value */
107101
delete(value) {
102+
var deleted = super.delete(value);
108103
var sources = this.#sources;
109104
var s = sources.get(value);
110-
var res = super.delete(value);
111105

112106
if (s !== undefined) {
113107
sources.delete(value);
114-
set(this.#size, super.size);
115108
set(s, false);
109+
}
110+
111+
if (deleted) {
112+
set(this.#size, super.size);
116113
increment(this.#version);
117114
}
118115

119-
return res;
116+
return deleted;
120117
}
121118

122119
clear() {
123-
var sources = this.#sources;
124-
125120
if (super.size !== 0) {
126-
set(this.#size, 0);
121+
var sources = this.#sources;
122+
127123
for (var s of sources.values()) {
128124
set(s, false);
129125
}
130-
increment(this.#version);
126+
131127
sources.clear();
128+
set(this.#size, 0);
129+
increment(this.#version);
132130
}
131+
133132
super.clear();
134133
}
135134

136135
keys() {
137-
get(this.#version);
138-
return super.keys();
136+
return this.values();
139137
}
140138

141139
values() {
Lines changed: 16 additions & 38 deletions
Original file line numberDiff line numberDiff line change
@@ -1,50 +1,28 @@
11
import { flushSync } from '../../../../src/index-client';
2-
import { test } from '../../test';
2+
import { ok, test } from '../../test';
33

44
export default test({
5-
html: `<button>add</button><button>delete</button><button>clear</button>`,
5+
html: `<button>delete initial</button><button>add</button><button>delete</button><button>clear</button><div id="output"><p>1</p><div>0</div></div>`,
66

77
test({ assert, target }) {
8-
const [btn, btn2, btn3] = target.querySelectorAll('button');
8+
const [btn, btn2, btn3, btn4] = target.querySelectorAll('button');
9+
const output = target.querySelector('#output');
10+
ok(output);
911

10-
flushSync(() => {
11-
btn?.click();
12-
});
12+
flushSync(() => btn?.click());
13+
assert.htmlEqual(output.innerHTML, `<p>0</p>`);
1314

14-
assert.htmlEqual(
15-
target.innerHTML,
16-
`<button>add</button><button>delete</button><button>clear</button><div>1</div>`
17-
);
15+
flushSync(() => btn2?.click());
16+
assert.htmlEqual(output.innerHTML, `<p>1</p><div>1</div>`);
1817

19-
flushSync(() => {
20-
btn?.click();
21-
});
18+
flushSync(() => btn2?.click());
19+
flushSync(() => btn2?.click());
20+
assert.htmlEqual(output.innerHTML, `<p>3</p><div>1</div><div>2</div><div>3</div>`);
2221

23-
flushSync(() => {
24-
btn?.click();
25-
});
22+
flushSync(() => btn3?.click());
23+
assert.htmlEqual(output.innerHTML, `<p>2</p><div>1</div><div>2</div>`);
2624

27-
assert.htmlEqual(
28-
target.innerHTML,
29-
`<button>add</button><button>delete</button><button>clear</button><div>1</div><div>2</div><div>3</div>`
30-
);
31-
32-
flushSync(() => {
33-
btn2?.click();
34-
});
35-
36-
assert.htmlEqual(
37-
target.innerHTML,
38-
`<button>add</button><button>delete</button><button>clear</button><div>1</div><div>2</div>`
39-
);
40-
41-
flushSync(() => {
42-
btn3?.click();
43-
});
44-
45-
assert.htmlEqual(
46-
target.innerHTML,
47-
`<button>add</button><button>delete</button><button>clear</button>`
48-
);
25+
flushSync(() => btn4?.click());
26+
assert.htmlEqual(output.innerHTML, `<p>0</p>`);
4927
}
5028
});
Lines changed: 12 additions & 15 deletions
Original file line numberDiff line numberDiff line change
@@ -1,21 +1,18 @@
11
<script>
2-
import {Set} from 'svelte/reactivity';
2+
import { Set } from 'svelte/reactivity';
33
4-
let state = new Set();
4+
let state = new Set([0]);
55
</script>
66

7-
<button onclick={() => {
8-
state.add(state.size + 1);
9-
}}>add</button>
7+
<button onclick={() => state.delete(0)}>delete initial</button>
8+
<button onclick={() => state.add(state.size + 1)}>add</button>
9+
<button onclick={() => state.delete(state.size)}>delete</button>
10+
<button onclick={() => state.clear()}>clear</button>
1011

11-
<button onclick={() => {
12-
state.delete(state.size);
13-
}}>delete</button>
12+
<div id="output">
13+
<p>{state.size}</p>
1414

15-
<button onclick={() => {
16-
state.clear();
17-
}}>clear</button>
18-
19-
{#each state as item}
20-
<div>{item}</div>
21-
{/each}
15+
{#each state as item}
16+
<div>{item}</div>
17+
{/each}
18+
</div>

0 commit comments

Comments
 (0)