2021-04-13 10:57:08 +02:00
|
|
|
// Copyright: Ankitects Pty Ltd and contributors
|
|
|
|
// License: GNU AGPL, version 3 or later; http://www.gnu.org/licenses/agpl.html
|
|
|
|
|
2021-03-22 00:40:19 +01:00
|
|
|
import { Readable, derived } from "svelte/store";
|
2021-03-21 20:16:40 +01:00
|
|
|
|
2021-04-15 18:30:53 +02:00
|
|
|
interface AsyncReactiveData<T, E> {
|
2021-03-21 21:07:35 +01:00
|
|
|
value: Readable<T | null>;
|
|
|
|
error: Readable<E | null>;
|
|
|
|
loading: Readable<boolean>;
|
2021-03-21 20:16:40 +01:00
|
|
|
}
|
|
|
|
|
2021-03-21 21:38:23 +01:00
|
|
|
function useAsyncReactive<T, E>(
|
2021-03-21 21:07:35 +01:00
|
|
|
asyncFunction: () => Promise<T>,
|
2021-10-19 01:06:00 +02:00
|
|
|
dependencies: [Readable<unknown>, ...Readable<unknown>[]],
|
2021-04-15 18:30:53 +02:00
|
|
|
): AsyncReactiveData<T, E> {
|
2021-03-22 02:44:08 +01:00
|
|
|
const promise = derived(
|
|
|
|
dependencies,
|
2021-03-22 15:30:35 +01:00
|
|
|
(_, set: (value: Promise<T> | null) => void): void => set(asyncFunction()),
|
2021-03-22 02:58:19 +01:00
|
|
|
// initialize with null to avoid duplicate fetch on init
|
2021-10-19 01:06:00 +02:00
|
|
|
null,
|
2021-03-22 02:44:08 +01:00
|
|
|
);
|
2021-03-21 20:16:40 +01:00
|
|
|
|
2021-03-21 21:07:35 +01:00
|
|
|
const value = derived(
|
|
|
|
promise,
|
2021-03-22 15:30:35 +01:00
|
|
|
($promise, set: (value: T) => void): void => {
|
2021-03-22 02:58:19 +01:00
|
|
|
$promise?.then((value: T) => set(value));
|
2021-03-21 21:07:35 +01:00
|
|
|
},
|
2021-10-19 01:06:00 +02:00
|
|
|
null,
|
2021-03-21 21:07:35 +01:00
|
|
|
);
|
2021-03-21 20:16:40 +01:00
|
|
|
|
2021-03-21 21:07:35 +01:00
|
|
|
const error = derived(
|
|
|
|
promise,
|
2021-03-22 15:30:35 +01:00
|
|
|
($promise, set: (error: E | null) => void): (() => void) => {
|
2021-03-22 02:58:19 +01:00
|
|
|
$promise?.catch((error: E) => set(error));
|
2021-03-22 15:30:35 +01:00
|
|
|
return (): void => set(null);
|
2021-03-21 21:07:35 +01:00
|
|
|
},
|
2021-10-19 01:06:00 +02:00
|
|
|
null,
|
2021-03-21 21:07:35 +01:00
|
|
|
);
|
2021-03-21 20:16:40 +01:00
|
|
|
|
2021-03-21 21:07:35 +01:00
|
|
|
const loading = derived(
|
2021-03-22 15:23:48 +01:00
|
|
|
promise,
|
2021-03-22 15:30:35 +01:00
|
|
|
($promise, set: (value: boolean) => void): (() => void) => {
|
2021-03-22 15:23:48 +01:00
|
|
|
$promise?.finally(() => set(false));
|
2021-03-22 15:30:35 +01:00
|
|
|
return (): void => set(true);
|
2021-03-21 21:07:35 +01:00
|
|
|
},
|
2021-10-19 01:06:00 +02:00
|
|
|
true,
|
2021-03-21 21:07:35 +01:00
|
|
|
);
|
|
|
|
|
2021-03-22 15:23:48 +01:00
|
|
|
return { value, error, loading };
|
2021-03-21 20:16:40 +01:00
|
|
|
}
|
|
|
|
|
2021-03-21 21:38:23 +01:00
|
|
|
export default useAsyncReactive;
|