@ngxs/storage-plugin 3.7.5 → 3.7.6-dev.master-95e4742
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/bundles/ngxs-storage-plugin.umd.js +177 -68
- package/bundles/ngxs-storage-plugin.umd.js.map +1 -1
- package/bundles/ngxs-storage-plugin.umd.min.js +1 -1
- package/bundles/ngxs-storage-plugin.umd.min.js.map +1 -1
- package/esm2015/index.js +2 -2
- package/esm2015/ngxs-storage-plugin.js +4 -3
- package/esm2015/src/engines.js +22 -0
- package/esm2015/src/internals/final-options.js +42 -0
- package/esm2015/src/internals/storage-key.js +47 -0
- package/esm2015/src/internals.js +13 -35
- package/esm2015/src/public_api.js +2 -1
- package/esm2015/src/storage.module.js +9 -3
- package/esm2015/src/storage.plugin.js +39 -32
- package/esm2015/src/symbols.js +9 -2
- package/esm5/index.js +2 -2
- package/esm5/ngxs-storage-plugin.js +4 -3
- package/esm5/src/engines.js +22 -0
- package/esm5/src/internals/final-options.js +43 -0
- package/esm5/src/internals/storage-key.js +47 -0
- package/esm5/src/internals.js +13 -35
- package/esm5/src/public_api.js +2 -1
- package/esm5/src/storage.module.js +9 -3
- package/esm5/src/storage.plugin.js +43 -35
- package/esm5/src/symbols.js +9 -2
- package/fesm2015/ngxs-storage-plugin.js +169 -65
- package/fesm2015/ngxs-storage-plugin.js.map +1 -1
- package/fesm5/ngxs-storage-plugin.js +173 -68
- package/fesm5/ngxs-storage-plugin.js.map +1 -1
- package/ngxs-storage-plugin.d.ts +2 -1
- package/ngxs-storage-plugin.metadata.json +1 -1
- package/package.json +4 -4
- package/src/engines.d.ts +4 -0
- package/src/internals/final-options.d.ts +10 -0
- package/src/internals/storage-key.d.ts +17 -0
- package/src/internals.d.ts +3 -9
- package/src/public_api.d.ts +1 -0
- package/src/storage.plugin.d.ts +3 -4
- package/src/symbols.d.ts +9 -3
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { InjectionToken, Injectable, Inject, PLATFORM_ID, NgModule } from '@angular/core';
|
|
1
|
+
import { InjectionToken, Injectable, Inject, PLATFORM_ID, Injector, NgModule } from '@angular/core';
|
|
2
2
|
import { StateToken, actionMatcher, InitState, UpdateState, getValue, setValue, NGXS_PLUGINS } from '@ngxs/store';
|
|
3
3
|
import { isPlatformServer } from '@angular/common';
|
|
4
4
|
import { tap } from 'rxjs/operators';
|
|
@@ -22,6 +22,13 @@ if (false) {
|
|
|
22
22
|
* @type {?|undefined}
|
|
23
23
|
*/
|
|
24
24
|
NgxsStoragePluginOptions.prototype.key;
|
|
25
|
+
/**
|
|
26
|
+
* The namespace is used to prefix the key for the state slice. This is
|
|
27
|
+
* necessary when running micro frontend applications which use storage plugin.
|
|
28
|
+
* The namespace will eliminate the conflict between keys that might overlap.
|
|
29
|
+
* @type {?|undefined}
|
|
30
|
+
*/
|
|
31
|
+
NgxsStoragePluginOptions.prototype.namespace;
|
|
25
32
|
/**
|
|
26
33
|
* Storage engine to use. Deaults to localStorage but can provide
|
|
27
34
|
*
|
|
@@ -62,7 +69,7 @@ if (false) {
|
|
|
62
69
|
NgxsStoragePluginOptions.prototype.afterDeserialize = function (obj, key) { };
|
|
63
70
|
}
|
|
64
71
|
/** @type {?} */
|
|
65
|
-
const NGXS_STORAGE_PLUGIN_OPTIONS = new InjectionToken('
|
|
72
|
+
const NGXS_STORAGE_PLUGIN_OPTIONS = new InjectionToken('NGXS_STORAGE_PLUGIN_OPTIONS');
|
|
66
73
|
/** @type {?} */
|
|
67
74
|
const STORAGE_ENGINE = new InjectionToken('STORAGE_ENGINE');
|
|
68
75
|
/**
|
|
@@ -99,47 +106,16 @@ if (false) {
|
|
|
99
106
|
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
|
|
100
107
|
*/
|
|
101
108
|
/**
|
|
102
|
-
*
|
|
103
|
-
*
|
|
109
|
+
* The following key is used to store the entire serialized
|
|
110
|
+
* state when there's no specific state provided.
|
|
104
111
|
* @type {?}
|
|
105
112
|
*/
|
|
106
113
|
const DEFAULT_STATE_KEY = '@@STATE';
|
|
107
|
-
/**
|
|
108
|
-
* This key is used to retrieve static metadatas on state classes.
|
|
109
|
-
* This constant is taken from the core codebase
|
|
110
|
-
* @type {?}
|
|
111
|
-
*/
|
|
112
|
-
const META_OPTIONS_KEY = 'NGXS_OPTIONS_META';
|
|
113
|
-
/**
|
|
114
|
-
* @param {?} key
|
|
115
|
-
* @return {?}
|
|
116
|
-
*/
|
|
117
|
-
function transformKeyOption(key) {
|
|
118
|
-
if (!Array.isArray(key)) {
|
|
119
|
-
key = [key];
|
|
120
|
-
}
|
|
121
|
-
return key.map((/**
|
|
122
|
-
* @param {?} token
|
|
123
|
-
* @return {?}
|
|
124
|
-
*/
|
|
125
|
-
(token) => {
|
|
126
|
-
// If it has the `NGXS_OPTIONS_META` key then it means the developer
|
|
127
|
-
// has provided state class like `key: [AuthState]`.
|
|
128
|
-
if (token.hasOwnProperty(META_OPTIONS_KEY)) {
|
|
129
|
-
// The `name` property will be an actual state name or a `StateToken`.
|
|
130
|
-
token = ((/** @type {?} */ (token)))[META_OPTIONS_KEY].name;
|
|
131
|
-
}
|
|
132
|
-
return token instanceof StateToken ? token.getName() : ((/** @type {?} */ (token)));
|
|
133
|
-
}));
|
|
134
|
-
}
|
|
135
114
|
/**
|
|
136
115
|
* @param {?} options
|
|
137
116
|
* @return {?}
|
|
138
117
|
*/
|
|
139
118
|
function storageOptionsFactory(options) {
|
|
140
|
-
if (options !== undefined && options.key) {
|
|
141
|
-
options.key = transformKeyOption(options.key);
|
|
142
|
-
}
|
|
143
119
|
return Object.assign({ key: [DEFAULT_STATE_KEY], storage: 0 /* LocalStorage */, serialize: JSON.stringify, deserialize: JSON.parse, beforeSerialize: (/**
|
|
144
120
|
* @param {?} obj
|
|
145
121
|
* @return {?}
|
|
@@ -167,6 +143,101 @@ function engineFactory(options, platformId) {
|
|
|
167
143
|
}
|
|
168
144
|
return null;
|
|
169
145
|
}
|
|
146
|
+
/**
|
|
147
|
+
* @param {?} key
|
|
148
|
+
* @param {?=} options
|
|
149
|
+
* @return {?}
|
|
150
|
+
*/
|
|
151
|
+
function getStorageKey(key, options) {
|
|
152
|
+
// Prepends the `namespace` option to any key if it's been provided by a user.
|
|
153
|
+
// So `@@STATE` becomes `my-app:@@STATE`.
|
|
154
|
+
return options && options.namespace ? `${options.namespace}:${key}` : key;
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
/**
|
|
158
|
+
* @fileoverview added by tsickle
|
|
159
|
+
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
|
|
160
|
+
*/
|
|
161
|
+
/**
|
|
162
|
+
* This enables the user to provide a storage engine per individual key.
|
|
163
|
+
* @record
|
|
164
|
+
*/
|
|
165
|
+
function KeyWithExplicitEngine() { }
|
|
166
|
+
if (false) {
|
|
167
|
+
/** @type {?} */
|
|
168
|
+
KeyWithExplicitEngine.prototype.key;
|
|
169
|
+
/** @type {?} */
|
|
170
|
+
KeyWithExplicitEngine.prototype.engine;
|
|
171
|
+
}
|
|
172
|
+
/**
|
|
173
|
+
* Determines whether the provided key has the following structure.
|
|
174
|
+
* @param {?} key
|
|
175
|
+
* @return {?}
|
|
176
|
+
*/
|
|
177
|
+
function isKeyWithExplicitEngine(key) {
|
|
178
|
+
return key != null && !!key.engine;
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* This symbol is used to store the metadata on state classes.
|
|
182
|
+
* @type {?}
|
|
183
|
+
*/
|
|
184
|
+
const META_OPTIONS_KEY = 'NGXS_OPTIONS_META';
|
|
185
|
+
/**
|
|
186
|
+
* @param {?} storageKey
|
|
187
|
+
* @return {?}
|
|
188
|
+
*/
|
|
189
|
+
function exctractStringKey(storageKey) {
|
|
190
|
+
// Extract the actual key out of the `{ key, engine }` structure.
|
|
191
|
+
if (isKeyWithExplicitEngine(storageKey)) {
|
|
192
|
+
storageKey = storageKey.key;
|
|
193
|
+
}
|
|
194
|
+
// Given the `storageKey` is a class, for instance, `AuthState`.
|
|
195
|
+
// We should retrieve its metadata and the `name` property.
|
|
196
|
+
// The `name` property might be a string (state name) or a state token.
|
|
197
|
+
if (storageKey.hasOwnProperty(META_OPTIONS_KEY)) {
|
|
198
|
+
storageKey = ((/** @type {?} */ (storageKey)))[META_OPTIONS_KEY].name;
|
|
199
|
+
}
|
|
200
|
+
return storageKey instanceof StateToken ? storageKey.getName() : (/** @type {?} */ (storageKey));
|
|
201
|
+
}
|
|
202
|
+
|
|
203
|
+
/**
|
|
204
|
+
* @fileoverview added by tsickle
|
|
205
|
+
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
|
|
206
|
+
*/
|
|
207
|
+
/**
|
|
208
|
+
* @record
|
|
209
|
+
*/
|
|
210
|
+
function FinalNgxsStoragePluginOptions() { }
|
|
211
|
+
if (false) {
|
|
212
|
+
/** @type {?} */
|
|
213
|
+
FinalNgxsStoragePluginOptions.prototype.keysWithEngines;
|
|
214
|
+
}
|
|
215
|
+
/** @type {?} */
|
|
216
|
+
const FINAL_NGXS_STORAGE_PLUGIN_OPTIONS = new InjectionToken('FINAL_NGXS_STORAGE_PLUGIN_OPTIONS');
|
|
217
|
+
/**
|
|
218
|
+
* @param {?} injector
|
|
219
|
+
* @param {?} options
|
|
220
|
+
* @return {?}
|
|
221
|
+
*/
|
|
222
|
+
function createFinalStoragePluginOptions(injector, options) {
|
|
223
|
+
/** @type {?} */
|
|
224
|
+
const storageKeys = Array.isArray(options.key) ? options.key : [(/** @type {?} */ (options.key))];
|
|
225
|
+
/** @type {?} */
|
|
226
|
+
const keysWithEngines = storageKeys.map((/**
|
|
227
|
+
* @param {?} storageKey
|
|
228
|
+
* @return {?}
|
|
229
|
+
*/
|
|
230
|
+
(storageKey) => {
|
|
231
|
+
/** @type {?} */
|
|
232
|
+
const key = exctractStringKey(storageKey);
|
|
233
|
+
/** @type {?} */
|
|
234
|
+
const engine = isKeyWithExplicitEngine(storageKey)
|
|
235
|
+
? injector.get(storageKey.engine)
|
|
236
|
+
: injector.get(STORAGE_ENGINE);
|
|
237
|
+
return { key, engine };
|
|
238
|
+
}));
|
|
239
|
+
return Object.assign({}, options, { keysWithEngines });
|
|
240
|
+
}
|
|
170
241
|
|
|
171
242
|
/**
|
|
172
243
|
* @fileoverview added by tsickle
|
|
@@ -175,18 +246,14 @@ function engineFactory(options, platformId) {
|
|
|
175
246
|
class NgxsStoragePlugin {
|
|
176
247
|
/**
|
|
177
248
|
* @param {?} _options
|
|
178
|
-
* @param {?} _engine
|
|
179
249
|
* @param {?} _platformId
|
|
180
250
|
*/
|
|
181
|
-
constructor(_options,
|
|
251
|
+
constructor(_options, _platformId) {
|
|
182
252
|
this._options = _options;
|
|
183
|
-
this._engine = _engine;
|
|
184
253
|
this._platformId = _platformId;
|
|
185
|
-
|
|
186
|
-
// transformed by the `storageOptionsFactory` function that provided token.
|
|
187
|
-
this._keys = (/** @type {?} */ (this._options.key));
|
|
254
|
+
this._keysWithEngines = this._options.keysWithEngines;
|
|
188
255
|
// We default to `[DEFAULT_STATE_KEY]` if the user explicitly does not provide the `key` option.
|
|
189
|
-
this._usesDefaultStateKey = this.
|
|
256
|
+
this._usesDefaultStateKey = this._keysWithEngines.length === 1 && this._keysWithEngines[0].key === DEFAULT_STATE_KEY;
|
|
190
257
|
}
|
|
191
258
|
/**
|
|
192
259
|
* @param {?} state
|
|
@@ -195,7 +262,7 @@ class NgxsStoragePlugin {
|
|
|
195
262
|
* @return {?}
|
|
196
263
|
*/
|
|
197
264
|
handle(state, event, next) {
|
|
198
|
-
if (isPlatformServer(this._platformId)
|
|
265
|
+
if (isPlatformServer(this._platformId)) {
|
|
199
266
|
return next(state, event);
|
|
200
267
|
}
|
|
201
268
|
/** @type {?} */
|
|
@@ -211,16 +278,29 @@ class NgxsStoragePlugin {
|
|
|
211
278
|
if (isInitOrUpdateAction) {
|
|
212
279
|
/** @type {?} */
|
|
213
280
|
const addedStates = isUpdateAction && event.addedStates;
|
|
214
|
-
for (const key of this.
|
|
281
|
+
for (const { key, engine } of this._keysWithEngines) {
|
|
215
282
|
// We're checking what states have been added by NGXS and if any of these states should be handled by
|
|
216
283
|
// the storage plugin. For instance, we only want to deserialize the `auth` state, NGXS has added
|
|
217
284
|
// the `user` state, the storage plugin will be rerun and will do redundant deserialization.
|
|
218
285
|
// `usesDefaultStateKey` is necessary to check since `event.addedStates` never contains `@@STATE`.
|
|
219
|
-
if (!this._usesDefaultStateKey && addedStates
|
|
220
|
-
|
|
286
|
+
if (!this._usesDefaultStateKey && addedStates) {
|
|
287
|
+
// We support providing keys that can be deeply nested via dot notation, for instance,
|
|
288
|
+
// `keys: ['myState.myProperty']` is a valid key.
|
|
289
|
+
// The state name should always go first. The below code checks if the `key` includes dot
|
|
290
|
+
// notation and extracts the state name out of the key.
|
|
291
|
+
// Given the `key` is `myState.myProperty`, the `addedStates` will only contain `myState`.
|
|
292
|
+
/** @type {?} */
|
|
293
|
+
const dotNotationIndex = key.indexOf(DOT);
|
|
294
|
+
/** @type {?} */
|
|
295
|
+
const stateName = dotNotationIndex > -1 ? key.slice(0, dotNotationIndex) : key;
|
|
296
|
+
if (!addedStates.hasOwnProperty(stateName)) {
|
|
297
|
+
continue;
|
|
298
|
+
}
|
|
221
299
|
}
|
|
222
300
|
/** @type {?} */
|
|
223
|
-
|
|
301
|
+
const storageKey = getStorageKey(key, this._options);
|
|
302
|
+
/** @type {?} */
|
|
303
|
+
let storedValue = engine.getItem(storageKey);
|
|
224
304
|
if (storedValue !== 'undefined' && storedValue != null) {
|
|
225
305
|
try {
|
|
226
306
|
/** @type {?} */
|
|
@@ -231,7 +311,7 @@ class NgxsStoragePlugin {
|
|
|
231
311
|
// Caretaker note: we have still left the `typeof` condition in order to avoid
|
|
232
312
|
// creating a breaking change for projects that still use the View Engine.
|
|
233
313
|
if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
|
234
|
-
console.error(`Error ocurred while deserializing the ${
|
|
314
|
+
console.error(`Error ocurred while deserializing the ${storageKey} store value, falling back to empty object, the value obtained from the store: `, storedValue);
|
|
235
315
|
}
|
|
236
316
|
storedValue = {};
|
|
237
317
|
}
|
|
@@ -252,7 +332,7 @@ class NgxsStoragePlugin {
|
|
|
252
332
|
}));
|
|
253
333
|
}
|
|
254
334
|
if (!this._usesDefaultStateKey) {
|
|
255
|
-
state = setValue(state,
|
|
335
|
+
state = setValue(state, key, storedValue);
|
|
256
336
|
}
|
|
257
337
|
else {
|
|
258
338
|
// The `UpdateState` action is dispatched whenever the feature state is added.
|
|
@@ -295,16 +375,18 @@ class NgxsStoragePlugin {
|
|
|
295
375
|
*/
|
|
296
376
|
nextState => {
|
|
297
377
|
if (!isInitOrUpdateAction || (isInitOrUpdateAction && hasMigration)) {
|
|
298
|
-
for (const key of this.
|
|
378
|
+
for (const { key, engine } of this._keysWithEngines) {
|
|
379
|
+
/** @type {?} */
|
|
380
|
+
let storedValue = nextState;
|
|
299
381
|
/** @type {?} */
|
|
300
|
-
|
|
382
|
+
const storageKey = getStorageKey(key, this._options);
|
|
301
383
|
if (key !== DEFAULT_STATE_KEY) {
|
|
302
|
-
|
|
384
|
+
storedValue = getValue(nextState, key);
|
|
303
385
|
}
|
|
304
386
|
try {
|
|
305
387
|
/** @type {?} */
|
|
306
|
-
const
|
|
307
|
-
|
|
388
|
+
const newStoredValue = (/** @type {?} */ (this._options.beforeSerialize))(storedValue, key);
|
|
389
|
+
engine.setItem(storageKey, (/** @type {?} */ (this._options.serialize))(newStoredValue));
|
|
308
390
|
}
|
|
309
391
|
catch (error) {
|
|
310
392
|
// Caretaker note: we have still left the `typeof` condition in order to avoid
|
|
@@ -313,10 +395,10 @@ class NgxsStoragePlugin {
|
|
|
313
395
|
if (error &&
|
|
314
396
|
(error.name === 'QuotaExceededError' ||
|
|
315
397
|
error.name === 'NS_ERROR_DOM_QUOTA_REACHED')) {
|
|
316
|
-
console.error(`The ${
|
|
398
|
+
console.error(`The ${storageKey} store value exceeds the browser storage quota: `, storedValue);
|
|
317
399
|
}
|
|
318
400
|
else {
|
|
319
|
-
console.error(`Error ocurred while serializing the ${
|
|
401
|
+
console.error(`Error ocurred while serializing the ${storageKey} store value, value not updated, the value obtained from the store: `, storedValue);
|
|
320
402
|
}
|
|
321
403
|
}
|
|
322
404
|
}
|
|
@@ -330,8 +412,7 @@ NgxsStoragePlugin.decorators = [
|
|
|
330
412
|
];
|
|
331
413
|
/** @nocollapse */
|
|
332
414
|
NgxsStoragePlugin.ctorParameters = () => [
|
|
333
|
-
{ type: undefined, decorators: [{ type: Inject, args: [
|
|
334
|
-
{ type: undefined, decorators: [{ type: Inject, args: [STORAGE_ENGINE,] }] },
|
|
415
|
+
{ type: undefined, decorators: [{ type: Inject, args: [FINAL_NGXS_STORAGE_PLUGIN_OPTIONS,] }] },
|
|
335
416
|
{ type: String, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
|
|
336
417
|
];
|
|
337
418
|
if (false) {
|
|
@@ -339,7 +420,7 @@ if (false) {
|
|
|
339
420
|
* @type {?}
|
|
340
421
|
* @private
|
|
341
422
|
*/
|
|
342
|
-
NgxsStoragePlugin.prototype.
|
|
423
|
+
NgxsStoragePlugin.prototype._keysWithEngines;
|
|
343
424
|
/**
|
|
344
425
|
* @type {?}
|
|
345
426
|
* @private
|
|
@@ -350,17 +431,14 @@ if (false) {
|
|
|
350
431
|
* @private
|
|
351
432
|
*/
|
|
352
433
|
NgxsStoragePlugin.prototype._options;
|
|
353
|
-
/**
|
|
354
|
-
* @type {?}
|
|
355
|
-
* @private
|
|
356
|
-
*/
|
|
357
|
-
NgxsStoragePlugin.prototype._engine;
|
|
358
434
|
/**
|
|
359
435
|
* @type {?}
|
|
360
436
|
* @private
|
|
361
437
|
*/
|
|
362
438
|
NgxsStoragePlugin.prototype._platformId;
|
|
363
439
|
}
|
|
440
|
+
/** @type {?} */
|
|
441
|
+
const DOT = '.';
|
|
364
442
|
|
|
365
443
|
/**
|
|
366
444
|
* @fileoverview added by tsickle
|
|
@@ -395,6 +473,11 @@ class NgxsStoragePluginModule {
|
|
|
395
473
|
provide: STORAGE_ENGINE,
|
|
396
474
|
useFactory: engineFactory,
|
|
397
475
|
deps: [NGXS_STORAGE_PLUGIN_OPTIONS, PLATFORM_ID]
|
|
476
|
+
},
|
|
477
|
+
{
|
|
478
|
+
provide: FINAL_NGXS_STORAGE_PLUGIN_OPTIONS,
|
|
479
|
+
useFactory: createFinalStoragePluginOptions,
|
|
480
|
+
deps: [Injector, NGXS_STORAGE_PLUGIN_OPTIONS]
|
|
398
481
|
}
|
|
399
482
|
]
|
|
400
483
|
};
|
|
@@ -404,6 +487,27 @@ NgxsStoragePluginModule.decorators = [
|
|
|
404
487
|
{ type: NgModule }
|
|
405
488
|
];
|
|
406
489
|
|
|
490
|
+
/**
|
|
491
|
+
* @fileoverview added by tsickle
|
|
492
|
+
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
|
|
493
|
+
*/
|
|
494
|
+
/** @type {?} */
|
|
495
|
+
const LOCAL_STORAGE_ENGINE = new InjectionToken('LOCAL_STORAGE_ENGINE', {
|
|
496
|
+
providedIn: 'root',
|
|
497
|
+
factory: (/**
|
|
498
|
+
* @return {?}
|
|
499
|
+
*/
|
|
500
|
+
() => localStorage)
|
|
501
|
+
});
|
|
502
|
+
/** @type {?} */
|
|
503
|
+
const SESSION_STORAGE_ENGINE = new InjectionToken('SESSION_STORAGE_ENGINE', {
|
|
504
|
+
providedIn: 'root',
|
|
505
|
+
factory: (/**
|
|
506
|
+
* @return {?}
|
|
507
|
+
*/
|
|
508
|
+
() => sessionStorage)
|
|
509
|
+
});
|
|
510
|
+
|
|
407
511
|
/**
|
|
408
512
|
* @fileoverview added by tsickle
|
|
409
513
|
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
|
|
@@ -419,5 +523,5 @@ NgxsStoragePluginModule.decorators = [
|
|
|
419
523
|
* @suppress {checkTypes,extraRequire,missingOverride,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
|
|
420
524
|
*/
|
|
421
525
|
|
|
422
|
-
export { NGXS_STORAGE_PLUGIN_OPTIONS, NgxsStoragePlugin, NgxsStoragePluginModule, STORAGE_ENGINE, USER_OPTIONS as ɵa, storageOptionsFactory as
|
|
526
|
+
export { LOCAL_STORAGE_ENGINE, NGXS_STORAGE_PLUGIN_OPTIONS, NgxsStoragePlugin, NgxsStoragePluginModule, SESSION_STORAGE_ENGINE, STORAGE_ENGINE, USER_OPTIONS as ɵa, FINAL_NGXS_STORAGE_PLUGIN_OPTIONS as ɵc, createFinalStoragePluginOptions as ɵd, storageOptionsFactory as ɵe, engineFactory as ɵf };
|
|
423
527
|
//# sourceMappingURL=ngxs-storage-plugin.js.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ngxs-storage-plugin.js","sources":["ng://@ngxs/storage-plugin/src/symbols.ts","ng://@ngxs/storage-plugin/src/internals.ts","ng://@ngxs/storage-plugin/src/storage.plugin.ts","ng://@ngxs/storage-plugin/src/storage.module.ts"],"sourcesContent":["import { InjectionToken } from '@angular/core';\n\nimport { StorageKey } from './internals';\n\nexport const enum StorageOption {\n LocalStorage,\n SessionStorage\n}\n\nexport interface NgxsStoragePluginOptions {\n /**\n * Key for the state slice to store in the storage engine.\n */\n key?: undefined | StorageKey;\n\n /**\n * Storage engine to use. Deaults to localStorage but can provide\n *\n * sessionStorage or custom implementation of the StorageEngine interface\n */\n storage?: StorageOption;\n\n /**\n * Migration strategies.\n */\n migrations?: {\n /**\n * Version to key off.\n */\n version: number | string;\n\n /**\n * Method to migrate the previous state.\n */\n migrate: (state: any) => any;\n\n /**\n * Key to migrate.\n */\n key?: string;\n\n /**\n * Key for the version. Defaults to 'version'.\n */\n versionKey?: string;\n }[];\n\n /**\n * Serailizer for the object before its pushed into the engine.\n */\n serialize?(obj: any): string;\n\n /**\n * Deserializer for the object before its pulled out of the engine.\n */\n deserialize?(obj: any): any;\n\n /**\n * Method to alter object before serialization.\n */\n beforeSerialize?(obj: any, key: string): any;\n\n /**\n * Method to alter object after deserialization.\n */\n afterDeserialize?(obj: any, key: string): any;\n}\n\nexport const NGXS_STORAGE_PLUGIN_OPTIONS = new InjectionToken('NGXS_STORAGE_PLUGIN_OPTION');\n\nexport const STORAGE_ENGINE = new InjectionToken('STORAGE_ENGINE');\n\nexport interface StorageEngine {\n readonly length: number;\n getItem(key: string): any;\n setItem(key: string, val: any): void;\n removeItem(key: string): void;\n clear(): void;\n}\n","import { isPlatformServer } from '@angular/common';\nimport { StateClass } from '@ngxs/store/internals';\nimport { StateToken } from '@ngxs/store';\n\nimport { StorageOption, StorageEngine, NgxsStoragePluginOptions } from './symbols';\n\n/**\n * If the `key` option is not provided then the below constant\n * will be used as a default key\n */\nexport const DEFAULT_STATE_KEY = '@@STATE';\n\n/**\n * Internal type definition for the `key` option provided\n * in the `forRoot` method when importing module\n */\nexport type StorageKey =\n | string\n | StateClass\n | StateToken<any>\n | (string | StateClass | StateToken<any>)[];\n\n/**\n * This key is used to retrieve static metadatas on state classes.\n * This constant is taken from the core codebase\n */\nconst META_OPTIONS_KEY = 'NGXS_OPTIONS_META';\n\nfunction transformKeyOption(key: StorageKey): string[] {\n if (!Array.isArray(key)) {\n key = [key];\n }\n\n return key.map((token: string | StateClass | StateToken<any>) => {\n // If it has the `NGXS_OPTIONS_META` key then it means the developer\n // has provided state class like `key: [AuthState]`.\n if (token.hasOwnProperty(META_OPTIONS_KEY)) {\n // The `name` property will be an actual state name or a `StateToken`.\n token = (token as any)[META_OPTIONS_KEY].name;\n }\n\n return token instanceof StateToken ? token.getName() : (token as string);\n });\n}\n\nexport function storageOptionsFactory(\n options: NgxsStoragePluginOptions | undefined\n): NgxsStoragePluginOptions {\n if (options !== undefined && options.key) {\n options.key = transformKeyOption(options.key);\n }\n\n return {\n key: [DEFAULT_STATE_KEY],\n storage: StorageOption.LocalStorage,\n serialize: JSON.stringify,\n deserialize: JSON.parse,\n beforeSerialize: obj => obj,\n afterDeserialize: obj => obj,\n ...options\n };\n}\n\nexport function engineFactory(\n options: NgxsStoragePluginOptions,\n platformId: string\n): StorageEngine | null {\n if (isPlatformServer(platformId)) {\n return null;\n }\n\n if (options.storage === StorageOption.LocalStorage) {\n return localStorage;\n } else if (options.storage === StorageOption.SessionStorage) {\n return sessionStorage;\n }\n\n return null;\n}\n","import { PLATFORM_ID, Inject, Injectable } from '@angular/core';\nimport { isPlatformServer } from '@angular/common';\nimport { PlainObject } from '@ngxs/store/internals';\nimport {\n NgxsPlugin,\n setValue,\n getValue,\n InitState,\n UpdateState,\n actionMatcher,\n NgxsNextPluginFn\n} from '@ngxs/store';\nimport { tap } from 'rxjs/operators';\n\nimport {\n StorageEngine,\n NgxsStoragePluginOptions,\n STORAGE_ENGINE,\n NGXS_STORAGE_PLUGIN_OPTIONS\n} from './symbols';\nimport { DEFAULT_STATE_KEY } from './internals';\n\n/**\n * @description Will be provided through Terser global definitions by Angular CLI\n * during the production build. This is how Angular does tree-shaking internally.\n */\ndeclare const ngDevMode: boolean;\n\n@Injectable()\nexport class NgxsStoragePlugin implements NgxsPlugin {\n // We cast to `string[]` here as we're sure that this option has been\n // transformed by the `storageOptionsFactory` function that provided token.\n private _keys = this._options.key as string[];\n // We default to `[DEFAULT_STATE_KEY]` if the user explicitly does not provide the `key` option.\n private _usesDefaultStateKey =\n this._keys.length === 1 && this._keys[0] === DEFAULT_STATE_KEY;\n\n constructor(\n @Inject(NGXS_STORAGE_PLUGIN_OPTIONS) private _options: NgxsStoragePluginOptions,\n @Inject(STORAGE_ENGINE) private _engine: StorageEngine,\n @Inject(PLATFORM_ID) private _platformId: string\n ) {}\n\n handle(state: any, event: any, next: NgxsNextPluginFn) {\n if (isPlatformServer(this._platformId) && this._engine === null) {\n return next(state, event);\n }\n\n const matches = actionMatcher(event);\n const isInitAction = matches(InitState);\n const isUpdateAction = matches(UpdateState);\n const isInitOrUpdateAction = isInitAction || isUpdateAction;\n let hasMigration = false;\n\n if (isInitOrUpdateAction) {\n const addedStates = isUpdateAction && event.addedStates;\n\n for (const key of this._keys) {\n // We're checking what states have been added by NGXS and if any of these states should be handled by\n // the storage plugin. For instance, we only want to deserialize the `auth` state, NGXS has added\n // the `user` state, the storage plugin will be rerun and will do redundant deserialization.\n // `usesDefaultStateKey` is necessary to check since `event.addedStates` never contains `@@STATE`.\n if (!this._usesDefaultStateKey && addedStates && !addedStates.hasOwnProperty(key)) {\n continue;\n }\n\n let storedValue: any = this._engine.getItem(key!);\n\n if (storedValue !== 'undefined' && storedValue != null) {\n try {\n const newVal = this._options.deserialize!(storedValue);\n storedValue = this._options.afterDeserialize!(newVal, key);\n } catch {\n // Caretaker note: we have still left the `typeof` condition in order to avoid\n // creating a breaking change for projects that still use the View Engine.\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n console.error(\n `Error ocurred while deserializing the ${key} store value, falling back to empty object, the value obtained from the store: `,\n storedValue\n );\n }\n storedValue = {};\n }\n\n if (this._options.migrations) {\n this._options.migrations.forEach(strategy => {\n const versionMatch =\n strategy.version === getValue(storedValue, strategy.versionKey || 'version');\n const keyMatch =\n (!strategy.key && this._usesDefaultStateKey) || strategy.key === key;\n if (versionMatch && keyMatch) {\n storedValue = strategy.migrate(storedValue);\n hasMigration = true;\n }\n });\n }\n\n if (!this._usesDefaultStateKey) {\n state = setValue(state, key!, storedValue);\n } else {\n // The `UpdateState` action is dispatched whenever the feature state is added.\n // The below condition is met only when the `UpdateState` is dispatched.\n // Let's assume that we have 2 states `counter` and `@ngxs/router-plugin` state.\n // `CounterState` is provided on the root level when calling `NgxsModule.forRoot()`\n // and `@ngxs/router-plugin` is provided as a feature state.\n // The storage plugin may save the `counter` state value as `10` before.\n // The `CounterState` may implement the `ngxsOnInit` hook and call `ctx.setState(999)`.\n // The storage plugin will re-hydrate the whole state when the `RouterState` is registered,\n // and the `counter` state will again equal `10` (not `999`).\n if (storedValue && addedStates && Object.keys(addedStates).length > 0) {\n storedValue = Object.keys(addedStates).reduce((accumulator, addedState) => {\n // The `storedValue` may equal the whole state (when the default state key is used).\n // If `addedStates` contains only `router` then we want to merge the state only\n // with the `router` value.\n // Let's assume that the `storedValue` is an object:\n // `{ counter: 10, router: {...} }`\n // This will pick only the `router` object from the `storedValue` and `counter`\n // state will not be re-hydrated unnecessary.\n if (storedValue.hasOwnProperty(addedState)) {\n accumulator[addedState] = storedValue[addedState];\n }\n return accumulator;\n }, <PlainObject>{});\n }\n\n state = { ...state, ...storedValue };\n }\n }\n }\n }\n\n return next(state, event).pipe(\n tap(nextState => {\n if (!isInitOrUpdateAction || (isInitOrUpdateAction && hasMigration)) {\n for (const key of this._keys) {\n let val = nextState;\n\n if (key !== DEFAULT_STATE_KEY) {\n val = getValue(nextState, key!);\n }\n\n try {\n const newVal = this._options.beforeSerialize!(val, key);\n this._engine.setItem(key!, this._options.serialize!(newVal));\n } catch (error) {\n // Caretaker note: we have still left the `typeof` condition in order to avoid\n // creating a breaking change for projects that still use the View Engine.\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n if (\n error &&\n (error.name === 'QuotaExceededError' ||\n error.name === 'NS_ERROR_DOM_QUOTA_REACHED')\n ) {\n console.error(\n `The ${key} store value exceeds the browser storage quota: `,\n val\n );\n } else {\n console.error(\n `Error ocurred while serializing the ${key} store value, value not updated, the value obtained from the store: `,\n val\n );\n }\n }\n }\n }\n }\n })\n );\n }\n}\n","import { NgModule, ModuleWithProviders, PLATFORM_ID, InjectionToken } from '@angular/core';\nimport { NGXS_PLUGINS } from '@ngxs/store';\n\nimport {\n NgxsStoragePluginOptions,\n STORAGE_ENGINE,\n NGXS_STORAGE_PLUGIN_OPTIONS\n} from './symbols';\nimport { NgxsStoragePlugin } from './storage.plugin';\nimport { storageOptionsFactory, engineFactory } from './internals';\n\nexport const USER_OPTIONS = new InjectionToken('USER_OPTIONS');\n\n@NgModule()\nexport class NgxsStoragePluginModule {\n static forRoot(\n options?: NgxsStoragePluginOptions\n ): ModuleWithProviders<NgxsStoragePluginModule> {\n return {\n ngModule: NgxsStoragePluginModule,\n providers: [\n {\n provide: NGXS_PLUGINS,\n useClass: NgxsStoragePlugin,\n multi: true\n },\n {\n provide: USER_OPTIONS,\n useValue: options\n },\n {\n provide: NGXS_STORAGE_PLUGIN_OPTIONS,\n useFactory: storageOptionsFactory,\n deps: [USER_OPTIONS]\n },\n {\n provide: STORAGE_ENGINE,\n useFactory: engineFactory,\n deps: [NGXS_STORAGE_PLUGIN_OPTIONS, PLATFORM_ID]\n }\n ]\n };\n }\n}\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"ngxs-storage-plugin.js","sources":["ng://@ngxs/storage-plugin/src/symbols.ts","ng://@ngxs/storage-plugin/src/internals.ts","ng://@ngxs/storage-plugin/src/internals/storage-key.ts","ng://@ngxs/storage-plugin/src/internals/final-options.ts","ng://@ngxs/storage-plugin/src/storage.plugin.ts","ng://@ngxs/storage-plugin/src/storage.module.ts","ng://@ngxs/storage-plugin/src/engines.ts"],"sourcesContent":["import { InjectionToken } from '@angular/core';\n\nimport { StorageKey } from './internals/storage-key';\n\nexport const enum StorageOption {\n LocalStorage,\n SessionStorage\n}\n\nexport interface NgxsStoragePluginOptions {\n /**\n * Key for the state slice to store in the storage engine.\n */\n key?: undefined | StorageKey | StorageKey[];\n\n /**\n * The namespace is used to prefix the key for the state slice. This is\n * necessary when running micro frontend applications which use storage plugin.\n * The namespace will eliminate the conflict between keys that might overlap.\n */\n namespace?: string;\n\n /**\n * Storage engine to use. Deaults to localStorage but can provide\n *\n * sessionStorage or custom implementation of the StorageEngine interface\n */\n storage?: StorageOption;\n\n /**\n * Migration strategies.\n */\n migrations?: {\n /**\n * Version to key off.\n */\n version: number | string;\n\n /**\n * Method to migrate the previous state.\n */\n migrate: (state: any) => any;\n\n /**\n * Key to migrate.\n */\n key?: string;\n\n /**\n * Key for the version. Defaults to 'version'.\n */\n versionKey?: string;\n }[];\n\n /**\n * Serailizer for the object before its pushed into the engine.\n */\n serialize?(obj: any): string;\n\n /**\n * Deserializer for the object before its pulled out of the engine.\n */\n deserialize?(obj: any): any;\n\n /**\n * Method to alter object before serialization.\n */\n beforeSerialize?(obj: any, key: string): any;\n\n /**\n * Method to alter object after deserialization.\n */\n afterDeserialize?(obj: any, key: string): any;\n}\n\nexport const NGXS_STORAGE_PLUGIN_OPTIONS = new InjectionToken('NGXS_STORAGE_PLUGIN_OPTIONS');\n\nexport const STORAGE_ENGINE = new InjectionToken<StorageEngine>('STORAGE_ENGINE');\n\nexport interface StorageEngine {\n readonly length: number;\n getItem(key: string): any;\n setItem(key: string, val: any): void;\n removeItem(key: string): void;\n clear(): void;\n}\n","import { isPlatformServer } from '@angular/common';\n\nimport { StorageOption, StorageEngine, NgxsStoragePluginOptions } from './symbols';\n\n/**\n * The following key is used to store the entire serialized\n * state when there's no specific state provided.\n */\nexport const DEFAULT_STATE_KEY = '@@STATE';\n\nexport function storageOptionsFactory(\n options: NgxsStoragePluginOptions | undefined\n): NgxsStoragePluginOptions {\n return {\n key: [DEFAULT_STATE_KEY],\n storage: StorageOption.LocalStorage,\n serialize: JSON.stringify,\n deserialize: JSON.parse,\n beforeSerialize: obj => obj,\n afterDeserialize: obj => obj,\n ...options\n };\n}\n\nexport function engineFactory(\n options: NgxsStoragePluginOptions,\n platformId: string\n): StorageEngine | null {\n if (isPlatformServer(platformId)) {\n return null;\n }\n\n if (options.storage === StorageOption.LocalStorage) {\n return localStorage;\n } else if (options.storage === StorageOption.SessionStorage) {\n return sessionStorage;\n }\n\n return null;\n}\n\nexport function getStorageKey(key: string, options?: NgxsStoragePluginOptions): string {\n // Prepends the `namespace` option to any key if it's been provided by a user.\n // So `@@STATE` becomes `my-app:@@STATE`.\n return options && options.namespace ? `${options.namespace}:${key}` : key;\n}\n","import { InjectionToken, Type } from '@angular/core';\nimport { StateToken } from '@ngxs/store';\nimport { StateClass } from '@ngxs/store/internals';\n\nimport { StorageEngine } from '../symbols';\n\n/** This enables the user to provide a storage engine per individual key. */\nexport interface KeyWithExplicitEngine {\n key: string | StateClass | StateToken<any>;\n engine: Type<StorageEngine> | InjectionToken<StorageEngine>;\n}\n\n/** Determines whether the provided key has the following structure. */\nexport function isKeyWithExplicitEngine(key: any): key is KeyWithExplicitEngine {\n return key != null && !!key.engine;\n}\n\n/**\n * This tuples all of the possible types allowed in the `key` property.\n * This is not exposed publicly and used internally only.\n */\nexport type StorageKey = string | StateClass | StateToken<any> | KeyWithExplicitEngine;\n\n/** This symbol is used to store the metadata on state classes. */\nconst META_OPTIONS_KEY = 'NGXS_OPTIONS_META';\nexport function exctractStringKey(storageKey: StorageKey): string {\n // Extract the actual key out of the `{ key, engine }` structure.\n if (isKeyWithExplicitEngine(storageKey)) {\n storageKey = storageKey.key;\n }\n\n // Given the `storageKey` is a class, for instance, `AuthState`.\n // We should retrieve its metadata and the `name` property.\n // The `name` property might be a string (state name) or a state token.\n if (storageKey.hasOwnProperty(META_OPTIONS_KEY)) {\n storageKey = (storageKey as any)[META_OPTIONS_KEY].name;\n }\n\n return storageKey instanceof StateToken ? storageKey.getName() : <string>storageKey;\n}\n","import { InjectionToken, Injector } from '@angular/core';\n\nimport { exctractStringKey, isKeyWithExplicitEngine, StorageKey } from './storage-key';\nimport { NgxsStoragePluginOptions, StorageEngine, STORAGE_ENGINE } from '../symbols';\n\nexport interface FinalNgxsStoragePluginOptions extends NgxsStoragePluginOptions {\n keysWithEngines: {\n key: string;\n engine: StorageEngine;\n }[];\n}\n\nexport const FINAL_NGXS_STORAGE_PLUGIN_OPTIONS = new InjectionToken<\n FinalNgxsStoragePluginOptions\n>('FINAL_NGXS_STORAGE_PLUGIN_OPTIONS');\n\nexport function createFinalStoragePluginOptions(\n injector: Injector,\n options: NgxsStoragePluginOptions\n): FinalNgxsStoragePluginOptions {\n const storageKeys: StorageKey[] = Array.isArray(options.key) ? options.key : [options.key!];\n\n const keysWithEngines = storageKeys.map((storageKey: StorageKey) => {\n const key = exctractStringKey(storageKey);\n const engine = isKeyWithExplicitEngine(storageKey)\n ? injector.get(storageKey.engine)\n : injector.get(STORAGE_ENGINE);\n return { key, engine };\n });\n\n return {\n ...options,\n keysWithEngines\n };\n}\n","import { PLATFORM_ID, Inject, Injectable } from '@angular/core';\nimport { isPlatformServer } from '@angular/common';\nimport { PlainObject } from '@ngxs/store/internals';\nimport {\n NgxsPlugin,\n setValue,\n getValue,\n InitState,\n UpdateState,\n actionMatcher,\n NgxsNextPluginFn\n} from '@ngxs/store';\nimport { tap } from 'rxjs/operators';\n\nimport { DEFAULT_STATE_KEY, getStorageKey } from './internals';\nimport {\n FinalNgxsStoragePluginOptions,\n FINAL_NGXS_STORAGE_PLUGIN_OPTIONS\n} from './internals/final-options';\n\n/**\n * @description Will be provided through Terser global definitions by Angular CLI\n * during the production build. This is how Angular does tree-shaking internally.\n */\ndeclare const ngDevMode: boolean;\n\n@Injectable()\nexport class NgxsStoragePlugin implements NgxsPlugin {\n private _keysWithEngines = this._options.keysWithEngines;\n // We default to `[DEFAULT_STATE_KEY]` if the user explicitly does not provide the `key` option.\n private _usesDefaultStateKey =\n this._keysWithEngines.length === 1 && this._keysWithEngines[0].key === DEFAULT_STATE_KEY;\n\n constructor(\n @Inject(FINAL_NGXS_STORAGE_PLUGIN_OPTIONS) private _options: FinalNgxsStoragePluginOptions,\n @Inject(PLATFORM_ID) private _platformId: string\n ) {}\n\n handle(state: any, event: any, next: NgxsNextPluginFn) {\n if (isPlatformServer(this._platformId)) {\n return next(state, event);\n }\n\n const matches = actionMatcher(event);\n const isInitAction = matches(InitState);\n const isUpdateAction = matches(UpdateState);\n const isInitOrUpdateAction = isInitAction || isUpdateAction;\n let hasMigration = false;\n\n if (isInitOrUpdateAction) {\n const addedStates = isUpdateAction && event.addedStates;\n\n for (const { key, engine } of this._keysWithEngines) {\n // We're checking what states have been added by NGXS and if any of these states should be handled by\n // the storage plugin. For instance, we only want to deserialize the `auth` state, NGXS has added\n // the `user` state, the storage plugin will be rerun and will do redundant deserialization.\n // `usesDefaultStateKey` is necessary to check since `event.addedStates` never contains `@@STATE`.\n if (!this._usesDefaultStateKey && addedStates) {\n // We support providing keys that can be deeply nested via dot notation, for instance,\n // `keys: ['myState.myProperty']` is a valid key.\n // The state name should always go first. The below code checks if the `key` includes dot\n // notation and extracts the state name out of the key.\n // Given the `key` is `myState.myProperty`, the `addedStates` will only contain `myState`.\n const dotNotationIndex = key.indexOf(DOT);\n const stateName = dotNotationIndex > -1 ? key.slice(0, dotNotationIndex) : key;\n if (!addedStates.hasOwnProperty(stateName)) {\n continue;\n }\n }\n\n const storageKey = getStorageKey(key, this._options);\n let storedValue: any = engine.getItem(storageKey);\n\n if (storedValue !== 'undefined' && storedValue != null) {\n try {\n const newVal = this._options.deserialize!(storedValue);\n storedValue = this._options.afterDeserialize!(newVal, key);\n } catch {\n // Caretaker note: we have still left the `typeof` condition in order to avoid\n // creating a breaking change for projects that still use the View Engine.\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n console.error(\n `Error ocurred while deserializing the ${storageKey} store value, falling back to empty object, the value obtained from the store: `,\n storedValue\n );\n }\n storedValue = {};\n }\n\n if (this._options.migrations) {\n this._options.migrations.forEach(strategy => {\n const versionMatch =\n strategy.version === getValue(storedValue, strategy.versionKey || 'version');\n const keyMatch =\n (!strategy.key && this._usesDefaultStateKey) || strategy.key === key;\n if (versionMatch && keyMatch) {\n storedValue = strategy.migrate(storedValue);\n hasMigration = true;\n }\n });\n }\n\n if (!this._usesDefaultStateKey) {\n state = setValue(state, key, storedValue);\n } else {\n // The `UpdateState` action is dispatched whenever the feature state is added.\n // The below condition is met only when the `UpdateState` is dispatched.\n // Let's assume that we have 2 states `counter` and `@ngxs/router-plugin` state.\n // `CounterState` is provided on the root level when calling `NgxsModule.forRoot()`\n // and `@ngxs/router-plugin` is provided as a feature state.\n // The storage plugin may save the `counter` state value as `10` before.\n // The `CounterState` may implement the `ngxsOnInit` hook and call `ctx.setState(999)`.\n // The storage plugin will re-hydrate the whole state when the `RouterState` is registered,\n // and the `counter` state will again equal `10` (not `999`).\n if (storedValue && addedStates && Object.keys(addedStates).length > 0) {\n storedValue = Object.keys(addedStates).reduce((accumulator, addedState) => {\n // The `storedValue` may equal the whole state (when the default state key is used).\n // If `addedStates` contains only `router` then we want to merge the state only\n // with the `router` value.\n // Let's assume that the `storedValue` is an object:\n // `{ counter: 10, router: {...} }`\n // This will pick only the `router` object from the `storedValue` and `counter`\n // state will not be re-hydrated unnecessary.\n if (storedValue.hasOwnProperty(addedState)) {\n accumulator[addedState] = storedValue[addedState];\n }\n return accumulator;\n }, <PlainObject>{});\n }\n\n state = { ...state, ...storedValue };\n }\n }\n }\n }\n\n return next(state, event).pipe(\n tap(nextState => {\n if (!isInitOrUpdateAction || (isInitOrUpdateAction && hasMigration)) {\n for (const { key, engine } of this._keysWithEngines) {\n let storedValue = nextState;\n\n const storageKey = getStorageKey(key, this._options);\n\n if (key !== DEFAULT_STATE_KEY) {\n storedValue = getValue(nextState, key);\n }\n\n try {\n const newStoredValue = this._options.beforeSerialize!(storedValue, key);\n engine.setItem(storageKey, this._options.serialize!(newStoredValue));\n } catch (error) {\n // Caretaker note: we have still left the `typeof` condition in order to avoid\n // creating a breaking change for projects that still use the View Engine.\n if (typeof ngDevMode === 'undefined' || ngDevMode) {\n if (\n error &&\n (error.name === 'QuotaExceededError' ||\n error.name === 'NS_ERROR_DOM_QUOTA_REACHED')\n ) {\n console.error(\n `The ${storageKey} store value exceeds the browser storage quota: `,\n storedValue\n );\n } else {\n console.error(\n `Error ocurred while serializing the ${storageKey} store value, value not updated, the value obtained from the store: `,\n storedValue\n );\n }\n }\n }\n }\n }\n })\n );\n }\n}\n\nconst DOT = '.';\n","import {\n NgModule,\n ModuleWithProviders,\n PLATFORM_ID,\n InjectionToken,\n Injector\n} from '@angular/core';\nimport { NGXS_PLUGINS } from '@ngxs/store';\n\nimport {\n NgxsStoragePluginOptions,\n STORAGE_ENGINE,\n NGXS_STORAGE_PLUGIN_OPTIONS\n} from './symbols';\nimport { NgxsStoragePlugin } from './storage.plugin';\nimport { engineFactory, storageOptionsFactory } from './internals';\nimport {\n createFinalStoragePluginOptions,\n FINAL_NGXS_STORAGE_PLUGIN_OPTIONS\n} from './internals/final-options';\n\nexport const USER_OPTIONS = new InjectionToken('USER_OPTIONS');\n\n@NgModule()\nexport class NgxsStoragePluginModule {\n static forRoot(\n options?: NgxsStoragePluginOptions\n ): ModuleWithProviders<NgxsStoragePluginModule> {\n return {\n ngModule: NgxsStoragePluginModule,\n providers: [\n {\n provide: NGXS_PLUGINS,\n useClass: NgxsStoragePlugin,\n multi: true\n },\n {\n provide: USER_OPTIONS,\n useValue: options\n },\n {\n provide: NGXS_STORAGE_PLUGIN_OPTIONS,\n useFactory: storageOptionsFactory,\n deps: [USER_OPTIONS]\n },\n {\n provide: STORAGE_ENGINE,\n useFactory: engineFactory,\n deps: [NGXS_STORAGE_PLUGIN_OPTIONS, PLATFORM_ID]\n },\n {\n provide: FINAL_NGXS_STORAGE_PLUGIN_OPTIONS,\n useFactory: createFinalStoragePluginOptions,\n deps: [Injector, NGXS_STORAGE_PLUGIN_OPTIONS]\n }\n ]\n };\n }\n}\n","import { InjectionToken } from '@angular/core';\n\nimport { StorageEngine } from './symbols';\n\nexport const LOCAL_STORAGE_ENGINE = new InjectionToken<StorageEngine>('LOCAL_STORAGE_ENGINE', {\n providedIn: 'root',\n factory: () => localStorage\n});\n\nexport const SESSION_STORAGE_ENGINE = new InjectionToken<StorageEngine>(\n 'SESSION_STORAGE_ENGINE',\n {\n providedIn: 'root',\n factory: () => sessionStorage\n }\n);\n"],"names":[],"mappings":"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"}
|