2022-03-25 13:31:21 +01:00
|
|
|
import * as plugins from './smartstate.plugins.js';
|
|
|
|
|
import { StatePart } from './smartstate.classes.statepart.js';
|
2026-02-27 11:40:07 +00:00
|
|
|
import { computed } from './smartstate.classes.computed.js';
|
2020-11-29 23:51:05 +00:00
|
|
|
|
2023-10-03 13:19:38 +02:00
|
|
|
export type TInitMode = 'soft' | 'mandatory' | 'force' | 'persistent';
|
|
|
|
|
|
2020-11-29 23:51:05 +00:00
|
|
|
/**
|
|
|
|
|
* Smartstate takes care of providing state
|
|
|
|
|
*/
|
2024-10-02 15:59:41 +02:00
|
|
|
export class Smartstate<StatePartNameType extends string> {
|
|
|
|
|
public statePartMap: { [key in StatePartNameType]?: StatePart<StatePartNameType, any> } = {};
|
2020-11-29 23:51:05 +00:00
|
|
|
|
2026-02-02 01:05:57 +00:00
|
|
|
private pendingStatePartCreation: Map<string, Promise<StatePart<StatePartNameType, any>>> = new Map();
|
|
|
|
|
|
2026-02-27 11:40:07 +00:00
|
|
|
// Batch support
|
|
|
|
|
private batchDepth = 0;
|
|
|
|
|
private pendingNotifications = new Set<StatePart<any, any>>();
|
|
|
|
|
|
2020-11-29 23:51:05 +00:00
|
|
|
constructor() {}
|
|
|
|
|
|
2026-02-27 11:40:07 +00:00
|
|
|
/**
|
|
|
|
|
* whether state changes are currently being batched
|
|
|
|
|
*/
|
|
|
|
|
public get isBatching(): boolean {
|
|
|
|
|
return this.batchDepth > 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* registers a state part for deferred notification during a batch
|
|
|
|
|
*/
|
|
|
|
|
public registerPendingNotification(statePart: StatePart<any, any>): void {
|
|
|
|
|
this.pendingNotifications.add(statePart);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* batches multiple state updates so subscribers are only notified once all updates complete
|
|
|
|
|
*/
|
|
|
|
|
public async batch(updateFn: () => Promise<void> | void): Promise<void> {
|
|
|
|
|
this.batchDepth++;
|
|
|
|
|
try {
|
|
|
|
|
await updateFn();
|
|
|
|
|
} finally {
|
|
|
|
|
this.batchDepth--;
|
|
|
|
|
if (this.batchDepth === 0) {
|
|
|
|
|
const pending = [...this.pendingNotifications];
|
|
|
|
|
this.pendingNotifications.clear();
|
|
|
|
|
for (const sp of pending) {
|
|
|
|
|
await sp.notifyChange();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* creates a computed observable derived from multiple state parts
|
|
|
|
|
*/
|
|
|
|
|
public computed<TResult>(
|
|
|
|
|
sources: StatePart<StatePartNameType, any>[],
|
|
|
|
|
computeFn: (...states: any[]) => TResult,
|
|
|
|
|
): plugins.smartrx.rxjs.Observable<TResult> {
|
|
|
|
|
return computed(sources, computeFn);
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-29 23:51:05 +00:00
|
|
|
/**
|
|
|
|
|
* Allows getting and initializing a new statepart
|
|
|
|
|
*/
|
2023-10-03 16:20:34 +02:00
|
|
|
public async getStatePart<PayloadType>(
|
2024-10-02 15:59:41 +02:00
|
|
|
statePartNameArg: StatePartNameType,
|
2020-11-29 23:51:05 +00:00
|
|
|
initialArg?: PayloadType,
|
2025-07-29 19:26:03 +00:00
|
|
|
initMode: TInitMode = 'soft'
|
2023-10-03 16:20:34 +02:00
|
|
|
): Promise<StatePart<StatePartNameType, PayloadType>> {
|
2026-02-02 01:05:57 +00:00
|
|
|
// Return pending creation if one exists to prevent duplicate state parts
|
|
|
|
|
const pending = this.pendingStatePartCreation.get(statePartNameArg);
|
|
|
|
|
if (pending) {
|
|
|
|
|
return pending as Promise<StatePart<StatePartNameType, PayloadType>>;
|
|
|
|
|
}
|
|
|
|
|
|
2025-07-29 19:26:03 +00:00
|
|
|
const existingStatePart = this.statePartMap[statePartNameArg];
|
2026-02-02 01:05:57 +00:00
|
|
|
|
2025-07-29 19:26:03 +00:00
|
|
|
if (existingStatePart) {
|
|
|
|
|
switch (initMode) {
|
|
|
|
|
case 'mandatory':
|
|
|
|
|
throw new Error(
|
|
|
|
|
`State part '${statePartNameArg}' already exists, but initMode is 'mandatory'`
|
|
|
|
|
);
|
|
|
|
|
case 'force':
|
2026-02-27 11:40:07 +00:00
|
|
|
break;
|
2025-07-29 19:26:03 +00:00
|
|
|
case 'soft':
|
|
|
|
|
case 'persistent':
|
|
|
|
|
default:
|
|
|
|
|
return existingStatePart as StatePart<StatePartNameType, PayloadType>;
|
2020-11-29 23:51:05 +00:00
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
if (!initialArg) {
|
|
|
|
|
throw new Error(
|
2025-07-29 19:26:03 +00:00
|
|
|
`State part '${statePartNameArg}' does not exist and no initial state provided`
|
2020-11-29 23:51:05 +00:00
|
|
|
);
|
|
|
|
|
}
|
2026-02-02 01:05:57 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
const creationPromise = this.createStatePart<PayloadType>(statePartNameArg, initialArg, initMode);
|
|
|
|
|
this.pendingStatePartCreation.set(statePartNameArg, creationPromise);
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
const result = await creationPromise;
|
|
|
|
|
return result;
|
|
|
|
|
} finally {
|
|
|
|
|
this.pendingStatePartCreation.delete(statePartNameArg);
|
2020-11-29 23:51:05 +00:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/**
|
2024-10-02 15:59:41 +02:00
|
|
|
* Creates a statepart
|
2020-11-29 23:51:05 +00:00
|
|
|
*/
|
2023-10-03 16:20:34 +02:00
|
|
|
private async createStatePart<PayloadType>(
|
2020-11-29 23:51:05 +00:00
|
|
|
statePartName: StatePartNameType,
|
2023-10-03 13:19:38 +02:00
|
|
|
initialPayloadArg: PayloadType,
|
2025-07-29 19:26:03 +00:00
|
|
|
initMode: TInitMode = 'soft'
|
2023-10-03 16:20:34 +02:00
|
|
|
): Promise<StatePart<StatePartNameType, PayloadType>> {
|
2023-10-03 13:19:38 +02:00
|
|
|
const newState = new StatePart<StatePartNameType, PayloadType>(
|
|
|
|
|
statePartName,
|
2024-10-02 15:59:41 +02:00
|
|
|
initMode === 'persistent'
|
|
|
|
|
? {
|
|
|
|
|
dbName: 'smartstate',
|
|
|
|
|
storeName: statePartName,
|
|
|
|
|
}
|
|
|
|
|
: null
|
2023-10-03 13:19:38 +02:00
|
|
|
);
|
2026-02-27 11:40:07 +00:00
|
|
|
newState.smartstateRef = this;
|
2023-10-03 19:19:54 +02:00
|
|
|
await newState.init();
|
2023-10-03 16:20:34 +02:00
|
|
|
const currentState = newState.getState();
|
2026-02-02 01:05:57 +00:00
|
|
|
|
|
|
|
|
if (initMode === 'persistent' && currentState !== undefined) {
|
|
|
|
|
await newState.setState({
|
|
|
|
|
...initialPayloadArg,
|
|
|
|
|
...currentState,
|
|
|
|
|
});
|
|
|
|
|
} else {
|
|
|
|
|
await newState.setState(initialPayloadArg);
|
|
|
|
|
}
|
|
|
|
|
|
2024-10-02 15:59:41 +02:00
|
|
|
this.statePartMap[statePartName] = newState;
|
2020-11-29 23:51:05 +00:00
|
|
|
return newState;
|
|
|
|
|
}
|
2026-02-27 11:40:07 +00:00
|
|
|
}
|