core-native
Version:
A lightweight framework based on React Native + Redux + Redux Saga, in strict TypeScript.
67 lines (60 loc) • 2.34 kB
text/typescript
import {applyMiddleware, compose, createStore, Store, StoreEnhancer} from "redux";
import createSagaMiddleware, {SagaMiddleware} from "redux-saga";
import {put, takeEvery} from "redux-saga/effects";
import {LoggerImpl, LoggerConfig} from "./Logger";
import {ActionHandler, ErrorHandler} from "./module";
import {Action, ERROR_ACTION_TYPE, errorAction, LOADING_ACTION, rootReducer, State} from "./reducer";
declare const window: any;
interface App {
readonly store: Store<State>;
readonly sagaMiddleware: SagaMiddleware<any>;
readonly actionHandlers: {[actionType: string]: ActionHandler};
readonly logger: LoggerImpl;
errorHandler: ErrorHandler | null;
loggerConfig: LoggerConfig | null;
}
function composeWithDevTools(enhancer: StoreEnhancer): StoreEnhancer {
let composeEnhancers = compose;
if (process.env.NODE_ENV !== "production") {
const extension = window.__REDUX_DEVTOOLS_EXTENSION_COMPOSE__;
if (extension) {
composeEnhancers = extension({
// Ref: https://github.com/zalmoxisus/redux-devtools-extension/blob/master/docs/API/Arguments.md
actionsBlacklist: [LOADING_ACTION],
});
}
}
return composeEnhancers(enhancer);
}
function createApp(): App {
const eventLogger = new LoggerImpl();
const sagaMiddleware = createSagaMiddleware();
const store: Store<State> = createStore(rootReducer(), composeWithDevTools(applyMiddleware(sagaMiddleware)));
sagaMiddleware.run(function* rootSaga() {
yield takeEvery("*", function*(action: Action<any>) {
if (action.type === ERROR_ACTION_TYPE) {
if (app.errorHandler) {
yield* app.errorHandler(action.payload);
}
} else {
const handler = app.actionHandlers[action.type];
if (handler) {
try {
yield* handler(...action.payload);
} catch (error) {
yield put(errorAction(error));
}
}
}
});
});
return {
store,
sagaMiddleware,
actionHandlers: {},
logger: eventLogger,
errorHandler: null,
loggerConfig: null,
};
}
export const app = createApp();