phpwordpresswoocommercecustom-fields

Cart item price calculation, based on different basic prices in Woocommerce


In Woocommerce, I use custom fields to calculate the price of a product, based on this code - Cart item price calculation, based on the chosen “days” custom field in Woocommerce. Many thanks for the help of LoicTheAztec.

// HERE your rental days settings
function get_rental_days_options() {
    return array(
            '2' => __("2 Days", "woocommerce"),
            '4' => __("4 Days", "woocommerce"),
    );
}

// Add a custom field before single add to cart
add_action('woocommerce_before_add_to_cart_button', 'display_single_product_custom_fields', 5);

function display_single_product_custom_fields() {
    // Get the rental days data options
    $options = array('' => __("Choosen period", "woocommerce")) + get_rental_days_options();

    echo '<div class="custom-text text"> < h3 > '.__("Rental", "woocommerce").
    ' < /h3> < label > '.__("Start Date", "woocommerce").
    ': < /label> < input type = "date"
    name = "rental_date"
    value = ""
    class = "rental_date" / >
            < label > Period: < /label> < select class = "rental-days"
    id = "rental-days"
    name = "rental_days" > ';

    foreach($options as $key => $option) {
            echo '<option value="'.$key.
            '">'.$option.
            '</option>';
    }

    echo '</select> < /div>';
}

// Get custom field value, calculate new item price, save it as custom cart item data
add_filter('woocommerce_add_cart_item_data', 'add_custom_field_data', 20, 3);

function add_custom_field_data($cart_item_data, $product_id, $variation_id) {
    // HERE set the percentage rate to be applied to get the new price
    $percentage = 2;

    if (isset($_POST['rental_date']) && !empty($_POST['rental_date'])) {
            $cart_item_data['custom_data']['start_date'] = $_POST['rental_date'];
    }

    if (isset($_POST['rental_days']) && !empty($_POST['rental_days'])) {
            $cart_item_data['custom_data']['rental_days'] = esc_attr($_POST['rental_days']);

            $_product_id = $variation_id > 0 ? $variation_id : $product_id;

            $product = wc_get_product($_product_id); // The WC_Product Object
            $base_price = (float) $product - > get_regular_price(); // Get the product regular price

            $price_rate = $cart_item_data['custom_data']['rental_days'] * $percentage / 100;

            $cart_item_data['custom_data']['base_price'] = $base_price;
            $cart_item_data['custom_data']['new_price'] = $base_price * $price_rate;
    }

    // Make each cart item unique
    if (isset($cart_item_data['custom_data']['rental_days']) || isset($cart_item_data['custom_data']['start_date'])) {
            $cart_item_data['custom_data']['unique_key'] = md5(microtime().rand());
    }

    return $cart_item_data;
}

// Set the new calculated cart item price
add_action('woocommerce_before_calculate_totals', 'extra_price_add_custom_price', 20, 1);

