@ngxs/storage-plugin 3.7.4-dev.master-a6cc926 → 3.7.4-dev.master-817c422
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 +67 -26
- package/bundles/ngxs-storage-plugin.umd.js.map +1 -1
- package/bundles/ngxs-storage-plugin.umd.min.js +2 -2
- package/bundles/ngxs-storage-plugin.umd.min.js.map +1 -1
- package/esm2015/src/storage.plugin.js +64 -23
- package/esm5/src/storage.plugin.js +68 -27
- package/fesm2015/ngxs-storage-plugin.js +63 -22
- package/fesm2015/ngxs-storage-plugin.js.map +1 -1
- package/fesm5/ngxs-storage-plugin.js +67 -26
- package/fesm5/ngxs-storage-plugin.js.map +1 -1
- package/ngxs-storage-plugin.metadata.json +1 -1
- package/package.json +1 -1
- package/src/storage.plugin.d.ts +2 -0
|
@@ -18,6 +18,11 @@ export class NgxsStoragePlugin {
|
|
|
18
18
|
this._options = _options;
|
|
19
19
|
this._engine = _engine;
|
|
20
20
|
this._platformId = _platformId;
|
|
21
|
+
// We cast to `string[]` here as we're sure that this option has been
|
|
22
|
+
// transformed by the `storageOptionsFactory` function that provided token.
|
|
23
|
+
this._keys = (/** @type {?} */ (this._options.key));
|
|
24
|
+
// We default to `[DEFAULT_STATE_KEY]` if the user explicitly does not provide the `key` option.
|
|
25
|
+
this._usesDefaultStateKey = this._keys.length === 1 && this._keys[0] === DEFAULT_STATE_KEY;
|
|
21
26
|
}
|
|
22
27
|
/**
|
|
23
28
|
* @param {?} state
|
|
@@ -29,10 +34,6 @@ export class NgxsStoragePlugin {
|
|
|
29
34
|
if (isPlatformServer(this._platformId) && this._engine === null) {
|
|
30
35
|
return next(state, event);
|
|
31
36
|
}
|
|
32
|
-
// We cast to `string[]` here as we're sure that this option has been
|
|
33
|
-
// transformed by the `storageOptionsFactory` function that provided token
|
|
34
|
-
/** @type {?} */
|
|
35
|
-
const keys = (/** @type {?} */ (this._options.key));
|
|
36
37
|
/** @type {?} */
|
|
37
38
|
const matches = actionMatcher(event);
|
|
38
39
|
/** @type {?} */
|
|
@@ -44,30 +45,31 @@ export class NgxsStoragePlugin {
|
|
|
44
45
|
/** @type {?} */
|
|
45
46
|
let hasMigration = false;
|
|
46
47
|
if (isInitOrUpdateAction) {
|
|
47
|
-
|
|
48
|
+
/** @type {?} */
|
|
49
|
+
const addedStates = isUpdateAction && event.addedStates;
|
|
50
|
+
for (const key of this._keys) {
|
|
48
51
|
// We're checking what states have been added by NGXS and if any of these states should be handled by
|
|
49
52
|
// the storage plugin. For instance, we only want to deserialize the `auth` state, NGXS has added
|
|
50
53
|
// the `user` state, the storage plugin will be rerun and will do redundant deserialization.
|
|
51
|
-
|
|
54
|
+
// `usesDefaultStateKey` is necessary to check since `event.addedStates` never contains `@@STATE`.
|
|
55
|
+
if (!this._usesDefaultStateKey && addedStates && !addedStates.hasOwnProperty(key)) {
|
|
52
56
|
continue;
|
|
53
57
|
}
|
|
54
58
|
/** @type {?} */
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
let val = this._engine.getItem((/** @type {?} */ (key)));
|
|
58
|
-
if (val !== 'undefined' && val != null) {
|
|
59
|
+
let storedValue = this._engine.getItem((/** @type {?} */ (key)));
|
|
60
|
+
if (storedValue !== 'undefined' && storedValue != null) {
|
|
59
61
|
try {
|
|
60
62
|
/** @type {?} */
|
|
61
|
-
const newVal = (/** @type {?} */ (this._options.deserialize))(
|
|
62
|
-
|
|
63
|
+
const newVal = (/** @type {?} */ (this._options.deserialize))(storedValue);
|
|
64
|
+
storedValue = (/** @type {?} */ (this._options.afterDeserialize))(newVal, key);
|
|
63
65
|
}
|
|
64
|
-
catch (
|
|
66
|
+
catch (_a) {
|
|
65
67
|
// Caretaker note: we have still left the `typeof` condition in order to avoid
|
|
66
68
|
// creating a breaking change for projects that still use the View Engine.
|
|
67
69
|
if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
|
68
|
-
console.error(`Error ocurred while deserializing the ${key} store value, falling back to empty object, the value obtained from the store: `,
|
|
70
|
+
console.error(`Error ocurred while deserializing the ${key} store value, falling back to empty object, the value obtained from the store: `, storedValue);
|
|
69
71
|
}
|
|
70
|
-
|
|
72
|
+
storedValue = {};
|
|
71
73
|
}
|
|
72
74
|
if (this._options.migrations) {
|
|
73
75
|
this._options.migrations.forEach((/**
|
|
@@ -76,20 +78,49 @@ export class NgxsStoragePlugin {
|
|
|
76
78
|
*/
|
|
77
79
|
strategy => {
|
|
78
80
|
/** @type {?} */
|
|
79
|
-
const versionMatch = strategy.version === getValue(
|
|
81
|
+
const versionMatch = strategy.version === getValue(storedValue, strategy.versionKey || 'version');
|
|
80
82
|
/** @type {?} */
|
|
81
|
-
const keyMatch = (!strategy.key &&
|
|
83
|
+
const keyMatch = (!strategy.key && this._usesDefaultStateKey) || strategy.key === key;
|
|
82
84
|
if (versionMatch && keyMatch) {
|
|
83
|
-
|
|
85
|
+
storedValue = strategy.migrate(storedValue);
|
|
84
86
|
hasMigration = true;
|
|
85
87
|
}
|
|
86
88
|
}));
|
|
87
89
|
}
|
|
88
|
-
if (!
|
|
89
|
-
state = setValue(state, (/** @type {?} */ (key)),
|
|
90
|
+
if (!this._usesDefaultStateKey) {
|
|
91
|
+
state = setValue(state, (/** @type {?} */ (key)), storedValue);
|
|
90
92
|
}
|
|
91
93
|
else {
|
|
92
|
-
|
|
94
|
+
// The `UpdateState` action is dispatched whenever the feature state is added.
|
|
95
|
+
// The below condition is met only when the `UpdateState` is dispatched.
|
|
96
|
+
// Let's assume that we have 2 states `counter` and `@ngxs/router-plugin` state.
|
|
97
|
+
// `CounterState` is provided on the root level when calling `NgxsModule.forRoot()`
|
|
98
|
+
// and `@ngxs/router-plugin` is provided as a feature state.
|
|
99
|
+
// The storage plugin may save the `counter` state value as `10` before.
|
|
100
|
+
// The `CounterState` may implement the `ngxsOnInit` hook and call `ctx.setState(999)`.
|
|
101
|
+
// The storage plugin will re-hydrate the whole state when the `RouterState` is registered,
|
|
102
|
+
// and the `counter` state will again equal `10` (not `999`).
|
|
103
|
+
if (storedValue && addedStates && Object.keys(addedStates).length > 0) {
|
|
104
|
+
storedValue = Object.keys(addedStates).reduce((/**
|
|
105
|
+
* @param {?} accumulator
|
|
106
|
+
* @param {?} addedState
|
|
107
|
+
* @return {?}
|
|
108
|
+
*/
|
|
109
|
+
(accumulator, addedState) => {
|
|
110
|
+
// The `storedValue` may equal the whole state (when the default state key is used).
|
|
111
|
+
// If `addedStates` contains only `router` then we want to merge the state only
|
|
112
|
+
// with the `router` value.
|
|
113
|
+
// Let's assume that the `storedValue` is an object:
|
|
114
|
+
// `{ counter: 10, router: {...} }`
|
|
115
|
+
// This will pick only the `router` object from the `storedValue` and `counter`
|
|
116
|
+
// state will not be re-hydrated unnecessary.
|
|
117
|
+
if (storedValue.hasOwnProperty(addedState)) {
|
|
118
|
+
accumulator[addedState] = storedValue[addedState];
|
|
119
|
+
}
|
|
120
|
+
return accumulator;
|
|
121
|
+
}), (/** @type {?} */ ({})));
|
|
122
|
+
}
|
|
123
|
+
state = Object.assign({}, state, storedValue);
|
|
93
124
|
}
|
|
94
125
|
}
|
|
95
126
|
}
|
|
@@ -100,7 +131,7 @@ export class NgxsStoragePlugin {
|
|
|
100
131
|
*/
|
|
101
132
|
nextState => {
|
|
102
133
|
if (!isInitOrUpdateAction || (isInitOrUpdateAction && hasMigration)) {
|
|
103
|
-
for (const key of
|
|
134
|
+
for (const key of this._keys) {
|
|
104
135
|
/** @type {?} */
|
|
105
136
|
let val = nextState;
|
|
106
137
|
if (key !== DEFAULT_STATE_KEY) {
|
|
@@ -140,6 +171,16 @@ NgxsStoragePlugin.ctorParameters = () => [
|
|
|
140
171
|
{ type: String, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
|
|
141
172
|
];
|
|
142
173
|
if (false) {
|
|
174
|
+
/**
|
|
175
|
+
* @type {?}
|
|
176
|
+
* @private
|
|
177
|
+
*/
|
|
178
|
+
NgxsStoragePlugin.prototype._keys;
|
|
179
|
+
/**
|
|
180
|
+
* @type {?}
|
|
181
|
+
* @private
|
|
182
|
+
*/
|
|
183
|
+
NgxsStoragePlugin.prototype._usesDefaultStateKey;
|
|
143
184
|
/**
|
|
144
185
|
* @type {?}
|
|
145
186
|
* @private
|
|
@@ -156,4 +197,4 @@ if (false) {
|
|
|
156
197
|
*/
|
|
157
198
|
NgxsStoragePlugin.prototype._platformId;
|
|
158
199
|
}
|
|
159
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storage.plugin.js","sourceRoot":"ng://@ngxs/storage-plugin/","sources":["src/storage.plugin.ts"],"names":[],"mappings":"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","sourcesContent":["import { PLATFORM_ID, Inject, Injectable } from '@angular/core';\nimport { isPlatformServer } from '@angular/common';\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  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    // We cast to `string[]` here as we're sure that this option has been\n    // transformed by the `storageOptionsFactory` function that provided token\n    const keys = this._options.key as string[];\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      for (const key of 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        if (isUpdateAction && event.addedStates && !event.addedStates.hasOwnProperty(key)) {\n          continue;\n        }\n\n        const isMaster = key === DEFAULT_STATE_KEY;\n        let val: any = this._engine.getItem(key!);\n\n        if (val !== 'undefined' && val != null) {\n          try {\n            const newVal = this._options.deserialize!(val);\n            val = this._options.afterDeserialize!(newVal, key);\n          } catch (e) {\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                val\n              );\n            }\n            val = {};\n          }\n\n          if (this._options.migrations) {\n            this._options.migrations.forEach(strategy => {\n              const versionMatch =\n                strategy.version === getValue(val, strategy.versionKey || 'version');\n              const keyMatch = (!strategy.key && isMaster) || strategy.key === key;\n              if (versionMatch && keyMatch) {\n                val = strategy.migrate(val);\n                hasMigration = true;\n              }\n            });\n          }\n\n          if (!isMaster) {\n            state = setValue(state, key!, val);\n          } else {\n            state = { ...state, ...val };\n          }\n        }\n      }\n    }\n\n    return next(state, event).pipe(\n      tap(nextState => {\n        if (!isInitOrUpdateAction || (isInitOrUpdateAction && hasMigration)) {\n          for (const key of 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"]}
|
|
200
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storage.plugin.js","sourceRoot":"ng://@ngxs/storage-plugin/","sources":["src/storage.plugin.ts"],"names":[],"mappings":"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","sourcesContent":["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"]}
|
|
@@ -14,6 +14,11 @@ var NgxsStoragePlugin = /** @class */ (function () {
|
|
|
14
14
|
this._options = _options;
|
|
15
15
|
this._engine = _engine;
|
|
16
16
|
this._platformId = _platformId;
|
|
17
|
+
// We cast to `string[]` here as we're sure that this option has been
|
|
18
|
+
// transformed by the `storageOptionsFactory` function that provided token.
|
|
19
|
+
this._keys = (/** @type {?} */ (this._options.key));
|
|
20
|
+
// We default to `[DEFAULT_STATE_KEY]` if the user explicitly does not provide the `key` option.
|
|
21
|
+
this._usesDefaultStateKey = this._keys.length === 1 && this._keys[0] === DEFAULT_STATE_KEY;
|
|
17
22
|
}
|
|
18
23
|
/**
|
|
19
24
|
* @param {?} state
|
|
@@ -33,10 +38,6 @@ var NgxsStoragePlugin = /** @class */ (function () {
|
|
|
33
38
|
if (isPlatformServer(this._platformId) && this._engine === null) {
|
|
34
39
|
return next(state, event);
|
|
35
40
|
}
|
|
36
|
-
// We cast to `string[]` here as we're sure that this option has been
|
|
37
|
-
// transformed by the `storageOptionsFactory` function that provided token
|
|
38
|
-
/** @type {?} */
|
|
39
|
-
var keys = (/** @type {?} */ (this._options.key));
|
|
40
41
|
/** @type {?} */
|
|
41
42
|
var matches = actionMatcher(event);
|
|
42
43
|
/** @type {?} */
|
|
@@ -48,30 +49,31 @@ var NgxsStoragePlugin = /** @class */ (function () {
|
|
|
48
49
|
/** @type {?} */
|
|
49
50
|
var hasMigration = false;
|
|
50
51
|
if (isInitOrUpdateAction) {
|
|
52
|
+
/** @type {?} */
|
|
53
|
+
var addedStates = isUpdateAction && event.addedStates;
|
|
51
54
|
var _loop_1 = function (key) {
|
|
52
55
|
// We're checking what states have been added by NGXS and if any of these states should be handled by
|
|
53
56
|
// the storage plugin. For instance, we only want to deserialize the `auth` state, NGXS has added
|
|
54
57
|
// the `user` state, the storage plugin will be rerun and will do redundant deserialization.
|
|
55
|
-
|
|
58
|
+
// `usesDefaultStateKey` is necessary to check since `event.addedStates` never contains `@@STATE`.
|
|
59
|
+
if (!this_1._usesDefaultStateKey && addedStates && !addedStates.hasOwnProperty(key)) {
|
|
56
60
|
return "continue";
|
|
57
61
|
}
|
|
58
62
|
/** @type {?} */
|
|
59
|
-
var
|
|
60
|
-
|
|
61
|
-
var val = this_1._engine.getItem((/** @type {?} */ (key)));
|
|
62
|
-
if (val !== 'undefined' && val != null) {
|
|
63
|
+
var storedValue = this_1._engine.getItem((/** @type {?} */ (key)));
|
|
64
|
+
if (storedValue !== 'undefined' && storedValue != null) {
|
|
63
65
|
try {
|
|
64
66
|
/** @type {?} */
|
|
65
|
-
var newVal = (/** @type {?} */ (this_1._options.deserialize))(
|
|
66
|
-
|
|
67
|
+
var newVal = (/** @type {?} */ (this_1._options.deserialize))(storedValue);
|
|
68
|
+
storedValue = (/** @type {?} */ (this_1._options.afterDeserialize))(newVal, key);
|
|
67
69
|
}
|
|
68
|
-
catch (
|
|
70
|
+
catch (_a) {
|
|
69
71
|
// Caretaker note: we have still left the `typeof` condition in order to avoid
|
|
70
72
|
// creating a breaking change for projects that still use the View Engine.
|
|
71
73
|
if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
|
72
|
-
console.error("Error ocurred while deserializing the " + key + " store value, falling back to empty object, the value obtained from the store: ",
|
|
74
|
+
console.error("Error ocurred while deserializing the " + key + " store value, falling back to empty object, the value obtained from the store: ", storedValue);
|
|
73
75
|
}
|
|
74
|
-
|
|
76
|
+
storedValue = {};
|
|
75
77
|
}
|
|
76
78
|
if (this_1._options.migrations) {
|
|
77
79
|
this_1._options.migrations.forEach((/**
|
|
@@ -80,34 +82,63 @@ var NgxsStoragePlugin = /** @class */ (function () {
|
|
|
80
82
|
*/
|
|
81
83
|
function (strategy) {
|
|
82
84
|
/** @type {?} */
|
|
83
|
-
var versionMatch = strategy.version === getValue(
|
|
85
|
+
var versionMatch = strategy.version === getValue(storedValue, strategy.versionKey || 'version');
|
|
84
86
|
/** @type {?} */
|
|
85
|
-
var keyMatch = (!strategy.key &&
|
|
87
|
+
var keyMatch = (!strategy.key && _this._usesDefaultStateKey) || strategy.key === key;
|
|
86
88
|
if (versionMatch && keyMatch) {
|
|
87
|
-
|
|
89
|
+
storedValue = strategy.migrate(storedValue);
|
|
88
90
|
hasMigration = true;
|
|
89
91
|
}
|
|
90
92
|
}));
|
|
91
93
|
}
|
|
92
|
-
if (!
|
|
93
|
-
state = setValue(state, (/** @type {?} */ (key)),
|
|
94
|
+
if (!this_1._usesDefaultStateKey) {
|
|
95
|
+
state = setValue(state, (/** @type {?} */ (key)), storedValue);
|
|
94
96
|
}
|
|
95
97
|
else {
|
|
96
|
-
|
|
98
|
+
// The `UpdateState` action is dispatched whenever the feature state is added.
|
|
99
|
+
// The below condition is met only when the `UpdateState` is dispatched.
|
|
100
|
+
// Let's assume that we have 2 states `counter` and `@ngxs/router-plugin` state.
|
|
101
|
+
// `CounterState` is provided on the root level when calling `NgxsModule.forRoot()`
|
|
102
|
+
// and `@ngxs/router-plugin` is provided as a feature state.
|
|
103
|
+
// The storage plugin may save the `counter` state value as `10` before.
|
|
104
|
+
// The `CounterState` may implement the `ngxsOnInit` hook and call `ctx.setState(999)`.
|
|
105
|
+
// The storage plugin will re-hydrate the whole state when the `RouterState` is registered,
|
|
106
|
+
// and the `counter` state will again equal `10` (not `999`).
|
|
107
|
+
if (storedValue && addedStates && Object.keys(addedStates).length > 0) {
|
|
108
|
+
storedValue = Object.keys(addedStates).reduce((/**
|
|
109
|
+
* @param {?} accumulator
|
|
110
|
+
* @param {?} addedState
|
|
111
|
+
* @return {?}
|
|
112
|
+
*/
|
|
113
|
+
function (accumulator, addedState) {
|
|
114
|
+
// The `storedValue` may equal the whole state (when the default state key is used).
|
|
115
|
+
// If `addedStates` contains only `router` then we want to merge the state only
|
|
116
|
+
// with the `router` value.
|
|
117
|
+
// Let's assume that the `storedValue` is an object:
|
|
118
|
+
// `{ counter: 10, router: {...} }`
|
|
119
|
+
// This will pick only the `router` object from the `storedValue` and `counter`
|
|
120
|
+
// state will not be re-hydrated unnecessary.
|
|
121
|
+
if (storedValue.hasOwnProperty(addedState)) {
|
|
122
|
+
accumulator[addedState] = storedValue[addedState];
|
|
123
|
+
}
|
|
124
|
+
return accumulator;
|
|
125
|
+
}), (/** @type {?} */ ({})));
|
|
126
|
+
}
|
|
127
|
+
state = tslib_1.__assign({}, state, storedValue);
|
|
97
128
|
}
|
|
98
129
|
}
|
|
99
130
|
};
|
|
100
131
|
var this_1 = this;
|
|
101
132
|
try {
|
|
102
|
-
for (var
|
|
103
|
-
var key =
|
|
133
|
+
for (var _b = tslib_1.__values(this._keys), _c = _b.next(); !_c.done; _c = _b.next()) {
|
|
134
|
+
var key = _c.value;
|
|
104
135
|
_loop_1(key);
|
|
105
136
|
}
|
|
106
137
|
}
|
|
107
138
|
catch (e_1_1) { e_1 = { error: e_1_1 }; }
|
|
108
139
|
finally {
|
|
109
140
|
try {
|
|
110
|
-
if (
|
|
141
|
+
if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
|
111
142
|
}
|
|
112
143
|
finally { if (e_1) throw e_1.error; }
|
|
113
144
|
}
|
|
@@ -120,8 +151,8 @@ var NgxsStoragePlugin = /** @class */ (function () {
|
|
|
120
151
|
var e_2, _a;
|
|
121
152
|
if (!isInitOrUpdateAction || (isInitOrUpdateAction && hasMigration)) {
|
|
122
153
|
try {
|
|
123
|
-
for (var
|
|
124
|
-
var key =
|
|
154
|
+
for (var _b = tslib_1.__values(_this._keys), _c = _b.next(); !_c.done; _c = _b.next()) {
|
|
155
|
+
var key = _c.value;
|
|
125
156
|
/** @type {?} */
|
|
126
157
|
var val = nextState;
|
|
127
158
|
if (key !== DEFAULT_STATE_KEY) {
|
|
@@ -151,7 +182,7 @@ var NgxsStoragePlugin = /** @class */ (function () {
|
|
|
151
182
|
catch (e_2_1) { e_2 = { error: e_2_1 }; }
|
|
152
183
|
finally {
|
|
153
184
|
try {
|
|
154
|
-
if (
|
|
185
|
+
if (_c && !_c.done && (_a = _b.return)) _a.call(_b);
|
|
155
186
|
}
|
|
156
187
|
finally { if (e_2) throw e_2.error; }
|
|
157
188
|
}
|
|
@@ -171,6 +202,16 @@ var NgxsStoragePlugin = /** @class */ (function () {
|
|
|
171
202
|
}());
|
|
172
203
|
export { NgxsStoragePlugin };
|
|
173
204
|
if (false) {
|
|
205
|
+
/**
|
|
206
|
+
* @type {?}
|
|
207
|
+
* @private
|
|
208
|
+
*/
|
|
209
|
+
NgxsStoragePlugin.prototype._keys;
|
|
210
|
+
/**
|
|
211
|
+
* @type {?}
|
|
212
|
+
* @private
|
|
213
|
+
*/
|
|
214
|
+
NgxsStoragePlugin.prototype._usesDefaultStateKey;
|
|
174
215
|
/**
|
|
175
216
|
* @type {?}
|
|
176
217
|
* @private
|
|
@@ -187,4 +228,4 @@ if (false) {
|
|
|
187
228
|
*/
|
|
188
229
|
NgxsStoragePlugin.prototype._platformId;
|
|
189
230
|
}
|
|
190
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storage.plugin.js","sourceRoot":"ng://@ngxs/storage-plugin/","sources":["src/storage.plugin.ts"],"names":[],"mappings":";;;;;AAAA,OAAO,EAAE,WAAW,EAAE,MAAM,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;AAChE,OAAO,EAAE,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AACnD,OAAO,EAEL,QAAQ,EACR,QAAQ,EACR,SAAS,EACT,WAAW,EACX,aAAa,EAEd,MAAM,aAAa,CAAC;AACrB,OAAO,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAErC,OAAO,EAGL,cAAc,EACd,2BAA2B,EAC5B,MAAM,WAAW,CAAC;AACnB,OAAO,EAAE,iBAAiB,EAAE,MAAM,aAAa,CAAC;AAQhD;IAEE,2BAC+C,QAAkC,EAC/C,OAAsB,EACzB,WAAmB;QAFH,aAAQ,GAAR,QAAQ,CAA0B;QAC/C,YAAO,GAAP,OAAO,CAAe;QACzB,gBAAW,GAAX,WAAW,CAAQ;IAC/C,CAAC;;;;;;;IAEJ,kCAAM;;;;;;IAAN,UAAO,KAAU,EAAE,KAAU,EAAE,IAAsB;QAArD,iBAqGC;;QApGC,IAAI,gBAAgB,CAAC,IAAI,CAAC,WAAW,CAAC,IAAI,IAAI,CAAC,OAAO,KAAK,IAAI,EAAE;YAC/D,OAAO,IAAI,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC;SAC3B;;;;YAIK,IAAI,GAAG,mBAAA,IAAI,CAAC,QAAQ,CAAC,GAAG,EAAY;;YACpC,OAAO,GAAG,aAAa,CAAC,KAAK,CAAC;;YAC9B,YAAY,GAAG,OAAO,CAAC,SAAS,CAAC;;YACjC,cAAc,GAAG,OAAO,CAAC,WAAW,CAAC;;YACrC,oBAAoB,GAAG,YAAY,IAAI,cAAc;;YACvD,YAAY,GAAG,KAAK;QAExB,IAAI,oBAAoB,EAAE;oCACb,GAAG;gBACZ,qGAAqG;gBACrG,iGAAiG;gBACjG,4FAA4F;gBAC5F,IAAI,cAAc,IAAI,KAAK,CAAC,WAAW,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,cAAc,CAAC,GAAG,CAAC,EAAE;;iBAElF;;oBAEK,QAAQ,GAAG,GAAG,KAAK,iBAAiB;;oBACtC,GAAG,GAAQ,OAAK,OAAO,CAAC,OAAO,CAAC,mBAAA,GAAG,EAAC,CAAC;gBAEzC,IAAI,GAAG,KAAK,WAAW,IAAI,GAAG,IAAI,IAAI,EAAE;oBACtC,IAAI;;4BACI,MAAM,GAAG,mBAAA,OAAK,QAAQ,CAAC,WAAW,EAAC,CAAC,GAAG,CAAC;wBAC9C,GAAG,GAAG,mBAAA,OAAK,QAAQ,CAAC,gBAAgB,EAAC,CAAC,MAAM,EAAE,GAAG,CAAC,CAAC;qBACpD;oBAAC,OAAO,CAAC,EAAE;wBACV,8EAA8E;wBAC9E,0EAA0E;wBAC1E,IAAI,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,EAAE;4BACjD,OAAO,CAAC,KAAK,CACX,2CAAyC,GAAG,oFAAiF,EAC7H,GAAG,CACJ,CAAC;yBACH;wBACD,GAAG,GAAG,EAAE,CAAC;qBACV;oBAED,IAAI,OAAK,QAAQ,CAAC,UAAU,EAAE;wBAC5B,OAAK,QAAQ,CAAC,UAAU,CAAC,OAAO;;;;wBAAC,UAAA,QAAQ;;gCACjC,YAAY,GAChB,QAAQ,CAAC,OAAO,KAAK,QAAQ,CAAC,GAAG,EAAE,QAAQ,CAAC,UAAU,IAAI,SAAS,CAAC;;gCAChE,QAAQ,GAAG,CAAC,CAAC,QAAQ,CAAC,GAAG,IAAI,QAAQ,CAAC,IAAI,QAAQ,CAAC,GAAG,KAAK,GAAG;4BACpE,IAAI,YAAY,IAAI,QAAQ,EAAE;gCAC5B,GAAG,GAAG,QAAQ,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;gCAC5B,YAAY,GAAG,IAAI,CAAC;6BACrB;wBACH,CAAC,EAAC,CAAC;qBACJ;oBAED,IAAI,CAAC,QAAQ,EAAE;wBACb,KAAK,GAAG,QAAQ,CAAC,KAAK,EAAE,mBAAA,GAAG,EAAC,EAAE,GAAG,CAAC,CAAC;qBACpC;yBAAM;wBACL,KAAK,wBAAQ,KAAK,EAAK,GAAG,CAAE,CAAC;qBAC9B;iBACF;YACH,CAAC;;;gBA7CD,KAAkB,IAAA,SAAA,iBAAA,IAAI,CAAA,0BAAA;oBAAjB,IAAM,GAAG,iBAAA;4BAAH,GAAG;iBA6Cb;;;;;;;;;SACF;QAED,OAAO,IAAI,CAAC,KAAK,EAAE,KAAK,CAAC,CAAC,IAAI,CAC5B,GAAG;;;;QAAC,UAAA,SAAS;;YACX,IAAI,CAAC,oBAAoB,IAAI,CAAC,oBAAoB,IAAI,YAAY,CAAC,EAAE;;oBACnE,KAAkB,IAAA,SAAA,iBAAA,IAAI,CAAA,0BAAA,4CAAE;wBAAnB,IAAM,GAAG,iBAAA;;4BACR,GAAG,GAAG,SAAS;wBAEnB,IAAI,GAAG,KAAK,iBAAiB,EAAE;4BAC7B,GAAG,GAAG,QAAQ,CAAC,SAAS,EAAE,mBAAA,GAAG,EAAC,CAAC,CAAC;yBACjC;wBAED,IAAI;;gCACI,MAAM,GAAG,mBAAA,KAAI,CAAC,QAAQ,CAAC,eAAe,EAAC,CAAC,GAAG,EAAE,GAAG,CAAC;4BACvD,KAAI,CAAC,OAAO,CAAC,OAAO,CAAC,mBAAA,GAAG,EAAC,EAAE,mBAAA,KAAI,CAAC,QAAQ,CAAC,SAAS,EAAC,CAAC,MAAM,CAAC,CAAC,CAAC;yBAC9D;wBAAC,OAAO,KAAK,EAAE;4BACd,8EAA8E;4BAC9E,0EAA0E;4BAC1E,IAAI,OAAO,SAAS,KAAK,WAAW,IAAI,SAAS,EAAE;gCACjD,IACE,KAAK;oCACL,CAAC,KAAK,CAAC,IAAI,KAAK,oBAAoB;wCAClC,KAAK,CAAC,IAAI,KAAK,4BAA4B,CAAC,EAC9C;oCACA,OAAO,CAAC,KAAK,CACX,SAAO,GAAG,qDAAkD,EAC5D,GAAG,CACJ,CAAC;iCACH;qCAAM;oCACL,OAAO,CAAC,KAAK,CACX,yCAAuC,GAAG,yEAAsE,EAChH,GAAG,CACJ,CAAC;iCACH;6BACF;yBACF;qBACF;;;;;;;;;aACF;QACH,CAAC,EAAC,CACH,CAAC;IACJ,CAAC;;gBA7GF,UAAU;;;;gDAGN,MAAM,SAAC,2BAA2B;gDAClC,MAAM,SAAC,cAAc;6CACrB,MAAM,SAAC,WAAW;;IAyGvB,wBAAC;CAAA,AA9GD,IA8GC;SA7GY,iBAAiB;;;;;;IAE1B,qCAA+E;;;;;IAC/E,oCAAsD;;;;;IACtD,wCAAgD","sourcesContent":["import { PLATFORM_ID, Inject, Injectable } from '@angular/core';\nimport { isPlatformServer } from '@angular/common';\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  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    // We cast to `string[]` here as we're sure that this option has been\n    // transformed by the `storageOptionsFactory` function that provided token\n    const keys = this._options.key as string[];\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      for (const key of 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        if (isUpdateAction && event.addedStates && !event.addedStates.hasOwnProperty(key)) {\n          continue;\n        }\n\n        const isMaster = key === DEFAULT_STATE_KEY;\n        let val: any = this._engine.getItem(key!);\n\n        if (val !== 'undefined' && val != null) {\n          try {\n            const newVal = this._options.deserialize!(val);\n            val = this._options.afterDeserialize!(newVal, key);\n          } catch (e) {\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                val\n              );\n            }\n            val = {};\n          }\n\n          if (this._options.migrations) {\n            this._options.migrations.forEach(strategy => {\n              const versionMatch =\n                strategy.version === getValue(val, strategy.versionKey || 'version');\n              const keyMatch = (!strategy.key && isMaster) || strategy.key === key;\n              if (versionMatch && keyMatch) {\n                val = strategy.migrate(val);\n                hasMigration = true;\n              }\n            });\n          }\n\n          if (!isMaster) {\n            state = setValue(state, key!, val);\n          } else {\n            state = { ...state, ...val };\n          }\n        }\n      }\n    }\n\n    return next(state, event).pipe(\n      tap(nextState => {\n        if (!isInitOrUpdateAction || (isInitOrUpdateAction && hasMigration)) {\n          for (const key of 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"]}
|
|
231
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"storage.plugin.js","sourceRoot":"ng://@ngxs/storage-plugin/","sources":["src/storage.plugin.ts"],"names":[],"mappings":"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","sourcesContent":["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"]}
|
|
@@ -182,6 +182,11 @@ class NgxsStoragePlugin {
|
|
|
182
182
|
this._options = _options;
|
|
183
183
|
this._engine = _engine;
|
|
184
184
|
this._platformId = _platformId;
|
|
185
|
+
// We cast to `string[]` here as we're sure that this option has been
|
|
186
|
+
// transformed by the `storageOptionsFactory` function that provided token.
|
|
187
|
+
this._keys = (/** @type {?} */ (this._options.key));
|
|
188
|
+
// We default to `[DEFAULT_STATE_KEY]` if the user explicitly does not provide the `key` option.
|
|
189
|
+
this._usesDefaultStateKey = this._keys.length === 1 && this._keys[0] === DEFAULT_STATE_KEY;
|
|
185
190
|
}
|
|
186
191
|
/**
|
|
187
192
|
* @param {?} state
|
|
@@ -193,10 +198,6 @@ class NgxsStoragePlugin {
|
|
|
193
198
|
if (isPlatformServer(this._platformId) && this._engine === null) {
|
|
194
199
|
return next(state, event);
|
|
195
200
|
}
|
|
196
|
-
// We cast to `string[]` here as we're sure that this option has been
|
|
197
|
-
// transformed by the `storageOptionsFactory` function that provided token
|
|
198
|
-
/** @type {?} */
|
|
199
|
-
const keys = (/** @type {?} */ (this._options.key));
|
|
200
201
|
/** @type {?} */
|
|
201
202
|
const matches = actionMatcher(event);
|
|
202
203
|
/** @type {?} */
|
|
@@ -208,30 +209,31 @@ class NgxsStoragePlugin {
|
|
|
208
209
|
/** @type {?} */
|
|
209
210
|
let hasMigration = false;
|
|
210
211
|
if (isInitOrUpdateAction) {
|
|
211
|
-
|
|
212
|
+
/** @type {?} */
|
|
213
|
+
const addedStates = isUpdateAction && event.addedStates;
|
|
214
|
+
for (const key of this._keys) {
|
|
212
215
|
// We're checking what states have been added by NGXS and if any of these states should be handled by
|
|
213
216
|
// the storage plugin. For instance, we only want to deserialize the `auth` state, NGXS has added
|
|
214
217
|
// the `user` state, the storage plugin will be rerun and will do redundant deserialization.
|
|
215
|
-
|
|
218
|
+
// `usesDefaultStateKey` is necessary to check since `event.addedStates` never contains `@@STATE`.
|
|
219
|
+
if (!this._usesDefaultStateKey && addedStates && !addedStates.hasOwnProperty(key)) {
|
|
216
220
|
continue;
|
|
217
221
|
}
|
|
218
222
|
/** @type {?} */
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
let val = this._engine.getItem((/** @type {?} */ (key)));
|
|
222
|
-
if (val !== 'undefined' && val != null) {
|
|
223
|
+
let storedValue = this._engine.getItem((/** @type {?} */ (key)));
|
|
224
|
+
if (storedValue !== 'undefined' && storedValue != null) {
|
|
223
225
|
try {
|
|
224
226
|
/** @type {?} */
|
|
225
|
-
const newVal = (/** @type {?} */ (this._options.deserialize))(
|
|
226
|
-
|
|
227
|
+
const newVal = (/** @type {?} */ (this._options.deserialize))(storedValue);
|
|
228
|
+
storedValue = (/** @type {?} */ (this._options.afterDeserialize))(newVal, key);
|
|
227
229
|
}
|
|
228
|
-
catch (
|
|
230
|
+
catch (_a) {
|
|
229
231
|
// Caretaker note: we have still left the `typeof` condition in order to avoid
|
|
230
232
|
// creating a breaking change for projects that still use the View Engine.
|
|
231
233
|
if (typeof ngDevMode === 'undefined' || ngDevMode) {
|
|
232
|
-
console.error(`Error ocurred while deserializing the ${key} store value, falling back to empty object, the value obtained from the store: `,
|
|
234
|
+
console.error(`Error ocurred while deserializing the ${key} store value, falling back to empty object, the value obtained from the store: `, storedValue);
|
|
233
235
|
}
|
|
234
|
-
|
|
236
|
+
storedValue = {};
|
|
235
237
|
}
|
|
236
238
|
if (this._options.migrations) {
|
|
237
239
|
this._options.migrations.forEach((/**
|
|
@@ -240,20 +242,49 @@ class NgxsStoragePlugin {
|
|
|
240
242
|
*/
|
|
241
243
|
strategy => {
|
|
242
244
|
/** @type {?} */
|
|
243
|
-
const versionMatch = strategy.version === getValue(
|
|
245
|
+
const versionMatch = strategy.version === getValue(storedValue, strategy.versionKey || 'version');
|
|
244
246
|
/** @type {?} */
|
|
245
|
-
const keyMatch = (!strategy.key &&
|
|
247
|
+
const keyMatch = (!strategy.key && this._usesDefaultStateKey) || strategy.key === key;
|
|
246
248
|
if (versionMatch && keyMatch) {
|
|
247
|
-
|
|
249
|
+
storedValue = strategy.migrate(storedValue);
|
|
248
250
|
hasMigration = true;
|
|
249
251
|
}
|
|
250
252
|
}));
|
|
251
253
|
}
|
|
252
|
-
if (!
|
|
253
|
-
state = setValue(state, (/** @type {?} */ (key)),
|
|
254
|
+
if (!this._usesDefaultStateKey) {
|
|
255
|
+
state = setValue(state, (/** @type {?} */ (key)), storedValue);
|
|
254
256
|
}
|
|
255
257
|
else {
|
|
256
|
-
|
|
258
|
+
// The `UpdateState` action is dispatched whenever the feature state is added.
|
|
259
|
+
// The below condition is met only when the `UpdateState` is dispatched.
|
|
260
|
+
// Let's assume that we have 2 states `counter` and `@ngxs/router-plugin` state.
|
|
261
|
+
// `CounterState` is provided on the root level when calling `NgxsModule.forRoot()`
|
|
262
|
+
// and `@ngxs/router-plugin` is provided as a feature state.
|
|
263
|
+
// The storage plugin may save the `counter` state value as `10` before.
|
|
264
|
+
// The `CounterState` may implement the `ngxsOnInit` hook and call `ctx.setState(999)`.
|
|
265
|
+
// The storage plugin will re-hydrate the whole state when the `RouterState` is registered,
|
|
266
|
+
// and the `counter` state will again equal `10` (not `999`).
|
|
267
|
+
if (storedValue && addedStates && Object.keys(addedStates).length > 0) {
|
|
268
|
+
storedValue = Object.keys(addedStates).reduce((/**
|
|
269
|
+
* @param {?} accumulator
|
|
270
|
+
* @param {?} addedState
|
|
271
|
+
* @return {?}
|
|
272
|
+
*/
|
|
273
|
+
(accumulator, addedState) => {
|
|
274
|
+
// The `storedValue` may equal the whole state (when the default state key is used).
|
|
275
|
+
// If `addedStates` contains only `router` then we want to merge the state only
|
|
276
|
+
// with the `router` value.
|
|
277
|
+
// Let's assume that the `storedValue` is an object:
|
|
278
|
+
// `{ counter: 10, router: {...} }`
|
|
279
|
+
// This will pick only the `router` object from the `storedValue` and `counter`
|
|
280
|
+
// state will not be re-hydrated unnecessary.
|
|
281
|
+
if (storedValue.hasOwnProperty(addedState)) {
|
|
282
|
+
accumulator[addedState] = storedValue[addedState];
|
|
283
|
+
}
|
|
284
|
+
return accumulator;
|
|
285
|
+
}), (/** @type {?} */ ({})));
|
|
286
|
+
}
|
|
287
|
+
state = Object.assign({}, state, storedValue);
|
|
257
288
|
}
|
|
258
289
|
}
|
|
259
290
|
}
|
|
@@ -264,7 +295,7 @@ class NgxsStoragePlugin {
|
|
|
264
295
|
*/
|
|
265
296
|
nextState => {
|
|
266
297
|
if (!isInitOrUpdateAction || (isInitOrUpdateAction && hasMigration)) {
|
|
267
|
-
for (const key of
|
|
298
|
+
for (const key of this._keys) {
|
|
268
299
|
/** @type {?} */
|
|
269
300
|
let val = nextState;
|
|
270
301
|
if (key !== DEFAULT_STATE_KEY) {
|
|
@@ -304,6 +335,16 @@ NgxsStoragePlugin.ctorParameters = () => [
|
|
|
304
335
|
{ type: String, decorators: [{ type: Inject, args: [PLATFORM_ID,] }] }
|
|
305
336
|
];
|
|
306
337
|
if (false) {
|
|
338
|
+
/**
|
|
339
|
+
* @type {?}
|
|
340
|
+
* @private
|
|
341
|
+
*/
|
|
342
|
+
NgxsStoragePlugin.prototype._keys;
|
|
343
|
+
/**
|
|
344
|
+
* @type {?}
|
|
345
|
+
* @private
|
|
346
|
+
*/
|
|
347
|
+
NgxsStoragePlugin.prototype._usesDefaultStateKey;
|
|
307
348
|
/**
|
|
308
349
|
* @type {?}
|
|
309
350
|
* @private
|