mirror of
https://github.com/pronamic/woocommerce-subscriptions.git
synced 2025-10-12 20:32:55 +00:00
244 lines
9.6 KiB
PHP
244 lines
9.6 KiB
PHP
<?php
|
|
/**
|
|
* A class to make it possible to limit a subscription product.
|
|
*
|
|
* @package WooCommerce Subscriptions
|
|
* @category Class
|
|
* @since 2.1
|
|
*/
|
|
class WCS_Limiter {
|
|
|
|
/* cache whether a given product is purchasable or not to save running lots of queries for the same product in the same request */
|
|
protected static $is_purchasable_cache = array();
|
|
|
|
/* cache the check on whether the session has an order awaiting payment for a given product */
|
|
protected static $order_awaiting_payment_for_product = array();
|
|
|
|
public static function init() {
|
|
|
|
//Add limiting subscriptions options on edit product page
|
|
add_action( 'woocommerce_product_options_advanced', __CLASS__ . '::admin_edit_product_fields' );
|
|
|
|
add_filter( 'woocommerce_subscription_is_purchasable', __CLASS__ . '::is_purchasable_switch', 12, 2 );
|
|
|
|
add_filter( 'woocommerce_subscription_variation_is_purchasable', __CLASS__ . '::is_purchasable_switch', 12, 2 );
|
|
|
|
add_filter( 'woocommerce_subscription_is_purchasable', __CLASS__ . '::is_purchasable_renewal', 12, 2 );
|
|
|
|
add_filter( 'woocommerce_subscription_variation_is_purchasable', __CLASS__ . '::is_purchasable_renewal', 12, 2 );
|
|
|
|
}
|
|
|
|
/**
|
|
* Adds limit options to 'Edit Product' screen.
|
|
*
|
|
* @since 2.1 Moved from WC_Subscriptions_Admin
|
|
*/
|
|
public static function admin_edit_product_fields() {
|
|
global $post;
|
|
|
|
echo '<div class="options_group limit_subscription show_if_subscription show_if_variable-subscription">';
|
|
|
|
// Only one Subscription per customer
|
|
woocommerce_wp_select( array(
|
|
'id' => '_subscription_limit',
|
|
'label' => __( 'Limit subscription', 'woocommerce-subscriptions' ),
|
|
// translators: placeholders are opening and closing link tags
|
|
'description' => sprintf( __( 'Only allow a customer to have one subscription to this product. %sLearn more%s.', 'woocommerce-subscriptions' ), '<a href="http://docs.woocommerce.com/document/subscriptions/store-manager-guide/#limit-subscription">', '</a>' ),
|
|
'options' => array(
|
|
'no' => __( 'Do not limit', 'woocommerce-subscriptions' ),
|
|
'active' => __( 'Limit to one active subscription', 'woocommerce-subscriptions' ),
|
|
'any' => __( 'Limit to one of any status', 'woocommerce-subscriptions' ),
|
|
),
|
|
) );
|
|
echo '</div>';
|
|
|
|
do_action( 'woocommerce_subscriptions_product_options_advanced' );
|
|
}
|
|
|
|
/**
|
|
* Canonical is_purchasable method to be called by product classes.
|
|
*
|
|
* @since 2.1
|
|
* @param bool $purchasable Whether the product is purchasable as determined by parent class
|
|
* @param mixed $product The product in question to be checked if it is purchasable.
|
|
* @param string $product_class Determines the subscription type of the product. Controls switch logic.
|
|
*
|
|
* @return bool
|
|
*/
|
|
public static function is_purchasable( $purchasable, $product ) {
|
|
switch ( $product->get_type() ) {
|
|
case 'subscription' :
|
|
case 'variable-subscription' :
|
|
if ( true === $purchasable && false === self::is_purchasable_product( $purchasable, $product ) ) {
|
|
$purchasable = false;
|
|
}
|
|
break;
|
|
case 'subscription_variation' :
|
|
$variable_product = wc_get_product( $product->get_parent_id() );
|
|
if ( 'no' != wcs_get_product_limitation( $variable_product ) && ! empty( WC()->cart->cart_contents ) && ! wcs_is_order_received_page() && ! wcs_is_paypal_api_page() ) {
|
|
foreach ( WC()->cart->cart_contents as $cart_item ) {
|
|
// If the variable product is limited, it can't be purchased if its the same variation
|
|
if ( $product->get_parent_id() == wcs_get_objects_property( $cart_item['data'], 'parent_id' ) && $product->get_id() != $cart_item['data']->get_id() ) {
|
|
$purchasable = false;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
return $purchasable;
|
|
}
|
|
|
|
|
|
/**
|
|
* If a product is limited and the customer already has a subscription, mark it as not purchasable.
|
|
*
|
|
* @since 2.1 Moved from WC_Subscriptions_Product
|
|
* @return bool
|
|
*/
|
|
public static function is_purchasable_product( $is_purchasable, $product ) {
|
|
|
|
//Set up cache
|
|
if ( ! isset( self::$is_purchasable_cache[ $product->get_id() ] ) ) {
|
|
self::$is_purchasable_cache[ $product->get_id() ] = array();
|
|
}
|
|
|
|
if ( ! isset( self::$is_purchasable_cache[ $product->get_id() ]['standard'] ) ) {
|
|
self::$is_purchasable_cache[ $product->get_id() ]['standard'] = $is_purchasable;
|
|
|
|
if ( WC_Subscriptions_Product::is_subscription( $product->get_id() ) && 'no' != wcs_get_product_limitation( $product ) && ! wcs_is_order_received_page() && ! wcs_is_paypal_api_page() ) {
|
|
|
|
if ( wcs_is_product_limited_for_user( $product ) && ! self::order_awaiting_payment_for_product( $product->get_id() ) ) {
|
|
self::$is_purchasable_cache[ $product->get_id() ]['standard'] = false;
|
|
}
|
|
}
|
|
}
|
|
return self::$is_purchasable_cache[ $product->get_id() ]['standard'];
|
|
|
|
}
|
|
|
|
/**
|
|
* If a product is being marked as not purchasable because it is limited and the customer has a subscription,
|
|
* but the current request is to switch the subscription, then mark it as purchasable.
|
|
*
|
|
* @since 2.1 Moved from WC_Subscriptions_Switcher::is_purchasable
|
|
* @return bool
|
|
*/
|
|
public static function is_purchasable_switch( $is_purchasable, $product ) {
|
|
$product_key = wcs_get_canonical_product_id( $product );
|
|
|
|
if ( ! isset( self::$is_purchasable_cache[ $product_key ] ) ) {
|
|
self::$is_purchasable_cache[ $product_key ] = array();
|
|
}
|
|
|
|
if ( ! isset( self::$is_purchasable_cache[ $product_key ]['switch'] ) ) {
|
|
|
|
if ( false === $is_purchasable && wcs_is_product_switchable_type( $product ) && WC_Subscriptions_Product::is_subscription( $product->get_id() ) && 'no' != wcs_get_product_limitation( $product ) && is_user_logged_in() && wcs_user_has_subscription( 0, $product->get_id(), wcs_get_product_limitation( $product ) ) ) {
|
|
|
|
//Adding to cart
|
|
if ( isset( $_GET['switch-subscription'] ) ) {
|
|
$is_purchasable = true;
|
|
|
|
//Validating when restring cart from session
|
|
} elseif ( WC_Subscriptions_Switcher::cart_contains_switches() ) {
|
|
$is_purchasable = true;
|
|
|
|
// Restoring cart from session, so need to check the cart in the session (WC_Subscriptions_Switcher::cart_contains_subscription_switch() only checks the cart)
|
|
} elseif ( isset( WC()->session->cart ) ) {
|
|
|
|
foreach ( WC()->session->cart as $cart_item_key => $cart_item ) {
|
|
if ( $product->get_id() == $cart_item['product_id'] && isset( $cart_item['subscription_switch'] ) ) {
|
|
$is_purchasable = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
self::$is_purchasable_cache[ $product_key ]['switch'] = $is_purchasable;
|
|
}
|
|
return self::$is_purchasable_cache[ $product_key ]['switch'];
|
|
}
|
|
|
|
/**
|
|
* Determines whether a product is purchasable based on whether the cart is to resubscribe or renew.
|
|
*
|
|
* @since 2.1 Combines WCS_Cart_Renewal::is_purchasable and WCS_Cart_Resubscribe::is_purchasable
|
|
* @return bool
|
|
*/
|
|
public static function is_purchasable_renewal( $is_purchasable, $product ) {
|
|
if ( false === $is_purchasable && false === self::is_purchasable_product( $is_purchasable, $product ) ) {
|
|
|
|
// Resubscribe logic
|
|
if ( isset( $_GET['resubscribe'] ) || false !== ( $resubscribe_cart_item = wcs_cart_contains_resubscribe() ) ) {
|
|
$subscription_id = ( isset( $_GET['resubscribe'] ) ) ? absint( $_GET['resubscribe'] ) : $resubscribe_cart_item['subscription_resubscribe']['subscription_id'];
|
|
$subscription = wcs_get_subscription( $subscription_id );
|
|
|
|
if ( false != $subscription && $subscription->has_product( $product->get_id() ) && wcs_can_user_resubscribe_to( $subscription ) ) {
|
|
$is_purchasable = true;
|
|
}
|
|
|
|
// Renewal logic
|
|
} elseif ( isset( $_GET['subscription_renewal'] ) || wcs_cart_contains_renewal() ) {
|
|
$is_purchasable = true;
|
|
|
|
// Restoring cart from session, so need to check the cart in the session (wcs_cart_contains_renewal() only checks the cart)
|
|
} elseif ( WC()->session->cart ) {
|
|
foreach ( WC()->session->cart as $cart_item_key => $cart_item ) {
|
|
if ( $product->get_id() == $cart_item['product_id'] && ( isset( $cart_item['subscription_renewal'] ) || isset( $cart_item['subscription_resubscribe'] ) ) ) {
|
|
$is_purchasable = true;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
return $is_purchasable;
|
|
}
|
|
|
|
/**
|
|
* Check if the current session has an order awaiting payment for a subscription to a specific product line item.
|
|
*
|
|
* @since 2.1 Moved from WC_Subscriptions_Product
|
|
* @return bool
|
|
**/
|
|
protected static function order_awaiting_payment_for_product( $product_id ) {
|
|
global $wp;
|
|
|
|
if ( ! isset( self::$order_awaiting_payment_for_product[ $product_id ] ) ) {
|
|
|
|
self::$order_awaiting_payment_for_product[ $product_id ] = false;
|
|
|
|
if ( ! empty( WC()->session->order_awaiting_payment ) || isset( $_GET['pay_for_order'] ) ) {
|
|
|
|
$order_id = ! empty( WC()->session->order_awaiting_payment ) ? WC()->session->order_awaiting_payment : $wp->query_vars['order-pay'];
|
|
$order = wc_get_order( absint( $order_id ) );
|
|
|
|
if ( is_object( $order ) && $order->has_status( array( 'pending', 'failed' ) ) ) {
|
|
foreach ( $order->get_items() as $item ) {
|
|
if ( $item['product_id'] == $product_id || $item['variation_id'] == $product_id ) {
|
|
|
|
$subscriptions = wcs_get_subscriptions( array(
|
|
'order_id' => wcs_get_objects_property( $order, 'id' ),
|
|
'product_id' => $product_id,
|
|
) );
|
|
|
|
if ( ! empty( $subscriptions ) ) {
|
|
$subscription = array_pop( $subscriptions );
|
|
|
|
if ( $subscription->has_status( array( 'pending', 'on-hold' ) ) ) {
|
|
self::$order_awaiting_payment_for_product[ $product_id ] = true;
|
|
}
|
|
}
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
return self::$order_awaiting_payment_for_product[ $product_id ];
|
|
}
|
|
|
|
}
|
|
WCS_Limiter::init();
|