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
2011-08-10 17:11:11 +00:00
* @ package WooCommerce
* @ category Class
* @ author WooThemes
*/
2012-01-27 16:38:39 +00:00
class WC_Coupon {
2012-08-14 19:42:38 +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
2012-12-14 21:41:59 +00:00
/** @public int Coupon usage count. */
public $usage_count ;
2012-08-15 17:08:42 +00:00
2012-12-14 21:41:59 +00:00
/** @public string Expirey date. */
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
2012-12-14 21:41:59 +00:00
/** @public string Minimum cart amount. */
public $minimum_amount ;
2012-08-15 17:08:42 +00:00
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-17 12:20:22 +00:00
* @ return void
2012-08-14 19:42:38 +00:00
*/
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
2012-11-08 16:57:53 +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
$coupon_data = apply_filters ( 'woocommerce_get_shop_coupon_data' , false , $code );
if ( $coupon_data ) {
2012-11-27 16:22:47 +00:00
2012-11-08 16:57:53 +00:00
$this -> id = absint ( $coupon_data [ 'id' ] );
$this -> type = esc_html ( $coupon_data [ 'type' ] );
$this -> amount = esc_html ( $coupon_data [ 'amount' ] );
$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' ] );
$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 -> minimum_amount = esc_html ( $coupon_data [ 'minimum_amount' ] );
$this -> customer_email = esc_html ( $coupon_data [ 'customer_email' ] );
2012-11-27 16:22:47 +00:00
2012-11-08 16:57:53 +00:00
} else {
2012-11-27 16:22:47 +00:00
2012-11-08 16:57:53 +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' " ), $this -> code ) );
2012-11-27 16:22:47 +00:00
if ( $coupon_id )
$coupon = get_post ( $coupon_id );
2012-12-11 17:02:08 +00:00
else
2012-11-27 16:22:47 +00:00
2012-11-08 16:57:53 +00:00
$coupon -> post_title = apply_filters ( 'woocommerce_coupon_code' , $coupon -> post_title );
2012-11-27 16:22:47 +00:00
if ( empty ( $coupon ) || $coupon -> post_status !== 'publish' || $this -> code !== $coupon -> post_title )
2012-11-08 16:57:53 +00:00
$this -> id = $coupon -> ID ;
$this -> coupon_custom_fields = get_post_custom ( $this -> id );
$load_data = array (
'discount_type' => 'fixed_cart' ,
'coupon_amount' => 0 ,
'individual_use' => 'no' ,
'product_ids' => '' ,
'exclude_product_ids' => '' ,
'usage_limit' => '' ,
'usage_count' => '' ,
'expiry_date' => '' ,
'apply_before_tax' => 'yes' ,
'free_shipping' => 'no' ,
'product_categories' => array (),
'exclude_product_categories' => array (),
'minimum_amount' => '' ,
'customer_email' => array ()
);
2012-11-27 16:22:47 +00:00
foreach ( $load_data as $key => $default )
2012-11-08 16:57:53 +00:00
$this -> $key = isset ( $this -> coupon_custom_fields [ $key ][ 0 ] ) && $this -> coupon_custom_fields [ $key ][ 0 ] !== '' ? $this -> coupon_custom_fields [ $key ][ 0 ] : $default ;
// Alias
$this -> type = $this -> discount_type ;
$this -> amount = $this -> coupon_amount ;
// 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 ) ) );
$this -> customer_email = array_filter ( array_map ( 'trim' , array_map ( 'strtolower' , ( array ) maybe_unserialize ( $this -> customer_email ) ) ) );
}
2012-11-27 16:22:47 +00:00
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
* @ return void
*/
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
}
2012-08-14 19:42:38 +00:00
/**
* Increase usage count fo current coupon .
*
* @ access public
* @ return void
*/
2012-12-14 21:41:59 +00:00
public function inc_usage_count () {
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 );
}
2012-08-14 19:42:38 +00:00
/**
* Decrease usage count fo current coupon .
*
* @ access public
* @ return void
*/
2012-12-14 21:41:59 +00:00
public function dcr_usage_count () {
2012-07-10 15:52:52 +00:00
$this -> usage_count -- ;
update_post_meta ( $this -> id , 'usage_count' , $this -> usage_count );
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 .
*
* Check if a coupon is valid . Return a reason code if invaid . Reason codes :
2012-08-14 19:42:38 +00:00
*
2012-06-10 18:07:19 +00:00
* @ access public
2012-08-14 19:42:38 +00:00
* @ return bool | WP_Error 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 () {
2011-09-06 11:11:22 +00:00
global $woocommerce ;
2012-08-14 19:42:38 +00:00
2012-07-31 12:28:03 +00:00
if ( $this -> id ) {
2012-08-14 19:42:38 +00:00
2012-01-17 15:20:04 +00:00
$valid = true ;
2012-06-10 18:07:19 +00:00
$error = false ;
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 ;
2012-06-10 18:07:19 +00:00
$error = __ ( 'Coupon usage limit has been reached.' , 'woocommerce' );
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 ;
2012-06-10 18:07:19 +00:00
$error = __ ( 'This coupon has expired.' , 'woocommerce' );
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 ) {
if ( $this -> minimum_amount > $woocommerce -> cart -> subtotal ) {
2012-03-07 21:38:34 +00:00
$valid = false ;
2012-09-06 16:47:05 +00:00
$error = sprintf ( __ ( 'The minimum spend for this coupon is %s.' , 'woocommerce' ), woocommerce_price ( $this -> minimum_amount ) );
2012-11-08 16:57:53 +00:00
}
}
2012-08-14 19:42:38 +00:00
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 ;
2012-11-08 16:57:53 +00:00
if ( sizeof ( $woocommerce -> cart -> get_cart () ) > 0 ) {
foreach ( $woocommerce -> 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 ;
2012-08-31 08:47:57 +00:00
$error = __ ( 'Sorry, this coupon is not applicable to your cart contents.' , 'woocommerce' );
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 ;
2012-11-08 16:57:53 +00:00
if ( sizeof ( $woocommerce -> cart -> get_cart () ) > 0 ) {
foreach ( $woocommerce -> 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 ;
2012-08-31 08:47:57 +00:00
$error = __ ( 'Sorry, this coupon is not applicable to your cart contents.' , 'woocommerce' );
2012-08-31 08:45:50 +00:00
}
2012-11-08 16:57:53 +00:00
}
2012-08-14 19:42:38 +00:00
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 ;
2012-11-08 16:57:53 +00:00
if ( sizeof ( $woocommerce -> cart -> get_cart () ) > 0 ) {
foreach ( $woocommerce -> 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 ) ) {
$valid_for_cart = false ;
}
}
}
2012-08-31 08:45:50 +00:00
if ( ! $valid_for_cart ) {
$valid = false ;
2012-08-31 08:47:57 +00:00
$error = __ ( 'Sorry, this coupon is not applicable to your cart contents.' , 'woocommerce' );
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
// 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 ;
2012-11-08 16:57:53 +00:00
if ( sizeof ( $woocommerce -> cart -> get_cart () ) > 0 ) {
foreach ( $woocommerce -> 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 -> exclude_product_categories ) ) > 0 )
2012-11-08 16:57:53 +00:00
$valid_for_cart = false ;
}
}
2012-08-31 08:45:50 +00:00
if ( ! $valid_for_cart ) {
$valid = false ;
2012-08-31 08:47:57 +00:00
$error = __ ( 'Sorry, this coupon is not applicable to your cart contents.' , 'woocommerce' );
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
if ( $valid )
2012-06-13 18:58:06 +00:00
return true ;
2012-08-14 19:42:38 +00:00
2012-07-31 12:28:03 +00:00
} else {
$error = __ ( 'Invalid coupon' , 'woocommerce' );
}
2012-08-14 19:42:38 +00:00
2012-12-19 14:57:46 +00:00
$this -> error_message = apply_filters ( 'woocommerce_coupon_error' , $error , $this );
return false ;
2011-08-10 17:11:11 +00:00
}
2012-01-27 16:38:39 +00:00
}