2020-08-05 18:38:55 +02:00
|
|
|
/*
|
|
|
|
Copyright 2020 Bruno Windels <bruno@windels.cloud>
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2020-04-19 19:02:10 +02:00
|
|
|
import {AbortError} from "../utils/error.js";
|
2020-04-20 21:26:39 +02:00
|
|
|
import {BaseObservable} from "./BaseObservable.js";
|
2020-04-19 19:02:10 +02:00
|
|
|
|
|
|
|
// like an EventEmitter, but doesn't have an event type
|
|
|
|
export class BaseObservableValue extends BaseObservable {
|
|
|
|
emit(argument) {
|
|
|
|
for (const h of this._handlers) {
|
|
|
|
h(argument);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
class WaitForHandle {
|
|
|
|
constructor(observable, predicate) {
|
|
|
|
this._promise = new Promise((resolve, reject) => {
|
|
|
|
this._reject = reject;
|
|
|
|
this._subscription = observable.subscribe(v => {
|
|
|
|
if (predicate(v)) {
|
|
|
|
this._reject = null;
|
|
|
|
resolve(v);
|
|
|
|
this.dispose();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
get promise() {
|
|
|
|
return this._promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose() {
|
|
|
|
if (this._subscription) {
|
|
|
|
this._subscription();
|
|
|
|
this._subscription = null;
|
|
|
|
}
|
|
|
|
if (this._reject) {
|
|
|
|
this._reject(new AbortError());
|
|
|
|
this._reject = null;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class ResolvedWaitForHandle {
|
|
|
|
constructor(promise) {
|
|
|
|
this.promise = promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose() {}
|
|
|
|
}
|
|
|
|
|
2020-04-20 21:26:39 +02:00
|
|
|
export class ObservableValue extends BaseObservableValue {
|
2020-04-19 19:02:10 +02:00
|
|
|
constructor(initialValue) {
|
|
|
|
super();
|
|
|
|
this._value = initialValue;
|
|
|
|
}
|
|
|
|
|
|
|
|
get() {
|
|
|
|
return this._value;
|
|
|
|
}
|
|
|
|
|
|
|
|
set(value) {
|
|
|
|
if (value !== this._value) {
|
|
|
|
this._value = value;
|
|
|
|
this.emit(this._value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
waitFor(predicate) {
|
|
|
|
if (predicate(this.get())) {
|
|
|
|
return new ResolvedWaitForHandle(Promise.resolve(this.get()));
|
|
|
|
} else {
|
|
|
|
return new WaitForHandle(this, predicate);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
export function tests() {
|
|
|
|
return {
|
|
|
|
"set emits an update": assert => {
|
|
|
|
const a = new ObservableValue();
|
|
|
|
let fired = false;
|
|
|
|
const subscription = a.subscribe(v => {
|
|
|
|
fired = true;
|
|
|
|
assert.strictEqual(v, 5);
|
|
|
|
});
|
|
|
|
a.set(5);
|
|
|
|
assert(fired);
|
|
|
|
subscription();
|
|
|
|
},
|
|
|
|
"set doesn't emit if value hasn't changed": assert => {
|
|
|
|
const a = new ObservableValue(5);
|
|
|
|
let fired = false;
|
|
|
|
const subscription = a.subscribe(() => {
|
|
|
|
fired = true;
|
|
|
|
});
|
|
|
|
a.set(5);
|
|
|
|
a.set(5);
|
|
|
|
assert(!fired);
|
|
|
|
subscription();
|
|
|
|
},
|
|
|
|
"waitFor promise resolves on matching update": async assert => {
|
|
|
|
const a = new ObservableValue(5);
|
|
|
|
const handle = a.waitFor(v => v === 6);
|
|
|
|
Promise.resolve().then(() => {
|
|
|
|
a.set(6);
|
|
|
|
});
|
|
|
|
await handle.promise;
|
|
|
|
assert.strictEqual(a.get(), 6);
|
|
|
|
},
|
|
|
|
"waitFor promise rejects when disposed": async assert => {
|
|
|
|
const a = new ObservableValue();
|
|
|
|
const handle = a.waitFor(() => false);
|
|
|
|
Promise.resolve().then(() => {
|
|
|
|
handle.dispose();
|
|
|
|
});
|
|
|
|
await assert.rejects(handle.promise, AbortError);
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|