From 2204cc177c0263b7fd3157d18382f951e18964c4 Mon Sep 17 00:00:00 2001 From: crusader Date: Mon, 24 Jul 2017 19:58:45 +0900 Subject: [PATCH] reducer --- src/ts/@overflow/app/index.tsx | 22 ++++---- src/ts/@overflow/app/redux/store.ts | 2 - .../@overflow/commons/redux/ReducerManager.ts | 56 +++++++++++++++++++ .../commons/redux/StateTreeManager.ts | 0 .../@overflow/commons/redux/createReducer.ts | 19 +++++++ .../@overflow/member/redux/reducer/signIn.ts | 49 +++++++++------- 6 files changed, 117 insertions(+), 31 deletions(-) create mode 100644 src/ts/@overflow/commons/redux/ReducerManager.ts create mode 100644 src/ts/@overflow/commons/redux/StateTreeManager.ts create mode 100644 src/ts/@overflow/commons/redux/createReducer.ts diff --git a/src/ts/@overflow/app/index.tsx b/src/ts/@overflow/app/index.tsx index 7126f59..61b1445 100644 --- a/src/ts/@overflow/app/index.tsx +++ b/src/ts/@overflow/app/index.tsx @@ -12,6 +12,9 @@ import Platform from '@overflow/commons/platform'; import AppContext from '@overflow/commons/context'; import * as AppContextLifecycleActions from '@overflow/commons/context/redux/action/lifecycle'; import WebSocketRPC from '@overflow/commons/websocket/WebSocketRPC'; +import ReducerManager from '@overflow/commons/redux/ReducerManager'; + + import { store, sagaMiddleware, history } from './redux/store'; import appConfig, { Config } from './config'; @@ -33,12 +36,14 @@ class Application { private store: Store; private sagaMiddleware: SagaMiddleware; private history: History; + private reducerManager: ReducerManager; public constructor() { this.config = appConfig; this.sagaMiddleware = sagaMiddleware; this.store = store; this.history = history; + this.reducerManager = new ReducerManager(); } public static Run(): void { @@ -89,16 +94,15 @@ class Application { } private initRedux(): Promise { - // state tree - // reducer - // middleware - // saga - const rpcClient = new Promise(resolve => { - this.sagaMiddleware.run(sagas); - resolve(); + const init = new Promise(resolve => { + // state tree + // reducer + // middleware + // saga + }); - return rpcClient; + return init; } private initSagaEffect(): Promise { @@ -158,9 +162,7 @@ class Application { } private displayDebugApp(): void { if (module.hot) { - // app module.hot.accept('./views/App', async () => { - // const NextApp = require('./app').App; const NextApp = (await import('./views/App')).default; ReactDOM.render( diff --git a/src/ts/@overflow/app/redux/store.ts b/src/ts/@overflow/app/redux/store.ts index 6e07a57..211c88f 100644 --- a/src/ts/@overflow/app/redux/store.ts +++ b/src/ts/@overflow/app/redux/store.ts @@ -19,8 +19,6 @@ import { composeWithDevTools, } from 'redux-devtools-extension/logOnlyInProduction'; -import reduxLogger from 'redux-logger'; - import reducer from './reducer'; import State from './state'; diff --git a/src/ts/@overflow/commons/redux/ReducerManager.ts b/src/ts/@overflow/commons/redux/ReducerManager.ts new file mode 100644 index 0000000..d136ecc --- /dev/null +++ b/src/ts/@overflow/commons/redux/ReducerManager.ts @@ -0,0 +1,56 @@ +import Action from './Action'; +import * as Redux from 'redux'; + +interface ReducerItem { + initialState: State; + reducer: Redux.Reducer; +} + +class ReducerManager { + private reducerMap: Map[]>; + + public constructor() { + this.reducerMap = new Map(); + } + + public putReducer(type: string, initialState: State, reducer: Redux.Reducer): void { + let reducerItems: ReducerItem[]; + if (!this.reducerMap.has(type)) { + reducerItems = new Array(); + this.reducerMap.set(type, reducerItems); + } else { + reducerItems = this.reducerMap.get(type); + } + let reducerItem: ReducerItem = { + initialState: initialState, + reducer: reducer, + }; + + reducerItems.push(reducerItem); + } + + public putReducers(initialState: State, handlers: Redux.ReducersMapObject): void { + for(let type in handlers) { + if (handlers.hasOwnProperty(type)) { + this.putReducer(type, initialState, handlers[type]); + } + } + } + + public reducer(state: State, action: A): State { + const actionType = action.type; + if (this.reducerMap.has(actionType)) { + let reducerItems = this.reducerMap.get(actionType); + for (let reducerItem of reducerItems) { + let initialState = reducerItem.initialState; + let reducer = reducerItem.reducer; + state = reducer(state, action); + } + } else { + console.log(`Reducer for [${actionType}] is not loaded.`); + } + return state; + } +} + +export default ReducerManager; diff --git a/src/ts/@overflow/commons/redux/StateTreeManager.ts b/src/ts/@overflow/commons/redux/StateTreeManager.ts new file mode 100644 index 0000000..e69de29 diff --git a/src/ts/@overflow/commons/redux/createReducer.ts b/src/ts/@overflow/commons/redux/createReducer.ts new file mode 100644 index 0000000..1058e05 --- /dev/null +++ b/src/ts/@overflow/commons/redux/createReducer.ts @@ -0,0 +1,19 @@ +import Action from './Action'; +import * as Redux from 'redux'; + +// export type Reducer = (state: S, action: A) => S; + +// export interface ReducersMapObject { +// [key: string]: Reducer; +// } + +const createReducer = (initialState: State, handlers: Redux.ReducersMapObject): Redux.Reducer => { + return (state: State = initialState, action: A): State => { + if (handlers[action.type]) { + return handlers[action.type](state, action); + } + return state; + }; +}; + +export default createReducer; diff --git a/src/ts/@overflow/member/redux/reducer/signIn.ts b/src/ts/@overflow/member/redux/reducer/signIn.ts index f0ffafc..d06efb4 100644 --- a/src/ts/@overflow/member/redux/reducer/signIn.ts +++ b/src/ts/@overflow/member/redux/reducer/signIn.ts @@ -1,4 +1,5 @@ import Action from '@overflow/commons/redux/Action'; +import createReducer from '@overflow/commons/redux/createReducer'; import Member from '@overflow/member/api/model/Member'; import * as SigninActionTypes from '../action/signIn'; @@ -6,24 +7,34 @@ import SigninState, { defaultState as signinDefaultState } from '../state/SignIn export type reducer = (state: SigninState, action: Action) => SigninState; -export const reducer: reducer = (state: SigninState = signinDefaultState, action: Action): SigninState => { - switch (action.type) { - case SigninActionTypes.REQUEST_SUCCESS: - { - let member = (>action).payload; +export const reducer: reducer = createReducer(signinDefaultState, { + [SigninActionTypes.REQUEST_SUCCESS]: (state: SigninState, action: Action): SigninState => { + return state; + }, + [SigninActionTypes.REQUEST_FAILURE]: (state: SigninState, action: Action): SigninState => { + return state; + }, +}); - const aaa: SigninState = { - ...state, - isAuthenticated: true, - }; - return aaa; - } - case SigninActionTypes.REQUEST_FAILURE: - { - return state; - } - default: - return state; - } -}; +// export const reducer: reducer = (state: SigninState = signinDefaultState, action: Action): SigninState => { +// switch (action.type) { +// case SigninActionTypes.REQUEST_SUCCESS: +// { +// let member = (>action).payload; + +// const aaa: SigninState = { +// ...state, +// isAuthenticated: true, +// }; + +// return aaa; +// } +// case SigninActionTypes.REQUEST_FAILURE: +// { +// return state; +// } +// default: +// return state; +// } +// };