woocommerce/includes/abstracts/abstract-wc-shipping-method...

468 lines
13 KiB
PHP
Raw Normal View History

2011-08-09 15:16:18 +00:00
<?php
2013-02-20 17:14:46 +00:00
if ( ! defined( 'ABSPATH' ) ) {
exit;
}
2013-02-20 17:14:46 +00:00
2011-08-10 17:11:11 +00:00
/**
* WooCommerce Shipping Method Class.
2012-08-12 14:12:52 +00:00
*
2011-08-10 17:11:11 +00:00
* Extended by shipping methods to handle shipping calculations etc.
*
* @class WC_Shipping_Method
* @version 2.6.0
* @package WooCommerce/Abstracts
* @category Abstract Class
* @author WooThemes
2012-08-12 14:12:52 +00:00
*/
abstract class WC_Shipping_Method extends WC_Settings_API {
2012-08-12 14:12:52 +00:00
/**
* Features this method supports. Possible features used by core:
2015-12-16 13:52:46 +00:00
* - shipping-zones Shipping zone functionality + instances
* - instance-settings Instance settings screens.
* - settings Non-instance settings screens. Enabled by default for BW compatibility with methods before instances existed.
* @var array
*/
2015-12-16 13:52:46 +00:00
public $supports = array( 'settings' );
2012-08-14 22:43:48 +00:00
/**
* Unique ID for the shipping method - must be set.
* @var string
*/
2015-12-16 13:52:46 +00:00
public $id = '';
2012-08-14 22:43:48 +00:00
/**
* Method title.
* @var string
*/
2015-12-16 13:52:46 +00:00
public $method_title = '';
/**
* Method description.
* @var string
*/
2015-12-16 13:52:46 +00:00
public $method_description = '';
/**
* yes or no based on whether the method is enabled.
* @var string
*/
2015-12-16 13:52:46 +00:00
public $enabled = 'yes';
2015-12-14 14:03:46 +00:00
/**
* Shipping method title for the frontend.
* @var string
*/
public $title;
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
/**
* This is an array of rates - methods must populate this array to register shipping costs.
* @var array
*/
2015-12-16 13:52:46 +00:00
public $rates = array();
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
/**
* If 'taxable' tax will be charged for this method (if applicable).
* @var string
*/
2015-12-16 13:52:46 +00:00
public $tax_status = 'taxable';
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
/**
* Fee for the method (if applicable).
* @var string
*/
2015-12-16 13:52:46 +00:00
public $fee = null;
2012-08-14 22:43:48 +00:00
2012-08-15 17:08:42 +00:00
/** @var float Minimum fee for the method */
2015-12-14 16:56:39 +00:00
/**
* Minimum fee for the method (if applicable).
* @var string
*/
2015-12-16 13:52:46 +00:00
public $minimum_fee = null;
2012-08-14 22:43:48 +00:00
2015-12-14 14:03:46 +00:00
/**
2015-12-14 16:56:39 +00:00
* Availability - legacy. Used for method Availability.
* No longer useful for instance based shipping methods.
* @deprecated 2.6.0
* @var string
*/
public $availability;
/**
* Availability countries - legacy. Used for method Availability.
* No longer useful for instance based shipping methods.
* @deprecated 2.6.0
* @var string
*/
2015-12-16 13:52:46 +00:00
public $countries = array();
2015-12-14 16:56:39 +00:00
/**
* Instance ID if used.
* @var int
*/
2015-12-16 13:52:46 +00:00
public $instance_id = 0;
/**
* Instance form fields.
* @var array
*/
public $instance_form_fields = array();
2015-12-14 16:56:39 +00:00
/**
* Instance settings.
2015-12-14 14:03:46 +00:00
* @var array
*/
2015-12-16 13:52:46 +00:00
public $instance_settings = array();
/**
* Constructor.
2015-12-14 16:56:39 +00:00
* @param int $instance_id
*/
public function __construct( $instance_id = 0 ) {
$this->instance_id = absint( $instance_id );
}
2012-08-12 14:12:52 +00:00
/**
* Check if a shipping method supports a given feature.
*
* Methods should override this to declare support (or lack of support) for a feature.
*
* @param $feature string The name of a feature to test support for.
* @return bool True if the shipping method supports the feature, false otherwise.
*/
public function supports( $feature ) {
return apply_filters( 'woocommerce_shipping_method_supports', in_array( $feature, $this->supports ), $feature, $this );
}
/**
* Called to calculate shipping rates for this method. Rates can be added using the add_rate() method.
*/
public function calculate_shipping() {}
/**
* Whether or not we need to calculate tax on top of the shipping rate.
* @return boolean
*/
public function is_taxable() {
return wc_tax_enabled() && 'taxable' === $this->tax_status && ! WC()->customer->is_vat_exempt();
}
2015-12-14 16:56:39 +00:00
/**
* Whether or not this method is enabled in settings.
* @since 2.6.0
* @return boolean
*/
public function is_enabled() {
return 'yes' === $this->enabled;
}
2015-12-15 15:00:45 +00:00
/**
* Return the shipping method instance ID.
* @since 2.6.0
* @return int
*/
public function get_instance_id() {
return $this->instance_id;
}
/**
* Return the shipping method title.
* @since 2.6.0
* @return string
*/
public function get_method_title() {
return apply_filters( 'woocommerce_shipping_method_title', $this->method_title, $this );
}
/**
* Return the shipping method description.
* @since 2.6.0
* @return string
*/
public function get_method_description() {
return apply_filters( 'woocommerce_shipping_method_description', $this->method_description, $this );
}
/**
* Return the shipping title which is user set.
*
* @return string
*/
public function get_title() {
return apply_filters( 'woocommerce_shipping_method_title', $this->title, $this->id );
}
/**
* Return calculated rates for a package.
* @since 2.6.0
* @param object $package
* @return array
*/
public function get_rates_for_package( $package ) {
$this->rates = array();
if ( $this->is_available( $package ) && ( empty( $package['ship_via'] ) || in_array( $this->id, $package['ship_via'] ) ) ) {
$this->calculate_shipping( $package );
}
return $this->rates;
}
/**
2012-08-14 22:43:48 +00:00
* Add a shipping rate. If taxes are not set they will be calculated based on cost.
* @param array $args (default: array())
*/
public function add_rate( $args = array() ) {
2015-09-07 15:02:23 +00:00
$args = wp_parse_args( $args, array(
'id' => '', // ID for the rate
'label' => '', // Label for the rate
'cost' => '0', // Amount or array of costs (per item shipping)
'taxes' => '', // Pass taxes, nothing to have it calculated for you, or 'false' to calc no tax
'calc_tax' => 'per_order' // Calc tax per_order or per_item. Per item needs an array of costs
2015-09-07 15:02:23 +00:00
) );
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
// ID and label are required
2015-09-07 15:02:23 +00:00
if ( ! $args['id'] || ! $args['label'] ) {
2014-11-17 16:30:15 +00:00
return;
}
2012-08-12 14:12:52 +00:00
2012-01-14 16:42:04 +00:00
// Handle cost
2015-09-07 15:02:23 +00:00
$total_cost = is_array( $args['cost'] ) ? array_sum( $args['cost'] ) : $args['cost'];
$taxes = $args['taxes'];
2012-08-12 14:12:52 +00:00
// Taxes - if not an array and not set to false, calc tax based on cost and passed calc_tax variable
2013-03-03 17:07:31 +00:00
// This saves shipping methods having to do complex tax calculations
if ( ! is_array( $taxes ) && $taxes !== false && $total_cost > 0 && $this->is_taxable() ) {
2015-09-07 15:02:23 +00:00
switch ( $args['calc_tax'] ) {
case "per_item" :
2015-12-14 16:56:39 +00:00
$taxes = $this->get_taxes_per_item( $args['cost'] );
break;
default :
$taxes = WC_Tax::calc_shipping_tax( $total_cost, WC_Tax::get_shipping_tax_rates() );
break;
}
}
$this->rates[ $args['id'] ] = new WC_Shipping_Rate( $args['id'], $args['label'], $total_cost, $taxes, $this->id );
2015-12-14 16:56:39 +00:00
}
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
/**
* Calc taxes per item being shipping in costs array.
* @since 2.6.0
* @access protected
* @param array $costs
* @return array of taxes
*/
protected function get_taxes_per_item( $costs ) {
$taxes = array();
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
// If we have an array of costs we can look up each items tax class and add tax accordingly
if ( is_array( $costs ) ) {
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
$cart = WC()->cart->get_cart();
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
foreach ( $costs as $cost_key => $amount ) {
if ( ! isset( $cart[ $cost_key ] ) ) {
continue;
}
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
$item_taxes = WC_Tax::calc_shipping_tax( $amount, WC_Tax::get_shipping_tax_rates( $cart[ $cost_key ]['data']->get_tax_class() ) );
2012-08-12 14:12:52 +00:00
2015-12-14 16:56:39 +00:00
// Sum the item taxes
foreach ( array_keys( $taxes + $item_taxes ) as $key ) {
$taxes[ $key ] = ( isset( $item_taxes[ $key ] ) ? $item_taxes[ $key ] : 0 ) + ( isset( $taxes[ $key ] ) ? $taxes[ $key ] : 0 );
}
}
// Add any cost for the order - order costs are in the key 'order'
if ( isset( $costs['order'] ) ) {
$item_taxes = WC_Tax::calc_shipping_tax( $costs['order'], WC_Tax::get_shipping_tax_rates() );
// Sum the item taxes
foreach ( array_keys( $taxes + $item_taxes ) as $key ) {
$taxes[ $key ] = ( isset( $item_taxes[ $key ] ) ? $item_taxes[ $key ] : 0 ) + ( isset( $taxes[ $key ] ) ? $taxes[ $key ] : 0 );
}
2012-05-29 12:32:54 +00:00
}
}
2015-12-14 16:56:39 +00:00
return $taxes;
}
2012-08-12 14:12:52 +00:00
/**
* is_available function.
*
* @param array $package
* @return bool
*/
public function is_available( $package ) {
2015-12-14 16:56:39 +00:00
if ( ! $this->is_enabled() ) {
return false;
}
2012-08-12 14:12:52 +00:00
// Country availability
$countries = is_array( $this->countries ) ? $this->countries : array();
switch ( $this->availability ) {
case 'specific' :
case 'including' :
$ship_to_countries = array_intersect( $countries, array_keys( WC()->countries->get_shipping_countries() ) );
break;
case 'excluding' :
$ship_to_countries = array_diff( array_keys( WC()->countries->get_shipping_countries() ), $countries );
break;
default :
2013-11-25 14:01:32 +00:00
$ship_to_countries = array_keys( WC()->countries->get_shipping_countries() );
break;
}
2012-08-12 14:12:52 +00:00
if ( ! in_array( $package['destination']['country'], $ship_to_countries ) ) {
return false;
}
2012-08-12 14:12:52 +00:00
return apply_filters( 'woocommerce_shipping_' . $this->id . '_is_available', true, $package );
}
/**
* get_fee function.
*
* @param mixed $fee
* @param mixed $total
* @return float
*/
public function get_fee( $fee, $total ) {
2014-11-17 16:30:15 +00:00
if ( strstr( $fee, '%' ) ) {
$fee = ( $total / 100 ) * str_replace( '%', '', $fee );
2014-11-17 16:30:15 +00:00
}
if ( ! empty( $this->minimum_fee ) && $this->minimum_fee > $fee ) {
$fee = $this->minimum_fee;
}
return $fee;
2012-08-12 14:12:52 +00:00
}
2015-12-14 16:56:39 +00:00
/**
* Does this method have a settings page?
* @return bool
*/
public function has_settings() {
return $this->instance_id ? $this->supports( 'instance-settings' ) : $this->supports( 'settings' );
}
/**
2015-12-14 16:56:39 +00:00
* Output the shipping settings screen.
*/
public function admin_options() {
if ( $this->instance_id ) {
echo wp_kses_post( wpautop( $this->get_method_description() ) );
echo '<table class="form-table">' . $this->generate_settings_html( $this->get_instance_form_fields(), false ) . '</table>';
} else {
echo '<h2>' . esc_html( $this->get_method_title() ) . '</h2>';
echo wp_kses_post( wpautop( $this->get_method_description() ) );
echo '<table class="form-table">' . $this->generate_settings_html( $this->get_form_fields(), false ) . '</table>';
}
}
/**
* get_option function.
*
* Gets and option from the settings API, using defaults if necessary to prevent undefined notices.
*
* @param string $key
* @param mixed $empty_value
* @return mixed The value specified for the option or a default value for the option.
*/
public function get_option( $key, $empty_value = null ) {
2015-12-15 12:39:52 +00:00
// Instance options take priority over global options
2015-12-14 16:56:39 +00:00
if ( in_array( $key, array_keys( $this->get_instance_form_fields() ) ) ) {
return $this->get_instance_option( $key, $empty_value );
}
// Return global option
return parent::get_option( $key, $empty_value );
}
/**
* get_option function.
*
* Gets and option from the settings API, using defaults if necessary to prevent undefined notices.
*
* @param string $key
* @param mixed $empty_value
* @return mixed The value specified for the option or a default value for the option.
*/
public function get_instance_option( $key, $empty_value = null ) {
if ( empty( $this->instance_settings ) ) {
$this->init_instance_settings();
}
// Get option default if unset.
if ( ! isset( $this->instance_settings[ $key ] ) ) {
$form_fields = $this->get_instance_form_fields();
$this->instance_settings[ $key ] = $this->get_field_default( $form_fields[ $key ] );
}
if ( ! is_null( $empty_value ) && '' === $this->instance_settings[ $key ] ) {
$this->instance_settings[ $key ] = $empty_value;
}
return $this->instance_settings[ $key ];
}
/**
* Get settings fields for instances of this shipping method (within zones).
* Should be overridden by shipping methods to add options.
* @since 2.6.0
* @return array
*/
public function get_instance_form_fields() {
2015-12-16 13:52:46 +00:00
return apply_filters( 'woocommerce_shipping_instance_form_fields_' . $this->id, $this->instance_form_fields );
}
/**
2015-12-14 16:56:39 +00:00
* Return the name of the option in the WP DB.
* @since 2.6.0
2015-12-14 16:56:39 +00:00
* @return string
*/
2015-12-14 16:56:39 +00:00
public function get_instance_option_key() {
return $this->instance_id ? $this->plugin_id . $this->id . $this->instance_id . '_settings' : '';
}
/**
2015-12-14 16:56:39 +00:00
* Initialise Settings for instances.
* @since 2.6.0
*/
public function init_instance_settings() {
$this->instance_settings = get_option( $this->get_instance_option_key(), null );
// If there are no settings defined, use defaults.
if ( ! is_array( $this->instance_settings ) ) {
$form_fields = $this->get_instance_form_fields();
$this->instance_settings = array_merge( array_fill_keys( array_keys( $form_fields ), '' ), wp_list_pluck( $form_fields, 'default' ) );
}
}
/**
* Processes and saves options.
* If there is an error thrown, will continue to save and validate fields, but will leave the erroring field out.
* @since 2.6.0
* @return bool was anything saved?
*/
2015-12-14 16:56:39 +00:00
public function process_admin_options() {
if ( $this->instance_id ) {
$this->init_instance_settings();
foreach ( $this->get_instance_form_fields() as $key => $field ) {
if ( ! in_array( $this->get_field_type( $field ), array( 'title' ) ) ) {
try {
$this->instance_settings[ $key ] = $this->get_field_value( $key, $field );
} catch ( Exception $e ) {
$this->add_error( $e->getMessage() );
}
}
}
2015-12-14 16:56:39 +00:00
return update_option( $this->get_instance_option_key(), apply_filters( 'woocommerce_shipping_' . $this->id . '_instance_settings_values', $this->instance_settings, $this ) );
} else {
2015-12-14 16:56:39 +00:00
return parent::process_admin_options();
}
}
2013-11-28 13:12:08 +00:00
}