createConfigGetter.js
2.46 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
/*
* @flow
*/
import invariant from '../utils/invariant';
import getScreenForRouteName from './getScreenForRouteName';
import addNavigationHelpers from '../addNavigationHelpers';
import validateScreenOptions from './validateScreenOptions';
import type {
NavigationScreenProp,
NavigationRoute,
NavigationStateRoute,
NavigationRouteConfigMap,
NavigationScreenConfig,
NavigationScreenConfigProps,
} from '../TypeDefinition';
function applyConfig<T: {}>(
configurer: ?NavigationScreenConfig<T>,
navigationOptions: any,
configProps: NavigationScreenConfigProps
): * {
if (typeof configurer === 'function') {
return {
...navigationOptions,
...configurer({
...configProps,
navigationOptions,
}),
};
}
if (typeof configurer === 'object') {
return {
...navigationOptions,
...configurer,
};
}
return navigationOptions;
}
export default (
routeConfigs: NavigationRouteConfigMap,
navigatorScreenConfig?: NavigationScreenConfig<*>
) => (navigation: NavigationScreenProp<NavigationRoute>, screenProps: *) => {
const { state, dispatch } = navigation;
const route = state;
invariant(
route.routeName && typeof route.routeName === 'string',
'Cannot get config because the route does not have a routeName.'
);
const Component = getScreenForRouteName(routeConfigs, route.routeName);
let outputConfig = {};
const router = Component.router;
if (router) {
// $FlowFixMe
const { routes, index } = (route: NavigationStateRoute);
if (!route || !routes || index == null) {
throw new Error(
`Expect nav state to have routes and index, ${JSON.stringify(route)}`
);
}
const childRoute = routes[index];
const childNavigation = addNavigationHelpers({
state: childRoute,
dispatch,
});
outputConfig = router.getScreenOptions(childNavigation, screenProps);
}
const routeConfig = routeConfigs[route.routeName];
const routeScreenConfig = routeConfig.navigationOptions;
const componentScreenConfig = Component.navigationOptions;
const configOptions = { navigation, screenProps: screenProps || {} };
outputConfig = applyConfig(
navigatorScreenConfig,
outputConfig,
configOptions
);
outputConfig = applyConfig(
componentScreenConfig,
outputConfig,
configOptions
);
outputConfig = applyConfig(routeScreenConfig, outputConfig, configOptions);
validateScreenOptions(outputConfig, route);
return outputConfig;
};