PluginProbe ʕ •ᴥ•ʔ
WooCommerce / 5.9.0-rc.2
WooCommerce v5.9.0-rc.2
10.9.0-beta.2 10.9.0-beta.1 10.8.1 10.8.0 10.8.0-rc.1 10.8.0-beta.2 10.8.0-beta.1 7.8.0-beta.1 7.8.0-beta.2 7.8.0-rc.1 7.8.0-rc.2 7.8.1 7.8.2 7.8.3 7.8.4 7.9.0 7.9.0-beta.1 7.9.0-beta.2 7.9.0-rc.2 7.9.0-rc.3 7.9.1 7.9.2 8.0.0 8.0.0-beta.1 8.0.0-beta.2 8.0.0-rc.1 8.0.0-rc.2 8.0.1 8.0.2 8.0.3 8.0.4 8.0.5 8.1.0 8.1.0-beta.1 8.1.0-rc.1 8.1.0-rc.2 8.1.1 8.1.2 8.1.3 8.1.4 8.2.0 8.2.0-beta.1 8.2.0-rc.1 8.2.0-rc.2 8.2.1 8.2.2 8.2.3 8.2.4 8.2.5 8.3.0 8.3.0-beta.1 8.3.0-rc.1 8.3.0-rc.2 8.3.1 8.3.2 8.3.3 8.3.4 8.4.0 8.4.0-beta.1 8.4.0-rc.1 8.4.1 8.4.2 8.4.3 8.5.0 8.5.0-beta.1 8.5.0-rc.1 8.5.1 8.5.2 8.5.3 8.5.4 8.5.5 8.6.0 8.6.0-beta.1 8.6.0-rc.1 8.6.1 8.6.2 8.6.3 8.6.4 8.7.0 8.7.0-beta.1 8.7.0-beta.2 8.7.0-rc.1 8.7.1 8.7.2 8.7.3 8.8.0 8.8.0-beta.1 8.8.0-rc.1 8.8.1 8.8.2 8.8.3 8.8.4 8.8.5 8.8.6 8.8.7 8.9.0 8.9.0-beta.1 8.9.0-rc.1 8.9.1 8.9.2 8.9.3 8.9.4 8.9.5 9.0.0 9.0.0-beta.1 9.0.0-beta.2 9.0.0-rc.1 9.0.1 9.0.2 9.0.3 9.0.4 9.1.0 9.1.0-beta.1 9.1.0-rc.1 9.1.1 9.1.2 9.1.3 9.1.4 9.1.5 9.1.6 9.2.0 9.2.0-beta.1 9.2.0-rc.1 9.2.1 9.2.2 9.2.3 9.2.4 9.2.5 9.3.0 9.3.0-beta.1 9.3.0-rc.1 9.3.1 9.3.2 9.3.3 9.3.4 9.3.5 9.3.6 9.4.0 9.4.0-beta.1 9.4.0-beta.2 9.4.0-rc.1 9.4.0-rc.2 9.4.0-rc.3 9.4.0-rc.4 9.4.1 9.4.2 9.4.3 9.4.4 9.4.5 9.5.0 9.5.0-beta.1 9.5.0-beta.2 9.5.0-rc.1 9.5.1 9.5.2 9.5.3 9.5.4 9.6.0 9.6.0-beta.1 9.6.0-beta.2 9.6.0-rc.1 9.6.1 9.6.2 9.6.3 9.6.4 9.7.0 9.7.0-beta.1 9.7.0-rc.1 9.7.1 9.7.2 9.7.3 9.8.0 9.8.0-beta.1 9.8.0-rc.1 9.8.1 9.8.2 9.8.3 9.8.4 9.8.5 9.8.6 9.8.7 9.9.0 9.9.0-beta.1 9.9.0-rc.1 9.9.1 9.9.2 9.9.3 9.9.4 9.9.5 9.9.6 9.9.7 3.7.3 7.1.2 3.8.0 7.2.0 3.8.0-beta.1 7.2.0-beta.1 3.8.0-rc.1 7.2.0-beta.2 3.8.0-rc.2 7.2.0-rc.1 3.8.1 7.2.0-rc.2 3.8.2 7.2.1 3.8.3 7.2.2 3.9.0 7.2.3 3.9.0-beta.1 7.2.4 3.9.0-beta.2 7.3.0 3.9.0-rc.1 7.3.0-beta.1 3.9.0-rc.2 7.3.0-beta.2 3.9.0-rc.3 7.3.0-rc.1 3.9.0-rc.4 7.3.0-rc.2 3.9.1 7.3.1 3.9.2 7.4.0 3.9.3 7.4.0-beta.1 3.9.4 7.4.0-beta.2 3.9.5 7.4.0-rc.1 4.0.0 7.4.0-rc.2 4.0.0-beta.1 7.4.1 4.0.0-rc.1 7.4.2 4.0.0-rc.2 7.5.0 4.0.1 7.5.0-beta.1 4.0.2 7.5.0-beta.2 4.0.3 7.5.0-rc.1 4.0.4 7.5.1 4.1.0 7.5.2 4.1.0-beta.1 7.6.0 4.1.0-beta.2 7.6.0-beta.1 4.1.0-rc.1 7.6.0-beta.2 4.1.0-rc.2 7.6.0-rc.1 4.1.1 7.6.0-rc.2 4.1.2 7.6.0-rc.3 4.1.3 7.6.1 4.1.4 7.6.2 4.2.0 7.7.0 4.2.0-RC.1 7.7.0-beta.1 4.2.0-RC.2 7.7.0-beta.2 4.2.0-beta.1 7.7.0-rc.1 4.2.1 7.7.1 4.2.2 7.7.2 4.2.3 7.7.3 4.2.4 7.8.0 4.2.5 4.3.0 4.3.0-beta.1 4.3.0-rc.1 4.3.0-rc.2 4.3.0-rc.3 4.3.1 4.3.2 4.3.3 4.3.4 4.3.5 4.3.6 4.4.0 4.4.0-beta.1 4.4.0-rc.1 4.4.1 4.4.2 4.4.3 4.4.4 4.5.0 4.5.0-beta.1 4.5.0-rc.1 4.5.0-rc.3 4.5.1 4.5.2 4.5.3 4.5.4 4.5.5 4.6.0 4.6.0-beta.1 4.6.0-rc.1 4.6.1 4.6.2 4.6.3 4.6.4 4.6.5 4.7.0 4.7.0-beta.1 4.7.0-beta.2 4.7.0-rc.1 4.7.1 4.7.1-beta.1 4.7.2 4.7.3 4.7.4 4.8.0 4.8.0-beta.1 4.8.0-rc.1 4.8.0-rc.2 4.8.1 4.8.2 4.8.3 4.9.0 4.9.0-beta.1 4.9.0-rc.1 4.9.0-rc.2 4.9.1 4.9.2 4.9.3 4.9.4 4.9.5 5.0.0 5.0.0-beta.1 5.0.0-beta.2 5.0.0-rc.1 5.0.0-rc.2 5.0.0-rc.3 5.0.1 5.0.2 5.0.3 5.1.0 5.1.0-beta.1 5.1.0-rc.1 trunk 5.1.1 10.0.0 5.1.2 10.0.0-rc.1 5.1.3 10.0.0-rc.2 5.2.0 10.0.1 5.2.0-beta.1 10.0.2 5.2.0-rc.1 10.0.3 5.2.0-rc.2 10.0.4 5.2.1 10.0.5 5.2.2 10.0.6 5.2.3 10.1.0 5.2.4 10.1.0-rc.1 5.2.5 10.1.0-rc.2 5.3.0 10.1.0-rc.3 5.3.0-beta.1 10.1.0-rc.4 5.3.0-rc.1 10.1.1 5.3.0-rc.2 10.1.2 5.3.1 10.1.3 5.3.2 10.1.4 5.3.3 10.2.0 5.4.0 10.2.0-beta.1 5.4.0-beta.1 10.2.0-beta.2 5.4.0-rc.1 10.2.0-rc.1 5.4.1 10.2.1 5.4.2 10.2.2 5.4.3 10.2.3 5.4.4 10.2.4 5.4.5 10.3.0 5.5.0 10.3.0-beta.1 5.5.0-beta.1 10.3.0-beta.2 5.5.0-rc.1 10.3.0-rc.1 5.5.0-rc.2 10.3.0-rc.2 5.5.1 10.3.1 5.5.2 10.3.2 5.5.3 10.3.3 5.5.4 10.3.4 5.5.5 10.3.5 5.6.0 10.3.6 5.6.0-beta.1 10.3.7 5.6.0-rc.1 10.3.8 5.6.0-rc.2 10.4.0 5.6.1 10.4.0-beta.1 5.6.2 10.4.0-beta.2 5.6.3 10.4.0-rc.1 5.7.0 10.4.1 5.7.0-beta.1 10.4.2 5.7.0-rc.1 10.4.3 5.7.1 10.4.4 5.7.2 10.5.0 5.7.3 10.5.0-beta.1 5.8.0 10.5.0-beta.2 5.8.0-beta.1 10.5.0-rc.1 5.8.0-beta.2 10.5.0-rc.2 5.8.0-rc.1 10.5.0-rc.3 5.8.1 10.5.1 5.8.2 10.5.2 5.9.0 10.5.3 5.9.0-beta.1 10.6.0 5.9.0-rc.1 10.6.0-beta.1 5.9.0-rc.2 10.6.0-beta.2 5.9.1 10.6.0-rc.1 5.9.2 10.6.1 6.0.0 10.6.2 6.0.0-beta.1 10.7.0 6.0.0-rc.1 10.7.0-beta.1 6.0.1 10.7.0-beta.2 6.0.2 10.7.0-rc.1 6.1.0 3.0.0 6.1.0-beta.1 3.0.1 6.1.0-rc.1 3.0.2 6.1.0-rc.2 3.0.3 6.1.1 3.0.4 6.1.2 3.0.5 6.1.3 3.0.6 6.2.0 3.0.7 6.2.0-beta.1 3.0.8 6.2.0-rc.1 3.0.9 6.2.0-rc.2 3.1.0 6.2.1 3.1.1 6.2.2 3.1.2 6.2.3 3.2.0 6.3.0 3.2.1 6.3.0-beta.1 3.2.2 6.3.0-rc.1 3.2.3 6.3.0-rc.2 3.2.4 6.3.1 3.2.5 6.3.2 3.2.6 6.4.0 3.3.0 6.4.0-beta.1 3.3.1 6.4.0-rc.1 3.3.2 6.4.1 3.3.2-rc.1 6.4.2 3.3.3 6.5.0 3.3.4 6.5.0-beta.1 3.3.5 6.5.0-rc.1 3.3.6 6.5.0-rc.2 3.4.0 6.5.1 3.4.0-beta.1 6.5.2 3.4.0-rc.2 6.6.0 3.4.1 6.6.0-beta.1 3.4.2 6.6.0-rc.1 3.4.3 6.6.0-rc.2 3.4.4 6.6.1 3.4.5 6.6.2 3.4.6 6.7.0 3.4.7 6.7.0-beta.1 3.4.8 6.7.0-beta.2 3.5.0 6.7.0-rc.1 3.5.0-beta.1 6.7.1 3.5.0-rc.1 6.8.0 3.5.0-rc.2 6.8.0-beta.1 3.5.1 6.8.0-beta.2 3.5.10 6.8.0-rc.1 3.5.2 6.8.1 3.5.3 6.8.2 3.5.4 6.8.3 3.5.5 6.9.0 3.5.6 6.9.0-beta.1 3.5.7 6.9.0-beta.2 3.5.8 6.9.0-rc.1 3.5.9 6.9.1 3.6.0 6.9.2 3.6.0-beta.1 6.9.3 3.6.0-rc.1 6.9.4 3.6.0-rc.2 6.9.5 3.6.0-rc.3 7.0.0 3.6.1 7.0.0-beta.1 3.6.2 7.0.0-beta.2 3.6.3 7.0.0-beta.3 3.6.4 7.0.0-rc.1 3.6.5 7.0.0-rc.2 3.6.6 7.0.1 3.6.7 7.0.2 3.7.0 7.1.0 3.7.0-beta.1 7.1.0-beta.1 3.7.0-rc.1 7.1.0-beta.2 3.7.0-rc.2 7.1.0-rc.1 3.7.1 7.1.0-rc.2 3.7.2 7.1.1
woocommerce / includes / abstracts / abstract-wc-shipping-method.php
woocommerce / includes / abstracts Last commit date
abstract-wc-data.php 5 years ago abstract-wc-deprecated-hooks.php 6 years ago abstract-wc-integration.php 5 years ago abstract-wc-log-handler.php 5 years ago abstract-wc-object-query.php 5 years ago abstract-wc-order.php 4 years ago abstract-wc-payment-gateway.php 5 years ago abstract-wc-payment-token.php 5 years ago abstract-wc-privacy.php 5 years ago abstract-wc-product.php 4 years ago abstract-wc-session.php 5 years ago abstract-wc-settings-api.php 5 years ago abstract-wc-shipping-method.php 5 years ago abstract-wc-widget.php 4 years ago class-wc-background-process.php 5 years ago
abstract-wc-shipping-method.php
570 lines
1 <?php
2 /**
3 * Abstract shipping method
4 *
5 * @class WC_Shipping_Method
6 * @package WooCommerce\Abstracts
7 */
8
9 if ( ! defined( 'ABSPATH' ) ) {
10 exit;
11 }
12
13 /**
14 * WooCommerce Shipping Method Class.
15 *
16 * Extended by shipping methods to handle shipping calculations etc.
17 *
18 * @class WC_Shipping_Method
19 * @version 3.0.0
20 * @package WooCommerce\Abstracts
21 */
22 abstract class WC_Shipping_Method extends WC_Settings_API {
23
24 /**
25 * Features this method supports. Possible features used by core:
26 * - shipping-zones Shipping zone functionality + instances
27 * - instance-settings Instance settings screens.
28 * - settings Non-instance settings screens. Enabled by default for BW compatibility with methods before instances existed.
29 * - instance-settings-modal Allows the instance settings to be loaded within a modal in the zones UI.
30 *
31 * @var array
32 */
33 public $supports = array( 'settings' );
34
35 /**
36 * Unique ID for the shipping method - must be set.
37 *
38 * @var string
39 */
40 public $id = '';
41
42 /**
43 * Method title.
44 *
45 * @var string
46 */
47 public $method_title = '';
48
49 /**
50 * Method description.
51 *
52 * @var string
53 */
54 public $method_description = '';
55
56 /**
57 * Yes or no based on whether the method is enabled.
58 *
59 * @var string
60 */
61 public $enabled = 'yes';
62
63 /**
64 * Shipping method title for the frontend.
65 *
66 * @var string
67 */
68 public $title;
69
70 /**
71 * This is an array of rates - methods must populate this array to register shipping costs.
72 *
73 * @var array
74 */
75 public $rates = array();
76
77 /**
78 * If 'taxable' tax will be charged for this method (if applicable).
79 *
80 * @var string
81 */
82 public $tax_status = 'taxable';
83
84 /**
85 * Fee for the method (if applicable).
86 *
87 * @var string
88 */
89 public $fee = null;
90
91 /**
92 * Minimum fee for the method (if applicable).
93 *
94 * @var string
95 */
96 public $minimum_fee = null;
97
98 /**
99 * Instance ID if used.
100 *
101 * @var int
102 */
103 public $instance_id = 0;
104
105 /**
106 * Instance form fields.
107 *
108 * @var array
109 */
110 public $instance_form_fields = array();
111
112 /**
113 * Instance settings.
114 *
115 * @var array
116 */
117 public $instance_settings = array();
118
119 /**
120 * Availability - legacy. Used for method Availability.
121 * No longer useful for instance based shipping methods.
122 *
123 * @deprecated 2.6.0
124 * @var string
125 */
126 public $availability;
127
128 /**
129 * Availability countries - legacy. Used for method Availability.
130 * No longer useful for instance based shipping methods.
131 *
132 * @deprecated 2.6.0
133 * @var array
134 */
135 public $countries = array();
136
137 /**
138 * Constructor.
139 *
140 * @param int $instance_id Instance ID.
141 */
142 public function __construct( $instance_id = 0 ) {
143 $this->instance_id = absint( $instance_id );
144 }
145
146 /**
147 * Check if a shipping method supports a given feature.
148 *
149 * Methods should override this to declare support (or lack of support) for a feature.
150 *
151 * @param string $feature The name of a feature to test support for.
152 * @return bool True if the shipping method supports the feature, false otherwise.
153 */
154 public function supports( $feature ) {
155 return apply_filters( 'woocommerce_shipping_method_supports', in_array( $feature, $this->supports ), $feature, $this );
156 }
157
158 /**
159 * Called to calculate shipping rates for this method. Rates can be added using the add_rate() method.
160 *
161 * @param array $package Package array.
162 */
163 public function calculate_shipping( $package = array() ) {}
164
165 /**
166 * Whether or not we need to calculate tax on top of the shipping rate.
167 *
168 * @return boolean
169 */
170 public function is_taxable() {
171 return wc_tax_enabled() && 'taxable' === $this->tax_status && ( WC()->customer && ! WC()->customer->get_is_vat_exempt() );
172 }
173
174 /**
175 * Whether or not this method is enabled in settings.
176 *
177 * @since 2.6.0
178 * @return boolean
179 */
180 public function is_enabled() {
181 return 'yes' === $this->enabled;
182 }
183
184 /**
185 * Return the shipping method instance ID.
186 *
187 * @since 2.6.0
188 * @return int
189 */
190 public function get_instance_id() {
191 return $this->instance_id;
192 }
193
194 /**
195 * Return the shipping method title.
196 *
197 * @since 2.6.0
198 * @return string
199 */
200 public function get_method_title() {
201 return apply_filters( 'woocommerce_shipping_method_title', $this->method_title, $this );
202 }
203
204 /**
205 * Return the shipping method description.
206 *
207 * @since 2.6.0
208 * @return string
209 */
210 public function get_method_description() {
211 return apply_filters( 'woocommerce_shipping_method_description', $this->method_description, $this );
212 }
213
214 /**
215 * Return the shipping title which is user set.
216 *
217 * @return string
218 */
219 public function get_title() {
220 return apply_filters( 'woocommerce_shipping_method_title', $this->title, $this->id );
221 }
222
223 /**
224 * Return calculated rates for a package.
225 *
226 * @since 2.6.0
227 * @param array $package Package array.
228 * @return array
229 */
230 public function get_rates_for_package( $package ) {
231 $this->rates = array();
232 if ( $this->is_available( $package ) && ( empty( $package['ship_via'] ) || in_array( $this->id, $package['ship_via'] ) ) ) {
233 $this->calculate_shipping( $package );
234 }
235 return $this->rates;
236 }
237
238 /**
239 * Returns a rate ID based on this methods ID and instance, with an optional
240 * suffix if distinguishing between multiple rates.
241 *
242 * @since 2.6.0
243 * @param string $suffix Suffix.
244 * @return string
245 */
246 public function get_rate_id( $suffix = '' ) {
247 $rate_id = array( $this->id );
248
249 if ( $this->instance_id ) {
250 $rate_id[] = $this->instance_id;
251 }
252
253 if ( $suffix ) {
254 $rate_id[] = $suffix;
255 }
256
257 return implode( ':', $rate_id );
258 }
259
260 /**
261 * Add a shipping rate. If taxes are not set they will be calculated based on cost.
262 *
263 * @param array $args Arguments (default: array()).
264 */
265 public function add_rate( $args = array() ) {
266 $args = apply_filters(
267 'woocommerce_shipping_method_add_rate_args',
268 wp_parse_args(
269 $args,
270 array(
271 'id' => $this->get_rate_id(), // ID for the rate. If not passed, this id:instance default will be used.
272 'label' => '', // Label for the rate.
273 'cost' => '0', // Amount or array of costs (per item shipping).
274 'taxes' => '', // Pass taxes, or leave empty to have it calculated for you, or 'false' to disable calculations.
275 'calc_tax' => 'per_order', // Calc tax per_order or per_item. Per item needs an array of costs.
276 'meta_data' => array(), // Array of misc meta data to store along with this rate - key value pairs.
277 'package' => false, // Package array this rate was generated for @since 2.6.0.
278 'price_decimals' => wc_get_price_decimals(),
279 )
280 ),
281 $this
282 );
283
284 // ID and label are required.
285 if ( ! $args['id'] || ! $args['label'] ) {
286 return;
287 }
288
289 // Total up the cost.
290 $total_cost = is_array( $args['cost'] ) ? array_sum( $args['cost'] ) : $args['cost'];
291 $taxes = $args['taxes'];
292
293 // Taxes - if not an array and not set to false, calc tax based on cost and passed calc_tax variable. This saves shipping methods having to do complex tax calculations.
294 if ( ! is_array( $taxes ) && false !== $taxes && $total_cost > 0 && $this->is_taxable() ) {
295 $taxes = 'per_item' === $args['calc_tax'] ? $this->get_taxes_per_item( $args['cost'] ) : WC_Tax::calc_shipping_tax( $total_cost, WC_Tax::get_shipping_tax_rates() );
296 }
297
298 // Round the total cost after taxes have been calculated.
299 $total_cost = wc_format_decimal( $total_cost, $args['price_decimals'] );
300
301 // Create rate object.
302 $rate = new WC_Shipping_Rate();
303 $rate->set_id( $args['id'] );
304 $rate->set_method_id( $this->id );
305 $rate->set_instance_id( $this->instance_id );
306 $rate->set_label( $args['label'] );
307 $rate->set_cost( $total_cost );
308 $rate->set_taxes( $taxes );
309
310 if ( ! empty( $args['meta_data'] ) ) {
311 foreach ( $args['meta_data'] as $key => $value ) {
312 $rate->add_meta_data( $key, $value );
313 }
314 }
315
316 // Store package data.
317 if ( $args['package'] ) {
318 $items_in_package = array();
319 foreach ( $args['package']['contents'] as $item ) {
320 $product = $item['data'];
321 $items_in_package[] = $product->get_name() . ' &times; ' . $item['quantity'];
322 }
323 $rate->add_meta_data( __( 'Items', 'woocommerce' ), implode( ', ', $items_in_package ) );
324 }
325
326 $this->rates[ $args['id'] ] = apply_filters( 'woocommerce_shipping_method_add_rate', $rate, $args, $this );
327 }
328
329 /**
330 * Calc taxes per item being shipping in costs array.
331 *
332 * @since 2.6.0
333 * @param array $costs Costs.
334 * @return array of taxes
335 */
336 protected function get_taxes_per_item( $costs ) {
337 $taxes = array();
338
339 // If we have an array of costs we can look up each items tax class and add tax accordingly.
340 if ( is_array( $costs ) ) {
341
342 $cart = WC()->cart->get_cart();
343
344 foreach ( $costs as $cost_key => $amount ) {
345 if ( ! isset( $cart[ $cost_key ] ) ) {
346 continue;
347 }
348
349 $item_taxes = WC_Tax::calc_shipping_tax( $amount, WC_Tax::get_shipping_tax_rates( $cart[ $cost_key ]['data']->get_tax_class() ) );
350
351 // Sum the item taxes.
352 foreach ( array_keys( $taxes + $item_taxes ) as $key ) {
353 $taxes[ $key ] = ( isset( $item_taxes[ $key ] ) ? $item_taxes[ $key ] : 0 ) + ( isset( $taxes[ $key ] ) ? $taxes[ $key ] : 0 );
354 }
355 }
356
357 // Add any cost for the order - order costs are in the key 'order'.
358 if ( isset( $costs['order'] ) ) {
359 $item_taxes = WC_Tax::calc_shipping_tax( $costs['order'], WC_Tax::get_shipping_tax_rates() );
360
361 // Sum the item taxes.
362 foreach ( array_keys( $taxes + $item_taxes ) as $key ) {
363 $taxes[ $key ] = ( isset( $item_taxes[ $key ] ) ? $item_taxes[ $key ] : 0 ) + ( isset( $taxes[ $key ] ) ? $taxes[ $key ] : 0 );
364 }
365 }
366 }
367
368 return $taxes;
369 }
370
371 /**
372 * Is this method available?
373 *
374 * @param array $package Package.
375 * @return bool
376 */
377 public function is_available( $package ) {
378 $available = $this->is_enabled();
379
380 // Country availability (legacy, for non-zone based methods).
381 if ( ! $this->instance_id && $available ) {
382 $countries = is_array( $this->countries ) ? $this->countries : array();
383
384 switch ( $this->availability ) {
385 case 'specific':
386 case 'including':
387 $available = in_array( $package['destination']['country'], array_intersect( $countries, array_keys( WC()->countries->get_shipping_countries() ) ) );
388 break;
389 case 'excluding':
390 $available = in_array( $package['destination']['country'], array_diff( array_keys( WC()->countries->get_shipping_countries() ), $countries ) );
391 break;
392 default:
393 $available = in_array( $package['destination']['country'], array_keys( WC()->countries->get_shipping_countries() ) );
394 break;
395 }
396 }
397
398 return apply_filters( 'woocommerce_shipping_' . $this->id . '_is_available', $available, $package, $this );
399 }
400
401 /**
402 * Get fee to add to shipping cost.
403 *
404 * @param string|float $fee Fee.
405 * @param float $total Total.
406 * @return float
407 */
408 public function get_fee( $fee, $total ) {
409 if ( strstr( $fee, '%' ) ) {
410 $fee = ( $total / 100 ) * str_replace( '%', '', $fee );
411 }
412 if ( ! empty( $this->minimum_fee ) && $this->minimum_fee > $fee ) {
413 $fee = $this->minimum_fee;
414 }
415 return $fee;
416 }
417
418 /**
419 * Does this method have a settings page?
420 *
421 * @return bool
422 */
423 public function has_settings() {
424 return $this->instance_id ? $this->supports( 'instance-settings' ) : $this->supports( 'settings' );
425 }
426
427 /**
428 * Return admin options as a html string.
429 *
430 * @return string
431 */
432 public function get_admin_options_html() {
433 if ( $this->instance_id ) {
434 $settings_html = $this->generate_settings_html( $this->get_instance_form_fields(), false );
435 } else {
436 $settings_html = $this->generate_settings_html( $this->get_form_fields(), false );
437 }
438
439 return '<table class="form-table">' . $settings_html . '</table>';
440 }
441
442 /**
443 * Output the shipping settings screen.
444 */
445 public function admin_options() {
446 if ( ! $this->instance_id ) {
447 echo '<h2>' . esc_html( $this->get_method_title() ) . '</h2>';
448 }
449 echo wp_kses_post( wpautop( $this->get_method_description() ) );
450 echo $this->get_admin_options_html(); // phpcs:ignore WordPress.XSS.EscapeOutput.OutputNotEscaped
451 }
452
453 /**
454 * Get_option function.
455 *
456 * Gets and option from the settings API, using defaults if necessary to prevent undefined notices.
457 *
458 * @param string $key Key.
459 * @param mixed $empty_value Empty value.
460 * @return mixed The value specified for the option or a default value for the option.
461 */
462 public function get_option( $key, $empty_value = null ) {
463 // Instance options take priority over global options.
464 if ( $this->instance_id && array_key_exists( $key, $this->get_instance_form_fields() ) ) {
465 return $this->get_instance_option( $key, $empty_value );
466 }
467
468 // Return global option.
469 $option = apply_filters( 'woocommerce_shipping_' . $this->id . '_option', parent::get_option( $key, $empty_value ), $key, $this );
470 return $option;
471 }
472
473 /**
474 * Gets an option from the settings API, using defaults if necessary to prevent undefined notices.
475 *
476 * @param string $key Key.
477 * @param mixed $empty_value Empty value.
478 * @return mixed The value specified for the option or a default value for the option.
479 */
480 public function get_instance_option( $key, $empty_value = null ) {
481 if ( empty( $this->instance_settings ) ) {
482 $this->init_instance_settings();
483 }
484
485 // Get option default if unset.
486 if ( ! isset( $this->instance_settings[ $key ] ) ) {
487 $form_fields = $this->get_instance_form_fields();
488 $this->instance_settings[ $key ] = $this->get_field_default( $form_fields[ $key ] );
489 }
490
491 if ( ! is_null( $empty_value ) && '' === $this->instance_settings[ $key ] ) {
492 $this->instance_settings[ $key ] = $empty_value;
493 }
494
495 $instance_option = apply_filters( 'woocommerce_shipping_' . $this->id . '_instance_option', $this->instance_settings[ $key ], $key, $this );
496 return $instance_option;
497 }
498
499 /**
500 * Get settings fields for instances of this shipping method (within zones).
501 * Should be overridden by shipping methods to add options.
502 *
503 * @since 2.6.0
504 * @return array
505 */
506 public function get_instance_form_fields() {
507 return apply_filters( 'woocommerce_shipping_instance_form_fields_' . $this->id, array_map( array( $this, 'set_defaults' ), $this->instance_form_fields ) );
508 }
509
510 /**
511 * Return the name of the option in the WP DB.
512 *
513 * @since 2.6.0
514 * @return string
515 */
516 public function get_instance_option_key() {
517 return $this->instance_id ? $this->plugin_id . $this->id . '_' . $this->instance_id . '_settings' : '';
518 }
519
520 /**
521 * Initialise Settings for instances.
522 *
523 * @since 2.6.0
524 */
525 public function init_instance_settings() {
526 $this->instance_settings = get_option( $this->get_instance_option_key(), null );
527
528 // If there are no settings defined, use defaults.
529 if ( ! is_array( $this->instance_settings ) ) {
530 $form_fields = $this->get_instance_form_fields();
531 $this->instance_settings = array_merge( array_fill_keys( array_keys( $form_fields ), '' ), wp_list_pluck( $form_fields, 'default' ) );
532 }
533 }
534
535 /**
536 * Processes and saves global shipping method options in the admin area.
537 *
538 * This method is usually attached to woocommerce_update_options_x hooks.
539 *
540 * @since 2.6.0
541 * @return bool was anything saved?
542 */
543 public function process_admin_options() {
544 if ( ! $this->instance_id ) {
545 return parent::process_admin_options();
546 }
547
548 // Check we are processing the correct form for this instance.
549 if ( ! isset( $_REQUEST['instance_id'] ) || absint( $_REQUEST['instance_id'] ) !== $this->instance_id ) { // WPCS: input var ok, CSRF ok.
550 return false;
551 }
552
553 $this->init_instance_settings();
554
555 $post_data = $this->get_post_data();
556
557 foreach ( $this->get_instance_form_fields() as $key => $field ) {
558 if ( 'title' !== $this->get_field_type( $field ) ) {
559 try {
560 $this->instance_settings[ $key ] = $this->get_field_value( $key, $field, $post_data );
561 } catch ( Exception $e ) {
562 $this->add_error( $e->getMessage() );
563 }
564 }
565 }
566
567 return update_option( $this->get_instance_option_key(), apply_filters( 'woocommerce_shipping_' . $this->id . '_instance_settings_values', $this->instance_settings, $this ), 'yes' );
568 }
569 }
570