首页 > 解决方案 > 反应-Redux | 产品列表 | 使用 BindActionCreator 添加/删除

问题描述

我正在尝试在 React 中创建一个产品列表,我可以在其中添加和删除产品。

我开始研究如何使用redux框架/平台来做到这一点并做出原生反应

我已经有一个函数productList容器、组件product和组件。cartListcartProduct

我的问题是:

产品:我只能添加产品,不能删除

购物车:反之亦然 + 购物车不会更新购物车物品的状态。

我已经添加了 bindActionCreator,但还不知道如何将它应用到我的 productList 中。

我期望会发生什么? 我正在尝试在同一个容器/组件中的反应存储中添加和删除产品。

我怎样才能做到这一点?我的方法是正确的还是完全错误的?

提前致谢。

ProductActionCreators

export const ADD_TO_CART = 'ADD_TO_CART'
export const REMOVE_FROM_CART = 'REMOVE_FROM_CART'

export function addItemToCart(row) {
    return {
        type:'ADD_TO_CART', 
        payload: row, qty
    }
  }

  export function removeTodo(row) {
    return {
        type:'REMOVE_FROM_CART' , 
        payload: row, qty
    }
  }

产品列表(简体)

     import React from 'react';
     import { Component } from 'react';
      import { 
        View,  
        StyleSheet, 
        Text
     } from 'react-native';
     import Products from '../components/Products';
     import { bindActionCreators} from 'redux';
     import { connect } from 'react-redux';

     import * as ProductActionCreators from '../actions/ProductActionCreators'

     export  class ProductList extends React.Component {
        static navigationOptions = {
            header: null,
        };
        constructor(props) {
            super(props);
            const { rows } = this.props.navigation.state.params;
            const arrays = Object.values( {rows});
            this.state = {
                arrays,
                filteredProducts: arrays,
            };
            const { dispatch } = props
            this.boundActionCreators = bindActionCreators(ProductActionCreators, dispatch)
            console.log(this.boundActionCreators)
        }



          render() {
            return (
                <View style={styles.container} >
                <Text style={styles.title} >
                    {this.state.arrays[0].name}
                </Text>
                    <Products products={this.state.arrays[0].data} onPress=
//Trying to change this to multiple actions
{this.props.addItemToCart}/>    
                </View>
            )
          }
        }

    const qty = 0;

    const mapDispatchToProps = (dispatch) =>{
        //need to add BindActionCreator
        return{
            addItemToCart:(row) => dispatch({
                type:'ADD_TO_CART', payload: row, qty

            }),
            removeItem:(product) => dispatch ({
                type:'REMOVE_FROM_CART' , payload: product, qty
            })  
        }

    }

    export default connect(null, mapDispatchToProps) (ProductList);

产品(简体)

import React, { Component } from "react";
import {
    View,
    Text,
    TouchableOpacity,
    TextInput,
    FlatList,
} from "react-native";
import Icon from "react-native-vector-icons/Ionicons";

class Products extends Component {
    constructor(props) {
        super(props);
        const { products } = this.props;
        this.state = {
            products, 
            filteredProducts: products,
        };
    }
    renderProducts = (products) => {
            return (
                <View key={products.index}>
                    <View> 
                        <Icon name={products.item.icon} color="#DD016B" size={25} />
                    </View>
                    <View>
                        <Text style={styles.name}>
                            {products.item.name}
                        </Text>
                        <Text>
                        € {products.item.price}
                        </Text>
                    </View>
                    <View style={styles.buttonContainer}>
                        <TouchableOpacity onPress={() => this.props.onPress(products.item)} > 
                            <Icon name="ios-add" color="white" size={25} />
                        </TouchableOpacity>

                        <TouchableOpacity onPress={() => this.props.onPress(products.item)} > 
                            <Icon name="ios-remove" color="white" size={25} />
                        </TouchableOpacity>

                    </View>
                </View>
            )
    }

    render() {
        return (
            <View>
                <FlatList
                style={styles.listContainer}
                data={this.state.filteredProducts}
                renderItem={this.renderProducts}
                keyExtractor={(item, index) => index.toString()}
                />
            </View>
        );
    }
}

export default Products;

减速器/cartItems

const cartItems = (state = [], action) => {
    switch (action.type)
    {
        case 'ADD_TO_CART':
            if (state.some(cartItem => cartItem.id === action.payload.id)) {
                // increase qty if item already exists in cart
                return state.map(cartItem => (
                    cartItem.id === action.payload.id ? { ...cartItem, qty: cartItem.qty + 1 } : cartItem

                    ));            
            }
            return [...state, { ...action.payload, qty: 1 }]; 
            // else add the new item to cart            
        case 'REMOVE_FROM_CART':
            return state
                .map(cartItem => (cartItem.id === action.payload.id ? { ...cartItem, qty: cartItem.qty - 1 } : cartItem))
                .filter(cartItem => cartItem.qty > 0);
    }
    return state
} 
export default cartItems 

