mirror of
https://github.com/status-im/react-native.git
synced 2025-01-26 17:30:25 +00:00
dcb68db758
Summary: Revise APIs of reducers, and ensure the stack reducer can support sub-reducers Reviewed By: javache Differential Revision: D2959915 fb-gh-sync-id: 20b28b9ead7ace3373489a806486999048d32aef shipit-source-id: 20b28b9ead7ace3373489a806486999048d32aef
108 lines
3.2 KiB
JavaScript
108 lines
3.2 KiB
JavaScript
/**
|
|
* Copyright (c) 2015-present, Facebook, Inc.
|
|
* All rights reserved.
|
|
*
|
|
* This source code is licensed under the BSD-style license found in the
|
|
* LICENSE file in the root directory of this source tree. An additional grant
|
|
* of patent rights can be found in the PATENTS file in the same directory.
|
|
*
|
|
* @providesModule NavigationStackReducer
|
|
* @flow
|
|
*/
|
|
'use strict';
|
|
|
|
var NavigationStateUtils = require('NavigationStateUtils');
|
|
|
|
import type {
|
|
NavigationState,
|
|
NavigationParentState,
|
|
NavigationReducer,
|
|
} from 'NavigationStateUtils';
|
|
|
|
import type {
|
|
BackAction,
|
|
} from 'NavigationRootContainer';
|
|
|
|
export type NavigationStackReducerAction = BackAction | {
|
|
type: string,
|
|
};
|
|
|
|
export type ReducerForStateHandler = (state: NavigationState) => NavigationReducer;
|
|
|
|
export type PushedReducerForActionHandler = (action: any) => ?NavigationReducer;
|
|
|
|
export type StackReducerConfig = {
|
|
/*
|
|
* The initialState is that the reducer will use when there is no previous state.
|
|
* Must be a NavigationParentState:
|
|
*
|
|
* {
|
|
* children: [
|
|
* {key: 'subState0'},
|
|
* {key: 'subState1'},
|
|
* ],
|
|
* index: 0,
|
|
* key: 'navStackKey'
|
|
* }
|
|
*/
|
|
initialState: NavigationParentState;
|
|
|
|
/*
|
|
* Returns the sub-reducer for a particular state to handle. This will be called
|
|
* when we need to handle an action on a sub-state. If no reducer is returned,
|
|
* no action will be taken
|
|
*/
|
|
getReducerForState?: ReducerForStateHandler;
|
|
|
|
/*
|
|
* Returns a sub-reducer that will be used when pushing a new route. If a reducer
|
|
* is returned, it be called to get the new state that will be pushed
|
|
*/
|
|
getPushedReducerForAction: PushedReducerForActionHandler;
|
|
};
|
|
|
|
const defaultGetReducerForState = (initialState) => (state) => state || initialState;
|
|
|
|
function NavigationStackReducer({initialState, getReducerForState, getPushedReducerForAction}: StackReducerConfig): NavigationReducer {
|
|
const getReducerForStateWithDefault = getReducerForState || defaultGetReducerForState;
|
|
return function (lastState: ?NavigationState, action: any): NavigationState {
|
|
if (!lastState) {
|
|
return initialState;
|
|
}
|
|
const lastParentState = NavigationStateUtils.getParent(lastState);
|
|
if (!lastParentState) {
|
|
return lastState;
|
|
}
|
|
switch (action.type) {
|
|
case 'BackAction':
|
|
if (lastParentState.index === 0 || lastParentState.children.length === 1) {
|
|
return lastParentState;
|
|
}
|
|
return NavigationStateUtils.pop(lastParentState);
|
|
}
|
|
|
|
const activeSubState = lastParentState.children[lastParentState.index];
|
|
const activeSubReducer = getReducerForStateWithDefault(activeSubState);
|
|
const nextActiveState = activeSubReducer(activeSubState, action);
|
|
if (nextActiveState !== activeSubState) {
|
|
const nextChildren = [...lastParentState.children];
|
|
nextChildren[lastParentState.index] = nextActiveState;
|
|
return {
|
|
...lastParentState,
|
|
children: nextChildren,
|
|
};
|
|
}
|
|
|
|
const subReducerToPush = getPushedReducerForAction(action);
|
|
if (subReducerToPush) {
|
|
return NavigationStateUtils.push(
|
|
lastParentState,
|
|
subReducerToPush(null, action)
|
|
);
|
|
}
|
|
return lastParentState;
|
|
};
|
|
}
|
|
|
|
module.exports = NavigationStackReducer;
|