/var/www/html/wp-content/plugins/woocommerce/includes/admin/settings/class-wc-settings-tax.php


1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
<?php
/**
 * WooCommerce Tax Settings
 *
 * @package     WooCommerce\Admin
 * @version     2.1.0
 */

if ( ! defined'ABSPATH' ) ) {
    exit;
}

if ( 
class_exists'WC_Settings_Tax'false ) ) {
    return new 
WC_Settings_Tax();
}

/**
 * WC_Settings_Tax.
 */
class WC_Settings_Tax extends WC_Settings_Page {

    
/**
     * Constructor.
     */
    
public function __construct() {
        
$this->id    'tax';
        
$this->label __'Tax''woocommerce' );

        
add_filter'woocommerce_settings_tabs_array', array( $this'add_settings_page' ), 20 );
        
add_action'woocommerce_admin_field_conflict_error', array( $this'conflict_error' ) );
        if ( 
wc_tax_enabled() ) {
            
add_action'woocommerce_sections_' $this->id, array( $this'output_sections' ) );
            
add_action'woocommerce_settings_' $this->id, array( $this'output' ) );
            
add_action'woocommerce_settings_save_' $this->id, array( $this'save' ) );
        }
    }

    
/**
     * Setting page icon.
     *
     * @var string
     */
    
public $icon 'percent';

    
/**
     * Creates the React mount point for the embedded banner.
     */
    
public function conflict_error() {
        
?>
        <tr valign="top">
                            <th scope="row" class="titledesc woocommerce_admin_tax_settings_slotfill_th">
                            </th>
                            <td class="forminp forminp-text woocommerce_admin_tax_settings_slotfill_td">
        <div id="wc_tax_settings_slotfill"> </div>
    </td>
    </tr>
        <?php
    
}

    
/**
     * Add this page to settings.
     *
     * @param array $pages Existing pages.
     * @return array|mixed
     */
    
public function add_settings_page$pages ) {
        if ( 
wc_tax_enabled() ) {
            return 
parent::add_settings_page$pages );
        } else {
            return 
$pages;
        }
    }

    
/**
     * Get own sections.
     *
     * @return array
     */
    
protected function get_own_sections() {
        
$sections = array(
            
''         => __'Tax options''woocommerce' ),
            
'standard' => __'Standard rates''woocommerce' ),
        );

        
// Get tax classes and display as links.
        
$tax_classes WC_Tax::get_tax_classes();

        foreach ( 
$tax_classes as $class ) {
            
/* translators: $s tax rate section name */
            
$sectionssanitize_title$class ) ] = sprintf__'%s rates''woocommerce' ), $class );
        }

        return 
$sections;
    }

    
/**
     * Get settings array.
     *
     * @return array
     */
    
public function get_settings_for_default_section() {
        return include 
__DIR__ '/views/settings-tax.php';
    }

    
/**
     * Output the settings.
     */
    
public function output() {
        global 
$current_section;

        
$tax_classes WC_Tax::get_tax_class_slugs();

        if ( 
'standard' === $current_section || in_array$current_sectionarray_filter$tax_classes ), true ) ) {
            
$this->output_tax_rates();
        } else {
            
parent::output();
        }
    }

    
/**
     * Save settings.
     */
    
public function save() {
        
// phpcs:disable WordPress.Security.NonceVerification.Missing
        
global $current_section;

        if ( ! 
$current_section ) {
            
$this->save_settings_for_current_section();

            if ( isset( 
$_POST['woocommerce_tax_classes'] ) ) {
                
$this->save_tax_classeswp_unslash$_POST['woocommerce_tax_classes'] ) ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized
            
}
        } elseif ( ! empty( 
$_POST['tax_rate_country'] ) ) {
            
$this->save_tax_rates();
        } else {
            
$this->save_settings_for_current_section();
        }

        
$this->do_update_options_action();

        
// Invalidate caches.
        
WC_Cache_Helper::invalidate_cache_group'taxes' );
        
WC_Cache_Helper::get_transient_version'shipping'true );
        
// phpcs:enable WordPress.Security.NonceVerification.Missing
    
}

    
/**
     * Saves tax classes defined in the textarea to the tax class table instead of an option.
     *
     * @param string $raw_tax_classes Posted value.
     * @return null
     */
    
