Woocommerce中基于定制字段的动态价格计算 [英] Dynamic price calculation based on custom field in Woocommerce

查看:62
本文介绍了Woocommerce中基于定制字段的动态价格计算的处理方法,对大家解决问题具有一定的参考价值,需要的朋友们下面随着小编来一起学习吧!

问题描述

在Woocommerce的每个产品页面上都有一个文本框,允许用户输入自己的自定义文本。然后将此文本应用于产品,并根据每个字母向客户收费。

In Woocommerce on each product page there is a Text Box, which allows users to enter their own custom text. This text is then applied to the product, with the customer being charged on a per letter basis.

除了数学逻辑之外,我还设法使所有工作正常进行。当访客输入x个字母时,当前的Math逻辑会正确计算产品价格+自定义字母的费用,并将此总和输出到Basket Widget。

I have managed to get everything working, apart from the Maths Logic. When a visitor enters x amount of letters, the current Maths logic correctly calculates the Product Price + Cost of Custom Letters and outputs this sum to the Basket Widget.

我遇到问题的地方是,当访客进入购物篮页面时,自定义字体的成本增加了一倍。

Where I am running into problems, is when the visitor goes to the Basket Page, the Cost of the Custom Lettering is doubled.

我似乎无法弄清为什么。

I just cannot seem to work out why this is.

在产品信息中心中创建文本字段:

<?php

/*Create Text Field in Product Dashboard*/

function add_text_field_product_dashboard(){

   global $post;

   $input_checkbox = get_post_meta( $post->ID, '_custom_text_option', true );
   if( empty( $input_checkbox ) || $input_checkbox == 'no' ) $input_checkbox = '';

    echo '<div class="product_custom_field">';

    /*Product Checkbox Field*/
    woocommerce_wp_checkbox(
        array(
            'id'        => '_custom_text_option',
            'desc'      =>  __('set custom custom text field', 'woocommerce'),
            'label'     => __('Display custom custom text field', 'woocommerce'),
            'desc_tip'  => 'true',
            'value'     => $input_checkbox
        )
    );
    /*Minimum Letter Text Box*/
    woocommerce_wp_text_input(
        array(
            'id'        => '_minimum_custom_text_option',
            'name'      => '_minimum_custom_text_option',
            'desc'      =>  __('set custom minimum Lettering text field', 'woocommerce'),
            'label'     => __('Minimum Letters', 'woocommerce'),
            'desc_tip'  => 'true'
        )
    );
    /*Maximum Letter Text Box*/
    woocommerce_wp_text_input(
        array(
            'id'        => '_maximum_custom_text_option',
            'desc'      =>  __('set custom maximum Lettering text field', 'woocommerce'),
            'label'     => __('Maximum Letters', 'woocommerce'),
            'desc_tip'  => 'true'
        )
    );
    /*Cost Per Letter Pricing*/
    woocommerce_wp_text_input(
        array(
            'id'        => '_pricing_custom_text_option',
            'desc'      =>  __('set custom pricing Lettering text field', 'woocommerce'),
            'label'     => __('Cost Per Letter', 'woocommerce'),
            'desc_tip'  => 'true'
        )
    );

    echo '</div>';
}
add_action('woocommerce_product_options_advanced', 'add_text_field_product_dashboard');
?>

保存文本字段条目:

<?php
/*Save Inputted Entries, in the Product Dashboard Text Fields.*/

/*Checkbox Field*/
 function woocommerce_product_custom_fields_save($post_id){               
    $_custom_text_option = isset( $_POST['_custom_text_option'] ) ? 'yes' : '';
    update_post_meta( $post_id, '_custom_text_option', $_custom_text_option );       
 }
add_action('woocommerce_process_product_meta', 'woocommerce_product_custom_fields_save');

/*Save Minimum Letters*/
function woocommerce_product_custom_fields_save1($post_id){
    if ( isset( $_POST['_minimum_custom_text_option'] ) )
        update_post_meta($post_id, '_minimum_custom_text_option', esc_attr( $_POST['_minimum_custom_text_option'] ));
}
add_action( 'woocommerce_process_product_meta', 'woocommerce_product_custom_fields_save1' );

