UNPKG

mattermost-redux

Version:

Common code (API client, Redux stores, logic, utility functions) for building a Mattermost client

101 lines (85 loc) 4.55 kB
// Copyright (c) 2015-present Mattermost, Inc. All Rights Reserved. // See LICENSE.txt for license information. import * as redux from 'redux'; import {createOfflineReducer, networkStatusChangedAction, offlineCompose} from 'redux-offline'; import defaultOfflineConfig from 'redux-offline/lib/defaults'; import reducerRegistry from './reducer_registry'; import devTools from 'remote-redux-devtools'; const windowAny = window as any; const devToolsEnhancer = typeof windowAny !== 'undefined' && windowAny.__REDUX_DEVTOOLS_EXTENSION__ ? // eslint-disable-line no-underscore-dangle windowAny.__REDUX_DEVTOOLS_EXTENSION__ : // eslint-disable-line no-underscore-dangle () => { return devTools({ name: 'Mattermost', hostname: 'localhost', port: 5678, realtime: true, }); }; import serviceReducer from '../reducers'; import deepFreezeAndThrowOnMutation from 'utils/deep_freeze'; import initialState from './initial_state'; import {offlineConfig, createReducer} from './helpers'; import {createMiddleware} from './middleware'; import {Reducer, Action} from 'types/actions'; import {GlobalState} from 'types/store'; /** * Configures and constructs the redux store. Accepts the following parameters: * preloadedState - Any preloaded state to be applied to the store after it is initially configured. * appReducer - An object containing any app-specific reducer functions that the client needs. * userOfflineConfig - Any additional configuration data to be passed into redux-offline aside from the default values. * getAppReducer - A function that returns the appReducer as defined above. Only used in development to enable hot reloading. * clientOptions - An object containing additional options used when configuring the redux store. The following options are available: * additionalMiddleware - func | array - Allows for single or multiple additional middleware functions to be passed in from the client side. * enableBuffer - bool - default = true - If true, the store will buffer all actions until offline state rehydration occurs. * enableThunk - bool - default = true - If true, include the thunk middleware automatically. If false, thunk must be provided as part of additionalMiddleware. */ export default function configureServiceStore(preloadedState: any, appReducer: any, userOfflineConfig: any, getAppReducer: any, clientOptions: any) { const baseOfflineConfig = Object.assign({}, defaultOfflineConfig, offlineConfig, userOfflineConfig); const baseState = Object.assign({}, initialState, preloadedState); const loadReduxDevtools = process.env.NODE_ENV !== 'test'; //eslint-disable-line no-process-env const store = redux.createStore( createOfflineReducer(createDevReducer(baseState, serviceReducer, appReducer)), baseState, offlineCompose(baseOfflineConfig)( createMiddleware(clientOptions), loadReduxDevtools ? [devToolsEnhancer()] : [], ), ); reducerRegistry.setChangeListener((reducers: any) => { store.replaceReducer(createOfflineReducer(createDevReducer(baseState, reducers))); }); // launch store persistor if (baseOfflineConfig.persist) { baseOfflineConfig.persist(store, baseOfflineConfig.persistOptions, baseOfflineConfig.persistCallback); } if (baseOfflineConfig.detectNetwork) { baseOfflineConfig.detectNetwork((online: boolean) => { store.dispatch(networkStatusChangedAction(online)); }); } if ((module as any).hot) { // Enable Webpack hot module replacement for reducers (module as any).hot.accept(() => { const nextServiceReducer = require('../reducers').default; // eslint-disable-line global-require let nextAppReducer; if (getAppReducer) { nextAppReducer = getAppReducer(); // eslint-disable-line global-require } store.replaceReducer(createDevReducer(baseState, reducerRegistry.getReducers(), nextServiceReducer, nextAppReducer)); }); } return store; } function createDevReducer(baseState: any, ...reducers: any) { return enableFreezing(createReducer(baseState, ...reducers)); } function enableFreezing(reducer: Reducer) { return (state: GlobalState, action: Action) => { const nextState = reducer(state, action); if (nextState !== state) { deepFreezeAndThrowOnMutation(nextState); } return nextState; }; }