woocommerce/includes/class-wc-discounts.php

372 lines
10 KiB
PHP
Raw Normal View History

2017-07-18 04:43:31 +00:00
<?php
/**
* Discount calculation
*
* @author Automattic
* @package WooCommerce/Classes
* @version 3.2.0
* @since 3.2.0
*/
/**
* Discounts class.
2017-07-18 19:48:19 +00:00
*
* @todo this class will need to be called instead get_discounted_price, in the cart?
2017-07-18 04:43:31 +00:00
*/
class WC_Discounts {
/**
2017-07-18 13:04:56 +00:00
* An array of items to discount.
2017-07-18 04:43:31 +00:00
*
2017-07-18 13:04:56 +00:00
* @var array
2017-07-18 04:43:31 +00:00
*/
2017-07-18 13:04:56 +00:00
protected $items = array();
2017-07-18 04:43:31 +00:00
2017-07-18 17:07:46 +00:00
/**
2017-07-19 14:55:56 +00:00
* An array of discounts which have been applied to items.
2017-07-18 17:07:46 +00:00
*
* @var array
*/
protected $discounts = array();
2017-07-18 17:07:46 +00:00
2017-07-19 14:55:56 +00:00
/**
* An array of applied coupons codes and total discount.
*
* @var array
*/
protected $applied_coupons = array();
2017-07-19 11:26:01 +00:00
/**
* Precision so we can work in cents.
*
* @var int
*/
protected $precision = 1;
/**
* Constructor.
*/
public function __construct() {
$this->precision = pow( 10, wc_get_price_decimals() );
}
2017-07-18 04:43:31 +00:00
/**
2017-07-18 13:04:56 +00:00
* Get items.
2017-07-18 04:43:31 +00:00
*
2017-07-18 13:04:56 +00:00
* @since 3.2.0
* @return object[]
2017-07-18 04:43:31 +00:00
*/
2017-07-18 13:04:56 +00:00
public function get_items() {
return $this->items;
2017-07-18 04:43:31 +00:00
}
2017-07-19 11:26:01 +00:00
/**
* Get discount by key without precision.
*
* @since 3.2.0
* @return array
*/
public function get_discount( $key ) {
return isset( $this->discounts[ $key ] ) ? $this->remove_precision( $this->discounts[ $key ] ) : 0;
}
/**
* Get all discount totals without precision.
*
* @since 3.2.0
* @return array
*/
public function get_discounts() {
return array_map( array( $this, 'remove_precision' ), $this->discounts );
}
/**
* Get discounted price of an item without precision.
*
* @since 3.2.0
* @param object $item
* @return float
*/
public function get_discounted_price( $item ) {
return $this->remove_precision( $this->get_discounted_price_in_cents( $item ) );
}
/**
* Get discounted price of an item to precision (in cents).
*
* @since 3.2.0
* @param object $item
* @return float
*/
public function get_discounted_price_in_cents( $item ) {
return $item->price - $this->discounts[ $item->key ];
}
2017-07-19 14:55:56 +00:00
/**
* Returns a list of applied coupons with name value pairs - name being
* the coupon code, and value being the total amount disounted.
*
* @since 3.2.0
* @return array
*/
public function get_applied_coupons() {
return array_map( array( $this, 'remove_precision' ), $this->applied_coupons );
}
2017-07-18 04:43:31 +00:00
/**
2017-07-18 13:04:56 +00:00
* Set cart/order items which will be discounted.
2017-07-18 04:43:31 +00:00
*
2017-07-18 13:04:56 +00:00
* @since 3.2.0
2017-07-18 17:07:46 +00:00
* @param array $raw_items List of raw cart or order items.
2017-07-18 04:43:31 +00:00
*/
2017-07-18 13:04:56 +00:00
public function set_items( $raw_items ) {
2017-07-19 12:49:22 +00:00
$this->items = array();
$this->discounts = array();
2017-07-18 17:47:05 +00:00
if ( ! empty( $raw_items ) && is_array( $raw_items ) ) {
foreach ( $raw_items as $raw_item ) {
$item = (object) array(
2017-07-19 11:26:01 +00:00
'price' => 0, // Line price without discounts, in cents.
'quantity' => 0, // Line qty.
'product' => false,
);
if ( is_a( $raw_item, 'WC_Cart_Item' ) ) {
//$item->quantity = $raw_item->get_quantity();
//$item->price = $raw_item->get_price() * $raw_item->get_quantity();
//$item->is_taxable = $raw_item->is_taxable();
//$item->tax_class = $raw_item->get_tax_class();
// @todo
} elseif ( is_a( $raw_item, 'WC_Order_Item_Product' ) ) {
$item->key = $raw_item->get_id();
$item->quantity = $raw_item->get_quantity();
2017-07-19 11:26:01 +00:00
$item->price = $raw_item->get_subtotal() * $this->precision;
$item->product = $raw_item->get_product();
} else {
$item->key = $raw_item['key'];
$item->quantity = $raw_item['quantity'];
2017-07-19 11:26:01 +00:00
$item->price = $raw_item['data']->get_price() * $this->precision * $raw_item['quantity'];
$item->product = $raw_item['data'];
}
$this->items[ $item->key ] = $item;
$this->discounts[ $item->key ] = 0;
2017-07-18 13:04:56 +00:00
}
2017-07-19 12:49:22 +00:00
uasort( $this->items, array( $this, 'sort_by_price' ) );
2017-07-18 13:04:56 +00:00
}
2017-07-18 04:43:31 +00:00
}
2017-07-19 14:55:56 +00:00
/**
* Apply a discount to all items using a coupon.
*
* @todo Coupon class has lots of WC()->cart calls and needs decoupling. This makes 'is valid' hard to use here.
* @todo is_valid_for_product accepts values - how can we deal with that?
*
* @since 3.2.0
* @param WC_Coupon $coupon
* @return bool True if applied.
*/
public function apply_coupon( $coupon ) {
if ( ! is_a( $coupon, 'WC_Coupon' ) ) {
return false;
}
if ( ! isset( $this->applied_coupons[ $coupon->get_code() ] ) ) {
$this->applied_coupons[ $coupon->get_code() ] = 0;
}
// @todo how can we support the old woocommerce_coupon_get_discount_amount filter?
// @todo is valid for product - filter items here and pass to function?
$items_to_apply = $this->get_items_to_apply_coupon( $coupon );
switch ( $coupon->get_discount_type() ) {
case 'percent' :
$this->applied_coupons[ $coupon->get_code() ] += $this->apply_percentage_discount( $items_to_apply, $coupon->get_amount() );
break;
case 'fixed_product' :
$this->applied_coupons[ $coupon->get_code() ] += $this->apply_fixed_product_discount( $items_to_apply, $coupon->get_amount() * $this->precision );
break;
case 'fixed_cart' :
$this->applied_coupons[ $coupon->get_code() ] += $this->apply_fixed_cart_discount( $items_to_apply, $coupon->get_amount() * $this->precision );
break;
}
}
/**
* Remove precision from a price.
*
* @param int $value
* @return float
*/
protected function remove_precision( $value ) {
return wc_format_decimal( $value / $this->precision, wc_get_price_decimals() );
}
/**
* Sort by price.
*
* @param array $a
* @param array $b
* @return int
*/
protected function sort_by_price( $a, $b ) {
$price_1 = $a->price * $a->quantity;
$price_2 = $b->price * $b->quantity;;
if ( $price_1 === $price_2 ) {
return 0;
}
return ( $price_1 < $price_2 ) ? 1 : -1;
}
2017-07-19 12:49:22 +00:00
/**
* Filter out all products which have been fully discounted to 0.
* Used as array_filter callback.
*
* @param object $item
* @return bool
*/
protected function filter_products_with_price( $item ) {
return $this->get_discounted_price_in_cents( $item ) > 0;
}
/**
* Get items which the coupon should be applied to.
*
* @param object $coupon
* @return array
*/
protected function get_items_to_apply_coupon( $coupon ) {
$items_to_apply = array();
$limit_usage_qty = 0;
$applied_count = 0;
if ( null !== $coupon->get_limit_usage_to_x_items() ) {
$limit_usage_qty = $coupon->get_limit_usage_to_x_items();
}
foreach ( $this->items as $item ) {
if ( 0 === $this->get_discounted_price_in_cents( $item ) ) {
continue;
}
if ( ! $coupon->is_valid_for_product( $item->product ) && ! $coupon->is_valid_for_cart() ) { // @todo is this enough?
continue;
}
if ( $limit_usage_qty && $applied_count > $limit_usage_qty ) {
break;
}
if ( $limit_usage_qty && $item->quantity > ( $limit_usage_qty - $applied_count ) ) {
$limit_to_qty = absint( $limit_usage_qty - $applied_count );
$item->price = ( $item->price / $item->quantity ) * $limit_to_qty;
$item->quantity = $limit_to_qty; // Lower the qty so the discount is applied less.
}
if ( 0 >= $item->quantity ) {
continue;
}
$items_to_apply[] = $item;
$applied_count += $item->quantity;
}
return $items_to_apply;
}
2017-07-18 17:07:46 +00:00
/**
* Apply a discount amount to an item and ensure it does not go negative.
2017-07-18 17:47:05 +00:00
*
* @since 3.2.0
* @param object $item
2017-07-19 11:26:01 +00:00
* @param int $discount
* @return int Amount discounted.
*/
2017-07-19 11:26:01 +00:00
protected function add_item_discount( &$item, $discount ) {
$discounted_price = $this->get_discounted_price_in_cents( $item );
$discount = $discount > $discounted_price ? $discounted_price : $discount;
$this->discounts[ $item->key ] = $this->discounts[ $item->key ] + $discount;
return $discount;
}
/**
* Apply percent discount to items.
*
* @since 3.2.0
2017-07-19 12:49:22 +00:00
* @param array $items_to_apply Array of items to apply the coupon to.
2017-07-19 14:55:56 +00:00
* @param int $amount
* @return int total discounted in cents
2017-07-18 17:47:05 +00:00
*/
2017-07-19 12:49:22 +00:00
protected function apply_percentage_discount( $items_to_apply, $amount ) {
2017-07-19 14:55:56 +00:00
$total_discounted = 0;
2017-07-19 12:49:22 +00:00
foreach ( $items_to_apply as $item ) {
2017-07-19 14:55:56 +00:00
$total_discounted += $this->add_item_discount( $item, $amount * ( $this->get_discounted_price_in_cents( $item ) / 100 ) );
2017-07-18 17:47:05 +00:00
}
2017-07-19 14:55:56 +00:00
return $total_discounted;
2017-07-18 17:47:05 +00:00
}
/**
* Apply fixed product discount to items.
2017-07-18 17:07:46 +00:00
*
* @since 3.2.0
2017-07-19 14:55:56 +00:00
* @param array $items_to_apply Array of items to apply the coupon to.
2017-07-19 11:26:01 +00:00
* @param int $amount
2017-07-19 14:55:56 +00:00
* @return int total discounted in cents
2017-07-18 17:07:46 +00:00
*/
2017-07-19 12:49:22 +00:00
protected function apply_fixed_product_discount( $items_to_apply, $discount ) {
2017-07-19 14:55:56 +00:00
$total_discounted = 0;
2017-07-19 12:49:22 +00:00
foreach ( $items_to_apply as $item ) {
2017-07-19 14:55:56 +00:00
$total_discounted += $this->add_item_discount( $item, $discount * $item->quantity );
2017-07-18 17:07:46 +00:00
}
2017-07-19 14:55:56 +00:00
return $total_discounted;
2017-07-18 04:43:31 +00:00
}
2017-07-18 17:52:50 +00:00
/**
2017-07-19 11:26:01 +00:00
* Apply fixed cart discount to items.
2017-07-18 17:52:50 +00:00
*
2017-07-19 14:55:56 +00:00
* @since 3.2.0
* @param array $items_to_apply Array of items to apply the coupon to.
* @param int $cart_discount
* @return int total discounted in cents
2017-07-18 17:52:50 +00:00
*/
2017-07-19 12:49:22 +00:00
protected function apply_fixed_cart_discount( $items_to_apply, $cart_discount ) {
2017-07-19 14:55:56 +00:00
$items_to_apply = array_filter( $items_to_apply, array( $this, 'filter_products_with_price' ) );
2017-07-19 11:26:01 +00:00
2017-07-19 12:49:22 +00:00
if ( ! $item_count = array_sum( wp_list_pluck( $items_to_apply, 'quantity' ) ) ) {
2017-07-19 14:55:56 +00:00
return 0;
2017-07-19 11:26:01 +00:00
}
$per_item_discount = floor( $cart_discount / $item_count ); // round it down to the nearest cent number.
$amount_discounted = 0;
if ( $per_item_discount > 0 ) {
2017-07-19 12:49:22 +00:00
foreach ( $items_to_apply as $item ) {
2017-07-19 11:26:01 +00:00
$amount_discounted += $this->add_item_discount( $item, $per_item_discount * $item->quantity );
}
/**
* If there is still discount remaining, repeat the process.
*/
if ( $amount_discounted > 0 && $amount_discounted < $cart_discount ) {
2017-07-19 14:55:56 +00:00
$amount_discounted += $this->apply_fixed_cart_discount( $items_to_apply, $cart_discount - $amount_discounted );
2017-07-19 11:26:01 +00:00
}
2017-07-19 11:26:01 +00:00
/**
* Deal with remaining fractional discounts by splitting it over items
* until the amount is expired, discounting 1 cent at a time.
*/
2017-07-19 14:55:56 +00:00
} elseif ( $cart_discount > 0 ) {
2017-07-19 12:49:22 +00:00
foreach ( $items_to_apply as $item ) {
2017-07-19 11:26:01 +00:00
for ( $i = 0; $i < $item->quantity; $i ++ ) {
$amount_discounted += $this->add_item_discount( $item, 1 );
if ( $amount_discounted >= $cart_discount ) {
break 2;
}
}
if ( $amount_discounted >= $cart_discount ) {
break;
}
}
}
2017-07-19 14:55:56 +00:00
return $amount_discounted;
2017-07-18 17:52:50 +00:00
}
2017-07-18 04:43:31 +00:00
}