2011-08-09 15:16:18 +00:00
< ? php
/**
* Checkout
2012-08-10 11:15:32 +00:00
*
2011-08-10 17:11:11 +00:00
* The WooCommerce checkout class handles the checkout process , collecting user data and processing the payment .
2011-08-09 15:16:18 +00:00
*
2012-08-14 19:42:38 +00:00
* @ class WC_Cart
* @ version 1.6 . 4
2012-08-14 22:43:48 +00:00
* @ package WooCommerce / Classes
2013-02-20 17:14:46 +00:00
* @ category Class
2012-08-14 19:42:38 +00:00
* @ author WooThemes
2011-08-09 15:16:18 +00:00
*/
2012-01-27 16:38:39 +00:00
class WC_Checkout {
2012-08-10 11:15:32 +00:00
2012-08-15 17:08:42 +00:00
/** @var array Array of posted form data. */
2012-12-14 21:27:29 +00:00
public $posted ;
2012-08-14 19:42:38 +00:00
2012-08-15 17:08:42 +00:00
/** @var array Array of fields to display on the checkout. */
2012-12-14 21:27:29 +00:00
public $checkout_fields ;
2012-08-14 19:42:38 +00:00
2012-08-15 17:08:42 +00:00
/** @var bool Whether or not the user must create an account to checkout. */
2012-12-14 21:27:29 +00:00
public $must_create_account ;
2012-08-14 19:42:38 +00:00
2012-12-10 12:34:55 +00:00
/** @var bool Whether or not signups are allowed. */
2012-12-14 21:27:29 +00:00
public $enable_signup ;
2012-12-10 12:34:55 +00:00
2012-12-27 19:24:33 +00:00
/** @var object The shipping method being used. */
private $shipping_method ;
/** @var array The payment gateway being used. */
private $payment_method ;
2013-02-15 21:18:46 +00:00
/** @var int ID of customer. */
private $customer_id ;
2012-08-14 19:42:38 +00:00
/**
2013-03-03 17:07:31 +00:00
* Constructor for the checkout class . Hooks in methods and defines checkout fields .
2012-08-14 19:42:38 +00:00
*
* @ access public
* @ return void
*/
2012-12-14 21:27:29 +00:00
public function __construct () {
2011-12-15 23:24:02 +00:00
global $woocommerce ;
2012-08-10 11:15:32 +00:00
2012-12-15 11:53:32 +00:00
add_action ( 'woocommerce_checkout_process' , array ( $this , 'checkout_process' ) );
add_action ( 'woocommerce_checkout_billing' , array ( $this , 'checkout_form_billing' ) );
add_action ( 'woocommerce_checkout_shipping' , array ( $this , 'checkout_form_shipping' ) );
2012-08-10 11:15:32 +00:00
2012-12-10 12:34:55 +00:00
$this -> enable_signup = get_option ( 'woocommerce_enable_signup_and_login_from_checkout' ) == 'yes' ? true : false ;
$this -> enable_guest_checkout = get_option ( 'woocommerce_enable_guest_checkout' ) == 'yes' ? true : false ;
$this -> must_create_account = $this -> enable_guest_checkout || is_user_logged_in () ? false : true ;
2011-12-21 16:03:45 +00:00
// Define all Checkout fields
$this -> checkout_fields [ 'billing' ] = $woocommerce -> countries -> get_address_fields ( $this -> get_value ( 'billing_country' ), 'billing_' );
$this -> checkout_fields [ 'shipping' ] = $woocommerce -> countries -> get_address_fields ( $this -> get_value ( 'shipping_country' ), 'shipping_' );
2012-08-10 11:15:32 +00:00
2013-06-03 15:09:04 +00:00
if ( get_option ( 'woocommerce_registration_generate_username' ) == 'no' ) {
2012-08-10 11:15:32 +00:00
$this -> checkout_fields [ 'account' ][ 'account_username' ] = array (
'type' => 'text' ,
'label' => __ ( 'Account username' , 'woocommerce' ),
'placeholder' => _x ( 'Username' , 'placeholder' , 'woocommerce' )
2011-12-21 16:03:45 +00:00
);
2012-08-10 11:15:32 +00:00
}
2013-06-03 15:09:04 +00:00
if ( get_option ( 'woocommerce_registration_generate_password' ) == 'no' ) {
$this -> checkout_fields [ 'account' ][ 'account_password' ] = array (
'type' => 'password' ,
'label' => __ ( 'Account password' , 'woocommerce' ),
'placeholder' => _x ( 'Password' , 'placeholder' , 'woocommerce' )
);
}
2012-08-10 11:15:32 +00:00
2011-12-21 16:03:45 +00:00
$this -> checkout_fields [ 'order' ] = array (
2012-08-10 11:15:32 +00:00
'order_comments' => array (
'type' => 'textarea' ,
'class' => array ( 'notes' ),
2012-10-16 09:45:33 +00:00
'label' => __ ( 'Order Notes' , 'woocommerce' ),
2012-08-10 11:15:32 +00:00
'placeholder' => _x ( 'Notes about your order, e.g. special notes for delivery.' , 'placeholder' , 'woocommerce' )
2011-12-21 16:03:45 +00:00
)
);
2012-12-10 12:34:55 +00:00
$this -> checkout_fields = apply_filters ( 'woocommerce_checkout_fields' , $this -> checkout_fields );
do_action ( 'woocommerce_checkout_init' , $this );
2011-08-09 15:16:18 +00:00
}
2012-08-10 11:15:32 +00:00
2012-08-14 19:42:38 +00:00
/**
* Checkout process
*
* @ access public
* @ return void
*/
2012-12-14 21:27:29 +00:00
public function checkout_process () {
2012-02-12 11:36:33 +00:00
// When we process the checkout, lets ensure cart items are rechecked to prevent checkout
do_action ( 'woocommerce_check_cart_items' );
}
2012-08-10 11:15:32 +00:00
2012-08-14 19:42:38 +00:00
/**
* Output the billing information form
*
* @ access public
* @ return void
*/
2012-12-14 21:27:29 +00:00
public function checkout_form_billing () {
2012-02-03 16:17:35 +00:00
woocommerce_get_template ( 'checkout/form-billing.php' , array ( 'checkout' => $this ) );
2011-08-09 15:16:18 +00:00
}
2012-08-10 11:15:32 +00:00
2012-08-14 19:42:38 +00:00
/**
* Output the shipping information form
*
* @ access public
* @ return void
*/
2012-12-14 21:27:29 +00:00
public function checkout_form_shipping () {
2012-02-03 16:17:35 +00:00
woocommerce_get_template ( 'checkout/form-shipping.php' , array ( 'checkout' => $this ) );
2011-08-09 15:16:18 +00:00
}
2012-08-14 19:42:38 +00:00
2012-12-12 21:14:19 +00:00
/**
* create_order function .
*
* @ access public
* @ return void
*/
2012-12-14 21:27:29 +00:00
public function create_order () {
2012-12-12 21:14:19 +00:00
global $woocommerce , $wpdb ;
2013-02-13 09:38:54 +00:00
// Give plugins the opportunity to create an order themselves
$order_id = apply_filters ( 'woocommerce_create_order' , null , $this );
if ( is_numeric ( $order_id ) )
return $order_id ;
// Create Order (send cart variable so we can record items and reduce inventory). Only create if this is a new order, not if the payment was rejected.
2012-12-12 21:14:19 +00:00
$order_data = apply_filters ( 'woocommerce_new_order_data' , array (
'post_type' => 'shop_order' ,
'post_title' => sprintf ( __ ( 'Order – %s' , 'woocommerce' ), strftime ( _x ( '%b %d, %Y @ %I:%M %p' , 'Order date parsed by strftime' , 'woocommerce' ) ) ),
'post_status' => 'publish' ,
'ping_status' => 'closed' ,
'post_excerpt' => isset ( $this -> posted [ 'order_comments' ] ) ? $this -> posted [ 'order_comments' ] : '' ,
'post_author' => 1 ,
'post_password' => uniqid ( 'order_' ) // Protects the post just in case
) );
// Insert or update the post data
$create_new_order = true ;
if ( $woocommerce -> session -> order_awaiting_payment > 0 ) {
$order_id = absint ( $woocommerce -> session -> order_awaiting_payment );
/* Check order is unpaid by getting its status */
$terms = wp_get_object_terms ( $order_id , 'shop_order_status' , array ( 'fields' => 'slugs' ) );
$order_status = isset ( $terms [ 0 ] ) ? $terms [ 0 ] : 'pending' ;
// Resume the unpaid order if its pending
2013-02-14 15:37:05 +00:00
if ( $order_status == 'pending' || $order_status == 'failed' ) {
2012-12-12 21:14:19 +00:00
// Update the existing order as we are resuming it
$create_new_order = false ;
$order_data [ 'ID' ] = $order_id ;
wp_update_post ( $order_data );
// Clear the old line items - we'll add these again in case they changed
$wpdb -> query ( $wpdb -> prepare ( " DELETE FROM { $wpdb -> prefix } woocommerce_order_itemmeta WHERE order_item_id IN ( SELECT order_item_id FROM { $wpdb -> prefix } woocommerce_order_items WHERE order_id = %d ) " , $order_id ) );
$wpdb -> query ( $wpdb -> prepare ( " DELETE FROM { $wpdb -> prefix } woocommerce_order_items WHERE order_id = %d " , $order_id ) );
// Trigger an action for the resumed order
do_action ( 'woocommerce_resume_order' , $order_id );
}
}
if ( $create_new_order ) {
$order_id = wp_insert_post ( $order_data );
if ( is_wp_error ( $order_id ) )
2013-05-09 12:39:41 +00:00
throw new Exception ( 'Error: Unable to create order. Please try again.' );
2012-12-12 21:14:19 +00:00
else
do_action ( 'woocommerce_new_order' , $order_id );
}
// Store user data
2013-06-05 11:07:23 +00:00
if ( $this -> checkout_fields [ 'billing' ] )
foreach ( $this -> checkout_fields [ 'billing' ] as $key => $field )
2012-12-12 21:14:19 +00:00
update_post_meta ( $order_id , '_' . $key , $this -> posted [ $key ] );
if ( $this -> checkout_fields [ 'shipping' ] && ( $woocommerce -> cart -> needs_shipping () || get_option ( 'woocommerce_require_shipping_address' ) == 'yes' ) ) {
foreach ( $this -> checkout_fields [ 'shipping' ] as $key => $field ) {
2013-03-04 11:17:48 +00:00
$postvalue = false ;
2012-12-12 21:14:19 +00:00
2013-05-28 13:19:08 +00:00
if ( $this -> posted [ 'ship_to_different_address' ] == false ) {
2013-03-04 11:17:48 +00:00
if ( isset ( $this -> posted [ str_replace ( 'shipping_' , 'billing_' , $key ) ] ) ) {
$postvalue = $this -> posted [ str_replace ( 'shipping_' , 'billing_' , $key ) ];
update_post_meta ( $order_id , '_' . $key , $postvalue );
}
2012-12-12 21:14:19 +00:00
} else {
2013-03-04 11:17:48 +00:00
$postvalue = $this -> posted [ $key ];
update_post_meta ( $order_id , '_' . $key , $postvalue );
2012-12-12 21:14:19 +00:00
}
2013-03-04 11:17:48 +00:00
// User
if ( $postvalue && $this -> customer_id )
update_user_meta ( $this -> customer_id , $key , $postvalue );
2012-12-12 21:14:19 +00:00
}
}
// Save any other user meta
if ( $this -> customer_id )
do_action ( 'woocommerce_checkout_update_user_meta' , $this -> customer_id , $this -> posted );
// Store the line items to the new/resumed order
foreach ( $woocommerce -> cart -> get_cart () as $cart_item_key => $values ) {
$_product = $values [ 'data' ];
// Add line item
$item_id = woocommerce_add_order_item ( $order_id , array (
'order_item_name' => $_product -> get_title (),
'order_item_type' => 'line_item'
) );
// Add line item meta
if ( $item_id ) {
woocommerce_add_order_item_meta ( $item_id , '_qty' , apply_filters ( 'woocommerce_stock_amount' , $values [ 'quantity' ] ) );
woocommerce_add_order_item_meta ( $item_id , '_tax_class' , $_product -> get_tax_class () );
woocommerce_add_order_item_meta ( $item_id , '_product_id' , $values [ 'product_id' ] );
woocommerce_add_order_item_meta ( $item_id , '_variation_id' , $values [ 'variation_id' ] );
2013-02-20 18:50:28 +00:00
woocommerce_add_order_item_meta ( $item_id , '_line_subtotal' , woocommerce_format_decimal ( $values [ 'line_subtotal' ], 4 ) );
woocommerce_add_order_item_meta ( $item_id , '_line_total' , woocommerce_format_decimal ( $values [ 'line_total' ], 4 ) );
2012-12-20 15:10:27 +00:00
woocommerce_add_order_item_meta ( $item_id , '_line_tax' , woocommerce_format_decimal ( $values [ 'line_tax' ], 4 ) );
woocommerce_add_order_item_meta ( $item_id , '_line_subtotal_tax' , woocommerce_format_decimal ( $values [ 'line_subtotal_tax' ], 4 ) );
2012-12-12 21:14:19 +00:00
// Store variation data in meta so admin can view it
if ( $values [ 'variation' ] && is_array ( $values [ 'variation' ] ) )
foreach ( $values [ 'variation' ] as $key => $value )
woocommerce_add_order_item_meta ( $item_id , esc_attr ( str_replace ( 'attribute_' , '' , $key ) ), $value );
// Add line item meta for backorder status
if ( $_product -> backorders_require_notification () && $_product -> is_on_backorder ( $values [ 'quantity' ] ) )
2013-03-18 05:38:22 +00:00
woocommerce_add_order_item_meta ( $item_id , apply_filters ( 'woocommerce_backordered_item_meta_name' , __ ( 'Backordered' , 'woocommerce' ), $cart_item_key , $order_id ), $values [ 'quantity' ] - max ( 0 , $_product -> get_total_stock () ) );
2012-12-12 21:14:19 +00:00
2013-04-16 14:39:07 +00:00
// Allow plugins to add order item meta
2013-05-15 13:19:06 +00:00
do_action ( 'woocommerce_add_order_item_meta' , $item_id , $values , $cart_item_key );
2012-12-12 21:14:19 +00:00
}
}
// Store fees
foreach ( $woocommerce -> cart -> get_fees () as $fee ) {
$item_id = woocommerce_add_order_item ( $order_id , array (
'order_item_name' => $fee -> name ,
'order_item_type' => 'fee'
) );
if ( $fee -> taxable )
woocommerce_add_order_item_meta ( $item_id , '_tax_class' , $fee -> tax_class );
else
woocommerce_add_order_item_meta ( $item_id , '_tax_class' , '0' );
woocommerce_add_order_item_meta ( $item_id , '_line_total' , woocommerce_format_decimal ( $fee -> amount ) );
woocommerce_add_order_item_meta ( $item_id , '_line_tax' , woocommerce_format_decimal ( $fee -> tax ) );
}
// Store tax rows
foreach ( array_keys ( $woocommerce -> cart -> taxes + $woocommerce -> cart -> shipping_taxes ) as $key ) {
$item_id = woocommerce_add_order_item ( $order_id , array (
'order_item_name' => $woocommerce -> cart -> tax -> get_rate_code ( $key ),
'order_item_type' => 'tax'
) );
// Add line item meta
if ( $item_id ) {
woocommerce_add_order_item_meta ( $item_id , 'rate_id' , $key );
woocommerce_add_order_item_meta ( $item_id , 'label' , $woocommerce -> cart -> tax -> get_rate_label ( $key ) );
woocommerce_add_order_item_meta ( $item_id , 'compound' , absint ( $woocommerce -> cart -> tax -> is_compound ( $key ) ? 1 : 0 ) );
woocommerce_add_order_item_meta ( $item_id , 'tax_amount' , woocommerce_clean ( isset ( $woocommerce -> cart -> taxes [ $key ] ) ? $woocommerce -> cart -> taxes [ $key ] : 0 ) );
woocommerce_add_order_item_meta ( $item_id , 'shipping_tax_amount' , woocommerce_clean ( isset ( $woocommerce -> cart -> shipping_taxes [ $key ] ) ? $woocommerce -> cart -> shipping_taxes [ $key ] : 0 ) );
}
}
// Store coupons
if ( $applied_coupons = $woocommerce -> cart -> get_applied_coupons () ) {
foreach ( $applied_coupons as $code ) {
$item_id = woocommerce_add_order_item ( $order_id , array (
'order_item_name' => $code ,
'order_item_type' => 'coupon'
) );
// Add line item meta
if ( $item_id ) {
woocommerce_add_order_item_meta ( $item_id , 'discount_amount' , isset ( $woocommerce -> cart -> coupon_discount_amounts [ $code ] ) ? $woocommerce -> cart -> coupon_discount_amounts [ $code ] : 0 );
}
}
}
// Store meta
2012-12-27 19:24:33 +00:00
if ( $this -> shipping_method ) {
update_post_meta ( $order_id , '_shipping_method' , $this -> shipping_method -> id );
update_post_meta ( $order_id , '_shipping_method_title' , $this -> shipping_method -> label );
2012-12-12 21:14:19 +00:00
}
2012-12-27 19:24:33 +00:00
if ( $this -> payment_method ) {
update_post_meta ( $order_id , '_payment_method' , $this -> payment_method -> id );
update_post_meta ( $order_id , '_payment_method_title' , $this -> payment_method -> get_title () );
2012-12-12 21:14:19 +00:00
}
update_post_meta ( $order_id , '_order_shipping' , woocommerce_format_total ( $woocommerce -> cart -> shipping_total ) );
update_post_meta ( $order_id , '_order_discount' , woocommerce_format_total ( $woocommerce -> cart -> get_order_discount_total () ) );
update_post_meta ( $order_id , '_cart_discount' , woocommerce_format_total ( $woocommerce -> cart -> get_cart_discount_total () ) );
2013-04-12 09:59:38 +00:00
update_post_meta ( $order_id , '_order_tax' , woocommerce_clean ( $woocommerce -> cart -> tax_total ) );
update_post_meta ( $order_id , '_order_shipping_tax' , woocommerce_clean ( $woocommerce -> cart -> shipping_tax_total ) );
2012-12-12 21:14:19 +00:00
update_post_meta ( $order_id , '_order_total' , woocommerce_format_total ( $woocommerce -> cart -> total ) );
update_post_meta ( $order_id , '_order_key' , apply_filters ( 'woocommerce_generate_order_key' , uniqid ( 'order_' ) ) );
update_post_meta ( $order_id , '_customer_user' , absint ( $this -> customer_id ) );
update_post_meta ( $order_id , '_order_currency' , get_woocommerce_currency () );
update_post_meta ( $order_id , '_prices_include_tax' , get_option ( 'woocommerce_prices_include_tax' ) );
update_post_meta ( $order_id , '_customer_ip_address' , isset ( $_SERVER [ 'HTTP_X_FORWARDED_FOR' ] ) ? $_SERVER [ 'HTTP_X_FORWARDED_FOR' ] : $_SERVER [ 'REMOTE_ADDR' ] );
update_post_meta ( $order_id , '_customer_user_agent' , isset ( $_SERVER [ 'HTTP_USER_AGENT' ] ) ? $_SERVER [ 'HTTP_USER_AGENT' ] : '' );
// Let plugins add meta
do_action ( 'woocommerce_checkout_update_order_meta' , $order_id , $this -> posted );
// Order status
wp_set_object_terms ( $order_id , 'pending' , 'shop_order_status' );
return $order_id ;
}
2011-09-12 15:34:29 +00:00
/**
* Process the checkout after the confirm order button is pressed
2012-08-14 19:42:38 +00:00
*
* @ access public
* @ return void
2011-09-12 15:34:29 +00:00
*/
2012-12-14 21:27:29 +00:00
public function process_checkout () {
2013-01-22 21:30:14 +00:00
global $wpdb , $woocommerce , $current_user ;
2011-09-14 14:55:03 +00:00
2013-06-11 16:55:55 +00:00
wp_verify_nonce ( $_POST [ '_wpnonce' ], 'woocommerce-process_checkout' );
2012-12-12 21:14:19 +00:00
2012-11-27 16:22:47 +00:00
if ( ! defined ( 'WOOCOMMERCE_CHECKOUT' ) )
2012-09-07 17:26:13 +00:00
define ( 'WOOCOMMERCE_CHECKOUT' , true );
2012-08-10 11:15:32 +00:00
2012-12-12 21:14:19 +00:00
// Prevent timeout
@ set_time_limit ( 0 );
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
do_action ( 'woocommerce_before_checkout_process' );
2012-01-12 00:43:30 +00:00
2012-06-10 11:40:07 +00:00
if ( sizeof ( $woocommerce -> cart -> get_cart () ) == 0 )
2013-06-11 14:59:54 +00:00
wc_add_error ( sprintf ( __ ( 'Sorry, your session has expired. <a href="%s">Return to homepage →</a>' , 'woocommerce' ), home_url () ) );
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
do_action ( 'woocommerce_checkout_process' );
2012-01-12 00:43:30 +00:00
2012-01-12 00:54:45 +00:00
// Checkout fields (not defined in checkout_fields)
2013-05-28 13:19:08 +00:00
$this -> posted [ 'terms' ] = isset ( $_POST [ 'terms' ] ) ? 1 : 0 ;
$this -> posted [ 'createaccount' ] = isset ( $_POST [ 'createaccount' ] ) ? 1 : 0 ;
2013-05-31 12:05:11 +00:00
$this -> posted [ 'payment_method' ] = isset ( $_POST [ 'payment_method' ] ) ? stripslashes ( $_POST [ 'payment_method' ] ) : '' ;
$this -> posted [ 'shipping_method' ] = isset ( $_POST [ 'shipping_method' ] ) ? stripslashes ( $_POST [ 'shipping_method' ] ) : '' ;
2013-05-28 13:19:08 +00:00
$this -> posted [ 'ship_to_different_address' ] = isset ( $_POST [ 'ship_to_different_address' ] ) ? true : false ;
if ( isset ( $_POST [ 'shiptobilling' ] ) ) {
_deprecated_argument ( 'WC_Checkout::process_checkout()' , '2.1' , 'The "shiptobilling" field is deprecated. THe template files are out of date' );
$this -> posted [ 'ship_to_different_address' ] = $_POST [ 'shiptobilling' ] ? false : true ;
}
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Ship to billing only option
2012-11-27 16:22:47 +00:00
if ( $woocommerce -> cart -> ship_to_billing_address_only () )
2013-05-28 13:19:08 +00:00
$this -> posted [ 'ship_to_different_address' ] = false ;
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Update customer shipping and payment method to posted method
2012-09-07 17:26:13 +00:00
$woocommerce -> session -> chosen_shipping_method = $this -> posted [ 'shipping_method' ];
2012-09-12 11:48:30 +00:00
$woocommerce -> session -> chosen_payment_method = $this -> posted [ 'payment_method' ];
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Note if we skip shipping
$skipped_shipping = false ;
2012-08-10 11:15:32 +00:00
2012-01-13 00:46:56 +00:00
// Get validation class
$validation = $woocommerce -> validation ();
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Get posted checkout_fields and do validation
2012-08-10 11:15:32 +00:00
foreach ( $this -> checkout_fields as $fieldset_key => $fieldset ) {
2013-05-28 13:19:08 +00:00
// Skip shipping if not needed
if ( $fieldset_key == 'shipping' && ( $this -> posted [ 'ship_to_different_address' ] == false || ( ! $woocommerce -> cart -> needs_shipping () && get_option ( 'woocommerce_require_shipping_address' ) == 'no' ) ) ) {
2012-01-12 00:54:45 +00:00
$skipped_shipping = true ;
continue ;
2012-04-21 18:30:37 +00:00
}
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
foreach ( $fieldset as $key => $field ) {
2012-08-10 11:15:32 +00:00
2013-05-23 08:40:55 +00:00
if ( ! isset ( $field [ 'type' ] ) )
$field [ 'type' ] = 'text' ;
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Get Value
2012-07-16 19:21:44 +00:00
switch ( $field [ 'type' ] ) {
2012-01-12 00:54:45 +00:00
case " checkbox " :
2013-05-23 08:40:55 +00:00
$this -> posted [ $key ] = isset ( $_POST [ $key ] ) ? 1 : 0 ;
break ;
case " multiselect " :
$this -> posted [ $key ] = isset ( $_POST [ $key ] ) ? implode ( ', ' , array_map ( 'woocommerce_clean' , $_POST [ $key ] ) ) : '' ;
2012-01-12 00:54:45 +00:00
break ;
default :
2013-05-23 08:40:55 +00:00
$this -> posted [ $key ] = isset ( $_POST [ $key ] ) ? woocommerce_clean ( $_POST [ $key ] ) : '' ;
2012-01-12 00:54:45 +00:00
break ;
2012-07-16 19:21:44 +00:00
}
2012-08-10 11:15:32 +00:00
2013-05-23 08:40:55 +00:00
// Hooks to allow modification of value
$this -> posted [ $key ] = apply_filters ( 'woocommerce_process_checkout_' . sanitize_title ( $field [ 'type' ] ) . '_field' , $this -> posted [ $key ] );
$this -> posted [ $key ] = apply_filters ( 'woocommerce_process_checkout_field_' . $key , $this -> posted [ $key ] );
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Validation: Required fields
2013-06-11 14:59:54 +00:00
if ( isset ( $field [ 'required' ] ) && $field [ 'required' ] && empty ( $this -> posted [ $key ] ) )
wc_add_error ( '<strong>' . $field [ 'label' ] . '</strong> ' . __ ( 'is a required field.' , 'woocommerce' ) );
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
if ( ! empty ( $this -> posted [ $key ] ) ) {
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Special handling for validation and formatting
2012-07-16 19:21:44 +00:00
switch ( $key ) {
2012-01-12 00:54:45 +00:00
case " billing_postcode " :
case " shipping_postcode " :
2012-08-10 11:15:32 +00:00
$validate_against = $key == 'billing_postcode' ? 'billing_country' : 'shipping_country' ;
2012-05-22 20:31:27 +00:00
$this -> posted [ $key ] = strtoupper ( str_replace ( ' ' , '' , $this -> posted [ $key ] ) );
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
if ( ! $validation -> is_postcode ( $this -> posted [ $key ], $_POST [ $validate_against ] ) )
2013-06-11 14:59:54 +00:00
wc_add_error ( '<strong>' . $field [ 'label' ] . '</strong> ' . sprintf ( __ ( '(%s) is not a valid postcode/ZIP.' , 'woocommerce' ), $this -> posted [ $key ] ) );
2012-07-16 19:21:44 +00:00
else
2013-05-23 08:40:55 +00:00
$this -> posted [ $key ] = $validation -> format_postcode ( $this -> posted [ $key ], $_POST [ $validate_against ] );
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
break ;
case " billing_state " :
case " shipping_state " :
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
// Get valid states
$validate_against = $key == 'billing_state' ? 'billing_country' : 'shipping_country' ;
$valid_states = $woocommerce -> countries -> get_states ( $_POST [ $validate_against ] );
if ( $valid_states )
$valid_state_values = array_flip ( array_map ( 'strtolower' , $valid_states ) );
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
// Convert value to key if set
if ( isset ( $valid_state_values [ strtolower ( $this -> posted [ $key ] ) ] ) )
$this -> posted [ $key ] = $valid_state_values [ strtolower ( $this -> posted [ $key ] ) ];
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
// Only validate if the country has specific state options
if ( $valid_states && sizeof ( $valid_states ) > 0 )
if ( ! in_array ( $this -> posted [ $key ], array_keys ( $valid_states ) ) )
2013-06-11 14:59:54 +00:00
wc_add_error ( '<strong>' . $field [ 'label' ] . '</strong> ' . __ ( 'is not valid. Please enter one of the following:' , 'woocommerce' ) . ' ' . implode ( ', ' , $valid_states ) );
2012-08-10 11:15:32 +00:00
2012-01-12 00:30:21 +00:00
break ;
2012-01-12 00:54:45 +00:00
case " billing_phone " :
2012-08-10 11:15:32 +00:00
2012-06-10 09:44:13 +00:00
$this -> posted [ $key ] = $validation -> format_phone ( $this -> posted [ $key ] );
2012-08-10 11:15:32 +00:00
2012-06-10 09:44:13 +00:00
if ( ! $validation -> is_phone ( $this -> posted [ $key ] ) )
2013-06-11 14:59:54 +00:00
wc_add_error ( '<strong>' . $field [ 'label' ] . '</strong> ' . __ ( 'is not a valid number.' , 'woocommerce' ) );
2012-01-12 00:54:45 +00:00
break ;
case " billing_email " :
2012-08-10 11:15:32 +00:00
2013-05-23 08:40:55 +00:00
$this -> posted [ $key ] = strtolower ( $this -> posted [ $key ] );
2012-08-10 11:15:32 +00:00
2013-05-23 08:40:55 +00:00
if ( ! $validation -> is_email ( $this -> posted [ $key ] ) )
2013-06-11 14:59:54 +00:00
wc_add_error ( '<strong>' . $field [ 'label' ] . '</strong> ' . __ ( 'is not a valid email address.' , 'woocommerce' ) );
2011-12-21 16:03:45 +00:00
break ;
2012-07-16 19:21:44 +00:00
}
}
}
}
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Update customer location to posted location so we can correctly check available shipping methods
2012-08-10 11:15:32 +00:00
if ( isset ( $this -> posted [ 'billing_country' ] ) )
2012-03-26 19:28:19 +00:00
$woocommerce -> customer -> set_country ( $this -> posted [ 'billing_country' ] );
2012-08-10 11:15:32 +00:00
if ( isset ( $this -> posted [ 'billing_state' ] ) )
2012-03-26 19:28:19 +00:00
$woocommerce -> customer -> set_state ( $this -> posted [ 'billing_state' ] );
2012-08-10 11:15:32 +00:00
if ( isset ( $this -> posted [ 'billing_postcode' ] ) )
2012-03-26 19:28:19 +00:00
$woocommerce -> customer -> set_postcode ( $this -> posted [ 'billing_postcode' ] );
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Shipping Information
2012-03-26 19:28:19 +00:00
if ( ! $skipped_shipping ) {
2012-08-10 11:15:32 +00:00
2011-12-21 16:03:45 +00:00
// Update customer location to posted location so we can correctly check available shipping methods
2012-08-10 11:15:32 +00:00
if ( isset ( $this -> posted [ 'shipping_country' ] ) )
2012-03-26 19:16:05 +00:00
$woocommerce -> customer -> set_shipping_country ( $this -> posted [ 'shipping_country' ] );
2012-08-10 11:15:32 +00:00
if ( isset ( $this -> posted [ 'shipping_state' ] ) )
$woocommerce -> customer -> set_shipping_state ( $this -> posted [ 'shipping_state' ] );
if ( isset ( $this -> posted [ 'shipping_postcode' ] ) )
$woocommerce -> customer -> set_shipping_postcode ( $this -> posted [ 'shipping_postcode' ] );
2012-03-26 19:28:19 +00:00
} else {
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Update customer location to posted location so we can correctly check available shipping methods
2012-08-10 11:15:32 +00:00
if ( isset ( $this -> posted [ 'billing_country' ] ) )
2012-03-26 19:16:05 +00:00
$woocommerce -> customer -> set_shipping_country ( $this -> posted [ 'billing_country' ] );
2012-08-10 11:15:32 +00:00
if ( isset ( $this -> posted [ 'billing_state' ] ) )
2012-03-26 19:16:05 +00:00
$woocommerce -> customer -> set_shipping_state ( $this -> posted [ 'billing_state' ] );
2012-08-10 11:15:32 +00:00
if ( isset ( $this -> posted [ 'billing_postcode' ] ) )
2012-03-26 19:16:05 +00:00
$woocommerce -> customer -> set_shipping_postcode ( $this -> posted [ 'billing_postcode' ] );
2012-08-10 11:15:32 +00:00
2012-03-26 19:28:19 +00:00
}
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
// Update cart totals now we have customer address
$woocommerce -> cart -> calculate_totals ();
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Terms
2012-08-10 11:15:32 +00:00
if ( ! isset ( $_POST [ 'woocommerce_checkout_update_totals' ] ) && empty ( $this -> posted [ 'terms' ] ) && woocommerce_get_page_id ( 'terms' ) > 0 )
2013-06-11 14:59:54 +00:00
wc_add_error ( __ ( 'You must accept our Terms & Conditions.' , 'woocommerce' ) );
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
if ( $woocommerce -> cart -> needs_shipping () ) {
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Shipping Method
$available_methods = $woocommerce -> shipping -> get_available_shipping_methods ();
2012-08-10 11:15:32 +00:00
2012-12-12 21:14:19 +00:00
if ( ! isset ( $available_methods [ $this -> posted [ 'shipping_method' ] ] ) ) {
2012-12-27 19:24:33 +00:00
$this -> shipping_method = '' ;
2013-06-11 14:59:54 +00:00
wc_add_error ( __ ( 'Invalid shipping method.' , 'woocommerce' ) );
2012-12-12 21:14:19 +00:00
} else {
2012-12-27 19:24:33 +00:00
$this -> shipping_method = $available_methods [ $this -> posted [ 'shipping_method' ] ];
2012-12-12 21:14:19 +00:00
}
2012-08-10 11:15:32 +00:00
}
2012-09-07 17:26:13 +00:00
if ( $woocommerce -> cart -> needs_payment () ) {
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Payment Method
$available_gateways = $woocommerce -> payment_gateways -> get_available_payment_gateways ();
2012-08-10 11:15:32 +00:00
2012-12-12 21:14:19 +00:00
if ( ! isset ( $available_gateways [ $this -> posted [ 'payment_method' ] ] ) ) {
2012-12-27 19:24:33 +00:00
$this -> payment_method = '' ;
2013-06-11 14:59:54 +00:00
wc_add_error ( __ ( 'Invalid payment method.' , 'woocommerce' ) );
2012-12-12 21:14:19 +00:00
} else {
2012-12-27 19:24:33 +00:00
$this -> payment_method = $available_gateways [ $this -> posted [ 'payment_method' ] ];
$this -> payment_method -> validate_fields ();
2012-12-12 21:14:19 +00:00
}
2012-07-16 19:21:44 +00:00
}
2012-08-10 11:15:32 +00:00
2012-07-16 19:21:44 +00:00
// Action after validation
2012-01-12 00:54:45 +00:00
do_action ( 'woocommerce_after_checkout_validation' , $this -> posted );
2012-08-10 11:15:32 +00:00
2013-06-11 14:59:54 +00:00
if ( ! isset ( $_POST [ 'woocommerce_checkout_update_totals' ] ) && wc_error_count () == 0 ) {
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
try {
2012-08-10 11:15:32 +00:00
2013-06-04 15:33:05 +00:00
// Customer accounts
$this -> customer_id = get_current_user_id ();
2012-08-10 11:15:32 +00:00
2013-06-04 15:33:05 +00:00
if ( ! is_user_logged_in () && ( $this -> must_create_account || ! empty ( $this -> posted [ 'createaccount' ] ) ) ) {
2012-08-10 11:15:32 +00:00
2013-06-04 15:33:05 +00:00
$username = ! empty ( $this -> posted [ 'account_username' ] ) ? $this -> posted [ 'account_username' ] : '' ;
$password = ! empty ( $this -> posted [ 'account_password' ] ) ? $this -> posted [ 'account_password' ] : '' ;
$new_customer = woocommerce_create_new_customer ( $this -> posted [ 'billing_email' ], $username , $password );
2012-08-10 11:15:32 +00:00
2013-06-04 15:33:05 +00:00
if ( is_wp_error ( $new_customer ) )
throw new Exception ( $new_customer -> get_error_message () );
2012-12-28 18:45:06 +00:00
2013-06-04 15:33:05 +00:00
$this -> customer_id = $new_customer ;
2012-08-10 11:15:32 +00:00
2013-06-04 15:33:05 +00:00
woocommerce_set_customer_auth_cookie ( $this -> customer_id );
2013-06-05 11:07:23 +00:00
// Add customer info from other billing fields
if ( $this -> posted [ 'billing_first_name' ] )
wp_update_user ( array ( 'ID' => $this -> customer_id , 'first_name' => $this -> posted [ 'billing_first_name' ], 'display_name' => $this -> posted [ 'billing_first_name' ] ) );
if ( $this -> posted [ 'billing_last_name' ] )
wp_update_user ( array ( 'ID' => $this -> customer_id , 'last_name' => $this -> posted [ 'billing_last_name' ] ) ) ;
2012-09-07 17:26:13 +00:00
}
2012-08-10 11:15:32 +00:00
2012-12-12 21:14:19 +00:00
// Abort if errors are present
2013-06-11 14:59:54 +00:00
if ( wc_error_count () > 0 )
2013-05-09 12:39:41 +00:00
throw new Exception ();
2012-08-10 11:15:32 +00:00
2012-12-12 21:14:19 +00:00
// Create the order
$order_id = $this -> create_order ();
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Order is saved
2012-09-07 17:26:13 +00:00
do_action ( 'woocommerce_checkout_order_processed' , $order_id , $this -> posted );
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Process payment
2012-09-07 17:26:13 +00:00
if ( $woocommerce -> cart -> needs_payment () ) {
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Store Order ID in session so it can be re-used after payment failure
2012-09-07 17:26:13 +00:00
$woocommerce -> session -> order_awaiting_payment = $order_id ;
2012-01-10 16:43:06 +00:00
2012-01-12 00:54:45 +00:00
// Process Payment
2012-09-07 17:26:13 +00:00
$result = $available_gateways [ $this -> posted [ 'payment_method' ] ] -> process_payment ( $order_id );
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Redirect to success/confirmation/payment page
2012-09-07 17:26:13 +00:00
if ( $result [ 'result' ] == 'success' ) {
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
$result = apply_filters ( 'woocommerce_payment_successful_result' , $result );
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
if ( is_ajax () ) {
2013-03-08 12:24:22 +00:00
echo '<!--WC_START-->' . json_encode ( $result ) . '<!--WC_END-->' ;
2011-08-09 15:16:18 +00:00
exit ;
2012-09-07 17:26:13 +00:00
} else {
2012-02-24 20:44:58 +00:00
wp_redirect ( $result [ 'redirect' ] );
2011-08-09 15:16:18 +00:00
exit ;
2012-09-07 17:26:13 +00:00
}
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
}
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
} else {
2012-08-10 11:15:32 +00:00
2012-08-12 14:12:52 +00:00
if ( empty ( $order ) )
$order = new WC_Order ( $order_id );
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// No payment was required for order
$order -> payment_complete ();
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Empty the Cart
$woocommerce -> cart -> empty_cart ();
2012-08-10 11:15:32 +00:00
2012-03-05 13:25:02 +00:00
// Get redirect
2013-05-31 15:13:14 +00:00
$return_url = $order -> get_checkout_order_received_url ();
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// Redirect to success/confirmation/payment page
2012-09-07 17:26:13 +00:00
if ( is_ajax () ) {
2013-03-08 12:24:22 +00:00
echo '<!--WC_START-->' . json_encode (
2012-01-18 12:32:39 +00:00
array (
2012-03-16 15:09:12 +00:00
'result' => 'success' ,
2013-05-31 15:13:14 +00:00
'redirect' => apply_filters ( 'woocommerce_checkout_no_payment_needed_redirect' , $return_url , $order )
2012-08-10 11:15:32 +00:00
)
2012-12-13 14:57:31 +00:00
) . '<!--WC_END-->' ;
2012-01-12 00:54:45 +00:00
exit ;
2012-09-07 17:26:13 +00:00
} else {
2012-08-10 11:15:32 +00:00
wp_safe_redirect (
2013-05-31 15:13:14 +00:00
apply_filters ( 'woocommerce_checkout_no_payment_needed_redirect' , $return_url , $order )
2012-01-18 12:32:39 +00:00
);
2012-01-12 00:54:45 +00:00
exit ;
2012-09-07 17:26:13 +00:00
}
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
}
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
} catch ( Exception $e ) {
2012-11-27 16:22:47 +00:00
2012-09-07 17:26:13 +00:00
if ( ! empty ( $e ) )
2013-06-11 14:59:54 +00:00
wc_add_error ( $e );
2012-11-27 16:22:47 +00:00
2012-09-07 17:26:13 +00:00
}
2012-01-12 00:54:45 +00:00
2012-09-07 17:26:13 +00:00
} // endif
2012-08-10 11:15:32 +00:00
2012-01-12 00:54:45 +00:00
// If we reached this point then there were errors
2012-09-07 17:26:13 +00:00
if ( is_ajax () ) {
2012-08-10 11:15:32 +00:00
2012-03-14 11:16:26 +00:00
ob_start ();
2013-06-11 14:59:54 +00:00
wc_print_messages ();
2012-03-14 11:16:26 +00:00
$messages = ob_get_clean ();
2012-08-10 11:15:32 +00:00
2013-03-08 12:24:22 +00:00
echo '<!--WC_START-->' . json_encode (
2012-03-14 11:16:26 +00:00
array (
'result' => 'failure' ,
'messages' => $messages ,
2012-09-07 17:26:13 +00:00
'refresh' => isset ( $woocommerce -> session -> refresh_totals ) ? 'true' : 'false'
2012-08-10 11:15:32 +00:00
)
2012-12-13 14:57:31 +00:00
) . '<!--WC_END-->' ;
2012-08-10 11:15:32 +00:00
2012-09-07 17:26:13 +00:00
unset ( $woocommerce -> session -> refresh_totals );
2012-01-12 00:54:45 +00:00
exit ;
2012-09-07 17:26:13 +00:00
}
2011-08-09 15:16:18 +00:00
}
2012-08-10 11:15:32 +00:00
2012-08-14 19:42:38 +00:00
/**
* Gets the value either from the posted data , or from the users meta data
*
* @ access public
* @ param string $input
* @ return string
*/
2012-12-14 21:27:29 +00:00
public function get_value ( $input ) {
2012-01-04 16:24:26 +00:00
global $woocommerce ;
2012-08-10 11:15:32 +00:00
2012-10-19 17:59:17 +00:00
if ( ! empty ( $_POST [ $input ] ) ) {
2012-08-10 11:15:32 +00:00
2012-10-19 17:59:17 +00:00
return esc_attr ( $_POST [ $input ] );
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
} else {
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
if ( is_user_logged_in () ) {
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
$current_user = wp_get_current_user ();
2012-08-10 11:15:32 +00:00
2013-01-07 17:26:02 +00:00
if ( $meta = get_user_meta ( $current_user -> ID , $input , true ) )
2013-01-07 15:58:19 +00:00
return $meta ;
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
if ( $input == " billing_email " )
return $current_user -> user_email ;
}
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
$default_billing_country = apply_filters ( 'default_checkout_country' , ( $woocommerce -> customer -> get_country ()) ? $woocommerce -> customer -> get_country () : $woocommerce -> countries -> get_base_country ());
$default_shipping_country = apply_filters ( 'default_checkout_country' , ( $woocommerce -> customer -> get_shipping_country ()) ? $woocommerce -> customer -> get_shipping_country () : $woocommerce -> countries -> get_base_country ());
2012-08-10 11:15:32 +00:00
2012-05-09 17:29:22 +00:00
if ( $woocommerce -> customer -> has_calculated_shipping () ) {
2013-01-07 15:58:19 +00:00
$default_billing_state = apply_filters ( 'default_checkout_state' , $woocommerce -> customer -> get_state ());
$default_shipping_state = apply_filters ( 'default_checkout_state' , $woocommerce -> customer -> get_shipping_state ());
2012-05-09 17:29:22 +00:00
} else {
2013-01-07 15:58:19 +00:00
$default_billing_state = apply_filters ( 'default_checkout_state' , '' );
$default_shipping_state = apply_filters ( 'default_checkout_state' , '' );
2012-04-19 11:09:52 +00:00
}
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
if ( $input == " billing_country " )
return $default_billing_country ;
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
if ( $input == " billing_state " )
return $default_billing_state ;
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
if ( $input == " billing_postcode " )
return $woocommerce -> customer -> get_postcode () ? $woocommerce -> customer -> get_postcode () : '' ;
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
if ( $input == " shipping_country " )
return $default_shipping_country ;
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
if ( $input == " shipping_state " )
return $default_shipping_state ;
2012-08-10 11:15:32 +00:00
2013-01-07 15:58:19 +00:00
if ( $input == " shipping_postcode " )
return $woocommerce -> customer -> get_shipping_postcode () ? $woocommerce -> customer -> get_shipping_postcode () : '' ;
2012-10-19 17:59:17 +00:00
}
2011-08-09 15:16:18 +00:00
}
2013-01-22 21:30:14 +00:00
}