Как определить активный маршрут в DrawerNavigator, чтобы подкрасить активный пункт меню и закрыть ящик - PullRequest
0 голосов
/ 03 октября 2019

Я использую библиотеку React Navigation . Я написал меню собственного ящика и добавил его в конфигурацию contentComponent для моего навигатора. Я не знаю, как определить, какая страница / экран активна из пользовательского меню. Вот мой код для DrawerNavigator :

const DrawerNavigator = createDrawerNavigator({
    "Search Locations": {
        screen: SearchLocationsScreen,
    },
    "About": {
        screen: AboutScreen
    },
    "Favorites": {
        screen: FavoritesScreen
    },
    "Sign In": {
        screen: SignIn
    },
}, {
    contentComponent: props => <CustomDrawerComponent {...props} />
});

Может быть важно отметить, что мой DrawerNavigator вложен в StackNavigator . Я экспортирую свои параметры навигации в отдельный файл, например:

export default (navigation) => {
    const {state} = navigation;
    let navOptions = {};

    if(state.index === 0){
        navOptions.headerRight = (
            <TouchableOpacity>
                <MaterialIcons
                    name="my-location"
                    size={32}
                    color="#fff"
                    style={{paddingRight: 10}} />
            </TouchableOpacity>
        )
    }

        if (state.isDrawerOpen){
        navOptions.headerLeft = (
            <>
                <StatusBar barStyle='light-content'/>
                <TouchableOpacity onPress={() => {
                    navigation.dispatch(DrawerActions.toggleDrawer())
                }}>
                    <Ionicons name="ios-close" style={styles.menuClose} size={38} color={'#fff'}/>
                </TouchableOpacity>
            </>
        )
    } else {
        navOptions.headerLeft = (
            <>
                <StatusBar barStyle='light-content'/>
                <TouchableOpacity onPress={() => {
                    navigation.dispatch(DrawerActions.toggleDrawer())
                }}>
                    <Ionicons name="ios-menu" style={styles.menuOpen} size={32} color={'#fff'}/>
                </TouchableOpacity>
            </>
        )
    }

    return navOptions;
};

, и назначаю эти параметры следующим образом:

const MainStackNavigator = createStackNavigator({
    DrawerNavigator: {
        screen: DrawerNavigator,
        navigationOptions: ({navigation}) => configureDrawerOptions(navigation)
    }
}, {
    defaultNavigationOptions: configureDefaultStackNavOptions
});

Мой пользовательский DrawerMenu выглядит следующим образом:

const DrawerMenu = (props) => {

    // let routes = props.navigation.state.routes;

    const navigateToScreen = (route) => () => {

        const navAction = NavigationActions.navigate({
            routeName: route
        });
        props.navigation.dispatch(navAction);
    };

    return (
        <ScrollView style={styles.root}>

            <View style={styles.rowContainer}>
                <TouchableOpacity onPress={navigateToScreen('Search Locations')}>
                    <View style={styles.row}>
                        <MaterialIcons name='location-searching' style={styles.icon} size={30}/>
                        <Text style={styles.label}>Search Locations</Text>
                    </View>
                </TouchableOpacity>
            </View>

            <View {...props} style={styles.rowContainer}>
                <TouchableOpacity onPress={navigateToScreen('About')}>
                    <View style={styles.row}>
                        <MaterialIcons name='info-outline' style={styles.icon} size={30}/>
                        <Text style={styles.label}>About</Text>
                    </View>
                </TouchableOpacity>
            </View>

            <View {...props} style={styles.rowContainer}>
                <TouchableOpacity onPress={navigateToScreen('Favorites')}>
                    <View style={styles.row}>
                        <MaterialIcons name='favorite-border' style={styles.icon} size={30}/>
                        <Text style={styles.label}>Favorites</Text>
                    </View>
                </TouchableOpacity>
            </View>

            <View {...props} style={styles.rowContainer}>
                <TouchableOpacity onPress={navigateToScreen('Sign In')}>
                    <View style={styles.row}>
                        <Ionicons name='md-log-in' style={styles.icon} size={30}/>
                        <Text style={styles.label}>Sign In</Text>
                    </View>
                </TouchableOpacity>
            </View>

        </ScrollView>
    )
};

