Files
smartstate/ts/smartstate.classes.smartstate.ts

86 lines
2.8 KiB
TypeScript
Raw Permalink Normal View History

2022-03-25 13:31:21 +01:00
import * as plugins from './smartstate.plugins.js';
import { StatePart } from './smartstate.classes.statepart.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
*/
export class Smartstate<StatePartNameType extends string> {
public statePartMap: { [key in StatePartNameType]?: StatePart<StatePartNameType, any> } = {};
2020-11-29 23:51:05 +00:00
constructor() {}
/**
* Allows getting and initializing a new statepart
* initMode === 'soft' (default) - returns existing statepart if exists, creates new if not
* initMode === 'mandatory' - requires statepart to not exist, fails if it does
* initMode === 'force' - always creates new statepart, overwriting any existing
* initMode === 'persistent' - like 'soft' but with webstore persistence
2020-11-29 23:51:05 +00:00
* @param statePartNameArg
* @param initialArg
* @param initMode
*/
2023-10-03 16:20:34 +02:00
public async getStatePart<PayloadType>(
statePartNameArg: StatePartNameType,
2020-11-29 23:51:05 +00:00
initialArg?: PayloadType,
initMode: TInitMode = 'soft'
2023-10-03 16:20:34 +02:00
): Promise<StatePart<StatePartNameType, PayloadType>> {
const existingStatePart = this.statePartMap[statePartNameArg];
if (existingStatePart) {
switch (initMode) {
case 'mandatory':
throw new Error(
`State part '${statePartNameArg}' already exists, but initMode is 'mandatory'`
);
case 'force':
// Force mode: create new state part
return this.createStatePart<PayloadType>(statePartNameArg, initialArg, initMode);
case 'soft':
case 'persistent':
default:
// Return existing state part
return existingStatePart as StatePart<StatePartNameType, PayloadType>;
2020-11-29 23:51:05 +00:00
}
} else {
// State part doesn't exist
2020-11-29 23:51:05 +00:00
if (!initialArg) {
throw new Error(
`State part '${statePartNameArg}' does not exist and no initial state provided`
2020-11-29 23:51:05 +00:00
);
}
2023-10-03 13:19:38 +02:00
return this.createStatePart<PayloadType>(statePartNameArg, initialArg, initMode);
2020-11-29 23:51:05 +00:00
}
}
/**
* Creates a statepart
2020-11-29 23:51:05 +00:00
* @param statePartName
* @param initialPayloadArg
* @param initMode
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,
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,
initMode === 'persistent'
? {
dbName: 'smartstate',
storeName: statePartName,
}
: null
2023-10-03 13:19:38 +02:00
);
2023-10-03 19:19:54 +02:00
await newState.init();
2023-10-03 16:20:34 +02:00
const currentState = newState.getState();
await newState.setState({
...currentState,
...initialPayloadArg,
2023-10-03 16:20:34 +02:00
});
this.statePartMap[statePartName] = newState;
2020-11-29 23:51:05 +00:00
return newState;
}
}