blob: 170bb97769a7dfe2c3f40769a87d68ae4ff7e9e3 [file] [log] [blame]
// Copyright (C) 2023 The Android Open Source Project
//
// 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.
import produce, {Draft} from 'immer';
import {Disposable} from '../base/disposable';
import {lookupPath, Path} from '../base/object_utils';
import {exists} from '../base/utils';
export class StoreError extends Error {
constructor(message: string) {
super(message);
this.name = 'StoreError';
}
}
// Describes a generic edit on a store of type T.
export type Edit<T> = (draft: Draft<T>) => void;
// Describes a callback from a store notifying subscribers of state changes.
export type SubscriptionCallback<T> = (state: T, oldState: T) => void;
// Describes a generic store of type T where T is the type of the state object
// the store manages.
// A store can be edited, subscribed to, and have proxies created from it.
export interface Store<T> extends Disposable {
// Access to the store's state.
// This state should be treated as immutable. It may be frozen in the future.
get state(): T;
// Apply one or multiple edits. Multiple edits are applied atomically.
// If any edits throw, the state is left unchanged.
// Subscribers are only called after all edits are made.
// Note: Purposely avoided using ...rest parameters here to avoid overflowing
// the stack when passing huge arrays of edits.
edit(edits: Edit<T>|Edit<T>[]): void;
// Create a new proxy store on a sub-tree of the state.
// This proxy store looks and feels like a regular store, but acts only on a
// specific subtree within its root store.
//
// The path doesn't need to exist in the object, but operations will be
// limited until it does.
// When the path is missing:
// - Accessing state on the proxy will throw.
// - Calling edit() on the proxy will throw.
// - The proxy won't notify subscribers the first time the subtree becomes
// defined, or when it becomes undefined.
createProxy<SubStateT>(path: Path): Store<SubStateT>;
// Register to be notified when edits are made to the store.
// The callback will be called whenever edits are made to the state managed by
// this store.
// Callbacks are passed the old state (state of the store before applying the
// edit(s)) and the new (current) state.
subscribe(callback: SubscriptionCallback<T>): Disposable;
}
// Factory method to create a new store, which lets the underlying store
// implementation change arbitrarily.
export function createStore<T>(initialState: T) {
return new StoreImpl<T>(initialState);
}
// Root store implementaiton.
class StoreImpl<T> implements Store<T> {
private _state: T;
private subscriptions = new Set<SubscriptionCallback<T>>();
constructor(initialState: T) {
this._state = initialState;
}
dispose() {
// No-op
}
get state() {
return this._state;
}
edit(edit: Edit<T>|Edit<T>[]): void {
if (Array.isArray(edit)) {
return this.applyEdits(edit);
} else {
return this.applyEdits([edit]);
}
}
private applyEdits(edits: Edit<T>[]): void {
const oldState = this._state;
let newState = oldState;
edits.forEach((edit) => {
newState = produce(newState, edit);
});
// Notify subscribers only if the state has changed.
if (oldState !== newState) {
this.subscriptions.forEach((sub) => {
sub(newState, oldState);
});
// It's important this is done last in order to keep updates atomic.
this._state = newState;
}
}
createProxy<ProxyT>(path: Path): Store<ProxyT> {
return new ProxyStoreImpl<T, ProxyT>(this, path);
}
subscribe(callback: SubscriptionCallback<T>): Disposable {
this.subscriptions.add(callback);
return {
dispose: () => {
this.subscriptions.delete(callback);
},
};
}
}
// A proxy store implemenation.
// This proxy implementation subscribes to the root store and thus must be
// disposed of properly by calling dispose() to avoid leaks.
// All edits are modified to operate on the subtree and passed back to the root
// store.
export class ProxyStoreImpl<RootT, T> implements Store<T> {
private subscriptions = new Set<SubscriptionCallback<T>>();
private rootSubscription;
constructor(
private rootStore: Store<RootT>,
private path: Path,
) {
this.rootSubscription = rootStore.subscribe(this.rootUpdateHandler);
}
dispose() {
this.rootSubscription.dispose();
}
private rootUpdateHandler = (newState: RootT, oldState: RootT) => {
const newSubState = lookupPath<T, RootT>(newState, this.path);
const oldSubState = lookupPath<T, RootT>(oldState, this.path);
if (exists(newSubState) && exists(oldSubState) &&
newSubState != oldSubState) {
this.subscriptions.forEach((subscription) => {
subscription(newSubState, oldSubState);
});
}
};
get state(): T {
const state = lookupPath<T, RootT>(this.rootStore.state, this.path);
if (state === undefined) {
throw new StoreError(`No such subtree: ${this.path}`);
}
return state;
}
edit(edit: Edit<T>|Edit<T>[]): void {
if (Array.isArray(edit)) {
this.applyEdits(edit);
} else {
this.applyEdits([edit]);
}
}
private applyEdits(edits: Edit<T>[]): void {
// Transform edits to work on the root store.
const rootEdits = edits.map(
(edit) => (state: Draft<RootT>) => {
// Extract subtree and apply edits to it.
const subtree = lookupPath<Draft<T>, Draft<RootT>>(state, this.path);
if (subtree === undefined) {
throw new StoreError(
`Unable to edit missing subtree: ${this.path}`);
}
edit(subtree);
},
);
// Apply edits to the root store.
this.rootStore.edit(rootEdits);
}
createProxy<NewSubStateT>(path: Path): Store<NewSubStateT> {
const fullPath = [...this.path, ...path];
return new ProxyStoreImpl<RootT, NewSubStateT>(this.rootStore, fullPath);
}
subscribe(callback: SubscriptionCallback<T>): Disposable {
this.subscriptions.add(callback);
return {
dispose: () => {
this.subscriptions.delete(callback);
},
};
}
}