woocommerce/classes/class-wc-coupon.php

501 lines
16 KiB
PHP
Raw Normal View History

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
// Coupon message codes
const E_WC_COUPON_INVALID_REMOVED = 100;
const E_WC_COUPON_NOT_YOURS_REMOVED = 101;
const E_WC_COUPON_ALREADY_APPLIED = 102;
const E_WC_COUPON_ALREADY_APPLIED_INDIV_USE_ONLY = 103;
const E_WC_COUPON_NOT_EXIST = 104;
const E_WC_COUPON_USAGE_LIMIT_REACHED = 105;
const E_WC_COUPON_EXPIRED = 106;
const E_WC_COUPON_MIN_SPEND_LIMIT_NOT_MET = 107;
const E_WC_COUPON_NOT_APPLICABLE = 108;
const E_WC_COUPON_NOT_VALID_SALE_ITEMS = 109;
const E_WC_COUPON_PLEASE_ENTER = 110;
const WC_COUPON_SUCCESS = 200;
/** @public string Coupon code. */
public $code;
2012-08-15 17:08:42 +00:00
/** @public int Coupon ID. */
public $id;
2012-08-15 17:08:42 +00:00
/** @public string Type of discount. */
public $type;
2012-08-15 17:08:42 +00:00
/** @public string Type of discount (alias). */
public $discount_type;
2012-08-15 17:08:42 +00:00
/** @public string Coupon amount. */
public $amount;
2012-08-15 17:08:42 +00:00
/** @public string "Yes" if for individual use. */
public $individual_use;
2012-08-15 17:08:42 +00:00
/** @public array Array of product IDs. */
public $product_ids;
2012-08-15 17:08:42 +00:00
/** @public int Coupon usage limit. */
public $usage_limit;
2012-08-15 17:08:42 +00:00
/** @public int Coupon usage count. */
public $usage_count;
2012-08-15 17:08:42 +00:00
/** @public string Expirey date. */
public $expiry_date;
2012-08-15 17:08:42 +00:00
/** @public string "yes" if applied before tax. */
public $apply_before_tax;
2012-08-15 17:08:42 +00:00
/** @public string "yes" if coupon grants free shipping. */
public $free_shipping;
2012-08-15 17:08:42 +00:00
/** @public array Array of category ids. */
public $product_categories;
2012-08-15 17:08:42 +00:00
/** @public array Array of category ids. */
public $exclude_product_categories;
2012-08-15 17:08:42 +00:00
/** @public string "yes" if coupon does NOT apply to items on sale. */
public $exclude_sale_items;
/** @public string Minimum cart amount. */
public $minimum_amount;
2012-08-15 17:08:42 +00:00
/** @public string Coupon owner's email. */
public $customer_email;
2012-08-15 17:08:42 +00:00
/** @public array Post meta. */
public $coupon_custom_fields;
2012-08-15 17:08:42 +00:00
/** @public string How much the coupon is worth. */
public $coupon_amount;
2012-08-14 19:42:38 +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
* @return void
2012-08-14 19:42:38 +00:00
*/
public function __construct( $code ) {
2012-02-08 15:55:02 +00:00
global $wpdb;
2012-08-14 19:42:38 +00:00
$this->code = apply_filters( 'woocommerce_coupon_code', $code );
2012-11-27 16:22:47 +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
$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->exclude_sale_items = esc_html( $coupon_data['exclude_sale_items'] );
$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
} else {
2012-11-27 16:22:47 +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
2013-02-15 03:56:35 +00:00
if ( ! $coupon_id )
2012-12-27 18:46:22 +00:00
return;
$coupon = get_post( $coupon_id );
$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-12-20 11:30:27 +00:00
return;
2012-12-27 18:46:22 +00:00
$this->id = $coupon->ID;
$this->coupon_custom_fields = get_post_meta( $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(),
'exclude_sale_items' => 'no',
'minimum_amount' => '',
'customer_email' => array()
);
2012-11-27 16:22:47 +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;
// 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 ) ) ) );
}
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
*/
public function apply_before_tax() {
return $this->apply_before_tax == 'yes' ? true : false;
}
2012-08-14 19:42:38 +00:00
/**
* Check if a coupon enables free shipping.
*
* @access public
* @return void
*/
public function enable_free_shipping() {
return $this->free_shipping == 'yes' ? true : false;
2011-11-28 16:10:31 +00:00
}
2013-01-29 13:58:57 +00:00
/**
* Check if a coupon excludes sale items.
*
* @access public
* @return void
*/
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
* @return void
*/
public function inc_usage_count() {
$this->usage_count++;
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
*/
public function dcr_usage_count() {
$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
/**
* 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
*/
public function is_valid() {
global $woocommerce;
2012-08-14 19:42:38 +00:00
2013-02-15 03:56:35 +00:00
$error_code = null;
$valid = true;
$error = false;
2012-08-14 19:42:38 +00:00
if ( $this->id ) {
2012-08-14 19:42:38 +00:00
2012-01-17 15:20:04 +00:00
// Usage Limit
if ( $this->usage_limit > 0 ) {
if ( $this->usage_count >= $this->usage_limit ) {
2012-01-17 15:20:04 +00:00
$valid = false;
$error_code = self::E_WC_COUPON_USAGE_LIMIT_REACHED;
}
}
2012-08-14 19:42:38 +00:00
2012-01-17 15:20:04 +00:00
// Expired
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;
$error_code = self::E_WC_COUPON_EXPIRED;
}
}
2012-08-14 19:42:38 +00:00
2012-03-07 21:38:34 +00:00
// Minimum spend
if ( $this->minimum_amount > 0 ) {
if ( $this->minimum_amount > $woocommerce->cart->subtotal ) {
2013-02-15 03:56:35 +00:00
$valid = false;
$error_code = self::E_WC_COUPON_MIN_SPEND_LIMIT_NOT_MET;
}
}
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
if ( sizeof( $this->product_ids ) > 0 ) {
2012-01-17 15:20:04 +00:00
$valid_for_cart = false;
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
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;
$error_code = self::E_WC_COUPON_NOT_APPLICABLE;
2012-08-31 08:45:50 +00:00
}
}
2012-08-14 19:42:38 +00:00
// Category ids - If a product included is found in the cart then its valid
if ( sizeof( $this->product_categories ) > 0 ) {
$valid_for_cart = false;
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
$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 )
$valid_for_cart = true;
}
}
2012-08-31 08:45:50 +00:00
if ( ! $valid_for_cart ) {
$valid = false;
$error_code = self::E_WC_COUPON_NOT_APPLICABLE;
2012-08-31 08:45:50 +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
if ( $this->type != 'fixed_product' && $this->type != 'percent_product' ) {
2012-08-14 19:42:38 +00:00
// Exclude Products
if ( sizeof( $this->exclude_product_ids ) > 0 ) {
2012-01-17 15:20:04 +00:00
$valid_for_cart = true;
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;
$error_code = self::E_WC_COUPON_NOT_APPLICABLE;
2012-08-31 08:45:50 +00:00
}
}
2012-08-14 19:42:38 +00:00
// Exclude Sale Items
if ( $this->exclude_sale_items == 'yes' ) {
$valid_for_cart = true;
$product_ids_on_sale = woocommerce_get_product_ids_on_sale();
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'], $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;
$error_code = self::E_WC_COUPON_NOT_VALID_SALE_ITEMS;
}
}
// Exclude Categories
if ( sizeof( $this->exclude_product_categories ) > 0 ) {
$valid_for_cart = true;
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
$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 )
$valid_for_cart = false;
}
}
2012-08-31 08:45:50 +00:00
if ( ! $valid_for_cart ) {
$valid = false;
$error_code = self::E_WC_COUPON_NOT_APPLICABLE;
2012-08-31 08:45:50 +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 )
return true;
2012-08-14 19:42:38 +00:00
2013-02-15 03:56:35 +00:00
} else {
$error_code = self::E_WC_COUPON_NOT_EXIST;
2012-07-31 12:28:03 +00:00
}
2012-08-14 19:42:38 +00:00
if ( $error_code )
$this->error_message = $this->get_coupon_error( $error_code );
return false;
2011-08-10 17:11:11 +00:00
}
2013-02-15 03:56:35 +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 ) {
global $woocommerce;
if ( $msg_code < 200 )
$woocommerce->add_error( $this->get_coupon_error( $msg_code ) );
else
$woocommerce->add_message( $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
/**
* Map one of the WC_Coupon message codes to a message string
2013-02-15 03:56:35 +00:00
*
* @access public
* @param mixed $msg_code
2013-02-15 03:56:35 +00:00
* @return string| Message/error string
*/
public function get_coupon_message( $msg_code ) {
switch ( $msg_code ) {
case self::WC_COUPON_SUCCESS:
2013-02-15 03:56:35 +00:00
$msg = __( 'Coupon code applied successfully.', 'woocommerce' );
break;
2013-02-15 03:56:35 +00:00
default:
$msg = '';
break;
2013-02-15 03:56:35 +00:00
}
return apply_filters( 'woocommerce_coupon_message', $msg, $msg_code, $this );
2013-02-15 00:29:55 +00:00
}
2013-02-15 03:56:35 +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 ) {
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:
$err = sprintf( __( 'The minimum spend for this coupon is %s.', 'woocommerce' ), woocommerce_price( $this->minimum_amount ) );
break;
case self::E_WC_COUPON_NOT_APPLICABLE:
$err = __( 'Sorry, this coupon is not applicable to your cart contents.', 'woocommerce' );
break;
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
*/
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:
$err = '';
break;
2013-02-15 03:56:35 +00:00
}
2013-02-15 03:56:35 +00:00
// When using this static method, there is no $this to pass to filter
return apply_filters( 'woocommerce_coupon_error', $err, $err_code, null );
2013-02-15 03:56:35 +00:00
}
2013-01-29 13:23:52 +00:00
}