gpt4 book ai didi

php - WooCommerce:结帐验证失败后更新自定义字段

转载 作者:行者123 更新时间:2023-12-01 03:07:36 25 4
gpt4 key购买 nike

在我的项目中,我正在自定义一些 WooCommerce 功能。
我的“运送方式”是:
1.发货
2.带走

我还在结账页面添加了一个自定义字段,即 <select>填充有效的送货时间(情况“1”)或外卖时间(情况“2”)。

用户可能会选择 2. take away在购物车页面中,然后选择对“2”有效的时间,但随后更改为 1. delivery并且所选时间可能不再有效,选项列表和自定义字段标签也不再有效。

当然,我正在使用 woocommerce_checkout_process通过 wc_add_notice() 警告用户的钩子(Hook),但即使woocommerce_checkout_fields钩子(Hook)被触发(这就是我创建选择列表的地方),<select>值未更新。

我认为有一个 AJAX 调用仅与运输方式相关,并且不会更新其他结帐字段,尽管 woocommerce_checkout_fields钩子(Hook)被触发。

如何更新自定义字段?我需要一些 js/jquery/AJAX 吗?

或者:自定义字段是否可以与运输方式相关(并通过 AJAX 进行更新)?怎么办?

编辑

自定义字段代码:

add_filter( 'woocommerce_checkout_fields', 'fty_filter_checkout_fields' );
function my_filter_checkout_fields($fields) {
$must_deliver = WC()->cart->shipping_total > 0.0; // true=deliver, false=take away

// some complex code to calculate time lists omitted, samples array instead:

$delivery_time_list = array(
"deliver 10:00",
"deliver 11:00",
"deliver 12:00",
"deliver 13:00"
);

$takeaway_time_list = array(
"takeaway 10:00",
"takeaway 10:30",
"takeaway 11:00",
"takeaway 11:30",
"takeaway 12:00",
"takeaway 12:30",
"takeaway 13:00",
"takeaway 13:30"
);

// add the new conditional field
if($must_deliver) {
$fields['my_delivery_datetime'] = array(
'my_delivery_time' => array(
'type' => 'select',
'options' => $delivery_time_list,
'required' => true,
'label' => __('Delivery time')
)
);
} else {
$fields['my_delivery_time'] = array(
'my_delivery_time' => array(
'type' => 'select',
'options' => $takeaway_time_list,
'required' => true,
'label' => __('Take away time')
)
);
}

return $fields;
}

验证代码的想法:

add_action('woocommerce_checkout_process', 'my_checkout_date_time_validation', 30, 1);
function my_checkout_date_time_validation($doh) {
$time = filter_input(INPUT_POST, 'my_delivery_time');
$shipping = filter_input(INPUT_POST, 'shipping_method', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);

if(strpos($time, "deliver")!==FALSE && strpos($shipping[0], "local_pickup")!==FALSE) {
wc_add_notice('Please re-select take away time', 'error');
} else if(strpos($time, "takeaway")!==FALSE && strpos($shipping[0], "distance_based_rate")!==FALSE) {
wc_add_notice('Please re-select delivery time', 'error');
}

}

这是关于运输方式的信息;

add_action( 'woocommerce_flat_rate_shipping_add_rate', 'add_distance_based_delivery_rate', 10, 2 );
function add_distance_based_delivery_rate( $method, $rate ) {
$new_rate = $rate;
$new_rate['id'] .= ':' . 'distance_based_rate';
$new_rate['label'] = 'delivery'; // Rename to 'Rushed Shipping'.

// incredibly complex code used to calculate delivery costs omitted
$dist_cost = 1000;

$new_rate['cost'] += $dist_cost;
$method->add_rate( $new_rate );
}

谢谢!

最佳答案

提供的代码大多没有用......我做了很多更改和优化。我的所有代码都带有注释,在 WooCommerce 3+ 上进行了测试并且完美运行。

You will have to add your "incredibly complex code used to calculate delivery costs omitted"…

<小时/>

1) 用于条件结帐字段现场事件的 JavaScript
参与客户现场事件(浏览器端)的唯一方法是 javascript/jQuery。所以这并不容易,因为 WooCommerce 在结账页面上已经使用了大量的 javascript/jQuery/Ajax...

我已将 javascript 代码包含到 Hook 函数中,但您应该将其保存在单独的文件中,并使用经典的 WordPress 注册脚本函数注册此脚本文件,如以下线程所示:

Checkout fields: Hiding and showing existing fields

