ng-simple-state 17.0.2 → 17.0.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +33 -71
- package/esm2022/lib/ng-simple-state-common.mjs +141 -0
- package/esm2022/lib/rxjs/ng-simple-state-base-store.mjs +15 -116
- package/esm2022/lib/signal/ng-simple-state-base-store.mjs +14 -120
- package/esm2022/lib/storage/ng-simple-state-browser-storage.mjs +71 -0
- package/esm2022/lib/{ng-simple-state-local-storage.mjs → storage/ng-simple-state-local-storage.mjs} +1 -1
- package/esm2022/lib/{ng-simple-state-session-storage.mjs → storage/ng-simple-state-session-storage.mjs} +1 -1
- package/esm2022/lib/{ng-simple-state-dev-tool.mjs → tool/ng-simple-state-dev-tool.mjs} +1 -1
- package/esm2022/public-api.mjs +5 -4
- package/fesm2022/ng-simple-state.mjs +114 -180
- package/fesm2022/ng-simple-state.mjs.map +1 -1
- package/lib/ng-simple-state-common.d.ts +90 -0
- package/lib/rxjs/ng-simple-state-base-store.d.ts +5 -53
- package/lib/signal/ng-simple-state-base-store.d.ts +6 -58
- package/package.json +1 -1
- package/public-api.d.ts +4 -3
- package/esm2022/lib/ng-simple-state-browser-storage.mjs +0 -71
- /package/lib/{ng-simple-state-browser-storage.d.ts → storage/ng-simple-state-browser-storage.d.ts} +0 -0
- /package/lib/{ng-simple-state-local-storage.d.ts → storage/ng-simple-state-local-storage.d.ts} +0 -0
- /package/lib/{ng-simple-state-session-storage.d.ts → storage/ng-simple-state-session-storage.d.ts} +0 -0
- /package/lib/{ng-simple-state-dev-tool.d.ts → tool/ng-simple-state-dev-tool.d.ts} +0 -0
package/README.md
CHANGED
|
@@ -11,17 +11,15 @@ See the demos:
|
|
|
11
11
|
- [Tour of heroes](https://stackblitz.com/edit/ng-simple-state-tour-of-heroes?file=src%2Fapp%2Fhero.service.ts)
|
|
12
12
|
- [To Do List](https://stackblitz.com/edit/ng-simple-state-todo?file=src%2Fapp%2Fapp.component.ts)
|
|
13
13
|
|
|
14
|
-
##
|
|
15
|
-
|
|
16
|
-
### Get Started
|
|
14
|
+
## Get Started
|
|
17
15
|
|
|
18
|
-
|
|
16
|
+
### Step 1: install `ng-simple-state`
|
|
19
17
|
|
|
20
18
|
```bash
|
|
21
19
|
npm i ng-simple-state
|
|
22
20
|
```
|
|
23
21
|
|
|
24
|
-
|
|
22
|
+
### Step 2: Import `NgSimpleStateModule` into your `AppModule`
|
|
25
23
|
|
|
26
24
|
`NgSimpleStateModule` has some global optional config defined by `NgSimpleStateConfig` interface:
|
|
27
25
|
|
|
@@ -57,7 +55,14 @@ import { environment } from '../environments/environment';
|
|
|
57
55
|
export class AppModule {}
|
|
58
56
|
```
|
|
59
57
|
|
|
60
|
-
|
|
58
|
+
### Step 3: Chose your store
|
|
59
|
+
|
|
60
|
+
There are two type of store `NgSimpleStateBaseRxjsStore` based on RxJS `BehaviorSubject` and `NgSimpleStateBaseSignalStore` based on Angular `Signal`:
|
|
61
|
+
|
|
62
|
+
- [RxJS Store](#rxjs-store)
|
|
63
|
+
- [Signal Store](#signal-store)
|
|
64
|
+
|
|
65
|
+
## RxJS Store
|
|
61
66
|
|
|
62
67
|
This is an example for a counter store in a `src/app/counter-store.ts` file.
|
|
63
68
|
Obviously, you can create every store you want with every complexity you need.
|
|
@@ -70,18 +75,18 @@ export interface CounterState {
|
|
|
70
75
|
}
|
|
71
76
|
```
|
|
72
77
|
|
|
73
|
-
2) Define your store service by extending `
|
|
78
|
+
2) Define your store service by extending `NgSimpleStateBaseRxjsStore`, eg.:
|
|
74
79
|
|
|
75
80
|
```ts
|
|
76
81
|
import { Injectable } from '@angular/core';
|
|
77
|
-
import {
|
|
82
|
+
import { NgSimpleStateBaseRxjsStore } from 'ng-simple-state';
|
|
78
83
|
|
|
79
84
|
export interface CounterState {
|
|
80
85
|
count: number;
|
|
81
86
|
}
|
|
82
87
|
|
|
83
88
|
@Injectable()
|
|
84
|
-
export class CounterStore extends
|
|
89
|
+
export class CounterStore extends NgSimpleStateBaseRxjsStore<CounterState> {
|
|
85
90
|
|
|
86
91
|
}
|
|
87
92
|
```
|
|
@@ -90,14 +95,14 @@ export class CounterStore extends NgSimpleStateBaseStore<CounterState> {
|
|
|
90
95
|
|
|
91
96
|
```ts
|
|
92
97
|
import { Injectable } from '@angular/core';
|
|
93
|
-
import {
|
|
98
|
+
import { NgSimpleStateBaseRxjsStore, NgSimpleStateStoreConfig } from 'ng-simple-state';
|
|
94
99
|
|
|
95
100
|
export interface CounterState {
|
|
96
101
|
count: number;
|
|
97
102
|
}
|
|
98
103
|
|
|
99
104
|
@Injectable()
|
|
100
|
-
export class CounterStore extends
|
|
105
|
+
export class CounterStore extends NgSimpleStateBaseRxjsStore<CounterState> {
|
|
101
106
|
|
|
102
107
|
storeConfig(): NgSimpleStateStoreConfig {
|
|
103
108
|
return {
|
|
@@ -118,7 +123,7 @@ export class CounterStore extends NgSimpleStateBaseStore<CounterState> {
|
|
|
118
123
|
|
|
119
124
|
```ts
|
|
120
125
|
import { Injectable } from '@angular/core';
|
|
121
|
-
import {
|
|
126
|
+
import { NgSimpleStateBaseRxjsStore, NgSimpleStateStoreConfig } from 'ng-simple-state';
|
|
122
127
|
import { Observable } from 'rxjs';
|
|
123
128
|
|
|
124
129
|
export interface CounterState {
|
|
@@ -126,7 +131,7 @@ export interface CounterState {
|
|
|
126
131
|
}
|
|
127
132
|
|
|
128
133
|
@Injectable()
|
|
129
|
-
export class CounterStore extends
|
|
134
|
+
export class CounterStore extends NgSimpleStateBaseRxjsStore<CounterState> {
|
|
130
135
|
|
|
131
136
|
storeConfig(): NgSimpleStateStoreConfig {
|
|
132
137
|
return {
|
|
@@ -150,7 +155,7 @@ export class CounterStore extends NgSimpleStateBaseStore<CounterState> {
|
|
|
150
155
|
|
|
151
156
|
```ts
|
|
152
157
|
import { Injectable } from '@angular/core';
|
|
153
|
-
import {
|
|
158
|
+
import { NgSimpleStateBaseRxjsStore, NgSimpleStateStoreConfig } from 'ng-simple-state';
|
|
154
159
|
import { Observable } from 'rxjs';
|
|
155
160
|
|
|
156
161
|
export interface CounterState {
|
|
@@ -158,7 +163,7 @@ export interface CounterState {
|
|
|
158
163
|
}
|
|
159
164
|
|
|
160
165
|
@Injectable()
|
|
161
|
-
export class CounterStore extends
|
|
166
|
+
export class CounterStore extends NgSimpleStateBaseRxjsStore<CounterState> {
|
|
162
167
|
|
|
163
168
|
storeConfig(): NgSimpleStateStoreConfig {
|
|
164
169
|
return {
|
|
@@ -244,12 +249,12 @@ export class AppComponent {
|
|
|
244
249
|
|
|
245
250
|
### Manage component state without service
|
|
246
251
|
|
|
247
|
-
If you want manage just a component state without make a new service, your component can extend directly `
|
|
252
|
+
If you want manage just a component state without make a new service, your component can extend directly `NgSimpleStateBaseRxjsStore`:
|
|
248
253
|
|
|
249
254
|
```ts
|
|
250
255
|
import { Component, Injector } from '@angular/core';
|
|
251
256
|
import { CommonModule } from '@angular/common';
|
|
252
|
-
import {
|
|
257
|
+
import { NgSimpleStateBaseRxjsStore } from 'ng-simple-state';
|
|
253
258
|
import { Observable } from 'rxjs';
|
|
254
259
|
|
|
255
260
|
export interface CounterState {
|
|
@@ -264,7 +269,7 @@ export interface CounterState {
|
|
|
264
269
|
<button (click)="decrement()">-</button>
|
|
265
270
|
`
|
|
266
271
|
})
|
|
267
|
-
export class CounterComponent extends
|
|
272
|
+
export class CounterComponent extends NgSimpleStateBaseRxjsStore<CounterState> {
|
|
268
273
|
|
|
269
274
|
public counter$: Observable<number> = this.selectState(state => state.count);
|
|
270
275
|
|
|
@@ -301,10 +306,10 @@ If you need to inject something into your store (eg. `HttpClient`), you need to
|
|
|
301
306
|
```ts
|
|
302
307
|
import { Injectable, Injector } from '@angular/core';
|
|
303
308
|
import { HttpClient } from '@angular/common/http';
|
|
304
|
-
import {
|
|
309
|
+
import { NgSimpleStateBaseRxjsStore } from 'ng-simple-state';
|
|
305
310
|
|
|
306
311
|
@Injectable()
|
|
307
|
-
export class CounterStore extends
|
|
312
|
+
export class CounterStore extends NgSimpleStateBaseRxjsStore<CounterState> {
|
|
308
313
|
|
|
309
314
|
constructor(injector: Injector, private http: HttpClient) {
|
|
310
315
|
super(injector);
|
|
@@ -331,7 +336,7 @@ import { NgSimpleStateStoreConfig } from 'ng-simple-state';
|
|
|
331
336
|
|
|
332
337
|
|
|
333
338
|
@Injectable()
|
|
334
|
-
export class CounterStore extends
|
|
339
|
+
export class CounterStore extends NgSimpleStateBaseRxjsStore<CounterState> {
|
|
335
340
|
|
|
336
341
|
override storeConfig(): NgSimpleStateStoreConfig {
|
|
337
342
|
return {
|
|
@@ -410,7 +415,7 @@ This is an example for a todo list store in a `src/app/todo-store.ts` file.
|
|
|
410
415
|
|
|
411
416
|
```ts
|
|
412
417
|
import { Injectable } from '@angular/core';
|
|
413
|
-
import {
|
|
418
|
+
import { NgSimpleStateBaseRxjsStore } from 'ng-simple-state';
|
|
414
419
|
import { Observable } from 'rxjs';
|
|
415
420
|
|
|
416
421
|
export interface Todo {
|
|
@@ -422,7 +427,7 @@ export interface Todo {
|
|
|
422
427
|
export type TodoState = Array<Todo>;
|
|
423
428
|
|
|
424
429
|
@Injectable()
|
|
425
|
-
export class TodoStore extends
|
|
430
|
+
export class TodoStore extends NgSimpleStateBaseRxjsStore<TodoState> {
|
|
426
431
|
|
|
427
432
|
storeConfig(): NgSimpleStateStoreConfig {
|
|
428
433
|
return {
|
|
@@ -480,12 +485,12 @@ export class AppComponent {
|
|
|
480
485
|
```
|
|
481
486
|
|
|
482
487
|
|
|
483
|
-
###
|
|
488
|
+
### NgSimpleStateBaseRxjsStore API
|
|
484
489
|
|
|
485
490
|
```ts
|
|
486
491
|
@Injectable()
|
|
487
492
|
@Directive()
|
|
488
|
-
export abstract class
|
|
493
|
+
export abstract class NgSimpleStateBaseRxjsStore<S extends object | Array<any>> implements OnDestroy {
|
|
489
494
|
|
|
490
495
|
/**
|
|
491
496
|
* Return the observable of the state
|
|
@@ -558,51 +563,6 @@ export abstract class NgSimpleStateBaseStore<S extends object | Array<any>> impl
|
|
|
558
563
|
```
|
|
559
564
|
## Signal Store
|
|
560
565
|
|
|
561
|
-
### Get Started
|
|
562
|
-
|
|
563
|
-
#### Step 1: install `ng-simple-state`
|
|
564
|
-
|
|
565
|
-
```bash
|
|
566
|
-
npm i ng-simple-state
|
|
567
|
-
```
|
|
568
|
-
|
|
569
|
-
#### Step 2: Import `NgSimpleStateModule` into your `AppModule`
|
|
570
|
-
|
|
571
|
-
`NgSimpleStateModule` has some global optional config defined by `NgSimpleStateConfig` interface:
|
|
572
|
-
|
|
573
|
-
| Option | Description | Default |
|
|
574
|
-
| -------------------- | ----------------------------------------------------------------------------------------------- | ---------- |
|
|
575
|
-
| *enableDevTool* | if `true` enable `Redux DevTools` browser extension for inspect the state of the store. | `false` |
|
|
576
|
-
| *enableLocalStorage* | if `true` latest state of store is saved in local storage and reloaded on store initialization. | `false` |
|
|
577
|
-
| *persistentStorage* | Set the persistent storage `local` or `session`. | `local` |
|
|
578
|
-
|
|
579
|
-
_Side note: each store can be override the global configuration implementing `storeConfig()` method (see "Override global config")._
|
|
580
|
-
|
|
581
|
-
```ts
|
|
582
|
-
import { BrowserModule } from '@angular/platform-browser';
|
|
583
|
-
import { NgModule } from '@angular/core';
|
|
584
|
-
import { AppComponent } from './app.component';
|
|
585
|
-
import { CommonModule } from '@angular/common';
|
|
586
|
-
import { NgSimpleStateModule } from 'ng-simple-state';
|
|
587
|
-
import { environment } from '../environments/environment';
|
|
588
|
-
|
|
589
|
-
@NgModule({
|
|
590
|
-
declarations: [AppComponent],
|
|
591
|
-
imports: [
|
|
592
|
-
BrowserModule,
|
|
593
|
-
CommonModule,
|
|
594
|
-
NgSimpleStateModule.forRoot({
|
|
595
|
-
enableDevTool: !environment.production, // Enable Redux DevTools only in development mode
|
|
596
|
-
enableLocalStorage: false // Local storage state persistence is globally disabled
|
|
597
|
-
})
|
|
598
|
-
],
|
|
599
|
-
bootstrap: [AppComponent],
|
|
600
|
-
})
|
|
601
|
-
export class AppModule {}
|
|
602
|
-
```
|
|
603
|
-
|
|
604
|
-
#### Step 3: Create your store
|
|
605
|
-
|
|
606
566
|
This is an example for a counter store in a `src/app/counter-store.ts` file.
|
|
607
567
|
Obviously, you can create every store you want with every complexity you need.
|
|
608
568
|
|
|
@@ -891,6 +851,7 @@ The options are defined by `NgSimpleStateStoreConfig` interface:
|
|
|
891
851
|
| *enableLocalStorage* | if `true` latest state of store is saved in local storage and reloaded on store initialization. | `false` |
|
|
892
852
|
| *storeName* | The name used into `Redux DevTools` and local storage key. | Class name |
|
|
893
853
|
| *persistentStorage* | Set the persistent storage `local` or `session` | `local` |
|
|
854
|
+
| *comparator* | A function used to compare the previous and current state for equality. | `a === b` |
|
|
894
855
|
|
|
895
856
|
### Testing
|
|
896
857
|
|
|
@@ -1062,9 +1023,10 @@ export abstract class NgSimpleStateBaseSignalStore<S extends object | Array<any>
|
|
|
1062
1023
|
/**
|
|
1063
1024
|
* Select a store state
|
|
1064
1025
|
* @param selectFn State selector (if not provided return full state)
|
|
1026
|
+
* @param comparator A function used to compare the previous and current state for equality. Defaults to a `===` check.
|
|
1065
1027
|
* @returns Signal of the selected state
|
|
1066
1028
|
*/
|
|
1067
|
-
selectState<K>(selectFn?: (state: Readonly<S>) => K): Signal<K>;
|
|
1029
|
+
selectState<K>(selectFn?: (state: Readonly<S>) => K, comparator?: (previous: K, current: K) => boolean): Signal<K>;
|
|
1068
1030
|
|
|
1069
1031
|
/**
|
|
1070
1032
|
* Return the current store state (snapshot)
|
|
@@ -0,0 +1,141 @@
|
|
|
1
|
+
import { Inject, Injectable, Injector, Directive, isDevMode } from '@angular/core';
|
|
2
|
+
import { NgSimpleStateDevTool } from './tool/ng-simple-state-dev-tool';
|
|
3
|
+
import { NgSimpleStateLocalStorage } from './storage/ng-simple-state-local-storage';
|
|
4
|
+
import { NG_SIMPLE_STORE_CONFIG } from './ng-simple-state-models';
|
|
5
|
+
import { NgSimpleStateSessionStorage } from './storage/ng-simple-state-session-storage';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
8
|
+
export class NgSimpleStateBaseCommonStore {
|
|
9
|
+
constructor(injector) {
|
|
10
|
+
this.devMode = isDevMode();
|
|
11
|
+
const globalConfig = injector.get(NG_SIMPLE_STORE_CONFIG, {});
|
|
12
|
+
const storeConfig = this.storeConfig() || {};
|
|
13
|
+
this.localStoreConfig = { ...globalConfig, ...storeConfig };
|
|
14
|
+
this.localStorageIsEnabled = typeof this.localStoreConfig.enableLocalStorage === 'boolean' ? this.localStoreConfig.enableLocalStorage : false;
|
|
15
|
+
if (this.localStorageIsEnabled) {
|
|
16
|
+
if (this.localStoreConfig.persistentStorage === 'local') {
|
|
17
|
+
this.persistentStorage = injector.get(NgSimpleStateLocalStorage);
|
|
18
|
+
}
|
|
19
|
+
else if (this.localStoreConfig.persistentStorage === 'session') {
|
|
20
|
+
this.persistentStorage = injector.get(NgSimpleStateSessionStorage);
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
this.devToolIsEnabled = typeof this.localStoreConfig.enableDevTool === 'boolean' ? this.localStoreConfig.enableDevTool : false;
|
|
24
|
+
if (this.devToolIsEnabled) {
|
|
25
|
+
this.devTool = injector.get(NgSimpleStateDevTool);
|
|
26
|
+
}
|
|
27
|
+
this.storeName = typeof this.localStoreConfig.storeName === 'string' ? this.localStoreConfig.storeName : this.constructor.name;
|
|
28
|
+
if (typeof this.localStoreConfig.comparator === 'function') {
|
|
29
|
+
this.comparator = this.localStoreConfig.comparator;
|
|
30
|
+
}
|
|
31
|
+
if (this.localStorageIsEnabled && this.persistentStorage) {
|
|
32
|
+
const firstState = this.persistentStorage.getItem(this.storeName);
|
|
33
|
+
if (firstState) {
|
|
34
|
+
this.firstState = firstState;
|
|
35
|
+
}
|
|
36
|
+
}
|
|
37
|
+
if (!this.firstState) {
|
|
38
|
+
this.firstState = this.initialState(injector);
|
|
39
|
+
}
|
|
40
|
+
this.devToolSend(this.firstState, `initialState`);
|
|
41
|
+
this.isArray = Array.isArray(this.firstState);
|
|
42
|
+
}
|
|
43
|
+
/**
|
|
44
|
+
* When you override this method, you have to call the `super.ngOnDestroy()` method in your `ngOnDestroy()` method.
|
|
45
|
+
*/
|
|
46
|
+
ngOnDestroy() {
|
|
47
|
+
this.devToolSend(undefined, 'ngOnDestroy');
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Return the first loaded store state:
|
|
51
|
+
* the last saved state, if `enableLocalStorage` config is `true`;
|
|
52
|
+
* otherwise the initial state provided from `initialState()` method.
|
|
53
|
+
* @returns The first state
|
|
54
|
+
*/
|
|
55
|
+
getFirstState() {
|
|
56
|
+
return this.deepFreeze(this.firstState);
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* Reset store to first loaded store state:
|
|
60
|
+
* - the last saved state, if `enableLocalStorage` config is `true`
|
|
61
|
+
* - otherwise the initial state provided from `initialState()` method.
|
|
62
|
+
*/
|
|
63
|
+
resetState() {
|
|
64
|
+
return this.setState(() => this.firstState, 'resetState');
|
|
65
|
+
}
|
|
66
|
+
/**
|
|
67
|
+
* Restart the store to initial state provided from `initialState()` method
|
|
68
|
+
*/
|
|
69
|
+
restartState() {
|
|
70
|
+
return this.setState(() => this.initialState(), 'restartState');
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Send to dev tool a new state
|
|
74
|
+
* @param newState new state
|
|
75
|
+
* @param actionName The action name
|
|
76
|
+
* @returns True if dev tools are enabled
|
|
77
|
+
*/
|
|
78
|
+
devToolSend(newState, actionName) {
|
|
79
|
+
if (!this.devToolIsEnabled || !this.devTool) {
|
|
80
|
+
return false;
|
|
81
|
+
}
|
|
82
|
+
if (!actionName) {
|
|
83
|
+
// retrieve the parent (of parent) method into the stack trace
|
|
84
|
+
actionName = new Error().stack
|
|
85
|
+
?.split('\n')[this.stackPoint]
|
|
86
|
+
.trim()
|
|
87
|
+
.split(' ')[1]
|
|
88
|
+
.split('.')[1] || 'unknown';
|
|
89
|
+
}
|
|
90
|
+
if (!this.devTool.send(this.storeName, actionName, newState)) {
|
|
91
|
+
console.log(this.storeName + '.' + actionName, newState);
|
|
92
|
+
}
|
|
93
|
+
return true;
|
|
94
|
+
}
|
|
95
|
+
/**
|
|
96
|
+
* Recursively Object.freeze simple Javascript structures consisting of plain objects, arrays, and primitives.
|
|
97
|
+
* Make the data immutable.
|
|
98
|
+
* @returns immutable object
|
|
99
|
+
*/
|
|
100
|
+
deepFreeze(object) {
|
|
101
|
+
// No freezing in production (for better performance).
|
|
102
|
+
if (!this.devMode || !object) {
|
|
103
|
+
return object;
|
|
104
|
+
}
|
|
105
|
+
// When already frozen, we assume its children are frozen (for better performance).
|
|
106
|
+
// This should be true if you always use `deepFreeze` to freeze objects.
|
|
107
|
+
//
|
|
108
|
+
// Note that Object.isFrozen will also return `true` for primitives (numbers,
|
|
109
|
+
// strings, booleans, undefined, null), so there is no need to check for
|
|
110
|
+
// those explicitly.
|
|
111
|
+
if (Object.isFrozen(object)) {
|
|
112
|
+
return object;
|
|
113
|
+
}
|
|
114
|
+
// At this point we know that we're dealing with either an array or plain object, so
|
|
115
|
+
// just freeze it and recurse on its values.
|
|
116
|
+
Object.freeze(object);
|
|
117
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
118
|
+
Object.keys(object).forEach(key => this.deepFreeze(object[key]));
|
|
119
|
+
return object;
|
|
120
|
+
}
|
|
121
|
+
/**
|
|
122
|
+
* Persist state to storage
|
|
123
|
+
*/
|
|
124
|
+
statePersist(state) {
|
|
125
|
+
if (this.localStorageIsEnabled && this.persistentStorage) {
|
|
126
|
+
this.persistentStorage.setItem(this.storeName, state);
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: NgSimpleStateBaseCommonStore, deps: [{ token: Injector }], target: i0.ɵɵFactoryTarget.Directive }); }
|
|
130
|
+
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.0.8", type: NgSimpleStateBaseCommonStore, ngImport: i0 }); }
|
|
131
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: NgSimpleStateBaseCommonStore }); }
|
|
132
|
+
}
|
|
133
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.8", ngImport: i0, type: NgSimpleStateBaseCommonStore, decorators: [{
|
|
134
|
+
type: Injectable
|
|
135
|
+
}, {
|
|
136
|
+
type: Directive
|
|
137
|
+
}], ctorParameters: () => [{ type: i0.Injector, decorators: [{
|
|
138
|
+
type: Inject,
|
|
139
|
+
args: [Injector]
|
|
140
|
+
}] }] });
|
|
141
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-simple-state-common.js","sourceRoot":"","sources":["../../../../projects/ng-simple-state/src/lib/ng-simple-state-common.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, Injector, OnDestroy, Directive, isDevMode } from '@angular/core';\r\nimport { NgSimpleStateBrowserStorage } from './storage/ng-simple-state-browser-storage';\r\nimport { NgSimpleStateDevTool } from './tool/ng-simple-state-dev-tool';\r\nimport { NgSimpleStateLocalStorage } from './storage/ng-simple-state-local-storage';\r\nimport { NgSimpleStateStoreConfig, NG_SIMPLE_STORE_CONFIG } from './ng-simple-state-models';\r\nimport { NgSimpleStateSessionStorage } from './storage/ng-simple-state-session-storage';\r\n\r\n@Injectable()\r\n@Directive()\r\n// eslint-disable-next-line @typescript-eslint/no-explicit-any\r\nexport abstract class NgSimpleStateBaseCommonStore<S extends object | Array<any>> implements OnDestroy {\r\n\r\n    protected abstract stackPoint: number;\r\n    protected localStorageIsEnabled: boolean;\r\n    protected devToolIsEnabled: boolean;\r\n    protected devTool!: NgSimpleStateDevTool;\r\n    protected persistentStorage!: NgSimpleStateBrowserStorage;\r\n    protected localStoreConfig: NgSimpleStateStoreConfig;\r\n    protected storeName: string;\r\n    protected firstState!: S;\r\n    protected isArray: boolean;\r\n    protected devMode: boolean = isDevMode();\r\n    protected comparator!: <K>(previous: K, current: K) => boolean;\r\n\r\n    constructor(@Inject(Injector) injector: Injector) {\r\n\r\n        const globalConfig = injector.get(NG_SIMPLE_STORE_CONFIG, {});\r\n        const storeConfig = this.storeConfig() || {};\r\n        this.localStoreConfig = { ...globalConfig, ...storeConfig };\r\n\r\n        this.localStorageIsEnabled = typeof this.localStoreConfig.enableLocalStorage === 'boolean' ? this.localStoreConfig.enableLocalStorage : false;\r\n\r\n        if (this.localStorageIsEnabled) {\r\n            if (this.localStoreConfig.persistentStorage === 'local') {\r\n                this.persistentStorage = injector.get(NgSimpleStateLocalStorage);\r\n            } else if (this.localStoreConfig.persistentStorage === 'session') {\r\n                this.persistentStorage = injector.get(NgSimpleStateSessionStorage);\r\n            }\r\n        }\r\n\r\n        this.devToolIsEnabled = typeof this.localStoreConfig.enableDevTool === 'boolean' ? this.localStoreConfig.enableDevTool : false;\r\n        if (this.devToolIsEnabled) {\r\n            this.devTool = injector.get(NgSimpleStateDevTool);\r\n        }\r\n\r\n        this.storeName = typeof this.localStoreConfig.storeName === 'string' ? this.localStoreConfig.storeName : this.constructor.name;\r\n        if (typeof this.localStoreConfig.comparator === 'function') {\r\n            this.comparator = this.localStoreConfig.comparator;\r\n        }\r\n\r\n        if (this.localStorageIsEnabled && this.persistentStorage) {\r\n            const firstState = this.persistentStorage.getItem<S>(this.storeName);\r\n            if (firstState) {\r\n                this.firstState = firstState;\r\n            }\r\n        }\r\n        if (!this.firstState) {\r\n            this.firstState = this.initialState(injector);\r\n        }\r\n\r\n        this.devToolSend(this.firstState, `initialState`);\r\n\r\n        this.isArray = Array.isArray(this.firstState);\r\n    }\r\n\r\n    /**\r\n     * When you override this method, you have to call the `super.ngOnDestroy()` method in your `ngOnDestroy()` method.\r\n     */\r\n    ngOnDestroy(): void {\r\n        this.devToolSend(undefined, 'ngOnDestroy');\r\n    }\r\n\r\n    /**\r\n     * Override this method for set a specific config for the store\r\n     * @returns NgSimpleStateStoreConfig\r\n     */\r\n    protected abstract storeConfig(): NgSimpleStateStoreConfig;\r\n\r\n    /**\r\n     * Set into the store the initial state\r\n     * @param injector current Injector\r\n     * @returns The state object\r\n     */\r\n    protected abstract initialState(injector?: Injector): S;\r\n\r\n    /**\r\n     * Select a store state\r\n     * @param selectFn State selector (if not provided return full state)\r\n     * @param comparator A function used to compare the previous and current state for equality. Defaults to a `===` check.\r\n     * @returns Observable of the selected state\r\n     */\r\n    // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n    abstract selectState<K>(selectFn?: (state: Readonly<S>) => K, comparator?: (previous: K, current: K) => boolean): any;\r\n\r\n    /**\r\n     * Set a new state\r\n     * @param selectFn State reducer\r\n     * @param actionName The action label into Redux DevTools (default is parent function name)\r\n     * @returns True if the state is changed\r\n     */\r\n    abstract setState(stateFn: (currentState: Readonly<S>) => Partial<S>, actionName?: string): boolean;\r\n\r\n    /**\r\n     * Return the current store state (snapshot)\r\n     * @returns The current state\r\n     */\r\n    abstract getCurrentState(): Readonly<S>;\r\n\r\n    /**\r\n     * Return the first loaded store state:\r\n     * the last saved state, if `enableLocalStorage` config is `true`;\r\n     * otherwise the initial state provided from `initialState()` method.\r\n     * @returns The first state\r\n     */\r\n    getFirstState(): Readonly<S> | null {\r\n        return this.deepFreeze(this.firstState);\r\n    }\r\n\r\n    /**\r\n     * Reset store to first loaded store state:\r\n     *  - the last saved state, if `enableLocalStorage` config is `true`\r\n     *  - otherwise the initial state provided from `initialState()` method.\r\n     */\r\n    resetState(): boolean {\r\n        return this.setState(() => this.firstState, 'resetState');\r\n    }\r\n\r\n    /**\r\n     * Restart the store to initial state provided from `initialState()` method\r\n     */\r\n    restartState(): boolean {\r\n        return this.setState(() => this.initialState(), 'restartState');\r\n    }\r\n\r\n    /**\r\n     * Send to dev tool a new state\r\n     * @param newState new state\r\n     * @param actionName The action name\r\n     * @returns True if dev tools are enabled\r\n     */\r\n    protected devToolSend(newState: S | undefined, actionName?: string): boolean {\r\n        if (!this.devToolIsEnabled || !this.devTool) {\r\n            return false;\r\n        }\r\n        if (!actionName) {\r\n            // retrieve the parent (of parent) method into the stack trace\r\n            actionName = new Error().stack\r\n                ?.split('\\n')[this.stackPoint]\r\n                .trim()\r\n                .split(' ')[1]\r\n                .split('.')[1] || 'unknown';\r\n        }\r\n        if (!this.devTool.send(this.storeName, actionName, newState)) {\r\n            console.log(this.storeName + '.' + actionName, newState);\r\n        }\r\n        return true;\r\n    }\r\n\r\n    /**\r\n     * Recursively Object.freeze simple Javascript structures consisting of plain objects, arrays, and primitives.\r\n     * Make the data immutable.\r\n     * @returns immutable object\r\n     */\r\n    protected deepFreeze(object: S): Readonly<S> {\r\n        // No freezing in production (for better performance).\r\n        if (!this.devMode || !object) {\r\n            return object as Readonly<S>;\r\n        }\r\n\r\n        // When already frozen, we assume its children are frozen (for better performance).\r\n        // This should be true if you always use `deepFreeze` to freeze objects.\r\n        //\r\n        // Note that Object.isFrozen will also return `true` for primitives (numbers,\r\n        // strings, booleans, undefined, null), so there is no need to check for\r\n        // those explicitly.\r\n        if (Object.isFrozen(object)) {\r\n            return object as Readonly<S>;\r\n        }\r\n\r\n        // At this point we know that we're dealing with either an array or plain object, so\r\n        // just freeze it and recurse on its values.\r\n        Object.freeze(object);\r\n        // eslint-disable-next-line @typescript-eslint/no-explicit-any\r\n        Object.keys(object).forEach(key => this.deepFreeze((object as any)[key]));\r\n\r\n        return object as Readonly<S>;\r\n    }\r\n\r\n    /**\r\n     * Persist state to storage\r\n     */\r\n    protected statePersist(state: S) {\r\n        if (this.localStorageIsEnabled && this.persistentStorage) {\r\n            this.persistentStorage.setItem<S>(this.storeName, state);\r\n        }\r\n    }\r\n}\r\n"]}
|