query_vars['order-pay'] ) || ! wcs_is_subscription( absint( $wp->query_vars['order-pay'] ) ) ) ) {
return;
}
/*
* Clear the output buffer.
*
* Because this function is hooked onto 'after_woocommerce_pay', WC would have started outputting
* the core order pay shortcode. Clearing the output buffer removes that partially outputted template.
*/
ob_clean();
// Because we've cleared the buffer, we need to re-include the opening container div.
echo '
';
// If the request to pay for the order belongs to a subscription but there's no GET params for changing payment method, show receipt page.
if ( ! self::$is_request_to_change_payment ) {
$valid_request = true;
$subscription = wcs_get_subscription( absint( $wp->query_vars['order-pay'] ) );
$subscription_key = isset( $_GET['key'] ) ? wc_clean( $_GET['key'] ) : '';
do_action( 'before_woocommerce_pay' );
/**
* wcs_before_replace_pay_shortcode
*
* Action that allows payment methods to modify the subscription object so, for example,
* if the new payment method still hasn't been set, they can set it temporarily (without saving).
*
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v2.4.0
*
* @param WC_Subscription $subscription
*/
do_action( 'wcs_before_replace_pay_shortcode', $subscription );
if ( $subscription && $subscription->get_id() === absint( $wp->query_vars['order-pay'] ) && $subscription->get_order_key() === $subscription_key ) {
WCS_Template_Loader::get_subscription_receipt_template( $subscription );
} else {
// The before_woocommerce_pay action would have printed all the notices so we need to print the notice directly.
wc_print_notice( __( 'Sorry, this subscription change payment method request is invalid and cannot be processed.', 'woocommerce-subscriptions' ), 'error' );
}
} else {
// Re-add all the notices that would have been displayed but have now been cleared from the output.
foreach ( self::$notices as $notice_type => $notices ) {
foreach ( $notices as $notice ) {
if ( wcs_is_woocommerce_pre( '3.9' ) ) {
wc_add_notice( $notice, $notice_type );
} else {
wc_add_notice( $notice['notice'], $notice_type, $notice['data'] );
}
}
}
$subscription = wcs_get_subscription( absint( $_GET['change_payment_method'] ) );
$valid_request = self::validate_change_payment_request( $subscription );
// WC display notices on this hook so trigger it after all notices have been added,
do_action( 'before_woocommerce_pay' );
if ( $valid_request ) {
if ( $subscription->get_time( 'next_payment' ) > 0 ) {
// translators: placeholder is next payment's date
$next_payment_string = sprintf( __( ' Next payment is due %s.', 'woocommerce-subscriptions' ), $subscription->get_date_to_display( 'next_payment' ) );
} else {
$next_payment_string = '';
}
// translators: placeholder is either empty or "Next payment is due..."
wc_print_notice( apply_filters( 'woocommerce_subscriptions_change_payment_method_page_notice_message', sprintf( __( 'Choose a new payment method.%s', 'woocommerce-subscriptions' ), $next_payment_string ), $subscription ), 'notice' );
// Set the customer location to subscription billing location
foreach ( array( 'country', 'state', 'postcode' ) as $address_property ) {
$subscription_address = $subscription->{"get_billing_$address_property"}();
if ( $subscription_address ) {
WC()->customer->{"set_billing_$address_property"}( $subscription_address );
}
}
wc_get_template( 'checkout/form-change-payment-method.php', array( 'subscription' => $subscription ), '', WC_Subscriptions_Plugin::instance()->get_plugin_directory( 'templates/' ) );
}
}
if ( false === $valid_request ) {
wc_print_notices();
}
}
/**
* Validates the request to change a subscription's payment method.
*
* Will display a customer facing notice if the request is invalid.
*
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v3.0.0
*
* @param WC_Subscription $subscription
* @return bool Whether the request is valid or not.
*/
private static function validate_change_payment_request( $subscription = null ) {
$is_valid = true;
if ( wp_verify_nonce( wc_clean( wp_unslash( $_GET['_wpnonce'] ) ) ) === false ) {
$is_valid = false;
wc_add_notice( __( 'There was an error with your request. Please try again.', 'woocommerce-subscriptions' ), 'error' );
} elseif ( empty( $subscription ) ) {
$is_valid = false;
wc_add_notice( __( 'Invalid Subscription.', 'woocommerce-subscriptions' ), 'error' );
} elseif ( ! current_user_can( 'edit_shop_subscription_payment_method', $subscription->get_id() ) ) {
$is_valid = false;
wc_add_notice( __( 'That doesn\'t appear to be one of your subscriptions.', 'woocommerce-subscriptions' ), 'error' );
} elseif ( ! $subscription->can_be_updated_to( 'new-payment-method' ) ) {
$is_valid = false;
wc_add_notice( __( 'The payment method can not be changed for that subscription.', 'woocommerce-subscriptions' ), 'error' );
} elseif ( $subscription->get_order_key() !== $_GET['key'] ) {
$is_valid = false;
wc_add_notice( __( 'Invalid order.', 'woocommerce-subscriptions' ), 'error' );
}
return $is_valid;
}
/**
* Add a "Change Payment Method" button to the "My Subscriptions" table.
*
* @param array $actions The $subscription_key => $actions array with all actions that will be displayed for a subscription on the "My Subscriptions" table
* @param WC_Subscription $subscription The subscription.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function change_payment_method_button( $actions, $subscription ) {
if ( $subscription->can_be_updated_to( 'new-payment-method' ) ) {
if ( $subscription->has_payment_gateway() && wc_get_payment_gateway_by_order( $subscription )->supports( 'subscriptions' ) ) {
$action_name = _x( 'Change payment', 'label on button, imperative', 'woocommerce-subscriptions' );
} else {
$action_name = _x( 'Add payment', 'label on button, imperative', 'woocommerce-subscriptions' );
}
$actions['change_payment_method'] = array(
'url' => wp_nonce_url( add_query_arg( array( 'change_payment_method' => $subscription->get_id() ), $subscription->get_checkout_payment_url() ) ),
'name' => $action_name,
);
}
return $actions;
}
/**
* Process the change payment form.
*
* Based on the @see woocommerce_pay_action() function.
*
* @return void
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function change_payment_method_via_pay_shortcode() {
if ( ! isset( $_POST['_wcsnonce'] ) || ! wp_verify_nonce( wc_clean( wp_unslash( $_POST['_wcsnonce'] ) ), 'wcs_change_payment_method' ) ) {
return;
}
$subscription_id = absint( wc_clean( wp_unslash( $_POST['woocommerce_change_payment'] ) ) );
$subscription = wcs_get_subscription( $subscription_id );
do_action( 'woocommerce_subscription_change_payment_method_via_pay_shortcode', $subscription );
ob_start();
if ( $subscription->get_order_key() == wc_clean( wp_unslash( $_GET['key'] ) ) ) {
$subscription_billing_country = $subscription->get_billing_country();
$subscription_billing_state = $subscription->get_billing_state();
$subscription_billing_postcode = $subscription->get_billing_postcode();
$subscription_billing_city = $subscription->get_billing_city();
// Set customer location to order location
if ( $subscription_billing_country ) {
$setter = is_callable( array( WC()->customer, 'set_billing_country' ) ) ? 'set_billing_country' : 'set_country';
WC()->customer->$setter( $subscription_billing_country );
}
if ( $subscription_billing_state ) {
$setter = is_callable( array( WC()->customer, 'set_billing_state' ) ) ? 'set_billing_state' : 'set_state';
WC()->customer->$setter( $subscription_billing_state );
}
if ( $subscription_billing_postcode ) {
$setter = is_callable( array( WC()->customer, 'set_billing_postcode' ) ) ? 'set_billing_postcode' : 'set_postcode';
WC()->customer->$setter( $subscription_billing_postcode );
}
if ( $subscription_billing_city ) {
$setter = is_callable( array( WC()->customer, 'set_billing_city' ) ) ? 'set_billing_city' : 'set_city';
WC()->customer->$setter( $subscription_billing_city );
}
// For each new change payment request, make sure we delete the delayed update payment method meta if it exists.
$subscription->delete_meta_data( '_delayed_update_payment_method_all' );
$subscription->save_meta_data();
// Update payment method
$new_payment_method = wc_clean( $_POST['payment_method'] );
$notice = $subscription->has_payment_gateway() ? __( 'Payment method updated.', 'woocommerce-subscriptions' ) : __( 'Payment method added.', 'woocommerce-subscriptions' );
// Allow some payment gateways which can't process the payment immediately, like PayPal, to do it later after the payment/sign-up is confirmed
if ( apply_filters( 'woocommerce_subscriptions_update_payment_via_pay_shortcode', true, $new_payment_method, $subscription ) ) {
self::update_payment_method( $subscription, $new_payment_method );
}
$available_gateways = WC()->payment_gateways->get_available_payment_gateways();
// Validate
$available_gateways[ $new_payment_method ]->validate_fields();
// Process payment for the new method (with a $0 order total)
if ( wc_notice_count( 'error' ) == 0 ) {
try {
$result = $available_gateways[ $new_payment_method ]->process_payment( $subscription->get_id() );
} catch ( Exception $e ) {
wc_add_notice( $e->getMessage(), 'error' );
return;
}
if ( 'success' == $result['result'] && wc_get_page_permalink( 'myaccount' ) == $result['redirect'] ) {
$result['redirect'] = $subscription->get_view_order_url();
}
$result = apply_filters( 'woocommerce_subscriptions_process_payment_for_change_method_via_pay_shortcode', $result, $subscription );
if ( 'success' != $result['result'] ) {
return;
}
/**
* After processing the payment result, make sure we get a new instance of the subscription.
*
* Because process_payment() is sent an ID, all subscription meta changes would occur on a different instance on the subscription.
* We need a new instance to ensure we have the latest changes when processing the update all subscription payment method request below.
*/
$subscription = wcs_get_subscription( $subscription->get_id() );
$subscription->set_requires_manual_renewal( false );
$subscription->save();
// Does the customer want all current subscriptions to be updated to this payment method?
if (
isset( $_POST['update_all_subscriptions_payment_method'] )
&& wc_clean( wp_unslash( $_POST['update_all_subscriptions_payment_method'] ) )
&& WC_Subscriptions_Change_Payment_Gateway::can_update_all_subscription_payment_methods( $available_gateways[ $new_payment_method ], $subscription )
) {
// Allow some payment gateways which can't process the payment immediately, like PayPal, to do it later after the payment/sign-up is confirmed
if ( ! apply_filters( 'woocommerce_subscriptions_update_payment_via_pay_shortcode', true, $new_payment_method, $subscription ) ) {
$subscription->update_meta_data( '_delayed_update_payment_method_all', $new_payment_method );
$subscription->save();
$notice = __( 'Payment method updated for all your current subscriptions.', 'woocommerce-subscriptions' );
} elseif ( self::update_all_payment_methods_from_subscription( $subscription, $new_payment_method ) ) {
$notice = __( 'Payment method updated for all your current subscriptions.', 'woocommerce-subscriptions' );
}
}
// Redirect to success/confirmation/payment page
wc_add_notice( $notice );
wp_safe_redirect( $result['redirect'] );
exit;
}
}
ob_get_clean();
}
/**
* Update the recurring payment method on all current subscriptions to the payment method on this subscription.
*
* @param WC_Subscription $subscription An instance of a WC_Subscription object.
* @param string $new_payment_method The ID of the new payment method.
* @return bool Were other subscriptions updated.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v2.5.0
*/
public static function update_all_payment_methods_from_subscription( $subscription, $new_payment_method ) {
// Require the delayed payment update method to match the current gateway if it is set
if ( self::will_subscription_update_all_payment_methods( $subscription ) ) {
if ( $subscription->get_meta( '_delayed_update_payment_method_all' ) != $new_payment_method ) {
return false;
}
$subscription->delete_meta_data( '_delayed_update_payment_method_all' );
$subscription->save_meta_data();
}
$payment_meta_table = WCS_Payment_Tokens::get_subscription_payment_meta( $subscription, $new_payment_method );
if ( ! is_array( $payment_meta_table ) ) {
return false;
}
$subscription_ids = WCS_Customer_Store::instance()->get_users_subscription_ids( $subscription->get_customer_id() );
foreach ( $subscription_ids as $subscription_id ) {
// Skip the subscription providing the new payment meta.
if ( $subscription->get_id() == $subscription_id ) {
continue;
}
$user_subscription = wcs_get_subscription( $subscription_id );
if ( ! $user_subscription instanceof WC_Subscription ) {
continue;
}
// Skip if subscription's current payment method is not supported
if ( ! $user_subscription->payment_method_supports( 'subscription_cancellation' ) ) {
continue;
}
// Skip if there are no remaining payments or the subscription is not current.
if ( $user_subscription->get_time( 'next_payment' ) <= 0 || ! $user_subscription->has_status( array( 'active', 'on-hold' ) ) ) {
continue;
}
// Clear any stale _delayed_update_payment_method_all meta existing on the users other subscriptions if it exists.
$user_subscription->delete_meta_data( '_delayed_update_payment_method_all' );
self::update_payment_method( $user_subscription, $new_payment_method, $payment_meta_table );
$user_subscription->set_requires_manual_renewal( false );
$user_subscription->save();
}
return true;
}
/**
* Check whether a payment method supports updating all current subscriptions' payment method.
*
* @param WC_Payment_Gateway $gateway The payment gateway to check.
* @param WC_Subscription $subscription An instance of a WC_Subscription object.
* @return bool Gateway supports updating all current subscriptions.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v2.5.0
*/
public static function can_update_all_subscription_payment_methods( $gateway, $subscription ) {
if ( $gateway->supports( 'subscription_payment_method_delayed_change' ) ) {
return true;
}
if ( ! $gateway->supports( 'subscription_payment_method_change_admin' ) ) {
return false;
}
if ( apply_filters( 'woocommerce_subscriptions_update_payment_via_pay_shortcode', true, $gateway->id, $subscription ) ) {
return true;
}
return false;
}
/**
* Check whether a subscription will update all current subscriptions' payment method.
*
* @param WC_Subscription $subscription An instance of a WC_Subscription object.
* @return bool Subscription will update all current subscriptions.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v2.5.0
*/
public static function will_subscription_update_all_payment_methods( $subscription ) {
if ( ! wcs_is_subscription( $subscription ) ) {
return false;
}
return (bool) $subscription->get_meta( '_delayed_update_payment_method_all' );
}
/**
* Update the recurring payment method on a subscription order.
*
* @param WC_Subscription $subscription An instance of a WC_Subscription object.
* @param string $new_payment_method The ID of the new payment method.
* @param array $new_payment_method_meta The meta for the new payment method. Optional. Default false.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function update_payment_method( $subscription, $new_payment_method, $new_payment_method_meta = [] ) {
$old_payment_method = $subscription->get_payment_method();
$old_payment_method_title = $subscription->get_payment_method_title();
$available_gateways = WC()->payment_gateways->get_available_payment_gateways(); // Also inits all payment gateways to make sure that hooks are attached correctly
$payment_gateways_handler = WC_Subscriptions_Core_Plugin::instance()->get_gateways_handler_class();
do_action( 'woocommerce_subscriptions_pre_update_payment_method', $subscription, $new_payment_method, $old_payment_method );
// Make sure the subscription is cancelled with the current gateway
$payment_gateways_handler::trigger_gateway_status_updated_hook( $subscription, 'cancelled' );
// Update meta
if ( isset( $available_gateways[ $new_payment_method ] ) ) {
$new_payment_method_title = $available_gateways[ $new_payment_method ]->get_title();
} else {
$new_payment_method_title = '';
}
if ( empty( $old_payment_method_title ) ) {
$old_payment_method_title = $old_payment_method;
}
if ( empty( $new_payment_method_title ) ) {
$new_payment_method_title = $new_payment_method;
}
// Changing the payment method can throw an exception via set_payment_method() -> set_payment_method_meta(). Catch it and display an error.
try {
$subscription->set_payment_method( $new_payment_method, $new_payment_method_meta );
$subscription->set_payment_method_title( $new_payment_method_title );
$subscription->update_meta_data( '_old_payment_method', $old_payment_method );
$subscription->update_meta_data( '_old_payment_method_title', $old_payment_method_title );
// Allow third-parties to filter the payment method titles used in the subscription note.
$old_payment_method_title = (string) apply_filters( 'woocommerce_subscription_note_old_payment_method_title', $old_payment_method_title, $old_payment_method, $subscription );
$new_payment_method_title = (string) apply_filters( 'woocommerce_subscription_note_new_payment_method_title', $new_payment_method_title, $new_payment_method, $subscription );
// Log change on order
// translators: 1: old payment title, 2: new payment title.
$subscription->add_order_note( sprintf( _x( 'Payment method changed from "%1$s" to "%2$s" by the subscriber.', '%1$s: old payment title, %2$s: new payment title', 'woocommerce-subscriptions' ), $old_payment_method_title, $new_payment_method_title ) );
$subscription->save();
do_action( 'woocommerce_subscription_payment_method_updated', $subscription, $new_payment_method, $old_payment_method );
do_action( 'woocommerce_subscription_payment_method_updated_to_' . $new_payment_method, $subscription, $old_payment_method );
if ( $old_payment_method ) {
do_action( 'woocommerce_subscription_payment_method_updated_from_' . $old_payment_method, $subscription, $new_payment_method );
}
} catch ( Exception $e ) {
$message = __( "An error occurred updating your subscription's payment method. Please contact us for assistance.", 'woocommerce-subscriptions' );
if ( ! wc_has_notice( $message, 'error' ) ) {
wc_add_notice( $message, 'error' );
}
// Add an error notice specific to this error if it hasn't been added yet. This will generate the unique list of errors which occurred.
$error_message = sprintf(
__( '%1$sError:%2$s %3$s', 'woocommerce-subscriptions' ),
'',
'',
$e->getMessage()
);
if ( ! wc_has_notice( $error_message, 'error' ) ) {
wc_add_notice( $error_message, 'error' );
}
}
}
/**
* Only display gateways which support changing payment method when paying for a failed renewal order or
* when requesting to change the payment method.
*
* @param array $available_gateways The payment gateways which are currently being allowed.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function get_available_payment_gateways( $available_gateways ) {
$is_change_payment_method_request = isset( $_GET['change_payment_method'] );
// If we're on a order-pay page but not changing a subscription's payment method, exit early - we don't want to filter the available payment gateways while the customer pays for an order.
if ( ! $is_change_payment_method_request && is_wc_endpoint_url( 'order-pay' ) ) {
return $available_gateways;
}
$renewal_order_cart_item = wcs_cart_contains_failed_renewal_order_payment();
$cart_contains_failed_renewal = (bool) $renewal_order_cart_item;
$cart_contains_failed_manual_renewal = false;
/**
* If there's no change payment request and the cart contains a failed renewal order, check if the subscription is manual.
*
* We update failing, non-manual subscriptions in @see WC_Subscriptions_Change_Payment_Gateway::change_failing_payment_method() so we
* don't need to apply our available payment gateways filter if the subscription is manual.
*/
if ( ! $is_change_payment_method_request && $cart_contains_failed_renewal ) {
$subscription = wcs_get_subscription( $renewal_order_cart_item['subscription_renewal']['subscription_id'] );
$cart_contains_failed_manual_renewal = $subscription->is_manual();
}
if ( apply_filters( 'wcs_payment_gateways_change_payment_method', $is_change_payment_method_request || ( $cart_contains_failed_renewal && ! $cart_contains_failed_manual_renewal ) ) ) {
foreach ( $available_gateways as $gateway_id => $gateway ) {
if ( true !== $gateway->supports( 'subscription_payment_method_change_customer' ) ) {
unset( $available_gateways[ $gateway_id ] );
}
}
}
return $available_gateways;
}
/**
* Make sure certain totals are set to 0 when the request is to change the payment method without charging anything.
*
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function maybe_zero_total( $total, $subscription ) {
global $wp;
if ( ! empty( $_POST['_wcsnonce'] ) && wp_verify_nonce( wc_clean( wp_unslash( $_POST['_wcsnonce'] ) ), 'wcs_change_payment_method' ) && isset( $_POST['woocommerce_change_payment'] ) && wcs_is_subscription( $subscription ) && $subscription->get_order_key() == $_GET['key'] && $subscription->get_id() == absint( $_POST['woocommerce_change_payment'] ) ) {
$total = 0;
} elseif ( ! self::$is_request_to_change_payment && isset( $wp->query_vars['order-pay'] ) && wcs_is_subscription( absint( $wp->query_vars['order-pay'] ) ) ) {
// if the request to pay for the order belongs to a subscription but there's no GET params for changing payment method, the receipt page is being used to collect credit card details so we still need to $0 the total
$total = 0;
}
return $total;
}
/**
* Redirect back to the "My Account" page instead of the "Thank You" page after changing the payment method.
*
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function get_return_url( $return_url ) {
if ( ! empty( $_POST['_wcsnonce'] ) && wp_verify_nonce( wc_clean( wp_unslash( $_POST['_wcsnonce'] ) ), 'wcs_change_payment_method' ) && isset( $_POST['woocommerce_change_payment'] ) ) {
$return_url = get_permalink( wc_get_page_id( 'myaccount' ) );
}
return $return_url;
}
/**
* Update the recurring payment method for a subscription after a customer has paid for a failed renewal order
* (which usually failed because of an issue with the existing payment, like an expired card or token).
*
* Also trigger a hook for payment gateways to update any meta on the original order for a subscription.
*
* @param WC_Order $renewal_order The order which recorded the successful payment (to make up for the failed automatic payment).
* @param WC_Subscription $subscription The subscription.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function change_failing_payment_method( $renewal_order, $subscription ) {
if ( ! $subscription->is_manual() ) {
if ( ! empty( $_POST['_wcsnonce'] ) && wp_verify_nonce( wc_clean( wp_unslash( $_POST['_wcsnonce'] ) ), 'wcs_change_payment_method' ) && isset( $_POST['payment_method'] ) ) {
$new_payment_method = wc_clean( wp_unslash( $_POST['payment_method'] ) );
} else {
$new_payment_method = wcs_get_objects_property( $renewal_order, 'payment_method' );
}
self::update_payment_method( $subscription, $new_payment_method );
do_action( 'woocommerce_subscription_failing_payment_method_updated', $subscription, $renewal_order );
do_action( 'woocommerce_subscription_failing_payment_method_updated_' . $new_payment_method, $subscription, $renewal_order );
}
}
/**
* Add a 'new-payment-method' handler to the @see WC_Subscription::can_be_updated_to() function
* to determine whether the recurring payment method on a subscription can be changed.
*
* For the recurring payment method to be changeable, the subscription must be active, have future (automatic) payments
* and use a payment gateway which allows the subscription to be cancelled.
*
* @param bool $subscription_can_be_changed Flag of whether the subscription can be changed.
* @param WC_Subscription $subscription The subscription to check.
* @return bool Flag indicating whether the subscription payment method can be updated.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function can_subscription_be_updated_to_new_payment_method( $subscription_can_be_changed, $subscription ) {
// Don't allow if automatic payments are disabled and the toggle is also disabled.
if ( wcs_is_manual_renewal_required() && ! WCS_My_Account_Auto_Renew_Toggle::is_enabled() ) {
return false;
}
// If there's no recurring payment, there's no need to add or update the payment method. Use the 'edit' context so we check the unfiltered total.
if ( $subscription->get_total( 'edit' ) == 0 ) {
return false;
}
// Don't allow if no gateways support changing methods.
$payment_gateways_handler = WC_Subscriptions_Core_Plugin::instance()->get_gateways_handler_class();
if ( ! $payment_gateways_handler::one_gateway_supports( 'subscription_payment_method_change_customer' ) ) {
return false;
}
// Don't allow if there are no remaining payments or the subscription is not active.
if ( $subscription->get_time( 'next_payment' ) <= 0 || ! $subscription->has_status( 'active' ) ) {
return false;
}
// Don't allow on subscription that doesn't support changing methods.
if ( ! $subscription->payment_method_supports( 'subscription_cancellation' ) ) {
return false;
}
return true;
}
/**
* Replace a page title with the endpoint title
*
* @param string $title
* @return string
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v2.0
*/
public static function change_payment_method_page_title( $title ) {
global $wp;
// Skip if not on checkout pay page or not a payment change request.
if ( ! self::$is_request_to_change_payment || ! is_main_query() || ! in_the_loop() || ! is_page() || ! is_checkout_pay_page() ) {
return $title;
}
$subscription = wcs_get_subscription( absint( $wp->query_vars['order-pay'] ) );
if ( ! $subscription ) {
return $title;
}
return self::get_change_payment_method_page_title( $subscription );
}
/**
* Replace the breadcrumbs structure to add a link to the subscription page and change the current page to "Change Payment Method"
*
* @param array $crumbs
* @return array
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v2.4.2
*/
public static function change_payment_method_breadcrumb( $crumbs ) {
if ( is_main_query() && is_page() && is_checkout_pay_page() && self::$is_request_to_change_payment ) {
global $wp_query;
$subscription = wcs_get_subscription( absint( $wp_query->query_vars['order-pay'] ) );
if ( ! $subscription ) {
return $crumbs;
}
$crumbs[1] = array(
get_the_title( wc_get_page_id( 'myaccount' ) ),
get_permalink( wc_get_page_id( 'myaccount' ) ),
);
$crumbs[2] = array(
// translators: %s: order number.
sprintf( _x( 'Subscription #%s', 'hash before order number', 'woocommerce-subscriptions' ), $subscription->get_order_number() ),
esc_url( $subscription->get_view_order_url() ),
);
$crumbs[3] = array(
self::get_change_payment_method_page_title( $subscription ),
'',
);
}
return $crumbs;
}
/**
* Get the Change Payment Method page title (also used for the page breadcrumb)
*
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v4.0.0
* @param WC_Subscription $subscription
* @return string
*/
public static function get_change_payment_method_page_title( $subscription ) {
if ( $subscription->has_payment_gateway() ) {
$title = _x( 'Change payment method', 'the page title of the change payment method form', 'woocommerce-subscriptions' );
} else {
$title = _x( 'Add payment method', 'the page title of the add payment method form', 'woocommerce-subscriptions' );
}
return apply_filters( 'woocommerce_subscriptions_change_payment_method_page_title', $title, $subscription );
}
/**
* When processing a change_payment_method request on a subscription that has a failed or pending renewal,
* we don't want the `$order->needs_payment()` check inside WC_Shortcode_Checkout::order_pay() to pass.
* This is causing `$gateway->payment_fields()` to be called multiple times.
*
* @param bool $needs_payment
* @return bool
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v2.0.7
*/
public static function maybe_override_needs_payment( $needs_payment ) {
if ( $needs_payment && self::$is_request_to_change_payment ) {
$needs_payment = false;
}
return $needs_payment;
}
/**
* Display a login form on the change payment method page if the customer isn't logged in.
*
* @param string $content The default HTML page content.
* @return string $content.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v2.5.0
*/
public static function maybe_request_log_in( $content ) {
global $wp;
if ( ! self::$is_request_to_change_payment || is_user_logged_in() || ! isset( $wp->query_vars['order-pay'] ) ) {
return $content;
}
$subscription = wcs_get_subscription( absint( $wp->query_vars['order-pay'] ) );
if ( $subscription ) {
wc_add_notice( __( 'Please log in to your account below to choose a new payment method for your subscription.', 'woocommerce-subscriptions' ), 'notice' );
ob_start();
woocommerce_login_form( array(
'redirect' => $subscription->get_change_payment_method_url(),
'message' => wc_print_notices( true ),
) );
$content = ob_get_clean();
}
return $content;
}
/** Deprecated Functions **/
/**
* Update the recurring payment method on a subscription order.
*
* @param string $subscription_key The subscription key.
* @param WC_Order $order The order.
* @param string $new_payment_method The new payment method.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
* @deprecated 1.0.0 - Migrated from WooCommerce Subscriptions v2.0
*/
public static function update_recurring_payment_method( $subscription_key, $order, $new_payment_method ) {
_deprecated_function( __METHOD__, '2.0', __CLASS__ . '::update_payment_method()' );
self::update_payment_method( wcs_get_subscription_from_key( $subscription_key ), $new_payment_method );
}
/**
* Keep a record of an order's dates if we're marking it as completed during a request to change the payment method.
*
* Deprecated as we now operate on a WC_Subscription object instead of the parent order, so we don't need to hack around date changes.
*
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
* @deprecated 1.0.0 - Migrated from WooCommerce Subscriptions v2.0
*/
public static function store_original_order_dates( $new_order_status, $subscription_id ) {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* Restore an order's dates if we marked it as completed during a request to change the payment method.
*
* Deprecated as we now operate on a WC_Subscription object instead of the parent order, so we don't need to hack around date changes.
*
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
* @deprecated 1.0.0 - Migrated from WooCommerce Subscriptions v2.0
*/
public static function restore_original_order_dates( $order_id ) {
_deprecated_function( __METHOD__, '2.0' );
}
/**
* Add a 'new-payment-method' handler to the @see WC_Subscription::can_be_updated_to() function
* to determine whether the recurring payment method on a subscription can be changed.
*
* For the recurring payment method to be changeable, the subscription must be active, have future (automatic) payments
* and use a payment gateway which allows the subscription to be cancelled.
*
* @param bool $subscription_can_be_changed Flag of whether the subscription can be changed to
* @param string $new_status_or_meta The status or meta data you want to change th subscription to. Can be 'active', 'on-hold', 'cancelled', 'expired', 'trash', 'deleted', 'failed', 'new-payment-date' or some other value attached to the 'woocommerce_can_subscription_be_changed_to' filter.
* @param object $args Set of values used in @see WC_Subscriptions_Manager::can_subscription_be_changed_to() for determining if a subscription can be changes, include:
* 'subscription_key' string A subscription key of the form created by @see WC_Subscriptions_Manager::get_subscription_key()
* 'subscription' array Subscription of the form returned by @see WC_Subscriptions_Manager::get_subscription()
* 'user_id' int The ID of the subscriber.
* 'order' WC_Order The order which recorded the successful payment (to make up for the failed automatic payment).
* 'payment_gateway' WC_Payment_Gateway The subscription's recurring payment gateway
* 'order_uses_manual_payments' bool A boolean flag indicating whether the subscription requires manual renewal payment.
* @since 1.0.0 - Migrated from WooCommerce Subscriptions v1.4
*/
public static function can_subscription_be_changed_to( $subscription_can_be_changed, $new_status_or_meta, $args ) {
_deprecated_function( __METHOD__, '2.0', __CLASS__ . '::can_subscription_be_updated_to_new_payment_method()' );
if ( 'new-payment-method' === $new_status_or_meta ) {
$subscription_can_be_changed = wcs_get_subscription_from_key( $args->subscription_key )->can_be_updated_to( 'new-payment-method' );
}
return $subscription_can_be_changed;
}
/**
* Attach WooCommerce version dependent hooks
*
* @since 1.0.0
*
* @deprecated 1.6.4
*/
public static function attach_dependant_hooks() {
_deprecated_function( __METHOD__, '1.6.4' );
if ( wcs_is_woocommerce_pre( '3.0' ) ) {
// If we're changing the payment method, we want to make sure a number of totals return $0 (to prevent payments being processed now)
add_filter( 'woocommerce_order_amount_total', __CLASS__ . '::maybe_zero_total', 11, 2 );
} else {
// If we're changing the payment method, we want to make sure a number of totals return $0 (to prevent payments being processed now)
add_filter( 'woocommerce_subscription_get_total', __CLASS__ . '::maybe_zero_total', 11, 2 );
}
}
}