php 添加自定义价格和其他数据到WooCommerce购物车

qoefvg9y  于 2023-10-15  发布在  PHP
关注(0)|答案(1)|浏览(111)

我有一个自定义的“添加到购物车”按钮使用Ajax在WooCommerce。我在一个固定价格的产品上测试了它,它的工作原理和预期的一样。然而,产品是可定制的,价格将根据它如何定制而变化。我环顾四周,作为PHP和WP/WC开发的新手,我不确定我看到了什么对我有帮助的东西。
我目前在functions.php文件中为我的活动子主题提供了以下两个函数:

add_action('wp_enqueue_scripts', 'enqueue_custom_add_to_cart_scripts');
function enqueue_custom_add_to_cart_scripts() {
  if(function_exists('is_product') && is_product()) {
    wp_enqueue_script('custom_add_to_cart', get_stylesheet_directory_uri() . '/js/add-to-cart-ajax.js', array('jquery'), '1.0', true);
    wp_localize_script('custom_add_to_cart', 'wc_params_custom', array(
      'ajax_url' => admin_url('admin-ajax.php'), 
      'nonce' => wp_create_nonce('nonce_custom_atc')
    ));
  }
}

add_action('wp_ajax_custom_add_to_cart', 'custom_add_to_cart');
add_action('wp_ajax_nopriv_custom_add_to_cart', 'custom_add_to_cart');
function custom_add_to_cart() {
  $cart_item_key = null;
  
  if (isset($_POST['nonce']) && wp_verify_nonce($_POST['nonce'], 'nonce_custom_atc')
  && isset($_POST['product_id']) && $_POST['product_id'] > 0
  && isset($_POST['quantity']) && $_POST['quantity'] > 0) {
    $cart_item_key = WC()->cart->add_to_cart(
      intval($_POST['product_id']), 
      intval($_POST['quantity']), 
      0, 
      array(), 
      array(
        'height'=>isset($_POST['height']) ? sanitize_text_field($_POST['height']) : '', 
        'width'=>isset($_POST['width']) ? sanitize_text_field($_POST['width']) : '', 
        'totalArea'=>isset($_POST['totalArea']) ? sanitize_text_field($_POST['totalArea']) : '', 
        'cost'=>isset($_POST['cost']) ? sanitize_text_field($_POST['cost']) : '', 
        'numPanels'=>isset($_POST['numPanels']) ? sanitize_text_field($_POST['numPanels']) : '', 
        'enteredHeight'=>isset($_POST['enteredHeight']) ? sanitize_text_field($_POST['enteredHeight']) : '', 
        'enteredWidth'=>isset($_POST['enteredWidth']) ? sanitize_text_field($_POST['enteredWidth']) : ''
      )
    );
  }

  wp_die($cart_item_key ? "Cart item key: {$cart_item_key}" : "Error: not added!");
}

我的JS文件看起来像这样:

jQuery( function($) {
    if (typeof wc_params_custom === 'undefined')
        return false;

    const PANEL_WIDTH = 24;

    let height = $('.height).val();
    let width = $('.width).val();
    let totalArea = ((height / 12) * (width / 12)).toFixed(2);
    let cost = (totalArea * 7.95).toFixed(2);
    let numPanels = width / PANEL_WIDTH;

    $('.custom.add-to-cart').on('click', function(e) {
        e.preventDefault();

        $.ajax({
            type: 'POST',
            url:  wc_params_custom.ajax_url,
            data: {
                'action'        :'custom_add_to_cart',
                'nonce'         : wc_params_custom.nonce,
                'product_id'    : '123',
                'quantity'      : 1,
                'height'        : `${height} in.`,
                'width'         : `${width} in.`,
                'totalArea'     : `${totalArea} sq ft`,
                'cost'          : `$${cost}`, // changes based upon height & width input
                numPanels,
                // These next two fields take their value from the inputs based upon whether it is inches or feet selected. They are not used for any sort of calculation and are only to be passed to the cart for customer info.
                'enteredHeight' : '8 ft',
                'enteredWidth'  : '10 ft'
            },
            success: function (response) {
                $(document.body).trigger("wc_fragment_refresh");
                console.log(response);
            },
            error: function (error) {
                console.log(error); 
            }
        });
    });
});

上面的代码是由一个简单的触发:

<input class="height" type="text" />
<input class="width" type="text" />
<button class="custom add-to-cart">Add to Cart</button>

(EDIT:我添加了如何收集维度输入并处理上述两个代码块的计算。)
显然,我不能把它放在后端的产品页面上,所以它必须来自用户输入。我还需要它在购物车页面上显示,以便客户可以看到正确的总数。我还需要totalAreanumPanelsenteredHeightenteredWidth显示在购物车上。我仍然需要以某种方式将heightwidth发送到生产团队,但对客户隐藏。字段enteredHeightenteredWidth只是为了向客户显示他们在定制器页面上输入的尺寸,以免他们混淆。
有没有人知道如何正确显示这些自定义数据以及生产所需的数据?
谢谢你,谢谢
更新:我发现了这篇文章,并试图让它与我的代码一起工作,但我仍然从wp_die()得到Error: not added!。我想我的方向是对的。Adding a product to cart with custom info and price

iyzzxitl

iyzzxitl1#

下面将显示一些自定义购物车项目数据,设置计算的购物车项目价格,并将保存一些自定义购物车项目数据作为自定义订单项目元数据(在订单和电子邮件通知上显示该元数据)。
HTML代码 (带有一点PHP)

<p class="final-cost">Cost: <span></span></p>
<label for="width"><?php _e('Width'); ?><br>
<input type="text" id="width" name="width" value="" /></label><br>
<label for="height"><?php _e('Height'); ?><br>
<input type="text" id="height" name="height" value="" /></label><br>

<input type="number" name="custom-qty" value="1" style="width:4em;" />
<button class="custom add-to-cart" data-product-id="<?php the_ID(); ?>"><?php _e('Add to Cart'); ?></button>
<input type="hidden" name="base-cost" value="<?php echo get_option('apv_price'); ?>" />
<input type="hidden" name="panel-width" value="<?php echo get_option('apv_panel'); ?>" />

PHP代码:

add_action('wp_enqueue_scripts', 'enqueue_custom_add_to_cart_scripts');
function enqueue_custom_add_to_cart_scripts() {
    if(function_exists('is_product') && is_product()) {
        // Don't use "add-to-cart-ajax.js" file name as it can interact with WooCommerce (or other plugins) existing file names.
        wp_enqueue_script('custom_add_to_cart', get_stylesheet_directory_uri() . '/js/custom-add-to-cart.js', array('jquery'), '1.0', true);
        wp_localize_script('custom_add_to_cart', 'wc_params_custom', array(
            'ajax_url' => admin_url('admin-ajax.php'), 
            'nonce' => wp_create_nonce('nonce_custom_atc')
        ));
    }
}

add_action('wp_ajax_custom_add_to_cart', 'custom_add_to_cart');
add_action('wp_ajax_nopriv_custom_add_to_cart', 'custom_add_to_cart');
function custom_add_to_cart() {
    $cart_item_key = null;
    
    if (isset($_POST['nonce']) && wp_verify_nonce($_POST['nonce'], 'nonce_custom_atc')
    && isset($_POST['product_id']) && $_POST['product_id'] > 0
    && isset($_POST['quantity']) && $_POST['quantity'] > 0 
    && isset($_POST['final_cost']) && $_POST['final_cost'] > 0) {
        $formatted_cost = wc_price(floatval($_POST['final_cost']));
        $cart_item_key = WC()->cart->add_to_cart(
            intval($_POST['product_id']), 
            intval($_POST['quantity']), 
            0, 
            array(), 
            array(
                'height'     => isset($_POST['height']) ?     sanitize_text_field($_POST['height']) : '', 
                'width'      => isset($_POST['width']) ?      sanitize_text_field($_POST['width']) : '', 
                'final_cost' => isset($_POST['final_cost']) ? floatval($_POST['final_cost']) : '', 
                'dim_unit'   => isset($_POST['dim_unit']) ?   esc_attr($_POST['dim_unit']) : '', 
                'num_panels'  => isset($_POST['num_panels']) ? floatval($_POST['num_panels']) : '',
                'total_area'  => isset($_POST['total_area']) ? floatval($_POST['totalArea']) : '', 
                'ent_height' => isset($_POST['ent_height']) ? esc_attr($_POST['ent_height']) : '', 
                'ent_width'  => isset($_POST['ent_width']) ?  esc_attr($_POST['ent_width']) : ''
            )
        );
    }

    wp_die( isset($cart_item_key) && $cart_item_key ? $formatted_cost : "Error: not added!");
}

// Optionally display some custom cart item data in cart and checkout pages
add_filter( 'woocommerce_get_item_data', 'display_custom_cart_item_datat', 90, 2 );
function display_custom_cart_item_datat( $cart_data, $cart_item ) {
    if( isset($cart_item['ent_width']) ){
        $cart_data[] = array(
            'name' => __('Width'),
            'value' => $cart_item['ent_width']
        );
    }
    if( isset($cart_item['ent_height']) ){
        $cart_data[] = array(
            'name' => __('Height'),
            'value' => $cart_item['ent_height']
        );
    }
    if( isset($cart_item['num_panels']) ){
        $cart_data[] = array(
            'name' => __('Panels'),
            'value' => $cart_item['num_panels']
        );
    }
    return $cart_data;
}

// Set custom calculated cart item price
add_action( 'woocommerce_before_calculate_totals', 'set_custom_calculated_item_price', 20, 1 );
function set_custom_calculated_item_price( $cart ) {
    if ( is_admin() && ! defined( 'DOING_AJAX' ) )
        return;

    if ( did_action('woocommerce_before_calculate_totals') > 1 )
        return;

    foreach($cart->get_cart() as $cart_item) {
        if( isset($cart_item['final_cost']) ) {
            $cart_item['data']->set_price( floatval($cart_item['final_cost']) );
        }
    }
}

// Handle mini cart displayed price
add_action( 'woocommerce_cart_item_price', 'display_custom_calculated_item_price', 20, 2 );
function display_custom_calculated_item_price( $price_html, $cart_item ) {
    if( isset($cart_item['final_cost']) ) {
        $args = array( 'price' => floatval($cart_item['final_cost']) ); 

        if ( WC()->cart->display_prices_including_tax() ) {
            $product_price = wc_get_price_including_tax( $cart_item['data'], $args );
        } else {
            $product_price = wc_get_price_excluding_tax( $cart_item['data'], $args );
        }
        return wc_price( $product_price );
    }
    return $price_html;
}

// Save custom cart item data as order item meta and display it on orders and emails
add_action( 'woocommerce_checkout_create_order_line_item', 'save_order_item_custom_meta_data', 10, 4 );
function save_order_item_custom_meta_data( $item, $cart_item_key, $values, $order ) {
    if ( isset($values['custom_data']['share_owner']) && isset($values['custom_data']['dynamic_amount']) ) {
        if( isset($cart_item['ent_width']) ){
            $item->update_meta_data( 'ent_width', $cart_item['ent_width'] ); 
        }
        if( isset($cart_item['ent_height']) ){
            $item->update_meta_data( 'ent_height', $cart_item['ent_height'] ); 
        }
        if( isset($cart_item['num_panels']) ){
            $item->update_meta_data( 'num_panels', $cart_item['num_panels'] );
        }

        // Displayed only in admin order items: meta_key starting with an underscore
        if( isset($cart_item['total_area']) ){
            $item->update_meta_data( '_total_area', $cart_item['total_area'] );
        }
    }
}

// Add readable "meta key" label name replacement for order item meta data
add_filter('woocommerce_order_item_display_meta_key', 'order_item_display_readable_custom_meta_keys', 10, 3 );
function order_item_display_readable_custom_meta_keys( $display_key, $meta, $item ) {
    if( $item->get_type() === 'line_item' ) {
        if( $meta->key === 'ent_width' ) {
            $display_key = __('Width');
        } elseif( $meta->key === 'ent_height' ) {
            $display_key = __('Height');
        } elseif( $meta->key === 'num_panels' ) {
            $display_key = __('Panels');
        } elseif( $meta->key === '_total_area' ) {
            $display_key = __('Total area');
        }
    } 
    return $display_key;
}

代码放在子主题的functions.php文件中(或插件中)。
JavaScript代码:

jQuery( function($) {
    const h = 'input#height',
          w = 'input#width',
          baseCost   = $('input[name=base-cost]').val(),
          panelWidth = $('input[name=panel-width]').val();

    var height, width, dimUnit, totalArea, finalCost, numPanels; // Initializing variables

    // Cost calculation and display
    $(document.body).on('input', h+','+w, function() {
        height    = $(h).val();     
        width     = $(w).val();
        dimUnit   = 'ft'; // => Dimension unit (to be replaced with a dynamic value)

        console.log('height: '+height+' | width: '+width);

        if ( height > 0 && width > 0 && dimUnit !== 'undefined' ) {
            // calculations
            totalArea = ((height / 12) * (width / 12)).toFixed(2),
            finalCost = (totalArea * baseCost).toFixed(2),
            numPanels = width / panelWidth;

            console.log('totalArea: '+totalArea+' | finalCost: '+finalCost+' | numPanels: '+numPanels);
            
            $('.final-cost span').html('$'+finalCost); // Display the formatted cost
        }
    });

    if (typeof wc_params_custom === 'undefined')
        return false;
    
    // Ajax add to cart with custom cart item data
    $('.custom.add-to-cart').on('click', function(e) {
        e.preventDefault();

        if ( height > 0 && width > 0 ) {
            $.ajax({
                type: 'POST',
                url:  wc_params_custom.ajax_url,
                data: {
                    'action'        :'custom_add_to_cart',
                    'nonce'         : wc_params_custom.nonce,
                    'product_id'    : $(this).data('product-id'),
                    'quantity'      : $('input[name=custom-qty]').val(),
                    'height'        : height,
                    'width'         : width,
                    'dim_unit'      : dimUnit,
                    'total_area'    : totalArea+' sq ft',
                    'final_cost'    : finalCost,
                    'num_panels'    : numPanels,
                    'ent_height'    : '8 ft',
                    'ent_width'     : '10 ft',
                },
                success: function (response) {
                    $(document.body).trigger("wc_fragment_refresh");
                    console.log(response); // returns the formatted price if added to cart
                },
                error: function (error) {
                    console.log(error); 
                }
            });
        } else {
            console.log('missing with or height'); 
        }
    });
});

代码放在子主题文件夹中的js/custom-add-to-cart.js文件中。

相关问题