function extra_price_add_custom_price($cart) {
    if (is_admin() && !defined('DOING_AJAX'))
            return;

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

// Display cart item custom price details
add_filter('woocommerce_cart_item_price', 'display_cart_items_custom_price_details', 20, 3);

function display_cart_items_custom_price_details($product_price, $cart_item, $cart_item_key) {
    if (isset($cart_item['custom_data']['base_price'])) {
            $product = $cart_item['data'];
            $base_price = $cart_item['custom_data']['base_price'];
            $product_price = wc_price(wc_get_price_to_display($product, array('price' => $base_price))).
            '<br>';

            if (isset($cart_item['custom_data']['rental_days'])) {
                    $rental_days = get_rental_days_options();
                    $product_price. = $rental_days[$cart_item['custom_data']['rental_days']];
            }
    }
    return $product_price;
}

// Display in cart item the selected date
add_filter('woocommerce_get_item_data', 'display_custom_item_data', 10, 2);

function display_custom_item_data($cart_item_data, $cart_item) {
    if (isset($cart_item['custom_data']['start_date'])) {
            $cart_item_data[] = array(
                    'name' => __("Rental start date", "woocommerce"),
                    'value' => date('d.m.Y', strtotime($cart_item['custom_data']['start_date'])),
            );
    }

    if (isset($cart_item['custom_data']['rental_days'])) {
            $rental_days = get_rental_days_options();
            $cart_item_data[] = array(
                    'name' => __("Rental period", "woocommerce"),
                    'value' => $rental_days[$cart_item['custom_data']['rental_days']],
            );
    }

    return $cart_item_data;
}

// Save and display custom field in orders and email notifications (everywhere)
add_action('woocommerce_checkout_create_order_line_item', 'custom_fields_update_order_item_meta', 20, 4);

function custom_fields_update_order_item_meta($item, $cart_item_key, $values, $order) {
    if (isset($values['custom_data']['date'])) {
            $date = date('d.m.Y', strtotime($values['custom_data']['date']));
            $item - > update_meta_data(__('Start date', 'woocommerce'), $date);
    }
    if (isset($values['custom_data']['rental_days'])) {
            $rental_days = get_rental_days_options();
            $item - > update_meta_data(__('Rental period', 'woocommerce'), $rental_days[$values['custom_data']['rental_days']]);
    }
}

I did not expect a new condition for calculating the user's price (( This is the last.

If the user buys a product with a base price greater than 6,660, then the standard calculation follows from the code shown above.

If the user buys a product whose base price is less than 6,660, then the new price is 999 per day. Next, the user selects the rental period. If he chooses 2 days, then 999 * 2. If he chooses 4 days, then 999 * 4.

As I understand it, I need to add the conditions if base_price> 6600 (hereafter goes the standard calculation code) and if base_price <6600 (then count based on the user's choice of 999 * 2 or 999 * 4).

I need to make a conclusion of price ['new_price'] next to the base price on the product single page.

But I do not quite understand how to implement it correctly.

I shall be very glad of your help!


UPDATE: I put a clean version of Wordpress without plugins. Activated the theme of the Storefront. Added demo data.

In the file, functions.php added the code, along with your edits:

// HERE your rental days settings
function get_rental_days_options() {
return array(
        '2' => __("2 Days", "woocommerce"),
        '4' => __("4 Days", "woocommerce"),
);
}

// Add a custom field before single add to cart
add_action('woocommerce_before_add_to_cart_button', 'display_single_product_custom_fields', 5);

function display_single_product_custom_fields() {
// Get the rental days data options
$options = array('' => __("Choosen period", "woocommerce")) + get_rental_days_options();

echo '<div class="custom-text text">
<h3>'.__("Rental", "woocommerce").'</h3>
<label>'.__("Start Date", "woocommerce").': </label>
<input type="date" name="rental_date" value="" class="rental_date" />
<label>Period: </label>
<select class="rental-days" id="rental-days" name="rental_days">';

foreach($options as $key => $option) {
        echo '<option value="'.$key.'">'.$option.'</option>';
}
echo '</select></div>';
}

// Get custom field value, calculate new item price, save it as custom cart item data
add_filter('woocommerce_add_cart_item_data', 'add_custom_field_data', 20, 3);

function add_custom_field_data($cart_item_data, $product_id, $variation_id) {
// HERE set the percentage rate to be applied to get the new price
$percentage = 2;

if (isset($_POST['rental_date']) && !empty($_POST['rental_date'])) {
        $cart_item_data['custom_data']['start_date'] = $_POST['rental_date'];
}

if (isset($_POST['rental_days']) && !empty($_POST['rental_days'])) {
        $cart_item_data['custom_data']['rental_days'] = esc_attr($_POST['rental_days']);

        $_product_id = $variation_id > 0 ? $variation_id : $product_id;

        $product = wc_get_product($_product_id); // The WC_Product Object
        $base_price = (float) $product->get_regular_price(); // Get the product regular price
        $rental_days = $cart_item_data['custom_data']['rental_days'];

        // Below is the old/original code for your reference.
        /*$price_rate = $cart_item_data['custom_data']['rental_days'] * $percentage / 100;

        $cart_item_data['custom_data']['base_price'] = $base_price;
        $cart_item_data['custom_data']['new_price'] = $base_price * $price_rate;*/
        // Above is the old/original code for your reference. The new code is what follows (below):

        // Set base price.
        $cart_item_data['custom_data']['base_price'] = $base_price;

        // Set new price.
        if ( $base_price > 6660 ) {
            $price_rate = $rental_days * $percentage / 100;
            $cart_item_data['custom_data']['new_price'] = $base_price * $price_rate;
        } else {
            $cart_item_data['custom_data']['new_price'] = 999 * $rental_days;
        }
}

// Make each cart item unique
if (isset($cart_item_data['custom_data']['rental_days']) || isset($cart_item_data['custom_data']['start_date'])) {
        $cart_item_data['custom_data']['unique_key'] = md5(microtime().rand());
}

return $cart_item_data;
}

// Set the new calculated cart item price
add_action('woocommerce_before_calculate_totals', 'extra_price_add_custom_price', 20, 1);

function extra_price_add_custom_price($cart) {
if (is_admin() && !defined('DOING_AJAX'))
        return;

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

// Display cart item custom price details
add_filter('woocommerce_cart_item_price', 'display_cart_items_custom_price_details', 20, 3);

function display_cart_items_custom_price_details($product_price, $cart_item, $cart_item_key) {
if (isset($cart_item['custom_data']['base_price'])) {
        $product = $cart_item['data'];
        $base_price = $cart_item['custom_data']['base_price'];
        $product_price = wc_price(wc_get_price_to_display($product, array('price' => $base_price))).
        '<br>';

        if (isset($cart_item['custom_data']['rental_days'])) {
                $rental_days = get_rental_days_options();
                $product_price .= $rental_days[$cart_item['custom_data']['rental_days']];
        }
}
return $product_price;
}

// Display in cart item the selected date
add_filter('woocommerce_get_item_data', 'display_custom_item_data', 10, 2);

function display_custom_item_data($cart_item_data, $cart_item) {
if (isset($cart_item['custom_data']['start_date'])) {
        $cart_item_data[] = array(
                'name' => __("Rental start date", "woocommerce"),
                'value' => date('d.m.Y', strtotime($cart_item['custom_data']['start_date'])),
        );
}

if (isset($cart_item['custom_data']['rental_days'])) {
        $rental_days = get_rental_days_options();
        $cart_item_data[] = array(
                'name' => __("Rental period", "woocommerce"),
                'value' => $rental_days[$cart_item['custom_data']['rental_days']],
        );
}

return $cart_item_data;
}

// Save and display custom field in orders and email notifications (everywhere)
add_action('woocommerce_checkout_create_order_line_item', 'custom_fields_update_order_item_meta', 20, 4);

function custom_fields_update_order_item_meta($item, $cart_item_key, $values, $order) {
if (isset($values['custom_data']['date'])) {
        $date = date('d.m.Y', strtotime($values['custom_data']['date']));
        $item->update_meta_data(__('Start date', 'woocommerce'), $date);
}
if (isset($values['custom_data']['rental_days'])) {
        $rental_days = get_rental_days_options();
        $item->update_meta_data(__('Rental period', 'woocommerce'), $rental_days[$values['custom_data']['rental_days']]);
}
}

Here are the screenshots:

enter image description here

enter image description here

enter image description here

Small problem with the calculation of the price.

If the base price is > 6660, it does not count and does not appear in the cart. If the base price is < 6660, then everything works correctly.

I need both conditions to be calculated and displayed in the cart.

If the customer buys both products, and one of them does not work the price, it's bad ((


UPDATE 2: I updated the code. Here is a screenshot of the cart:

enter image description here

Instead of $680, there should be $5100.

Instead of $1360, there should be $10200.

17000 / 100 * 15 = 2550

2550 * 2 = 5100

2550 * 4 = 10200

And how to show a new price on a single product page, next to base price?


Solution

  • (Updated/new answer)

    STEP 1: Above this code:

    // HERE your rental days settings
    function get_rental_days_options() {
    ... code removed, but nothing changed here.
    }
    

    Add this code, which calculates the new price — and based on the new calculation:

    // Calculate new price based on rental period/days.
    function calc_rental_new_price( $rental_days, $base_price ) {
        $percentage = 15;
    
        if ( $base_price > 6660 ) {
            $price_rate = $rental_days * $percentage / 100;
            return $base_price * $price_rate;
        } else {
            return 999 * $rental_days;
        }
    }
    

    STEP 2: Use this new add_custom_field_data() code — it's re-indented for clarity:

    function add_custom_field_data($cart_item_data, $product_id, $variation_id) {
        if (isset($_POST['rental_date']) && !empty($_POST['rental_date'])) {
            $cart_item_data['custom_data']['start_date'] = $_POST['rental_date'];
        }
    
        if (isset($_POST['rental_days']) && !empty($_POST['rental_days'])) {
            $cart_item_data['custom_data']['rental_days'] = esc_attr($_POST['rental_days']);
    
            $_product_id = $variation_id > 0 ? $variation_id : $product_id;
    
            $product = wc_get_product($_product_id); // The WC_Product Object
            $base_price = (float) $product->get_regular_price();
    
            // Set base price.
            $cart_item_data['custom_data']['base_price'] = $base_price;
    
            // Set new price.
            $cart_item_data['custom_data']['new_price'] = calc_rental_new_price(
                $cart_item_data['custom_data']['rental_days'], $base_price
            );
        }
    
        // Make each cart item unique
        if (isset($cart_item_data['custom_data']['rental_days']) || isset($cart_item_data['custom_data']['start_date'])) {
            $cart_item_data['custom_data']['unique_key'] = md5(microtime().rand());
        }
    
        return $cart_item_data;
    }
    

    STEP 3: After this code:

    function custom_fields_update_order_item_meta($item, $cart_item_key, $values, $order) {
    ... code removed, but nothing changed here.
    }
    

    Add this code, which shows a new price on a single product page, next to base price:

    // Adds the new price next to the base price. That new price is for 1 day rental period.
    add_filter( 'woocommerce_get_price_html', 'add_new_price_html', 10, 2 );
    
    function add_new_price_html( $html, $product ) {
        // Check if we're on a single Product page, the shop page, or a products archive page.
        if ( ! is_product() && ! is_shop() && ! is_product_taxonomy() ) {
            return $html;
        }
    
        $base_price = (float) ( 'variable' === $product->get_type() ?
            $product->get_variation_regular_price() : $product->get_regular_price() );
        $new_price = calc_rental_new_price( 1, $base_price );
    
        // If it's a Variable product, appends "From " to the new price.
        $text_from = ( 'variable' === $product->get_type() ) ? 'From ' : '';
    
        return $html . ' <span class="new-price">' . $text_from . wc_price( $new_price ) . ' / day</span>';
    }
    

    UPDATED Sep 04 2018