存储/索引

import {createStore} from 'redux';
import cartItems from '../reducers/carItems';

export default store = createStore(cartItems)

应用程序结构(简化)

Main folder
        ↳
          Containers(folder)
            ↳
             ProductsList.js
             CartList.js

          Components(folder)
            ↳
             Product.js
             cartProduct.js

          Reducers(folder)
            ↳
             carItems.js

          Actions(folder)
            ↳ 
             ProductActionCreators.js

          Navigation(folder)
            ↳
             AppNavigator,js
             MainTabNavigator.js

          Assets(folder for images etc.)
          Store(folder)
            ↳
             index.js
          App.JS
          Data.JS (using static JSON file for this development phase)

标签: reactjsreact-nativereduxreact-redux

解决方案


您有两个不同的操作,addItemToCartremoveItem在 mapDispatchToProps 中定义它们。现在你指定了一个 mapDispatchToProps 参数来连接,dispatch 方法不能作为连接组件的 prop 使用,而是 mapDispatchToProps 返回的方法只可用

其次,您不需要使用bindActionCreators并且绝对不在组件中。MapDispatchToProps可以简单地是一个对象,connect 将在内部使用调度。

第三,您需要将添加和删除操作都传递给子组件。

第四,您可以将多个操作简单地作为道具传递给 Product 组件

你的代码看起来像

ProductActionCreators.js

export const ADD_TO_CART = 'ADD_TO_CART'
export const REMOVE_FROM_CART = 'REMOVE_FROM_CART'

export function addItemToCart(row) {
    return {
        type:'ADD_TO_CART', 
        payload: row
    }
  }

  export function removeItem(item) {
    return {
        type:'REMOVE_FROM_CART' , 
        payload: item
    }
  }

产品列表

  import React from 'react';
  import { Component } from 'react';
  import { 
    View,  
    StyleSheet, 
    Text
 } from 'react-native';
 import Products from '../components/Products';
 import { connect } from 'react-redux';

 import { addItemToCart, removeItem } from '../actions/ProductActionCreators';

 export  class ProductList extends React.Component {
    static navigationOptions = {
        header: null,
    };
    constructor(props) {
        super(props);
        const { rows } = this.props.navigation.state.params;
        const arrays = Object.values( {rows});
        this.state = {
            arrays,
            filteredProducts: arrays,
        };
    }



      render() {
        return (
            <View style={styles.container} >
            <Text style={styles.title} >
                {this.state.arrays[0].name}
            </Text>
                <Products products={this.state.arrays[0].data} addItemToCart={this.props.addItemToCart} removeItem={this.props.removeItem}/>    
            </View>
        )
      }
    }


const mapDispatchToProps =  {
     addItemToCart,
     removeItem
}

export default connect(null, mapDispatchToProps) (ProductList);

产品

import React, { Component } from "react";
import {
    View,
    Text,
    TouchableOpacity,
    TextInput,
    FlatList,
} from "react-native";
import Icon from "react-native-vector-icons/Ionicons";

class Products extends Component {
    constructor(props) {
        super(props);
        const { products } = this.props;
        this.state = {
            products, 
            filteredProducts: products,
        };
    }
    renderProducts = (products) => {
            return (
                <View key={products.index}>
                    <View> 
                        <Icon name={products.item.icon} color="#DD016B" size={25} />
                    </View>
                    <View>
                        <Text style={styles.name}>
                            {products.item.name}
                        </Text>
                        <Text>
                        € {products.item.price}
                        </Text>
                    </View>
                    <View style={styles.buttonContainer}>
                        <TouchableOpacity onPress={() => this.props.addItemToCart(products.item)} > 
                            <Icon name="ios-add" color="white" size={25} />
                        </TouchableOpacity>

                        <TouchableOpacity onPress={() => this.props.removeItem(products.item)} > 
                            <Icon name="ios-remove" color="white" size={25} />
                        </TouchableOpacity>

                    </View>
                </View>
            )
    }

    render() {
        return (
            <View>
                <FlatList
                style={styles.listContainer}
                data={this.state.filteredProducts}
                renderItem={this.renderProducts}
                keyExtractor={(item, index) => index.toString()}
                />
            </View>
        );
    }
}

export default Products;

推荐阅读