The library for persisting state in ngrx. Highly inspired by @ngneat/elf-persist-state. Supports local storage, session storage and async storages with localForage.
angular
19+@ngrx/store
19+
npm i @ngrx-addons/persist-state
or
yarn add @ngrx-addons/persist-state
The module gives ability to persist some of the app’s states, by saving it to localStorage/sessionStorage
or anything that implements the StorageEngine API
, and restore it after a refresh. It supports both root and feature states. The only thing you need to do is to add PersistStateModule.forRoot
/providePersistStore
to your AppModule
and PersistStateModule.forFeature
/providePersistState
to your feature module.
import { NgModule } from '@angular/core';
import { StoreModule } from '@ngrx/store';
import { BeforeAppInit } from '@ngrx-addons/common';
import { PersistStateModule, localStorageStrategy } from '@ngrx-addons/persist-store';
const counterReducer = ...;
const reducers = {
counter: counterReducer,
} as const;
@NgModule({
imports: [
StoreModule.forRoot(reducers),
// Define after EffectsModule.forRoot() if you want to listen on `rehydrate` action
// type provided for hints on states
PersistStateModule.forRoot<typeof reducers>({
states: [
{
key: 'counter',
// the package exposes localStorageStrategy and
// sessionStorageStrategy, optionally you can
// provide your own implementation or even
// use localForage for indexed db.
storage: localStorageStrategy
// optional options (default values)
runGuard: () => typeof window !== 'undefined',
source: (state) => state,
storageKey: `${storageKeyPrefix}-${key}@store`,
migrations: [],
skip: 1
},
// next states to persist, same reducer key can be
// specified multiple times to save parts of the state
// to different storages
],
// optional root options (for all, also feature states)
storageKeyPrefix: 'some-prefix',
// optional rehydration strategy
strategy: BeforeAppInit, // or AfterAppInit
}),
],
})
export class AppModule {}
or in case of using standalone API:
import { NgModule } from '@angular/core';
import { provideStore } from '@ngrx/store';
import { BeforeAppInit } from '@ngrx-addons/common';
import { providePersistStore, localStorageStrategy } from '@ngrx-addons/persist-store';
const counterReducer = ...;
const reducers = {
counter: counterReducer,
} as const;
@NgModule({
providers: [
provideStore(reducers),
// Define after EffectsModule.forRoot() if you want to listen on `rehydrate` action
// type provided for hints on states
providePersistStore<typeof reducers>({
states: [
{
key: 'counter',
// the package exposes localStorageStrategy and
// sessionStorageStrategy, optionally you can
// provide your own implementation or even
// use localForage for indexed db.
storage: localStorageStrategy
// optional options (default values)
runGuard: () => typeof window !== 'undefined',
source: (state) => state,
storageKey: `${storageKeyPrefix}-${key}@store`,
migrations: [],
skip: 1
},
// next states to persist, same reducer key can be
// specified multiple times to save parts of the state
// to different storages
],
// optional root options (for all, also feature states)
storageKeyPrefix: 'some-prefix',
// optional rehydration strategy
strategy: BeforeAppInit, // or AfterAppInit
}),
],
})
export class AppModule {}
The forRoot
/providePersistStore
method accepts an object with the following properties:
states
- array of states configs (defined below, required)storageKeyPrefix
- prefix for all storage keys (optional)strategy
- defines if rehydrate actions should be fired before or after app initialization (optional, default:BeforeAppInit
)
Each state can be described by multiple state configs with the following properties:
key
- the reducer key in app state (required)storage
: an object or function resolving to an object with async setItem, getItem and removeItem methods for storing the state. The package exposeslocalStorageStrategy
orsessionStorageStrategy
, excepts also alocalForage
instance (required).source
: a method that receives the observable of a state and return what to save from it (by default - the entire state).storageKey
: the name under which the store state is saved (by default - the prefix plus store name plus a@store
suffix).runGuard
- returns whether the actual implementation should be run. The default is() => typeof window !== 'undefined'
skip
- the number of state changes skipped before the state is persisted. Used to skip the initial state change. The default is1
.migrations
- the array of migrations to run on the state beforerehydrated
event is fired. The default is[]
.version
- the version of the state to migrate from.versionKey
- the key in the state that contains the version. The default isversion
.migrate
- the function that receives the state and returns the migrated state.
Remember to add features only once, in any case only the last registration will be used.
import { NgModule } from '@angular/core';
import { StoreModule } from '@ngrx/store';
import { PersistStateModule, localStorageStrategy } from '@ngrx-addons/persist-store';
interface CounterState {
count: number;
}
const counterReducer = ...;
@NgModule({
imports: [
StoreModule.forRoot(),
// forRoot should be always called, similar to ngrx StoreModule and it's forFeature implementation.
PersistStateModule.forRoot(),
],
})
export class AppModule {}
@NgModule({
imports: [
StoreModule.forFeature('counter', reducer),
// type provided for hints on states
PersistStateModule.forFeature<CounterState>({
key: 'counter',
states: [
{
// The same options as for root states, except the key
storage: localStorageStrategy
},
],
}),
]
})
export class CounterModule {}
or in case of using standalone API:
import { NgModule } from '@angular/core';
import { provideStore, provideState } from '@ngrx/store';
import { providePersistStore, providePersistState, localStorageStrategy } from '@ngrx-addons/persist-store';
interface CounterState {
count: number;
}
const counterReducer = ...;
@NgModule({
providers: [
provideStore(),
// forRoot should be always called, similar to ngrx StoreModule and it's forFeature implementation.
providePersistStore(),
],
})
export class AppModule {}
@NgModule({
providers: [
provideState('counter', reducer),
// type provided for hints on states
providePersistState<CounterState>({
key: 'counter',
states: [
{
// The same options as for root states, except the key
storage: localStorageStrategy
},
],
}),
]
})
export class CounterModule {}
The forFeature
/providePersistState
method accepts an object with the following properties:
key
- the feature key (required)states
- array of states configs as inforRoot
, exceptkey
property (required)
Once the state is rehydrated, the action (rehydrated
, type: @ngrx-addons/persist-state/rehydrate
) with the proper features
is dispatched (multiple times). You can use it to react in effects
or meta-reducers
.
The excludeKeys()
/includeKeys()
operator can be used to exclude keys from the state:
import { NgModule } from '@angular/core';
import { StoreModule } from '@ngrx/store';
import { excludeKeys, includeKeys } from '@ngrx-addons/common';
import { PersistStateModule, localStorageStrategy } from '@ngrx-addons/persist-store';
const counterReducer = ...;
const reducers = {
counter: counterReducer,
} as const;
@NgModule({
imports: [
StoreModule.forRoot(reducers),
PersistStateModule.forRoot<typeof reducers>({
states: [
{
key: 'counter',
storage: localStorageStrategy,
source: (state) => state.pipe(excludeKeys(['a', 'b'])),
// source: (state) => state.pipe(includeKeys(['a', 'b'])),
},
],
}),
],
})
export class AppModule {}
By default, the module will update the storage upon each state changes (distinctUntilChanged
with object equality check is applied). Some applications perform multiple updates in a second, and update the storage on each change can be costly.
For such cases, it's recommended to use the debounceTime
operator. For example:
import { NgModule } from '@angular/core';
import { StoreModule } from '@ngrx/store';
import { debounceTime } from 'rxjs/operators';
import { PersistStateModule, localStorageStrategy } from '@ngrx-addons/persist-store';
const counterReducer = ...;
const reducers = {
counter: counterReducer,
} as const;
@NgModule({
imports: [
StoreModule.forRoot(reducers),
PersistStateModule.forRoot<typeof reducers>({
states: [
{
key: 'counter',
storage: localStorageStrategy,
source: (state) => state.pipe(debounceTime(1000)),
},
],
}),
],
})
export class AppModule {}
Check apps