我正在尝试更新订购产品的数量,这样做时,我希望订单能够反映实际成本。我发现发生的事情是降低了产品成本以匹配总数,并且订单总数从未真正更新过。我在下面提供了一个简单的示例:
function prefix_update_woo_order() {
$order_id = 123; // This needs to be a real order or there will be errors
$order_item_id = 5; // This needs to be a real order item ID or there will be errors.
$order = new WC_Order( $order_id );
$order_items = $order->get_items();
$order_items[ $order_item_id ]->set_quantity( 2 );
$order->calculate_taxes();
$order->calculate_totals();
$order->save();
}
add_action( 'admin_init', 'prefix_update_woo_order' );
例如,“带徽标的无檐小便帽”产品的售价为$ 18.00,我最初购买了1。在下订单后,我想以编程方式将订单商品更新为数量2,而不是数量1。我希望总价为36.00美元,但我发现产品成本发生了变化,以匹配总价。数量更新为2,而不是“带有徽标的无檐小便帽”的成本为18.00美元,而成本降低为9.00美元。
简而言之,我想做的是更新现有的订单项目数量,并更新总数以反映新的数量,成本,折扣和税金。我需要使用什么方法来实现这一目标?
答案 0 :(得分:1)
您好,我认为这段代码会改变您的问题
add_action( 'admin_init', 'test_order_update_order' );
function test_order_update_order() {
$order_id = 80; // This needs to be a real order or there will be errors
$order_item_id = 11; // This needs to be a real order item ID or there will be errors.
$quantity = 2; //quantity which you want to set.
$order = new WC_Order( $order_id );
$order_items = $order->get_items();
foreach ( $order_items as $key => $value ) {
if ( $order_item_id == $key ) {
$product_value = $value->get_data();
$product_id = $product_value['product_id'];
}
}
$product = wc_get_product( $product_id );
$price = $product->get_price();
$price = ( int ) $quantity * $price;
$order_items[ $order_item_id ]->set_quantity( 2 );
$order_items[ $order_item_id ]->set_subtotal( $price );
$order->calculate_taxes();
$order->calculate_totals();
$order->save();
}
答案 1 :(得分:0)
我以前没有尝试过要实现的目标。我在您的代码中看到的是$order_items
是由WC_Order::get_items()
生成的Items Objects数组,但是我没有看到WC_Order实例收到有关订单项已更改的通知。我希望有一种$order->update_cart($order_items);
这样的方法,我相信我发现了一些有用的链接,可以进行更多研究
https://hotexamples.com/examples/-/WC_Order/-/php-wc_order-class-examples.html
woocommerce - programmatically update cart item quantity
对不起,我没有什么帮助!
答案 2 :(得分:0)
使用以下代码段完成上述任务-
function prefix_update_woo_order() {
$order_id = 123; // This needs to be a real order or there will be errors
$order_item_id = 5; // This needs to be a real order item ID or there will be errors.
$order = wc_get_order( $order_id );
if( $order && !wc_get_order_item_meta( $order_item_id, '_order_item_data_updated', true ) ) {
$item_price = wc_get_order_item_meta( $order_item_id, '_line_total', true );
$updated_item_quantity = 2;
wc_update_order_item_meta( $order_item_id, '_qty', $updated_item_quantity );
wc_update_order_item_meta( $order_item_id, '_line_total', $item_price * $updated_item_quantity );
$order->calculate_totals();
$order->save();
// set flag
wc_add_order_item_meta( $order_item_id, '_order_item_data_updated', true, true );
}
}
add_action( 'admin_init', 'prefix_update_woo_order' );
代码进入活动主题的功能。php
答案 3 :(得分:0)
这是我想出的。我尝试在适用的地方使用wc_format_decimal()
。似乎需要做很多工作来简单地更新订单商品的数量,但这就是它的本质。
底部的注释是不必要的,但是如果您使用的是Cost of Goods plugin,则可以解决此问题。
/**
* Update the order item quantity and totals
* @param Integer $order_id
* @param Integer $order_item_id
* @param Integer $quantity - Quantity to set
*
* @return void
*/
function prefix_update_woo_order( $order_id, $order_item_id, $quantity ) {
// Get Order, Item, and Product Data
$order = new WC_Order( $order_id );
$order_items = $order->get_items();
$line_item = $order_items[ $order_item_id ];
$variation_id = $line_item->get_variation_id();
$product_id = $line_item->get_product_id();
$product = wc_get_product( $variation_id ? $variation_id : $product_id );
$quantity_old = $line_item->get_quantity();
// Calculate Old and New Discounts
$discount = wc_format_decimal( $line_item->get_subtotal() - $line_item->get_total(), '' );
if( ! empty( $discount ) ) {
$discount_per_qty = wc_format_decimal( ( $discount / $quantity_old ), '' );
$discount = wc_format_decimal( ( $discount_per_qty * $quantity ), '' );
}
// Set Quantity and Order Totals
$line_item->set_quantity( $quantity );
$total = wc_get_price_excluding_tax( $product, array( 'qty' => $line_item->get_quantity() ) ); // Also see `wc_get_price_excluding_tax()`
$line_item->set_subtotal( $total ); // Without Discount
$line_item->set_total( $total - $discount ); // With Discount
// Save Everything
$line_item->save();
wc_save_order_items( $order_id, $order_items );
/**
* If using the 'Cost of Goods' Plugin
* - - - - - -
* $cog = $line_item->get_meta( '_wc_cog_item_cost', true );
* $new_cog = wc_format_decimal( ( $quantity * $cog ), '' );
* $line_item->update_meta_data( '_wc_cog_item_total_cost', $new_cog );
* wc_cog()->set_order_cost_meta( $order_id, true );
*/
}