From 32f6f3a1b54a2d2b3812d2264b89e53f35bded38 Mon Sep 17 00:00:00 2001 From: John Date: Sat, 27 Jul 2019 13:43:48 +0200 Subject: [PATCH] fix(store): Not change store reference on every UPDATE_ACTION --- spec/index_spec.ts | 53 ++++++++++++++++++++++++++++++++++++++-------- src/index.ts | 22 ++++++++++--------- 2 files changed, 56 insertions(+), 19 deletions(-) diff --git a/spec/index_spec.ts b/spec/index_spec.ts index d397ac7..4bf920b 100644 --- a/spec/index_spec.ts +++ b/spec/index_spec.ts @@ -5,6 +5,7 @@ import * as deepmerge from 'deepmerge'; import 'localstorage-polyfill'; import { dateReviver, localStorageSync, rehydrateApplicationState, syncStateUpdate } from '../src/index'; const INIT_ACTION = '@ngrx/store/init'; +const UPDATE_ACTION = '@ngrx/store/update-reducers'; // Very simple classes to test serialization options. They cover string, number, date, and nested classes // The top level class has static functions to help test reviver, replacer, serialize and deserialize @@ -453,7 +454,7 @@ describe('ngrxLocalStorage', () => { const metaReducer = localStorageSync({keys: ['state'], rehydrate: true}); const action = {type: INIT_ACTION}; - // Resultant state should merge the oldstring state and our initual state + // Resultant state should merge the oldstring state and our initial state const finalState = metaReducer(reducer)(initialState, action); expect(finalState.state.astring).toEqual(initialState.state.astring); }); @@ -493,32 +494,32 @@ describe('ngrxLocalStorage', () => { feature1: { slice11: false, slice12: [], slice13: {} }, feature2: { slice21: false, slice22: [], slice23: {} }, }; - + // A legit case where state is saved in chunks rather than as a single object localStorage.setItem('feature1', JSON.stringify({ slice11: true, slice12: [1, 2] })); localStorage.setItem('feature2', JSON.stringify({ slice21: true, slice22: [1, 2] })); - + // Set up reducers const reducer = (state = initialState, action) => state; const mergeReducer = (state, rehydratedState, action) => { // Perform a merge where we only want a single property from feature1 // but a deepmerge with feature2 - return { + return { ...state, feature1: { slice11: rehydratedState.feature1.slice11 }, feature2: deepmerge(state.feature2, rehydratedState.feature2) - } - } + }; + }; const metaReducer = localStorageSync({keys: [ {'feature1': ['slice11', 'slice12']}, {'feature2': ['slice21', 'slice22']}, ], rehydrate: true, mergeReducer}); - + const action = {type: INIT_ACTION}; - + // Resultant state should merge the rehydrated partial state and our initial state const finalState = metaReducer(reducer)(initialState, action); expect(finalState).toEqual({ @@ -526,5 +527,39 @@ describe('ngrxLocalStorage', () => { feature1: { slice11: true }, feature2: { slice21: true, slice22: [1, 2], slice23: {} }, }); - }); + }); + + it('should have same reference after rehydrate', () => { + const myInitialState = { + app: t1, + feature3: { hello: 'World' }, + feature4: { slice21: false, slice22: [], slice23: {} }, + }; + const feature3LocalStorage = { hello: 'Peter' }; + + localStorage.setItem('feature3', JSON.stringify(feature3LocalStorage)); + + // Set up reducers + const reducer = (state = myInitialState, action) => state; + const metaReducer = localStorageSync({keys: ['feature3', 'feature4'], rehydrate: true}); + + const action = {type: UPDATE_ACTION}; + + // Resultant state should merge the rehydrated partial state and our initial state + const finalState = metaReducer(reducer)(myInitialState, action); + + // Global state should not have same reference + expect(myInitialState).not.toBe(finalState); + + // App state should have the same with same reference + expect(myInitialState.app).toBe(finalState.app); + + // Feature3 state should not have same reference + expect(myInitialState.feature3).not.toBe(finalState.feature3); + // Feature3 state should have localStorage value + expect(finalState.feature3).toEqual(feature3LocalStorage); + + // Feature4 state should be the same with same reference + expect(myInitialState.feature4).toBe(finalState.feature4); + }); }); diff --git a/src/index.ts b/src/index.ts index b9d7ce5..5fdd291 100644 --- a/src/index.ts +++ b/src/index.ts @@ -15,7 +15,7 @@ export const dateReviver = (key: string, value: any) => { const dummyReviver = (key: string, value: any) => value; const checkIsBrowserEnv = () => { - return typeof window !== 'undefined' + return typeof window !== 'undefined'; }; const validateStateKeys = (keys: any[]) => { @@ -220,15 +220,17 @@ export const syncStateUpdate = ( }; // Default merge strategy is a full deep merge. -export const defaultMergeReducer = (state: any, rehydratedState: any, action: any) => { - - if ((action.type === INIT_ACTION || action.type === UPDATE_ACTION) && rehydratedState) { - const overwriteMerge = (destinationArray, sourceArray, options) => sourceArray; - const options: deepmerge.Options = { - arrayMerge: overwriteMerge - }; - +export const defaultMergeReducer = (state: any, rehydratedState: any, action: any) => { + const overwriteMerge = (destinationArray, sourceArray, options) => sourceArray; + const options: deepmerge.Options = { + arrayMerge: overwriteMerge + }; + if (action.type === INIT_ACTION && rehydratedState) { state = deepmerge(state, rehydratedState, options); + } else if (action.type === UPDATE_ACTION && rehydratedState) { + Object.keys(rehydratedState).forEach(partialState => { + state[partialState] = deepmerge(state[partialState] || {}, rehydratedState[partialState], options); + }); } return state; @@ -282,7 +284,7 @@ export const localStorageSync = (config: LocalStorageConfig) => ( // Merge the store state with the rehydrated state using // either a user-defined reducer or the default. nextState = mergeReducer(nextState, rehydratedState, action); - + nextState = reducer(nextState, action); if (action.type !== INIT_ACTION) {