2015-12-18 14:24:34 +00:00
< ? php
2018-03-07 18:29:10 +00:00
/**
* Class WC_Shipping_Legacy_Flat_Rate file .
*
* @ package WooCommerce\Shipping
*/
2015-12-18 14:24:34 +00:00
if ( ! defined ( 'ABSPATH' ) ) {
exit ;
}
/**
* Flat Rate Shipping Method .
*
2017-07-17 10:10:52 +00:00
* This class is here for backwards compatibility for methods existing before zones existed .
2015-12-18 14:24:34 +00:00
*
* @ deprecated 2.6 . 0
2018-03-07 18:04:56 +00:00
* @ version 2.4 . 0
2020-08-05 16:36:24 +00:00
* @ package WooCommerce\Classes\Shipping
2015-12-18 14:24:34 +00:00
*/
class WC_Shipping_Legacy_Flat_Rate extends WC_Shipping_Method {
2018-03-07 18:29:10 +00:00
/**
* Cost passed to [ fee ] shortcode .
*
* @ var string
*/
2015-12-18 14:24:34 +00:00
protected $fee_cost = '' ;
/**
* Constructor .
*/
public function __construct () {
2018-03-07 18:29:10 +00:00
$this -> id = 'legacy_flat_rate' ;
$this -> method_title = __ ( 'Flat rate (legacy)' , 'woocommerce' );
/* translators: %s: Admin shipping settings URL */
2016-10-24 23:56:38 +00:00
$this -> method_description = '<strong>' . sprintf ( __ ( 'This method is deprecated in 2.6.0 and will be removed in future versions - we recommend disabling it and instead setting up a new rate within your <a href="%s">Shipping zones</a>.' , 'woocommerce' ), admin_url ( 'admin.php?page=wc-settings&tab=shipping' ) ) . '</strong>' ;
2015-12-18 14:24:34 +00:00
$this -> init ();
add_action ( 'woocommerce_update_options_shipping_' . $this -> id , array ( $this , 'process_admin_options' ) );
2016-04-21 12:59:13 +00:00
add_action ( 'woocommerce_flat_rate_shipping_add_rate' , array ( $this , 'calculate_extra_shipping' ), 10 , 2 );
2015-12-18 14:24:34 +00:00
}
2016-04-28 11:36:49 +00:00
/**
* Process and redirect if disabled .
*/
public function process_admin_options () {
parent :: process_admin_options ();
2016-08-27 03:23:21 +00:00
if ( 'no' === $this -> settings [ 'enabled' ] ) {
2016-04-28 11:36:49 +00:00
wp_redirect ( admin_url ( 'admin.php?page=wc-settings&tab=shipping§ion=options' ) );
exit ;
}
}
2015-12-18 14:24:34 +00:00
/**
* Return the name of the option in the WP DB .
2018-03-07 18:04:56 +00:00
*
2015-12-18 14:24:34 +00:00
* @ since 2.6 . 0
* @ return string
*/
public function get_option_key () {
2018-03-07 18:29:10 +00:00
return $this -> plugin_id . 'flat_rate_settings' ;
2015-12-18 14:24:34 +00:00
}
/**
2018-03-07 18:29:10 +00:00
* Init function .
2015-12-18 14:24:34 +00:00
*/
public function init () {
// Load the settings.
$this -> init_form_fields ();
$this -> init_settings ();
2018-03-07 18:29:10 +00:00
// Define user set variables.
2015-12-18 14:24:34 +00:00
$this -> title = $this -> get_option ( 'title' );
$this -> availability = $this -> get_option ( 'availability' );
$this -> countries = $this -> get_option ( 'countries' );
$this -> tax_status = $this -> get_option ( 'tax_status' );
$this -> cost = $this -> get_option ( 'cost' );
$this -> type = $this -> get_option ( 'type' , 'class' );
2020-07-16 20:13:08 +00:00
$this -> options = $this -> get_option ( 'options' , false ); // @deprecated 2.4.0
2015-12-18 14:24:34 +00:00
}
/**
* Initialise Settings Form Fields .
*/
public function init_form_fields () {
2018-03-07 18:04:56 +00:00
$this -> form_fields = include 'includes/settings-flat-rate.php' ;
2015-12-18 14:24:34 +00:00
}
/**
* Evaluate a cost from a sum / string .
2018-03-07 18:04:56 +00:00
*
2018-03-07 18:29:10 +00:00
* @ param string $sum Sum to evaluate .
* @ param array $args Arguments .
2015-12-18 14:24:34 +00:00
* @ return string
*/
protected function evaluate_cost ( $sum , $args = array () ) {
2018-03-07 18:04:56 +00:00
include_once WC () -> plugin_path () . '/includes/libraries/class-wc-eval-math.php' ;
2015-12-18 14:24:34 +00:00
$locale = localeconv ();
$decimals = array ( wc_get_price_decimal_separator (), $locale [ 'decimal_point' ], $locale [ 'mon_decimal_point' ] );
$this -> fee_cost = $args [ 'cost' ];
2018-03-07 18:29:10 +00:00
// Expand shortcodes.
2015-12-18 14:24:34 +00:00
add_shortcode ( 'fee' , array ( $this , 'fee' ) );
2018-03-07 18:04:56 +00:00
$sum = do_shortcode (
str_replace (
array (
'[qty]' ,
'[cost]' ,
),
array (
$args [ 'qty' ],
$args [ 'cost' ],
),
$sum
)
);
2015-12-18 14:24:34 +00:00
remove_shortcode ( 'fee' , array ( $this , 'fee' ) );
2018-03-07 18:29:10 +00:00
// Remove whitespace from string.
2015-12-18 14:24:34 +00:00
$sum = preg_replace ( '/\s+/' , '' , $sum );
2018-03-07 18:29:10 +00:00
// Remove locale from string.
2015-12-18 14:24:34 +00:00
$sum = str_replace ( $decimals , '.' , $sum );
2018-03-07 18:29:10 +00:00
// Trim invalid start/end characters.
2015-12-18 14:24:34 +00:00
$sum = rtrim ( ltrim ( $sum , " \t \n \r \0 \x0B +*/ " ), " \t \n \r \0 \x0B +-*/ " );
2018-03-07 18:29:10 +00:00
// Do the math.
2015-12-18 14:24:34 +00:00
return $sum ? WC_Eval_Math :: evaluate ( $sum ) : 0 ;
}
/**
* Work out fee ( shortcode ) .
2018-03-07 18:04:56 +00:00
*
2018-03-07 18:29:10 +00:00
* @ param array $atts Shortcode attributes .
2015-12-18 14:24:34 +00:00
* @ return string
*/
public function fee ( $atts ) {
2018-03-07 18:04:56 +00:00
$atts = shortcode_atts (
array (
'percent' => '' ,
'min_fee' => '' ,
2019-12-20 18:25:23 +00:00
),
$atts ,
'fee'
2018-03-07 18:04:56 +00:00
);
2015-12-18 14:24:34 +00:00
$calculated_fee = 0 ;
if ( $atts [ 'percent' ] ) {
$calculated_fee = $this -> fee_cost * ( floatval ( $atts [ 'percent' ] ) / 100 );
}
if ( $atts [ 'min_fee' ] && $calculated_fee < $atts [ 'min_fee' ] ) {
$calculated_fee = $atts [ 'min_fee' ];
}
return $calculated_fee ;
}
/**
2018-03-07 18:29:10 +00:00
* Calculate shipping .
2015-12-18 14:24:34 +00:00
*
2018-03-07 18:29:10 +00:00
* @ param array $package ( default : array ()) .
2015-12-18 14:24:34 +00:00
*/
public function calculate_shipping ( $package = array () ) {
$rate = array (
2016-04-20 14:03:10 +00:00
'id' => $this -> id ,
'label' => $this -> title ,
'cost' => 0 ,
'package' => $package ,
2015-12-18 14:24:34 +00:00
);
2018-03-07 18:29:10 +00:00
// Calculate the costs.
2015-12-18 14:24:34 +00:00
$has_costs = false ; // True when a cost is set. False if all costs are blank strings.
$cost = $this -> get_option ( 'cost' );
2016-09-09 00:14:28 +00:00
if ( '' !== $cost ) {
2015-12-18 14:24:34 +00:00
$has_costs = true ;
2018-03-07 18:04:56 +00:00
$rate [ 'cost' ] = $this -> evaluate_cost (
2019-12-20 18:25:23 +00:00
$cost ,
array (
2018-03-07 18:04:56 +00:00
'qty' => $this -> get_package_item_qty ( $package ),
'cost' => $package [ 'contents_cost' ],
)
);
2015-12-18 14:24:34 +00:00
}
2018-03-07 18:29:10 +00:00
// Add shipping class costs.
2015-12-18 14:24:34 +00:00
$found_shipping_classes = $this -> find_shipping_classes ( $package );
$highest_class_cost = 0 ;
foreach ( $found_shipping_classes as $shipping_class => $products ) {
2018-03-07 18:29:10 +00:00
// Also handles BW compatibility when slugs were used instead of ids.
2015-12-18 14:24:34 +00:00
$shipping_class_term = get_term_by ( 'slug' , $shipping_class , 'product_shipping_class' );
$class_cost_string = $shipping_class_term && $shipping_class_term -> term_id ? $this -> get_option ( 'class_cost_' . $shipping_class_term -> term_id , $this -> get_option ( 'class_cost_' . $shipping_class , '' ) ) : $this -> get_option ( 'no_class_cost' , '' );
2016-09-09 00:14:28 +00:00
if ( '' === $class_cost_string ) {
2015-12-18 14:24:34 +00:00
continue ;
}
$has_costs = true ;
2018-03-07 18:04:56 +00:00
$class_cost = $this -> evaluate_cost (
2019-12-20 18:25:23 +00:00
$class_cost_string ,
array (
2018-03-07 18:04:56 +00:00
'qty' => array_sum ( wp_list_pluck ( $products , 'quantity' ) ),
'cost' => array_sum ( wp_list_pluck ( $products , 'line_total' ) ),
)
);
2015-12-18 14:24:34 +00:00
2016-09-09 00:14:28 +00:00
if ( 'class' === $this -> type ) {
2015-12-18 14:24:34 +00:00
$rate [ 'cost' ] += $class_cost ;
} else {
$highest_class_cost = $class_cost > $highest_class_cost ? $class_cost : $highest_class_cost ;
}
}
2016-09-09 00:14:28 +00:00
if ( 'order' === $this -> type && $highest_class_cost ) {
2015-12-18 14:24:34 +00:00
$rate [ 'cost' ] += $highest_class_cost ;
}
2016-04-20 07:59:19 +00:00
$rate [ 'package' ] = $package ;
2018-03-07 18:29:10 +00:00
// Add the rate.
2015-12-18 14:24:34 +00:00
if ( $has_costs ) {
2016-04-20 07:59:19 +00:00
$this -> add_rate ( $rate );
2015-12-18 14:24:34 +00:00
}
/**
* Developers can add additional flat rates based on this one via this action since @ version 2.4 .
*
* Previously there were ( overly complex ) options to add additional rates however this was not user .
* friendly and goes against what Flat Rate Shipping was originally intended for .
*
* This example shows how you can add an extra rate based on this flat rate via custom function :
*
2018-03-07 18:04:56 +00:00
* add_action ( 'woocommerce_flat_rate_shipping_add_rate' , 'add_another_custom_flat_rate' , 10 , 2 );
2015-12-18 14:24:34 +00:00
*
2018-03-07 18:04:56 +00:00
* function add_another_custom_flat_rate ( $method , $rate ) {
* $new_rate = $rate ;
* $new_rate [ 'id' ] .= ':' . 'custom_rate_name' ; // Append a custom ID.
* $new_rate [ 'label' ] = 'Rushed Shipping' ; // Rename to 'Rushed Shipping'.
* $new_rate [ 'cost' ] += 2 ; // Add $2 to the cost.
2015-12-18 14:24:34 +00:00
*
2018-03-07 18:04:56 +00:00
* // Add it to WC.
* $method -> add_rate ( $new_rate );
* } .
2015-12-18 14:24:34 +00:00
*/
2016-04-20 07:59:19 +00:00
do_action ( 'woocommerce_flat_rate_shipping_add_rate' , $this , $rate );
2015-12-18 14:24:34 +00:00
}
/**
* Get items in package .
2018-03-07 18:04:56 +00:00
*
2018-03-07 18:29:10 +00:00
* @ param array $package Package information .
2015-12-18 14:24:34 +00:00
* @ return int
*/
public function get_package_item_qty ( $package ) {
$total_quantity = 0 ;
foreach ( $package [ 'contents' ] as $item_id => $values ) {
if ( $values [ 'quantity' ] > 0 && $values [ 'data' ] -> needs_shipping () ) {
$total_quantity += $values [ 'quantity' ];
}
}
return $total_quantity ;
}
/**
* Finds and returns shipping classes and the products with said class .
2018-03-07 18:04:56 +00:00
*
2018-03-07 18:29:10 +00:00
* @ param mixed $package Package information .
2015-12-18 14:24:34 +00:00
* @ return array
*/
public function find_shipping_classes ( $package ) {
$found_shipping_classes = array ();
foreach ( $package [ 'contents' ] as $item_id => $values ) {
if ( $values [ 'data' ] -> needs_shipping () ) {
$found_class = $values [ 'data' ] -> get_shipping_class ();
if ( ! isset ( $found_shipping_classes [ $found_class ] ) ) {
$found_shipping_classes [ $found_class ] = array ();
}
$found_shipping_classes [ $found_class ][ $item_id ] = $values ;
}
}
return $found_shipping_classes ;
}
/**
* Adds extra calculated flat rates .
*
* @ deprecated 2.4 . 0
*
2017-03-28 17:58:51 +00:00
* Additional rates defined like this :
2018-03-07 18:04:56 +00:00
* Option Name | Additional Cost [ +- Percents % ] | Per Cost Type ( order , class , or item ) .
2018-03-07 18:29:10 +00:00
*
* @ param null $method Deprecated .
* @ param array $rate Rate information .
2015-12-18 14:24:34 +00:00
*/
2016-04-21 12:59:13 +00:00
public function calculate_extra_shipping ( $method , $rate ) {
2015-12-18 14:24:34 +00:00
if ( $this -> options ) {
$options = array_filter ( ( array ) explode ( " \n " , $this -> options ) );
foreach ( $options as $option ) {
$this_option = array_map ( 'trim' , explode ( WC_DELIMITER , $option ) );
2018-03-07 18:29:10 +00:00
if ( count ( $this_option ) !== 3 ) {
2015-12-18 14:24:34 +00:00
continue ;
}
$extra_rate = $rate ;
$extra_rate [ 'id' ] = $this -> id . ':' . urldecode ( sanitize_title ( $this_option [ 0 ] ) );
$extra_rate [ 'label' ] = $this_option [ 0 ];
2016-04-26 13:21:42 +00:00
$extra_cost = $this -> get_extra_cost ( $this_option [ 1 ], $this_option [ 2 ], $rate [ 'package' ] );
2015-12-18 14:24:34 +00:00
if ( is_array ( $extra_rate [ 'cost' ] ) ) {
$extra_rate [ 'cost' ][ 'order' ] = $extra_rate [ 'cost' ][ 'order' ] + $extra_cost ;
} else {
$extra_rate [ 'cost' ] += $extra_cost ;
}
2016-04-20 07:59:19 +00:00
$this -> add_rate ( $extra_rate );
2015-12-18 14:24:34 +00:00
}
}
}
/**
* Calculate the percentage adjustment for each shipping rate .
*
* @ deprecated 2.4 . 0
2018-03-07 18:29:10 +00:00
* @ param float $cost Cost .
* @ param float $percent_adjustment Percent adjusment .
* @ param string $percent_operator Percent operator .
* @ param float $base_price Base price .
2015-12-18 14:24:34 +00:00
* @ return float
*/
public function calc_percentage_adjustment ( $cost , $percent_adjustment , $percent_operator , $base_price ) {
2018-03-07 18:29:10 +00:00
if ( '+' === $percent_operator ) {
2015-12-18 14:24:34 +00:00
$cost += $percent_adjustment * $base_price ;
} else {
$cost -= $percent_adjustment * $base_price ;
}
return $cost ;
}
/**
* Get extra cost .
*
* @ deprecated 2.4 . 0
2018-03-07 18:29:10 +00:00
* @ param string $cost_string Cost string .
* @ param string $type Type .
* @ param array $package Package information .
2015-12-18 14:24:34 +00:00
* @ return float
*/
public function get_extra_cost ( $cost_string , $type , $package ) {
$cost = $cost_string ;
$cost_percent = false ;
2016-09-01 20:50:14 +00:00
// @codingStandardsIgnoreStart
2015-12-18 14:24:34 +00:00
$pattern =
2016-09-01 20:50:14 +00:00
'/' . // Start regex.
'(\d+\.?\d*)' . // Capture digits, optionally capture a `.` and more digits.
'\s*' . // Match whitespace.
'(\+|-)' . // Capture the operand.
'\s*' . // Match whitespace.
'(\d+\.?\d*)' . // Capture digits, optionally capture a `.` and more digits.
'\%/' ; // Match the percent sign & end regex.
// @codingStandardsIgnoreEnd
2015-12-18 14:24:34 +00:00
if ( preg_match ( $pattern , $cost_string , $this_cost_matches ) ) {
$cost_operator = $this_cost_matches [ 2 ];
$cost_percent = $this_cost_matches [ 3 ] / 100 ;
$cost = $this_cost_matches [ 1 ];
}
switch ( $type ) {
2018-03-07 18:04:56 +00:00
case 'class' :
2018-03-07 18:29:10 +00:00
$cost = $cost * count ( $this -> find_shipping_classes ( $package ) );
2018-03-07 18:04:56 +00:00
break ;
case 'item' :
2015-12-18 14:24:34 +00:00
$cost = $cost * $this -> get_package_item_qty ( $package );
2018-03-07 18:04:56 +00:00
break ;
2015-12-18 14:24:34 +00:00
}
if ( $cost_percent ) {
switch ( $type ) {
2018-03-07 18:04:56 +00:00
case 'class' :
2015-12-18 14:24:34 +00:00
$shipping_classes = $this -> find_shipping_classes ( $package );
2016-08-27 04:47:24 +00:00
foreach ( $shipping_classes as $shipping_class => $items ) {
2015-12-18 14:24:34 +00:00
foreach ( $items as $item_id => $values ) {
$cost = $this -> calc_percentage_adjustment ( $cost , $cost_percent , $cost_operator , $values [ 'line_total' ] );
}
}
2018-03-07 18:04:56 +00:00
break ;
case 'item' :
2015-12-18 14:24:34 +00:00
foreach ( $package [ 'contents' ] as $item_id => $values ) {
if ( $values [ 'data' ] -> needs_shipping () ) {
$cost = $this -> calc_percentage_adjustment ( $cost , $cost_percent , $cost_operator , $values [ 'line_total' ] );
}
}
2018-03-07 18:04:56 +00:00
break ;
case 'order' :
2015-12-18 14:24:34 +00:00
$cost = $this -> calc_percentage_adjustment ( $cost , $cost_percent , $cost_operator , $package [ 'contents_cost' ] );
2018-03-07 18:04:56 +00:00
break ;
2015-12-18 14:24:34 +00:00
}
}
return $cost ;
}
}