为了账号安全,请及时绑定邮箱和手机立即绑定

WooCommerce 购物车和结账中运输方式的额外承运人字段

WooCommerce 购物车和结账中运输方式的额外承运人字段

PHP
慕的地10843 2023-09-08 18:23:01
受Woocommerce 结帐页面答案代码中的运输承运人自定义字段验证的启发,我使用以下代码显示运输公司的选择字段(仅当我选择特定的运输方式时才显示该字段):add_action( 'woocommerce_after_shipping_rate', 'carrier_custom_fields', 20, 2 );function carrier_custom_fields( $method, $index ) {    if( ! is_checkout()) return; // Only on the checkout page    $customer_carrier_method = 'flat_rate:14';    if( $method->id != $customer_carrier_method ) return; // Mostrar solo para "flat_rate:14"    $chosen_method_id = WC()->session->chosen_shipping_methods[ $index ];    // If the chosen shipping method is 'flat_rate: 14', we will show    if($chosen_method_id == $customer_carrier_method ):    echo '<div class="custom-carrier2">';    woocommerce_form_field( 'carrier_name1', array(        'type'          => 'select',        'class'         => array('carrier_name2-class form-row-wide'),        'label'         => __('<strong>Shipping Company</strong>'),        'required'      => 'true',        'options'       => array(            '1'                     => '', // no data means that the field is not selected            'Shipping Company 1'    => 'Shipping Company 1',            'Shipping Company 2'    => 'Shipping Company 2',            'Shipping Company 3'    => 'Shipping Company 3',            'Shipping Company 4'    => 'Shipping Company 4'        )    ), WC()->checkout->get_value( 'carrier_name1' ) );    echo '</div>';    endif;}// Validate the custom selection fieldadd_action('woocommerce_checkout_process', 'carrier_checkout_process');function carrier_checkout_process() {    if( isset( $_POST['carrier_name1'] ) && empty( $_POST['carrier_name1'] ) )        wc_add_notice( ( "<strong>Shipping Company</strong> it is a required field." ), "error" );}问题是它只显示在结帐页面上,我希望它在购物车页面中显示它,将购物车页面上选定的值保留到结帐页面。我想我发现了一些内容,表明购物车和支付页面之间所选数据的传输是通过 Ajax 完成的,但我不熟悉 Ajax,也不知道如何实现这一点。
查看完整描述

1 回答

?
MM们

TA贡献1886条经验 获得超2个赞

为了使其在购物车和结账页面上工作,您将需要一些使用 jQuery、Ajax 和 WC Session 变量的附加代码:


最终更新- 为了使代码更加动态,我们从一个自定义函数开始,它将处理所有必需的设置:


// Custom function that handle your settings

function carrier_settings(){

    return array(

        'targeted_methods' => array('flat_rate:14'), // Your targeted shipping method(s) in this array

        'field_id'         => 'carrier_name', // Field Id

        'field_type'       => 'select', // Field type

        'field_label'      => '', // Leave empty value if the first option has a text (see below).

        'label_name'       => __("Carrier company","woocommerce"), // for validation and as meta key for orders

        'field_options'    => array(

             // The option displayed at first ( or keep an empty value '',)

            __("Choose a carrier company", "woocommerce"),

            // The carrier companies below (one by line)

            'Company name 1',

            'Company name 2',

            'Company name 3',

            'Company name 4',

        ),

    );

}

然后我们可以将该设置加载到任何需要的函数上。


现在,购物车和结帐页面上的“选择”字段中显示了特定运输方式的承运商公司:


// Display the custom checkout field

add_action( 'woocommerce_after_shipping_rate', 'carrier_company_custom_select_field', 20, 2 );

function carrier_company_custom_select_field( $method, $index ) {

    extract( carrier_settings() ); // Load settings and convert them in variables


    $chosen  = WC()->session->get('chosen_shipping_methods'); // The chosen methods

    $value   = WC()->session->get($field_id);

    $value   = WC()->session->__isset($field_id) ? $value : WC()->checkout->get_value('_'.$field_id);

    $options = array(); // Initializing


    if( ! empty($chosen) && $method->id === $chosen[$index] && in_array($method->id, $targeted_methods)  ) {

        echo '<div class="custom-carrier">';


        // Loop through field otions to add the correct keys

        foreach( $field_options as $key => $option_value ) {

            $option_key = $key == 0 ? '' : $key;

            $options[$option_key] = $option_value;

        }


        woocommerce_form_field( $field_id, array(

            'type'     => $field_type,

            'label'    => '', // Not required if the first option has a text.

            'class'    => array('form-row-wide ' . $field_id . '-' . $field_type ),

            'required' => true,

            'options'  => $options,

        ), $value );


        echo '</div>';

    }

}

Ajax 部分:所选运营商公司的 jQuery 发送器 + PHP WordPress 管理 Ajax 接收器代码:


// jQuery code (client side) - Ajax sender 

add_action( 'wp_footer', 'carrier_company_script_js' );

function carrier_company_script_js() {

    // Only cart & checkout pages

    if( is_cart() || ( is_checkout() && ! is_wc_endpoint_url() ) ):


    // Load settings and convert them in variables

    extract( carrier_settings() );


    $js_variable = is_cart() ? 'wc_cart_params' : 'wc_checkout_params';


    // jQuery Ajax code

    ?>

    <script type="text/javascript">

    jQuery( function($){

        if (typeof <?php echo $js_variable; ?> === 'undefined')

            return false;


        $(document.body).on( 'change', 'select#<?php echo $field_id; ?>', function(){

            var value = $(this).val();

            $.ajax({

                type: 'POST',

                url: <?php echo $js_variable; ?>.ajax_url,

                data: {

                    'action': 'carrier_name',

                    'value': value

                },

                success: function (result) {

                    console.log(result); // Only for testing (to be removed)

                }

            });

        });

    });

    </script>

    <?php

    endif;

}


// The Wordpress Ajax PHP receiver

add_action( 'wp_ajax_carrier_name', 'set_carrier_company_name' );

add_action( 'wp_ajax_nopriv_carrier_name', 'set_carrier_company_name' );

function set_carrier_company_name() {

    if ( isset($_POST['value']) ){

        // Load settings and convert them in variables

        extract( carrier_settings() );


        if( empty($_POST['value']) ) {

            $value = 0;

            $label = 'Empty';

        } else {

            $value = $label = esc_attr( $_POST['value'] );

        }


        // Update session variable

        WC()->session->set( $field_id, $value );


        // Send back the data to javascript (json encoded)

        echo $label . ' | ' . $field_options[$value];

        die();

    }

}

然后在结帐页面上进行字段验证并将所选承运公司保存到订单中:


// Conditional function for validation

function has_carrier_field(){

    $settings = carrier_settings();

    return array_intersect(WC()->session->get( 'chosen_shipping_methods' ), $settings['targeted_methods']);

}


// Validate the custom selection field

add_action('woocommerce_checkout_process', 'carrier_company_checkout_validation');

function carrier_company_checkout_validation() {

    // Load settings and convert them in variables

    extract( carrier_settings() );


    if( has_carrier_field() && isset( $_POST[$field_id] ) && empty( $_POST[$field_id] ) )

        wc_add_notice(

            sprintf( __("Please select a %s as it is a required field.","woocommerce"),

            '<strong>' . $label_name . '</strong>'

        ), "error" );

}


// Save custom field as order meta data

add_action( 'woocommerce_checkout_create_order', 'save_carrier_company_as_order_meta', 30, 1 );

function save_carrier_company_as_order_meta( $order ) {

    // Load settings and convert them in variables

    extract( carrier_settings() );


    if( has_carrier_field() && isset( $_POST[$field_id] ) && ! empty( $_POST[$field_id] ) ) {

        $order->update_meta_data( '_'.$field_id, $field_options[esc_attr($_POST[$field_id])] );

        WC()->session->__unset( $field_id ); // remove session variable

    }

}

在管理订单页面、客户订单和电子邮件通知上显示所选承运商:


// Display custom field in admin order pages

add_action( 'woocommerce_admin_order_data_after_shipping_address', 'admin_order_display_carrier_company', 30, 1 );

function admin_order_display_carrier_company( $order ) {

    // Load settings and convert them in variables

    extract( carrier_settings() );


    $carrier = $order->get_meta( '_'.$field_id ); // Get carrier company


    if( ! empty($carrier) ) {

        // Display

        echo '<p><strong>' . $label_name . '</strong>: ' . $carrier . '</p>';

    }

}


// Display carrier company after shipping line everywhere (orders and emails)

add_filter( 'woocommerce_get_order_item_totals', 'display_carrier_company_on_order_item_totals', 1000, 3 );

function display_carrier_company_on_order_item_totals( $total_rows, $order, $tax_display ){

    // Load settings and convert them in variables

    extract( carrier_settings() );


    $carrier = $order->get_meta( '_'.$field_id ); // Get carrier company


    if( ! empty($carrier) ) {

        $new_total_rows = [];


        // Loop through order total rows

        foreach( $total_rows as $key => $values ) {

            $new_total_rows[$key] = $values;

            

            // Inserting the carrier company under shipping method

            if( $key === 'shipping' ) {

                $new_total_rows[$field_id] = array(

                    'label' => $label_name,

                    'value' => $carrier,

                );

            }

        }

        return $new_total_rows;

    }

    return $total_rows;

}

所有代码都位于活动子主题(或主题)的functions.php 文件中。经过测试并有效

在购物车页面上 (针对所选的特定运输方式)

https://img1.sycdn.imooc.com//64faf64c0001b72605230263.jpg

在结帐页面 (针对所选的特定运输方式)

https://img1.sycdn.imooc.com//64faf655000132a605120423.jpg

关于客户订单 (电子邮件通知和管理订单页面)

https://img1.sycdn.imooc.com//64faf66300011c4706190269.jpg


查看完整回答
反对 回复 2023-09-08
  • 1 回答
  • 0 关注
  • 66 浏览

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信