/*Save Maximum Letters*/
function woocommerce_product_custom_fields_save2($post_id){
    if ( isset( $_POST['_maximum_custom_text_option'] ) )
        update_post_meta($post_id, '_maximum_custom_text_option', esc_attr( $_POST['_maximum_custom_text_option'] ));
}
add_action( 'woocommerce_process_product_meta', 'woocommerce_product_custom_fields_save2' );

/*Save Cost Per Letter*/
function woocommerce_product_custom_fields_save3($post_id){
    if ( isset( $_POST['_pricing_custom_text_option'] ) )
        update_post_meta($post_id, '_pricing_custom_text_option', esc_attr( $_POST['_pricing_custom_text_option'] ));
}
add_action( 'woocommerce_process_product_meta', 'woocommerce_product_custom_fields_save3' );
?>

将自定义文本输出到产品页面:

<?php
/*Output Custom Text Field to Product Page*/

function add_custom_text_field() {
    global $post;

    // Get the checkbox value
    $custom_option = get_post_meta( $post->ID, '_custom_text_option', true );

    // If is single product page and have the "custom text option" enabled we display the field
    if ( is_product() && ! empty($custom_option) ) {
?>      
        <div> 
            <label class="product-custom-text-label" for="custom_text"><?php _e( 'Custom Letters:', 'woocommerce'); ?><br>
                <input style="min-width:220px" type="text" class="product-counter" name="custom_text" placeholder="<?php _e( 'Enter Your Custom Letters ...', 'woocommerce'); ?>" minlength="<?php global $post; echo get_post_meta($post->ID,'_minimum_custom_text_option',true);?>" maxlength="<?php global $post; echo get_post_meta($post->ID,'_maximum_custom_text_option',true);?>" />
            </label>
        </div><br>
<?php
    }
}
add_action( 'woocommerce_before_add_to_cart_button', 'add_custom_text_field', 0 );
?>
?>

附加到购物车:

<?php  
/*Append to Cart once Shoper adds to Cart*/

function save_custom_text( $cart_item_data, $product_id ) {
    if( isset( $_POST['custom_text'] ) && !empty( $_POST['custom_text'] ) ) {
        $cart_item_data[ "custom_text" ] = esc_attr( $_POST['custom_text'] );     
    }

    return $cart_item_data;
}
add_filter( 'woocommerce_add_cart_item_data', 'save_custom_text', 99, 2 );
?>

产品总和加输入的每封信的成本:

<?php 
/*Sum of Product Price plus Custom Text*/

function calculate_custom_text_fee( $cart_object ) {  
    foreach ( $cart_object->get_cart() as $cart_item ) {
        // Checking that we got the custom text in cart object
        if( ! empty( $cart_item["custom_text"] ) ) {

            // Quantity of characters entered into Custom Text Field:
            $lenght = strlen( $cart_item["custom_text"] );

            // get the custom pricing for this product
            $pricing_custom = get_post_meta( $cart_item['product_id'], '_pricing_custom_text_option', true );

            // Characters Entered Multiplied by Cost of Each Letter:
            $custom_text_fee = $lenght * $pricing_custom; 

            // get product price
            $price = floatval( $cart_item['data']->get_price() );

            // set new price
            $cart_item['data']->set_price( $price + $custom_text_fee );
        }
    }
}
add_action( 'woocommerce_before_calculate_totals', 'calculate_custom_text_fee', 99, 1 );
?>

输出到购物车描述:

<?php 
/*Output to Cart Description*/

function render_meta_on_cart_and_checkout( $cart_data, $cart_item = null ) {
    $meta_items = array();


    if( !empty( $cart_data ) ) {
        $meta_items = $cart_data;
    }

    if( isset( $cart_item["custom_text"] ) ) {
        $meta_items[] = array( "name" => "Your Custom Text", "value" => $cart_item["custom_text"] );
    }

    return $meta_items;
}
add_filter( 'woocommerce_get_item_data', 'render_meta_on_cart_and_checkout', 99, 2 );
?>

确保产品已包含在电子邮件中:

