UNPKG

@furystack/shades

Version:

A lightweight UI framework for FuryStack with JSX support

113 lines 5.35 kB
import { AggregatedError } from '@furystack/core'; import { ObservableValue, isAsyncDisposable, isDisposable } from '@furystack/utils'; /** * Class for managing observables and disposables for components, based on key-value maps */ export class ResourceManager { disposables = new Map(); disposableDeps = new Map(); /** * Returns an existing disposable resource by key, or creates and caches a new one. * Resources are automatically disposed when the component is removed from the DOM. * When `deps` is provided, the resource is re-created (and the old one disposed) whenever * the serialized deps value changes. This is useful for resources that depend on dynamic * parameters (e.g., entity-sync subscriptions with changing query options). * @param key Unique key for caching this resource * @param factory Factory function called once to create the resource * @param deps Optional dependency array -- when deps change, the old resource is disposed and a new one is created. * Values are compared via `JSON.stringify`, so `undefined` and `null` are treated as equal within arrays. * @returns The cached or newly created resource */ useDisposable(key, factory, deps) { const existing = this.disposables.get(key); const depsKey = deps !== undefined ? JSON.stringify(deps) : undefined; if (existing) { if (depsKey !== undefined && this.disposableDeps.get(key) !== depsKey) { if (isDisposable(existing)) existing[Symbol.dispose](); if (isAsyncDisposable(existing)) void existing[Symbol.asyncDispose](); const created = factory(); this.disposables.set(key, created); this.disposableDeps.set(key, depsKey); return created; } return existing; } const created = factory(); this.disposables.set(key, created); if (depsKey !== undefined) this.disposableDeps.set(key, depsKey); return created; } observers = new Map(); /** * Subscribes to an observable value by key. If the observable changes between renders, * the previous subscription is disposed and a new one is created. * @param key Unique key for caching this subscription * @param observable The observable to subscribe to * @param onChange Callback invoked when the value changes * @param options Additional observer options * @returns Tuple of [currentValue, setValue] */ useObservable = (key, observable, onChange, options) => { const alreadyUsed = this.observers.get(key); if (alreadyUsed) { if (alreadyUsed.observable !== observable) { alreadyUsed[Symbol.dispose](); const observer = observable.subscribe(onChange, options); this.observers.set(key, observer); return [observable.getValue(), observable.setValue.bind(observable)]; } return [alreadyUsed.observable.getValue(), alreadyUsed.observable.setValue.bind(alreadyUsed.observable)]; } const observer = observable.subscribe(onChange, options); this.observers.set(key, observer); return [observable.getValue(), observable.setValue.bind(observable)]; }; stateObservers = new Map(); /** * Creates or retrieves a local state observable by key. * State is persisted across re-renders and disposed with the component. * @param key Unique key for caching this state * @param initialValue Initial value used on first call * @param callback Callback invoked when the state changes * @returns Tuple of [currentValue, setValue] */ useState = (key, initialValue, callback) => { if (!this.stateObservers.has(key)) { const newObservable = new ObservableValue(initialValue); this.stateObservers.set(key, newObservable); newObservable.subscribe(callback); } const observable = this.stateObservers.get(key); const setValue = (newValue) => { if (!observable.isDisposed) { observable.setValue(newValue); } }; return [observable.getValue(), setValue]; }; async [Symbol.asyncDispose]() { const disposeResult = await Promise.allSettled([...this.disposables].map(async ([_key, resource]) => { if (isDisposable(resource)) { resource[Symbol.dispose](); } if (isAsyncDisposable(resource)) { await resource[Symbol.asyncDispose](); } })); const fails = disposeResult.filter((r) => r.status === 'rejected'); if (fails && fails.length) { const error = new AggregatedError(`There was an error during disposing ${fails.length} stores: ${fails.map((f) => f.reason).join(', ')}`, fails); throw error; } this.disposables.clear(); this.disposableDeps.clear(); this.observers.forEach((r) => r[Symbol.dispose]()); this.observers.clear(); this.stateObservers.forEach((r) => r[Symbol.dispose]()); this.stateObservers.clear(); } } //# sourceMappingURL=resource-manager.js.map