2016-02-05 22:25:17 +00: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 09:24:07 +00:00
|
|
|
* @providesModule NavigationStateUtils
|
2016-02-05 22:25:17 +00:00
|
|
|
* @flow
|
|
|
|
*/
|
|
|
|
'use strict';
|
|
|
|
|
2016-03-02 18:14:12 +00:00
|
|
|
const invariant = require('fbjs/lib/invariant');
|
2016-02-05 22:25:17 +00:00
|
|
|
|
2016-03-04 22:56:37 +00:00
|
|
|
import type {
|
2016-05-20 21:24:24 +00:00
|
|
|
NavigationRoute,
|
2016-03-04 22:56:37 +00:00
|
|
|
NavigationState,
|
|
|
|
} from 'NavigationTypeDefinition';
|
2016-02-05 22:25:17 +00:00
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Utilities to perform atomic operation with navigate state and routes.
|
|
|
|
*/
|
2016-02-05 22:25:17 +00:00
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Gets a route by key
|
|
|
|
*/
|
2016-05-22 23:27:53 +00:00
|
|
|
function get(state: NavigationState, key: string): ?NavigationRoute {
|
2016-06-08 21:58:22 +00:00
|
|
|
return state.routes.find(route => route.key === key) || null;
|
2016-02-05 22:25:17 +00:00
|
|
|
}
|
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Returns the first index at which a given route's key can be found in the
|
|
|
|
* routes of the navigation state, or -1 if it is not present.
|
|
|
|
*/
|
|
|
|
function indexOf(state: NavigationState, key: string): number {
|
|
|
|
return state.routes.map(route => route.key).indexOf(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns `true` at which a given route's key can be found in the
|
|
|
|
* routes of the navigation state.
|
|
|
|
*/
|
|
|
|
function has(state: NavigationState, key: string): boolean {
|
|
|
|
return !!state.routes.some(route => route.key === key);
|
2016-02-05 22:25:17 +00:00
|
|
|
}
|
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Pushes a new route into the navigation state.
|
|
|
|
* Note that this moves the index to the positon to where the last route in the
|
|
|
|
* stack is at.
|
|
|
|
*/
|
|
|
|
function push(state: NavigationState, route: NavigationRoute): NavigationState {
|
|
|
|
invariant(
|
|
|
|
indexOf(state, route.key) === -1,
|
|
|
|
'should not push route with duplicated key %s',
|
|
|
|
route.key,
|
|
|
|
);
|
|
|
|
|
|
|
|
const routes = [
|
|
|
|
...state.routes,
|
|
|
|
route,
|
|
|
|
];
|
|
|
|
|
2016-02-05 22:25:17 +00:00
|
|
|
return {
|
2016-02-23 00:15:42 +00:00
|
|
|
...state,
|
2016-06-08 21:58:22 +00:00
|
|
|
index: routes.length - 1,
|
|
|
|
routes,
|
2016-02-05 22:25:17 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Pops out a route from the navigation state.
|
|
|
|
* Note that this moves the index to the positon to where the last route in the
|
|
|
|
* stack is at.
|
|
|
|
*/
|
2016-05-20 21:24:24 +00:00
|
|
|
function pop(state: NavigationState): NavigationState {
|
2016-05-31 18:32:19 +00:00
|
|
|
if (state.index <= 0) {
|
2016-06-08 21:58:22 +00:00
|
|
|
// [Note]: Over-popping does not throw error. Instead, it will be no-op.
|
2016-05-31 18:32:19 +00:00
|
|
|
return state;
|
|
|
|
}
|
2016-06-08 21:58:22 +00:00
|
|
|
const routes = state.routes.slice(0, -1);
|
2016-02-05 22:25:17 +00:00
|
|
|
return {
|
2016-02-23 00:15:42 +00:00
|
|
|
...state,
|
2016-06-08 21:58:22 +00:00
|
|
|
index: routes.length - 1,
|
|
|
|
routes,
|
2016-02-05 22:25:17 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Sets the focused route of the navigation state by index.
|
|
|
|
*/
|
|
|
|
function jumpToIndex(state: NavigationState, index: number): NavigationState {
|
|
|
|
if (index === state.index) {
|
2016-02-05 22:25:17 +00:00
|
|
|
return state;
|
|
|
|
}
|
2016-06-08 21:58:22 +00:00
|
|
|
|
|
|
|
invariant(!!state.routes[index], 'invalid index %s to jump to', index);
|
|
|
|
|
2016-02-05 22:25:17 +00:00
|
|
|
return {
|
2016-06-08 21:58:22 +00:00
|
|
|
...state,
|
2016-02-05 22:25:17 +00:00
|
|
|
index,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Sets the focused route of the navigation state by key.
|
|
|
|
*/
|
|
|
|
function jumpTo(state: NavigationState, key: string): NavigationState {
|
|
|
|
const index = indexOf(state, key);
|
|
|
|
return jumpToIndex(state, index);
|
2016-02-05 22:25:17 +00:00
|
|
|
}
|
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Replace a route by a key.
|
|
|
|
* Note that this moves the index to the positon to where the new route in the
|
|
|
|
* stack is at.
|
|
|
|
*/
|
|
|
|
function replaceAt(
|
|
|
|
state: NavigationState,
|
|
|
|
key: string,
|
|
|
|
route: NavigationRoute,
|
|
|
|
): NavigationState {
|
|
|
|
const index = indexOf(state, key);
|
|
|
|
return replaceAtIndex(state, index, route);
|
2016-02-05 22:25:17 +00:00
|
|
|
}
|
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Replace a route by a index.
|
|
|
|
* Note that this moves the index to the positon to where the new route in the
|
|
|
|
* stack is at.
|
|
|
|
*/
|
|
|
|
function replaceAtIndex(
|
|
|
|
state: NavigationState,
|
|
|
|
index: number,
|
|
|
|
route: NavigationRoute,
|
|
|
|
): NavigationState {
|
2016-02-05 22:25:17 +00:00
|
|
|
invariant(
|
2016-06-08 21:58:22 +00:00
|
|
|
!!state.routes[index],
|
|
|
|
'invalid index %s for replacing route %s',
|
2016-02-05 22:25:17 +00:00
|
|
|
index,
|
2016-06-08 21:58:22 +00:00
|
|
|
route.key,
|
|
|
|
);
|
2016-02-05 22:25:17 +00:00
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
if (state.routes[index] === route) {
|
2016-02-05 22:25:17 +00:00
|
|
|
return state;
|
|
|
|
}
|
2016-06-08 21:58:22 +00:00
|
|
|
|
|
|
|
const routes = state.routes.slice();
|
|
|
|
routes[index] = route;
|
|
|
|
|
2016-02-05 22:25:17 +00:00
|
|
|
return {
|
2016-06-08 21:58:22 +00:00
|
|
|
...state,
|
|
|
|
index,
|
2016-05-22 23:27:53 +00:00
|
|
|
routes,
|
2016-02-05 22:25:17 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2016-06-08 21:58:22 +00:00
|
|
|
/**
|
|
|
|
* Resets all routes.
|
|
|
|
* Note that this moves the index to the positon to where the last route in the
|
|
|
|
* stack is at if the param `index` isn't provided.
|
|
|
|
*/
|
|
|
|
function reset(
|
|
|
|
state: NavigationState,
|
|
|
|
routes: Array<NavigationRoute>,
|
|
|
|
index?: number,
|
|
|
|
): NavigationState {
|
|
|
|
invariant(
|
|
|
|
routes.length && Array.isArray(routes),
|
|
|
|
'invalid routes to replace',
|
|
|
|
);
|
|
|
|
|
|
|
|
const nextIndex: number = index === undefined ? routes.length - 1 : index;
|
|
|
|
|
|
|
|
if (state.routes.length === routes.length && state.index === nextIndex) {
|
|
|
|
const compare = (route, ii) => routes[ii] === route;
|
|
|
|
if (state.routes.every(compare)) {
|
|
|
|
return state;
|
|
|
|
}
|
2016-02-05 22:25:17 +00:00
|
|
|
}
|
2016-06-08 21:58:22 +00:00
|
|
|
|
|
|
|
invariant(!!routes[nextIndex], 'invalid index %s to reset', nextIndex);
|
|
|
|
|
2016-02-05 22:25:17 +00:00
|
|
|
return {
|
2016-06-08 21:58:22 +00:00
|
|
|
...state,
|
|
|
|
index: nextIndex,
|
2016-05-22 23:27:53 +00:00
|
|
|
routes,
|
2016-02-05 22:25:17 +00:00
|
|
|
};
|
|
|
|
}
|
2016-02-19 09:24:07 +00:00
|
|
|
|
|
|
|
const NavigationStateUtils = {
|
|
|
|
get: get,
|
2016-06-08 21:58:22 +00:00
|
|
|
has,
|
2016-02-19 09:24:07 +00:00
|
|
|
indexOf,
|
|
|
|
jumpTo,
|
2016-05-27 01:07:48 +00:00
|
|
|
jumpToIndex,
|
|
|
|
pop,
|
|
|
|
push,
|
2016-02-19 09:24:07 +00:00
|
|
|
replaceAt,
|
|
|
|
replaceAtIndex,
|
2016-05-27 01:07:48 +00:00
|
|
|
reset,
|
2016-02-19 09:24:07 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
module.exports = NavigationStateUtils;
|