2011-08-10 17:11:11 +00:00
< ? php
/**
* WooCommerce coupons
2012-08-14 19:42:38 +00:00
*
2012-01-30 19:24:52 +00:00
* The WooCommerce coupons class gets coupon data from storage and checks coupon validity
2011-08-10 17:11:11 +00:00
*
2012-01-27 16:38:39 +00:00
* @ class WC_Coupon
2013-02-20 17:14:46 +00:00
* @ package WooCommerce / Classes
2011-08-10 17:11:11 +00:00
* @ category Class
* @ author WooThemes
*/
2012-01-27 16:38:39 +00:00
class WC_Coupon {
2012-08-14 19:42:38 +00:00
2013-02-18 12:29:10 +00:00
// Coupon message codes
2013-02-18 19:05:23 +00:00
const E_WC_COUPON_INVALID_FILTERED = 100 ;
const E_WC_COUPON_INVALID_REMOVED = 101 ;
const E_WC_COUPON_NOT_YOURS_REMOVED = 102 ;
const E_WC_COUPON_ALREADY_APPLIED = 103 ;
const E_WC_COUPON_ALREADY_APPLIED_INDIV_USE_ONLY = 104 ;
const E_WC_COUPON_NOT_EXIST = 105 ;
const E_WC_COUPON_USAGE_LIMIT_REACHED = 106 ;
const E_WC_COUPON_EXPIRED = 107 ;
const E_WC_COUPON_MIN_SPEND_LIMIT_NOT_MET = 108 ;
const E_WC_COUPON_NOT_APPLICABLE = 109 ;
const E_WC_COUPON_NOT_VALID_SALE_ITEMS = 110 ;
const E_WC_COUPON_PLEASE_ENTER = 111 ;
2014-10-14 14:09:56 +00:00
const E_WC_COUPON_MAX_SPEND_LIMIT_MET = 112 ;
const E_WC_COUPON_EXCLUDED_PRODUCTS = 113 ;
2014-10-14 14:18:31 +00:00
const E_WC_COUPON_EXCLUDED_CATEGORIES = 114 ;
2013-02-18 12:29:10 +00:00
const WC_COUPON_SUCCESS = 200 ;
2013-08-14 20:00:34 +00:00
const WC_COUPON_REMOVED = 201 ;
2013-02-18 12:29:10 +00:00
2012-12-14 21:41:59 +00:00
/** @public string Coupon code. */
public $code ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public int Coupon ID. */
public $id ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public string Type of discount. */
public $type ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public string Type of discount (alias). */
public $discount_type ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public string Coupon amount. */
public $amount ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public string "Yes" if for individual use. */
public $individual_use ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public array Array of product IDs. */
public $product_ids ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public int Coupon usage limit. */
public $usage_limit ;
2012-08-15 17:08:42 +00:00
2013-10-01 10:48:27 +00:00
/** @public int Coupon usage limit per user. */
public $usage_limit_per_user ;
2013-10-18 17:10:55 +00:00
/** @public int Coupon usage limit per item. */
public $limit_usage_to_x_items ;
2012-12-14 21:41:59 +00:00
/** @public int Coupon usage count. */
public $usage_count ;
2012-08-15 17:08:42 +00:00
2013-03-03 17:07:31 +00:00
/** @public string Expiry date. */
2012-12-14 21:41:59 +00:00
public $expiry_date ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public string "yes" if applied before tax. */
public $apply_before_tax ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public string "yes" if coupon grants free shipping. */
public $free_shipping ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public array Array of category ids. */
public $product_categories ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public array Array of category ids. */
public $exclude_product_categories ;
2012-08-15 17:08:42 +00:00
2013-01-29 13:17:52 +00:00
/** @public string "yes" if coupon does NOT apply to items on sale. */
public $exclude_sale_items ;
2012-12-14 21:41:59 +00:00
/** @public string Minimum cart amount. */
public $minimum_amount ;
2012-08-15 17:08:42 +00:00
2014-08-08 11:38:58 +00:00
/** @public string Maximum cart amount. */
public $maximum_amount ;
2012-12-14 21:41:59 +00:00
/** @public string Coupon owner's email. */
public $customer_email ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public array Post meta. */
public $coupon_custom_fields ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public string How much the coupon is worth. */
public $coupon_amount ;
2012-08-14 19:42:38 +00:00
2012-12-19 14:57:46 +00:00
/** @public string Error message. */
public $error_message ;
2012-08-14 19:42:38 +00:00
/**
2012-08-15 17:08:42 +00:00
* Coupon constructor . Loads coupon data .
2012-08-14 19:42:38 +00:00
*
* @ access public
* @ param mixed $code code of the coupon to load
*/
2012-12-14 21:41:59 +00:00
public function __construct ( $code ) {
2012-02-08 15:55:02 +00:00
global $wpdb ;
2012-08-14 19:42:38 +00:00
2013-02-22 08:44:40 +00:00
$this -> code = apply_filters ( 'woocommerce_coupon_code' , $code );
2012-11-27 16:22:47 +00:00
2012-11-08 16:57:53 +00:00
// Coupon data lets developers create coupons through code
2013-02-22 08:44:40 +00:00
$coupon_data = apply_filters ( 'woocommerce_get_shop_coupon_data' , false , $code );
2012-11-08 16:57:53 +00:00
2014-07-30 16:07:30 +00:00
if ( $coupon_data ) {
2012-11-27 16:22:47 +00:00
2013-10-01 10:48:27 +00:00
$this -> id = absint ( $coupon_data [ 'id' ] );
$this -> type = esc_html ( $coupon_data [ 'type' ] );
2014-07-30 16:07:30 +00:00
$this -> amount = esc_html ( $coupon_data [ 'amount' ] ? $coupon_data [ 'amount' ] : $coupon_data [ 'coupon_amount' ] ) ;
$this -> coupon_amount = $this -> amount ;
2013-10-01 10:48:27 +00:00
$this -> individual_use = esc_html ( $coupon_data [ 'individual_use' ] );
$this -> product_ids = is_array ( $coupon_data [ 'product_ids' ] ) ? $coupon_data [ 'product_ids' ] : array ();
$this -> exclude_product_ids = is_array ( $coupon_data [ 'exclude_product_ids' ] ) ? $coupon_data [ 'exclude_product_ids' ] : array ();
$this -> usage_limit = absint ( $coupon_data [ 'usage_limit' ] );
2013-10-11 11:31:32 +00:00
$this -> usage_limit_per_user = isset ( $coupon_data [ 'usage_limit_per_user' ] ) ? absint ( $coupon_data [ 'usage_limit_per_user' ] ) : 0 ;
2013-10-18 17:10:55 +00:00
$this -> limit_usage_to_x_items = isset ( $coupon_data [ 'limit_usage_to_x_items' ] ) ? absint ( $coupon_data [ 'limit_usage_to_x_items' ] ) : '' ;
2013-10-01 10:48:27 +00:00
$this -> usage_count = absint ( $coupon_data [ 'usage_count' ] );
$this -> expiry_date = esc_html ( $coupon_data [ 'expiry_date' ] );
$this -> apply_before_tax = esc_html ( $coupon_data [ 'apply_before_tax' ] );
$this -> free_shipping = esc_html ( $coupon_data [ 'free_shipping' ] );
$this -> product_categories = is_array ( $coupon_data [ 'product_categories' ] ) ? $coupon_data [ 'product_categories' ] : array ();
$this -> exclude_product_categories = is_array ( $coupon_data [ 'exclude_product_categories' ] ) ? $coupon_data [ 'exclude_product_categories' ] : array ();
$this -> exclude_sale_items = esc_html ( $coupon_data [ 'exclude_sale_items' ] );
$this -> minimum_amount = esc_html ( $coupon_data [ 'minimum_amount' ] );
2014-08-08 11:38:58 +00:00
$this -> maximum_amount = esc_html ( $coupon_data [ 'maximum_amount' ] );
2013-10-01 10:48:27 +00:00
$this -> customer_email = esc_html ( $coupon_data [ 'customer_email' ] );
2012-11-27 16:22:47 +00:00
2014-07-30 16:07:30 +00:00
} else {
2012-11-27 16:22:47 +00:00
2014-07-30 16:07:30 +00:00
$coupon_id = $wpdb -> get_var ( $wpdb -> prepare ( apply_filters ( 'woocommerce_coupon_code_query' , " SELECT ID FROM $wpdb->posts WHERE post_title = %s AND post_type = 'shop_coupon' AND post_status = 'publish' " ), $this -> code ) );
2012-11-27 16:22:47 +00:00
2014-07-30 16:07:30 +00:00
if ( ! $coupon_id )
return ;
2012-12-27 18:46:22 +00:00
$coupon = get_post ( $coupon_id );
2013-10-18 17:10:55 +00:00
$this -> post_title = apply_filters ( 'woocommerce_coupon_code' , $coupon -> post_title );
2012-11-08 16:57:53 +00:00
2014-07-30 16:07:30 +00:00
if ( empty ( $coupon ) || $this -> code !== $this -> post_title )
return ;
2012-11-08 16:57:53 +00:00
2014-07-30 16:07:30 +00:00
$this -> id = $coupon -> ID ;
$this -> coupon_custom_fields = get_post_meta ( $this -> id );
2012-11-08 16:57:53 +00:00
2014-07-30 16:07:30 +00:00
$load_data = array (
2013-10-01 10:48:27 +00:00
'discount_type' => 'fixed_cart' ,
'coupon_amount' => 0 ,
'individual_use' => 'no' ,
'product_ids' => '' ,
'exclude_product_ids' => '' ,
'usage_limit' => '' ,
'usage_limit_per_user' => '' ,
2013-10-18 17:10:55 +00:00
'limit_usage_to_x_items' => '' ,
2013-10-01 10:48:27 +00:00
'usage_count' => '' ,
'expiry_date' => '' ,
'apply_before_tax' => 'yes' ,
'free_shipping' => 'no' ,
'product_categories' => array (),
'exclude_product_categories' => array (),
'exclude_sale_items' => 'no' ,
'minimum_amount' => '' ,
2014-08-08 11:38:58 +00:00
'maximum_amount' => '' ,
2013-10-01 10:48:27 +00:00
'customer_email' => array ()
2014-07-30 16:07:30 +00:00
);
2012-11-08 16:57:53 +00:00
2014-07-30 16:07:30 +00:00
foreach ( $load_data as $key => $default )
$this -> $key = isset ( $this -> coupon_custom_fields [ $key ][ 0 ] ) && $this -> coupon_custom_fields [ $key ][ 0 ] !== '' ? $this -> coupon_custom_fields [ $key ][ 0 ] : $default ;
2012-11-08 16:57:53 +00:00
2014-07-30 16:07:30 +00:00
// Alias
$this -> type = $this -> discount_type ;
$this -> amount = $this -> coupon_amount ;
2012-11-08 16:57:53 +00:00
2014-07-30 16:07:30 +00:00
// Formatting
$this -> product_ids = array_filter ( array_map ( 'trim' , explode ( ',' , $this -> product_ids ) ) );
$this -> exclude_product_ids = array_filter ( array_map ( 'trim' , explode ( ',' , $this -> exclude_product_ids ) ) );
$this -> expiry_date = $this -> expiry_date ? strtotime ( $this -> expiry_date ) : '' ;
$this -> product_categories = array_filter ( array_map ( 'trim' , ( array ) maybe_unserialize ( $this -> product_categories ) ) );
$this -> exclude_product_categories = array_filter ( array_map ( 'trim' , ( array ) maybe_unserialize ( $this -> exclude_product_categories ) ) );
2013-02-22 08:44:40 +00:00
$this -> customer_email = array_filter ( array_map ( 'trim' , array_map ( 'strtolower' , ( array ) maybe_unserialize ( $this -> customer_email ) ) ) );
2014-07-30 16:07:30 +00:00
}
2013-02-22 08:44:40 +00:00
2014-07-30 16:07:30 +00:00
do_action ( 'woocommerce_coupon_loaded' , $this );
2011-08-10 17:11:11 +00:00
}
2012-08-14 19:42:38 +00:00
/**
* Check if coupon needs applying before tax .
*
* @ access public
* @ return bool
*/
2012-12-14 21:41:59 +00:00
public function apply_before_tax () {
2012-11-08 16:57:53 +00:00
return $this -> apply_before_tax == 'yes' ? true : false ;
2011-11-19 20:59:16 +00:00
}
2012-08-14 19:42:38 +00:00
/**
* Check if a coupon enables free shipping .
*
* @ access public
2013-11-25 15:00:54 +00:00
* @ return bool
2012-08-14 19:42:38 +00:00
*/
2012-12-14 21:41:59 +00:00
public function enable_free_shipping () {
2012-11-08 16:57:53 +00:00
return $this -> free_shipping == 'yes' ? true : false ;
2011-11-28 16:10:31 +00:00
}
2013-01-29 13:58:57 +00:00
/**
2013-01-29 13:17:52 +00:00
* Check if a coupon excludes sale items .
*
* @ access public
2013-11-25 15:00:54 +00:00
* @ return bool
2013-01-29 13:17:52 +00:00
*/
public function exclude_sale_items () {
return $this -> exclude_sale_items == 'yes' ? true : false ;
}
2012-08-14 19:42:38 +00:00
/**
* Increase usage count fo current coupon .
*
* @ access public
2013-10-01 10:48:27 +00:00
* @ param string $used_by Either user ID or billing email
2012-08-14 19:42:38 +00:00
* @ return void
*/
2013-10-01 10:48:27 +00:00
public function inc_usage_count ( $used_by = '' ) {
2011-09-06 11:11:22 +00:00
$this -> usage_count ++ ;
2012-07-10 15:52:52 +00:00
update_post_meta ( $this -> id , 'usage_count' , $this -> usage_count );
2013-10-01 10:48:27 +00:00
2014-04-24 14:26:34 +00:00
if ( $used_by ) {
2013-10-01 10:48:27 +00:00
add_post_meta ( $this -> id , '_used_by' , strtolower ( $used_by ) );
2014-04-24 14:26:34 +00:00
}
2012-07-10 15:52:52 +00:00
}
2012-08-14 19:42:38 +00:00
/**
* Decrease usage count fo current coupon .
*
* @ access public
2013-10-01 10:48:27 +00:00
* @ param string $used_by Either user ID or billing email
2012-08-14 19:42:38 +00:00
* @ return void
*/
2013-10-01 10:48:27 +00:00
public function dcr_usage_count ( $used_by = '' ) {
global $wpdb ;
2012-07-10 15:52:52 +00:00
$this -> usage_count -- ;
update_post_meta ( $this -> id , 'usage_count' , $this -> usage_count );
2013-10-01 10:48:27 +00:00
// Delete 1 used by meta
$meta_id = $wpdb -> get_var ( $wpdb -> prepare ( " SELECT meta_id FROM $wpdb->postmeta WHERE meta_key = '_used_by' AND meta_value = %s AND post_id = %d LIMIT 1; " , $used_by , $this -> id ) );
2014-04-24 14:26:34 +00:00
if ( $meta_id ) {
2013-10-01 10:48:27 +00:00
delete_metadata_by_mid ( 'post' , $meta_id );
2014-04-24 14:26:34 +00:00
}
2011-08-15 16:48:24 +00:00
}
2012-08-14 19:42:38 +00:00
2012-12-19 14:57:46 +00:00
/**
* Returns the error_message string
*
* @ access public
* @ return string
*/
public function get_error_message () {
return $this -> error_message ;
}
2012-06-10 18:07:19 +00:00
/**
* is_valid function .
*
2013-03-03 17:07:31 +00:00
* Check if a coupon is valid . Return a reason code if invalid . Reason codes :
2012-08-14 19:42:38 +00:00
*
2012-06-10 18:07:19 +00:00
* @ access public
2014-09-07 23:37:55 +00:00
* @ return boolean validity or a WP_Error if not valid
2012-06-10 18:07:19 +00:00
*/
2012-12-14 21:41:59 +00:00
public function is_valid () {
2012-08-14 19:42:38 +00:00
2013-02-15 03:56:35 +00:00
$error_code = null ;
2013-02-18 12:29:10 +00:00
$valid = true ;
$error = false ;
2012-08-14 19:42:38 +00:00
2013-02-18 12:29:10 +00:00
if ( $this -> id ) {
2012-08-14 19:42:38 +00:00
2012-01-17 15:20:04 +00:00
// Usage Limit
2012-11-08 16:57:53 +00:00
if ( $this -> usage_limit > 0 ) {
if ( $this -> usage_count >= $this -> usage_limit ) {
2012-01-17 15:20:04 +00:00
$valid = false ;
2013-02-18 12:29:10 +00:00
$error_code = self :: E_WC_COUPON_USAGE_LIMIT_REACHED ;
2013-10-01 10:48:27 +00:00
}
}
2014-04-15 15:56:35 +00:00
// Per user usage limit - check here if user is logged in (against user IDs)
2013-10-01 10:48:27 +00:00
// Checked again for emails later on in WC_Cart::check_customer_coupons()
if ( $this -> usage_limit_per_user > 0 && is_user_logged_in () ) {
2014-04-24 14:26:34 +00:00
$used_by = ( array ) get_post_meta ( $this -> id , '_used_by' );
2013-10-01 10:48:27 +00:00
$usage_count = sizeof ( array_keys ( $used_by , get_current_user_id () ) );
if ( $usage_count >= $this -> usage_limit_per_user ) {
$valid = false ;
$error_code = self :: E_WC_COUPON_USAGE_LIMIT_REACHED ;
2012-11-08 16:57:53 +00:00
}
}
2012-08-14 19:42:38 +00:00
2012-01-17 15:20:04 +00:00
// Expired
2012-11-08 16:57:53 +00:00
if ( $this -> expiry_date ) {
2012-12-04 20:24:37 +00:00
if ( current_time ( 'timestamp' ) > $this -> expiry_date ) {
2012-01-17 15:20:04 +00:00
$valid = false ;
2013-02-18 12:29:10 +00:00
$error_code = self :: E_WC_COUPON_EXPIRED ;
2012-11-08 16:57:53 +00:00
}
}
2012-08-14 19:42:38 +00:00
2012-03-07 21:38:34 +00:00
// Minimum spend
2012-11-08 16:57:53 +00:00
if ( $this -> minimum_amount > 0 ) {
2013-11-25 14:01:32 +00:00
if ( $this -> minimum_amount > WC () -> cart -> subtotal ) {
2013-02-15 03:56:35 +00:00
$valid = false ;
2013-02-18 12:29:10 +00:00
$error_code = self :: E_WC_COUPON_MIN_SPEND_LIMIT_NOT_MET ;
2012-11-08 16:57:53 +00:00
}
}
2012-08-14 19:42:38 +00:00
2014-08-08 11:38:58 +00:00
// Maximum spend
if ( $this -> maximum_amount > 0 ) {
if ( $this -> maximum_amount < WC () -> cart -> subtotal ) {
$valid = false ;
$error_code = self :: E_WC_COUPON_MAX_SPEND_LIMIT_MET ;
}
}
2011-11-15 22:20:59 +00:00
// Product ids - If a product included is found in the cart then its valid
2012-11-08 16:57:53 +00:00
if ( sizeof ( $this -> product_ids ) > 0 ) {
2012-01-17 15:20:04 +00:00
$valid_for_cart = false ;
2013-11-25 14:01:32 +00:00
if ( sizeof ( WC () -> cart -> get_cart () ) > 0 ) {
foreach ( WC () -> cart -> get_cart () as $cart_item_key => $cart_item ) {
2012-11-27 16:22:47 +00:00
2012-11-08 16:57:53 +00:00
if ( in_array ( $cart_item [ 'product_id' ], $this -> product_ids ) || in_array ( $cart_item [ 'variation_id' ], $this -> product_ids ) || in_array ( $cart_item [ 'data' ] -> get_parent (), $this -> product_ids ) )
$valid_for_cart = true ;
}
}
2012-08-31 08:45:50 +00:00
if ( ! $valid_for_cart ) {
$valid = false ;
2013-02-18 12:29:10 +00:00
$error_code = self :: E_WC_COUPON_NOT_APPLICABLE ;
2012-08-31 08:45:50 +00:00
}
2012-11-08 16:57:53 +00:00
}
2012-08-14 19:42:38 +00:00
2012-03-06 17:27:02 +00:00
// Category ids - If a product included is found in the cart then its valid
2012-11-08 16:57:53 +00:00
if ( sizeof ( $this -> product_categories ) > 0 ) {
2012-03-06 17:27:02 +00:00
$valid_for_cart = false ;
2013-11-25 14:01:32 +00:00
if ( sizeof ( WC () -> cart -> get_cart () ) > 0 ) {
foreach ( WC () -> cart -> get_cart () as $cart_item_key => $cart_item ) {
2012-11-27 16:22:47 +00:00
2012-11-08 16:57:53 +00:00
$product_cats = wp_get_post_terms ( $cart_item [ 'product_id' ], 'product_cat' , array ( " fields " => " ids " ));
2012-11-27 16:22:47 +00:00
if ( sizeof ( array_intersect ( $product_cats , $this -> product_categories ) ) > 0 )
2012-11-08 16:57:53 +00:00
$valid_for_cart = true ;
}
}
2012-08-31 08:45:50 +00:00
if ( ! $valid_for_cart ) {
$valid = false ;
2013-02-18 12:29:10 +00:00
$error_code = self :: E_WC_COUPON_NOT_APPLICABLE ;
2012-08-31 08:45:50 +00:00
}
2012-11-08 16:57:53 +00:00
}
2012-08-14 19:42:38 +00:00
2014-06-25 12:46:47 +00:00
// Exclude Sale Items check for product coupons - valid if a non-sale item is present
if ( 'yes' === $this -> exclude_sale_items && in_array ( $this -> type , array ( 'fixed_product' , 'percent_product' ) ) ) {
$valid_for_cart = false ;
$product_ids_on_sale = wc_get_product_ids_on_sale ();
if ( sizeof ( WC () -> cart -> get_cart () ) > 0 ) {
foreach ( WC () -> cart -> get_cart () as $cart_item_key => $cart_item ) {
if ( sizeof ( array_intersect ( array ( absint ( $cart_item [ 'product_id' ] ), absint ( $cart_item [ 'variation_id' ] ), $cart_item [ 'data' ] -> get_parent () ), $product_ids_on_sale ) ) === 0 ) {
// not on sale
$valid_for_cart = true ;
}
}
}
if ( ! $valid_for_cart ) {
$valid = false ;
$error_code = self :: E_WC_COUPON_NOT_VALID_SALE_ITEMS ;
}
}
2011-11-15 22:20:59 +00:00
// Cart discounts cannot be added if non-eligble product is found in cart
2012-11-08 16:57:53 +00:00
if ( $this -> type != 'fixed_product' && $this -> type != 'percent_product' ) {
2012-08-14 19:42:38 +00:00
2012-03-06 17:27:02 +00:00
// Exclude Products
2012-11-08 16:57:53 +00:00
if ( sizeof ( $this -> exclude_product_ids ) > 0 ) {
2012-01-17 15:20:04 +00:00
$valid_for_cart = true ;
2013-11-25 14:01:32 +00:00
if ( sizeof ( WC () -> cart -> get_cart () ) > 0 ) {
foreach ( WC () -> cart -> get_cart () as $cart_item_key => $cart_item ) {
2012-11-08 16:57:53 +00:00
if ( in_array ( $cart_item [ 'product_id' ], $this -> exclude_product_ids ) || in_array ( $cart_item [ 'variation_id' ], $this -> exclude_product_ids ) || in_array ( $cart_item [ 'data' ] -> get_parent (), $this -> exclude_product_ids ) ) {
$valid_for_cart = false ;
}
}
}
2012-08-31 08:45:50 +00:00
if ( ! $valid_for_cart ) {
$valid = false ;
2014-10-14 14:09:56 +00:00
$error_code = self :: E_WC_COUPON_EXCLUDED_PRODUCTS ;
2012-08-31 08:45:50 +00:00
}
2012-11-08 16:57:53 +00:00
}
2012-08-14 19:42:38 +00:00
2013-01-29 13:17:52 +00:00
// Exclude Sale Items
if ( $this -> exclude_sale_items == 'yes' ) {
$valid_for_cart = true ;
2013-11-25 13:56:59 +00:00
$product_ids_on_sale = wc_get_product_ids_on_sale ();
2013-11-25 14:01:32 +00:00
if ( sizeof ( WC () -> cart -> get_cart () ) > 0 ) {
foreach ( WC () -> cart -> get_cart () as $cart_item_key => $cart_item ) {
2013-01-29 13:17:52 +00:00
if ( in_array ( $cart_item [ 'product_id' ], $product_ids_on_sale , true ) || in_array ( $cart_item [ 'variation_id' ], $product_ids_on_sale , true ) || in_array ( $cart_item [ 'data' ] -> get_parent (), $product_ids_on_sale , true ) ) {
$valid_for_cart = false ;
}
}
}
if ( ! $valid_for_cart ) {
$valid = false ;
2013-02-18 12:29:10 +00:00
$error_code = self :: E_WC_COUPON_NOT_VALID_SALE_ITEMS ;
2013-01-29 13:17:52 +00:00
}
}
2012-03-06 17:27:02 +00:00
// Exclude Categories
2012-11-08 16:57:53 +00:00
if ( sizeof ( $this -> exclude_product_categories ) > 0 ) {
2012-03-06 17:27:02 +00:00
$valid_for_cart = true ;
2013-11-25 14:01:32 +00:00
if ( sizeof ( WC () -> cart -> get_cart () ) > 0 ) {
foreach ( WC () -> cart -> get_cart () as $cart_item_key => $cart_item ) {
2012-11-27 16:22:47 +00:00
2012-11-08 16:57:53 +00:00
$product_cats = wp_get_post_terms ( $cart_item [ 'product_id' ], 'product_cat' , array ( " fields " => " ids " ) );
2012-11-27 16:22:47 +00:00
2014-10-14 14:11:46 +00:00
if ( sizeof ( array_intersect ( $product_cats , $this -> exclude_product_categories ) ) > 0 ) {
2012-11-08 16:57:53 +00:00
$valid_for_cart = false ;
2014-10-14 14:11:46 +00:00
}
2012-11-08 16:57:53 +00:00
}
}
2012-08-31 08:45:50 +00:00
if ( ! $valid_for_cart ) {
$valid = false ;
2014-10-14 14:18:31 +00:00
$error_code = self :: E_WC_COUPON_EXCLUDED_CATEGORIES ;
2012-08-31 08:45:50 +00:00
}
2012-11-08 16:57:53 +00:00
}
}
2012-08-14 19:42:38 +00:00
2012-06-10 18:07:19 +00:00
$valid = apply_filters ( 'woocommerce_coupon_is_valid' , $valid , $this );
2012-08-14 19:42:38 +00:00
2013-02-18 19:05:23 +00:00
if ( $valid ) {
2012-06-13 18:58:06 +00:00
return true ;
2013-02-18 19:05:23 +00:00
} else {
2013-03-25 03:50:17 +00:00
if ( is_null ( $error_code ) )
$error_code = self :: E_WC_COUPON_INVALID_FILTERED ;
2013-02-18 19:05:23 +00:00
}
2012-08-14 19:42:38 +00:00
2013-02-15 03:56:35 +00:00
} else {
2013-02-18 12:29:10 +00:00
$error_code = self :: E_WC_COUPON_NOT_EXIST ;
2012-07-31 12:28:03 +00:00
}
2012-08-14 19:42:38 +00:00
2013-02-18 12:29:10 +00:00
if ( $error_code )
$this -> error_message = $this -> get_coupon_error ( $error_code );
2012-12-19 14:57:46 +00:00
return false ;
2011-08-10 17:11:11 +00:00
}
2013-02-15 03:56:35 +00:00
2013-11-25 15:00:54 +00:00
/**
* Check if a coupon is valid
*
* @ return bool
*/
2013-10-18 17:10:55 +00:00
public function is_valid_for_cart () {
2014-07-02 14:03:48 +00:00
$valid = $this -> type != 'fixed_cart' && $this -> type != 'percent' ? false : true ;
return apply_filters ( 'woocommerce_coupon_is_valid_for_cart' , $valid , $this );
2013-10-18 17:10:55 +00:00
}
/**
* Check if a coupon is valid for a product
2014-09-20 18:46:22 +00:00
*
2013-10-18 17:10:55 +00:00
* @ param WC_Product $product
* @ return boolean
*/
2014-10-13 08:32:40 +00:00
public function is_valid_for_product ( $product , $values = array () ) {
2013-10-18 17:10:55 +00:00
if ( $this -> type != 'fixed_product' && $this -> type != 'percent_product' )
2014-10-10 14:45:01 +00:00
return apply_filters ( 'woocommerce_coupon_is_valid_for_product' , false , $product , $this , $values );
2013-10-18 17:10:55 +00:00
$valid = false ;
$product_cats = wp_get_post_terms ( $product -> id , 'product_cat' , array ( " fields " => " ids " ) );
// Specific products get the discount
if ( sizeof ( $this -> product_ids ) > 0 ) {
if ( in_array ( $product -> id , $this -> product_ids ) || ( isset ( $product -> variation_id ) && in_array ( $product -> variation_id , $this -> product_ids ) ) || in_array ( $product -> get_parent (), $this -> product_ids ) )
$valid = true ;
// Category discounts
} elseif ( sizeof ( $this -> product_categories ) > 0 ) {
if ( sizeof ( array_intersect ( $product_cats , $this -> product_categories ) ) > 0 )
$valid = true ;
} else {
// No product ids - all items discounted
$valid = true ;
}
// Specific product ID's excluded from the discount
if ( sizeof ( $this -> exclude_product_ids ) > 0 )
if ( in_array ( $product -> id , $this -> exclude_product_ids ) || ( isset ( $product -> variation_id ) && in_array ( $product -> variation_id , $this -> exclude_product_ids ) ) || in_array ( $product -> get_parent (), $this -> exclude_product_ids ) )
$valid = false ;
// Specific categories excluded from the discount
if ( sizeof ( $this -> exclude_product_categories ) > 0 )
if ( sizeof ( array_intersect ( $product_cats , $this -> exclude_product_categories ) ) > 0 )
$valid = false ;
// Sale Items excluded from discount
if ( $this -> exclude_sale_items == 'yes' ) {
2013-11-25 13:56:59 +00:00
$product_ids_on_sale = wc_get_product_ids_on_sale ();
2013-10-18 17:10:55 +00:00
if ( in_array ( $product -> id , $product_ids_on_sale , true ) || ( isset ( $product -> variation_id ) && in_array ( $product -> variation_id , $product_ids_on_sale , true ) ) || in_array ( $product -> get_parent (), $product_ids_on_sale , true ) )
$valid = false ;
}
2014-10-10 14:45:01 +00:00
return apply_filters ( 'woocommerce_coupon_is_valid_for_product' , $valid , $product , $this , $values );
2013-10-18 17:10:55 +00:00
}
/**
* Get discount amount for a cart item
2014-09-20 18:46:22 +00:00
*
2013-10-18 17:10:55 +00:00
* @ param float $discounting_amount Amount the coupon is being applied to
* @ param array | null $cart_item Cart item being discounted if applicable
* @ param boolean $single True if discounting a single qty item , false if its the line
* @ return float Amount this coupon has discounted
*/
public function get_discount_amount ( $discounting_amount , $cart_item = null , $single = false ) {
$discount = 0 ;
if ( $this -> type == 'fixed_product' ) {
$discount = $discounting_amount < $this -> amount ? $discounting_amount : $this -> amount ;
2014-03-06 12:29:25 +00:00
// If dealing with a line and not a single item, we need to multiple fixed discount by cart item qty.
if ( ! $single && ! is_null ( $cart_item ) ) {
// Discount for the line.
$discount = $discount * $cart_item [ 'quantity' ];
}
2013-10-18 17:10:55 +00:00
} elseif ( $this -> type == 'percent_product' || $this -> type == 'percent' ) {
2014-01-30 20:50:58 +00:00
$discount = round ( ( $discounting_amount / 100 ) * $this -> amount , WC () -> cart -> dp );
2013-10-18 17:10:55 +00:00
} elseif ( $this -> type == 'fixed_cart' ) {
if ( ! is_null ( $cart_item ) ) {
/**
* This is the most complex discount - we need to divide the discount between rows based on their price in
* proportion to the subtotal . This is so rows with different tax rates get a fair discount , and so rows
* with no price ( free ) don ' t get discounted .
*
* Get item discount by dividing item cost by subtotal to get a %
*/
$discount_percent = 0 ;
2014-04-07 09:26:43 +00:00
if ( WC () -> cart -> subtotal_ex_tax ) {
2013-10-18 17:10:55 +00:00
$discount_percent = ( $cart_item [ 'data' ] -> get_price_excluding_tax () * $cart_item [ 'quantity' ] ) / WC () -> cart -> subtotal_ex_tax ;
2014-04-07 09:26:43 +00:00
}
2014-09-20 18:46:22 +00:00
2013-10-18 17:10:55 +00:00
$discount = min ( ( $this -> amount * $discount_percent ) / $cart_item [ 'quantity' ], $discounting_amount );
} else {
$discount = min ( $this -> amount , $discounting_amount );
}
}
2014-03-06 12:29:25 +00:00
// Handle the limit_usage_to_x_items option
2013-10-18 17:10:55 +00:00
if ( in_array ( $this -> type , array ( 'percent_product' , 'fixed_product' ) ) && ! is_null ( $cart_item ) ) {
$qty = empty ( $this -> limit_usage_to_x_items ) ? $cart_item [ 'quantity' ] : min ( $this -> limit_usage_to_x_items , $cart_item [ 'quantity' ] );
if ( $single ) {
$discount = ( $discount * $qty ) / $cart_item [ 'quantity' ];
} else {
$discount = ( $discount / $cart_item [ 'quantity' ] ) * $qty ;
}
}
2014-04-07 09:26:43 +00:00
return apply_filters ( 'woocommerce_coupon_get_discount_amount' , $discount , $discounting_amount , $cart_item , $single , $this );
2013-10-18 17:10:55 +00:00
}
2013-02-15 00:29:55 +00:00
/**
2013-02-15 03:56:35 +00:00
* Converts one of the WC_Coupon message / error codes to a message string and
* displays the message / error .
2013-02-15 00:29:55 +00:00
*
* @ access public
2013-02-15 03:56:35 +00:00
* @ param int $msg_code Message / error code .
* @ return void
2013-02-15 00:29:55 +00:00
*/
2013-02-15 03:56:35 +00:00
public function add_coupon_message ( $msg_code ) {
2013-02-18 12:29:10 +00:00
if ( $msg_code < 200 )
2013-11-13 04:29:03 +00:00
wc_add_notice ( $this -> get_coupon_error ( $msg_code ), 'error' );
2013-02-18 12:29:10 +00:00
else
2013-11-13 04:32:29 +00:00
wc_add_notice ( $this -> get_coupon_message ( $msg_code ) );
2013-02-15 03:56:35 +00:00
}
2013-02-15 00:29:55 +00:00
2013-02-15 03:56:35 +00:00
/**
2013-02-18 12:29:10 +00:00
* Map one of the WC_Coupon message codes to a message string
2013-02-15 03:56:35 +00:00
*
* @ access public
2014-09-07 23:37:55 +00:00
* @ param integer $msg_code
2013-02-15 03:56:35 +00:00
* @ return string | Message / error string
*/
2013-02-18 12:29:10 +00:00
public function get_coupon_message ( $msg_code ) {
switch ( $msg_code ) {
2013-08-14 20:00:34 +00:00
case self :: WC_COUPON_SUCCESS :
2013-02-15 03:56:35 +00:00
$msg = __ ( 'Coupon code applied successfully.' , 'woocommerce' );
2013-02-18 12:29:10 +00:00
break ;
2013-08-14 20:00:34 +00:00
case self :: WC_COUPON_REMOVED :
$msg = __ ( 'Coupon code removed successfully.' , 'woocommerce' );
break ;
2013-02-15 03:56:35 +00:00
default :
$msg = '' ;
2013-02-18 12:29:10 +00:00
break ;
2013-02-15 03:56:35 +00:00
}
2013-02-18 12:29:10 +00:00
return apply_filters ( 'woocommerce_coupon_message' , $msg , $msg_code , $this );
2013-02-15 00:29:55 +00:00
}
2013-02-18 12:29:10 +00:00
2013-02-15 03:56:35 +00:00
/**
2013-02-18 12:29:10 +00:00
* Map one of the WC_Coupon error codes to a message string
*
* @ access public
* @ param int $err_code Message / error code .
* @ return string | Message / error string
*/
public function get_coupon_error ( $err_code ) {
switch ( $err_code ) {
2013-02-18 19:05:23 +00:00
case self :: E_WC_COUPON_INVALID_FILTERED :
$err = __ ( 'Coupon is not valid.' , 'woocommerce' );
break ;
2013-02-18 12:29:10 +00:00
case self :: E_WC_COUPON_NOT_EXIST :
$err = __ ( 'Coupon does not exist!' , 'woocommerce' );
break ;
case self :: E_WC_COUPON_INVALID_REMOVED :
$err = sprintf ( __ ( 'Sorry, it seems the coupon "%s" is invalid - it has now been removed from your order.' , 'woocommerce' ), $this -> code );
break ;
case self :: E_WC_COUPON_NOT_YOURS_REMOVED :
$err = sprintf ( __ ( 'Sorry, it seems the coupon "%s" is not yours - it has now been removed from your order.' , 'woocommerce' ), $this -> code );
break ;
case self :: E_WC_COUPON_ALREADY_APPLIED :
$err = __ ( 'Coupon code already applied!' , 'woocommerce' );
break ;
case self :: E_WC_COUPON_ALREADY_APPLIED_INDIV_USE_ONLY :
$err = sprintf ( __ ( 'Sorry, coupon "%s" has already been applied and cannot be used in conjunction with other coupons.' , 'woocommerce' ), $this -> code );
break ;
case self :: E_WC_COUPON_USAGE_LIMIT_REACHED :
$err = __ ( 'Coupon usage limit has been reached.' , 'woocommerce' );
break ;
case self :: E_WC_COUPON_EXPIRED :
$err = __ ( 'This coupon has expired.' , 'woocommerce' );
break ;
case self :: E_WC_COUPON_MIN_SPEND_LIMIT_NOT_MET :
2013-11-25 13:34:21 +00:00
$err = sprintf ( __ ( 'The minimum spend for this coupon is %s.' , 'woocommerce' ), wc_price ( $this -> minimum_amount ) );
2013-02-18 12:29:10 +00:00
break ;
2014-08-08 11:38:58 +00:00
case self :: E_WC_COUPON_MAX_SPEND_LIMIT_MET :
$err = sprintf ( __ ( 'The maximum spend for this coupon is %s.' , 'woocommerce' ), wc_price ( $this -> minimum_amount ) );
break ;
2013-02-18 12:29:10 +00:00
case self :: E_WC_COUPON_NOT_APPLICABLE :
$err = __ ( 'Sorry, this coupon is not applicable to your cart contents.' , 'woocommerce' );
break ;
2014-10-14 14:09:56 +00:00
case self :: E_WC_COUPON_EXCLUDED_PRODUCTS :
// Store excluded products that are in cart in $products
$products = array ();
if ( sizeof ( WC () -> cart -> get_cart () ) > 0 ) {
foreach ( WC () -> cart -> get_cart () as $cart_item_key => $cart_item ) {
if ( in_array ( $cart_item [ 'product_id' ], $this -> exclude_product_ids ) || in_array ( $cart_item [ 'variation_id' ], $this -> exclude_product_ids ) || in_array ( $cart_item [ 'data' ] -> get_parent (), $this -> exclude_product_ids ) ) {
$products [] = $cart_item [ 'data' ] -> get_title ();
}
}
}
2014-10-14 14:18:31 +00:00
$err = sprintf ( __ ( 'Sorry, this coupon is not applicable to the products: %s.' , 'woocommerce' ), implode ( ', ' , $products ) );
break ;
case self :: E_WC_COUPON_EXCLUDED_CATEGORIES :
// Store excluded categories that are in cart in $categories
$categories = array ();
if ( sizeof ( WC () -> cart -> get_cart () ) > 0 ) {
foreach ( WC () -> cart -> get_cart () as $cart_item_key => $cart_item ) {
$product_cats = wp_get_post_terms ( $cart_item [ 'product_id' ], 'product_cat' , array ( " fields " => " ids " ) );
if ( sizeof ( $intersect = array_intersect ( $product_cats , $this -> exclude_product_categories ) ) > 0 ) {
foreach ( $intersect as $cat_id ) {
$cat = get_term ( $cat_id , 'product_cat' );
$categories [] = $cat -> name ;
}
}
}
}
$err = sprintf ( __ ( 'Sorry, this coupon is not applicable to the categories: %s.' , 'woocommerce' ), implode ( ', ' , $categories ) );
2014-10-14 14:09:56 +00:00
break ;
2013-02-18 12:29:10 +00:00
case self :: E_WC_COUPON_NOT_VALID_SALE_ITEMS :
$err = __ ( 'Sorry, this coupon is not valid for sale items.' , 'woocommerce' );
break ;
default :
$err = '' ;
break ;
}
return apply_filters ( 'woocommerce_coupon_error' , $err , $err_code , $this );
}
/**
* Map one of the WC_Coupon error codes to an error string
2013-02-15 03:56:35 +00:00
* No coupon instance will be available where a coupon does not exist ,
* so this static method exists .
*
* @ access public
* @ param int $err_code Error code
* @ return string | Error string
*/
2013-02-18 12:29:10 +00:00
public static function get_generic_coupon_error ( $err_code ) {
switch ( $err_code ) {
case self :: E_WC_COUPON_NOT_EXIST :
$err = __ ( 'Coupon does not exist!' , 'woocommerce' );
break ;
case self :: E_WC_COUPON_PLEASE_ENTER :
$err = __ ( 'Please enter a coupon code.' , 'woocommerce' );
break ;
2013-02-15 03:56:35 +00:00
default :
2013-02-18 12:29:10 +00:00
$err = '' ;
break ;
2013-02-15 03:56:35 +00:00
}
2013-02-18 12:29:10 +00:00
2013-02-15 03:56:35 +00:00
// When using this static method, there is no $this to pass to filter
2013-02-18 12:29:10 +00:00
return apply_filters ( 'woocommerce_coupon_error' , $err , $err_code , null );
2013-02-15 03:56:35 +00:00
}
2013-02-18 12:29:10 +00:00
2013-01-29 13:23:52 +00:00
}