Files
react-navigation/example/src/ParallaxDrawer.tsx
2020-02-24 17:27:10 +01:00

266 lines
7.0 KiB
TypeScript

import React from 'react';
import {
Button,
Dimensions,
TouchableOpacity,
ScrollView,
StyleSheet,
View,
} from 'react-native';
import {
ThemeColors,
useTheme,
Themed,
SafeAreaView,
NavigationRoute,
} from 'react-navigation';
import {
createDrawerNavigator,
DrawerContentComponentProps,
NavigationDrawerOptions,
NavigationDrawerProp,
NavigationDrawerScreenComponent,
} from 'react-navigation-drawer';
import Animated from 'react-native-reanimated';
import { MaterialIcons } from '@expo/vector-icons';
type Params = { drawerLockMode: 'unlocked' | 'locked-open' | 'locked-closed' };
const SampleText = ({ children }: { children: React.ReactNode }) => (
<Themed.Text>{children}</Themed.Text>
);
const MyNavScreen = ({
navigation,
banner,
}: {
navigation: NavigationDrawerProp<NavigationRoute, Params>;
banner: string;
}) => {
let theme = useTheme();
return (
<ScrollView>
<SafeAreaView forceInset={{ top: 'always' }}>
<View
style={{ flex: 1, alignItems: 'center', justifyContent: 'center' }}
>
<SampleText>{banner}</SampleText>
</View>
<Themed.TextInput
style={{
flex: 1,
height: 35,
marginHorizontal: 10,
marginVertical: 10,
borderWidth: StyleSheet.hairlineWidth,
borderColor: ThemeColors[theme].bodyBorder,
textAlign: 'center',
}}
placeholder="Focus this TextInput then drag the drawer!"
/>
<Button onPress={() => navigation.openDrawer()} title="Open drawer" />
<Button
onPress={() => navigation.toggleDrawer()}
title="Toggle drawer"
/>
<Button
onPress={() => {
navigation.openDrawer();
navigation.closeDrawer();
}}
title="Open and immediately close"
/>
<Button
onPress={() => {
navigation.closeDrawer();
navigation.openDrawer();
}}
title="Close and immediately open"
/>
<Button
onPress={() => {
navigation.openDrawer();
setTimeout(() => {
navigation.closeDrawer();
}, 150);
}}
title="Open then close drawer shortly after"
/>
<Button
onPress={() => navigation.navigate('Email')}
title="Open other screen"
/>
<Button onPress={() => navigation.goBack(null)} title="Go back" />
<Button
onPress={() => navigation.navigate('Index')}
title="Go back to list"
/>
{
{
'locked-open': (
<Button
onPress={() =>
navigation.setParams({ drawerLockMode: 'locked-closed' })
}
title="Set locked-closed"
/>
),
'locked-closed': (
<Button
onPress={() =>
navigation.setParams({ drawerLockMode: 'unlocked' })
}
title="Set unlocked"
/>
),
unlocked: (
<Button
onPress={() =>
navigation.setParams({ drawerLockMode: 'locked-open' })
}
title="Set locked-open"
/>
),
}[navigation.getParam('drawerLockMode', 'unlocked')]
}
</SafeAreaView>
<Themed.StatusBar />
</ScrollView>
);
};
const InboxScreen: NavigationDrawerScreenComponent<Params> = ({
navigation,
}) => <MyNavScreen banner="Inbox Screen" navigation={navigation} />;
const EmailScreen: NavigationDrawerScreenComponent<Params> = ({
navigation,
}) => <MyNavScreen banner="Email Screen" navigation={navigation} />;
const DraftsScreen: NavigationDrawerScreenComponent<Params> = ({
navigation,
}) => <MyNavScreen banner="Drafts Screen" navigation={navigation} />;
const DrawerContents = ({
drawerOpenProgress,
descriptors,
navigation,
}: DrawerContentComponentProps) => {
// `contentComponent` is passed an Animated.Value called drawerOpenProgress
// that can be used to do interesting things like a simple parallax drawe
const translateX = Animated.interpolate(drawerOpenProgress, {
inputRange: [0, 1],
outputRange: [-100, 0],
});
return (
<Animated.View style={{ transform: [{ translateX }] }}>
<ScrollView>
<SafeAreaView forceInset={{ top: 'always' }}>
{navigation.state.routes.map(route => (
<DrawerItem
key={route.key}
navigation={descriptors[route.key].navigation}
item={route.routeName}
/>
))}
</SafeAreaView>
</ScrollView>
</Animated.View>
);
};
const DrawerItem = (props: {
navigation: NavigationDrawerProp;
item: string;
}) => {
return (
<TouchableOpacity onPress={() => props.navigation.navigate(props.item)}>
<Themed.Text style={{ padding: 10, fontSize: 18, fontWeight: '600' }}>
{props.item}
</Themed.Text>
</TouchableOpacity>
);
};
function createDrawerExample(options = {}) {
let DrawerExample = createDrawerNavigator(
{
Inbox: {
path: '/',
screen: InboxScreen,
navigationOptions: ({ navigation }) => {
const options: NavigationDrawerOptions = {
drawerLabel: 'Inbox',
drawerLockMode: (
navigation.state.routes[navigation.state.index].params || {}
).drawerLockMode,
drawerIcon: ({ tintColor }) => (
<MaterialIcons
name="move-to-inbox"
size={24}
style={{ color: tintColor }}
/>
),
};
return options;
},
},
Drafts: {
path: '/sent',
screen: DraftsScreen,
navigationOptions: ({ navigation }) => {
const options: NavigationDrawerOptions = {
drawerLabel: 'Drafts',
drawerLockMode: (
navigation.state.routes[navigation.state.index].params || {}
).drawerLockMode,
drawerIcon: ({ tintColor }) => (
<MaterialIcons
name="drafts"
size={24}
style={{ color: tintColor }}
/>
),
};
return options;
},
},
Email: {
path: '/sent',
screen: EmailScreen,
},
},
{
overlayColor: 'rgba(0,0,0,0)',
drawerType: 'back',
drawerBackgroundColor: {
light: '#eee',
dark: '#888',
},
screenContainerStyle: {
shadowColor: '#000000',
shadowOpacity: 0.4,
shadowRadius: 8,
shadowOffset: { height: 0, width: -4 },
},
contentComponent: DrawerContents,
drawerWidth: Dimensions.get('window').width * 0.8,
navigationOptions: {
headerShown: false,
},
contentOptions: {
activeTintColor: '#e91e63',
},
...options,
}
);
return DrawerExample;
}
export default createDrawerExample();