2016-02-05 14:25:17 -08:00
|
|
|
/**
|
|
|
|
* 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.
|
|
|
|
*
|
2016-02-19 01:24:07 -08:00
|
|
|
* @providesModule NavigationStateUtils
|
2016-02-05 14:25:17 -08:00
|
|
|
* @flow
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
2016-03-02 10:14:12 -08:00
|
|
|
const invariant = require('fbjs/lib/invariant');
|
2016-02-05 14:25:17 -08:00
|
|
|
|
2016-03-04 14:56:37 -08:00
|
|
|
import type {
|
2016-05-20 14:24:24 -07:00
|
|
|
NavigationRoute,
|
2016-03-04 14:56:37 -08:00
|
|
|
NavigationState,
|
|
|
|
} from 'NavigationTypeDefinition';
|
2016-02-05 14:25:17 -08:00
|
|
|
|
2016-05-22 16:27:53 -07:00
|
|
|
function getParent(state: NavigationState): ?NavigationState {
|
2016-02-05 14:25:17 -08:00
|
|
|
if (
|
|
|
|
(state instanceof Object) &&
|
2016-05-22 16:27:53 -07:00
|
|
|
(state.routes instanceof Array) &&
|
|
|
|
(state.routes[0] !== undefined) &&
|
2016-02-05 14:25:17 -08:00
|
|
|
(typeof state.index === 'number') &&
|
2016-05-22 16:27:53 -07:00
|
|
|
(state.routes[state.index] !== undefined)
|
2016-02-05 14:25:17 -08:00
|
|
|
) {
|
|
|
|
return state;
|
|
|
|
}
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
2016-05-22 16:27:53 -07:00
|
|
|
function get(state: NavigationState, key: string): ?NavigationRoute {
|
2016-02-05 14:25:17 -08:00
|
|
|
const parentState = getParent(state);
|
|
|
|
if (!parentState) {
|
|
|
|
return null;
|
|
|
|
}
|
2016-05-22 16:27:53 -07:00
|
|
|
const childState = parentState.routes.find(child => child.key === key);
|
2016-02-05 14:25:17 -08:00
|
|
|
return childState || null;
|
|
|
|
}
|
|
|
|
|
2016-05-22 16:27:53 -07:00
|
|
|
function indexOf(state: NavigationState, key: string): ?number {
|
2016-02-05 14:25:17 -08:00
|
|
|
const parentState = getParent(state);
|
|
|
|
if (!parentState) {
|
|
|
|
return null;
|
|
|
|
}
|
2016-05-22 16:27:53 -07:00
|
|
|
const index = parentState.routes.map(child => child.key).indexOf(key);
|
2016-02-05 14:25:17 -08:00
|
|
|
if (index === -1) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
return index;
|
|
|
|
}
|
|
|
|
|
2016-05-20 14:24:24 -07:00
|
|
|
function push(state: NavigationState, newChildState: NavigationRoute): NavigationState {
|
2016-05-22 16:27:53 -07:00
|
|
|
var lastChildren: Array<NavigationRoute> = state.routes;
|
2016-02-05 14:25:17 -08:00
|
|
|
return {
|
2016-02-22 16:15:42 -08:00
|
|
|
...state,
|
2016-05-22 16:27:53 -07:00
|
|
|
routes: [
|
2016-02-05 14:25:17 -08:00
|
|
|
...lastChildren,
|
|
|
|
newChildState,
|
|
|
|
],
|
|
|
|
index: lastChildren.length,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-20 14:24:24 -07:00
|
|
|
function pop(state: NavigationState): NavigationState {
|
2016-05-31 11:32:19 -07:00
|
|
|
if (state.index <= 0) {
|
|
|
|
return state;
|
|
|
|
}
|
2016-05-22 16:27:53 -07:00
|
|
|
const lastChildren = state.routes;
|
2016-02-05 14:25:17 -08:00
|
|
|
return {
|
2016-02-22 16:15:42 -08:00
|
|
|
...state,
|
2016-05-22 16:27:53 -07:00
|
|
|
routes: lastChildren.slice(0, lastChildren.length - 1),
|
2016-02-05 14:25:17 -08:00
|
|
|
index: lastChildren.length - 2,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-22 16:27:53 -07:00
|
|
|
function reset(state: NavigationState, nextChildren: ?Array<NavigationRoute>, nextIndex: ?number): NavigationState {
|
2016-02-05 14:25:17 -08:00
|
|
|
const parentState = getParent(state);
|
|
|
|
if (!parentState) {
|
|
|
|
return state;
|
|
|
|
}
|
2016-05-22 16:27:53 -07:00
|
|
|
const routes = nextChildren || parentState.routes;
|
2016-02-05 14:25:17 -08:00
|
|
|
const index = nextIndex == null ? parentState.index : nextIndex;
|
2016-05-22 16:27:53 -07:00
|
|
|
if (routes === parentState.routes && index === parentState.index) {
|
2016-02-05 14:25:17 -08:00
|
|
|
return state;
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
...parentState,
|
2016-05-22 16:27:53 -07:00
|
|
|
routes,
|
2016-02-05 14:25:17 -08:00
|
|
|
index,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-22 16:27:53 -07:00
|
|
|
function set(state: ?NavigationState, key: string, nextChildren: Array<NavigationRoute>, nextIndex: number): NavigationState {
|
2016-02-05 14:25:17 -08:00
|
|
|
if (!state) {
|
|
|
|
return {
|
2016-05-22 16:27:53 -07:00
|
|
|
routes: nextChildren,
|
2016-02-05 14:25:17 -08:00
|
|
|
index: nextIndex,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
const parentState = getParent(state);
|
|
|
|
if (!parentState) {
|
|
|
|
return {
|
2016-05-22 16:27:53 -07:00
|
|
|
routes: nextChildren,
|
2016-02-05 14:25:17 -08:00
|
|
|
index: nextIndex,
|
|
|
|
};
|
|
|
|
}
|
2016-05-26 18:07:48 -07:00
|
|
|
if (nextChildren === parentState.routes && nextIndex === parentState.index) {
|
2016-02-05 14:25:17 -08:00
|
|
|
return parentState;
|
|
|
|
}
|
|
|
|
return {
|
|
|
|
...parentState,
|
2016-05-22 16:27:53 -07:00
|
|
|
routes: nextChildren,
|
2016-02-05 14:25:17 -08:00
|
|
|
index: nextIndex,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-22 16:27:53 -07:00
|
|
|
function jumpToIndex(state: NavigationState, index: number): NavigationState {
|
2016-02-05 14:25:17 -08:00
|
|
|
const parentState = getParent(state);
|
2016-02-08 20:02:45 -08:00
|
|
|
if (parentState && parentState.index === index) {
|
|
|
|
return parentState;
|
|
|
|
}
|
2016-02-05 14:25:17 -08:00
|
|
|
return {
|
|
|
|
...parentState,
|
|
|
|
index,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-22 16:27:53 -07:00
|
|
|
function jumpTo(state: NavigationState, key: string): NavigationState {
|
2016-02-05 14:25:17 -08:00
|
|
|
const parentState = getParent(state);
|
|
|
|
if (!parentState) {
|
|
|
|
return state;
|
|
|
|
}
|
2016-05-22 16:27:53 -07:00
|
|
|
const index = parentState.routes.indexOf(parentState.routes.find(child => child.key === key));
|
2016-02-05 14:25:17 -08:00
|
|
|
invariant(
|
|
|
|
index !== -1,
|
2016-05-20 14:24:24 -07:00
|
|
|
'Cannot find child with matching key in this NavigationRoute'
|
2016-02-05 14:25:17 -08:00
|
|
|
);
|
|
|
|
return {
|
|
|
|
...parentState,
|
|
|
|
index,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-26 18:07:48 -07:00
|
|
|
function replaceAt(state: NavigationState, key: string, newState: NavigationRoute): NavigationState {
|
2016-02-05 14:25:17 -08:00
|
|
|
const parentState = getParent(state);
|
|
|
|
if (!parentState) {
|
|
|
|
return state;
|
|
|
|
}
|
2016-05-22 16:27:53 -07:00
|
|
|
const routes = [...parentState.routes];
|
|
|
|
const index = parentState.routes.indexOf(parentState.routes.find(child => child.key === key));
|
2016-02-05 14:25:17 -08:00
|
|
|
invariant(
|
|
|
|
index !== -1,
|
2016-05-20 14:24:24 -07:00
|
|
|
'Cannot find child with matching key in this NavigationRoute'
|
2016-02-05 14:25:17 -08:00
|
|
|
);
|
2016-05-22 16:27:53 -07:00
|
|
|
routes[index] = newState;
|
2016-02-05 14:25:17 -08:00
|
|
|
return {
|
|
|
|
...parentState,
|
2016-05-22 16:27:53 -07:00
|
|
|
routes,
|
2016-02-05 14:25:17 -08:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-05-26 18:07:48 -07:00
|
|
|
function replaceAtIndex(state: NavigationState, index: number, newState: NavigationRoute): NavigationState {
|
2016-02-05 14:25:17 -08:00
|
|
|
const parentState = getParent(state);
|
|
|
|
if (!parentState) {
|
|
|
|
return state;
|
|
|
|
}
|
2016-05-22 16:27:53 -07:00
|
|
|
const routes = [...parentState.routes];
|
|
|
|
routes[index] = newState;
|
2016-02-05 14:25:17 -08:00
|
|
|
return {
|
|
|
|
...parentState,
|
2016-05-22 16:27:53 -07:00
|
|
|
routes,
|
2016-02-05 14:25:17 -08:00
|
|
|
};
|
|
|
|
}
|
2016-02-19 01:24:07 -08:00
|
|
|
|
|
|
|
const NavigationStateUtils = {
|
|
|
|
get: get,
|
2016-05-26 18:07:48 -07:00
|
|
|
getParent,
|
2016-02-19 01:24:07 -08:00
|
|
|
indexOf,
|
|
|
|
jumpTo,
|
2016-05-26 18:07:48 -07:00
|
|
|
jumpToIndex,
|
|
|
|
pop,
|
|
|
|
push,
|
2016-02-19 01:24:07 -08:00
|
|
|
replaceAt,
|
|
|
|
replaceAtIndex,
|
2016-05-26 18:07:48 -07:00
|
|
|
reset,
|
|
|
|
set: set,
|
2016-02-19 01:24:07 -08:00
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = NavigationStateUtils;
|