React Navigation 5:未定义不是对象

问题描述

在我的UserProductsScreen.js文件中,我传递了一个参数ID,以便可以在导航中找到它:

const UserProductsScreen = props => {
    const userProducts = useSelector(state => state.products.userProducts);
    
    // Navigate to EditProduct
    const editProductHandler = (id) => {
        props.navigation.navigate('EditProduct',{ productId: id })
    };  

然后,在我的navigator.js文件中,我尝试检查它是否存在,以便我可以决定显示“添加产品”或“编辑产品”的标题标题:

  <AdminNavigator.Screen
                name="EditProduct"
                component={EditProductScreen}
                options={({ route }) => {
                    const productId = route.params.productId;
                    return {
                        title: productId ? "Edit Product" : "Add Product"
                    };
                }}
            />

我测试了它是否收到了productId,但是,当我单击加号图标TypeError: undefined is not an object (evaluating 'route.params.productId')

时,我收到了此错误

我不确定这是什么意思,但我只是在检查它是否存在,以便可以动态更改标题。

记录下来,这是我完整的navigation

import React from 'react';
import { NavigationContainer } from '@react-navigation/native';
import { createStackNavigator } from '@react-navigation/stack';
import { createDrawerNavigator } from '@react-navigation/drawer';
import { Platform } from 'react-native';

import { Ionicons } from '@expo/vector-icons';

import ProductsOverviewScreen from '../screens/shop/ProductsOverviewScreen';
import ProductDetailScreen from '../screens/shop/ProductDetailScreen';
import CartScreen from '../screens/shop/CartScreen';
import OrdersScreen from '../screens/shop/OrdersScreen';
import UserProductsScreen from '../screens/user/UserProductsScreen';
import EditProductScreen from '../screens/user/EditProductScreen';

import Colors from '../constants/Colors';
import { HeaderButtons,Item } from 'react-navigation-header-buttons';
import HeaderButton from '../components/UI/HeaderButton';

const ProductsNavigator = createStackNavigator();

const ProductsNavMenu = () => {
    return (
        <ProductsNavigator.Navigator
            screenOptions={{
                headerStyle: {
                    backgroundColor: Platform.OS === 'android' ? Colors.primary : ''
                },headerTintColor: Platform.OS === 'android' ? 'white' : Colors.primary,headerTitleStyle: {
                    fontSize: 17,fontFamily: 'poppins-bold'
                },headerBackTitleStyle: {
                    fontFamily: 'poppins-regular'
                }
            }}>
            <ProductsNavigator.Screen
                name="Products"
                component={ProductsOverviewScreen}
                options={({ navigation }) => {
                    return {
                        headerRight: () => (
                            <HeaderButtons HeaderButtonComponent={HeaderButton}>
                                <Item
                                    title="Cart"
                                    iconName={Platform.OS === 'android' ? 'md-cart' : 'ios-cart'}
                                    onPress={() => navigation.navigate('Cart')}
                                />
                            </HeaderButtons>
                        ),headerLeft: () => (
                            <HeaderButtons HeaderButtonComponent={HeaderButton}>
                                <Item
                                    title='Menu'
                                    iconName={Platform.OS === 'android' ? 'md-menu' : 'ios-menu'}
                                    onPress={() => {
                                        navigation.toggleDrawer();
                                    }}
                                />
                            </HeaderButtons>
                        )
                    };
                }}
            />
            <ProductsNavigator.Screen
                name="ProductDetail"
                component={ProductDetailScreen}
                options={({ route }) => {
                    const productTitle = route.params.productTitle;
                    return {
                        title: productTitle
                    };
                }}
            />
            <ProductsNavigator.Screen
                name="Cart"
                component={CartScreen}
            />
        </ProductsNavigator.Navigator>
    );
};


// Create a separate stack navigation 
// for orders 
const OrdersNavigator = createStackNavigator();

const OrdersNavMenu = () => {
    return (
        <OrdersNavigator.Navigator
            mode="modal"
            screenOptions={{
                headerStyle: {
                    backgroundColor: Platform.OS === 'android' ? Colors.primary : ''
                },headerBackTitleStyle: {
                    fontFamily: 'poppins-regular'
                }
            }}
        >
            <OrdersNavigator.Screen
                name="Orders"
                component={OrdersScreen}
                options={({ navigation }) => {
                    return {
                        headerLeft: () => (
                            <HeaderButtons HeaderButtonComponent={HeaderButton}>
                                <Item
                                    title='Menu'
                                    iconName={Platform.OS === 'android' ? 'md-menu' : 'ios-menu'}
                                    onPress={() => {
                                        navigation.toggleDrawer();
                                    }}
                                />
                            </HeaderButtons>
                        )
                    };
                }}

            />
        </OrdersNavigator.Navigator>
    );
};





// Create a separate stack navigation 
// for userProductsScreen
const AdminNavigator = createStackNavigator();

const AdminNavMenu = () => {
    return (
        <AdminNavigator.Navigator
            mode="modal"
            screenOptions={{
                headerStyle: {
                    backgroundColor: Platform.OS === 'android' ? Colors.primary : ''
                },headerBackTitleStyle: {
                    fontFamily: 'poppins-regular'
                }
            }}
        >
            <AdminNavigator.Screen
                name="UserProducts"
                component={UserProductsScreen}
                options={({ navigation }) => {
                    return {
                        title: "User Products",headerLeft: () => (
                            <HeaderButtons HeaderButtonComponent={HeaderButton}>
                                <Item
                                    title='User Products'
                                    iconName={Platform.OS === 'android' ? 'md-list' : 'ios-list'}
                                    onPress={() => {
                                        navigation.toggleDrawer();
                                    }}
                                />
                            </HeaderButtons>
                        ),headerRight: () => (
                            <HeaderButtons HeaderButtonComponent={HeaderButton}>
                                <Item
                                    title='Add'
                                    iconName={Platform.OS === 'android' ? 'md-create' : 'ios-create'}
                                    onPress={() => {
                                        navigation.navigate('EditProduct');
                                    }}
                                />
                            </HeaderButtons>
                        )
                    };
                }}

            />

            <AdminNavigator.Screen
                name="EditProduct"
                component={EditProductScreen}
                options={({ route }) => {
                    const productId = route.params.productId;
                    return {
                        title: productId ? "Edit Product" : "Add Product"
                    };
                }}
            />
        </AdminNavigator.Navigator>
    );
};







const ShopNavigator = createDrawerNavigator();

const ShopNavMenu = () => {
    return (
        <NavigationContainer>
            <ShopNavigator.Navigator>
                <ShopNavigator.Screen
                    name="Products"
                    component={ProductsNavMenu}
                    options={{
                        drawerIcon: ({ focused,size }) => (
                            <Ionicons
                                name={Platform.OS === 'android' ? 'md-cart' : 'ios-cart'}
                                size={23}
                                color={focused ? '#7cc' : '#ccc'}
                            />
                        )
                    }}
                />
                <ShopNavigator.Screen
                    name="Orders"
                    component={OrdersNavMenu}
                    options={{
                        drawerIcon: ({ focused,size }) => (
                            <Ionicons
                                name={Platform.OS === 'android' ? 'md-list' : 'ios-list'}
                                size={23}
                                color={focused ? '#7cc' : '#ccc'}
                            />
                        )
                    }}
                />
                <ShopNavigator.Screen
                    name="Admin"
                    component={AdminNavMenu}
                    options={{
                        drawerIcon: ({ focused,size }) => (
                            <Ionicons
                                name={Platform.OS === 'android' ? 'md-create' : 'ios-create'}
                                size={23}
                                color={focused ? '#7cc' : '#ccc'}
                            />
                        )
                    }}
                />
            </ShopNavigator.Navigator>
        </NavigationContainer>
    );
};

export default ShopNavMenu;

更新:因此,我尝试在发生魔法的setParams上使用UserProductsScreen。我使用useEffect将初始productId设置为空白:

import React,{ useEffect } from 'react';
import { CommonActions } from '@react-navigation/native';

const UserProductsScreen = props => {
    const userProducts = useSelector(state => state.products.userProducts);
    
    // Navigate to EditProduct
    const editProductHandler = (id) => {
        props.navigation.navigate('EditProduct',{ productId: id })
    };  

        // Set initial productId params not until the editProductHandler is triggerred
        useEffect(() => {
            props.navigation.dispatch(CommonActions.setParams({ productId: '' }));
        },[]); 
    

    // dispatch
    const dispatch = useDispatch();

然后在我的navigation上,我尝试通过headerRight捕获productId:

    <AdminNavigator.Screen
    name="UserProducts"
    component={UserProductsScreen}
    options={({ navigation,route }) => {
        const productId = route.params.productId;
        return {
            title: "User Products",headerLeft: () => (
                <HeaderButtons HeaderButtonComponent={HeaderButton}>
                    <Item
                        title='User Products'
                        iconName={Platform.OS === 'android' ? 'md-list' : 'ios-list'}
                        onPress={() => {
                            navigation.toggleDrawer();
                        }}
                    />
                </HeaderButtons>
            ),headerRight: () => (
                <HeaderButtons HeaderButtonComponent={HeaderButton}>
                    <Item
                        title='Add'
                        iconName={Platform.OS === 'android' ? 'md-create' : 'ios-create'}
                        onPress={() => {
                            navigation.navigate('EditProduct');
                        }}
                    />
                </HeaderButtons>
            )
        };
    }}

/>

但这会影响常规导航上的ID。现在,单击“编辑”按钮时,还会显示上面的错误。

解决方法

问题在这里

     <HeaderButtons HeaderButtonComponent={HeaderButton}>
                        <Item
                            title='Add'
                            iconName={Platform.OS === 'android' ? 'md-create' : 'ios-create'}
                            onPress={() => {
                                navigation.navigate('EditProduct');
                            }}
                        />
                    </HeaderButtons>

从标题右侧的按钮导航时,没有传递任何参数。

如果此时您无法访问路由参数,则在导航到页面后,必须使用setParams设置参数。 你可以参考文档 https://reactnavigation.org/docs/navigation-prop#setparams

更新: 因此,当您导航不传递任何内容时,route.params是未定义的 当您访问未定义的属性时,它将引发错误

验证此方法的方法是使用如下所示的可选链接

const productId = route.params?.productId;

仅当参数可用时,才会设置该值。

,

我在加载第一个屏幕时遇到了这个问题。 参数未定义,我通过使用 initialParams 选项为我的第一个屏幕设置参数来修复它。

<Stack.Screen
  component={LCWebView}
  options={{
    title: 'Awesome app',myopt: 'test',}}
  initialParams={{ url: url_goes_here }}
/>

相关问答

依赖报错 idea导入项目后依赖报错,解决方案:https://blog....
错误1:代码生成器依赖和mybatis依赖冲突 启动项目时报错如下...
错误1:gradle项目控制台输出为乱码 # 解决方案:https://bl...
错误还原:在查询的过程中,传入的workType为0时,该条件不起...
报错如下,gcc版本太低 ^ server.c:5346:31: 错误:‘struct...