<?php 
/*Ensure Product is in Email Notifications*/
function custom_text_order_meta_handler( $item_id, $values, $cart_item_key ) {

    if( isset( $values["custom_text"] ) ) {
        wc_add_order_item_meta( $item_id, "Custom Text", $values["custom_text"] );
    }   
}
add_action( 'woocommerce_add_order_item_meta', 'custom_text_order_meta_handler', 99, 3 );
?>

有人能看到我可能会出问题的地方,这导致我的数学费用增加了一倍在购物篮页面中?

Is anyone able to see where I could be going wrong, which is causing my Maths to double the costs in the Basket page?

请明确说明,产品价格称为罚款。那没有翻倍。只是自定义文本的费用。我觉得 $ price 可能有问题,但是我不太确定。

Just to be clear, the Product Price is called fine. That is not doubled. Just the cost of the Custom Text. I feel there may be an issue with $price but I am just not so sure.

任何指针,在

推荐答案

更新

为避免此问题,我在进行新的价格计算之前进行了处理,并使用 woocommerce_add_cart_item_data 钩子(将其与保存购物车对象中的自定义文本)

To avoid this problem, I make before the new price calculation and save it in cart object using woocommerce_add_cart_item_data hook (the same hook used to save the custom text in cart object).

注意:对于您的文本,我正在使用 trim() php函数从中删除空格长度字母计数…如果不需要,可以将其删除。

Note: For your text I am using trim() php function to remove white spaces from the length letter count… If you dont need it you can remove it.

我重新查看了您的代码,进行了一些更改:

I have revisited your code making some changes:

// Add custom fields in "product data" settings metabox ("Advanced" tab)
add_action('woocommerce_product_options_advanced', 'add_text_field_product_dashboard');
function add_text_field_product_dashboard(){

   global $post;

    echo '<div class="product_custom_field">';

    // Checkbox Field
    woocommerce_wp_checkbox( array(
        'id'        => '_custom_text_option',
        'description'      =>  __('set custom custom text field', 'woocommerce'),
        'label'     => __('Display custom custom text field', 'woocommerce'),
        'desc_tip'  => 'true',
    ) );

    // Minimum Letter Text Box
    woocommerce_wp_text_input( array(
        'id'        => '_minimum_custom_text_option',
        'label'     => __('Minimum Letters', 'woocommerce'),
        'description' =>  __('set custom minimum Lettering text field', 'woocommerce'),
        'desc_tip'  => 'true',
    ) );

    // Maximum Letter Text Box
    woocommerce_wp_text_input( array(
        'id'        => '_maximum_custom_text_option',
        'label'     => __('Maximum Letters', 'woocommerce'),
        'description' => __('set custom maximum Lettering text field', 'woocommerce'),
        'desc_tip'  => 'true'
    ) );

    // Cost Per Letter Pricing
    woocommerce_wp_text_input( array(
        'id'        => '_pricing_custom_text_option',
        'label'     => __('Cost Per Letter', 'woocommerce'),
        'description' => __('set custom pricing Lettering text field', 'woocommerce'),
        'desc_tip'  => 'true'
    ) );

    echo '</div>';
}

// Save Inputted Entries, in the Product Dashboard Text Fields.
add_action('woocommerce_process_product_meta', 'woocommerce_product_custom_fields_save');
 function woocommerce_product_custom_fields_save($post_id){
    // Checkbox Field
    $checkbox = isset( $_POST['_custom_text_option'] ) ? 'yes' : 'no';
    update_post_meta( $post_id, '_custom_text_option', $checkbox );

    // Save Minimum Letters
    if ( isset( $_POST['_minimum_custom_text_option'] ) )
        update_post_meta($post_id, '_minimum_custom_text_option', sanitize_text_field( $_POST['_minimum_custom_text_option'] ) );

    // Save Maximum Letters
    if ( isset( $_POST['_maximum_custom_text_option'] ) )
        update_post_meta($post_id, '_maximum_custom_text_option', sanitize_text_field( $_POST['_maximum_custom_text_option'] ) );

    // Save Cost Per Letter
    if ( isset( $_POST['_pricing_custom_text_option'] ) )
        update_post_meta($post_id, '_pricing_custom_text_option', sanitize_text_field( $_POST['_pricing_custom_text_option'] ) );
}


