feat: export underlying views used to build navigators (#191)

Exporting the underlying views makes it easy to build custom navigators on top of our views. Libraries such as react-native-router-flux rely on such exports to build custom routing solutions while being able to take advantage of our work.

This can also be the solution to adding custom behaviour without us needing to add separate config to override the router.
This commit is contained in:
Satyajit Sahoo
2019-12-04 00:22:53 +01:00
committed by Michał Osadnik
parent c7a5cfd5b2
commit d618ab382e
13 changed files with 60 additions and 52 deletions

View File

@@ -0,0 +1,74 @@
import * as React from 'react';
import { View, StyleSheet } from 'react-native';
import { StackNavigationState, StackActions } from '@react-navigation/routers';
import {
// @ts-ignore
ScreenStack,
Screen as ScreenComponent,
ScreenProps,
// eslint-disable-next-line import/no-unresolved
} from 'react-native-screens';
import HeaderConfig from './HeaderConfig';
import {
NativeStackNavigationHelpers,
NativeStackDescriptorMap,
} from '../types';
const Screen = (ScreenComponent as unknown) as React.ComponentType<
ScreenProps & {
stackPresentation?: 'push' | 'modal' | 'transparentModal';
stackAnimation?: 'default' | 'fade' | 'none';
onDismissed?: () => void;
}
>;
type Props = {
state: StackNavigationState;
navigation: NativeStackNavigationHelpers;
descriptors: NativeStackDescriptorMap;
};
export default function NativeStackView({
state,
navigation,
descriptors,
}: Props) {
return (
<ScreenStack style={styles.scenes}>
{state.routes.map(route => {
const { options, render: renderScene } = descriptors[route.key];
const { presentation = 'push', animation, contentStyle } = options;
return (
<Screen
key={route.key}
style={StyleSheet.absoluteFill}
stackPresentation={presentation}
stackAnimation={animation}
onDismissed={() => {
navigation.dispatch({
...StackActions.pop(),
source: route.key,
target: state.key,
});
}}
>
<HeaderConfig {...options} route={route} />
<View style={[styles.content, contentStyle]}>{renderScene()}</View>
</Screen>
);
})}
</ScreenStack>
);
}
const styles = StyleSheet.create({
content: {
flex: 1,
backgroundColor: '#eee',
},
scenes: {
flex: 1,
},
});