2022-11-27 15:03:07 +00:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* Download Object
|
|
|
|
*
|
|
|
|
* @package EDD
|
|
|
|
* @subpackage Classes/Download
|
|
|
|
* @copyright Copyright (c) 2018, Easy Digital Downloads, LLC
|
|
|
|
* @license http://opensource.org/licenses/gpl-2.0.php GNU Public License
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
|
|
|
|
// Exit if accessed directly
|
|
|
|
defined( 'ABSPATH' ) || exit;
|
|
|
|
|
|
|
|
use EDD\Models\Download;
|
|
|
|
/**
|
|
|
|
* EDD_Download Class
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
class EDD_Download {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The download ID
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
public $ID = 0;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The download price
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $price;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The download prices, if Variable Prices are enabled
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $prices;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The download files
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $files;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The file download limit
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $file_download_limit;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The refund window
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $refund_window;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The download type, default or bundle
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $type;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The bundled downloads, if this is a bundle type
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $bundled_downloads;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The sale count
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $sales;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The total earnings
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $earnings;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The notes
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $notes;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The download SKU
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $sku;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The purchase button behavior
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
private $button_behavior;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Declare the default properties in WP_Post as we can't extend it
|
|
|
|
* Anything we've declared above has been removed.
|
|
|
|
*/
|
|
|
|
public $post_author = 0;
|
|
|
|
public $post_date = '0000-00-00 00:00:00';
|
|
|
|
public $post_date_gmt = '0000-00-00 00:00:00';
|
|
|
|
public $post_content = '';
|
|
|
|
public $post_title = '';
|
|
|
|
public $post_excerpt = '';
|
|
|
|
public $post_status = 'publish';
|
|
|
|
public $comment_status = 'open';
|
|
|
|
public $ping_status = 'open';
|
|
|
|
public $post_password = '';
|
|
|
|
public $post_name = '';
|
|
|
|
public $to_ping = '';
|
|
|
|
public $pinged = '';
|
|
|
|
public $post_modified = '0000-00-00 00:00:00';
|
|
|
|
public $post_modified_gmt = '0000-00-00 00:00:00';
|
|
|
|
public $post_content_filtered = '';
|
|
|
|
public $post_parent = 0;
|
|
|
|
public $guid = '';
|
|
|
|
public $menu_order = 0;
|
|
|
|
public $post_mime_type = '';
|
|
|
|
public $comment_count = 0;
|
|
|
|
public $filter;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get things going
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
public function __construct( $_id = false ) {
|
|
|
|
$download = WP_Post::get_instance( $_id );
|
|
|
|
|
|
|
|
return $this->setup_download( $download );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Given the download data, let's set the variables
|
|
|
|
*
|
|
|
|
* @since 2.3.6
|
|
|
|
* @param WP_Post $download The WP_Post object for download.
|
|
|
|
* @return bool If the setup was successful or not
|
|
|
|
*/
|
|
|
|
private function setup_download( $download ) {
|
|
|
|
|
|
|
|
if ( ! is_object( $download ) ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $download instanceof WP_Post ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( 'download' !== $download->post_type ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
foreach ( $download as $key => $value ) {
|
|
|
|
$this->{$key} = $value;
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Magic __get function to dispatch a call to retrieve a private property
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*/
|
|
|
|
public function __get( $key = '' ) {
|
|
|
|
if ( method_exists( $this, "get_{$key}" ) ) {
|
|
|
|
return call_user_func( array( $this, "get_{$key}" ) );
|
|
|
|
} else {
|
|
|
|
return new WP_Error( 'edd-download-invalid-property', sprintf( __( 'Can\'t get property %s', 'easy-digital-downloads' ), $key ) );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a download
|
|
|
|
*
|
|
|
|
* @since 2.3.6
|
|
|
|
* @param array $data Array of attributes for a download
|
|
|
|
* @return mixed false if data isn't passed and class not instantiated for creation, or New Download ID
|
|
|
|
*/
|
|
|
|
public function create( $data = array() ) {
|
|
|
|
|
|
|
|
if ( $this->id != 0 ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
$defaults = array(
|
|
|
|
'post_type' => 'download',
|
|
|
|
'post_status' => 'draft',
|
|
|
|
'post_title' => __( 'New Download Product', 'easy-digital-downloads' )
|
|
|
|
);
|
|
|
|
|
|
|
|
$args = wp_parse_args( $data, $defaults );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fired before a download is created
|
|
|
|
*
|
|
|
|
* @param array $args The post object arguments used for creation.
|
|
|
|
*/
|
|
|
|
do_action( 'edd_download_pre_create', $args );
|
|
|
|
|
|
|
|
$id = wp_insert_post( $args, true );
|
|
|
|
|
|
|
|
$download = WP_Post::get_instance( $id );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fired after a download is created
|
|
|
|
*
|
|
|
|
* @param int $id The post ID of the created item.
|
|
|
|
* @param array $args The post object arguments used for creation.
|
|
|
|
*/
|
|
|
|
do_action( 'edd_download_post_create', $id, $args );
|
|
|
|
|
|
|
|
return $this->setup_download( $download );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the ID
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return int ID of the download
|
|
|
|
*/
|
|
|
|
public function get_ID() {
|
|
|
|
return $this->ID;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the download name
|
|
|
|
*
|
|
|
|
* @since 2.5.8
|
|
|
|
* @return string Name of the download
|
|
|
|
*/
|
|
|
|
public function get_name() {
|
|
|
|
return get_the_title( $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the price
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return float Price of the download
|
|
|
|
*/
|
|
|
|
public function get_price() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->price ) ) {
|
|
|
|
$this->price = get_post_meta( $this->ID, 'edd_price', true );
|
|
|
|
|
|
|
|
if ( $this->price ) {
|
|
|
|
$this->price = edd_sanitize_amount( $this->price );
|
|
|
|
} else {
|
|
|
|
$this->price = 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Override the download price.
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*
|
|
|
|
* @param string $price The download price(s).
|
|
|
|
* @param string|int $id The downloads ID.
|
|
|
|
*/
|
|
|
|
return apply_filters( 'edd_get_download_price', $this->price, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the variable prices
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return array List of the variable prices
|
|
|
|
*/
|
|
|
|
public function get_prices() {
|
|
|
|
|
|
|
|
$this->prices = array();
|
|
|
|
|
|
|
|
if ( true === $this->has_variable_prices() ) {
|
|
|
|
if ( empty( $this->prices ) ) {
|
|
|
|
$this->prices = get_post_meta( $this->ID, 'edd_variable_prices', true );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Override variable prices
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
*
|
|
|
|
* @param array $prices The array of variables prices.
|
|
|
|
* @param int|string The ID of the download.
|
|
|
|
*/
|
2023-06-28 12:45:44 +00:00
|
|
|
return (array) apply_filters( 'edd_get_variable_prices', $this->prices, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the default Price ID for variable priced products.
|
|
|
|
*
|
|
|
|
* Since it is possible for the value to not be set on older products, we'll set it to the first price in the array
|
|
|
|
* if one is not set, as that has been the default behavior since default prices were introduced.
|
|
|
|
*
|
|
|
|
* Storing it as the first if found, is just more consistent and intentional.
|
|
|
|
*
|
|
|
|
* @since 3.1.2
|
|
|
|
*
|
|
|
|
* @return int|null The default price ID, or null if the product does not have variable prices.
|
|
|
|
*/
|
|
|
|
public function get_default_price_id() {
|
|
|
|
if ( ! $this->has_variable_prices() ) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
$default_price_id = get_post_meta( $this->ID, '_edd_default_price_id', true );
|
|
|
|
|
|
|
|
// If no default price ID is set, or the default price ID is not in the prices array, set the first price as the default.
|
|
|
|
$prices = $this->get_prices();
|
|
|
|
if ( is_array( $prices ) && ( ! is_numeric( $default_price_id ) || ! array_key_exists( (int) $default_price_id, $prices ) ) ) {
|
|
|
|
$default_price_id = key( $prices );
|
|
|
|
|
|
|
|
// Set the default price ID
|
|
|
|
update_post_meta( $this->ID, '_edd_default_price_id', $default_price_id );
|
|
|
|
}
|
|
|
|
|
|
|
|
return absint( apply_filters( 'edd_variable_default_price_id', $default_price_id, $this->ID ) );
|
2022-11-27 15:03:07 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if single price mode is enabled or disabled
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return bool True if download is in single price mode, false otherwise
|
|
|
|
*/
|
|
|
|
public function is_single_price_mode() {
|
|
|
|
$ret = $this->has_variable_prices() && get_post_meta( $this->ID, '_edd_price_options_mode', true );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Override the price mode for a download when checking if is in single price mode.
|
|
|
|
*
|
|
|
|
* @since 2.3
|
|
|
|
*
|
|
|
|
* @param bool $ret Is download in single price mode?
|
|
|
|
* @param int|string The ID of the download.
|
|
|
|
*/
|
|
|
|
return (bool) apply_filters( 'edd_single_price_option_mode', $ret, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if the download has variable prices enabled
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return bool True when the download has variable pricing enabled, false otherwise
|
|
|
|
*/
|
|
|
|
public function has_variable_prices() {
|
|
|
|
$ret = get_post_meta( $this->ID, '_variable_pricing', true );
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Override whether the download has variables prices.
|
|
|
|
*
|
|
|
|
* @since 2.3
|
|
|
|
*
|
|
|
|
* @param bool $ret Does download have variable prices?
|
|
|
|
* @param int|string The ID of the download.
|
|
|
|
*/
|
|
|
|
return (bool) apply_filters( 'edd_has_variable_prices', $ret, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the file downloads
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @param integer $variable_price_id
|
|
|
|
* @return array List of download files
|
|
|
|
*/
|
|
|
|
public function get_files( $variable_price_id = null ) {
|
|
|
|
if ( ! isset( $this->files ) ) {
|
|
|
|
|
|
|
|
$this->files = array();
|
|
|
|
|
|
|
|
// Bundled products are not allowed to have files
|
|
|
|
if ( $this->is_bundled_download() ) {
|
|
|
|
return $this->files;
|
|
|
|
}
|
|
|
|
|
|
|
|
$download_files = get_post_meta( $this->ID, 'edd_download_files', true );
|
|
|
|
|
|
|
|
if ( ! empty( $download_files ) ) {
|
|
|
|
if ( ! is_null( $variable_price_id ) && $this->has_variable_prices() ) {
|
|
|
|
foreach ( $download_files as $key => $file_info ) {
|
|
|
|
if ( isset( $file_info['condition'] ) ) {
|
|
|
|
if ( $file_info['condition'] == $variable_price_id || 'all' === $file_info['condition'] ) {
|
|
|
|
$this->files[ $key ] = $file_info;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
$this->files = $download_files;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return apply_filters( 'edd_download_files', $this->files, $this->ID, $variable_price_id );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the file download limit
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return int Number of download limit
|
|
|
|
*/
|
|
|
|
public function get_file_download_limit() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->file_download_limit ) ) {
|
|
|
|
$limit = get_post_meta( $this->ID, '_edd_download_limit', true );
|
|
|
|
$global = edd_get_option( 'file_download_limit', 0 );
|
|
|
|
|
|
|
|
// Download specific limit
|
|
|
|
if ( is_numeric( $limit ) ) {
|
|
|
|
$retval = absint( $limit );
|
|
|
|
|
|
|
|
// Use global
|
|
|
|
} elseif ( '' === $limit ) {
|
|
|
|
$retval = '';
|
|
|
|
|
|
|
|
// Global limit
|
|
|
|
} elseif ( ! empty( $global ) ) {
|
|
|
|
$retval = absint( $global );
|
|
|
|
|
|
|
|
// Default
|
|
|
|
} else {
|
|
|
|
$retval = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->file_download_limit = $retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
return apply_filters( 'edd_file_download_limit', $this->file_download_limit, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the refund window
|
|
|
|
*
|
|
|
|
* @since 3.0
|
|
|
|
* @return int Number of days
|
|
|
|
*/
|
|
|
|
public function get_refund_window() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->refund_window ) ) {
|
|
|
|
$window = get_post_meta( $this->ID, '_edd_refund_window', true );
|
|
|
|
$global = edd_get_option( 'refund_window', 0 ); // needs to be 0 here
|
|
|
|
|
|
|
|
// Download specific window
|
|
|
|
if ( is_numeric( $window ) ) {
|
|
|
|
$retval = absint( $window );
|
|
|
|
|
|
|
|
// Use global
|
|
|
|
} elseif ( '' === $window ) {
|
|
|
|
$retval = '';
|
|
|
|
|
|
|
|
// Global limit
|
|
|
|
} elseif ( ! empty( $global ) ) {
|
|
|
|
$retval = absint( $global );
|
|
|
|
|
|
|
|
// Default
|
|
|
|
} else {
|
|
|
|
$retval = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->refund_window = $retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->refund_window; // No filter
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve whether the product is refundable.
|
|
|
|
*
|
|
|
|
* @since 3.0
|
|
|
|
*
|
|
|
|
* @return string `refundable` or `nonrefundable`
|
|
|
|
*/
|
|
|
|
public function get_refundability() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->refundability ) ) {
|
|
|
|
$default = 'refundable';
|
|
|
|
$refundable = get_post_meta( $this->ID, '_edd_refundability', true );
|
|
|
|
$global = edd_get_option( 'refundability', $default );
|
|
|
|
|
|
|
|
// Download specific window
|
|
|
|
if ( ! empty( $refundable ) ) {
|
|
|
|
$retval = $refundable;
|
|
|
|
|
|
|
|
// Use global
|
|
|
|
} elseif ( ! empty( $global ) ) {
|
|
|
|
$retval = $global;
|
|
|
|
|
|
|
|
// Default
|
|
|
|
} else {
|
|
|
|
$retval = $default;
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->refundability = $retval;
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->refundability; // No filter
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the price option that has access to the specified file
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return int|string
|
|
|
|
*/
|
|
|
|
public function get_file_price_condition( $file_key = 0 ) {
|
|
|
|
$files = $this->get_files();
|
|
|
|
$condition = isset( $files[ $file_key ]['condition'] )
|
|
|
|
? $files[ $file_key ]['condition']
|
|
|
|
: 'all';
|
|
|
|
|
|
|
|
return apply_filters( 'edd_get_file_price_condition', $condition, $this->ID, $files );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the download type, default or bundle
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return string Type of download, either 'default' or 'bundle'
|
|
|
|
*/
|
|
|
|
public function get_type() {
|
|
|
|
if ( ! isset( $this->type ) ) {
|
|
|
|
$this->type = get_post_meta( $this->ID, '_edd_product_type', true );
|
|
|
|
|
|
|
|
if ( empty( $this->type ) ) {
|
|
|
|
$this->type = 'default';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return apply_filters( 'edd_get_download_type', $this->type, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if this is a bundled download
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return bool True when download is a bundle, false otherwise
|
|
|
|
*/
|
|
|
|
public function is_bundled_download() {
|
|
|
|
return 'bundle' === $this->get_type();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieves the Download IDs that are bundled with this Download
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return array List of bundled downloads
|
|
|
|
*/
|
|
|
|
public function get_bundled_downloads() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->bundled_downloads ) ) {
|
|
|
|
$this->bundled_downloads = (array) get_post_meta( $this->ID, '_edd_bundled_products', true );
|
|
|
|
}
|
|
|
|
|
|
|
|
return (array) apply_filters( 'edd_get_bundled_products', array_filter( $this->bundled_downloads ), $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the Download IDs that are bundled with this Download based on the variable pricing ID passed
|
|
|
|
*
|
|
|
|
* @since 2.7
|
|
|
|
* @param int $price_id Variable pricing ID
|
|
|
|
* @return array List of bundled downloads
|
|
|
|
*/
|
|
|
|
public function get_variable_priced_bundled_downloads( $price_id = null ) {
|
|
|
|
if ( null === $price_id ) {
|
|
|
|
return $this->get_bundled_downloads();
|
|
|
|
}
|
|
|
|
|
|
|
|
$downloads = array();
|
|
|
|
$bundled_downloads = $this->get_bundled_downloads();
|
|
|
|
$price_assignments = $this->get_bundle_pricing_variations();
|
|
|
|
|
|
|
|
if ( ! $price_assignments ) {
|
|
|
|
return $bundled_downloads;
|
|
|
|
}
|
|
|
|
|
|
|
|
$price_assignments = $price_assignments[0];
|
|
|
|
|
|
|
|
foreach ( $price_assignments as $key => $value ) {
|
|
|
|
if ( isset( $bundled_downloads[ $key ] ) && ( $value == $price_id || $value == 'all' ) ) {
|
|
|
|
$downloads[] = $bundled_downloads[ $key ];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return $downloads;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the download notes
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return string Note related to the download
|
|
|
|
*/
|
|
|
|
public function get_notes() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->notes ) ) {
|
|
|
|
$this->notes = get_post_meta( $this->ID, 'edd_product_notes', true );
|
|
|
|
}
|
|
|
|
|
|
|
|
return (string) apply_filters( 'edd_product_notes', $this->notes, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the download sku
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return string SKU of the download
|
|
|
|
*/
|
|
|
|
public function get_sku() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->sku ) ) {
|
|
|
|
|
|
|
|
$this->sku = get_post_meta( $this->ID, 'edd_sku', true );
|
|
|
|
|
|
|
|
if ( empty( $this->sku ) ) {
|
|
|
|
$this->sku = '-';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return apply_filters( 'edd_get_download_sku', $this->sku, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the purchase button behavior
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return string
|
|
|
|
*/
|
|
|
|
public function get_button_behavior() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->button_behavior ) ) {
|
|
|
|
|
|
|
|
$this->button_behavior = get_post_meta( $this->ID, '_edd_button_behavior', true );
|
|
|
|
|
|
|
|
if ( empty( $this->button_behavior ) || ! edd_shop_supports_buy_now() ) {
|
|
|
|
$this->button_behavior = 'add_to_cart';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return apply_filters( 'edd_get_download_button_behavior', $this->button_behavior, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the sale count for the download
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return int Number of times this has been purchased
|
|
|
|
*/
|
|
|
|
public function get_sales() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->sales ) ) {
|
|
|
|
|
|
|
|
if ( '' == get_post_meta( $this->ID, '_edd_download_sales', true ) ) {
|
|
|
|
add_post_meta( $this->ID, '_edd_download_sales', 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->sales = get_post_meta( $this->ID, '_edd_download_sales', true );
|
|
|
|
|
|
|
|
// Never let sales be less than zero
|
|
|
|
$this->sales = max( $this->sales, 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->sales;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Increment the sale count by one
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @param int $quantity The quantity to increase the sales by
|
|
|
|
* @return int New number of total sales
|
|
|
|
*/
|
|
|
|
public function increase_sales( $quantity = 1 ) {
|
|
|
|
|
|
|
|
_edd_deprecated_function( __METHOD__, '3.0', 'EDD_Download::recalculate_net_sales_earnings()' );
|
|
|
|
edd_recalculate_download_sales_earnings( $this->ID );
|
|
|
|
|
|
|
|
$this->get_sales();
|
|
|
|
do_action( 'edd_download_increase_sales', $this->ID, $this->sales, $this );
|
|
|
|
|
|
|
|
return $this->sales;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decrement the sale count by one
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @param int $quantity The quantity to decrease by
|
|
|
|
* @return int New number of total sales
|
|
|
|
*/
|
|
|
|
public function decrease_sales( $quantity = 1 ) {
|
|
|
|
|
|
|
|
_edd_deprecated_function( __METHOD__, '3.0', 'EDD_Download::recalculate_net_sales_earnings()' );
|
|
|
|
$this->recalculate_net_sales_earnings();
|
|
|
|
|
|
|
|
$this->get_sales();
|
|
|
|
do_action( 'edd_download_decrease_sales', $this->ID, $this->sales, $this );
|
|
|
|
|
|
|
|
return $this->sales;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Retrieve the total earnings for the download
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @return float Total download earnings
|
|
|
|
*/
|
|
|
|
public function get_earnings() {
|
|
|
|
|
|
|
|
if ( ! isset( $this->earnings ) ) {
|
|
|
|
if ( '' == get_post_meta( $this->ID, '_edd_download_earnings', true ) ) {
|
|
|
|
add_post_meta( $this->ID, '_edd_download_earnings', 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
$this->earnings = get_post_meta( $this->ID, '_edd_download_earnings', true );
|
|
|
|
|
|
|
|
// Never let earnings be less than zero
|
|
|
|
$this->earnings = max( $this->earnings, 0 );
|
|
|
|
}
|
|
|
|
|
|
|
|
return $this->earnings;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Increase the earnings by the given amount
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @param int|float $amount Amount to increase the earnings by
|
|
|
|
* @return float New number of total earnings
|
|
|
|
*/
|
|
|
|
public function increase_earnings( $amount = 0 ) {
|
|
|
|
|
|
|
|
_edd_deprecated_function( __METHOD__, '3.0', 'edd_recalculate_download_sales_earnings()' );
|
|
|
|
edd_recalculate_download_sales_earnings( $this->ID );
|
|
|
|
|
|
|
|
$this->get_earnings();
|
|
|
|
do_action( 'edd_download_increase_earnings', $this->ID, $this->earnings, $this );
|
|
|
|
|
|
|
|
return $this->earnings;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Decrease the earnings by the given amount
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @param int|float $amount Number to decrease earning with
|
|
|
|
* @return float New number of total earnings
|
|
|
|
*/
|
|
|
|
public function decrease_earnings( $amount ) {
|
|
|
|
|
|
|
|
_edd_deprecated_function( __METHOD__, '3.0', 'EDD_Download::recalculate_net_sales_earnings()' );
|
|
|
|
|
|
|
|
$this->recalculate_net_sales_earnings();
|
|
|
|
$this->get_earnings();
|
|
|
|
|
|
|
|
do_action( 'edd_download_decrease_earnings', $this->ID, $this->earnings, $this );
|
|
|
|
|
|
|
|
return $this->earnings;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates the gross sales and earnings for a download.
|
|
|
|
*
|
|
|
|
* @since 3.0
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function recalculate_gross_sales_earnings() {
|
|
|
|
$download_model = new Download( $this->ID );
|
|
|
|
|
|
|
|
// This currently uses the post meta functions as we do not yet guarantee that the meta exists.
|
|
|
|
update_post_meta( $this->ID, '_edd_download_gross_sales', $download_model->get_gross_sales() );
|
|
|
|
update_post_meta( $this->ID, '_edd_download_gross_earnings', floatval( $download_model->get_gross_earnings() ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recalculates the net sales and earnings for a download.
|
|
|
|
*
|
|
|
|
* @since 3.0
|
|
|
|
* @return void
|
|
|
|
*/
|
|
|
|
public function recalculate_net_sales_earnings() {
|
|
|
|
$download_model = new Download( $this->ID );
|
|
|
|
|
|
|
|
$this->update_meta( '_edd_download_sales', intval( $download_model->get_net_sales() ) );
|
|
|
|
$this->update_meta( '_edd_download_earnings', floatval( $download_model->get_net_earnings() ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Determine if the download is free or if the given price ID is free
|
|
|
|
*
|
|
|
|
* @since 2.2
|
|
|
|
* @param bool $price_id ID of variation if needed
|
|
|
|
* @return bool True when the download is free, false otherwise
|
|
|
|
*/
|
|
|
|
public function is_free( $price_id = false ) {
|
|
|
|
|
|
|
|
$is_free = false;
|
|
|
|
$variable_pricing = edd_has_variable_prices( $this->ID );
|
|
|
|
|
|
|
|
if ( $variable_pricing && ! is_null( $price_id ) && $price_id !== false ) {
|
|
|
|
$price = edd_get_price_option_amount( $this->ID, $price_id );
|
|
|
|
|
|
|
|
} elseif ( $variable_pricing && $price_id === false ) {
|
|
|
|
$lowest_price = (float) edd_get_lowest_price_option( $this->ID );
|
|
|
|
$highest_price = (float) edd_get_highest_price_option( $this->ID );
|
|
|
|
|
|
|
|
if ( $lowest_price === 0.00 && $highest_price === 0.00 ) {
|
|
|
|
$price = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
} elseif ( ! $variable_pricing ) {
|
|
|
|
|
|
|
|
$price = get_post_meta( $this->ID, 'edd_price', true );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( isset( $price ) && (float) $price == 0 ) {
|
|
|
|
$is_free = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (bool) apply_filters( 'edd_is_free_download', $is_free, $this->ID, $price_id );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Is quantity input disabled on this product?
|
|
|
|
*
|
|
|
|
* @since 2.7
|
|
|
|
* @return bool
|
|
|
|
*/
|
|
|
|
public function quantities_disabled() {
|
|
|
|
$ret = (bool) get_post_meta( $this->ID, '_edd_quantities_disabled', true );
|
|
|
|
return apply_filters( 'edd_download_quantity_disabled', $ret, $this->ID );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Updates a single meta entry for the download
|
|
|
|
*
|
|
|
|
* @since 2.3
|
|
|
|
* @access private
|
|
|
|
* @param string $meta_key The meta_key to update
|
|
|
|
* @param string|array|object $meta_value The value to put into the meta
|
|
|
|
* @return bool The result of the update query
|
|
|
|
*/
|
|
|
|
private function update_meta( $meta_key = '', $meta_value = '' ) {
|
|
|
|
global $wpdb;
|
|
|
|
|
|
|
|
if ( empty( $meta_key ) || ( ! is_numeric( $meta_value ) && empty( $meta_value ) ) ) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make sure if it needs to be serialized, we do
|
|
|
|
$meta_value = maybe_serialize( $meta_value );
|
|
|
|
|
|
|
|
if ( is_numeric( $meta_value ) ) {
|
|
|
|
$value_type = is_float( $meta_value ) ? '%f' : '%d';
|
|
|
|
} else {
|
|
|
|
$value_type = "'%s'";
|
|
|
|
}
|
|
|
|
|
|
|
|
$sql = $wpdb->prepare( "UPDATE $wpdb->postmeta SET meta_value = $value_type WHERE post_id = $this->ID AND meta_key = '%s'", $meta_value, $meta_key );
|
|
|
|
|
|
|
|
if ( $wpdb->query( $sql ) ) {
|
|
|
|
|
|
|
|
clean_post_cache( $this->ID );
|
|
|
|
return true;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Checks if the download can be purchased
|
|
|
|
*
|
|
|
|
* NOTE: Currently only checks on edd_get_cart_contents() and edd_add_to_cart()
|
|
|
|
*
|
|
|
|
* @since 2.6.4
|
|
|
|
* @return bool If the current user can purchase the download ID
|
|
|
|
*/
|
|
|
|
public function can_purchase() {
|
|
|
|
$can_purchase = true;
|
|
|
|
|
|
|
|
if ( 'publish' !== $this->post_status && ! current_user_can( 'edit_post', $this->ID ) ) {
|
|
|
|
$can_purchase = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (bool) apply_filters( 'edd_can_purchase_download', $can_purchase, $this );
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get pricing variations for bundled items
|
|
|
|
*
|
|
|
|
* @since 2.7
|
|
|
|
* @return array
|
|
|
|
*/
|
|
|
|
public function get_bundle_pricing_variations() {
|
|
|
|
return get_post_meta( $this->ID, '_edd_bundled_products_conditions' );
|
|
|
|
}
|
|
|
|
}
|