2022-11-27 15:03:07 +00:00
< ? php
/**
* Discount Functions
*
* @ package EDD
* @ subpackage Functions
* @ copyright Copyright ( c ) 2018 , Easy Digital Downloads , LLC
* @ license http :// opensource . org / licenses / gpl - 2.0 . php GNU Public License
* @ since 1.0
*/
// Exit if accessed directly
defined ( 'ABSPATH' ) || exit ;
/**
* Add a discount .
*
* @ since 1.0
* @ since 3.0 This function has been repurposed . Previously it was an internal admin callback for adding
* a discount via the UI . It ' s now used as a public function for inserting a new discount
* into the database .
*
* @ param array $data Discount data .
* @ return int Discount ID .
*/
function edd_add_discount ( $data = array () ) {
// Juggle requirements and products.
$product_requirements = isset ( $data [ 'product_reqs' ] ) ? $data [ 'product_reqs' ] : null ;
$excluded_products = isset ( $data [ 'excluded_products' ] ) ? $data [ 'excluded_products' ] : null ;
$product_condition = isset ( $data [ 'product_condition' ] ) ? $data [ 'product_condition' ] : null ;
$pre_convert_args = $data ;
unset ( $data [ 'product_reqs' ], $data [ 'excluded_products' ], $data [ 'product_condition' ] );
if ( isset ( $data [ 'expiration' ] ) ) {
$data [ 'end_date' ] = $data [ 'expiration' ];
unset ( $data [ 'expiration' ] );
}
if ( isset ( $data [ 'start' ] ) ) {
$data [ 'start_date' ] = $data [ 'start' ];
unset ( $data [ 'start' ] );
}
// Setup the discounts query.
$discounts = new EDD\Compat\Discount_Query ();
// Attempt to add the discount.
$discount_id = $discounts -> add_item ( $data );
// Maybe add requirements & exclusions.
if ( ! empty ( $discount_id ) ) {
// Product requirements.
if ( ! empty ( $product_requirements ) ) {
if ( is_string ( $product_requirements ) ) {
$product_requirements = maybe_unserialize ( $product_requirements );
}
if ( is_array ( $product_requirements ) ) {
foreach ( array_filter ( $product_requirements ) as $product_requirement ) {
edd_add_adjustment_meta ( $discount_id , 'product_requirement' , $product_requirement );
}
}
}
// Excluded products.
if ( ! empty ( $excluded_products ) ) {
if ( is_string ( $excluded_products ) ) {
$excluded_products = maybe_unserialize ( $excluded_products );
}
if ( is_array ( $excluded_products ) ) {
foreach ( array_filter ( $excluded_products ) as $excluded_product ) {
edd_add_adjustment_meta ( $discount_id , 'excluded_product' , $excluded_product );
}
}
}
if ( ! empty ( $product_condition ) ) {
edd_add_adjustment_meta ( $discount_id , 'product_condition' , $product_condition );
}
// If the end date has passed, mark the discount as expired.
edd_is_discount_expired ( $discount_id );
}
/**
* Fires after the discount code is inserted . This hook exists for
* backwards compatibility purposes . It uses the $pre_convert_args variable
* to ensure the arguments maintain backwards compatible array keys .
*
* @ since 2.7
*
* @ param array $pre_convert_args Discount args .
* @ param int $return Discount ID .
*/
do_action ( 'edd_post_insert_discount' , $pre_convert_args , $discount_id );
// Return the new discount ID.
return $discount_id ;
}
/**
* Delete a discount .
*
* @ since 3.0
*
* @ param int $discount_id Discount ID .
* @ return int
*/
function edd_delete_discount ( $discount_id = 0 ) {
$discount = edd_get_discount ( $discount_id );
// Do not allow for a discount to be deleted if it has been used.
if ( $discount && 0 < $discount -> use_count ) {
return false ;
}
$discounts = new EDD\Compat\Discount_Query ();
// Pre-3.0 pre action.
do_action ( 'edd_pre_delete_discount' , $discount_id );
$retval = $discounts -> delete_item ( $discount_id );
// Pre-3.0 post action.
do_action ( 'edd_post_delete_discount' , $discount_id );
return $retval ;
}
/**
* Get Discount .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object
* @ since 3.0 Updated to call use new query class .
*
* @ param int $discount_id Discount ID .
* @ return \EDD_Discount | bool EDD_Discount object or false if not found .
*/
function edd_get_discount ( $discount_id = 0 ) {
$discounts = new EDD\Compat\Discount_Query ();
// Return discount
return $discounts -> get_item ( $discount_id );
}
/**
* Get discount by code .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object
* @ since 3.0 Updated to call use new query class .
* @ since 3.0 Updated to include a filter .
*
* @ param string $code Discount code .
* @ return EDD_Discount | bool EDD_Discount object or false if not found .
*/
function edd_get_discount_by_code ( $code = '' ) {
$discount = edd_get_discount_by ( 'code' , $code );
/**
* Filters the get discount by request .
*
* @ since 3.0
*
* @ param \EDD_Discount $discount Discount object .
* @ param string $code Discount code .
*/
return apply_filters ( 'edd_get_discount_by_code' , $discount , $code );
}
/**
* Retrieve discount by a given field
*
* @ since 2.0
* @ since 2.7 Updated to use EDD_Discount object
* @ since 3.0 Updated to call use new query class .
*
* @ param string $field The field to retrieve the discount with .
* @ param mixed $value The value for $field .
* @ return mixed EDD_Discount | bool EDD_Discount object or false if not found .
*/
function edd_get_discount_by ( $field = '' , $value = '' ) {
$discounts = new EDD\Compat\Discount_Query ();
// Return discount
return $discounts -> get_item_by ( $field , $value );
}
/**
* Retrieve discount by a given field
*
* @ since 2.0
* @ since 2.7 Updated to use EDD_Discount object
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID .
* @ param string $field The field to retrieve the discount with .
* @ return mixed object | bool EDD_Discount object or false if not found .
*/
function edd_get_discount_field ( $discount_id , $field = '' ) {
$discount = edd_get_discount ( $discount_id );
// Check that field exists
return isset ( $discount -> { $field } )
? $discount -> { $field }
: null ;
}
/**
* Update a discount
*
* @ since 3.0
* @ param int $discount_id Discount ID .
* @ param array $data
* @ return int
*/
function edd_update_discount ( $discount_id = 0 , $data = array () ) {
// Pre-3.0 pre action
do_action ( 'edd_pre_update_discount' , $data , $discount_id );
// Product requirements.
if ( isset ( $data [ 'product_reqs' ] ) && ! empty ( $data [ 'product_reqs' ] ) ) {
if ( is_string ( $data [ 'product_reqs' ] ) ) {
$data [ 'product_reqs' ] = maybe_unserialize ( $data [ 'product_reqs' ] );
}
if ( is_array ( $data [ 'product_reqs' ] ) ) {
edd_delete_adjustment_meta ( $discount_id , 'product_requirement' );
foreach ( $data [ 'product_reqs' ] as $product_requirement ) {
edd_add_adjustment_meta ( $discount_id , 'product_requirement' , $product_requirement );
}
}
unset ( $data [ 'product_reqs' ] );
} elseif ( isset ( $data [ 'product_reqs' ] ) ) {
edd_delete_adjustment_meta ( $discount_id , 'product_requirement' );
// We don't have product conditions when there are no product requirements.
edd_delete_adjustment_meta ( $discount_id , 'product_condition' );
unset ( $data [ 'product_condition' ] );
}
// Excluded products are handled differently.
if ( isset ( $data [ 'excluded_products' ] ) && ! empty ( $data [ 'excluded_products' ] ) ) {
if ( is_string ( $data [ 'excluded_products' ] ) ) {
$data [ 'excluded_products' ] = maybe_unserialize ( $data [ 'excluded_products' ] );
}
if ( is_array ( $data [ 'excluded_products' ] ) ) {
edd_delete_adjustment_meta ( $discount_id , 'excluded_product' );
foreach ( $data [ 'excluded_products' ] as $excluded_product ) {
edd_add_adjustment_meta ( $discount_id , 'excluded_product' , $excluded_product );
}
}
unset ( $data [ 'excluded_products' ] );
} elseif ( isset ( $data [ 'excluded_products' ] ) ) {
edd_delete_adjustment_meta ( $discount_id , 'excluded_product' );
}
if ( isset ( $data [ 'product_condition' ] ) ) {
$product_condition = sanitize_text_field ( $data [ 'product_condition' ] );
edd_update_adjustment_meta ( $discount_id , 'product_condition' , $product_condition );
}
$discounts = new EDD\Compat\Discount_Query ();
$retval = $discounts -> update_item ( $discount_id , $data );
// Pre-3.0 post action
do_action ( 'edd_post_update_discount' , $data , $discount_id );
return $retval ;
}
/**
* Get Discounts
*
* Retrieves an array of all available discount codes .
*
* @ since 1.0
* @ param array $args Query arguments
* @ return mixed array if discounts exist , false otherwise
*/
function edd_get_discounts ( $args = array () ) {
// Parse arguments.
$r = wp_parse_args ( $args , array (
'number' => 30
) );
// Back compat for old query arg.
if ( isset ( $r [ 'posts_per_page' ] ) ) {
$r [ 'number' ] = $r [ 'posts_per_page' ];
}
// Instantiate a query object.
$discounts = new EDD\Compat\Discount_Query ();
// Return discounts
return $discounts -> query ( $r );
}
/**
* Return total number of discounts
*
* @ since 3.0
*
* @ param array $args Arguments .
* @ return int
*/
function edd_get_discount_count ( $args = array () ) {
// Parse args.
$r = wp_parse_args ( $args , array (
'count' => true
) );
// Query for count(s).
$discounts = new EDD\Compat\Discount_Query ( $r );
// Return count(s).
return absint ( $discounts -> found_items );
}
/**
* Query for and return array of discount counts , keyed by status .
*
* @ since 3.0
*
* @ return array
*/
function edd_get_discount_counts ( $args = array () ) {
// Parse arguments
$r = wp_parse_args ( $args , array (
'count' => true ,
'groupby' => 'status'
) );
// Query for count.
$counts = new EDD\Compat\Discount_Query ( $r );
// Format & return
return edd_format_counts ( $counts , $r [ 'groupby' ] );
}
/**
* Query for discount notes .
*
* @ since 3.0
*
* @ param int $discount_id Discount ID .
* @ return array Retrieved notes .
*/
function edd_get_discount_notes ( $discount_id = 0 ) {
return edd_get_notes ( array (
'object_id' => $discount_id ,
'object_type' => 'discount' ,
'order' => 'asc'
) );
}
/**
* Checks if there is any active discounts , returns a boolean .
*
* @ since 1.0
* @ since 3.0 Updated to be more efficient and make direct calls to the EDD_Discount object .
*
* @ return bool
*/
function edd_has_active_discounts () {
// Query for active discounts.
$discounts = edd_get_discounts ( array (
'number' => 10 ,
'status' => 'active'
) );
// Bail if none.
if ( empty ( $discounts ) ) {
return false ;
}
// Check each discount for active status, applying filters, etc...
foreach ( $discounts as $discount ) {
/** @var $discount EDD_Discount */
if ( $discount -> is_active ( false , true ) ) {
return true ;
}
}
return false ;
}
/**
* Stores a discount code . If the code already exists , it updates it , otherwise
* it creates a new one .
*
* @ internal This method exists for backwards compatibility . `edd_add_discount()` should be used .
*
2023-07-03 14:52:54 +00:00
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to use new query class .
* @ deprecated 3.0 Use edd_add_discount ()
2022-11-27 15:03:07 +00:00
*
* @ param array $details Discount args .
* @ param int $discount_id Discount ID .
* @ return mixed bool | int The discount ID of the discount code , or false on failure .
*/
function edd_store_discount ( $details , $discount_id = null ) {
2023-07-03 14:52:54 +00:00
edd_debug_log (
sprintf (
__ ( '%1$s is deprecated since Easy Digital Downloads version %2$s! Use %3$s instead.' , 'easy-digital-downloads' ),
__FUNCTION__ ,
'3.0' ,
'edd_add_discount()'
),
true
);
2022-11-27 15:03:07 +00:00
// Set default return value to false.
$return = false ;
// Back-compat for start date.
if ( isset ( $details [ 'start' ] ) && strstr ( $details [ 'start' ], '/' ) ) {
2023-07-03 14:52:54 +00:00
$time_format = date ( 'H:i:s' , strtotime ( $details [ 'start' ] ) );
$date_format = date ( 'Y-m-d' , strtotime ( $details [ 'start' ] ) ) . ' ' . $time_format ;
$details [ 'start_date' ] = edd_get_utc_equivalent_date ( EDD () -> utils -> date ( $date_format , edd_get_timezone_id (), false ));
2022-11-27 15:03:07 +00:00
unset ( $details [ 'start' ] );
}
// Back-compat for end date.
if ( isset ( $details [ 'expiration' ] ) && strstr ( $details [ 'expiration' ], '/' ) ) {
2023-07-03 14:52:54 +00:00
$time_format = date ( 'H:i:s' , strtotime ( $details [ 'expiration' ] ) );
$date_format = date ( 'Y-m-d' , strtotime ( $details [ 'expiration' ] ) ) . ' ' . ( '00:00:00' !== $time_format ? $time_format : '23:59:59' );
$details [ 'end_date' ] = edd_get_utc_equivalent_date ( EDD () -> utils -> date ( $date_format , edd_get_timezone_id (), false ) );
2022-11-27 15:03:07 +00:00
unset ( $details [ 'expiration' ] );
}
/**
* Filters the args before being inserted into the database . This hook
* exists for backwards compatibility purposes .
*
* @ since 2.7
*
* @ param array $details Discount args .
*/
$details = apply_filters ( 'edd_insert_discount' , $details );
/**
* Fires before the discount has been added to the database . This hook
* exists for backwards compatibility purposes . It fires before the
* call to `EDD_Discount::convert_legacy_args` to ensure the arguments
* maintain backwards compatible array keys .
*
* @ since 2.7
*
* @ param array $details Discount args .
*/
do_action ( 'edd_pre_insert_discount' , $details );
// Convert legacy arguments to new ones accepted by `edd_add_discount()`.
$details = EDD_Discount :: convert_legacy_args ( $details );
if ( null === $discount_id ) {
$return = ( int ) edd_add_discount ( $details );
} else {
edd_update_discount ( $discount_id , $details );
$return = $discount_id ;
}
return $return ;
}
/**
* Deletes a discount code .
*
* @ internal This method exists for backwards compatibility . `edd_delete_discount()` should be used .
*
* @ since 1.0
* @ deprecated 3.0
*
* @ param int $discount_id Discount ID .
*/
function edd_remove_discount ( $discount_id = 0 ) {
edd_delete_discount ( $discount_id );
}
/**
* Updates a discount status from one status to another .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID ( default : 0 )
* @ param string $new_status New status ( default : active )
*
* @ return bool Whether the status has been updated or not .
*/
function edd_update_discount_status ( $discount_id = 0 , $new_status = 'active' ) {
// Bail if an invalid ID is passed.
if ( $discount_id <= 0 ) {
return false ;
}
// Set defaults.
$updated = false ;
$new_status = sanitize_key ( $new_status );
$discount = edd_get_discount ( $discount_id );
// No change.
if ( $new_status === $discount -> status ) {
return true ;
}
// Try to update status.
if ( ! empty ( $discount -> id ) ) {
$updated = ( bool ) edd_update_discount ( $discount -> id , array (
'status' => $new_status
) );
}
// Return.
return $updated ;
}
/**
* Checks to see if a discount code already exists .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount () .
*
* @ param int $discount_id Discount ID .
*
* @ return bool Whether or not the discount exists .
*/
function edd_discount_exists ( $discount_id ) {
$discount = edd_get_discount ( $discount_id );
return $discount instanceof EDD_Discount && $discount -> exists ();
}
/**
* Checks whether a discount code is active .
*
* @ since 1.0
* @ since 2.6 . 11 Added $update parameter .
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount () .
*
* @ param int $discount_id Discount ID .
* @ param bool $update Update the discount to expired if an one is found but has an active status /
* @ param bool $set_error Whether an error message should be set in session .
* @ return bool Whether or not the discount is active .
*/
function edd_is_discount_active ( $discount_id = 0 , $update = true , $set_error = true ) {
$discount = edd_get_discount ( $discount_id );
if ( ! $discount instanceof EDD_Discount ) {
return false ;
}
return $discount -> is_active ( $update , $set_error );
}
/**
* Retrieve the discount code .
*
* @ since 1.4
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field ()
*
* @ param int $discount_id Discount ID .
* @ return string $code Discount Code .
*/
function edd_get_discount_code ( $discount_id = 0 ) {
return edd_get_discount_field ( $discount_id , 'code' );
}
/**
* Retrieve the discount code start date .
*
* @ since 1.4
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field ()
*
* @ param int $discount_id Discount ID .
* @ return string $start Discount start date .
*/
function edd_get_discount_start_date ( $discount_id = 0 ) {
return edd_get_discount_field ( $discount_id , 'start_date' );
}
/**
* Retrieve the discount code expiration date .
*
* @ since 1.4
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field ()
*
* @ param int $discount_id Discount ID .
* @ return string $expiration Discount expiration .
*/
function edd_get_discount_expiration ( $discount_id = 0 ) {
return edd_get_discount_field ( $discount_id , 'end_date' );
}
/**
* Retrieve the maximum uses that a certain discount code .
*
* @ since 1.4
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field ()
*
* @ param int $discount_id Discount ID .
* @ return int $max_uses Maximum number of uses for the discount code .
*/
function edd_get_discount_max_uses ( $discount_id = 0 ) {
return edd_get_discount_field ( $discount_id , 'max_uses' );
}
/**
* Retrieve number of times a discount has been used .
*
* @ since 1.4
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field () .
*
* @ param int $discount_id Discount ID .
* @ return int $uses Number of times a discount has been used .
*/
function edd_get_discount_uses ( $discount_id = 0 ) {
return ( int ) edd_get_discount_field ( $discount_id , 'use_count' );
}
/**
* Retrieve the minimum purchase amount for a discount .
*
* @ since 1.4
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field () .
*
* @ param int $discount_id Discount ID .
* @ return float $min_price Minimum purchase amount .
*/
function edd_get_discount_min_price ( $discount_id = 0 ) {
return edd_format_amount ( edd_get_discount_field ( $discount_id , 'min_charge_amount' ) );
}
/**
* Retrieve the discount amount .
*
* @ since 1.4
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field () .
*
* @ param int $discount_id Discount ID .
* @ return float $amount Discount amount .
*/
function edd_get_discount_amount ( $discount_id = 0 ) {
return edd_get_discount_field ( $discount_id , 'amount' );
}
/**
* Retrieve the discount type
*
* @ since 1.4
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field () .
*
* @ param int $discount_id Discount ID .
* @ return string $type Discount type
*/
function edd_get_discount_type ( $discount_id = 0 ) {
return edd_get_discount_field ( $discount_id , 'type' );
}
/**
* Retrieve the products the discount cannot be applied to .
*
* @ since 1.9
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID .
* @ return array $excluded_products IDs of the required products .
*/
function edd_get_discount_excluded_products ( $discount_id = 0 ) {
$discount = edd_get_discount ( $discount_id );
return $discount instanceof EDD_Discount ? $discount -> excluded_products : array ();
}
/**
* Retrieve the discount product requirements .
*
* @ since 1.5
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID .
* @ return array $product_reqs IDs of the required products .
*/
function edd_get_discount_product_reqs ( $discount_id = 0 ) {
$discount = edd_get_discount ( $discount_id );
return $discount instanceof EDD_Discount ? $discount -> product_reqs : array ();
}
/**
* Retrieve the product condition .
*
* @ since 1.5
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field ()
*
* @ param int $discount_id Discount ID .
*
* @ return string Product condition .
*/
function edd_get_discount_product_condition ( $discount_id = 0 ) {
$discount = edd_get_discount ( $discount_id );
return $discount instanceof EDD_Discount ? $discount -> product_condition : '' ;
}
/**
* Retrieves the discount status label .
*
* @ since 2.9
*
* @ param int $discount_id Discount ID .
* @ return string Product condition .
*/
function edd_get_discount_status_label ( $discount_id = null ) {
$discount = edd_get_discount ( $discount_id );
return $discount instanceof EDD_Discount ? $discount -> get_status_label () : '' ;
}
/**
* Check if a discount is not global .
*
* By default discounts are applied to all products in the cart . Non global discounts are
* applied only to the products selected as requirements .
*
* @ since 1.5
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Please use edd_get_discount_scope () instead .
*
* @ param int $discount_id Discount ID .
*
* @ return boolean Whether or not discount code is not global .
*/
function edd_is_discount_not_global ( $discount_id = 0 ) {
return ( 'not_global' === edd_get_discount_field ( $discount_id , 'scope' ) );
}
/**
* Retrieve the discount scope .
*
* By default this will return " global " as discounts are applied to all products in the cart . Non global discounts are
* applied only to the products selected as requirements .
*
* @ since 3.0
*
* @ param int $discount_id Discount ID .
*
* @ return string global or not_global .
*/
function edd_get_discount_scope ( $discount_id = 0 ) {
return edd_get_discount_field ( $discount_id , 'scope' );
}
/**
* Checks whether a discount code is expired .
*
* @ since 1.0
* @ since 2.6 . 11 Added $update parameter .
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID .
* @ param bool $update Update the discount to expired if an one is found but has an active status .
* @ return bool Whether on not the discount has expired .
*/
function edd_is_discount_expired ( $discount_id = 0 , $update = true ) {
$discount = edd_get_discount ( $discount_id );
return ! empty ( $discount -> id )
? $discount -> is_expired ( $update )
: false ;
}
/**
* Checks whether a discount code is available to use yet ( start date ) .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID .
* @ param bool $set_error Whether an error message be set in session .
* @ return bool Is discount started ?
*/
function edd_is_discount_started ( $discount_id = 0 , $set_error = true ) {
$discount = edd_get_discount ( $discount_id );
return ! empty ( $discount -> id )
? $discount -> is_started ( $set_error )
: false ;
}
/**
* Is Discount Maxed Out .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID .
* @ param bool $set_error Whether an error message be set in session .
* @ return bool Is discount maxed out ?
*/
function edd_is_discount_maxed_out ( $discount_id = 0 , $set_error = true ) {
$discount = edd_get_discount ( $discount_id );
return ! empty ( $discount -> id )
? $discount -> is_maxed_out ( $set_error )
: false ;
}
/**
* Checks to see if the minimum purchase amount has been met .
*
* @ since 1.1 . 7
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID .
* @ param bool $set_error Whether an error message be set in session .
* @ return bool Whether the minimum amount has been met or not .
*/
function edd_discount_is_min_met ( $discount_id = 0 , $set_error = true ) {
$discount = edd_get_discount ( $discount_id );
return ! empty ( $discount -> id )
? $discount -> is_min_price_met ( $set_error )
: false ;
}
/**
* Is the discount limited to a single use per customer ?
*
* @ since 1.5
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_field ()
*
* @ param int $discount_id Discount ID .
*
* @ return bool Whether the discount is single use or not .
*/
function edd_discount_is_single_use ( $discount_id = 0 ) {
return ( bool ) edd_get_discount_field ( $discount_id , 'once_per_customer' );
}
/**
* Checks to see if the required products are in the cart
*
* @ since 1.5
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param int $discount_id Discount ID .
* @ param bool $set_error Whether an error message be set in session .
* @ return bool Are required products in the cart for the discount to hold .
*/
function edd_discount_product_reqs_met ( $discount_id = 0 , $set_error = true ) {
$discount = edd_get_discount ( $discount_id );
return $discount instanceof EDD_Discount && $discount -> is_product_requirements_met ( $set_error );
}
/**
* Checks to see if a user has already used a discount .
*
* @ since 1.1 . 5
* @ since 1.5 Added $discount_id parameter .
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount ()
*
* @ param string $code Discount Code .
* @ param string $user User info .
* @ param int $discount_id Discount ID .
* @ param bool $set_error Whether an error message be set in session
*
* @ return bool $return Whether the the discount code is used .
*/
function edd_is_discount_used ( $code = null , $user = '' , $discount_id = 0 , $set_error = true ) {
$discount = ( null == $code )
? edd_get_discount ( $discount_id )
: edd_get_discount_by_code ( $code );
return $discount instanceof EDD_Discount && $discount -> is_used ( $user , $set_error );
}
/**
* Check whether a discount code is valid ( when purchasing ) .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_by_code ()
*
* @ param string $code Discount Code .
* @ param string $user User info .
* @ param bool $set_error Whether an error message be set in session .
* @ return bool Whether the discount code is valid .
*/
function edd_is_discount_valid ( $code = '' , $user = '' , $set_error = true ) {
$discount = edd_get_discount_by_code ( $code );
if ( ! empty ( $discount -> id ) ) {
return $discount -> is_valid ( $user , $set_error );
} elseif ( $set_error ) {
edd_set_error ( 'edd-discount-error' , _x ( 'This discount is invalid.' , 'error for when a discount is invalid based on its configuration' , 'easy-digital-downloads' ) );
return false ;
} else {
return false ;
}
}
/**
* Retrieves a discount ID from the code .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_by_code ()
*
* @ param string $code Discount code .
* @ return int | bool Discount ID , or false if discount does not exist .
*/
function edd_get_discount_id_by_code ( $code = '' ) {
$discount = edd_get_discount_by_code ( $code );
return ( $discount instanceof EDD_Discount ) ? $discount -> id : false ;
}
/**
* Get Discounted Amount .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_by_code ()
*
* @ param string $code Code to calculate a discount for .
* @ param mixed string | int $base_price Price before discount .
* @ return string Amount after discount .
*/
function edd_get_discounted_amount ( $code = '' , $base_price = 0 ) {
$discount = edd_get_discount_by_code ( $code );
return ! empty ( $discount -> id )
? $discount -> get_discounted_amount ( $base_price )
: $base_price ;
}
/**
* Increases the use count of a discount code .
*
* @ since 1.0
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_by_code ()
*
* @ param string $code Discount code to be incremented .
* @ return int New usage .
*/
function edd_increase_discount_usage ( $code = '' ) {
$discount = edd_get_discount_by_code ( $code );
// Increase if discount exists
return ! empty ( $discount -> id )
? ( int ) $discount -> increase_usage ()
: false ;
}
/**
* Decreases the use count of a discount code .
*
* @ since 2.5 . 7
* @ since 2.7 Updated to use EDD_Discount object .
* @ since 3.0 Updated to call edd_get_discount_by_code ()
*
* @ param string $code Discount code to be decremented .
* @ return int New usage .
*/
function edd_decrease_discount_usage ( $code = '' ) {
$discount = edd_get_discount_by_code ( $code );
// Decrease if discount exists
return ! empty ( $discount -> id )
? ( int ) $discount -> decrease_usage ()
: false ;
}
/**
* Format Discount Rate
*
* @ since 1.0
* @ param string $type Discount code type
* @ param string | int $amount Discount code amount
* @ return string $amount Formatted amount
*/
function edd_format_discount_rate ( $type = '' , $amount = '' ) {
return ( 'flat' === $type )
? edd_currency_filter ( edd_format_amount ( $amount ) )
: edd_format_amount ( $amount ) . '%' ;
}
/**
* Retrieves a discount amount for an item .
*
* Calculates an amount based on the context of other items .
*
* @ since 3.0
*
* @ global float $edd_flat_discount_total Track flat rate discount total for penny adjustments .
* @ link https :// github . com / easydigitaldownloads / easy - digital - downloads / issues / 2757
*
* @ param array $item {
* Order Item data , matching Cart line item format .
*
* @ type string $id Download ID .
* @ type array $options {
* Download options .
*
* @ type string $price_id Download Price ID .
* }
* @ type int $quantity Purchase quantity .
* }
* @ param array $items All items ( including item being calculated ) .
* @ param \EDD_Discount [] | string [] $discounts Discount to determine adjustment from .
* A discount code can be passed as a string .
* @ param int $item_unit_price ( Optional ) Pass in a defined price for a specific context , such as the cart .
* @ return float Discount amount . 0 if Discount is invalid or no Discount is applied .
*/
function edd_get_item_discount_amount ( $item , $items , $discounts , $item_unit_price = false ) {
global $edd_flat_discount_total ;
// Validate item.
if ( empty ( $item ) || empty ( $item [ 'id' ] ) ) {
return 0 ;
}
if ( ! isset ( $item [ 'quantity' ] ) ) {
return 0 ;
}
if ( ! isset ( $item [ 'options' ] ) ) {
$item [ 'options' ] = array ();
/*
* Support for variable pricing when the `item_number` key is set ( cart details ) .
*/
if ( isset ( $item [ 'item_number' ][ 'options' ] ) ) {
$item [ 'options' ] = $item [ 'item_number' ][ 'options' ];
}
}
// Validate and normalize Discounts.
$discounts = array_map (
function ( $discount ) {
// Convert a Discount code to a Discount object.
if ( is_string ( $discount ) ) {
$discount = edd_get_discount_by_code ( $discount );
}
if ( ! $discount instanceof \EDD_Discount ) {
return false ;
}
return $discount ;
},
$discounts
);
$discounts = array_filter ( $discounts );
if ( false === $item_unit_price ) {
// Determine the price of the item.
if ( edd_has_variable_prices ( $item [ 'id' ] ) ) {
// Mimics the original behavior of `\EDD_Cart::get_item_amount()` that
// does not fallback to the first Price ID if none is provided.
if ( ! isset ( $item [ 'options' ][ 'price_id' ] ) ) {
return 0 ;
}
$item_unit_price = edd_get_price_option_amount ( $item [ 'id' ], $item [ 'options' ][ 'price_id' ] );
} else {
$item_unit_price = edd_get_download_price ( $item [ 'id' ] );
}
}
$item_amount = ( $item_unit_price * $item [ 'quantity' ] );
$discount_amount = 0 ;
foreach ( $discounts as $discount ) {
$reqs = $discount -> get_product_reqs ();
$excluded_products = $discount -> get_excluded_products ();
// Make sure requirements are set and that this discount shouldn't apply to the whole cart.
if ( ! empty ( $reqs ) && 'global' !== $discount -> get_scope () ) {
// This is a product(s) specific discount.
foreach ( $reqs as $download_id ) {
if ( $download_id == $item [ 'id' ] && ! in_array ( $item [ 'id' ], $excluded_products ) ) {
$discount_amount += ( $item_amount - $discount -> get_discounted_amount ( $item_amount ) );
}
}
} else {
// This is a global cart discount.
if ( ! in_array ( $item [ 'id' ], $excluded_products ) ) {
if ( 'flat' === $discount -> get_type () ) {
// In order to correctly record individual item amounts, global flat rate discounts
// are distributed across all items.
//
// The discount amount is divided by the number of items in the cart and then a
// portion is evenly applied to each item.
$items_amount = 0 ;
foreach ( $items as $i ) {
if ( ! in_array ( $i [ 'id' ], $excluded_products ) ) {
$i_amount = 0 ;
if ( edd_has_variable_prices ( $i [ 'id' ] ) ) {
$price_id = isset ( $i [ 'options' ][ 'price_id' ] ) ? $i [ 'options' ][ 'price_id' ] : $i [ 'item_number' ][ 'options' ][ 'price_id' ];
$i_amount = edd_get_price_option_amount ( $i [ 'id' ], $price_id );
} else {
$i_amount = edd_get_download_price ( $i [ 'id' ] );
}
$items_amount += ( $i_amount * $i [ 'quantity' ] );
}
}
$subtotal_percent = ! empty ( $items_amount ) ? ( $item_amount / $items_amount ) : 0 ;
$discount_amount += ( $discount -> get_amount () * $subtotal_percent );
$edd_flat_discount_total += round ( $discount_amount , edd_currency_decimal_filter () );
if ( $item [ 'id' ] === end ( $items )[ 'id' ] && $edd_flat_discount_total < $discount -> get_amount () ) {
$adjustment = ( $discount -> get_amount () - $edd_flat_discount_total );
$discount_amount += $adjustment ;
}
if ( $discount_amount > $item_amount ) {
$discount_amount = $item_amount ;
}
} else {
$discount_amount += ( $item_amount - $discount -> get_discounted_amount ( $item_amount ) );
}
}
}
}
return $discount_amount ;
}
/** Cart **********************************************************************/
/**
* Set the active discount for the shopping cart
*
* @ since 1.4 . 1
* @ param string $code Discount code
* @ return string [] All currently active discounts
*/
function edd_set_cart_discount ( $code = '' ) {
// Get all active cart discounts
if ( edd_multiple_discounts_allowed () ) {
$discounts = edd_get_cart_discounts ();
// Only one discount allowed per purchase, so override any existing
} else {
$discounts = false ;
}
if ( $discounts ) {
$key = array_search ( strtolower ( $code ), array_map ( 'strtolower' , $discounts ) );
// Can't set the same discount more than once
if ( false !== $key ) {
unset ( $discounts [ $key ] );
}
$discounts [] = $code ;
} else {
$discounts = array ();
$discounts [] = $code ;
}
EDD () -> session -> set ( 'cart_discounts' , implode ( '|' , $discounts ) );
do_action ( 'edd_cart_discount_set' , $code , $discounts );
do_action ( 'edd_cart_discounts_updated' , $discounts );
return $discounts ;
}
/**
* Remove an active discount from the shopping cart
*
* @ since 1.4 . 1
* @ param string $code Discount code
* @ return array $discounts All remaining active discounts
*/
function edd_unset_cart_discount ( $code = '' ) {
$discounts = edd_get_cart_discounts ();
if ( $discounts ) {
$discounts = array_map ( 'strtoupper' , $discounts );
$key = array_search ( strtoupper ( $code ), $discounts );
if ( false !== $key ) {
unset ( $discounts [ $key ] );
}
$discounts = implode ( '|' , array_values ( $discounts ) );
// update the active discounts
EDD () -> session -> set ( 'cart_discounts' , $discounts );
}
do_action ( 'edd_cart_discount_removed' , $code , $discounts );
do_action ( 'edd_cart_discounts_updated' , $discounts );
return $discounts ;
}
/**
* Remove all active discounts
*
* @ since 1.4 . 1
* @ return void
*/
function edd_unset_all_cart_discounts () {
EDD () -> cart -> remove_all_discounts ();
}
/**
* Retrieve the currently applied discount
*
* @ since 1.4 . 1
* @ return array $discounts The active discount codes
*/
function edd_get_cart_discounts () {
return EDD () -> cart -> get_discounts ();
}
/**
* Check if the cart has any active discounts applied to it
*
* @ since 1.4 . 1
* @ return bool
*/
function edd_cart_has_discounts () {
return EDD () -> cart -> has_discounts ();
}
/**
* Retrieves the total discounted amount on the cart
*
* @ since 1.4 . 1
*
* @ param bool $discounts Discount codes
*
* @ return float | mixed | void Total discounted amount
*/
function edd_get_cart_discounted_amount ( $discounts = false ) {
return EDD () -> cart -> get_discounted_amount ( $discounts );
}
/**
* Get the discounted amount on a price
*
* @ since 1.9
* @ param array $item Cart item array
* @ param bool | string $discount False to use the cart discounts or a string to check with a discount code
* @ return float The discounted amount
*/
function edd_get_cart_item_discount_amount ( $item = array (), $discount = false ) {
return EDD () -> cart -> get_item_discount_amount ( $item , $discount );
}
/**
* Outputs the HTML for all discounts applied to the cart
*
* @ since 1.4 . 1
*
* @ return void
*/
function edd_cart_discounts_html () {
echo edd_get_cart_discounts_html ();
}
/**
* Retrieves the HTML for all discounts applied to the cart
*
* @ since 1.4 . 1
*
* @ param mixed $discounts Array of cart discounts .
2023-03-17 22:34:04 +00:00
* @ return string
2022-11-27 15:03:07 +00:00
*/
function edd_get_cart_discounts_html ( $discounts = false ) {
if ( ! $discounts ) {
$discounts = EDD () -> cart -> get_discounts ();
}
if ( empty ( $discounts ) ) {
2023-03-17 22:34:04 +00:00
return apply_filters ( 'edd_get_cart_discounts_html' , '' , $discounts , 0 , '' );
2022-11-27 15:03:07 +00:00
}
$html = _n ( 'Discount' , 'Discounts' , count ( $discounts ), 'easy-digital-downloads' ) . ': ' ;
foreach ( $discounts as $discount ) {
$discount_id = edd_get_discount_id_by_code ( $discount );
$discount_amount = 0 ;
$items = EDD () -> cart -> get_contents_details ();
if ( is_array ( $items ) && ! empty ( $items ) ) {
foreach ( $items as $key => $item ) {
$discount_amount += edd_get_item_discount_amount ( $item , $items , array ( $discount ), $item [ 'item_price' ] );
}
}
$type = edd_get_discount_type ( $discount_id );
$rate = edd_format_discount_rate ( $type , edd_get_discount_amount ( $discount_id ) );
$remove_url = add_query_arg (
array (
'edd_action' => 'remove_cart_discount' ,
'discount_id' => urlencode ( $discount_id ),
'discount_code' => urlencode ( $discount ),
),
edd_get_checkout_uri ()
);
$discount_html = '' ;
$discount_html .= " <span class= \" edd_discount \" > \n " ;
$discount_amount = edd_currency_filter ( edd_format_amount ( $discount_amount ) );
$discount_html .= " <span class= \" edd_discount_total \" > { $discount } – { $discount_amount } </span> \n " ;
if ( 'percent' === $type ) {
$discount_html .= " <span class= \" edd_discount_rate \" >( $rate )</span> \n " ;
}
$discount_html .= sprintf (
'<a href="%s" data-code="%s" class="edd_discount_remove"><span class="screen-reader-text">%s</span></a>' ,
esc_url ( $remove_url ),
esc_attr ( $discount ),
esc_attr__ ( 'Remove discount' , 'easy-digital-downloads' )
);
$discount_html .= " </span> \n " ;
$html .= apply_filters ( 'edd_get_cart_discount_html' , $discount_html , $discount , $rate , $remove_url );
}
return apply_filters ( 'edd_get_cart_discounts_html' , $html , $discounts , $rate , $remove_url );
}
/**
* Show the fully formatted cart discount
*
* Note the $formatted parameter was removed from the display_cart_discount () function
* within EDD_Cart in 2.7 as it was a redundant parameter .
*
* @ since 1.4 . 1
* @ param bool $formatted
* @ param bool $echo Echo ?
* @ return string $amount Fully formatted cart discount
*/
function edd_display_cart_discount ( $formatted = false , $echo = false ) {
if ( ! $echo ) {
return EDD () -> cart -> display_cart_discount ( $echo );
} else {
EDD () -> cart -> display_cart_discount ( $echo );
}
}
/**
* Processes a remove discount from cart request
*
* @ since 1.4 . 1
* @ return void
*/
function edd_remove_cart_discount () {
// Get ID
$discount_id = isset ( $_GET [ 'discount_id' ] )
? absint ( $_GET [ 'discount_id' ] )
: 0 ;
// Get code
$discount_code = isset ( $_GET [ 'discount_code' ] )
? urldecode ( $_GET [ 'discount_code' ] )
: '' ;
// Bail if either ID or code are empty
if ( empty ( $discount_id ) || empty ( $discount_code ) ) {
return ;
}
// Pre-3.0 pre action
do_action ( 'edd_pre_remove_cart_discount' , $discount_id );
edd_unset_cart_discount ( $discount_code );
// Pre-3.0 post action
do_action ( 'edd_post_remove_cart_discount' , $discount_id );
// Redirect
edd_redirect ( edd_get_checkout_uri () );
}
add_action ( 'edd_remove_cart_discount' , 'edd_remove_cart_discount' );
/**
* Checks whether discounts are still valid when removing items from the cart
*
* If a discount requires a certain product , and that product is no longer in
* the cart , the discount is removed .
*
* @ since 1.5 . 2
*
* @ param int $cart_key
*/
function edd_maybe_remove_cart_discount ( $cart_key = 0 ) {
$discounts = edd_get_cart_discounts ();
if ( empty ( $discounts ) ) {
return ;
}
foreach ( $discounts as $discount ) {
if ( ! edd_is_discount_valid ( $discount ) ) {
edd_unset_cart_discount ( $discount );
}
}
}
add_action ( 'edd_post_remove_from_cart' , 'edd_maybe_remove_cart_discount' );
/**
* Checks whether multiple discounts can be applied to the same purchase
*
* @ since 1.7
* @ return bool
*/
function edd_multiple_discounts_allowed () {
$ret = edd_get_option ( 'allow_multiple_discounts' , false );
return ( bool ) apply_filters ( 'edd_multiple_discounts_allowed' , $ret );
}
/**
* Listens for a discount and automatically applies it if present and valid
*
* @ since 2.0
* @ return void
*/
function edd_listen_for_cart_discount () {
// Bail if in admin
if ( is_admin () ) {
return ;
}
// Array stops the bulk delete of discount codes from storing as a preset_discount
if ( empty ( $_REQUEST [ 'discount' ] ) || is_array ( $_REQUEST [ 'discount' ] ) ) {
return ;
}
$code = preg_replace ( '/[^a-zA-Z0-9-_]+/' , '' , $_REQUEST [ 'discount' ] );
EDD () -> session -> set ( 'preset_discount' , $code );
}
add_action ( 'init' , 'edd_listen_for_cart_discount' , 0 );
/**
* Applies the preset discount , if any . This is separated from edd_listen_for_cart_discount () in order to allow items to be
* added to the cart and for it to persist across page loads if necessary
*
* @ return void
*/
function edd_apply_preset_discount () {
// Bail if in admin
if ( is_admin () ) {
return ;
}
$code = sanitize_text_field ( EDD () -> session -> get ( 'preset_discount' ) );
if ( empty ( $code ) ) {
return ;
}
if ( ! edd_is_discount_valid ( $code , '' , false ) ) {
return ;
}
$code = apply_filters ( 'edd_apply_preset_discount' , $code );
edd_set_cart_discount ( $code );
EDD () -> session -> set ( 'preset_discount' , null );
}
add_action ( 'init' , 'edd_apply_preset_discount' , 999 );
/**
* Validate discount code , optionally against an array of download IDs .
* Note : this function does not evaluate whether a current user can use the discount ,
* or check the discount minimum cart requirement .
*
* @ param int $discount_id Discount ID .
* @ param array $download_ids Array of download IDs .
*
* @ return boolean True if discount holds , false otherwise .
*/
function edd_validate_discount ( $discount_id = 0 , $download_ids = array () ) {
// Bail if discount ID not passed.
if ( empty ( $discount_id ) ) {
return false ;
}
$discount = edd_get_discount ( $discount_id );
// Bail if discount not found.
if ( ! $discount ) {
return false ;
}
// Check if discount is active, started, and not maxed out.
if ( ! $discount -> is_active ( true , false ) || ! $discount -> is_started ( false ) || $discount -> is_maxed_out ( false ) ) {
return false ;
}
$product_requirements = $discount -> get_product_reqs ();
$excluded_products = $discount -> get_excluded_products ();
// Return true if there are no requirements/excluded products set.
if ( empty ( $product_requirements ) && empty ( $excluded_products ) ) {
return true ;
}
// At this point, we assume the discount is valid.
$is_valid = true ;
$product_requirements = array_map ( 'absint' , $product_requirements );
asort ( $product_requirements );
$product_requirements = array_filter ( array_values ( $product_requirements ) );
if ( ! empty ( $product_requirements ) ) {
$matches = array_intersect ( $product_requirements , $download_ids );
switch ( $discount -> get_product_condition () ) {
case 'all' :
$is_valid = count ( $matches ) === count ( $product_requirements );
break ;
default :
$is_valid = 0 < count ( $matches );
}
}
$excluded_products = array_map ( 'absint' , $excluded_products );
asort ( $excluded_products );
$excluded_products = array_filter ( array_values ( $excluded_products ) );
if ( ! empty ( $excluded_products ) ) {
$is_valid = false === ( bool ) array_intersect ( $excluded_products , $download_ids );
}
/**
* Filters the validity of a discount .
*
* @ since 3.0
*
* @ param bool $is_valid True if valid , false otherwise .
* @ param \EDD_Discount $discount Discount object .
* @ param array $download_ids Download IDs to check against .
*/
return apply_filters ( 'edd_validate_discount' , $is_valid , $discount , $download_ids );
}