我遇到问题让我的价格更新坚持下去。或多或少这是我用来更新价格的样本:
add_action( 'woocommerce_before_calculate_totals', 'cst_product_quantity_discounter', 10 );
function cst_product_quantity_discounter( $cart_object ) {
if ( is_admin() && ! defined( 'DOING_AJAX' ) ){
return;
}
if( ! isset( WC()->session->reload_checkout ) ) {
// declare woocommerce global
global $woocommerce;
// array of ID's to match that represent each item
$special_ids_array = [
0 => array( "id" => 15372, "product" => "TAF", "single_bag_price" => 42.99, "3_multiplier" => 0.82181902768 ),
1 => array( "id" => 14285, "product" => "POW", "single_bag_price" => 29.99, "3_multiplier" => 0.8890741358 )
];
// gather cart object
$wc_cart = WC()->cart->get_cart();
foreach( $wc_cart as $cart_item ){
// gather relevant data for testing and comparisons
$product_id = $cart_item['product_id'];
$product_key = $cart_item['key'];
$pack_size_attribute = $cart_item['variation']['attribute_pa_sample-check']; // returns: "full-pack" || "sample" || null
$item_quantity = $cart_item['quantity'];
foreach ( $special_ids_array as $id_test_2 ) {
if ( $product_id === $id_test_2["id"] && $pack_size_attribute !== "sample" ){
foreach ( $cart_object->cart_contents as $key => $value ) {
// if the key matches we will set the price of the correct product
if ( $key === $product_key ){
$discounted_price = 10;
$value['data']->set_price($discounted_price);
}
}
}
}
}
}
}
还有其他一些东西可以看到这些价值,但是有效。这将在我的测试车页面上更新,一切看起来都很好。
我的问题是,当我进入我的测试结帐页面时价格会短暂更新,然后它们会被原始价格覆盖。我错过了一些在结帐时运行的更新,woocommerce_before_calculate_totals
挂钩似乎没有在结帐页面上进行永久性更改。
我需要在哪里挂钩我的功能,以便更改在结帐时发生的任何负载都会持续覆盖,从而覆盖最初的成功价格变化?
答案 0 :(得分:2)
您的代码中有一些奇怪的东西:
$cart_object
(即WC_Cart
对象)已经是钩子函数中的参数(它取代了WC()->cart
或过时的$woocommerce->cart
) ... WC()->cart->get_cart()
和$cart_object->cart_contents
完全相同。最好的是在你的foreach循环中使用$cart_object->get_cart()
。global $woocommerce;
,因为它已经包含在WC()
中(全球Woocommerce对象已经在$cart_object
中)reload_checkout
在WC_Sessions
或WC_Session_Handler
类中不存在(因此您的条件始终为真),无论如何这对此挂钩无效。 $cart_item['data']
是此购物车项目的WC_Product
对象的实例。$cart_item['key']
和$key
(在购物车循环中)始终是一回事。在这种情况下你真的不需要使用它...... 所以我尝试在更实际的情况下添加一种动态价格计算 ,但是以基于
$special_ids_array
数组键的不同方式/值。计算出的动态价格折扣的开头数量为3个符合条件的商品(Product Ids et“pack size”属性)。
所以这一切都简化并压缩了这个重新审视的代码:
add_action( 'woocommerce_before_calculate_totals', 'custom_product_quantity_discounter', 10, 1 );
function custom_product_quantity_discounter( $cart ) {
if ( is_admin() && ! defined( 'DOING_AJAX' ) )
return;
if ( did_action( 'woocommerce_before_calculate_totals' ) >= 2 )
return;
// array of ID's to match that represent each item
$special_ids = array(
0 => array( "id" => 15372, "product" => "TAF", "single_bag_price" => 42.99, "3_multiplier" => 0.82181902768 ),
1 => array( "id" => 14285, "product" => "POW", "single_bag_price" => 29.99, "3_multiplier" => 0.8890741358 ),
);
// Loop through the cart items
foreach( $cart->get_cart() as $cart_item ){
$product_id = $cart_item['product_id'];
$quantity = $cart_item['quantity'];
$pack_size = $cart_item['variation']['attribute_pa_sample-check']; // returns "full-pack", "sample" or null
foreach ( $special_ids as $special_id ){
if( $special_id['id'] == $product_id && $pack_size !== 'sample' && $quantity >= 3 ){
$discounted_price = $special_id['single_bag_price'] * $special_id['3_multiplier']; // calculation (fake and custom)
$cart_item['data']->set_price($discounted_price); // set the calculated price
}
}
}
}
代码放在活动子主题(或活动主题)的function.php文件中。
经过测试和工作。
即使多次重新加载页面,基于购物车商品数量和自定义计算的动态计算定价也会在结帐时保留。