@mmuscat/angular-actions
Version:
A tiny (1kb) state management library for Angular Composition API.
102 lines • 13.4 kB
JavaScript
import { ErrorHandler, Inject, InjectionToken, isDevMode, NgModule, Self, } from "@angular/core";
import { inject, Service, subscribe, use, } from "@mmuscat/angular-composition-api";
import { merge, Notification } from "rxjs";
import * as i0 from "@angular/core";
class ActionObserver {
constructor(state, reduce) {
this.state = state;
this.reduce = reduce;
}
next(action) {
this.state(this.reduce(this.state.value, action));
}
}
class EffectObserver {
constructor(name, errorHandler) {
this.name = name;
this.errorHandler = errorHandler;
}
next(value) {
if (value instanceof Notification && value.kind === "E") {
this.error(value.error);
}
}
error(error) {
isDevMode() && console.warn(`Unhandled error in effect "${this.name}"`);
this.errorHandler.handleError(error);
}
}
function createStore(name, { reducers, effects, state }) {
const sink = subscribe();
const initialState = state();
const errorHandler = inject(ErrorHandler);
const injector = new Map();
for (const reducer of reducers) {
for (const [action, reduce] of inject(reducer).reducers) {
let actions = [];
const actionTypes = (Array.isArray(action) ? action : [action]);
for (const action of actionTypes) {
if (!injector.has(action)) {
const emitter = use(inject(action));
injector.set(action, emitter);
actions.push(emitter);
}
}
const state = use(initialState[reducer.overriddenName]);
injector.set(reducer, state);
sink.add(merge(...actions).subscribe(new ActionObserver(state, reduce)));
}
}
for (const effect of effects !== null && effects !== void 0 ? effects : []) {
const source = effect(store);
if (source) {
sink.add(source.subscribe(new EffectObserver(effect.name, errorHandler)));
}
}
function store(token) {
if (injector.has(token)) {
return injector.get(token);
}
throw new Error(`No provider found for ${token} in store ${name}`);
}
return store;
}
const STORE = new InjectionToken("STORE");
function createStoreProvider(name, options) {
function Store() {
return createStore(name, options);
}
Store.overriddenName = name;
Store.Provider = [
{ provide: Store, useClass: new Service(Store) },
options.reducers.map((reducer) => reducer.Provider),
];
return Store;
}
export const Store = createStoreProvider;
export class StoreModule {
constructor(store) {
this.store = store;
}
static config(store) {
return {
ngModule: StoreModule,
providers: [
store.Provider,
{ provide: STORE, useExisting: store },
],
};
}
}
StoreModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.0.4", ngImport: i0, type: StoreModule, deps: [{ token: STORE, self: true }], target: i0.ɵɵFactoryTarget.NgModule });
StoreModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", version: "12.0.4", ngImport: i0, type: StoreModule });
StoreModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "12.0.4", ngImport: i0, type: StoreModule });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.0.4", ngImport: i0, type: StoreModule, decorators: [{
type: NgModule
}], ctorParameters: function () { return [{ type: Store, decorators: [{
type: Self
}, {
type: Inject,
args: [STORE]
}] }]; } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"store.js","sourceRoot":"","sources":["../../../packages/store/src/store.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n   ErrorHandler,\n   Inject,\n   InjectionToken,\n   isDevMode,\n   ModuleWithProviders,\n   NgModule,\n   Self,\n   Type,\n} from \"@angular/core\"\nimport {\n   inject,\n   Service,\n   subscribe,\n   use,\n   Value,\n   ValueToken,\n} from \"@mmuscat/angular-composition-api\"\nimport { Action, ActionDispatcher } from \"./action\"\nimport { merge, Notification, Observable } from \"rxjs\"\n\nexport interface Store {\n   <T>(token: ValueToken<T>): T\n}\n\nexport type EffectFactory = (store: Store) => Observable<any> | void\n\nexport interface StoreOptions {\n   state: StateFactory\n   reducers: ValueToken<any>[]\n   effects?: EffectFactory[]\n}\n\nclass ActionObserver<T extends Action<any>> {\n   next(action: T) {\n      this.state(this.reduce(this.state.value, action))\n   }\n   constructor(\n      private state: Value<any>,\n      private reduce: (state: any, action: any) => any,\n   ) {}\n}\n\nclass EffectObserver {\n   next(value: any) {\n      if (value instanceof Notification && value.kind === \"E\") {\n         this.error(value.error)\n      }\n   }\n   error(error: unknown) {\n      isDevMode() && console.warn(`Unhandled error in effect \"${this.name}\"`)\n      this.errorHandler.handleError(error)\n   }\n   constructor(private name: string, private errorHandler: ErrorHandler) {}\n}\n\nfunction createStore(\n   name: string,\n   { reducers, effects, state }: StoreOptions,\n): Store {\n   const sink = subscribe()\n   const initialState = state()\n   const errorHandler = inject(ErrorHandler)\n   const injector = new Map()\n   for (const reducer of <any>reducers) {\n      for (const [action, reduce] of (<any>inject(reducer)).reducers) {\n         let actions: any[] = []\n         const actionTypes = (\n            Array.isArray(action) ? action : [action]\n         ) as Type<ActionDispatcher<any, any>>[]\n         for (const action of actionTypes) {\n            if (!injector.has(action)) {\n               const emitter = use(inject(action))\n               injector.set(action, emitter)\n               actions.push(emitter)\n            }\n         }\n         const state = use(initialState[reducer.overriddenName])\n         injector.set(reducer, state)\n         sink.add(\n            merge(...actions).subscribe(\n               new ActionObserver(state, reduce) as any,\n            ),\n         )\n      }\n   }\n\n   for (const effect of effects ?? []) {\n      const source = effect(store)\n      if (source) {\n         sink.add(\n            source.subscribe(new EffectObserver(effect.name, errorHandler)),\n         )\n      }\n   }\n\n   function store(token: ValueToken<any>) {\n      if (injector.has(token)) {\n         return injector.get(token)\n      }\n      throw new Error(`No provider found for ${token} in store ${name}`)\n   }\n\n   return store\n}\n\nconst STORE = new InjectionToken(\"STORE\")\n\nfunction createStoreProvider(name: string, options: StoreOptions) {\n   function Store() {\n      return createStore(name, options)\n   }\n   Store.overriddenName = name\n   Store.Provider = [\n      { provide: Store, useClass: new Service(Store) },\n      options.reducers.map((reducer) => reducer.Provider),\n   ]\n   return Store\n}\n\nexport type StateFactory = () => { [key: string]: any }\n\nexport type StoreFactory = ValueToken<Store>\n\ninterface StoreStatic {\n   new (name: string, options: StoreOptions): ValueToken<Store>\n}\n\nexport const Store: StoreStatic = createStoreProvider as any\n\n@NgModule()\nexport class StoreModule {\n   static config(store: StoreFactory): ModuleWithProviders<StoreModule> {\n      return {\n         ngModule: StoreModule,\n         providers: [\n            (<any>store).Provider,\n            { provide: STORE, useExisting: store },\n         ],\n      }\n   }\n\n   constructor(@Self() @Inject(STORE) private store: Store) {}\n}\n"]}