Fix double dispatch bug in observable compositions

This commit is contained in:
Talia 2025-02-20 23:01:12 +01:00
parent fc1fc056a2
commit d457c386e0
Signed by: darkwiiplayer
GPG key ID: 7808674088232B3E

View file

@ -134,6 +134,7 @@ export class Observable extends EventTarget {
enqueue(property, from, to, mutation=false) { enqueue(property, from, to, mutation=false) {
const change = {property, from, to, mutation} const change = {property, from, to, mutation}
if (!this.dispatchEvent(new ChangeEvent(change))) return false if (!this.dispatchEvent(new ChangeEvent(change))) return false
if (!this.synchronous) { if (!this.synchronous) {
if (!this.#queue) { if (!this.#queue) {
this.#queue = [] this.#queue = []
@ -260,9 +261,14 @@ export class ObservableValue extends Observable {
} }
} }
/** @param {Change[]} changes */ /**
emit(...changes) { * @param {(value: any) => any} func
this.dispatchEvent(new ChangedEvent(...changes)) */
compose(func) {
return new Composition(func, {}, this)
}
proxy(methods) {
} }
} }
@ -365,34 +371,17 @@ class Composition extends ObservableValue {
const ref = new WeakRef(this) const ref = new WeakRef(this)
obesrvables.forEach(state => { obesrvables.forEach(state => {
state.addEventListener("change", () => { state.addEventListener("changed", () => {
ref.deref()?.scheduleUpdate() ref.deref()?.update()
}, {signal: abortController.signal}) }, {signal: abortController.signal})
}) })
this.update() this.update()
} }
#microtaskQueued = false
scheduleUpdate() {
if (this.synchronous) {
this.update()
} else {
if (!this.#microtaskQueued) {
queueMicrotask(() => {
this.#microtaskQueued = false
this.update()
})
this.#microtaskQueued = true
}
}
}
update() { update() {
const value = this.#func(...this.#states.map(state => state.value)) const value = this.#func(...this.#states.map(state => state.value))
const change = {property: "value", from: this.value, to: value}
this.value = value this.value = value
this.emit(change)
} }
} }