3587 lines
95 KiB
PHP
3587 lines
95 KiB
PHP
|
<?php
|
||
|
/**
|
||
|
* Payments Object.
|
||
|
*
|
||
|
* This class is for working with payments in EDD.
|
||
|
*
|
||
|
* @package EDD
|
||
|
* @subpackage Payments
|
||
|
* @copyright Copyright (c) 2018, Easy Digital Downloads, LLC
|
||
|
* @license http://opensource.org/licenses/gpl-2.0.php GNU Public License
|
||
|
* @since 2.5
|
||
|
*/
|
||
|
|
||
|
// Exit if accessed directly
|
||
|
defined( 'ABSPATH' ) || exit;
|
||
|
|
||
|
/**
|
||
|
* EDD_Payment Class
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Updated to work with new custom tables.
|
||
|
*
|
||
|
* @property int $ID
|
||
|
* @property int $_ID
|
||
|
* @property bool $new
|
||
|
* @property string $number
|
||
|
* @property string $mode
|
||
|
* @property string $key
|
||
|
* @property float $total
|
||
|
* @property float $subtotal
|
||
|
* @property float $tax
|
||
|
* @property float $discounted_amount
|
||
|
* @property float $tax_rate
|
||
|
* @property array $fees
|
||
|
* @property float $fees_total
|
||
|
* @property string $discounts
|
||
|
* @property string $date
|
||
|
* @property string $completed_date
|
||
|
* @property string $status
|
||
|
* @property string $post_status
|
||
|
* @property string $old_status
|
||
|
* @property string $status_nicename
|
||
|
* @property int $customer_id
|
||
|
* @property int $user_id
|
||
|
* @property string $first_name
|
||
|
* @property string $last_name
|
||
|
* @property string $email
|
||
|
* @property array $user_info
|
||
|
* @property array $payment_meta
|
||
|
* @property array $address
|
||
|
* @property string $transaction_id
|
||
|
* @property array $downloads
|
||
|
* @property string $ip
|
||
|
* @property string $gateway
|
||
|
* @property string $currency
|
||
|
* @property array $cart_details
|
||
|
* @property bool $has_unlimited_downloads
|
||
|
* @property int $parent_payment
|
||
|
*/
|
||
|
class EDD_Payment {
|
||
|
|
||
|
/**
|
||
|
* The Payment ID
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var integer
|
||
|
*/
|
||
|
public $ID = 0;
|
||
|
protected $_ID = 0;
|
||
|
|
||
|
/**
|
||
|
* Identify if the payment is a new one or existing
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var boolean
|
||
|
*/
|
||
|
protected $new = false;
|
||
|
|
||
|
/**
|
||
|
* The Payment number (for use with sequential payments)
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $number = '';
|
||
|
|
||
|
/**
|
||
|
* The Gateway mode the payment was made in
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $mode = 'live';
|
||
|
|
||
|
/**
|
||
|
* The Unique Payment Key
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $key = '';
|
||
|
|
||
|
/**
|
||
|
* The total amount the payment is for
|
||
|
* Includes items, taxes, fees, and discounts
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var float
|
||
|
*/
|
||
|
protected $total = 0.00;
|
||
|
|
||
|
/**
|
||
|
* The Subtotal fo the payment before taxes
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var float
|
||
|
*/
|
||
|
protected $subtotal = 0;
|
||
|
|
||
|
/**
|
||
|
* The amount of tax for this payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var float
|
||
|
*/
|
||
|
protected $tax = 0;
|
||
|
|
||
|
/**
|
||
|
* The amount the payment has been discounted through discount codes
|
||
|
*
|
||
|
* @since 2.8.7
|
||
|
* @var int
|
||
|
*/
|
||
|
protected $discounted_amount = 0;
|
||
|
|
||
|
/**
|
||
|
* The tax rate charged on this payment
|
||
|
*
|
||
|
* @since 2.7
|
||
|
* @var float
|
||
|
*/
|
||
|
protected $tax_rate = '';
|
||
|
|
||
|
/**
|
||
|
* Array of global fees for this payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $fees = array();
|
||
|
|
||
|
/**
|
||
|
* The sum of the fee amounts
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var float
|
||
|
*/
|
||
|
protected $fees_total = 0;
|
||
|
|
||
|
/**
|
||
|
* Any discounts applied to the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $discounts = 'none';
|
||
|
|
||
|
/**
|
||
|
* The date the payment was created
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $date = '';
|
||
|
|
||
|
/**
|
||
|
* The date the payment was marked as 'complete'
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $completed_date = '';
|
||
|
|
||
|
/**
|
||
|
* The status of the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $status = 'pending';
|
||
|
protected $post_status = 'pending'; // Same as $status but here for backwards compat
|
||
|
|
||
|
/**
|
||
|
* When updating, the old status prior to the change
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $old_status = '';
|
||
|
|
||
|
/**
|
||
|
* The display name of the current payment status
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $status_nicename = '';
|
||
|
|
||
|
/**
|
||
|
* The customer ID that made the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var integer
|
||
|
*/
|
||
|
protected $customer_id = null;
|
||
|
|
||
|
/**
|
||
|
* The User ID (if logged in) that made the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var integer
|
||
|
*/
|
||
|
protected $user_id = 0;
|
||
|
|
||
|
/**
|
||
|
* The first name of the payee
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $first_name = '';
|
||
|
|
||
|
/**
|
||
|
* The last name of the payee
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $last_name = '';
|
||
|
|
||
|
/**
|
||
|
* The email used for the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $email = '';
|
||
|
|
||
|
/**
|
||
|
* Legacy (not to be accessed) array of user information
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var array
|
||
|
*/
|
||
|
private $user_info = array();
|
||
|
|
||
|
/**
|
||
|
* Legacy (not to be accessed) payment meta array
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var array
|
||
|
*/
|
||
|
private $payment_meta = array();
|
||
|
|
||
|
/**
|
||
|
* The physical address used for the payment if provided
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $address = array();
|
||
|
|
||
|
/**
|
||
|
* The transaction ID returned by the gateway
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $transaction_id = '';
|
||
|
|
||
|
/**
|
||
|
* Array of downloads for this payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $downloads = array();
|
||
|
|
||
|
/**
|
||
|
* IP Address payment was made from
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $ip = '';
|
||
|
|
||
|
/**
|
||
|
* The gateway used to process the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $gateway = '';
|
||
|
|
||
|
/**
|
||
|
* The the payment was made with
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var string
|
||
|
*/
|
||
|
protected $currency = '';
|
||
|
|
||
|
/**
|
||
|
* The cart details array
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var array
|
||
|
*/
|
||
|
protected $cart_details = array();
|
||
|
|
||
|
/**
|
||
|
* Allows the files for this payment to be downloaded unlimited times (when download limits are enabled)
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var boolean
|
||
|
*/
|
||
|
protected $has_unlimited_downloads = false;
|
||
|
|
||
|
/**
|
||
|
* Array of items that have changed since the last save() was run
|
||
|
* This is for internal use, to allow fewer update_payment_meta calls to be run
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var array
|
||
|
*/
|
||
|
private $pending;
|
||
|
|
||
|
/**
|
||
|
* The parent payment (if applicable)
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @var integer
|
||
|
*/
|
||
|
protected $parent_payment = 0;
|
||
|
|
||
|
/**
|
||
|
* Order object.
|
||
|
*
|
||
|
* @since 3.0
|
||
|
* @var EDD\Orders\Order
|
||
|
*/
|
||
|
protected $order;
|
||
|
|
||
|
/**
|
||
|
* Whether the payment being retrieved is a post object.
|
||
|
*
|
||
|
* @var bool
|
||
|
*/
|
||
|
private $is_edd_payment = false;
|
||
|
|
||
|
/**
|
||
|
* Constructor.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Updated to fetch transaction ID from edd_ordermeta table.
|
||
|
*
|
||
|
* @param mixed $payment_or_txn_id Payment ID or transaction ID. Default false.
|
||
|
* @param bool $by_txn Whether the constructor should retrieve the order ID from the transaction ID. Default false.
|
||
|
*
|
||
|
* @return mixed void|false
|
||
|
*/
|
||
|
public function __construct( $payment_or_txn_id = false, $by_txn = false ) {
|
||
|
if ( empty( $payment_or_txn_id ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if ( $by_txn ) {
|
||
|
$payment_id = edd_get_order_id_from_transaction_id( $payment_or_txn_id );
|
||
|
|
||
|
if ( empty( $payment_id ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
} else {
|
||
|
$payment_id = absint( $payment_or_txn_id );
|
||
|
}
|
||
|
|
||
|
$this->setup_payment( $payment_id );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Magic GET function
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param string $key The property
|
||
|
*
|
||
|
* @return mixed The value
|
||
|
*/
|
||
|
public function __get( $key ) {
|
||
|
if ( method_exists( $this, "get_{$key}" ) ) {
|
||
|
$value = call_user_func( array( $this, "get_{$key}" ) );
|
||
|
} elseif ( 'id' === $key ) {
|
||
|
$value = $this->ID;
|
||
|
} elseif ( 'post_type' === $key ) {
|
||
|
$value = 'edd_payment';
|
||
|
} elseif ( 'post_date' === $key ) {
|
||
|
$value = $this->date;
|
||
|
} else {
|
||
|
$value = $this->$key;
|
||
|
}
|
||
|
|
||
|
return $value;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Magic SET function
|
||
|
*
|
||
|
* Sets up the pending array for the save method
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param string $key The property name
|
||
|
* @param mixed $value The value of the property
|
||
|
*/
|
||
|
public function __set( $key, $value ) {
|
||
|
$ignore = array( 'downloads', 'cart_details', 'fees', '_ID' );
|
||
|
|
||
|
if ( $key === 'status' ) {
|
||
|
$this->old_status = $this->status;
|
||
|
}
|
||
|
|
||
|
if ( ! in_array( $key, $ignore ) ) {
|
||
|
$this->pending[ $key ] = $value;
|
||
|
}
|
||
|
|
||
|
if ( '_ID' !== $key ) {
|
||
|
$this->$key = $value;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Magic ISSET function, which allows empty checks on protected elements
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param string $name The attribute to get
|
||
|
*
|
||
|
* @return boolean If the item is set or not
|
||
|
*/
|
||
|
public function __isset( $name ) {
|
||
|
if ( property_exists( $this, $name ) ) {
|
||
|
return false === empty( $this->$name );
|
||
|
} else {
|
||
|
return null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup payment properties
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param int $payment_id The payment ID
|
||
|
*
|
||
|
* @return bool If the setup was successful or not
|
||
|
*/
|
||
|
private function setup_payment( $payment_id ) {
|
||
|
$this->pending = array();
|
||
|
|
||
|
if ( empty( $payment_id ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$this->order = $this->_shim_edd_get_order( $payment_id );
|
||
|
|
||
|
if ( ! $this->order || is_wp_error( $this->order ) ) {
|
||
|
return _edd_get_final_payment_id() ? $this->_setup_compat_payment( $payment_id ) : false;
|
||
|
}
|
||
|
|
||
|
// Allow extensions to perform actions before the payment is loaded
|
||
|
do_action( 'edd_pre_setup_payment', $this, $payment_id );
|
||
|
|
||
|
// Primary Identifier
|
||
|
$this->ID = absint( $payment_id );
|
||
|
|
||
|
// Protected ID that can never be changed
|
||
|
$this->_ID = absint( $payment_id );
|
||
|
|
||
|
// Status and Dates
|
||
|
$this->date = $this->order->date_created;
|
||
|
$this->completed_date = $this->setup_completed_date();
|
||
|
$this->status = $this->order->status;
|
||
|
$this->post_status = $this->order->status;
|
||
|
$this->mode = $this->order->mode;
|
||
|
$this->parent_payment = $this->order->parent;
|
||
|
|
||
|
$all_payment_statuses = edd_get_payment_statuses();
|
||
|
$this->status_nicename = array_key_exists( $this->status, $all_payment_statuses ) ? $all_payment_statuses[ $this->status ] : ucfirst( $this->status );
|
||
|
|
||
|
// Items
|
||
|
$this->fees = $this->setup_fees();
|
||
|
$this->cart_details = $this->setup_cart_details();
|
||
|
$this->downloads = $this->setup_downloads();
|
||
|
|
||
|
// Currency Based
|
||
|
$this->total = $this->order->total;
|
||
|
$this->tax = $this->order->tax;
|
||
|
$this->tax_rate = $this->setup_tax_rate();
|
||
|
$this->fees_total = $this->setup_fees_total();
|
||
|
$this->subtotal = $this->order->subtotal;
|
||
|
$this->currency = $this->setup_currency();
|
||
|
|
||
|
// Gateway based
|
||
|
$this->gateway = $this->order->gateway;
|
||
|
$this->transaction_id = $this->setup_transaction_id();
|
||
|
|
||
|
// User based
|
||
|
$this->ip = $this->order->ip;
|
||
|
$this->customer_id = $this->order->customer_id;
|
||
|
$this->user_id = $this->setup_user_id();
|
||
|
$this->email = $this->setup_email();
|
||
|
$this->discounts = $this->setup_discounts();
|
||
|
$this->user_info = $this->setup_user_info();
|
||
|
$this->first_name = $this->user_info['first_name'];
|
||
|
$this->last_name = $this->user_info['last_name'];
|
||
|
$this->address = $this->setup_address();
|
||
|
|
||
|
// Other Identifiers
|
||
|
$this->key = $this->order->payment_key;
|
||
|
$this->number = $this->setup_payment_number();
|
||
|
|
||
|
// Additional Attributes
|
||
|
$this->has_unlimited_downloads = $this->setup_has_unlimited();
|
||
|
|
||
|
// We have a payment, get the generic payment_meta item to reduce calls to it
|
||
|
// This only exists for backwards compatibility purposes.
|
||
|
$this->payment_meta = $this->get_meta();
|
||
|
|
||
|
// Allow extensions to add items to this object via hook
|
||
|
do_action( 'edd_setup_payment', $this, $payment_id );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Create the base of a payment.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Updated to insert orders to the new custom tables.
|
||
|
*
|
||
|
* @return int|bool False on failure, the order ID on success.
|
||
|
*/
|
||
|
private function insert_payment() {
|
||
|
|
||
|
$payment_data = array(
|
||
|
'price' => $this->total,
|
||
|
'date' => $this->date,
|
||
|
'user_email' => $this->email,
|
||
|
'purchase_key' => $this->key,
|
||
|
'currency' => $this->currency,
|
||
|
'downloads' => $this->downloads,
|
||
|
'user_info' => array(
|
||
|
'id' => $this->user_id,
|
||
|
'email' => $this->email,
|
||
|
'first_name' => $this->first_name,
|
||
|
'last_name' => $this->last_name,
|
||
|
'discount' => $this->discounts,
|
||
|
'address' => $this->address,
|
||
|
),
|
||
|
'cart_details' => $this->cart_details,
|
||
|
'status' => $this->status,
|
||
|
'fees' => $this->fees,
|
||
|
);
|
||
|
|
||
|
// Create an order
|
||
|
$order_args = array(
|
||
|
'parent' => $this->parent_payment,
|
||
|
'status' => $this->status,
|
||
|
'user_id' => $this->user_id,
|
||
|
'email' => $this->email,
|
||
|
'ip' => $this->ip,
|
||
|
'gateway' => $this->gateway,
|
||
|
'mode' => $this->mode,
|
||
|
'currency' => $this->currency,
|
||
|
'payment_key' => $this->key,
|
||
|
);
|
||
|
|
||
|
$order_id = edd_add_order( $order_args );
|
||
|
|
||
|
if ( ! empty( $order_id ) ) {
|
||
|
$this->ID = $order_id;
|
||
|
$this->_ID = $order_id;
|
||
|
|
||
|
$customer = $this->maybe_create_customer();
|
||
|
|
||
|
$this->customer_id = $customer->id;
|
||
|
$customer->attach_payment( $this->ID, false );
|
||
|
|
||
|
$order_data = array(
|
||
|
'customer_id' => $this->customer_id,
|
||
|
);
|
||
|
|
||
|
/**
|
||
|
* This run of the edd_payment_meta filter is for backwards compatibility purposes. The filter will also run
|
||
|
* in the EDD_Payment::save method. By keeping this here, it retains compatibility of adding payment meta
|
||
|
* prior to the payment being inserted, as was previously supported by edd_insert_payment().
|
||
|
*
|
||
|
* @reference: https://github.com/easydigitaldownloads/easy-digital-downloads/issues/5838
|
||
|
*/
|
||
|
$this->payment_meta = apply_filters( 'edd_payment_meta', $this->payment_meta, $payment_data );
|
||
|
if ( ! empty( $this->payment_meta['fees'] ) ) {
|
||
|
$this->fees = array_merge( $this->payment_meta['fees'], $this->fees );
|
||
|
foreach ( $this->fees as $key => $fee ) {
|
||
|
add_filter( 'edd_prices_include_tax', '__return_false' );
|
||
|
|
||
|
$tax = ( isset( $fee['no_tax'] ) && false === $fee['no_tax'] ) || $fee['amount'] < 0
|
||
|
? floatval( edd_calculate_tax( $fee['amount'] ) )
|
||
|
: 0.00;
|
||
|
|
||
|
remove_filter( 'edd_prices_include_tax', '__return_false' );
|
||
|
|
||
|
$adjustment_id = edd_add_order_adjustment( array(
|
||
|
'object_id' => $this->ID,
|
||
|
'object_type' => 'order',
|
||
|
'type_key' => $key,
|
||
|
'type' => 'fee',
|
||
|
'description' => $fee['label'],
|
||
|
'subtotal' => floatval( $fee['amount'] ),
|
||
|
'tax' => $tax,
|
||
|
'total' => floatval( $fee['amount'] ) + $tax,
|
||
|
) );
|
||
|
|
||
|
$this->increase_fees( $fee['amount'] );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( edd_get_option( 'enable_sequential' ) ) {
|
||
|
$number = edd_get_next_payment_number();
|
||
|
$this->number = edd_format_payment_number( $number );
|
||
|
|
||
|
$this->update_meta( '_edd_payment_number', $this->number );
|
||
|
$order_data['order_number'] = $this->number;
|
||
|
|
||
|
update_option( 'edd_last_payment_number', $number );
|
||
|
}
|
||
|
|
||
|
edd_update_order( $order_id, $order_data );
|
||
|
|
||
|
$this->update_meta( '_edd_payment_meta', $this->payment_meta );
|
||
|
|
||
|
$tax_rate = $this->tax_rate;
|
||
|
if ( ! empty( $tax_rate ) && $this->tax_rate > 0 && $this->tax_rate < 1 ) {
|
||
|
$tax_rate = $tax_rate * 100;
|
||
|
}
|
||
|
$order_meta = array(
|
||
|
'tax_rate' => $tax_rate,
|
||
|
);
|
||
|
|
||
|
foreach ( $order_meta as $key => $value ) {
|
||
|
if ( ! empty( $value ) ) {
|
||
|
edd_add_order_meta( $order_id, $key, $value );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$this->new = true;
|
||
|
}
|
||
|
|
||
|
return $this->ID;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* One items have been set, an update is needed to save them to the database.
|
||
|
*
|
||
|
* @since 3.0 Refactored to work with the new query methods.
|
||
|
*
|
||
|
* @return bool True of the save occurred, false if it failed or wasn't needed.
|
||
|
*/
|
||
|
public function save() {
|
||
|
$saved = false;
|
||
|
|
||
|
if ( empty( $this->ID ) ) {
|
||
|
$payment_id = $this->insert_payment();
|
||
|
|
||
|
if ( false === $payment_id ) {
|
||
|
$saved = false;
|
||
|
} else {
|
||
|
$this->ID = $payment_id;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $this->ID !== $this->_ID ) {
|
||
|
$this->ID = $this->_ID;
|
||
|
}
|
||
|
|
||
|
// If the order is null, it means a new order is being added
|
||
|
$this->order = $this->_shim_edd_get_order( $this->ID );
|
||
|
|
||
|
$customer = $this->maybe_create_customer();
|
||
|
if ( $this->customer_id !== $customer->id ) {
|
||
|
$this->customer_id = $customer->id;
|
||
|
$this->pending['customer_id'] = $this->customer_id;
|
||
|
}
|
||
|
|
||
|
// If we have something pending, let's save it
|
||
|
if ( ! empty( $this->pending ) ) {
|
||
|
foreach ( $this->pending as $key => $value ) {
|
||
|
switch ( $key ) {
|
||
|
case 'downloads':
|
||
|
case 'fees':
|
||
|
break;
|
||
|
|
||
|
case 'status':
|
||
|
$this->update_status( $this->status );
|
||
|
break;
|
||
|
|
||
|
case 'gateway':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'gateway' => $this->gateway,
|
||
|
) );
|
||
|
break;
|
||
|
|
||
|
case 'mode':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'mode' => $this->mode,
|
||
|
) );
|
||
|
break;
|
||
|
|
||
|
case 'transaction_id':
|
||
|
$this->update_meta( 'transaction_id', $this->transaction_id );
|
||
|
break;
|
||
|
|
||
|
case 'customer_id':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'customer_id' => $this->customer_id,
|
||
|
) );
|
||
|
|
||
|
$customer = new EDD_Customer( $this->customer_id );
|
||
|
$customer->attach_payment( $this->ID, false );
|
||
|
break;
|
||
|
|
||
|
case 'user_id':
|
||
|
edd_update_order(
|
||
|
$this->ID,
|
||
|
array(
|
||
|
'user_id' => $this->user_id,
|
||
|
)
|
||
|
);
|
||
|
|
||
|
$this->user_info['id'] = $this->user_id;
|
||
|
break;
|
||
|
|
||
|
case 'first_name':
|
||
|
$this->user_info['first_name'] = $this->first_name;
|
||
|
break;
|
||
|
|
||
|
case 'last_name':
|
||
|
$this->user_info['last_name'] = $this->last_name;
|
||
|
break;
|
||
|
|
||
|
case 'discounts':
|
||
|
if ( ! is_array( $this->discounts ) ) {
|
||
|
$this->discounts = explode( ',', $this->discounts );
|
||
|
}
|
||
|
|
||
|
$cart_subtotal = 0.00;
|
||
|
|
||
|
foreach ( $this->cart_details as $item ) {
|
||
|
$cart_subtotal += $item['subtotal'];
|
||
|
}
|
||
|
|
||
|
if ( 'none' === $this->discounts[0] ) {
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
foreach ( $this->discounts as $discount ) {
|
||
|
/** @var EDD_Discount $discount_obj */
|
||
|
$discount_obj = edd_get_discount_by( 'code', $discount );
|
||
|
$args = array(
|
||
|
'object_id' => $this->ID,
|
||
|
'object_type' => 'order',
|
||
|
'description' => $discount,
|
||
|
);
|
||
|
|
||
|
if ( false === $discount_obj ) {
|
||
|
$args['type'] = 'fee';
|
||
|
$args['subtotal'] = floatval( $this->total - $cart_subtotal - $this->tax );
|
||
|
$args['total'] = floatval( $this->total - $cart_subtotal - $this->tax );
|
||
|
} else {
|
||
|
$args['type_id'] = $discount_obj->id;
|
||
|
$args['type'] = 'discount';
|
||
|
$args['subtotal'] = floatval( $cart_subtotal - $discount_obj->get_discounted_amount( $cart_subtotal ) );
|
||
|
$args['total'] = floatval( $cart_subtotal - $discount_obj->get_discounted_amount( $cart_subtotal ) );
|
||
|
}
|
||
|
edd_add_order_adjustment( $args );
|
||
|
}
|
||
|
|
||
|
$this->user_info['discount'] = implode( ',', $this->discounts );
|
||
|
break;
|
||
|
|
||
|
case 'address':
|
||
|
$this->user_info['address'] = $this->address;
|
||
|
break;
|
||
|
|
||
|
case 'email':
|
||
|
$this->payment_meta['email'] = $this->email;
|
||
|
$this->user_info['email'] = $this->email;
|
||
|
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'email' => $this->email,
|
||
|
) );
|
||
|
break;
|
||
|
|
||
|
case 'key':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'payment_key' => $this->key,
|
||
|
) );
|
||
|
break;
|
||
|
|
||
|
|
||
|
case 'tax_rate':
|
||
|
$tax_rate = $this->tax_rate > 1 ? $this->tax_rate : ( $this->tax_rate * 100 );
|
||
|
$this->update_meta( '_edd_payment_tax_rate', $tax_rate );
|
||
|
break;
|
||
|
|
||
|
case 'number':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'order_number' => $this->number,
|
||
|
) );
|
||
|
break;
|
||
|
|
||
|
case 'date':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'date_created' => $this->date,
|
||
|
) );
|
||
|
break;
|
||
|
|
||
|
case 'completed_date':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'date_completed' => $this->completed_date,
|
||
|
) );
|
||
|
break;
|
||
|
|
||
|
case 'has_unlimited_downloads':
|
||
|
$this->update_meta( 'unlimited_downloads', $this->has_unlimited_downloads );
|
||
|
break;
|
||
|
|
||
|
case 'parent_payment':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'parent' => $this->parent_payment,
|
||
|
) );
|
||
|
break;
|
||
|
|
||
|
default:
|
||
|
/**
|
||
|
* Used to save non-standard data. Developers can hook here if they want to save
|
||
|
* specific payment data when $payment->save() is run and their item is in the $pending array
|
||
|
*/
|
||
|
do_action( 'edd_payment_save', $this, $key );
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$discount = 0.00;
|
||
|
|
||
|
foreach ( $this->cart_details as $item ) {
|
||
|
$discount += $item['discount'];
|
||
|
}
|
||
|
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'subtotal' => (float) $this->subtotal,
|
||
|
'tax' => (float) $this->tax,
|
||
|
'discount' => $discount,
|
||
|
'total' => (float) $this->total,
|
||
|
) );
|
||
|
|
||
|
$this->downloads = array_values( $this->downloads );
|
||
|
|
||
|
$new_meta = array(
|
||
|
'downloads' => $this->downloads,
|
||
|
'cart_details' => $this->cart_details,
|
||
|
'fees' => $this->fees,
|
||
|
'user_info' => is_array( $this->user_info ) ? $this->user_info : array(),
|
||
|
'date' => $this->date,
|
||
|
'email' => $this->email,
|
||
|
'tax' => $this->tax,
|
||
|
);
|
||
|
|
||
|
// Do some merging of user_info before we merge it all, to honor the edd_payment_meta filter
|
||
|
if ( ! empty( $this->payment_meta['user_info'] ) ) {
|
||
|
$stored_discount = ! empty( $new_meta['user_info']['discount'] ) ? $new_meta['user_info']['discount'] : '';
|
||
|
|
||
|
$new_meta['user_info'] = array_replace_recursive( (array) $this->payment_meta['user_info'], $new_meta['user_info'] );
|
||
|
|
||
|
if ( 'none' !== $stored_discount ) {
|
||
|
$new_meta['user_info']['discount'] = $stored_discount;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$merged_meta = array_merge( $this->payment_meta, $new_meta );
|
||
|
|
||
|
$payment_data = array(
|
||
|
'price' => $this->total,
|
||
|
'date' => $this->date,
|
||
|
'user_email' => $this->email,
|
||
|
'downloads' => $this->downloads,
|
||
|
'user_info' => array(
|
||
|
'id' => $this->user_id,
|
||
|
'email' => $this->email,
|
||
|
'first_name' => $this->first_name,
|
||
|
'last_name' => $this->last_name,
|
||
|
'discount' => $this->discounts,
|
||
|
'address' => $this->address,
|
||
|
),
|
||
|
'cart_details' => $this->cart_details,
|
||
|
'status' => $this->status,
|
||
|
'fees' => $this->fees,
|
||
|
'tax' => $this->tax,
|
||
|
);
|
||
|
$merged_meta = apply_filters( 'edd_payment_meta', $merged_meta, $payment_data );
|
||
|
|
||
|
// Only save the payment meta if it's changed
|
||
|
if ( md5( serialize( $this->payment_meta ) ) !== md5( serialize( $merged_meta ) ) ) {
|
||
|
// First, update the order.
|
||
|
$order_info = array(
|
||
|
'email' => $merged_meta['email'],
|
||
|
);
|
||
|
|
||
|
if ( isset( $merged_meta['user_info']['id'] ) ) {
|
||
|
$order_info['user_id'] = $merged_meta['user_info']['id'];
|
||
|
}
|
||
|
|
||
|
if ( ! empty( $merged_meta['date'] ) ) {
|
||
|
$order_info['date'] = $merged_meta['date'];
|
||
|
}
|
||
|
|
||
|
edd_update_order( $this->ID, $order_info );
|
||
|
|
||
|
// We need to check if all of the order items exist in the database.
|
||
|
$items = edd_get_order_items( array(
|
||
|
'order_id' => $this->ID,
|
||
|
) );
|
||
|
|
||
|
// If an empty set was returned, this is a new payment.
|
||
|
if ( empty( $items ) ) {
|
||
|
foreach ( $merged_meta['cart_details'] as $key => $item ) {
|
||
|
edd_add_order_item( array(
|
||
|
'order_id' => $this->ID,
|
||
|
'product_id' => $item['id'],
|
||
|
'product_name' => $item['name'],
|
||
|
'price_id' => isset( $item['item_number']['options']['price_id'] ) && is_numeric( $item['item_number']['options']['price_id'] )
|
||
|
? absint( $item['item_number']['options']['price_id'] )
|
||
|
: null,
|
||
|
'cart_index' => $key,
|
||
|
'quantity' => $item['quantity'],
|
||
|
'amount' => $item['item_price'],
|
||
|
'subtotal' => $item['subtotal'],
|
||
|
'discount' => $item['discount'],
|
||
|
'tax' => $item['tax'],
|
||
|
'total' => $item['price'],
|
||
|
'status' => ! empty( $item['status'] ) ? $item['status'] : $this->status,
|
||
|
) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Re-fetch the order with the new items from the database as it is used for the synchronization
|
||
|
* between cart_details and the database.
|
||
|
*/
|
||
|
$this->order = $this->_shim_edd_get_order( $this->ID );
|
||
|
|
||
|
$updated = $this->update_meta( '_edd_payment_meta', $merged_meta );
|
||
|
|
||
|
if ( false !== $updated ) {
|
||
|
$saved = true;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$this->pending = array();
|
||
|
$saved = true;
|
||
|
}
|
||
|
|
||
|
if ( true === $saved ) {
|
||
|
$this->setup_payment( $this->ID );
|
||
|
|
||
|
/**
|
||
|
* This action fires anytime that $payment->save() is run, allowing developers to run actions
|
||
|
* when a payment is updated
|
||
|
*/
|
||
|
do_action( 'edd_payment_saved', $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
$customer = new EDD_Customer( $this->customer_id );
|
||
|
$customer->recalculate_stats();
|
||
|
|
||
|
/**
|
||
|
* Update the payment in the object cache
|
||
|
*/
|
||
|
$cache_key = md5( 'edd_payment' . $this->ID );
|
||
|
wp_cache_set( $cache_key, $this, 'payments' );
|
||
|
|
||
|
return $saved;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add a download to a given payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param int $download_id The download to add
|
||
|
* @param array $args Other arguments to pass to the function
|
||
|
* @param array $options List of download options
|
||
|
*
|
||
|
* @return bool True when successful, false otherwise
|
||
|
*/
|
||
|
public function add_download( $download_id = 0, $args = array(), $options = array() ) {
|
||
|
$download = new EDD_Download( $download_id );
|
||
|
|
||
|
// Bail if this post isn't a download.
|
||
|
if ( ! $download || 'download' !== $download->post_type ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Set up defaults.
|
||
|
$defaults = array(
|
||
|
'quantity' => 1,
|
||
|
'price_id' => false,
|
||
|
'item_price' => false,
|
||
|
'discount' => 0,
|
||
|
'tax' => 0.00,
|
||
|
'fees' => array(),
|
||
|
'status' => $this->status,
|
||
|
);
|
||
|
|
||
|
$args = wp_parse_args( apply_filters( 'edd_payment_add_download_args', $args, $download->ID ), $defaults );
|
||
|
|
||
|
// Allow overriding the price.
|
||
|
if ( false !== $args['item_price'] ) {
|
||
|
$item_price = $args['item_price'];
|
||
|
} else {
|
||
|
|
||
|
// Deal with variable pricing.
|
||
|
if ( edd_has_variable_prices( $download->ID ) ) {
|
||
|
$prices = get_post_meta( $download->ID, 'edd_variable_prices', true );
|
||
|
|
||
|
if ( $args['price_id'] && array_key_exists( $args['price_id'], (array) $prices ) ) {
|
||
|
$item_price = $prices[ $args['price_id'] ]['amount'];
|
||
|
} else {
|
||
|
$item_price = edd_get_lowest_price_option( $download->ID );
|
||
|
$args['price_id'] = edd_get_lowest_price_id( $download->ID );
|
||
|
}
|
||
|
} else {
|
||
|
$item_price = edd_get_download_price( $download->ID );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Sanitizing the price here so we don't have a dozen calls later
|
||
|
$item_price = edd_sanitize_amount( $item_price );
|
||
|
$quantity = edd_item_quantities_enabled() ? absint( $args['quantity'] ) : 1;
|
||
|
$amount = round( $item_price * $quantity, edd_currency_decimal_filter() );
|
||
|
|
||
|
// Setup the downloads meta item
|
||
|
$new_download = array(
|
||
|
'id' => $download->ID,
|
||
|
'quantity' => $quantity,
|
||
|
);
|
||
|
|
||
|
$default_options = array(
|
||
|
'quantity' => $quantity,
|
||
|
);
|
||
|
|
||
|
if ( false !== $args['price_id'] ) {
|
||
|
$default_options['price_id'] = (int) $args['price_id'];
|
||
|
}
|
||
|
|
||
|
$options = wp_parse_args( $options, $default_options );
|
||
|
$new_download['options'] = $options;
|
||
|
|
||
|
$this->downloads[] = $new_download;
|
||
|
|
||
|
$discount = $args['discount'];
|
||
|
$subtotal = $amount;
|
||
|
$tax = $args['tax'];
|
||
|
|
||
|
if ( edd_prices_include_tax() ) {
|
||
|
$subtotal -= round( $tax, edd_currency_decimal_filter() );
|
||
|
}
|
||
|
|
||
|
$fees = 0;
|
||
|
if ( ! empty( $args['fees'] ) && is_array( $args['fees'] ) ) {
|
||
|
foreach ( $args['fees'] as $feekey => $fee ) {
|
||
|
$fees += $fee['amount'];
|
||
|
}
|
||
|
|
||
|
$fees = round( $fees, edd_currency_decimal_filter() );
|
||
|
}
|
||
|
|
||
|
$total = $subtotal - $discount + $tax + $fees;
|
||
|
|
||
|
// Do not allow totals to go negative
|
||
|
if ( $total < 0 ) {
|
||
|
$total = 0;
|
||
|
}
|
||
|
|
||
|
// Silly item_number array
|
||
|
$item_number = array(
|
||
|
'id' => $download->ID,
|
||
|
'quantity' => $quantity,
|
||
|
'options' => $options,
|
||
|
);
|
||
|
|
||
|
$this->cart_details[] = array(
|
||
|
'name' => edd_get_download_name( $download->ID, $args['price_id'] ),
|
||
|
'id' => $download->ID,
|
||
|
'item_number' => $item_number,
|
||
|
'item_price' => round( $item_price, edd_currency_decimal_filter() ),
|
||
|
'quantity' => $quantity,
|
||
|
'discount' => $discount,
|
||
|
'subtotal' => round( $subtotal, edd_currency_decimal_filter() ),
|
||
|
'tax' => round( $tax, edd_currency_decimal_filter() ),
|
||
|
'fees' => $args['fees'],
|
||
|
'price' => round( $total, edd_currency_decimal_filter() ),
|
||
|
);
|
||
|
|
||
|
$added_download = end( $this->cart_details );
|
||
|
$added_download['action'] = 'add';
|
||
|
|
||
|
// We need to add the cart index from 3.0+ as it gets stored in the database.
|
||
|
$added_download['cart_index'] = key( $this->cart_details );
|
||
|
|
||
|
$this->pending['downloads'][] = $added_download;
|
||
|
reset( $this->cart_details );
|
||
|
|
||
|
$this->increase_subtotal( $subtotal - $discount );
|
||
|
$this->increase_tax( $tax );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Remove a download from the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param int $download_id The download ID to remove
|
||
|
* @param array $args Arguments to pass to identify (quantity, amount, price_id)
|
||
|
*
|
||
|
* @return bool If the item was removed or not
|
||
|
*/
|
||
|
public function remove_download( $download_id, $args = array() ) {
|
||
|
|
||
|
// Set some defaults
|
||
|
$defaults = array(
|
||
|
'quantity' => 1,
|
||
|
'item_price' => false,
|
||
|
'price_id' => false,
|
||
|
'cart_index' => false,
|
||
|
);
|
||
|
|
||
|
$args = wp_parse_args( $args, $defaults );
|
||
|
|
||
|
$download = new EDD_Download( $download_id );
|
||
|
|
||
|
/**
|
||
|
* Bail if this post isn't a download post type.
|
||
|
*
|
||
|
* We need to allow this to process though for a missing post ID, in case it's a download that was deleted.
|
||
|
*/
|
||
|
if ( ! empty( $download->ID ) && 'download' !== $download->post_type ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
foreach ( $this->downloads as $key => $item ) {
|
||
|
if ( (int) $download_id !== (int) $item['id'] ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if ( false !== $args['price_id'] ) {
|
||
|
if ( isset( $item['options']['price_id'] ) && (int) $args['price_id'] !== (int) $item['options']['price_id'] ) {
|
||
|
continue;
|
||
|
}
|
||
|
} elseif ( false !== $args['cart_index'] ) {
|
||
|
$cart_index = absint( $args['cart_index'] );
|
||
|
$cart_item = ! empty( $this->cart_details[ $cart_index ] ) ? $this->cart_details[ $cart_index ] : false;
|
||
|
|
||
|
if ( ! empty( $cart_item ) ) {
|
||
|
|
||
|
// If the cart index item isn't the same download ID, don't remove it
|
||
|
if ( $cart_item['id'] !== $item['id'] ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
// If this item has a price ID, make sure it matches the cart indexed item's price ID before removing
|
||
|
if ( ( isset( $item['options']['price_id'] ) && isset( $cart_item['item_number']['options']['price_id'] ) )
|
||
|
&& (int) $item['options']['price_id'] !== (int) $cart_item['item_number']['options']['price_id'] ) {
|
||
|
continue;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$item_quantity = $this->downloads[ $key ]['quantity'];
|
||
|
|
||
|
if ( $item_quantity > $args['quantity'] ) {
|
||
|
$this->downloads[ $key ]['quantity'] -= $args['quantity'];
|
||
|
break;
|
||
|
} else {
|
||
|
unset( $this->downloads[ $key ] );
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$found_cart_key = false;
|
||
|
|
||
|
if ( false === $args['cart_index'] ) {
|
||
|
foreach ( $this->cart_details as $cart_key => $item ) {
|
||
|
if ( (int) $download_id !== (int) $item['id'] ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
if ( false !== $args['price_id'] ) {
|
||
|
if ( isset( $item['item_number']['options']['price_id'] ) && (int) $args['price_id'] !== (int) $item['item_number']['options']['price_id'] ) {
|
||
|
continue;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( false !== $args['item_price'] ) {
|
||
|
if ( isset( $item['item_price'] ) && (float) $args['item_price'] !== (float) $item['item_price'] ) {
|
||
|
continue;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$found_cart_key = (int) $cart_key;
|
||
|
break;
|
||
|
}
|
||
|
} else {
|
||
|
$cart_index = absint( $args['cart_index'] );
|
||
|
|
||
|
if ( ! array_key_exists( $cart_index, $this->cart_details ) ) {
|
||
|
return false; // Invalid cart index passed.
|
||
|
}
|
||
|
|
||
|
if ( (int) $this->cart_details[ $cart_index ]['id'] !== (int) $download_id ) {
|
||
|
return false; // We still need the proper Download ID to be sure.
|
||
|
}
|
||
|
|
||
|
$found_cart_key = $cart_index;
|
||
|
}
|
||
|
|
||
|
$orig_quantity = $this->cart_details[ $found_cart_key ]['quantity'];
|
||
|
|
||
|
if ( $orig_quantity > $args['quantity'] ) {
|
||
|
$this->cart_details[ $found_cart_key ]['quantity'] -= $args['quantity'];
|
||
|
|
||
|
$item_price = $this->cart_details[ $found_cart_key ]['item_price'];
|
||
|
$tax = $this->cart_details[ $found_cart_key ]['tax'];
|
||
|
$discount = ! empty( $this->cart_details[ $found_cart_key ]['discount'] ) ? $this->cart_details[ $found_cart_key ]['discount'] : 0;
|
||
|
|
||
|
// The total reduction equals the number removed * the item_price
|
||
|
$total_reduced = round( $item_price * $args['quantity'], edd_currency_decimal_filter() );
|
||
|
$tax_reduced = round( ( $tax / $orig_quantity ) * $args['quantity'], edd_currency_decimal_filter() );
|
||
|
|
||
|
$new_quantity = $this->cart_details[ $found_cart_key ]['quantity'];
|
||
|
$new_tax = $this->cart_details[ $found_cart_key ]['tax'] - $tax_reduced;
|
||
|
$new_subtotal = $new_quantity * $item_price;
|
||
|
$new_discount = 0;
|
||
|
$new_total = 0;
|
||
|
|
||
|
$this->cart_details[ $found_cart_key ]['subtotal'] = $new_subtotal;
|
||
|
$this->cart_details[ $found_cart_key ]['discount'] = $new_discount;
|
||
|
$this->cart_details[ $found_cart_key ]['tax'] = $new_tax;
|
||
|
$this->cart_details[ $found_cart_key ]['price'] = $new_subtotal - $new_discount + $new_tax;
|
||
|
} else {
|
||
|
$total_reduced = $this->cart_details[ $found_cart_key ]['item_price'];
|
||
|
$tax_reduced = $this->cart_details[ $found_cart_key ]['tax'];
|
||
|
|
||
|
$found_fees = array();
|
||
|
|
||
|
if ( ! empty( $this->cart_details[ $found_cart_key ]['fees'] ) ) {
|
||
|
$found_fees = $this->cart_details[ $found_cart_key ]['fees'];
|
||
|
|
||
|
foreach ( $found_fees as $key => $fee ) {
|
||
|
$this->remove_fee( $key );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
unset( $this->cart_details[ $found_cart_key ] );
|
||
|
}
|
||
|
|
||
|
$pending_args = $args;
|
||
|
$pending_args['id'] = $download_id;
|
||
|
$pending_args['amount'] = $total_reduced;
|
||
|
$pending_args['price_id'] = false !== $args['price_id'] ? $args['price_id'] : false;
|
||
|
$pending_args['quantity'] = $args['quantity'];
|
||
|
$pending_args['action'] = 'remove';
|
||
|
$pending_args['fees'] = isset( $found_fees ) ? $found_fees : array();
|
||
|
$pending_args['cart_index'] = $found_cart_key;
|
||
|
|
||
|
$this->pending['downloads'][] = $pending_args;
|
||
|
|
||
|
/**
|
||
|
* Remove/modify the order item from the database at this point in lieu of having to synchronise with cart_details
|
||
|
* later on in update_meta().
|
||
|
*/
|
||
|
|
||
|
// Find the order item based on the cart index.
|
||
|
$order_item = array_filter( $this->order->items, function ( $i ) use ( $found_cart_key ) {
|
||
|
/** @var EDD\Orders\Order_Item $i */
|
||
|
|
||
|
return (int) $i->cart_index === (int) $found_cart_key;
|
||
|
} );
|
||
|
|
||
|
// Reset array index.
|
||
|
$order_item = array_values( $order_item );
|
||
|
|
||
|
$order_item = ( 1 === count( $order_item ) )
|
||
|
? $order_item[0]
|
||
|
: null;
|
||
|
|
||
|
/** @var EDD\Orders\Order_Item $order_item */
|
||
|
|
||
|
// Ensure an order item exists in the database.
|
||
|
if ( ! is_null( $order_item ) ) {
|
||
|
|
||
|
// Update the order item if the quantity is being modified.
|
||
|
if ( isset( $this->cart_details[ $found_cart_key ] ) ) {
|
||
|
edd_update_order_item( $order_item->id, array(
|
||
|
'quantity' => $this->cart_details[ $found_cart_key ]['quantity'],
|
||
|
'amount' => $this->cart_details[ $found_cart_key ]['item_price'],
|
||
|
'subtotal' => $this->cart_details[ $found_cart_key ]['subtotal'],
|
||
|
'discount' => $this->cart_details[ $found_cart_key ]['discount'],
|
||
|
'tax' => $this->cart_details[ $found_cart_key ]['tax'],
|
||
|
'total' => $this->cart_details[ $found_cart_key ]['price'],
|
||
|
) );
|
||
|
|
||
|
// Remove the order item.
|
||
|
} else {
|
||
|
edd_delete_order_item( $order_item->id );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$this->decrease_subtotal( $total_reduced );
|
||
|
$this->decrease_tax( $tax_reduced );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Alter a limited set of properties of a cart item
|
||
|
*
|
||
|
* @since 2.7
|
||
|
*
|
||
|
* @param bool $cart_index
|
||
|
* @param array $args
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function modify_cart_item( $cart_index = false, $args = array() ) {
|
||
|
if ( false === $cart_index ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if ( ! array_key_exists( $cart_index, $this->cart_details ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$current_args = $this->cart_details[ $cart_index ];
|
||
|
$allowed_items = apply_filters( 'edd_allowed_cart_item_modifications', array(
|
||
|
'item_price',
|
||
|
'tax',
|
||
|
'discount',
|
||
|
'quantity',
|
||
|
) );
|
||
|
|
||
|
// Remove any items we don't want to modify.
|
||
|
foreach ( $args as $key => $arg ) {
|
||
|
if ( ! in_array( $key, $allowed_items, true ) ) {
|
||
|
unset( $args[ $key ] );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$merged_item = array_merge( $current_args, $args );
|
||
|
|
||
|
if ( md5( json_encode( $current_args ) ) === md5( json_encode( $merged_item ) ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Format the item_price correctly now
|
||
|
$merged_item['item_price'] = edd_sanitize_amount( $merged_item['item_price'] );
|
||
|
|
||
|
$discount = isset( $merged_item['discount'] )
|
||
|
? (float) $merged_item['discount']
|
||
|
: 0.00;
|
||
|
|
||
|
$new_subtotal = floatval( $merged_item['item_price'] ) * $merged_item['quantity'];
|
||
|
$merged_item['tax'] = edd_sanitize_amount( $merged_item['tax'] );
|
||
|
$merged_item['price'] = edd_prices_include_tax() ? $new_subtotal - $discount : $new_subtotal + $merged_item['tax'] - $discount;
|
||
|
$this->cart_details[ $cart_index ] = $merged_item;
|
||
|
|
||
|
// Sort the current and new args, and checksum them. If no changes. No need to fire a modification.
|
||
|
ksort( $current_args );
|
||
|
ksort( $merged_item );
|
||
|
|
||
|
$modified_download = $merged_item;
|
||
|
$modified_download['action'] = 'modify';
|
||
|
$modified_download['previous_data'] = $current_args;
|
||
|
|
||
|
$this->pending['downloads'][] = $modified_download;
|
||
|
|
||
|
if ( $new_subtotal > $current_args['subtotal'] ) {
|
||
|
$this->increase_subtotal( ( $new_subtotal - (float) $modified_download['discount'] ) - (float) $current_args['subtotal'] );
|
||
|
} else {
|
||
|
$this->decrease_subtotal( (float) $current_args['subtotal'] - ( $new_subtotal - (float) $modified_download['discount'] ) );
|
||
|
}
|
||
|
|
||
|
if ( (float) $modified_download['tax'] > (float) $current_args['tax'] ) {
|
||
|
$this->increase_tax( (float) $modified_download['tax'] - (float) $current_args['tax'] );
|
||
|
} else {
|
||
|
$this->decrease_tax( (float) $current_args['tax'] - (float) $modified_download['tax'] );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Remove/modify the order item from the database at this point in lieu of having to synchronise with cart_details
|
||
|
* later on in update_meta().
|
||
|
*/
|
||
|
|
||
|
// Find the order item.
|
||
|
$order_item_id = 0;
|
||
|
|
||
|
foreach ( $this->order->items as $item ) {
|
||
|
if ( (int) $item->cart_index === (int) $cart_index ) {
|
||
|
$order_item_id = $item->id;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( $order_item_id ) {
|
||
|
edd_update_order_item( $order_item_id, array(
|
||
|
'quantity' => $modified_download['quantity'],
|
||
|
'amount' => (float) $modified_download['item_price'],
|
||
|
'subtotal' => (float) $new_subtotal,
|
||
|
'tax' => (float) $modified_download['tax'],
|
||
|
'discount' => (float) $modified_download['discount'],
|
||
|
'total' => (float) $modified_download['price'],
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add a fee to a given payment.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param array $args Array of arguments for the fee to add.
|
||
|
* @param bool $global
|
||
|
*
|
||
|
* @return bool If the fee was added.
|
||
|
*/
|
||
|
public function add_fee( $args, $global = true ) {
|
||
|
$default_args = array(
|
||
|
'label' => '',
|
||
|
'amount' => 0,
|
||
|
'type' => 'fee',
|
||
|
'id' => '',
|
||
|
'no_tax' => false,
|
||
|
'download_id' => 0,
|
||
|
);
|
||
|
|
||
|
$fee = wp_parse_args( $args, $default_args );
|
||
|
$this->fees[] = $fee;
|
||
|
|
||
|
$added_fee = $fee;
|
||
|
$added_fee['action'] = 'add';
|
||
|
$this->pending['fees'][] = $added_fee;
|
||
|
reset( $this->fees );
|
||
|
|
||
|
$this->increase_fees( $fee['amount'] );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Remove a fee from the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param int $key The array key index to remove
|
||
|
*
|
||
|
* @return bool If the fee was removed successfully
|
||
|
*/
|
||
|
public function remove_fee( $key ) {
|
||
|
$removed = $this->remove_fee_by( 'index', $key );
|
||
|
|
||
|
return $removed;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Remove a fee by the defined attributed
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param string $key The key to remove by
|
||
|
* @param int|string $value The value to search for
|
||
|
* @param boolean $global False - removes the first value it finds, True - removes all matches
|
||
|
*
|
||
|
* @return boolean If the item is removed.
|
||
|
*/
|
||
|
public function remove_fee_by( $key, $value, $global = false ) {
|
||
|
$allowed_fee_keys = apply_filters( 'edd_payment_fee_keys', array(
|
||
|
'index',
|
||
|
'label',
|
||
|
'amount',
|
||
|
'type',
|
||
|
) );
|
||
|
|
||
|
if ( ! in_array( $key, $allowed_fee_keys, true ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$removed = false;
|
||
|
|
||
|
if ( 'index' === $key && array_key_exists( $value, $this->fees ) ) {
|
||
|
$removed_fee = $this->fees[ $value ];
|
||
|
$removed_fee['action'] = 'remove';
|
||
|
$this->pending['fees'][] = $removed_fee;
|
||
|
|
||
|
$this->decrease_fees( $removed_fee['amount'] );
|
||
|
|
||
|
unset( $this->fees[ $value ] );
|
||
|
$removed = true;
|
||
|
} elseif ( 'index' !== $key ) {
|
||
|
foreach ( $this->fees as $index => $fee ) {
|
||
|
if ( isset( $fee[ $key ] ) && $fee[ $key ] === $value ) {
|
||
|
$removed_fee = $fee;
|
||
|
$removed_fee['action'] = 'remove';
|
||
|
$this->pending['fees'][] = $removed_fee;
|
||
|
|
||
|
$this->decrease_fees( $removed_fee['amount'] );
|
||
|
|
||
|
unset( $this->fees[ $index ] );
|
||
|
$removed = true;
|
||
|
|
||
|
if ( false === $global ) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Remove the fee from the database at this point in lieu of having to synchronise with payment meta
|
||
|
* later on in update_meta()/save().
|
||
|
*/
|
||
|
if ( true === $removed ) {
|
||
|
$fee = end( $this->pending['fees'] );
|
||
|
|
||
|
$fee_id = 'index' === $key
|
||
|
? $value
|
||
|
: null;
|
||
|
|
||
|
// Find by fee ID, if set.
|
||
|
if ( ! is_null( $fee_id ) ) {
|
||
|
foreach ( $this->order->get_fees() as $id => $f ) {
|
||
|
if ( $id === $fee_id ) {
|
||
|
edd_delete_order_adjustment( $f->id );
|
||
|
|
||
|
if ( false === $global ) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Find by fee label.
|
||
|
} else {
|
||
|
foreach ( $this->order->get_fees() as $f ) {
|
||
|
if ( $fee['label'] === $f->description ) {
|
||
|
edd_delete_order_adjustment( $f->id );
|
||
|
|
||
|
if ( false === $global ) {
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $removed;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the fees, filterable by type.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param string $type All, item, fee.
|
||
|
*
|
||
|
* @return array Fees for the type specified.
|
||
|
*/
|
||
|
public function get_fees( $type = 'all' ) {
|
||
|
$fees = array();
|
||
|
|
||
|
if ( ! empty( $this->fees ) && is_array( $this->fees ) ) {
|
||
|
foreach ( $this->fees as $fee_id => $fee ) {
|
||
|
if ( 'all' !== $type && ! empty( $fee['type'] ) && $type !== $fee['type'] ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$fee['id'] = $fee_id;
|
||
|
$fees[] = $fee;
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return apply_filters( 'edd_get_payment_fees', $fees, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add a note to an order.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Return true if note was inserted successfully.
|
||
|
*
|
||
|
* @param string $note The note to add.
|
||
|
*
|
||
|
* @return bool Whether or not the note was inserted.
|
||
|
*/
|
||
|
public function add_note( $note = '' ) {
|
||
|
|
||
|
// Bail if no note specified.
|
||
|
if ( empty( $note ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$note_id = edd_insert_payment_note( $this->ID, $note );
|
||
|
|
||
|
if ( $note_id ) {
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Increase the payment's subtotal
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param float $amount The amount to increase the payment subtotal by
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
private function increase_subtotal( $amount = 0.00 ) {
|
||
|
$amount = (float) $amount;
|
||
|
$this->subtotal += $amount;
|
||
|
|
||
|
$this->recalculate_total();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Decrease the payment's subtotal
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param float $amount The amount to decrease the payment subtotal by
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
private function decrease_subtotal( $amount = 0.00 ) {
|
||
|
$amount = (float) $amount;
|
||
|
$this->subtotal -= $amount;
|
||
|
|
||
|
if ( $this->subtotal < 0 ) {
|
||
|
$this->subtotal = 0;
|
||
|
}
|
||
|
|
||
|
$this->recalculate_total();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Increase the payment's subtotal
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param float $amount The amount to increase the payment subtotal by
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
private function increase_fees( $amount = 0.00 ) {
|
||
|
$amount = (float) $amount;
|
||
|
$this->fees_total += $amount;
|
||
|
|
||
|
$this->recalculate_total();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Decrease the payment's subtotal
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param float $amount The amount to decrease the payment subtotal by
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
private function decrease_fees( $amount = 0.00 ) {
|
||
|
$amount = (float) $amount;
|
||
|
$this->fees_total -= $amount;
|
||
|
|
||
|
if ( $this->fees_total < 0 ) {
|
||
|
$this->fees_total = 0;
|
||
|
}
|
||
|
|
||
|
$this->recalculate_total();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set or update the total for a payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @return void
|
||
|
*/
|
||
|
private function recalculate_total() {
|
||
|
$this->total = $this->subtotal + $this->tax + $this->fees_total;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Increase the payment's tax by the provided amount
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param float $amount The amount to increase the payment tax by
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
public function increase_tax( $amount = 0.00 ) {
|
||
|
$amount = (float) $amount;
|
||
|
$this->tax += $amount;
|
||
|
|
||
|
$this->recalculate_total();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Decrease the payment's tax by the provided amount
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param float $amount The amount to reduce the payment tax by
|
||
|
*
|
||
|
* @return void
|
||
|
*/
|
||
|
public function decrease_tax( $amount = 0.00 ) {
|
||
|
$amount = (float) $amount;
|
||
|
$this->tax -= $amount;
|
||
|
|
||
|
if ( $this->tax < 0 ) {
|
||
|
$this->tax = 0;
|
||
|
}
|
||
|
|
||
|
$this->recalculate_total();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Change the status of an order to refunded, and run the necessary changes.
|
||
|
*
|
||
|
* @since 2.5.7
|
||
|
*/
|
||
|
public function refund() {
|
||
|
$this->old_status = $this->status;
|
||
|
$this->status = 'refunded';
|
||
|
$this->pending['status'] = $this->status;
|
||
|
|
||
|
$this->save();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Set the order status and run any status specific changes necessary.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Updated to work with the new refunds API and new query methods
|
||
|
* introduced.
|
||
|
*
|
||
|
* @param string $status New order status.
|
||
|
* @return bool True if the status was successfully updated, false otherwise.
|
||
|
*/
|
||
|
public function update_status( $status = '' ) {
|
||
|
|
||
|
if ( ! $this->order ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Bail if an empty status is passed.
|
||
|
if ( empty( $status ) || ! $status ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Override to `complete` since 3.0.
|
||
|
if ( 'completed' === $status || 'publish' === $status ) {
|
||
|
$status = 'complete';
|
||
|
}
|
||
|
|
||
|
// Get the old (current) status.
|
||
|
$old_status = ! empty( $this->old_status )
|
||
|
? $this->old_status
|
||
|
: false;
|
||
|
|
||
|
// We do not allow status changes if the status is the same to that stored in the database.
|
||
|
// This prevents the `edd_update_payment_status` action from being triggered unnecessarily.
|
||
|
if ( $old_status === $status ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$do_change = apply_filters( 'edd_should_update_payment_status', true, $this->ID, $status, $old_status );
|
||
|
|
||
|
$updated = false;
|
||
|
|
||
|
if ( $do_change ) {
|
||
|
do_action( 'edd_before_payment_status_change', $this->ID, $status, $old_status );
|
||
|
|
||
|
$update_fields = apply_filters( 'edd_update_payment_status_fields', array(
|
||
|
'status' => $status,
|
||
|
) );
|
||
|
|
||
|
// Account for someone filtering and using `post_status`
|
||
|
if ( isset( $update_fields['post_status'] ) ) {
|
||
|
_edd_generic_deprecated( 'EDD_Payment::update_status', '3.0', __( 'Array key "post_status" is no longer a supported attribute for the "edd_update_payment_status_fields" filter. Please use "status" instead.', 'easy-digital-downloads' ) );
|
||
|
|
||
|
$update_fields['status'] = $update_fields['post_status'];
|
||
|
unset( $update_fields['post_status'] );
|
||
|
}
|
||
|
|
||
|
// Strip data that does not need to be passed to `edd_update_order()`.
|
||
|
unset( $update_fields['ID'] );
|
||
|
|
||
|
/**
|
||
|
* As per the new refund API introduced in 3.0, the order is only
|
||
|
* marked as refunded when `EDD_Payment::process_refund()` has called
|
||
|
* `edd_refund_order()` and a new order has been generated with a
|
||
|
* type of `refund`.
|
||
|
*
|
||
|
* @since 3.0
|
||
|
* @see EDD_Payment::process_refund()
|
||
|
* @see edd_refund_order()
|
||
|
* @see https://github.com/easydigitaldownloads/easy-digital-downloads/issues/2721
|
||
|
*/
|
||
|
if ( 'refunded' !== $status ) {
|
||
|
edd_update_order( $this->ID, $update_fields );
|
||
|
|
||
|
// Update each order item.
|
||
|
foreach ( $this->order->items as $item ) {
|
||
|
edd_update_order_item( $item->id, $update_fields );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Albeit the order itself is not updated (for refunds), the EDD_Payment
|
||
|
* class vars are updated for backwards compatibility purposes and
|
||
|
* for anyone/anything that is checking that the status of the object
|
||
|
* has successfully changed.
|
||
|
*/
|
||
|
$this->status = $status;
|
||
|
$this->post_status = $status;
|
||
|
|
||
|
$all_payment_statuses = edd_get_payment_statuses();
|
||
|
$this->status_nicename = array_key_exists( $status, $all_payment_statuses )
|
||
|
? $all_payment_statuses[ $status ]
|
||
|
: ucfirst( $status );
|
||
|
|
||
|
// Process any specific status functions.
|
||
|
switch ( $status ) {
|
||
|
case 'refunded':
|
||
|
$this->process_refund();
|
||
|
do_action( 'edd_update_payment_status', $this->ID, $status, $old_status );
|
||
|
break;
|
||
|
case 'failed':
|
||
|
$this->process_failure();
|
||
|
break;
|
||
|
case 'pending' || 'processing':
|
||
|
$this->process_pending();
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $updated;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get a post meta item for the payment
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @param string $meta_key The Meta Key
|
||
|
* @param boolean $single Return single item or array
|
||
|
*
|
||
|
* @return mixed The value from the post meta
|
||
|
*/
|
||
|
public function get_meta( $meta_key = '_edd_payment_meta', $single = true ) {
|
||
|
if ( $this->is_edd_payment ) {
|
||
|
return get_post_meta( $this->ID, $meta_key, $single );
|
||
|
}
|
||
|
$meta = edd_get_order_meta( $this->ID, $meta_key, $single );
|
||
|
|
||
|
// Backwards compatibility.
|
||
|
switch ( $meta_key ) {
|
||
|
case '_edd_payment_purchase_key':
|
||
|
$meta = $this->order->payment_key;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_transaction_id':
|
||
|
$transactions = array_values( edd_get_order_transactions( array(
|
||
|
'number' => 1,
|
||
|
'object_id' => $this->ID,
|
||
|
'object_type' => 'order',
|
||
|
'orderby' => 'date_created',
|
||
|
'order' => 'ASC',
|
||
|
'fields' => 'transaction_id',
|
||
|
) ) );
|
||
|
|
||
|
$transaction_id = '';
|
||
|
|
||
|
if ( $transactions ) {
|
||
|
$transaction_id = esc_attr( $transactions[0] );
|
||
|
}
|
||
|
|
||
|
$meta = $transaction_id;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_user_email':
|
||
|
$meta = $this->order->email;
|
||
|
break;
|
||
|
|
||
|
case '_edd_completed_date':
|
||
|
$meta = $this->completed_date;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_gateway':
|
||
|
$meta = $this->order->gateway;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_user_id':
|
||
|
$meta = $this->order->user_id;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_user_ip':
|
||
|
$meta = $this->order->ip;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_mode':
|
||
|
$meta = $this->order->mode;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_tax_rate':
|
||
|
$meta = $this->order->get_tax_rate();
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_customer_id':
|
||
|
$meta = $this->order->customer_id;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_tax':
|
||
|
$meta = $this->order->tax;
|
||
|
break;
|
||
|
|
||
|
case '_edd_payment_number':
|
||
|
$meta = $this->order->get_number();
|
||
|
break;
|
||
|
}
|
||
|
|
||
|
if ( '_edd_payment_meta' === $meta_key ) {
|
||
|
if ( empty( $meta ) ) {
|
||
|
$meta = array();
|
||
|
}
|
||
|
|
||
|
// Payment meta was simplified in EDD v1.5, so these are here for backwards compatibility
|
||
|
if ( empty( $meta['key'] ) ) {
|
||
|
$meta['key'] = $this->key;
|
||
|
}
|
||
|
|
||
|
if ( empty( $meta['email'] ) ) {
|
||
|
$meta['email'] = $this->email;
|
||
|
}
|
||
|
|
||
|
if ( empty( $meta['date'] ) ) {
|
||
|
$meta['date'] = $this->date;
|
||
|
}
|
||
|
|
||
|
// We need to back fill the returned meta for backwards compatibility purposes.
|
||
|
$meta['key'] = $this->key;
|
||
|
$meta['email'] = $this->email;
|
||
|
$meta['date'] = $this->date;
|
||
|
$meta['user_info'] = $this->user_info;
|
||
|
$meta['downloads'] = $this->downloads;
|
||
|
$meta['cart_details'] = $this->cart_details;
|
||
|
$meta['fees'] = $this->fees;
|
||
|
$meta['currency'] = $this->currency;
|
||
|
$meta['tax'] = $this->tax;
|
||
|
|
||
|
$migrated_payment_meta = edd_get_order_meta( $this->ID, 'payment_meta', true );
|
||
|
|
||
|
// This is no longer stored in _edd_payment_meta.
|
||
|
$core_meta_keys = array( 'key', 'email', 'date', 'user_info', 'downloads', 'cart_details', 'quantity', 'discount', 'subtotal', 'tax', 'fees', 'currency' );
|
||
|
|
||
|
$migrated_payment_meta = array_diff_key( (array) $migrated_payment_meta, array_flip( $core_meta_keys ) );
|
||
|
|
||
|
if ( is_array( $migrated_payment_meta ) && 0 < count( $migrated_payment_meta ) ) {
|
||
|
$meta = array_merge( $meta, $migrated_payment_meta );
|
||
|
}
|
||
|
|
||
|
// #5228 Fix possible data issue introduced in 2.6.12
|
||
|
if ( is_array( $meta ) && isset( $meta[0] ) ) {
|
||
|
$bad_meta = $meta[0];
|
||
|
unset( $meta[0] );
|
||
|
|
||
|
if ( is_array( $bad_meta ) ) {
|
||
|
$meta = array_merge( $meta, $bad_meta );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$meta = apply_filters( 'edd_get_payment_meta_' . $meta_key, $meta, $this->ID );
|
||
|
|
||
|
if ( is_serialized( $meta ) ) {
|
||
|
preg_match( '/[oO]\s*:\s*\d+\s*:\s*"\s*(?!(?i)(stdClass))/', $meta, $matches );
|
||
|
if ( ! empty( $matches ) ) {
|
||
|
$meta = array();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return apply_filters( 'edd_get_payment_meta', $meta, $this->ID, $meta_key );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Update the order meta.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Updated to use the new custom tables.
|
||
|
*
|
||
|
* @param string $meta_key The meta key to update.
|
||
|
* @param string $meta_value The meta value.
|
||
|
* @param string $prev_value Previous meta value.
|
||
|
*
|
||
|
* @return int|bool Meta ID if the key didn't exist, true on successful update, false on failure.
|
||
|
*/
|
||
|
public function update_meta( $meta_key = '', $meta_value = '', $prev_value = '' ) {
|
||
|
if ( empty( $meta_key ) || empty( $this->ID ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
$meta_value = apply_filters( 'edd_update_payment_meta_' . $meta_key, $meta_value, $this->ID );
|
||
|
|
||
|
switch ( $meta_key ) {
|
||
|
case '_edd_payment_meta':
|
||
|
if ( isset( $meta_value['tax'] ) && ! empty( $meta_value['tax'] ) ) {
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'tax' => $meta_value['tax'],
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
if ( isset( $meta_value['key'] ) && ! empty( $meta_value['key'] ) ) {
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'key' => $meta_value['key'],
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
if ( isset( $meta_value['email'] ) && ! empty( $meta_value['email'] ) ) {
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'email' => $meta_value['email'],
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
if ( isset( $meta_value['currency'] ) && ! empty( $meta_value['currency'] ) ) {
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'currency' => $meta_value['currency'],
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
if ( isset( $meta_value['user_info'] ) && ! empty( $meta_value['user_info'] ) ) {
|
||
|
|
||
|
// Handle discounts.
|
||
|
$discounts = isset( $meta_value['user_info']['discount'] ) && ! empty( $meta_value['user_info']['discount'] )
|
||
|
? $meta_value['user_info']['discount']
|
||
|
: array();
|
||
|
|
||
|
if ( ! is_array( $discounts ) ) {
|
||
|
$discounts = explode( ',', $discounts );
|
||
|
}
|
||
|
|
||
|
if ( ! empty( $discounts ) && ( 'none' !== $discounts[0] ) ) {
|
||
|
foreach ( $discounts as $discount ) {
|
||
|
|
||
|
/** @var EDD_Discount $discount */
|
||
|
$discount = edd_get_discount_by( 'code', $discount );
|
||
|
|
||
|
if ( false === $discount ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
$adjustments = $this->order->adjustments;
|
||
|
|
||
|
$found_discount = array_filter( $adjustments, function( $adjustment ) use ( $discount ) {
|
||
|
/** @var EDD\Orders\Order_Adjustment $adjustment */
|
||
|
|
||
|
return (string) $adjustment->description === (string) $discount->code;
|
||
|
} );
|
||
|
|
||
|
// Discount exists so update the amount.
|
||
|
if ( 1 === count( $found_discount ) ) {
|
||
|
$found_discount = $found_discount[0];
|
||
|
|
||
|
/** @var EDD\Orders\Order_Adjustment $found_discount */
|
||
|
|
||
|
edd_update_order_adjustment( $found_discount->id, array(
|
||
|
'amount' => $this->subtotal - $discount->get_discounted_amount( $this->subtotal ),
|
||
|
) );
|
||
|
} else {
|
||
|
// Add the discount as an adjustment.
|
||
|
edd_add_order_adjustment(
|
||
|
array(
|
||
|
'object_id' => $this->ID,
|
||
|
'object_type' => 'order',
|
||
|
'type_id' => $discount->id,
|
||
|
'type' => 'discount',
|
||
|
'description' => $discount->code,
|
||
|
'subtotal' => $this->subtotal - $discount->get_discounted_amount( $this->subtotal ),
|
||
|
'total' => $this->subtotal - $discount->get_discounted_amount( $this->subtotal ),
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$user_info = array_diff_key( $meta_value['user_info'], array_flip( array(
|
||
|
'id',
|
||
|
'email',
|
||
|
'discount'
|
||
|
) ) );
|
||
|
|
||
|
$defaults = array(
|
||
|
'first_name' => '',
|
||
|
'last_name' => '',
|
||
|
'address' => array(
|
||
|
'line1' => '',
|
||
|
'line2' => '',
|
||
|
'city' => '',
|
||
|
'state' => '',
|
||
|
'country' => '',
|
||
|
'zip' => '',
|
||
|
),
|
||
|
);
|
||
|
|
||
|
if ( isset( $user_info['address'] ) ) {
|
||
|
$user_info['address'] = wp_parse_args( $user_info['address'], $defaults['address'] );
|
||
|
}
|
||
|
|
||
|
$user_info = wp_parse_args( $user_info, $defaults );
|
||
|
$name = $user_info['first_name'] . ' ' . $user_info['last_name'];
|
||
|
|
||
|
if ( null !== $this->order && $this->order->get_address()->id ) {
|
||
|
$order_address = $this->order->get_address();
|
||
|
|
||
|
edd_update_order_address( $order_address->id, array(
|
||
|
'name' => $name,
|
||
|
'address' => $user_info['address']['line1'],
|
||
|
'address2' => $user_info['address']['line2'],
|
||
|
'city' => $user_info['address']['city'],
|
||
|
'region' => $user_info['address']['state'],
|
||
|
'postal_code' => $user_info['address']['zip'],
|
||
|
'country' => $user_info['address']['country'],
|
||
|
) );
|
||
|
} else {
|
||
|
edd_add_order_address( array(
|
||
|
'order_id' => $this->ID,
|
||
|
'name' => $name,
|
||
|
'address' => $user_info['address']['line1'],
|
||
|
'address2' => $user_info['address']['line2'],
|
||
|
'city' => $user_info['address']['city'],
|
||
|
'region' => $user_info['address']['state'],
|
||
|
'postal_code' => $user_info['address']['zip'],
|
||
|
'country' => $user_info['address']['country'],
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
$remaining_user_info = array_diff_key( $meta_value['user_info'], array_flip( array(
|
||
|
'id',
|
||
|
'first_name',
|
||
|
'last_name',
|
||
|
'email',
|
||
|
'address',
|
||
|
'discount'
|
||
|
) ) );
|
||
|
|
||
|
if ( ! empty( $remaining_user_info ) ) {
|
||
|
edd_update_order_meta( $this->ID, 'user_info', $remaining_user_info );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( isset( $meta_value['fees'] ) && ! empty( $meta_value['fees'] ) ) {
|
||
|
foreach ( $meta_value['fees'] as $fee_id => $fee ) {
|
||
|
if ( ! empty( $fee['download_id'] ) && 0 < $fee['download_id'] ) {
|
||
|
$order_item_id = edd_get_order_items( array(
|
||
|
'number' => 1,
|
||
|
'order_id' => $this->ID,
|
||
|
'product_id' => $fee['download_id'],
|
||
|
'price_id' => isset( $fee['price_id'] ) && ! is_null( $fee['price_id'] ) ? intval( $fee['price_id'] ) : 0,
|
||
|
'fields' => 'ids',
|
||
|
) );
|
||
|
|
||
|
if ( is_array( $order_item_id ) ) {
|
||
|
$order_item_id = (int) $order_item_id[0];
|
||
|
}
|
||
|
|
||
|
$adjustment_id = edd_get_order_adjustments( array(
|
||
|
'number' => 1,
|
||
|
'object_id' => $order_item_id,
|
||
|
'object_type' => 'order_item',
|
||
|
'type' => 'fee',
|
||
|
'fields' => 'ids',
|
||
|
'type_key' => $fee_id,
|
||
|
) );
|
||
|
|
||
|
if ( is_array( $adjustment_id ) && ! empty( $adjustment_id ) ) {
|
||
|
$adjustment_id = $adjustment_id[0];
|
||
|
|
||
|
edd_update_order_adjustment( $adjustment_id, array(
|
||
|
'description' => $fee['label'],
|
||
|
'subtotal' => (float) $fee['amount'],
|
||
|
) );
|
||
|
} else {
|
||
|
add_filter( 'edd_prices_include_tax', '__return_false' );
|
||
|
|
||
|
$tax = ( isset( $fee['no_tax'] ) && false === $fee['no_tax'] ) || $fee['amount'] < 0
|
||
|
? floatval( edd_calculate_tax( $fee['amount'] ) )
|
||
|
: 0.00;
|
||
|
|
||
|
remove_filter( 'edd_prices_include_tax', '__return_false' );
|
||
|
|
||
|
$adjustment_id = edd_add_order_adjustment( array(
|
||
|
'object_id' => $order_item_id,
|
||
|
'object_type' => 'order_item',
|
||
|
'type_key' => $fee_id,
|
||
|
'type' => 'fee',
|
||
|
'description' => $fee['label'],
|
||
|
'subtotal' => floatval( $fee['amount'] ),
|
||
|
'tax' => $tax,
|
||
|
'total' => floatval( $fee['amount'] ) + $tax
|
||
|
) );
|
||
|
}
|
||
|
} else {
|
||
|
$adjustment_id = edd_get_order_adjustments( array(
|
||
|
'number' => 1,
|
||
|
'object_id' => $this->ID,
|
||
|
'object_type' => 'order',
|
||
|
'type' => 'fee',
|
||
|
'fields' => 'ids',
|
||
|
'type_key' => $fee_id,
|
||
|
) );
|
||
|
|
||
|
if ( is_array( $adjustment_id ) && ! empty( $adjustment_id ) ) {
|
||
|
$adjustment_id = $adjustment_id[0];
|
||
|
|
||
|
edd_update_order_adjustment( $adjustment_id, array(
|
||
|
'description' => $fee['label'],
|
||
|
'subtotal' => (float) $fee['amount'],
|
||
|
) );
|
||
|
} else {
|
||
|
add_filter( 'edd_prices_include_tax', '__return_false' );
|
||
|
|
||
|
$tax = ( isset( $fee['no_tax'] ) && false === $fee['no_tax'] ) || $fee['amount'] < 0
|
||
|
? floatval( edd_calculate_tax( $fee['amount'] ) )
|
||
|
: 0.00;
|
||
|
|
||
|
remove_filter( 'edd_prices_include_tax', '__return_false' );
|
||
|
|
||
|
$adjustment_id = edd_add_order_adjustment( array(
|
||
|
'object_id' => $this->ID,
|
||
|
'object_type' => 'order',
|
||
|
'type_key' => $fee_id,
|
||
|
'type' => 'fee',
|
||
|
'description' => $fee['label'],
|
||
|
'subtotal' => floatval( $fee['amount'] ),
|
||
|
'tax' => $tax,
|
||
|
'total' => floatval( $fee['amount'] ) + $tax
|
||
|
) );
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* As of 3.0, the cart details array is no longer used for payments; it's purpose is for backwards compatibility
|
||
|
* purposes only. Due to the way EDD_Payment, the cart_details array needs to be synchronized with the data
|
||
|
* stored in the database as it could be different to the other class vars in the instance of EDD_Payment.
|
||
|
*/
|
||
|
|
||
|
if ( isset( $meta_value['cart_details'] ) && ! empty( $meta_value['cart_details'] ) ) {
|
||
|
|
||
|
// Totals need to be updated based on cart details.
|
||
|
$new_tax = 0.00;
|
||
|
$new_subtotal = 0.00;
|
||
|
|
||
|
foreach ( $meta_value['cart_details'] as $key => $item ) {
|
||
|
$order_item_id = edd_get_order_items( array(
|
||
|
'number' => 1,
|
||
|
'fields' => 'ids',
|
||
|
'order_id' => $this->ID,
|
||
|
'product_id' => $item['id'],
|
||
|
'product_name' => $item['name'],
|
||
|
) );
|
||
|
|
||
|
$item['item_number']['options']['price_id'] = isset( $item['item_number']['options']['price_id'] ) && is_numeric( $item['item_number']['options']['price_id'] )
|
||
|
? absint( $item['item_number']['options']['price_id'] )
|
||
|
: null;
|
||
|
|
||
|
if ( is_array( $order_item_id ) && ! empty( $order_item_id ) ) {
|
||
|
$order_item_id = $order_item_id[0];
|
||
|
|
||
|
edd_update_order_item( $order_item_id, array(
|
||
|
'order_id' => $this->ID,
|
||
|
'product_id' => $item['id'],
|
||
|
'product_name' => $item['name'],
|
||
|
'price_id' => $item['item_number']['options']['price_id'],
|
||
|
'cart_index' => $key,
|
||
|
'quantity' => $item['quantity'],
|
||
|
'amount' => $item['item_price'],
|
||
|
'subtotal' => $item['subtotal'],
|
||
|
'discount' => $item['discount'],
|
||
|
'tax' => $item['tax'],
|
||
|
'total' => $item['price'],
|
||
|
) );
|
||
|
|
||
|
$new_subtotal = $item['subtotal'];
|
||
|
$new_tax += $item['tax'];
|
||
|
} else {
|
||
|
$order_item_id = edd_add_order_item( array(
|
||
|
'order_id' => $this->ID,
|
||
|
'product_id' => $item['id'],
|
||
|
'product_name' => $item['name'],
|
||
|
'price_id' => $item['item_number']['options']['price_id'],
|
||
|
'cart_index' => $key,
|
||
|
'quantity' => $item['quantity'],
|
||
|
'amount' => $item['item_price'],
|
||
|
'subtotal' => $item['subtotal'],
|
||
|
'discount' => $item['discount'],
|
||
|
'tax' => $item['tax'],
|
||
|
'total' => $item['price'],
|
||
|
'status' => ! empty( $item['status'] ) ? $item->status : $this->status,
|
||
|
) );
|
||
|
|
||
|
$new_tax += $item['tax'];
|
||
|
$new_subtotal += $item['subtotal'];
|
||
|
|
||
|
if ( isset( $item['fees'] ) && ! empty( $item['fees'] ) ) {
|
||
|
foreach ( $item['fees'] as $fee_id => $fee ) {
|
||
|
add_filter( 'edd_prices_include_tax', '__return_false' );
|
||
|
|
||
|
$tax = ( isset( $fee['no_tax'] ) && false === $fee['no_tax'] ) || $fee['amount'] < 0
|
||
|
? floatval( edd_calculate_tax( $fee['amount'] ) )
|
||
|
: 0.00;
|
||
|
|
||
|
remove_filter( 'edd_prices_include_tax', '__return_false' );
|
||
|
|
||
|
$adjustment_id = edd_add_order_adjustment( array(
|
||
|
'object_id' => $order_item_id,
|
||
|
'object_type' => 'order_item',
|
||
|
'type_key' => $fee_id,
|
||
|
'type' => 'fee',
|
||
|
'description' => $fee['label'],
|
||
|
'subtotal' => floatval( $fee['amount'] ),
|
||
|
'tax' => $tax,
|
||
|
'total' => floatval( $fee['amount'] ) + $tax,
|
||
|
) );
|
||
|
|
||
|
$new_tax += $tax;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// This is no longer stored in _edd_payment_meta.
|
||
|
$core_meta_keys = array( 'key', 'email', 'date', 'user_info', 'downloads', 'cart_details', 'quantity', 'discount', 'subtotal', 'tax', 'fees', 'currency' );
|
||
|
|
||
|
$meta_value = array_diff_key( $meta_value, array_flip( $core_meta_keys ) );
|
||
|
|
||
|
// If the above checks fall through, store anything else in a "payment_meta" meta key.
|
||
|
return edd_update_order_meta( $this->ID, 'payment_meta', $meta_value );
|
||
|
case '_edd_completed_date':
|
||
|
$meta_value = empty( $meta_value )
|
||
|
? null
|
||
|
: $meta_value;
|
||
|
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'date_completed' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_gateway':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'gateway' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_user_id':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'user_id' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_user_email':
|
||
|
case 'email':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'email' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_user_ip':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'ip' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_purchase_key':
|
||
|
case 'key':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'payment_key' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_mode':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'mode' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_tax_rate':
|
||
|
$tax_rate = $meta_value > 0 ? $meta_value : ( $meta_value * 100 );
|
||
|
edd_update_order_meta( $this->ID, 'tax_rate', $tax_rate, $prev_value );
|
||
|
return true;
|
||
|
case '_edd_payment_customer_id':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'customer_id' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_total':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'total' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_tax':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'tax' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_number':
|
||
|
edd_update_order( $this->ID, array(
|
||
|
'order_number' => $meta_value,
|
||
|
) );
|
||
|
return true;
|
||
|
case '_edd_payment_transaction_id':
|
||
|
case 'transaction_id':
|
||
|
$transaction_ids = array_values( edd_get_order_transactions( array(
|
||
|
'fields' => 'ids',
|
||
|
'number' => 1,
|
||
|
'object_id' => $this->ID,
|
||
|
'object_type' => 'order',
|
||
|
'orderby' => 'date_created',
|
||
|
'order' => 'ASC',
|
||
|
) ) );
|
||
|
|
||
|
if ( $transaction_ids ) {
|
||
|
$transaction_id = $transaction_ids[0];
|
||
|
|
||
|
return edd_update_order_transaction( $transaction_id, array(
|
||
|
'transaction_id' => $meta_value,
|
||
|
'gateway' => $this->gateway,
|
||
|
) );
|
||
|
} else {
|
||
|
return edd_add_order_transaction( array(
|
||
|
'object_id' => $this->ID,
|
||
|
'object_type' => 'order',
|
||
|
'transaction_id' => $meta_value,
|
||
|
'gateway' => $this->gateway,
|
||
|
'status' => 'complete',
|
||
|
'total' => $this->total,
|
||
|
) );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return edd_update_order_meta( $this->ID, $meta_key, $meta_value, $prev_value );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Add an item to the payment meta
|
||
|
*
|
||
|
* @since 2.8
|
||
|
*
|
||
|
* @param string $meta_key
|
||
|
* @param string $meta_value
|
||
|
* @param bool $unique
|
||
|
*
|
||
|
* @return bool|false|int
|
||
|
*/
|
||
|
public function add_meta( $meta_key = '', $meta_value = '', $unique = false ) {
|
||
|
if ( empty( $meta_key ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return edd_add_order_meta( $this->ID, $meta_key, $meta_value, $unique );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Delete an item from payment meta
|
||
|
*
|
||
|
* @since 2.8
|
||
|
*
|
||
|
* @param string $meta_key
|
||
|
* @param string $meta_value
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function delete_meta( $meta_key = '', $meta_value = '' ) {
|
||
|
if ( empty( $meta_key ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
return edd_delete_order_meta( $this->ID, $meta_key, $meta_value );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Determines if this payment is able to be resumed by the user.
|
||
|
*
|
||
|
* @since 2.7
|
||
|
*
|
||
|
* @return bool
|
||
|
*/
|
||
|
public function is_recoverable() {
|
||
|
return $this->order->is_recoverable();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the URL that a customer can use to resume a payment, or false if it's not recoverable.
|
||
|
*
|
||
|
* @since 2.7
|
||
|
*
|
||
|
* @return bool|string
|
||
|
*/
|
||
|
public function get_recovery_url() {
|
||
|
return $this->order->get_recovery_url();
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* When a payment is set to a status of 'refunded' process the necessary actions to reduce stats
|
||
|
*
|
||
|
* @since 2.5.7
|
||
|
* @access private
|
||
|
*/
|
||
|
private function process_refund() {
|
||
|
$process_refund = true;
|
||
|
|
||
|
// If the payment was not in publish or revoked status, don't decrement stats as they were never incremented
|
||
|
if ( ( 'complete' !== $this->old_status && 'revoked' !== $this->old_status ) || 'refunded' !== $this->status ) {
|
||
|
$process_refund = false;
|
||
|
}
|
||
|
|
||
|
// Allow extensions to filter for their own payment types, Example: Recurring Payments
|
||
|
$process_refund = apply_filters( 'edd_should_process_refund', $process_refund, $this );
|
||
|
|
||
|
if ( false === $process_refund ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
do_action( 'edd_pre_refund_payment', $this );
|
||
|
|
||
|
$decrease_store_earnings = apply_filters( 'edd_decrease_store_earnings_on_refund', true, $this );
|
||
|
$decrease_customer_value = apply_filters( 'edd_decrease_customer_value_on_refund', true, $this );
|
||
|
$decrease_purchase_count = apply_filters( 'edd_decrease_customer_purchase_count_on_refund', true, $this );
|
||
|
|
||
|
$this->maybe_alter_stats( $decrease_store_earnings, $decrease_customer_value, $decrease_purchase_count );
|
||
|
|
||
|
// Clear the This Month earnings (this_monththis_month is NOT a typo)
|
||
|
delete_transient( md5( 'edd_earnings_this_monththis_month' ) );
|
||
|
|
||
|
do_action( 'edd_post_refund_payment', $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Process when a payment is set to failed, decrement discount usages and other stats.
|
||
|
*
|
||
|
* @since 2.5.7
|
||
|
* @access private
|
||
|
*/
|
||
|
private function process_failure() {
|
||
|
$discounts = $this->discounts;
|
||
|
|
||
|
if ( 'none' === $discounts || empty( $discounts ) ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
if ( ! is_array( $discounts ) ) {
|
||
|
$discounts = array_map( 'trim', explode( ',', $discounts ) );
|
||
|
}
|
||
|
|
||
|
foreach ( $discounts as $discount ) {
|
||
|
edd_decrease_discount_usage( $discount );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Process when a payment moves to pending.
|
||
|
*
|
||
|
* @since 2.5.10
|
||
|
* @access private
|
||
|
*/
|
||
|
private function process_pending() {
|
||
|
$process_pending = true;
|
||
|
|
||
|
// If the payment was not in publish or revoked status, don't decrement stats as they were never incremented
|
||
|
if ( ( 'complete' !== $this->old_status && 'revoked' !== $this->old_status ) || ! $this->in_process() ) {
|
||
|
$process_pending = false;
|
||
|
}
|
||
|
|
||
|
// Allow extensions to filter for their own payment types, Example: Recurring Payments
|
||
|
$process_pending = apply_filters( 'edd_should_process_pending', $process_pending, $this );
|
||
|
|
||
|
if ( false === $process_pending ) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
$decrease_store_earnings = apply_filters( 'edd_decrease_store_earnings_on_pending', true, $this );
|
||
|
$decrease_customer_value = apply_filters( 'edd_decrease_customer_value_on_pending', true, $this );
|
||
|
$decrease_purchase_count = apply_filters( 'edd_decrease_customer_purchase_count_on_pending', true, $this );
|
||
|
|
||
|
$this->maybe_alter_stats( $decrease_store_earnings, $decrease_customer_value, $decrease_purchase_count );
|
||
|
|
||
|
$this->completed_date = false;
|
||
|
$this->update_meta( '_edd_completed_date', '' );
|
||
|
|
||
|
// Clear the This Month earnings (this_monththis_month is NOT a typo)
|
||
|
delete_transient( md5( 'edd_earnings_this_monththis_month' ) );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Used during the process of moving to refunded or pending, to decrement stats
|
||
|
*
|
||
|
* @since 2.5.10
|
||
|
* @access private
|
||
|
*
|
||
|
* @param bool $alter_store_earnings If the method should alter the store earnings
|
||
|
* @param bool $alter_customer_value If the method should reduce the customer value
|
||
|
* @param bool $alter_customer_purchase_count If the method should reduce the customer's purchase count
|
||
|
*/
|
||
|
private function maybe_alter_stats( $alter_store_earnings, $alter_customer_value, $alter_customer_purchase_count ) {
|
||
|
if ( edd_undo_purchase( false, $this->ID ) ) {
|
||
|
$this->status = 'refunded';
|
||
|
$this->post_status = 'refunded';
|
||
|
|
||
|
$statuses = edd_get_payment_statuses();
|
||
|
$this->status_nicename = array_key_exists( 'refunded', $statuses )
|
||
|
? $statuses['refunded']
|
||
|
: ucfirst( 'refunded' );
|
||
|
}
|
||
|
|
||
|
// Decrease store earnings
|
||
|
if ( true === $alter_store_earnings ) {
|
||
|
edd_decrease_total_earnings( $this->total );
|
||
|
}
|
||
|
|
||
|
// Decrement the stats for the customer
|
||
|
if ( ! empty( $this->customer_id ) ) {
|
||
|
$customer = new EDD_Customer( $this->customer_id );
|
||
|
|
||
|
if ( ! empty( $alter_customer_value || $alter_customer_purchase_count ) ) {
|
||
|
$customer->recalculate_stats();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Delete sales logs for this purchase
|
||
|
*
|
||
|
* @since 2.5.10
|
||
|
* @deprecated Deprecated since 3.0 as sales logs are no longer used.
|
||
|
*/
|
||
|
private function delete_sales_logs() {
|
||
|
_doing_it_wrong( __FUNCTION__, 'Sales logs are deprecated and are no longer used.', 'EDD 3.0' );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup functions only, these are not to be used by developers.
|
||
|
* These functions exist only to allow the setup routine to be backwards compatible with our old
|
||
|
* helper functions.
|
||
|
*
|
||
|
* These will run whenever setup_payment is called, which should only be called once.
|
||
|
* To update an attribute, update it directly instead of re-running the setup routine
|
||
|
*/
|
||
|
|
||
|
/**
|
||
|
* Setup the payment completed date.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Updated to use the new custom tables.
|
||
|
*
|
||
|
* @return string The date the payment was completed.
|
||
|
*/
|
||
|
private function setup_completed_date() {
|
||
|
/** @var EDD\Orders\Order $order */
|
||
|
$order = $this->_shim_edd_get_order( $this->ID );
|
||
|
|
||
|
if ( 'pending' === $order->status || 'preapproved' === $order->status || 'processing' === $order->status ) {
|
||
|
return false; // This payment was never completed
|
||
|
}
|
||
|
|
||
|
return $order->date_completed ? $order->date_completed : '';
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the payment total.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @return float Payment total.
|
||
|
*/
|
||
|
private function setup_total() {
|
||
|
$amount = $this->get_meta( '_edd_payment_total', true );
|
||
|
|
||
|
if ( empty( $amount ) && '0.00' !== $amount ) {
|
||
|
$meta = $this->get_meta( '_edd_payment_meta', true );
|
||
|
$meta = maybe_unserialize( $meta );
|
||
|
|
||
|
if ( isset( $meta['amount'] ) ) {
|
||
|
$amount = $meta['amount'];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $amount;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the payment tax rate.
|
||
|
*
|
||
|
* @since 2.7
|
||
|
*
|
||
|
* @return float Tax rate for the payment.
|
||
|
*/
|
||
|
private function setup_tax_rate() {
|
||
|
$tax_rate = $this->order->get_tax_rate();
|
||
|
|
||
|
if ( ! empty( $tax_rate ) && $tax_rate > 1 ) {
|
||
|
$tax_rate = $tax_rate / 100;
|
||
|
}
|
||
|
|
||
|
return $tax_rate;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the total fee amount applied to the payment.
|
||
|
*
|
||
|
* @since 2.5.10
|
||
|
*
|
||
|
* @return float Total fee amount applied to the payment.
|
||
|
*/
|
||
|
private function setup_fees_total() {
|
||
|
$fees_total = array_reduce( $this->fees, function( $carry, $item ) {
|
||
|
$carry += (float) $item['amount'];
|
||
|
|
||
|
return $carry;
|
||
|
}, (float) 0.00 );
|
||
|
|
||
|
return $fees_total;
|
||
|
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the payment subtotal.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @return float Payment subtotal.
|
||
|
*/
|
||
|
private function setup_subtotal() {
|
||
|
$subtotal = 0;
|
||
|
$cart_details = $this->cart_details;
|
||
|
|
||
|
if ( is_array( $cart_details ) ) {
|
||
|
foreach ( $cart_details as $item ) {
|
||
|
if ( isset( $item['subtotal'] ) ) {
|
||
|
$subtotal += $item['subtotal'];
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
$subtotal = $this->total;
|
||
|
$tax = edd_use_taxes() ? $this->tax : 0;
|
||
|
$subtotal -= $tax;
|
||
|
}
|
||
|
|
||
|
return $subtotal;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the payments discount codes.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @return string Discount codes on this payment.
|
||
|
*/
|
||
|
private function setup_discounts() {
|
||
|
$discounts = array();
|
||
|
|
||
|
$order_discounts = $this->order->get_discounts();
|
||
|
|
||
|
foreach ( $order_discounts as $discount ) {
|
||
|
$discounts[] = $discount->description;
|
||
|
}
|
||
|
|
||
|
$discounts = implode( ', ', $discounts );
|
||
|
|
||
|
return $discounts;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the currency code
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @return string The currency for the payment.
|
||
|
*/
|
||
|
private function setup_currency() {
|
||
|
$currency = $this->order->currency;
|
||
|
|
||
|
return ! empty( $currency )
|
||
|
? $currency
|
||
|
: apply_filters( 'edd_payment_currency_default', edd_get_currency(), $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup any fees associated with the payment.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @return array Payment fees.
|
||
|
*/
|
||
|
private function setup_fees() {
|
||
|
$fees = array();
|
||
|
|
||
|
if ( $this->order->get_fees() ) {
|
||
|
/*
|
||
|
* Build up an array of order item IDs with values set to their respective download/price IDs.
|
||
|
* This is so we can easily get that information when configuring order item fees.
|
||
|
*/
|
||
|
$order_items = array();
|
||
|
foreach ( $this->order->get_items() as $order_item ) {
|
||
|
/**
|
||
|
* @var \EDD\Orders\Order_Item $order_item
|
||
|
*/
|
||
|
$order_items[ intval( $order_item->id ) ] = array(
|
||
|
'download_id' => $order_item->product_id,
|
||
|
'price_id' => $order_item->price_id
|
||
|
);
|
||
|
}
|
||
|
|
||
|
foreach ( $this->order->get_fees() as $order_fee ) {
|
||
|
/**
|
||
|
* @var \EDD\Orders\Order_Adjustment $order_fee
|
||
|
*/
|
||
|
|
||
|
$download_id = 0;
|
||
|
$price_id = null;
|
||
|
|
||
|
if ( 'order_item' === $order_fee->object_type && array_key_exists( intval( $order_fee->object_id ), $order_items ) ) {
|
||
|
$download_id = $order_items[ intval( $order_fee->object_id ) ]['download_id'];
|
||
|
$price_id = $order_items[ intval( $order_fee->object_id ) ]['price_id'];
|
||
|
}
|
||
|
|
||
|
$no_tax = (bool) 0.00 === $order_fee->tax;
|
||
|
$id = is_null( $order_fee->type_key ) ? $order_fee->id : $order_fee->type_key;
|
||
|
if ( array_key_exists( $id, $fees ) ) {
|
||
|
$id .= '_2';
|
||
|
}
|
||
|
|
||
|
if ( $id != $order_fee->type_key ) {
|
||
|
/*
|
||
|
* We run an update here because if we don't, then we'll send back a key of `23_2` when in the
|
||
|
* DB it's actually `null`, and if this value gets updated via the payment meta array, it
|
||
|
* will actually add a brand *new* fee instead of updating the existing one.
|
||
|
*
|
||
|
* @link https://github.com/easydigitaldownloads/easy-digital-downloads/issues/8412
|
||
|
*/
|
||
|
edd_update_order_adjustment( $order_fee->id, array(
|
||
|
'type_key' => $id
|
||
|
) );
|
||
|
}
|
||
|
|
||
|
$fees[ $id ] = array(
|
||
|
'amount' => $order_fee->subtotal,
|
||
|
'label' => $order_fee->description,
|
||
|
'no_tax' => $no_tax,
|
||
|
'type' => 'fee',
|
||
|
'price_id' => $price_id,
|
||
|
'download_id' => $download_id,
|
||
|
);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return $fees;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the transaction ID.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @return string The transaction ID for the payment.
|
||
|
*/
|
||
|
private function setup_transaction_id() {
|
||
|
$transaction_id = $this->get_meta( '_edd_payment_transaction_id', true );
|
||
|
|
||
|
if ( empty( $transaction_id ) || (int) $transaction_id === (int) $this->ID ) {
|
||
|
$gateway = $this->gateway;
|
||
|
$transaction_id = apply_filters( 'edd_get_payment_transaction_id-' . $gateway, $this->ID );
|
||
|
}
|
||
|
|
||
|
return $transaction_id;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the User ID associated with the purchase.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @return int User ID.
|
||
|
*/
|
||
|
private function setup_user_id() {
|
||
|
$user_id = $this->get_meta( '_edd_payment_user_id', true );
|
||
|
$customer = new EDD_Customer( $this->customer_id );
|
||
|
|
||
|
// Make sure it exists, and that it matches that of the associated customer record
|
||
|
if ( ! empty( $customer->user_id ) && ( empty( $user_id ) || (int) $user_id !== (int) $customer->user_id ) ) {
|
||
|
$user_id = $customer->user_id;
|
||
|
|
||
|
// Backfill the user ID, or reset it to be correct in the event of data corruption
|
||
|
$this->update_meta( '_edd_payment_user_id', $user_id );
|
||
|
}
|
||
|
|
||
|
return $user_id;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the email address for the purchase
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @return string The email address for the payment
|
||
|
*/
|
||
|
private function setup_email() {
|
||
|
$email = $this->order->email;
|
||
|
|
||
|
if ( empty( $email ) ) {
|
||
|
$email = EDD()->customers->get_column( 'email', $this->customer_id );
|
||
|
}
|
||
|
|
||
|
return $email;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the user info.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @return array The user info associated with the payment.
|
||
|
*/
|
||
|
private function setup_user_info() {
|
||
|
$order_address = $this->order->get_address();
|
||
|
|
||
|
$user_info = array(
|
||
|
'id' => $this->user_id,
|
||
|
'first_name' => $order_address->first_name,
|
||
|
'last_name' => $order_address->last_name,
|
||
|
'discount' => $this->discounts,
|
||
|
);
|
||
|
|
||
|
// Ensure email index is in the old user info array
|
||
|
if ( empty( $user_info['email'] ) ) {
|
||
|
$user_info['email'] = $this->email;
|
||
|
}
|
||
|
|
||
|
if ( empty( $user_info ) ) {
|
||
|
// Get the customer, but only if it's been created
|
||
|
$customer = new EDD_Customer( $this->customer_id );
|
||
|
|
||
|
if ( $customer->id > 0 ) {
|
||
|
$name = explode( ' ', $customer->name, 2 );
|
||
|
$user_info = array(
|
||
|
'first_name' => $name[0],
|
||
|
'last_name' => $name[1],
|
||
|
'email' => $customer->email,
|
||
|
'discount' => 'none',
|
||
|
);
|
||
|
}
|
||
|
} else {
|
||
|
// Get the customer, but only if it's been created
|
||
|
$customer = new EDD_Customer( $this->customer_id );
|
||
|
if ( $customer->id > 0 ) {
|
||
|
foreach ( $user_info as $key => $value ) {
|
||
|
if ( ! empty( $value ) ) {
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
switch ( $key ) {
|
||
|
case 'first_name':
|
||
|
$name = explode( ' ', $customer->name, 2 );
|
||
|
|
||
|
$user_info[ $key ] = $name[0];
|
||
|
break;
|
||
|
|
||
|
case 'last_name':
|
||
|
$name = explode( ' ', $customer->name, 2 );
|
||
|
$last_name = ! empty( $name[1] ) ? $name[1] : '';
|
||
|
|
||
|
$user_info[ $key ] = $last_name;
|
||
|
break;
|
||
|
|
||
|
case 'email':
|
||
|
$user_info[ $key ] = $customer->email;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$country = $order_address->country;
|
||
|
|
||
|
// Add address to array if one exists.
|
||
|
if ( ! empty( $country ) ) {
|
||
|
$user_info['address'] = array(
|
||
|
'line1' => $order_address->address,
|
||
|
'line2' => $order_address->address2,
|
||
|
'city' => $order_address->city,
|
||
|
'state' => $order_address->region,
|
||
|
'country' => $country,
|
||
|
'zip' => $order_address->postal_code,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
// Check for old `user_info` meta which may still exist.
|
||
|
$old_meta = edd_get_order_meta( $this->ID, 'payment_meta', true );
|
||
|
if ( ! empty( $old_meta['user_info'] ) ) {
|
||
|
$user_info = array_merge( $user_info, $old_meta['user_info'] );
|
||
|
}
|
||
|
|
||
|
return $user_info;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the address for the payment.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @return array The address information for the payment.
|
||
|
*/
|
||
|
private function setup_address() {
|
||
|
$address = ! empty( $this->user_info['address'] ) ? $this->user_info['address'] : array();
|
||
|
$defaults = array( 'line1' => '', 'line2' => '', 'city' => '', 'country' => '', 'state' => '', 'zip' => '' );
|
||
|
|
||
|
$address = wp_parse_args( $address, $defaults );
|
||
|
|
||
|
return $address;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the payment number.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Refactor to use EDD\Orders\Order.
|
||
|
*
|
||
|
* @return int|string Integer by default, or string if sequential order numbers is enabled.
|
||
|
*/
|
||
|
private function setup_payment_number() {
|
||
|
return $this->order->order_number;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the cart details
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @since 3.0 Refactored as cart_details is no longer used and this is here for backwards compatibility purposes.
|
||
|
*
|
||
|
* @return array Cart details of an order.
|
||
|
*/
|
||
|
private function setup_cart_details() {
|
||
|
$order_items = $this->order->items;
|
||
|
|
||
|
$cart_details = array();
|
||
|
|
||
|
foreach ( $order_items as $item ) {
|
||
|
/** @var EDD\Orders\Order_Item $item */
|
||
|
|
||
|
$item_fees = array();
|
||
|
|
||
|
foreach ( $item->fees as $key => $item_fee ) {
|
||
|
/** @var EDD\Orders\Order_Adjustment $item_fee */
|
||
|
|
||
|
$download_id = $item->product_id;
|
||
|
$price_id = $item->price_id;
|
||
|
$no_tax = (bool) 0.00 === $item_fee->tax;
|
||
|
$id = is_null( $item_fee->type_key ) ? $item_fee->id : $item_fee->type_key;
|
||
|
if ( array_key_exists( $id, $item_fees ) ) {
|
||
|
$id .= '_2';
|
||
|
}
|
||
|
|
||
|
$item_fees[ $id ] = array(
|
||
|
'amount' => $item_fee->amount,
|
||
|
'label' => $item_fee->description,
|
||
|
'no_tax' => $no_tax ? $no_tax : false,
|
||
|
'type' => 'fee',
|
||
|
'price_id' => $price_id ? $price_id : null,
|
||
|
'download_id' => 0,
|
||
|
);
|
||
|
|
||
|
if ( $download_id ) {
|
||
|
$item_fees[ $id ]['download_id'] = $download_id;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$item_options = array(
|
||
|
'quantity' => $item->quantity,
|
||
|
'price_id' => $item->price_id,
|
||
|
);
|
||
|
|
||
|
/*
|
||
|
* For backwards compatibility from pre-3.0: add in order item meta prefixed with `_option_`.
|
||
|
* While saving, we've migrated these values to order item meta, but people may still be looking
|
||
|
* for them in this cart details array, so we need to fill them back in.
|
||
|
*/
|
||
|
$order_item_meta = edd_get_order_item_meta( $item->id );
|
||
|
if ( ! empty( $order_item_meta ) ) {
|
||
|
foreach ( $order_item_meta as $item_meta_key => $item_meta_value ) {
|
||
|
if ( '_option_' === substr( $item_meta_key, 0, 8 ) && isset( $item_meta_value[0] ) ) {
|
||
|
$item_options[ str_replace( '_option_', '', $item_meta_key ) ] = $item_meta_value[0];
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
$cart_details[ $item->cart_index ] = array(
|
||
|
'name' => $item->product_name,
|
||
|
'id' => $item->product_id,
|
||
|
'item_number' => array(
|
||
|
'id' => $item->product_id,
|
||
|
'quantity' => $item->quantity,
|
||
|
'options' => $item_options,
|
||
|
),
|
||
|
'item_price' => $item->amount,
|
||
|
'quantity' => $item->quantity,
|
||
|
'discount' => $item->discount,
|
||
|
'subtotal' => $item->subtotal,
|
||
|
'tax' => $item->tax,
|
||
|
'fees' => $item_fees,
|
||
|
'price' => $item->total,
|
||
|
'order_item_id' => $item->id,
|
||
|
);
|
||
|
}
|
||
|
|
||
|
return $cart_details;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the downloads array.
|
||
|
*
|
||
|
* @since 2.5
|
||
|
*
|
||
|
* @internal This exists for backwards compatibility purposes.
|
||
|
*
|
||
|
* @return array Downloads associated with this payment.
|
||
|
*/
|
||
|
private function setup_downloads() {
|
||
|
$order_items = $this->order->items;
|
||
|
|
||
|
$downloads = array();
|
||
|
|
||
|
foreach ( $order_items as $item ) {
|
||
|
/** @var EDD\Orders\Order_Item $item */
|
||
|
|
||
|
$downloads[] = array(
|
||
|
'id' => $item->product_id,
|
||
|
'quantity' => $item->quantity,
|
||
|
'options' => array(
|
||
|
'quantity' => $item->quantity,
|
||
|
'price_id' => $item->price_id,
|
||
|
)
|
||
|
);
|
||
|
}
|
||
|
|
||
|
return $downloads;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Setup the Unlimited downloads setting
|
||
|
*
|
||
|
* @since 2.5
|
||
|
* @return bool If this payment has unlimited downloads
|
||
|
*/
|
||
|
private function setup_has_unlimited() {
|
||
|
$unlimited = (bool) $this->order->has_unlimited_downloads();
|
||
|
|
||
|
return $unlimited;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Converts this object into an array for special cases.
|
||
|
*
|
||
|
* @return array The payment object as an array.
|
||
|
*/
|
||
|
public function array_convert() {
|
||
|
return get_object_vars( $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment cart details.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return array Cart details array.
|
||
|
*/
|
||
|
private function get_cart_details() {
|
||
|
return apply_filters( 'edd_payment_cart_details', $this->cart_details, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment completion date
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
* @since 3.0 Updated for backwards compatibility.
|
||
|
*
|
||
|
* @return string Date payment was completed.
|
||
|
*/
|
||
|
private function get_completed_date() {
|
||
|
if ( is_null( $this->completed_date ) ) {
|
||
|
$date = false;
|
||
|
} else {
|
||
|
$date = $this->completed_date;
|
||
|
}
|
||
|
|
||
|
return apply_filters( 'edd_payment_completed_date', $date, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment tax.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return float Payment tax
|
||
|
*/
|
||
|
private function get_tax() {
|
||
|
return apply_filters( 'edd_get_payment_tax', $this->tax, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment subtotal.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return float Payment subtotal.
|
||
|
*/
|
||
|
private function get_subtotal() {
|
||
|
return apply_filters( 'edd_get_payment_subtotal', $this->subtotal, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment discounts.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return array Discount codes on payment.
|
||
|
*/
|
||
|
private function get_discounts() {
|
||
|
return apply_filters( 'edd_payment_discounts', $this->discounts, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Return the discounted amount of the payment.
|
||
|
*
|
||
|
* @since 2.8.7
|
||
|
*
|
||
|
* @return float Discounted amount.
|
||
|
*/
|
||
|
private function get_discounted_amount() {
|
||
|
return floatval( apply_filters( 'edd_payment_discounted_amount', $this->order->discount, $this ) );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment currency.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return string Payment currency code.
|
||
|
*/
|
||
|
private function get_currency() {
|
||
|
return apply_filters( 'edd_payment_currency_code', $this->currency, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment gateway.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return string Payment gateway used.
|
||
|
*/
|
||
|
private function get_gateway() {
|
||
|
return apply_filters( 'edd_payment_gateway', $this->gateway, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment transaction ID.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return string Transaction ID from merchant processor.
|
||
|
*/
|
||
|
private function get_transaction_id() {
|
||
|
return apply_filters( 'edd_get_payment_transaction_id', $this->transaction_id, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment IP.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return string Payment IP address.
|
||
|
*/
|
||
|
private function get_ip() {
|
||
|
return apply_filters( 'edd_payment_user_ip', $this->ip, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment customer ID.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return int Payment customer ID.
|
||
|
*/
|
||
|
private function get_customer_id() {
|
||
|
return apply_filters( 'edd_payment_customer_id', $this->customer_id, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment user ID.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return int Payment user ID.
|
||
|
*/
|
||
|
private function get_user_id() {
|
||
|
return apply_filters( 'edd_payment_user_id', $this->user_id, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment email.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return string Payment customer email.
|
||
|
*/
|
||
|
private function get_email() {
|
||
|
return apply_filters( 'edd_payment_user_email', $this->email, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment user info.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return array Payment user info.
|
||
|
*/
|
||
|
private function get_user_info() {
|
||
|
return apply_filters( 'edd_payment_meta_user_info', $this->user_info, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment billing address.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return array Payment billing address.
|
||
|
*/
|
||
|
private function get_address() {
|
||
|
return apply_filters( 'edd_payment_address', $this->address, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment key.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return string Payment key.
|
||
|
*/
|
||
|
private function get_key() {
|
||
|
return apply_filters( 'edd_payment_key', $this->key, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve payment number.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return int|string Payment number.
|
||
|
*/
|
||
|
private function get_number() {
|
||
|
return $this->order instanceof EDD\Orders\Order ? $this->order->get_number() : $this->ID;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve downloads on payment.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return array Payment downloads.
|
||
|
*/
|
||
|
private function get_downloads() {
|
||
|
return apply_filters( 'edd_payment_meta_downloads', $this->downloads, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Retrieve unlimited file downloads status.
|
||
|
*
|
||
|
* @since 2.5.1
|
||
|
*
|
||
|
* @return bool True if unlimited downloads are enabled, false otherwise.
|
||
|
*/
|
||
|
private function get_unlimited() {
|
||
|
return apply_filters( 'edd_payment_unlimited_downloads', $this->unlimited, $this->ID, $this );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Easily determine if the payment is in a status of pending some action. Processing is specifically used for
|
||
|
* eChecks.
|
||
|
*
|
||
|
* @since 2.7
|
||
|
* @return bool
|
||
|
*/
|
||
|
private function in_process() {
|
||
|
$in_process_statuses = array( 'pending', 'processing' );
|
||
|
|
||
|
return in_array( $this->status, $in_process_statuses, true );
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Determines if a customer needs to be created given the current payment details.
|
||
|
*
|
||
|
* @since 2.8.4
|
||
|
*
|
||
|
* @return EDD_Customer The customer object of the existing customer or new customer.
|
||
|
*/
|
||
|
private function maybe_create_customer() {
|
||
|
$customer = new stdClass();
|
||
|
|
||
|
if ( did_action( 'edd_pre_process_purchase' ) && is_user_logged_in() ) {
|
||
|
$customer = new EDD_customer( get_current_user_id(), true );
|
||
|
|
||
|
// Customer is logged in but used a different email to purchase with so assign to their customer record
|
||
|
if ( ! empty( $customer->id ) && $this->email !== $customer->email ) {
|
||
|
$customer->add_email( $this->email );
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if ( empty( $customer->id ) ) {
|
||
|
$customer = new EDD_Customer( $this->email );
|
||
|
}
|
||
|
|
||
|
if ( empty( $customer->id ) ) {
|
||
|
if ( empty( $this->first_name ) && empty( $this->last_name ) ) {
|
||
|
$name = $this->email;
|
||
|
} else {
|
||
|
$name = $this->first_name . ' ' . $this->last_name;
|
||
|
}
|
||
|
|
||
|
$customer_data = array(
|
||
|
'name' => $name,
|
||
|
'email' => $this->email,
|
||
|
'user_id' => $this->user_id,
|
||
|
);
|
||
|
|
||
|
$customer->create( $customer_data );
|
||
|
}
|
||
|
|
||
|
return $customer;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Sets up a payment object from a post.
|
||
|
* This is only intended to be used when a 3.0 migration is in process and the
|
||
|
* new order object is not yet available.
|
||
|
*
|
||
|
* @todo deprecate in 3.1
|
||
|
*
|
||
|
* @since 3.0
|
||
|
* @param int $payment_id
|
||
|
* @return bool
|
||
|
*/
|
||
|
private function _setup_compat_payment( $payment_id ) {
|
||
|
$payment = get_post( $payment_id );
|
||
|
|
||
|
if ( ! $payment || is_wp_error( $payment ) ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
if ( 'edd_payment' !== $payment->post_type ) {
|
||
|
return false;
|
||
|
}
|
||
|
|
||
|
// Set the compatibility property to true.
|
||
|
$this->is_edd_payment = true;
|
||
|
|
||
|
// Allow extensions to perform actions before the payment is loaded
|
||
|
do_action( 'edd_pre_setup_payment', $this, $payment_id );
|
||
|
|
||
|
// Primary Identifier
|
||
|
$this->ID = absint( $payment_id );
|
||
|
|
||
|
// Protected ID that can never be changed
|
||
|
$this->_ID = absint( $payment_id );
|
||
|
|
||
|
include_once EDD_PLUGIN_DIR . 'includes/compat/class-edd-payment-compat.php';
|
||
|
$payment_compat = new EDD_Payment_Compat( $this->ID );
|
||
|
|
||
|
// We have a payment; get the generic payment_meta item to reduce calls to it
|
||
|
$this->payment_meta = $payment_compat->payment_meta;
|
||
|
|
||
|
// Status and Dates
|
||
|
$this->date = $payment->post_date;
|
||
|
$this->completed_date = $payment_compat->completed_date;
|
||
|
$this->status = $payment_compat->status;
|
||
|
$this->post_status = $this->status;
|
||
|
$this->mode = $payment_compat->mode;
|
||
|
$this->parent_payment = $payment->post_parent;
|
||
|
|
||
|
$all_payment_statuses = edd_get_payment_statuses();
|
||
|
$this->status_nicename = array_key_exists( $this->status, $all_payment_statuses ) ? $all_payment_statuses[ $this->status ] : ucfirst( $this->status );
|
||
|
|
||
|
// Items
|
||
|
$this->fees = $payment_compat->fees;
|
||
|
$this->cart_details = $payment_compat->cart_details;
|
||
|
$this->downloads = $payment_compat->downloads;
|
||
|
|
||
|
// Currency Based
|
||
|
$this->total = $payment_compat->total;
|
||
|
$this->tax = $payment_compat->tax;
|
||
|
$this->tax_rate = $payment_compat->tax_rate;
|
||
|
$this->fees_total = $payment_compat->fees_total;
|
||
|
$this->subtotal = $payment_compat->subtotal;
|
||
|
$this->currency = $payment_compat->currency;
|
||
|
|
||
|
// Gateway based
|
||
|
$this->gateway = $payment_compat->gateway;
|
||
|
$this->transaction_id = $payment_compat->transaction_id;
|
||
|
|
||
|
// User based
|
||
|
$this->ip = $payment_compat->ip;
|
||
|
$this->customer_id = $payment_compat->customer_id;
|
||
|
$this->user_id = $payment_compat->user_id;
|
||
|
$this->email = $payment_compat->email;
|
||
|
$this->user_info = $payment_compat->user_info;
|
||
|
$this->address = $payment_compat->address;
|
||
|
$this->discounts = $this->user_info['discount'];
|
||
|
$this->first_name = $this->user_info['first_name'];
|
||
|
$this->last_name = $this->user_info['last_name'];
|
||
|
|
||
|
// Other Identifiers
|
||
|
$this->key = $payment_compat->key;
|
||
|
$this->number = $payment_compat->number;
|
||
|
|
||
|
// Additional Attributes
|
||
|
$this->has_unlimited_downloads = $payment_compat->has_unlimited_downloads;
|
||
|
$this->order = $payment_compat->order;
|
||
|
|
||
|
// Allow extensions to add items to this object via hook
|
||
|
do_action( 'edd_setup_payment', $this, $payment_id );
|
||
|
|
||
|
return true;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Gets the order from the database.
|
||
|
* This is a duplicate of edd_get_order, but is defined separately here
|
||
|
* for pending migration purposes.
|
||
|
*
|
||
|
* @todo deprecate in 3.1
|
||
|
*
|
||
|
* @param int $order_id
|
||
|
* @return false|EDD\Orders\Order
|
||
|
*/
|
||
|
private function _shim_edd_get_order( $order_id ) {
|
||
|
$orders = new EDD\Database\Queries\Order();
|
||
|
|
||
|
// Return order
|
||
|
return $orders->get_item( $order_id );
|
||
|
}
|
||
|
}
|