如果在WooCommerce中没有付款,则在X天后自动取消订单

问题描述 投票:2回答:2

我在搜索网络后设法把它放在一起,然而,它不起作用。我的目的是在三天后没有支付订单的情况下,无论支付网关如何,都自动取消所有状态为暂停状态的订单。

代码显然是不完整的,我正在寻求帮助使其完整。我用-1 minute测试它,看看有没有发生。它没。

function get_unpaid_orders() {
    global $wpdb;

    $unpaid_orders = $wpdb->get_col( $wpdb->prepare( "
        SELECT posts.ID
        FROM {$wpdb->posts} AS posts
        WHERE posts.post_status = 'wc-on-hold'
        AND posts.post_date < %s
    ", date( 'Y-m-d H:i:s', strtotime('-1 minute') ) ) );

    return $unpaid_orders;
}

add_action( 'woocommerce_cancel_unpaid_submitted', 'cancel_unpaid_orders' );
function cancel_unpaid_orders() {
    $unpaid_orders = get_unpaid_orders();

    if ( $unpaid_orders ) {
        foreach ( $unpaid_orders as $unpaid_order ) {
            $order = wc_get_order( $unpaid_order );
            $cancel_order = true;

            foreach  ( $order->get_items() as $item_key => $item_values) {
                $manage_stock = get_post_meta( $item_values, '_manage_stock', true );
                if ( $manage_stock == "yes" ) {
                    $payment_method = $order->get_payment_method();
                    if ( $payment_method == "bacs" ) {
                        $cancel_order = false;
                    }
                }
            }
            if ( $cancel_order == true ) {
                $order -> update_status( 'cancelled', __( 'The order was cancelled due to no payment from customer.', 'woocommerce') );
            }
        }
    }
}
php sql wordpress woocommerce orders
2个回答
2
投票

更新4

注意:在WooCommerce中,there is already a function迷上了woocommerce_cancel_unpaid_orders动作钩子,在7天后取消了未付的订单。

我没有找到woocommerce_cancel_unpaid_submitted动作钩子,所以我不知道它是否存在以及它是否被触发。

现在您的代码中存在一些错误,您可以更好地使用wc_get_orders(),它可以直接为您提供正确的WC_Order对象数组......

以下是一些不同的方法(最后一个未经测试):

1)最后一个解决方案经过测试并且可以工作当店铺经理或管理员用户角色浏览管理员订单列表(每天只执行一次)时:

add_action( 'restrict_manage_posts', 'cancel_unpaid_orders' );
function cancel_unpaid_orders() {
    global $pagenow, $post_type;

    // Enable the process to be executed daily when browsing Admin order list 
    if( 'shop_order' === $post_type && 'edit.php' === $pagenow 
        && get_option( 'unpaid_orders_daily_process' ) < time() ) :

    $days_delay = 5; // <=== SET the delay (number of days to wait before cancelation)

    $one_day    = 24 * 60 * 60;
    $today      = strtotime( date('Y-m-d') );

    // Get unpaid orders (5 days old)
    $unpaid_orders = (array) wc_get_orders( array(
        'limit'        => -1,
        'status'       => 'on-hold',
        'date_created' => '<' . ( $today - ($days_delay * $one_day) ),
    ) );

    if ( sizeof($unpaid_orders) > 0 ) {
        $cancelled_text = __("The order was cancelled due to no payment from customer.", "woocommerce");

        // Loop through orders
        foreach ( $unpaid_orders as $order ) {
            $order->update_status( 'cancelled', $cancelled_text );
        }
    }
    // Schedule the process to the next day (executed once restriction)
    update_option( 'unpaid_orders_daily_process', $today + $one_day );

    endif;
}

代码位于活动子主题(或活动主题)的function.php文件中。


2)测试并运行第三个解决方案:当任何订单更改为“处理”或“已完成”状态(仅每天执行一次)时,将触发该功能:

// Triggered on orders status change to "processing" or "completed"
add_action( 'woocommerce_order_status_changed', 'daily_cancel_unpaid_orders', 10, 4 );
function daily_cancel_unpaid_orders( $order_id, $old_status, $new_status, $order ) {
    // Enable the process to be executed daily
    if( in_array( $new_status, array('processing', 'completed') ) 
        && get_option( 'unpaid_orders_daily_process' ) < time() ) :

    $days_delay = 5; // <=== SET the delay (number of days to wait before cancelation)

    $one_day    = 24 * 60 * 60;
    $today      = strtotime( date('Y-m-d') );

    // Get unpaid orders (5 days old)
    $unpaid_orders = (array) wc_get_orders( array(
        'limit'        => -1,
        'status'       => 'on-hold',
        'date_created' => '<' . ( $today - ($days_delay * $one_day) ),
    ) );

    if ( sizeof($unpaid_orders) > 0 ) {
        $cancelled_text = __("The order was cancelled due to no payment from customer.", "woocommerce");

        // Loop through WC_Order Objects
        foreach ( $unpaid_orders as $unpaid_order ) {
            $order->update_status( 'cancelled', $cancelled_text );
        }
    }
    // Schedule the process to the next day (executed once restriction)
    update_option( 'unpaid_orders_daily_process', $today + $one_day );

    endif;
}

代码位于活动子主题(或活动主题)的function.php文件中。


3)所以你可以尝试使用woocommerce_cancel_unpaid_submitted动作钩子:

add_action( 'woocommerce_cancel_unpaid_submitted', 'cancel_unpaid_orders' );
function cancel_unpaid_orders() {
    $days_delay = 5; // <=== SET the delay (number of days to wait before cancelation)

    $one_day    = 24 * 60 * 60;
    $today      = strtotime( date('Y-m-d') );

    // Get unpaid orders (5 days old here)
    $unpaid_orders = (array) wc_get_orders( array(
        'limit'        => -1,
        'status'       => 'on-hold',
        'date_created' => '<' . ( $today - ($days_delay * $one_day) ),
    ) );

    if ( sizeof($unpaid_orders) > 0 ) {
        $cancelled_text = __("The order was cancelled due to no payment from customer.", "woocommerce");

        // Loop through orders
        foreach ( $unpaid_orders as $order ) {
            $order->update_status( 'cancelled', $cancelled_text );
        }
    }
}

代码位于活动子主题(或活动主题)的function.php文件中。

功能代码应该更好用。对于我真的不知道的钩子。


4)您也可以尝试使用woocommerce_cancel_unpaid_orders动作钩子。


0
投票

我需要在创建订单4小时后取消特定情况下的暂停订单。

示例:2个客户

第一个客户在下午5:00订购,因此其暂停状态(我希望在订单创建之日起4小时后取消,如果他没有付款,那么它必须在晚上9点取消)

第二次客户订单在5:37所以它的暂停状态(我希望它在订单创建之日起4小时后取消,如果他没有付款,那么它必须在9:37取消)

我希望它在订单创建日期后4小时后立即取消订单,而不是作为cron作业每4小时工作一次,所以在上一个例子中,如果cron事件在下午6:00开始,将每4小时运行一次,以便第一个客户订单将在晚上10:00到期,第二个客户订单也将在晚上10:00到期。

我试过上面的解决方案

但不能让它在4小时后过期,因为它只在几天内定义,也不知道哪个答案对我来说是最好的方案,每个订单在创建之日起4个小时后单独取消而不是一个cron事件,

我使用了上面相关问题中提到的许多解决方案但只有第一个答案,因为我记得当管理员/商店经理访问订单页面时...我需要它自动完成而不需要人工做任何动作。

第三个解决方案经过测试并且工作正常:但是我希望它在订单处于暂停状态未完成/处理时被取消,并且如果在订单创建日期后4小时通过则要取消,并且每个解决方案都要安排并运行30分钟不是每天

© www.soinside.com 2019 - 2024. All rights reserved.