// Output Custom Text Field to Product Page
add_action( 'woocommerce_before_add_to_cart_button', 'add_custom_text_field', 0 );
function add_custom_text_field() {
    global $post;

    // Get the checkbox value
    $custom_option = get_post_meta( $post->ID, '_custom_text_option', true );

    // If is single product page and have the "custom text option" enabled we display the field
    if ( is_product() && ! empty($custom_option) ) {
?>
        <div>
            <label class="product-custom-text-label" for="custom_text"><?php _e( 'Custom Letters:', 'woocommerce'); ?><br>
                <input style="min-width:220px" type="text" class="product-counter" name="custom_text" placeholder="<?php _e( 'Enter Your Custom Letters ...', 'woocommerce'); ?>" minlength="<?php global $post; echo get_post_meta($post->ID,'_minimum_custom_text_option',true);?>" maxlength="<?php global $post; echo get_post_meta($post->ID,'_maximum_custom_text_option',true);?>" />
            </label>
        </div><br>
<?php
    }
}

// Set custom text and  calculated price as custom cart data in the cart item
add_filter( 'woocommerce_add_cart_item_data', 'save_custom_data_in_cart_object', 30, 3 );
function save_custom_data_in_cart_object( $cart_item_data, $product_id, $variation_id ) {
    if( ! isset( $_POST['custom_text'] ) || empty( $_POST['custom_text'] ) )
        return $cart_item_data;

    // Get the custom text cost by letter
    $pricing_custom = (float) get_post_meta( $product_id, '_pricing_custom_text_option', true );

    // Get an instance of the WC_Product object
    $product = $variation_id > 0 ? wc_get_product($variation_id) : wc_get_product($product_id);
    $product_price = (float) $product->get_price(); // Get the product price

    // Get the text
    $custom_text = sanitize_text_field ( $_POST['custom_text'] );
    // Get lenght (trimming white spaces)
    $lenght = (float) strlen ( trim( $custom_text ) );

    // Set the text and the calculated price as custom cart data in the cart item
    $cart_item_data['custom_data']['price'] = $product_price + ( $lenght * $pricing_custom );
    $cart_item_data['custom_data']['text']  = $custom_text;

    return $cart_item_data;
}

// Display Custom text in cart and checkout pages
add_filter( 'woocommerce_get_item_data', 'render_meta_on_cart_and_checkout', 99, 2 );
function render_meta_on_cart_and_checkout( $cart_data, $cart_item = null ) {

    if( isset( $cart_item['custom_data']['text'] ) )
        $cart_data[] = array( "name" => "Your Custom Text", "value" => $cart_item["custom_data"]["text"] );

    return $cart_data;
}

// Set the new calculated price of the cart item
add_action( 'woocommerce_before_calculate_totals', 'calculate_custom_text_fee', 99, 1 );
function calculate_custom_text_fee( $cart ) {
    if ( is_admin() && ! defined( 'DOING_AJAX' ) )
        return;

    if ( did_action( 'woocommerce_before_calculate_totals' ) >= 2 )
        return;

    foreach ( $cart->get_cart() as $cart_item ) {
        if( isset( $cart_item['custom_data']['price'] ) ) {
            // Get the new calculated price
            $new_price = (float) $cart_item['custom_data']['price'];

            // Set the new calculated price
            $cart_item['data']->set_price( $new_price );
        }
    }
}

// Save the custom text as order item data (displaying it in order and notifications)
add_action( 'woocommerce_add_order_item_meta', 'custom_text_order_meta_handler', 99, 3 );
function custom_text_order_meta_handler( $item_id, $values, $cart_item_key ) {

    if( isset( $values['custom_data']['text'] ) )
        wc_add_order_item_meta( $item_id, "Custom Text", $values["custom_data"]["text"] );
}

代码包含在您活动的子主题的function.php文件中(或活动主题)。

经过测试并有效

这篇关于Woocommerce中基于定制字段的动态价格计算的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持IT屋!

查看全文
相关文章
登录 关闭
扫码关注1秒登录
发送“验证码”获取 | 15天全站免登陆