Я могу нормально перемещаться, если выбираю другую страницу (если я на странице «Поиск местоположения» и хочу перейти на страницу «Войти», все работает как положено). Однако, если я нахожусь на странице «Поиск местоположений» и нажимаю на пункт меню «Поиск местоположений», я хочу просто закрыть ящик. Я также хочу подкрасить значок / ярлык активной страницы. Моя проблема в том, что внутри файла DrawerMenu.js я не знаю, как определить, на какой странице я сейчас нахожусь, чтобы сделать это.

Правильно ли я это реализую? Я новичок в React Native. Заранее спасибо.

1 Ответ

0 голосов
/ 04 октября 2019

Я нашел способ сделать это, но мне интересно, есть ли лучший / более простой способ, чем я придумал. Я добавил несколько хуков (useEffect, useState) и изменил свою функцию navigateToScreen. Вот мой код. Если это поможет вам, поддержите это. Если вы предоставили лучшее решение, чем мое, я сделаю его приемлемым ответом:

const DrawerMenu = (props) => {

    let activeIndex = props.navigation.state.index;
    let activeRouteName = props.navigation.state.routes[activeIndex].routeName;

    const [activeRoute, setActiveRoute] = useState('');

    useEffect(() => {
        setActiveRoute(activeRouteName);
    }, [activeRouteName, activeRoute]);


    const navigateToScreen = (route) => () => {

        if (activeRouteName === route){
            return props.navigation.closeDrawer();
        }

        const navAction = NavigationActions.navigate({
            routeName: route
        });

        props.navigation.dispatch(navAction);
    };

    return (
        <ScrollView style={styles.root}>

            <View style={styles.rowContainer}>
                <TouchableOpacity
                    onPress={navigateToScreen('Search Locations')}>
                    <View style={styles.row}>
                        <MaterialIcons
                            name='location-searching'
                            style={activeRouteName === "Search Locations" ? styles.activeIcon : styles.icon}
                            size={30}
                        />
                        <Text style={activeRouteName === "Search Locations" ? styles.activeLabel : styles.label}>
                            Search Locations
                        </Text>
                    </View>
                </TouchableOpacity>
            </View>

            <View style={styles.rowContainer}>
                <TouchableOpacity onPress={navigateToScreen('About')}>
                    <View style={styles.row}>
                        <MaterialIcons
                            name='info-outline'
                            style={activeRouteName === "About" ? styles.activeIcon : styles.icon}
                            size={30}
                        />
                        <Text style={activeRouteName === "About" ? styles.activeLabel : styles.label}>
                            About
                        </Text>
                    </View>
                </TouchableOpacity>
            </View>

            <View style={styles.rowContainer}>
                <TouchableOpacity onPress={navigateToScreen('Favorites')}>
                    <View style={styles.row}>
                        <MaterialIcons
                            name='favorite-border'
                            style={activeRouteName === "Favorites" ? styles.activeIcon : styles.icon}
                            size={30}
                        />
                        <Text style={activeRouteName === "Favorites" ? styles.activeLabel : styles.label}>
                            Favorites
                        </Text>
                    </View>
                </TouchableOpacity>
            </View>

            <View style={styles.rowContainer}>
                <TouchableOpacity onPress={navigateToScreen('Profile')}>
                    <View style={styles.row}>
                        <MaterialIcons
                            name='person-outline'
                            style={activeRouteName === "Profile" ? styles.activeIcon : styles.icon}
                            size={32}
                        />
                        <Text style={activeRouteName === "Profile" ? styles.activeLabel : styles.label}>
                            User Profile
                        </Text>
                    </View>
                </TouchableOpacity>
            </View> 
        </ScrollView>
    )
};

const styles = StyleSheet.create({
    root: {
        marginTop: 20,
    },
    row: {
        flexDirection: 'row',
        paddingTop: 16,
        paddingBottom: 16,
        alignItems: 'center',
        justifyContent: 'flex-start'
    },
    icon: {
        marginRight: 25,
        color: '#fff',
        padding: 2,
    },
    activeIcon: {
        marginRight: 25,
        color: colors.accent,
        padding: 2,
    },
    label: {
        fontSize: 20,
        color: '#fff',
    },
    activeLabel: {
        fontSize: 20,
        color: colors.accent,
    },
    rowContainer: {
        marginHorizontal: 16,
    }

});

export default DrawerMenu;
...