public function save_tax_classes$raw_tax_classes ) {
        
$tax_classes          array_filterarray_map'trim'explode"\n"$raw_tax_classes ) ) );
        
$existing_tax_classes WC_Tax::get_tax_classes();
        
$removed              array_diff$existing_tax_classes$tax_classes );
        
$added                array_diff$tax_classes$existing_tax_classes );

        foreach ( 
$removed as $name ) {
            
WC_Tax::delete_tax_class_by'name'$name );
        }

        foreach ( 
$added as $name ) {
            
$tax_class WC_Tax::create_tax_class$name );

            
// Display any error that could be triggered while creating tax classes.
            
if ( is_wp_error$tax_class ) ) {
                
WC_Admin_Settings::add_error(
                    
sprintf(
                        
/* translators: 1: tax class name 2: error message */
                        
esc_html__'Additional tax class "%1$s" couldn\'t be saved. %2$s.''woocommerce' ),
                        
esc_html$name ),
                        
$tax_class->get_error_message()
                    )
                );
            }
        }

        return 
null;
    }

    
/**
     * Output tax rate tables.
     */
    
public function output_tax_rates() {
        global 
$current_section;

        
$current_class self::get_current_tax_class();

        
$countries = array();
        foreach ( 
WC()->countries->get_allowed_countries() as $value => $label ) {
            
$countries[] = array(
                
'value' => $value,
                
'label' => esc_jshtml_entity_decode$label ) ),
            );
        }

        
$states = array();
        foreach ( 
WC()->countries->get_allowed_country_states() as $label ) {
            foreach ( 
$label as $code => $state ) {
                
$states[] = array(
                    
'value' => $code,
                    
'label' => esc_jshtml_entity_decode$state ) ),
                );
            }
        }

        
$base_url admin_url(
            
add_query_arg(
                array(
                    
'page'    => 'wc-settings',
                    
'tab'     => 'tax',
                    
'section' => $current_section,
                ),
                
'admin.php'
            
)
        );

        
// Localize and enqueue our js.
        
wp_localize_script(
            
'wc-settings-tax',
            
'htmlSettingsTaxLocalizeScript',
            array(
                
'current_class' => $current_class,
                
'wc_tax_nonce'  => wp_create_nonce'wc_tax_nonce-class:' $current_class ),
                
'base_url'      => $base_url,
                
'rates'         => array_valuesWC_Tax::get_rates_for_tax_class$current_class ) ),
                
'page'          => ! empty( $_GET['p'] ) ? absint$_GET['p'] ) : 1// phpcs:ignore WordPress.Security.NonceVerification.Recommended
                
'limit'         => 100,
                
'countries'     => $countries,
                
'states'        => $states,
                
'default_rate'  => array(
                    
'tax_rate_id'       => 0,
                    
'tax_rate_country'  => '',
                    
'tax_rate_state'    => '',
                    
'tax_rate'          => '',
                    
'tax_rate_name'     => '',
                    
'tax_rate_priority' => 1,
                    
'tax_rate_compound' => 0,
                    
'tax_rate_shipping' => 1,
                    
'tax_rate_order'    => null,
                    
'tax_rate_class'    => $current_class,
                ),
                
'strings'       => array(
                    
'no_rows_selected'        => __'No row(s) selected''woocommerce' ),
                    
'unload_confirmation_msg' => __'Your changed data will be lost if you leave this page without saving.''woocommerce' ),
                    
'csv_data_cols'           => array(
                        
__'Country code''woocommerce' ),
                        
__'State code''woocommerce' ),
                        
__'Postcode / ZIP''woocommerce' ),
                        
__'City''woocommerce' ),
                        
__'Rate %''woocommerce' ),
                        
__'Tax name''woocommerce' ),
                        
__'Priority''woocommerce' ),
                        
__'Compound''woocommerce' ),
                        
__'Shipping''woocommerce' ),
                        
__'Tax class''woocommerce' ),
                    ),
                ),
            )
        );
        
wp_enqueue_script'wc-settings-tax' );

        include 
__DIR__ '/views/html-settings-tax.php';
    }

    
/**
     * Get tax class being edited.
     *
     * @return string
     */
    
private static function get_current_tax_class() {
        global 
$current_section;

        
$tax_classes   WC_Tax::get_tax_classes();
        
$current_class '';

        foreach ( 
$tax_classes as $class ) {
            if ( 
sanitize_title$class ) === $current_section ) {
                
$current_class $class;
            }
        }

        return 
