360 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			360 lines
		
	
	
		
			8.5 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
| <?php
 | |
| /**
 | |
|  * Fees
 | |
|  *
 | |
|  * This class is for adding arbitrary fees to the cart. Fees can be positive or negative (discounts)
 | |
|  *
 | |
|  * @package     EDD
 | |
|  * @subpackage  Classes/Fees
 | |
|  * @copyright   Copyright (c) 2018, Easy Digital Downloads, LLC
 | |
|  * @license     http://opensource.org/licenses/gpl-2.0.php GNU Public License
 | |
|  * @since       1.5
 | |
|  */
 | |
| 
 | |
| // Exit if accessed directly
 | |
| defined( 'ABSPATH' ) || exit;
 | |
| 
 | |
| /**
 | |
|  * EDD_Fees Class
 | |
|  *
 | |
|  * @since 1.5
 | |
|  */
 | |
| class EDD_Fees {
 | |
| 
 | |
| 	/**
 | |
| 	 * Setup the EDD Fees
 | |
| 	 *
 | |
| 	 * @since 1.5
 | |
| 	 */
 | |
| 	public function __construct() {
 | |
| 		add_filter( 'edd_payment_meta', array( $this, 'record_fees' ), 10, 2 );
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Adds a new Fee
 | |
| 	 *
 | |
| 	 * @since 1.5
 | |
| 	 *
 | |
| 	 * @param array $args Fee arguments
 | |
| 	 *
 | |
| 	 * @uses EDD_Fees::get_fees()
 | |
| 	 * @uses EDD_Session::set()
 | |
| 	 *
 | |
| 	 * @return array The fees.
 | |
| 	 */
 | |
| 	public function add_fee( $args = array() ) {
 | |
| 
 | |
| 		// Backwards compatibility with pre 2.0
 | |
| 		if ( func_num_args() > 1 ) {
 | |
| 
 | |
| 			$args     = func_get_args();
 | |
| 			$amount   = $args[0];
 | |
| 			$label    = isset( $args[1] ) ? $args[1] : '';
 | |
| 			$id       = isset( $args[2] ) ? $args[2] : '';
 | |
| 			$type     = 'fee';
 | |
| 
 | |
| 			$args = array(
 | |
| 				'amount' => $amount,
 | |
| 				'label'  => $label,
 | |
| 				'id'     => $id,
 | |
| 				'type'   => $type,
 | |
| 				'no_tax' => false,
 | |
| 				'download_id' => 0,
 | |
| 				'price_id'    => NULL
 | |
| 			);
 | |
| 
 | |
| 		} else {
 | |
| 
 | |
| 			$defaults = array(
 | |
| 				'amount'      => 0,
 | |
| 				'label'       => '',
 | |
| 				'id'          => '',
 | |
| 				'no_tax'      => false,
 | |
| 				'type'        => 'fee',
 | |
| 				'download_id' => 0,
 | |
| 				'price_id'    => NULL
 | |
| 			);
 | |
| 
 | |
| 			$args = wp_parse_args( $args, $defaults );
 | |
| 
 | |
| 			if( $args['type'] != 'fee' && $args['type'] != 'item' ) {
 | |
| 				$args['type'] = 'fee';
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		// If the fee is for an "item" but we passed in a download id
 | |
| 		if( 'item' === $args['type'] && ! empty( $args['download_id'] ) ) {
 | |
| 			unset( $args['download_id'] );
 | |
| 			unset( $args['price_id'] );
 | |
| 		}
 | |
| 
 | |
| 		if ( ! empty( $args['download_id'] ) ) {
 | |
| 			$options = isset( $args['price_id'] ) ? array( 'price_id' => $args['price_id'] ) : array();
 | |
| 			if ( ! edd_item_in_cart( $args['download_id'], $options ) ) {
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		$fees = $this->get_fees( 'all' );
 | |
| 
 | |
| 		// Determine the key
 | |
| 		$key = empty( $args['id'] ) ? sanitize_key( $args['label'] ) : sanitize_key( $args['id'] );
 | |
| 
 | |
| 		// Remove the unneeded id key
 | |
| 		unset( $args['id'] );
 | |
| 
 | |
| 		// Sanitize the amount
 | |
| 		$args['amount'] = edd_sanitize_amount( $args['amount'] );
 | |
| 
 | |
| 		// Force the amount to have the proper number of decimal places.
 | |
| 		$args['amount'] = number_format( (float) $args['amount'], edd_currency_decimal_filter(), '.', '' );
 | |
| 
 | |
| 		// Force no_tax to true if the amount is negative
 | |
| 		if( $args['amount'] < 0 ) {
 | |
| 			$args['no_tax'] = true;
 | |
| 		}
 | |
| 
 | |
| 		// Set the fee
 | |
| 		$fees[ $key ] = apply_filters( 'edd_fees_add_fee', $args, $this );
 | |
| 
 | |
| 		// Allow 3rd parties to process the fees before storing them in the session
 | |
| 		$fees = apply_filters( 'edd_fees_set_fees', $fees, $this );
 | |
| 
 | |
| 		// Update fees
 | |
| 		EDD()->session->set( 'edd_cart_fees', $fees );
 | |
| 
 | |
| 		do_action( 'edd_post_add_fee', $fees, $key, $args );
 | |
| 
 | |
| 		return $fees;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Remove an Existing Fee
 | |
| 	 *
 | |
| 	 * @since 1.5
 | |
| 	 * @param string $id Fee ID
 | |
| 	 * @uses EDD_Fees::get_fees()
 | |
| 	 * @uses EDD_Session::set()
 | |
| 	 * @return array Remaining fees
 | |
| 	 */
 | |
| 	public function remove_fee( $id = '' ) {
 | |
| 
 | |
| 		$fees = $this->get_fees( 'all' );
 | |
| 
 | |
| 		if ( isset( $fees[ $id ] ) ) {
 | |
| 			unset( $fees[ $id ] );
 | |
| 			EDD()->session->set( 'edd_cart_fees', $fees );
 | |
| 
 | |
| 			do_action( 'edd_post_remove_fee', $fees, $id );
 | |
| 		}
 | |
| 
 | |
| 		return $fees;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Check if any fees are present
 | |
| 	 *
 | |
| 	 * @since 1.5
 | |
| 	 * @param string $type Fee type, "fee" or "item"
 | |
| 	 * @uses EDD_Fees::get_fees()
 | |
| 	 * @return bool True if there are fees, false otherwise
 | |
| 	 */
 | |
| 	public function has_fees( $type = 'fee' ) {
 | |
| 
 | |
| 		if( 'all' == $type || 'fee' == $type ) {
 | |
| 			if( ! edd_get_cart_contents() ) {
 | |
| 				$type = 'item';
 | |
| 			}
 | |
| 
 | |
| 		}
 | |
| 
 | |
| 		$fees = $this->get_fees( $type );
 | |
| 		return ! empty( $fees ) && is_array( $fees );
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Retrieve all active fees
 | |
| 	 *
 | |
| 	 * @since 1.5
 | |
| 	 * @param string $type Fee type, "fee" or "item"
 | |
| 	 * @param int $download_id The download ID whose fees to retrieve
 | |
| 	 * @param int $price_id The variable price ID whose fees to retrieve
 | |
| 	 * @uses EDD_Session::get()
 | |
| 	 * @return array|bool List of fees when available, false when there are no fees
 | |
| 	 */
 | |
| 	public function get_fees( $type = 'fee', $download_id = 0, $price_id = NULL ) {
 | |
| 		$fees = EDD()->session->get( 'edd_cart_fees' );
 | |
| 
 | |
| 		if ( EDD()->cart->is_empty() ) {
 | |
| 			// We can only get item type fees when the cart is empty
 | |
| 			$type = 'item';
 | |
| 		}
 | |
| 
 | |
| 		if ( ! empty( $fees ) && ! empty( $type ) && 'all' !== $type ) {
 | |
| 			foreach ( $fees as $key => $fee ) {
 | |
| 				if ( ! empty( $fee['type'] ) && $type != $fee['type'] ) {
 | |
| 					unset( $fees[ $key ] );
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if ( ! empty( $fees ) && ! empty( $download_id ) ) {
 | |
| 			// Remove fees that don't belong to the specified Download
 | |
| 			$applied_fees = array();
 | |
| 			foreach ( $fees as $key => $fee ) {
 | |
| 
 | |
| 				if ( empty( $fee['download_id'] ) || (int) $download_id !== (int) $fee['download_id'] ) {
 | |
| 					unset( $fees[ $key ] );
 | |
| 				}
 | |
| 
 | |
| 				$fee_hash   = md5( $fee['amount'] . $fee['label'] . $fee['type'] );
 | |
| 
 | |
| 				if ( in_array( $fee_hash, $applied_fees ) ) {
 | |
| 					unset( $fees[ $key ] );
 | |
| 				}
 | |
| 
 | |
| 				$applied_fees[] = $fee_hash;
 | |
| 
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Now that we've removed any fees that are for other Downloads, lets also remove any fees that don't match this price id
 | |
| 		if ( ! empty( $fees ) && ! empty( $download_id ) && ! is_null( $price_id ) ) {
 | |
| 			// Remove fees that don't belong to the specified Download AND Price ID
 | |
| 			foreach ( $fees as $key => $fee ) {
 | |
| 				if ( is_null( $fee['price_id'] ) ) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				if ( (int) $price_id !== (int) $fee['price_id'] ){
 | |
| 					unset( $fees[ $key ] );
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		if ( ! empty( $fees ) ) {
 | |
| 			// Remove fees that belong to a specific download but are not in the cart
 | |
| 			foreach ( $fees as $key => $fee ) {
 | |
| 				if ( empty( $fee['download_id'] ) ) {
 | |
| 					continue;
 | |
| 				}
 | |
| 
 | |
| 				if ( ! edd_item_in_cart( $fee['download_id'] ) ) {
 | |
| 					unset( $fees[ $key ] );
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Allow 3rd parties to process the fees before returning them
 | |
| 		return apply_filters( 'edd_fees_get_fees', ! empty( $fees ) ? $fees : array(), $this );
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Retrieve a specific fee
 | |
| 	 *
 | |
| 	 * @since 1.5
 | |
| 	 *
 | |
| 	 * @param string $id ID of the fee to get
 | |
| 	 * @return array|bool The fee array when available, false otherwise
 | |
| 	 */
 | |
| 	public function get_fee( $id = '' ) {
 | |
| 		$fees = $this->get_fees( 'all' );
 | |
| 
 | |
| 		if ( ! isset( $fees[ $id ] ) )
 | |
| 			return false;
 | |
| 
 | |
| 		return $fees[ $id ];
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Calculate the total fee amount for a specific fee type
 | |
| 	 *
 | |
| 	 * Can be negative
 | |
| 	 *
 | |
| 	 * @since 2.0
 | |
| 	 * @param string $type Fee type, "fee" or "item"
 | |
| 	 * @uses EDD_Fees::get_fees()
 | |
| 	 * @uses EDD_Fees::has_fees()
 | |
| 	 * @return float Total fee amount
 | |
| 	 */
 | |
| 	public function type_total( $type = 'fee' ) {
 | |
| 		$fees  = $this->get_fees( $type );
 | |
| 		$total = (float) 0.00;
 | |
| 
 | |
| 		if ( $this->has_fees( $type ) ) {
 | |
| 			foreach ( $fees as $fee ) {
 | |
| 				$total += edd_sanitize_amount( $fee['amount'] );
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return edd_sanitize_amount( $total );
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Calculate the total fee amount
 | |
| 	 *
 | |
| 	 * Can be negative
 | |
| 	 *
 | |
| 	 * @since 1.5
 | |
| 	 * @uses EDD_Fees::get_fees()
 | |
| 	 * @uses EDD_Fees::has_fees()
 | |
| 	 * @param int $download_id The download ID whose fees to retrieve
 | |
| 	 * @return float Total fee amount
 | |
| 	 */
 | |
| 	public function total( $download_id = 0 ) {
 | |
| 		$fees  = $this->get_fees( 'all', $download_id );
 | |
| 		$total = (float) 0.00;
 | |
| 
 | |
| 		if ( $this->has_fees( 'all' ) ) {
 | |
| 			foreach ( $fees as $fee ) {
 | |
| 				$total += edd_sanitize_amount( $fee['amount'] );
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return edd_sanitize_amount( $total );
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Stores the fees in the payment meta
 | |
| 	 *
 | |
| 	 * @since 1.5
 | |
| 	 * @uses EDD_Session::set()
 | |
| 	 * @param array $payment_meta The meta data to store with the payment
 | |
| 	 * @param array $payment_data The info sent from process-purchase.php
 | |
| 	 * @return array Return the payment meta with the fees added
 | |
| 	*/
 | |
| 	public function record_fees( $payment_meta, $payment_data ) {
 | |
| 		if ( $this->has_fees( 'all' ) ) {
 | |
| 
 | |
| 			$payment_meta['fees'] = $this->get_fees( 'all' );
 | |
| 
 | |
| 			// Only clear fees from session when status is not pending
 | |
| 			if( ! empty( $payment_data['status'] ) && 'pending' !== strtolower( $payment_data['status'] ) ) {
 | |
| 
 | |
| 				EDD()->session->set( 'edd_cart_fees', null );
 | |
| 
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		return $payment_meta;
 | |
| 	}
 | |
| 
 | |
| 	/**
 | |
| 	 * Gets the tax to be added to a fee.
 | |
| 	 *
 | |
| 	 * @since 3.0
 | |
| 	 * @param  array   $fee
 | |
| 	 * @param  float   $tax_rate
 | |
| 	 * @return float
 | |
| 	 */
 | |
| 	public function get_calculated_tax( $fee, $tax_rate ) {
 | |
| 		$tax = 0.00;
 | |
| 		if ( ! ( $tax_rate || empty( $fee['no_tax'] ) ) || $fee['amount'] < 0 ) {
 | |
| 			return $tax;
 | |
| 		}
 | |
| 
 | |
| 		return ( floatval( $fee['amount'] ) * $tax_rate ) / 100;
 | |
| 	}
 | |
| }
 |