2) 使用现有的运输方式(动态价格计算):
您无需创建任何运费。您可以使用:

  • local_pickup适用于您的“TAKE WAY”
  • flat_rate“送货”可用的方法(动态价格计算)

对于您的每个送货区域,在 Woocommerce > 设置 > 送货中启用、设置和重命名(标签名称) 2 种方法:

enter image description here

对于统一费率,您可以设置任何最低金额(该金额将被您的计算覆盖)...

If you make changes you need to refresh shipping cached data: disable, save and enable, save those methods for the current shipping zone.

3) 节省订购元数据的运输时间:
我为此添加了一些代码,并将其保存在 2 个自定义元字段中:

  • 一件用于选择的运输
  • 暂时的另一个

4) 在元框中显示所选的运输类型和时间(在订单编辑页面中):
我还为此添加了一些代码。

enter image description here

<小时/>

最后是代码:

add_action( 'woocommerce_after_order_notes', 'my_custom_checkout_field' );

function my_custom_checkout_field( $checkout ) {

// The 2 Options arrays in imput select
$delivery_time_list[''] = $takeaway_time_list[''] = __('Select an hour');
for($h = 10, $i = 0; $i < 8; $i++ ){
if( $i % 2 == 0 ){
$time = $h.':00';
$delivery_time_list[$time] = 'deliver '.$time;
} else {
$time = $h.':30';
$h++;
}
$takeaway_time_list[$time] = 'takeaway '.$time;
}

echo '<div id="delivery_checkout_fields"><h3>' . __('Shipping time options') . '</h3>';

woocommerce_form_field( 'delivery_time', array(
'type' => 'select',
'class' => array('delivery-time form-row-wide'),
'label' => __('Delivery time'),
'options' => $delivery_time_list,
), $checkout->get_value( 'delivery_time' ) );

woocommerce_form_field( 'takeaway_time', array(
'type' => 'select',
'class' => array('takeaway-time form-row-wide'),
'label' => __('Take away time'),
'options' => $takeaway_time_list,
), $checkout->get_value( 'takeaway_time' ) );

echo '</div>';

$required = esc_attr__( 'required', 'woocommerce' );

?>
<script>
jQuery(function($){

var choosenShipMethod = $('input[name^="shipping_method"]:checked').val().split(':')[0], // Choosen shipping method slug
required = '<abbr class="required" title="<?php echo $required; ?>">*</abbr>'; // Required html

// TESTING: displaying in console the choosen shipping
console.log('Chosen shipping: '+choosenShipMethod);

// Function that shows or hide imput select fields
function showHide( actionToDo='show', selector='' ){
if( actionToDo == 'show' )
$(selector).show(function(){
$(this).addClass("validate-required");
$(this).removeClass("woocommerce-validated");
$(this).removeClass("woocommerce-invalid woocommerce-invalid-required-field");
$(selector+' label').append(required);
//console.log('Selector (show): '+selector);
});
else
$(selector).hide(function(){
$(this).removeClass("validate-required");
$(this).removeClass("woocommerce-validated");
$(this).removeClass("woocommerce-invalid woocommerce-invalid-required-field");
$(selector+' label > .required').remove();
//console.log('Selector (hide): '+selector);
});
}

// Initialising at start (Based on the choosen shipping method)
if( choosenShipMethod == 'flat_rate' ) // Choosen "Delivery" (Hidding "Take away")
{
showHide('show','#delivery_time_field' );
showHide('hide','#takeaway_time_field' );
}
else if( choosenShipMethod == 'local_pickup' ) // Choosen "Take away" (Hidding "Delivery")
{
showHide('show','#takeaway_time_field' );
showHide('hide','#delivery_time_field' );
}
else // No shipping choosen yet (Hidding BOTH shipping dropdown hour selectors
{
showHide('hide','#delivery_time_field' );
showHide('hide','#takeaway_time_field' );
$('#delivery_checkout_fields').hide();
}

// When shipping method is changed (Live event)
$( 'form.checkout' ).on( 'change', 'input[name^="shipping_method"]', function() {
var changedShipMethod = $('input[name^="shipping_method"]:checked').val().split(':')[0];
if( changedShipMethod == 'flat_rate' )
{
// Choose "Delivery" | Show "Delivery" and Hide "Take away"
$('#delivery_checkout_fields').show();
showHide('show','#delivery_time_field' );
showHide('hide','#takeaway_time_field' );
}
else if( changedShipMethod == 'local_pickup' )
{
// Choose "Take away" | Show "Take away" and Hide "Delivery"
$('#delivery_checkout_fields').show();
showHide('show','#takeaway_time_field' );
showHide('hide','#delivery_time_field' );
}
console.log("Chosen shipping: "+changedShipMethod);
});

// When an hour is selected (LIVE event)
$('#delivery_checkout_fields select').change( function(){
if( $(this).val() != '')
$(this).parent().removeClass("validate-required");
else
$(this).parent().addClass("validate-required");

console.log("Selector value: "+$(this).val());
});
// "select.shipping_method, input[name^="shipping_method"], #ship-to-different-address input, .update_totals_on_change select, .update_totals_on_change input[type="radio"], .update_totals_on_change input[type="checkbox"]"
//"function (){t.reset_update_checkout_timer(),t.dirtyInput=!1,e(document.body).trigger("update_checkout")}"
});
</script>
<?php

}