$current_class;
    }

    
/**
     * Get a posted tax rate.
     *
     * @param string $key   Key of tax rate in the post data array.
     * @param int    $order Position/order of rate.
     * @param string $class Tax class for rate.
     * @return array
     */
    
private function get_posted_tax_rate$key$order$class ) {
        
// phpcs:disable WordPress.Security.NonceVerification.Missing -- this is called from 'save_tax_rates' only, where nonce is already verified.
        
$tax_rate      = array();
        
$tax_rate_keys = array(
            
'tax_rate_country',
            
'tax_rate_state',
            
'tax_rate',
            
'tax_rate_name',
            
'tax_rate_priority',
        );

        
// phpcs:disable WordPress.Security.NonceVerification.Missing
        
foreach ( $tax_rate_keys as $tax_rate_key ) {
            if ( isset( 
$_POST$tax_rate_key ], $_POST$tax_rate_key ][ $key ] ) ) {
                
$tax_rate$tax_rate_key ] = wc_cleanwp_unslash$_POST$tax_rate_key ][ $key ] ) );
            }
        }

        
$tax_rate['tax_rate_compound'] = isset( $_POST['tax_rate_compound'][ $key ] ) ? 0;
        
$tax_rate['tax_rate_shipping'] = isset( $_POST['tax_rate_shipping'][ $key ] ) ? 0;
        
$tax_rate['tax_rate_order']    = $order;
        
$tax_rate['tax_rate_class']    = $class;
        
// phpcs:enable WordPress.Security.NonceVerification.Missing

        
return $tax_rate;
        
// phpcs:enable WordPress.Security.NonceVerification.Missing
    
}

    
/**
     * Save tax rates.
     */
    
public function save_tax_rates() {
        
// phpcs:disable WordPress.Security.NonceVerification.Missing -- this is called via "do_action('woocommerce_settings_save_'...") in base class, where nonce is verified first.
        
global $wpdb;

        
$current_class sanitize_titleself::get_current_tax_class() );
        
// phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotValidated, WordPress.Security.NonceVerification.Missing
        
$posted_countries wc_cleanwp_unslash$_POST['tax_rate_country'] ) );

        
// get the tax rate id of the first submitted row.
        
$first_tax_rate_id key$posted_countries );

        
// get the order position of the first tax rate id.
        
$tax_rate_order absint$wpdb->get_var$wpdb->prepare"SELECT tax_rate_order FROM {$wpdb->prefix}woocommerce_tax_rates WHERE tax_rate_id = %s"$first_tax_rate_id ) ) );

        
$index = isset( $tax_rate_order ) ? $tax_rate_order 0;

        
// Loop posted fields.
        // phpcs:disable WordPress.Security.NonceVerification.Missing
        
foreach ( $posted_countries as $key => $value ) {
            
$mode     = ( === strpos$key'new-' ) ) ? 'insert' 'update';
            
$tax_rate $this->get_posted_tax_rate$key$index ++, $current_class );

            if ( 
'insert' === $mode ) {
                
$tax_rate_id WC_Tax::_insert_tax_rate$tax_rate );
            } elseif ( isset( 
$_POST['remove_tax_rate'][ $key ] ) && === absint$_POST['remove_tax_rate'][ $key ] ) ) {
                
$tax_rate_id absint$key );
                
WC_Tax::_delete_tax_rate$tax_rate_id );
                continue;
            } else {
                
$tax_rate_id absint$key );
                
WC_Tax::_update_tax_rate$tax_rate_id$tax_rate );
            }

            if ( isset( 
$_POST['tax_rate_postcode'][ $key ] ) ) {
                
WC_Tax::_update_tax_rate_postcodes$tax_rate_idwc_cleanwp_unslash$_POST['tax_rate_postcode'][ $key ] ) ) );
            }
            if ( isset( 
$_POST['tax_rate_city'][ $key ] ) ) {
                
WC_Tax::_update_tax_rate_cities$tax_rate_idwc_cleanwp_unslash$_POST['tax_rate_city'][ $key ] ) ) );
            }
        }
        
// phpcs:enable WordPress.Security.NonceVerification.Missing
    
}
}

return new 
WC_Settings_Tax();