以编程方式更新订单不会更新总计

时间:2019-11-04 18:55:42

标签: wordpress woocommerce

我正在尝试更新订购产品的数量,这样做时,我希望订单能够反映实际成本。我发现发生的事情是降低了产品成本以匹配总数,并且订单总数从未真正更新过。我在下面提供了一个简单的示例:

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美元。

简而言之,我想做的是更新现有的订单项目数量,并更新总数以反映新的数量,成本,折扣和税金。我需要使用什么方法来实现这一目标?

4 个答案:

答案 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 );
     */

}