Автоматически отменяет заказ через X дней, если нет оплаты в WooCommerce

#php #sql #wordpress #woocommerce #orders

#php #sql #wordpress #woocommerce #заказы

Вопрос:

Мне удалось собрать это вместе после поиска в Интернете, и все же это не работает. Моя цель — автоматически отменить ВСЕ заказы со статусом on-hold, независимо от платежного шлюза, если заказ не был оплачен через три дня.

Код явно неполный, и я прошу помощи в его завершении. Я тестировал его, -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') );
            }
        }
    }
}
  

Ответ №1:

Обновление 4

Примечание: в WooCommerce уже есть функция, подключенная к woocommerce_cancel_unpaid_orders action hook, которая отменяет неоплаченные заказы через 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 amp;amp; 'edit.php' === $pagenow 
        amp;amp; 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 $unpaid_order ) {
            $unpaid_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') ) 
        amp;amp; 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 $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 помощью action hook:

 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 action hook.

Ответ №2:

Просто добавляю к блестящему ответу @LoicTheAztec. Я переделал его в плагин и добавил страницу настроек, где вы можете включить / отключить эту функцию, а также изменить количество дней задержки в настройках администратора. Вот код:

 <?php
/**
* Plugin Name: Cancel On-Hold Orders After X Days
* Plugin URI: https://smartairfilters.com
* Description: Cancel orders that are in an 'on-hold' status after a certain number of days. Uses wordpress cron to trigger
* Version: 1.0.0
* Author: Paddy Robertson
* Author URI: https://patrickrobertson.uk
* Requires at least: 5.3
* Requires PHP: 7.0
*/

add_action('woocommerce_cancel_unpaid_orders', 'cancel_onhold_orders');
function cancel_onhold_orders() {
  if (!get_option('cancel_onhold_enable')) {
    // only run if feature is enabled
    return;
  }

  $days_delay = get_option('cancel_onhold_days_delay'); // <=== SET the delay (number of days to wait before cancelation)
  echo "days delay   xxxx " . $days_delay;
  $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(
    'orderby' => 'date',
    'order' => 'DESC',
    '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 );
    }
  }
}

// Settings page
function cancel_onhold_register_settings() {
  register_setting( 'cancel_onhold_group', 'cancel_onhold_days_delay');
  register_setting( 'cancel_onhold_group', 'cancel_onhold_enable');
}

add_action( 'admin_init', 'cancel_onhold_register_settings' );
function cancel_onhold_register_options_page() {
  add_options_page('Cancel On-Hold Orders', 'Cancel On-Hold', 'manage_options', 'cancel_onhold_options_page', 'cancel_onhold_options_page');
}

add_action('admin_menu', 'cancel_onhold_register_options_page');
function cancel_onhold_options_page() {
  ?>
  <div>
    <h2>Cancel On-Hold Orders after X Days</h2>
    <form method="post" action="options.php">
      <?php settings_fields( 'cancel_onhold_group' ); ?>
      <table class="form-table">
        <tr valign="top">
          <th scope="row"><label for="cancel_onhold_enable">Enable/Disable this feature</label></th>
          <td><input type="checkbox" id="cancel_onhold_enable" name="cancel_onhold_enable" value="1" <?php checked(get_option('cancel_onhold_enable')); ?>" /></td>
        </tr>
        <tr valign="top">
          <th scope="row"><label for="cancel_onhold_days_delay">Enter number of days after which on hold orders will be cancelled</label></th>
          <td><input type="number" id="cancel_onhold_days_delay" name="cancel_onhold_days_delay" value="<?php echo get_option('cancel_onhold_days_delay'); ?>" /> days</td>
        </tr>
      </table>
      <?php submit_button(); ?>
    </form>
  </div>
  <?php
}