// Process the checkout (Checking if required fields are not empty)
add_action('woocommerce_checkout_process', 'ba_custom_checkout_field_process');
function ba_custom_checkout_field_process() {

$delivery_time = $takeaway_time = 0;
if ( $_POST['delivery_time'] ) $delivery_time = 1;
if ( $_POST['takeaway_time'] ) $takeaway_time = 1;

// Only one message is possible for both
if ( ( $delivery_time + $takeaway_time ) == 0 ){
wc_add_notice( __('Please select a <strong>shipping time</strong>.' ), 'error');
}
}


## CALCULATING THE DELIVERY FEE (BASED ON COUNTING THE DIFFERENT DATES For all items) ##
add_filter( 'woocommerce_package_rates', 'custom_shipping_flat_rate_cost_calculation', 10, 2 );
function custom_shipping_flat_rate_cost_calculation( $rates, $package )
{
## --- CALCULATIONS Based on CART DATA (if needed) --- ##

foreach(WC()->cart->get_cart() as $cart_item ):
// HERE your incredibly complex code used to calculate delivery costs
endforeach;


## --- CHANGING DYNAMICALLY THE METHODS COSTS --- ##

foreach($rates as $rate_key => $rate_values):

$method_id = $rate_values->method_id;
$rate_id = $rate_values->id;

// "DELIVERY" - "local_pickup" method (if needed)
if ( 'flat_rate' === $method_id ){

// HERE your incredibly complex code used to calculate delivery costs

// Change the price cost
$price_excl_tax = $rates[$rate_id]->cost + 2.5;
$rates[$rate_id]->cost = number_format($price_excl_tax, 2);

$tax_calculation = $rates[$rate_id]->taxes[0] * 0.1;
$rates[$rate_id]->taxes[0] = number_format($tax_calculation, 2);
}
// "TAKE WAY" - "local_pickup" method (if needed)
elseif ( 'local_pickup' === $method_id )
{
// do something if needed
}

endforeach;

return $rates;
}


// Save the "shipping time" in order meta data
add_action( 'woocommerce_checkout_update_order_meta', 'save_shipping_time_in_order_meta', 100, 1 );
function save_shipping_time_in_order_meta( $order_id ) {

// Take away time
$takeaway_time = $_POST['takeaway_time'];
if ( ! empty( $takeaway_time ) ){
add_post_meta( $order_id, '_shipping_time', $takeaway_time );
add_post_meta( $order_id, '_shipping_type', __('Take away', 'woocommerce' ) );
}
// Delivery time
$delivery_time = $_POST['delivery_time'];
if ( ! empty( $delivery_time ) ){
add_post_meta( $order_id, '_shipping_time', $delivery_time );
add_post_meta( $order_id, '_shipping_type', __('Delivery', 'woocommerce' ) );
}

}

// Adding shipping time metabox (on right side) to Order edit pages
add_action( 'add_meta_boxes', 'add_order_shipping_time_meta_boxe' );
function add_order_shipping_time_meta_boxe(){

add_meta_box(
'woocommerce-order-shipping-time-values', __( 'Shipping type and time', 'woocommerce' ),
'order_shipping_time_values', 'shop_order', 'side', 'default'
);
}

// Adding content to shipping time metabox to Order edit pages
function order_shipping_time_values(){
global $post;

$type = get_post_meta($post->ID, '_shipping_type', true);
$time = get_post_meta($post->ID, '_shipping_time', true);

echo "<p><strong>Type:</strong> $type | <strong>time:</strong> $time</p>";
}

代码位于事件子主题(或主题)的 function.php 文件中或任何插件文件中。

此代码在 WooCommerce 3+ 上经过测试并且有效。

关于php - WooCommerce:结帐验证失败后更新自定义字段,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/45243303/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com