Passed
Pull Request — master (#169)
by
unknown
02:04
created

Algolia_Woo_Indexer::setup_settings_sections()   B

Complexity

Conditions 3
Paths 3

Size

Total Lines 126
Code Lines 86

Duplication

Lines 0
Ratio 0 %

Importance

Changes 8
Bugs 0 Features 2
Metric Value
cc 3
eloc 86
c 8
b 0
f 2
nc 3
nop 0
dl 0
loc 126
rs 8.3054

How to fix   Long Method   

Long Method

Small methods make your code easier to understand, in particular if combined with a good name. Besides, if your method is small, finding a good name is usually much easier.

For example, if you find yourself adding comments to a method's body, this is usually a good sign to extract the commented part to a new method, and use the comment as a starting point when coming up with a good name for this new method.

Commonly applied refactorings include:

1
<?php
2
3
/**
4
 * Main Algolia Woo Indexer class
5
 * Called from main plugin file algolia-woo-indexer.php
6
 *
7
 * @package algolia-woo-indexer
8
 */
9
10
namespace Algowoo;
11
12
use \Algowoo\Algolia_Check_Requirements as Algolia_Check_Requirements;
0 ignored issues
show
Bug introduced by
The type \Algowoo\Algolia_Check_Requirements was not found. Maybe you did not declare it correctly or list all dependencies?

The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g. excluded_paths: ["lib/*"], you can move it to the dependency path list as follows:

filter:
    dependency_paths: ["lib/*"]

For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths

Loading history...
Bug introduced by
This use statement conflicts with another class in this namespace, Algowoo\Algolia_Check_Requirements. Consider defining an alias.

Let?s assume that you have a directory layout like this:

.
|-- OtherDir
|   |-- Bar.php
|   `-- Foo.php
`-- SomeDir
    `-- Foo.php

and let?s assume the following content of Bar.php:

// Bar.php
namespace OtherDir;

use SomeDir\Foo; // This now conflicts the class OtherDir\Foo

If both files OtherDir/Foo.php and SomeDir/Foo.php are loaded in the same runtime, you will see a PHP error such as the following:

PHP Fatal error:  Cannot use SomeDir\Foo as Foo because the name is already in use in OtherDir/Foo.php

However, as OtherDir/Foo.php does not necessarily have to be loaded and the error is only triggered if it is loaded before OtherDir/Bar.php, this problem might go unnoticed for a while. In order to prevent this error from surfacing, you must import the namespace with a different alias:

// Bar.php
namespace OtherDir;

use SomeDir\Foo as SomeDirFoo; // There is no conflict anymore.
Loading history...
13
use \Algowoo\Algolia_Verify_Nonces as Algolia_Verify_Nonces;
0 ignored issues
show
Bug introduced by
The type \Algowoo\Algolia_Verify_Nonces was not found. Maybe you did not declare it correctly or list all dependencies?

The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g. excluded_paths: ["lib/*"], you can move it to the dependency path list as follows:

filter:
    dependency_paths: ["lib/*"]

For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths

Loading history...
Bug introduced by
This use statement conflicts with another class in this namespace, Algowoo\Algolia_Verify_Nonces. Consider defining an alias.

Let?s assume that you have a directory layout like this:

.
|-- OtherDir
|   |-- Bar.php
|   `-- Foo.php
`-- SomeDir
    `-- Foo.php

and let?s assume the following content of Bar.php:

// Bar.php
namespace OtherDir;

use SomeDir\Foo; // This now conflicts the class OtherDir\Foo

If both files OtherDir/Foo.php and SomeDir/Foo.php are loaded in the same runtime, you will see a PHP error such as the following:

PHP Fatal error:  Cannot use SomeDir\Foo as Foo because the name is already in use in OtherDir/Foo.php

However, as OtherDir/Foo.php does not necessarily have to be loaded and the error is only triggered if it is loaded before OtherDir/Bar.php, this problem might go unnoticed for a while. In order to prevent this error from surfacing, you must import the namespace with a different alias:

// Bar.php
namespace OtherDir;

use SomeDir\Foo as SomeDirFoo; // There is no conflict anymore.
Loading history...
14
use \Algowoo\Algolia_Send_Products as Algolia_Send_Products;
0 ignored issues
show
Bug introduced by
This use statement conflicts with another class in this namespace, Algowoo\Algolia_Send_Products. Consider defining an alias.

Let?s assume that you have a directory layout like this:

.
|-- OtherDir
|   |-- Bar.php
|   `-- Foo.php
`-- SomeDir
    `-- Foo.php

and let?s assume the following content of Bar.php:

// Bar.php
namespace OtherDir;

use SomeDir\Foo; // This now conflicts the class OtherDir\Foo

If both files OtherDir/Foo.php and SomeDir/Foo.php are loaded in the same runtime, you will see a PHP error such as the following:

PHP Fatal error:  Cannot use SomeDir\Foo as Foo because the name is already in use in OtherDir/Foo.php

However, as OtherDir/Foo.php does not necessarily have to be loaded and the error is only triggered if it is loaded before OtherDir/Bar.php, this problem might go unnoticed for a while. In order to prevent this error from surfacing, you must import the namespace with a different alias:

// Bar.php
namespace OtherDir;

use SomeDir\Foo as SomeDirFoo; // There is no conflict anymore.
Loading history...
Bug introduced by
The type \Algowoo\Algolia_Send_Products was not found. Maybe you did not declare it correctly or list all dependencies?

The issue could also be caused by a filter entry in the build configuration. If the path has been excluded in your configuration, e.g. excluded_paths: ["lib/*"], you can move it to the dependency path list as follows:

filter:
    dependency_paths: ["lib/*"]

For further information see https://scrutinizer-ci.com/docs/tools/php/php-scrutinizer/#list-dependency-paths

Loading history...
15
16
/**
17
 * Abort if this file is called directly
18
 */
19
if (!defined('ABSPATH')) {
20
    exit;
21
}
22
23
/**
24
 * Include plugin file if function is_plugin_active does not exist
25
 */
26
if (!function_exists('is_plugin_active')) {
27
    require_once(ABSPATH . '/wp-admin/includes/plugin.php');
28
}
29
30
if (!class_exists('Algolia_Woo_Indexer')) {
31
    /**
32
     * Algolia WooIndexer main class
33
     */
34
    // TODO Rename class "Algolia_Woo_Indexer" to match the regular expression ^[A-Z][a-zA-Z0-9]*$.
35
    class Algolia_Woo_Indexer
36
    {
37
        const PLUGIN_NAME      = 'Algolia Woo Indexer';
38
        const PLUGIN_TRANSIENT = 'algowoo-plugin-notice';
39
40
        /**
41
         * Class instance
42
         *
43
         * @var object
44
         */
45
        private static $instance;
46
47
        /**
48
         * The plugin URL
49
         *
50
         * @var string
51
         */
52
        private static $plugin_url = '';
53
54
        /**
55
         * Class constructor
56
         *
57
         * @return void
58
         */
59
        public function __construct()
60
        {
61
            $this->init();
62
        }
63
64
        /**
65
         * Setup sections and fields to store and retrieve values from Settings API
66
         *
67
         * @return void
68
         */
69
        public static function setup_settings_sections()
70
        {
71
            /**
72
             * Setup arguments for settings sections and fields
73
             *
74
             * @see https://developer.wordpress.org/reference/functions/register_setting/
75
             */
76
            if (is_admin()) {
77
                $arguments = array(
78
                    'type'              => 'string',
79
                    'sanitize_callback' => 'settings_fields_validate_options',
80
                    'default'           => null,
81
                );
82
                register_setting('algolia_woo_options', 'algolia_woo_options', $arguments);
83
84
                /**
85
                 * Make sure we reference the instance of the current class by using self::get_instance()
86
                 * This way we can setup the correct callback function for add_settings_section and add_settings_field
87
                 */
88
                $algowooindexer = self::get_instance();
89
90
                /**
91
                 * Add our necessary settings sections and fields
92
                 */
93
                add_settings_section(
94
                    'algolia_woo_indexer_main',
95
                    esc_html__('Algolia Woo Plugin Settings', 'algolia-woo-indexer'),
96
                    array($algowooindexer, 'algolia_woo_indexer_section_text'),
97
                    'algolia_woo_indexer'
98
                );
99
                add_settings_field(
100
                    'algolia_woo_indexer_application_id',
101
                    esc_html__('Application ID', 'algolia-woo-indexer'),
102
                    array($algowooindexer, 'algolia_woo_indexer_application_id_output'),
103
                    'algolia_woo_indexer',
104
                    'algolia_woo_indexer_main'
105
                );
106
                add_settings_field(
107
                    'algolia_woo_indexer_admin_api_key',
108
                    esc_html__('Admin API Key', 'algolia-woo-indexer'),
109
                    array($algowooindexer, 'algolia_woo_indexer_admin_api_key_output'),
110
                    'algolia_woo_indexer',
111
                    'algolia_woo_indexer_main'
112
                );
113
                add_settings_field(
114
                    'algolia_woo_indexer_index_name',
115
                    esc_html__('Index name (will be created if not existing)', 'algolia-woo-indexer'),
116
                    array($algowooindexer, 'algolia_woo_indexer_index_name_output'),
117
                    'algolia_woo_indexer',
118
                    'algolia_woo_indexer_main'
119
                );
120
                add_settings_field(
121
                    'algolia_woo_indexer_automatically_send_new_products',
122
                    esc_html__('Automatically index new products', 'algolia-woo-indexer'),
123
                    array($algowooindexer, 'algolia_woo_indexer_automatically_send_new_products_output'),
124
                    'algolia_woo_indexer',
125
                    'algolia_woo_indexer_main'
126
                );
127
128
                /**
129
                 * Add sections and fields for the basic fields
130
                 */
131
                add_settings_section(
132
                    'algolia_woo_indexer_fields',
133
                    esc_html__('Fields indexing settings', 'algolia-woo-indexer'),
134
                    array($algowooindexer, 'algolia_woo_indexer_fields_section_text'),
135
                    'algolia_woo_indexer'
136
                );
137
                foreach (BASIC_FIELDS as $field) {
138
                    add_settings_field(
139
                        'algolia_woo_indexer_field_' . $field,
140
                        esc_html__($field, 'algolia-woo-indexer'),
141
                        array($algowooindexer, 'algolia_woo_indexer_field_output'),
142
                        'algolia_woo_indexer',
143
                        'algolia_woo_indexer_fields',
144
                        array(
145
                            'label_for' => 'algolia_woo_indexer_field_' . $field,
146
                            'name' => $field
147
                        )
148
                    );
149
                }
150
151
                /**
152
                 * Add sections and fields for the attributes
153
                 */
154
                add_settings_section(
155
                    'algolia_woo_indexer_attributes',
156
                    esc_html__('Attributes indexing settings', 'algolia-woo-indexer'),
157
                    array($algowooindexer, 'algolia_woo_indexer_attributes_section_text'),
158
                    'algolia_woo_indexer'
159
                );
160
                add_settings_field(
161
                    'algolia_woo_indexer_attributes_enabled',
162
                    esc_html__('Enable indexing of attributes', 'algolia-woo-indexer'),
163
                    array($algowooindexer, 'algolia_woo_indexer_attributes_enabled_output'),
164
                    'algolia_woo_indexer',
165
                    'algolia_woo_indexer_attributes'
166
                );
167
                add_settings_field(
168
                    'algolia_woo_indexer_attributes_visibility',
169
                    esc_html__('Visibility', 'algolia-woo-indexer'),
170
                    array($algowooindexer, 'algolia_woo_indexer_attributes_visibility_output'),
171
                    'algolia_woo_indexer',
172
                    'algolia_woo_indexer_attributes'
173
                );
174
175
                add_settings_field(
176
                    'algolia_woo_indexer_attributes_variation',
177
                    esc_html__('Used for variations', 'algolia-woo-indexer'),
178
                    array($algowooindexer, 'algolia_woo_indexer_attributes_variation_output'),
179
                    'algolia_woo_indexer',
180
                    'algolia_woo_indexer_attributes'
181
                );
182
                add_settings_field(
183
                    'algolia_woo_indexer_attributes_list',
184
                    esc_html__('Valid Attributes', 'algolia-woo-indexer'),
185
                    array($algowooindexer, 'algolia_woo_indexer_attributes_list_output'),
186
                    'algolia_woo_indexer',
187
                    'algolia_woo_indexer_attributes'
188
                );
189
                add_settings_field(
190
                    'algolia_woo_indexer_attributes_list_interpolate',
191
                    esc_html__('Numeric Interpolation', 'algolia-woo-indexer'),
192
                    array($algowooindexer, 'algolia_woo_indexer_attributes_list_interpolate_output'),
193
                    'algolia_woo_indexer',
194
                    'algolia_woo_indexer_attributes'
195
                );
196
            }
197
        }
198
199
        /**
200
         * Output for admin API key field
201
         *
202
         * @see https://developer.wordpress.org/reference/functions/wp_nonce_field/
203
         *
204
         * @return void
205
         */
206
        public static function algolia_woo_indexer_admin_api_key_output()
207
        {
208
            $api_key = get_option(ALGOWOO_DB_OPTION . ALGOLIA_API_KEY);
209
            $api_key = is_string($api_key) ? $api_key : CHANGE_ME;
210
211
            wp_nonce_field('algolia_woo_indexer_admin_api_nonce_action', 'algolia_woo_indexer_admin_api_nonce_name');
212
213
            echo "<input id='algolia_woo_indexer_admin_api_key' name='algolia_woo_indexer_admin_api_key[key]'
214
				type='text' value='" . esc_attr($api_key) . "' />";
215
        }
216
217
        /**
218
         * Output for application ID field
219
         *
220
         * @return void
221
         */
222
        public static function algolia_woo_indexer_application_id_output()
223
        {
224
            $application_id = get_option(ALGOWOO_DB_OPTION . ALGOLIA_APP_ID);
225
            $application_id = is_string($application_id) ? $application_id : CHANGE_ME;
226
227
            echo "<input id='algolia_woo_indexer_application_id' name='algolia_woo_indexer_application_id[id]'
228
				type='text' value='" . esc_attr($application_id) . "' />";
229
        }
230
231
        /**
232
         * Output for index name field
233
         *
234
         * @return void
235
         */
236
        public static function algolia_woo_indexer_index_name_output()
237
        {
238
            $index_name = get_option(ALGOWOO_DB_OPTION . INDEX_NAME);
239
            $index_name = is_string($index_name) ? $index_name : CHANGE_ME;
240
241
            echo "<input id='algolia_woo_indexer_index_name' name='algolia_woo_indexer_index_name[name]'
242
				type='text' value='" . esc_attr($index_name) . "' />";
243
        }
244
245
        /**
246
         * Output for checkbox to check if we automatically send new products to Algolia
247
         *
248
         * @return void
249
         */
250
        public static function algolia_woo_indexer_automatically_send_new_products_output()
251
        {
252
            /**
253
             * Sanitization is not really needed as the variable is not directly echoed
254
             * But I have still done it to be 100% safe
255
             */
256
            $auto_send = get_option(ALGOWOO_DB_OPTION . AUTOMATICALLY_SEND_NEW_PRODUCTS);
257
            $auto_send = (!empty($auto_send)) ? 1 : 0; ?>
258
            <input id="algolia_woo_indexer_automatically_send_new_products" name="algolia_woo_indexer_automatically_send_new_products[checked]" type="checkbox" <?php checked(1, $auto_send); ?> />
259
        <?php
260
        }
261
262
        /**
263
         * Output for fields which data shall be sent to Algolia
264
         *
265
         * @return void
266
         */
267
        public static function algolia_woo_indexer_field_output()
268
        {
269
            $value = get_option(ALGOWOO_DB_OPTION . BASIC_FIELD_PREFIX . $args["name"]);
0 ignored issues
show
Comprehensibility Best Practice introduced by
The variable $args seems to be never defined.
Loading history...
270
            $isChecked = (!empty($value)) ? 1 : 0;
271
        ?>
272
            <input id="<?php echo $args["label_for"] ?>" name="<?php echo $args["label_for"] ?>[checked]" type="checkbox" <?php checked(1, $isChecked); ?> />
273
        <?php
274
        }
275
276
        /**
277
         * Output for attributes if functionality is enabled
278
         *
279
         * @return void
280
         */
281
        public static function algolia_woo_indexer_attributes_enabled_output()
282
        {
283
            $value = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_ENABLED);
284
            $isChecked = (!empty($value)) ? 1 : 0;
285
        ?>
286
            <input id="algolia_woo_indexer_attributes_enabled" name="algolia_woo_indexer_attributes_enabled[checked]" type="checkbox" <?php checked(1, $isChecked); ?> />
287
            <?php
288
        }
289
290
        /**
291
         * Output for attributes how to handle visibility setting
292
         *
293
         * @return void
294
         */
295
        public static function algolia_woo_indexer_attributes_visibility_output()
296
        {
297
            $value = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_VISIBILITY);
298
            foreach (ATTRIBUTES_VISIBILITY_STATES as $state) {
299
                $id = 'algolia_woo_indexer_attributes_visibility_' . $state;
300
            ?>
301
                <p><input id="<?php echo $id; ?>" name="algolia_woo_indexer_attributes_visibility[value]" type="radio" value="<?php echo $state; ?>" <?php checked($state, $value); ?> /><label for="<?php echo $id; ?>"><?php echo esc_html__($state, 'algolia-woo-indexer'); ?></label></p>
302
            <?php
303
            }
304
        }
305
306
        /**
307
         * Output for attributes how to handle variant setting
308
         *
309
         * @return void
310
         */
311
        public static function algolia_woo_indexer_attributes_variation_output()
312
        {
313
            $value = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_VARIATION);
314
            foreach (ATTRIBUTES_VARIATION_STATES as $state) {
315
                $id = 'algolia_woo_indexer_attributes_variation_' . $state;
316
            ?>
317
                <p><input id="<?php echo $id; ?>" name="algolia_woo_indexer_attributes_variation[value]" type="radio" value="<?php echo $state; ?>" <?php checked($state, $value); ?> /><label for="<?php echo $id; ?>"><?php echo esc_html__($state, 'algolia-woo-indexer'); ?></label></p>
318
            <?php
319
            }
320
        }
321
322
        /**
323
         * Output for attributes list which attributes are whitelisted
324
         *
325
         * @return void
326
         */
327
        public static function algolia_woo_indexer_attributes_list_output()
328
        {
329
            $value = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_LIST);
330
            $selectedIds = explode(",", $value);
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type false; however, parameter $string of explode() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

330
            $selectedIds = explode(",", /** @scrutinizer ignore-type */ $value);
Loading history...
331
            $name = "algolia_woo_indexer_attributes_list[list]";
332
            $description = __('Here you can whitelist all the attributes. Use the <b>shift</b> or <b>control</b> buttons to select multiple attributes.', 'algolia-woo-indexer');
333
            self::generic_attributes_select_output($name, $selectedIds, $description);
334
        }
335
336
        /**
337
         * Output for attributes list which are using a numeric interpolation
338
         *
339
         * @return void
340
         */
341
        public static function algolia_woo_indexer_attributes_list_interpolate_output()
342
        {
343
            $value = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_LIST_INTERPOLATE);
344
            $selectedIds = explode(",", $value);
0 ignored issues
show
Bug introduced by
It seems like $value can also be of type false; however, parameter $string of explode() does only seem to accept string, maybe add an additional type check? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-type  annotation

344
            $selectedIds = explode(",", /** @scrutinizer ignore-type */ $value);
Loading history...
345
            $name = "algolia_woo_indexer_attributes_list_interpolate[list]";
346
            $description = __('If you have some attributes based on number which shall be interpolated between the lowest to the highest number, you can select it here. A common usecase for this is if you want to have a <b>range slider</b> in aloglia which works for a certain range. Example: a plant grows between 20 and 25cm tall. for this you enter 20 and 25 as attribute values to your product and it will automatically extend the data to [20,21,22,23,24,25]', 'algolia-woo-indexer');
347
            self::generic_attributes_select_output($name, $selectedIds, $description);
348
        }
349
350
        /**
351
         * Generic Output for attributes list where attributes are whitelisted
352
         *
353
         * @return void
354
         */
355
        public static function generic_attributes_select_output($name, $selectedIds, $description)
356
        {
357
            $attribute_taxonomies = wc_get_attribute_taxonomies();
0 ignored issues
show
Bug introduced by
The function wc_get_attribute_taxonomies was not found. Maybe you did not declare it correctly or list all dependencies? ( Ignorable by Annotation )

If this is a false-positive, you can also ignore this issue in your code via the ignore-call  annotation

357
            $attribute_taxonomies = /** @scrutinizer ignore-call */ wc_get_attribute_taxonomies();
Loading history...
358
359
            if (!$attribute_taxonomies) {
360
                echo esc_html__('You don\'t have any attributes defined yet. Go to WooCommerce and add some to use this feature.', 'algolia-woo-indexer');
361
                return;
362
            }
363
            ?>
364
            <p><?php echo $description; ?></p>
365
            <select multiple="multiple" name="<?php echo $name; ?>[]" size="<?php echo count($attribute_taxonomies); ?>">
366
                <?php
367
                foreach ($attribute_taxonomies as $tax) {
368
369
                    $id = $tax->attribute_id;
370
                    $label = $tax->attribute_label;
371
                    $name = $tax->attribute_name;
372
                    $selected = in_array($id, $selectedIds) ? ' selected="selected" ' : '';
373
                ?>
374
                    <option value="<?php echo $id; ?>" <?php echo $selected; ?>>
375
                        <?php echo $label . ' (' . $name . ')'; ?>
376
                    </option>
377
                <?php
378
                }
379
                ?>
380
            </select>
381
        <?php
382
383
        }
384
385
        /**
386
         * Section text for plugin settings section text
387
         *
388
         * @return void
389
         */
390
        public static function algolia_woo_indexer_section_text()
391
        {
392
            echo esc_html__('Enter your API settings here.', 'algolia-woo-indexer');
393
        }
394
395
        /**
396
         * Section text for basic fields settings section text
397
         *
398
         * @return void
399
         */
400
        public static function algolia_woo_indexer_fields_section_text()
401
        {
402
            echo esc_html__('Choose which basic fields shall be indexed.', 'algolia-woo-indexer');
403
        }
404
405
        /**
406
         * Section text for attributes settings section text
407
         *
408
         * @return void
409
         */
410
        public static function algolia_woo_indexer_attributes_section_text()
411
        {
412
            echo esc_html__('Control if and how the attributes shall be indexed.', 'algolia-woo-indexer');
413
        }
414
415
        /**
416
         * Check if we are going to send products by verifying send products nonce
417
         *
418
         * @return void
419
         */
420
        public static function maybe_send_products()
421
        {
422
            if (true === Algolia_Verify_Nonces::verify_send_products_nonce()) {
423
                Algolia_Send_Products::send_products_to_algolia();
424
            }
425
        }
426
427
        /**
428
         * Initialize class, setup settings sections and fields
429
         *
430
         * @return void
431
         */
432
        public static function init()
433
        {
434
            /**
435
             * Fetch the option to see if we are going to automatically send new products
436
             */
437
            $auto_send = get_option(ALGOWOO_DB_OPTION . AUTOMATICALLY_SEND_NEW_PRODUCTS);
438
439
            /**
440
             * Check that we have the minimum versions required and all of the required PHP extensions
441
             */
442
            Algolia_Check_Requirements::check_unmet_requirements();
443
444
            if (!Algolia_Check_Requirements::algolia_wp_version_check() || !Algolia_Check_Requirements::algolia_php_version_check()) {
445
                add_action(
446
                    'admin_notices',
447
                    function () {
448
                        echo '<div class="error notice">
449
                                  <p>' . esc_html__('Please check the server requirements for Algolia Woo Indexer. <br/> It requires minimum PHP version 7.2 and WordPress version 5.0', 'algolia-woo-indexer') . '</p>
450
                                </div>';
451
                    }
452
                );
453
            }
454
455
            $ob_class = get_called_class();
456
457
            /**
458
             * Setup translations
459
             */
460
            add_action('plugins_loaded', array($ob_class, 'load_textdomain'));
461
462
            /**
463
             * Add actions to setup admin menu
464
             */
465
            if (is_admin()) {
466
                add_action('admin_menu', array($ob_class, 'admin_menu'));
467
                add_action('admin_init', array($ob_class, 'setup_settings_sections'));
468
                add_action('admin_init', array($ob_class, 'update_settings_options'));
469
                add_action('admin_init', array($ob_class, 'maybe_send_products'));
470
471
                /**
472
                 * Register hook to automatically send new products if the option is set
473
                 */
474
475
                if ('1' === $auto_send) {
476
                    add_action('save_post', array($ob_class, 'send_new_product_to_algolia'), 10, 3);
477
                }
478
479
                self::$plugin_url = admin_url('options-general.php?page=algolia-woo-indexer-settings');
480
481
                if (!is_plugin_active('woocommerce/woocommerce.php')) {
482
                    add_action(
483
                        'admin_notices',
484
                        function () {
485
                            echo '<div class="error notice">
486
								  <p>' . esc_html__('WooCommerce plugin must be enabled for Algolia Woo Indexer to work.', 'algolia-woo-indexer') . '</p>
487
								</div>';
488
                        }
489
                    );
490
                }
491
            }
492
        }
493
494
        /**
495
         * Send a single product to Algolia once a new product has been published
496
         *
497
         * @param int   $post_id ID of the product.
498
         * @param array $post Post array.
499
         *
500
         * @return void
501
         */
502
        public static function send_new_product_to_algolia($post_id, $post)
503
        {
504
            if ('publish' !== $post->post_status || 'product' !== $post->post_type) {
505
                return;
506
            }
507
            Algolia_Send_Products::send_products_to_algolia($post_id);
508
        }
509
510
        /**
511
         * Verify nonces before we update options and settings
512
         * Also retrieve the value from the send_products_to_algolia hidden field to check if we are sending products to Algolia
513
         *
514
         * @return void
515
         */
516
        public static function update_settings_options()
517
        {
518
            /**
519
             * Do not proceed if nonce for settings is not set
520
             */
521
            if (false === Algolia_Verify_Nonces::verify_settings_nonce()) {
522
                return;
523
            }
524
525
            /**
526
             * Do not proceed if we are going to send products
527
             */
528
            if (true === Algolia_Verify_Nonces::verify_send_products_nonce()) {
529
                return;
530
            }
531
532
            /**
533
             * Filter the application id, api key, index name and verify that the input is an array
534
             *
535
             * @see https://www.php.net/manual/en/function.filter-input.php
536
             */
537
            $post_application_id             = filter_input(INPUT_POST, 'algolia_woo_indexer_application_id', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
538
            $post_api_key                    = filter_input(INPUT_POST, 'algolia_woo_indexer_admin_api_key', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
539
            $post_index_name                 = filter_input(INPUT_POST, 'algolia_woo_indexer_index_name', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
540
            $auto_send                       = filter_input(INPUT_POST, 'algolia_woo_indexer_automatically_send_new_products', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
541
            $attributes_enabled              = filter_input(INPUT_POST, 'algolia_woo_indexer_attributes_enabled', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
542
            $attributes_visibility           = filter_input(INPUT_POST, 'algolia_woo_indexer_attributes_visibility', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
543
            $attributes_variation            = filter_input(INPUT_POST, 'algolia_woo_indexer_attributes_variation', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
544
            $attributes_list                 = filter_input(INPUT_POST, 'algolia_woo_indexer_attributes_list', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
545
            $attributes_list_interpolate     = filter_input(INPUT_POST, 'algolia_woo_indexer_attributes_list_interpolate', FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
546
547
            /**
548
             * Properly sanitize text fields before updating data
549
             *
550
             * @see https://developer.wordpress.org/reference/functions/sanitize_text_field/
551
             */
552
            $filtered_app_id         = sanitize_text_field($post_application_id['id']);
553
            $filtered_api_key        = sanitize_text_field($post_api_key['key']);
554
            $filtered_index_name     = sanitize_text_field($post_index_name['name']);
555
            $filtered_attributes_visibility     = sanitize_text_field($attributes_visibility['value']);
556
            $filtered_attributes_variation     = sanitize_text_field($attributes_variation['value']);
557
558
            /**
559
             * sanitize select list of id's by getting integers and them implode seperated with comma
560
             */
561
562
            $attributes_list_integers = [];
563
            foreach ($attributes_list['list'] as $id) {
564
                array_push($attributes_list_integers, (int) $id);
565
            }
566
            $filtered_attributes_list = implode(',', $attributes_list_integers);
567
            $attributes_list_interp_int = [];
568
            foreach ($attributes_list_interpolate['list'] as $id) {
569
                array_push($attributes_list_interp_int, (int) $id);
570
            }
571
            $clean_attributes_list_interp = implode(',', $attributes_list_interp_int);
572
573
            /**
574
             * Sanitizing by setting the value to either 1 or 0
575
             */
576
            $filtered_product = (!empty($auto_send)) ? 1 : 0;
577
            $filtered_attributes_enabled = (!empty($attributes_enabled)) ? 1 : 0;
578
579
            /**
580
             * Filter the data fields checkboxes and verify that the input is an array and assign it to an associative array
581
             *
582
             * @see https://www.php.net/manual/en/function.filter-input.php
583
             */
584
            $filtered_fields = array();
585
            foreach (BASIC_FIELDS as $field) {
586
                $raw_field = filter_input(INPUT_POST, 'algolia_woo_indexer_field_' . $field, FILTER_DEFAULT, FILTER_REQUIRE_ARRAY);
587
                $filtered_field = (!empty($raw_field)) ? 1 : 0;
588
                $filtered_fields[$field] = $filtered_field;
589
            }
590
591
            /**
592
             * Values have been filtered and sanitized
593
             * Check if set and not empty and update the database
594
             *
595
             * @see https://developer.wordpress.org/reference/functions/update_option/
596
             */
597
            if (isset($filtered_app_id) && (!empty($filtered_app_id))) {
598
                update_option(
599
                    ALGOWOO_DB_OPTION . ALGOLIA_APP_ID,
600
                    $filtered_app_id
601
                );
602
            }
603
604
            if (isset($filtered_api_key) && (!empty($filtered_api_key))) {
605
                update_option(
606
                    ALGOWOO_DB_OPTION . ALGOLIA_API_KEY,
607
                    $filtered_api_key
608
                );
609
            }
610
611
            if (isset($filtered_index_name) && (!empty($filtered_index_name))) {
612
                update_option(
613
                    ALGOWOO_DB_OPTION . INDEX_NAME,
614
                    $filtered_index_name
615
                );
616
            }
617
618
            if (isset($filtered_product)) {
619
                update_option(
620
                    ALGOWOO_DB_OPTION . AUTOMATICALLY_SEND_NEW_PRODUCTS,
621
                    $filtered_product
622
                );
623
            }
624
625
            if (isset($filtered_attributes_enabled)) {
626
                update_option(
627
                    ALGOWOO_DB_OPTION . ATTRIBUTES_ENABLED,
628
                    $filtered_attributes_enabled
629
                );
630
            }
631
632
            if (isset($filtered_attributes_visibility) && (!empty($filtered_attributes_visibility))) {
633
                update_option(
634
                    ALGOWOO_DB_OPTION . ATTRIBUTES_VISIBILITY,
635
                    $filtered_attributes_visibility
636
                );
637
            }
638
639
            if (isset($filtered_attributes_variation) && (!empty($filtered_attributes_variation))) {
640
                update_option(
641
                    ALGOWOO_DB_OPTION . ATTRIBUTES_VARIATION,
642
                    $filtered_attributes_variation
643
                );
644
            }
645
646
            if (isset($filtered_attributes_list) && (!empty($filtered_attributes_list))) {
647
                update_option(
648
                    ALGOWOO_DB_OPTION . ATTRIBUTES_LIST,
649
                    $filtered_attributes_list
650
                );
651
            }
652
653
            if (isset($clean_attributes_list_interp) && (!empty($clean_attributes_list_interp))) {
654
                update_option(
655
                    ALGOWOO_DB_OPTION . ATTRIBUTES_LIST_INTERPOLATE,
656
                    $clean_attributes_list_interp
657
                );
658
            }
659
660
            if (isset($filtered_fields) && (!empty($filtered_fields))) {
661
                foreach ($filtered_fields as $key => $value) {
662
                    update_option(
663
                        ALGOWOO_DB_OPTION . BASIC_FIELD_PREFIX . $key,
664
                        $value
665
                    );
666
                }
667
            }
668
        }
669
670
        /**
671
         * Sanitize input in settings fields and filter through regex to accept only a-z and A-Z
672
         *
673
         * @param string $input Settings text data
674
         * @return array
675
         */
676
        public static function settings_fields_validate_options($input)
677
        {
678
            $valid         = array();
679
            $valid['name'] = preg_replace(
680
                '/[^a-zA-Z\s]/',
681
                '',
682
                $input['name']
683
            );
684
            return $valid;
685
        }
686
687
        /**
688
         * Load text domain for translations
689
         *
690
         * @return void
691
         */
692
        public static function load_textdomain()
693
        {
694
            load_plugin_textdomain('algolia-woo-indexer', false, basename(dirname(__FILE__)) . '/languages/');
695
        }
696
697
        /**
698
         * Add the new menu to settings section so that we can configure the plugin
699
         *
700
         * @return void
701
         */
702
        public static function admin_menu()
703
        {
704
            add_submenu_page(
705
                'options-general.php',
706
                esc_html__('Algolia Woo Indexer Settings', 'algolia-woo-indexer'),
707
                esc_html__('Algolia Woo Indexer Settings', 'algolia-woo-indexer'),
708
                'manage_options',
709
                'algolia-woo-indexer-settings',
710
                array(get_called_class(), 'algolia_woo_indexer_settings')
711
            );
712
        }
713
714
        /**
715
         * Display settings and allow user to modify them
716
         *
717
         * @return void
718
         */
719
        public static function algolia_woo_indexer_settings()
720
        {
721
            /**
722
             * Verify that the user can access the settings page
723
             */
724
            if (!current_user_can('manage_options')) {
725
                wp_die(esc_html__('Action not allowed.', 'algolia_woo_indexer_settings'));
726
            } ?>
727
            <div class="wrap">
728
                <h1><?php esc_html__('Algolia Woo Indexer Settings', 'algolia-woo-indexer'); ?></h1>
729
                <form action="<?php echo esc_url(self::$plugin_url); ?>" method="POST">
730
                    <?php
731
                    settings_fields('algolia_woo_options');
732
                    do_settings_sections('algolia_woo_indexer');
733
                    submit_button('', 'primary wide'); ?>
734
                </form>
735
                <form action="<?php echo esc_url(self::$plugin_url); ?>" method="POST">
736
                    <?php wp_nonce_field('send_products_to_algolia_nonce_action', 'send_products_to_algolia_nonce_name'); ?>
737
                    <input type="hidden" name="send_products_to_algolia" id="send_products_to_algolia" value="true" />
738
                    <?php submit_button(esc_html__('Send products to Algolia', 'algolia_woo_indexer_settings'), 'primary wide', '', false); ?>
739
                </form>
740
            </div>
741
<?php
742
        }
743
744
        /**
745
         * Get active object instance
746
         *
747
         * @return object
748
         */
749
        public static function get_instance()
750
        {
751
            if (!self::$instance) {
752
                self::$instance = new Algolia_Woo_Indexer();
753
            }
754
            return self::$instance;
755
        }
756
757
        /**
758
         * The actions to execute when the plugin is activated.
759
         *
760
         * @return void
761
         */
762
        public static function activate_plugin()
763
        {
764
765
            /**
766
             * Set default values for options if not already set
767
             */
768
            $auto_send = get_option(ALGOWOO_DB_OPTION . AUTOMATICALLY_SEND_NEW_PRODUCTS);
769
            $algolia_application_id          = get_option(ALGOWOO_DB_OPTION . ALGOLIA_APP_ID);
770
            $algolia_api_key                 = get_option(ALGOWOO_DB_OPTION . ALGOLIA_API_KEY);
771
            $algolia_index_name              = get_option(ALGOWOO_DB_OPTION . INDEX_NAME);
772
            $attributes_enabled              = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_ENABLED);
773
            $attributes_visibility           = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_VISIBILITY);
774
            $attributes_variation            = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_VARIATION);
775
            $attributes_list                 = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_LIST);
776
            $attributes_list_interpolate     = get_option(ALGOWOO_DB_OPTION . ATTRIBUTES_LIST_INTERPOLATE);
777
778
            if (empty($auto_send)) {
779
                add_option(
780
                    ALGOWOO_DB_OPTION . AUTOMATICALLY_SEND_NEW_PRODUCTS,
781
                    '0'
782
                );
783
            }
784
785
            if (empty($algolia_application_id)) {
786
                add_option(
787
                    ALGOWOO_DB_OPTION . ALGOLIA_APP_ID,
788
                    'Change me'
789
                );
790
            }
791
792
            if (empty($algolia_api_key)) {
793
                add_option(
794
                    ALGOWOO_DB_OPTION . ALGOLIA_API_KEY,
795
                    'Change me'
796
                );
797
            }
798
799
            if (empty($algolia_index_name)) {
800
                add_option(
801
                    ALGOWOO_DB_OPTION . INDEX_NAME,
802
                    'Change me'
803
                );
804
            }
805
806
            if (empty($attributes_enabled)) {
807
                add_option(
808
                    ALGOWOO_DB_OPTION . ATTRIBUTES_ENABLED,
809
                    0
810
                );
811
            }
812
813
            if (empty($attributes_visibility)) {
814
                add_option(
815
                    ALGOWOO_DB_OPTION . ATTRIBUTES_VISIBILITY,
816
                    ATTRIBUTES_VISIBILITY_STATES[1]
817
                );
818
            }
819
            if (empty($attributes_variation)) {
820
                add_option(
821
                    ALGOWOO_DB_OPTION . ATTRIBUTES_VARIATION,
822
                    ATTRIBUTES_VARIATION_STATES[0]
823
                );
824
            }
825
            if (empty($attributes_list)) {
826
                add_option(
827
                    ALGOWOO_DB_OPTION . ATTRIBUTES_LIST,
828
                    ''
829
                );
830
            }
831
            if (empty($attributes_list_interpolate)) {
832
                add_option(
833
                    ALGOWOO_DB_OPTION . ATTRIBUTES_LIST_INTERPOLATE,
834
                    ''
835
                );
836
            }
837
838
            /**
839
             * Set default values for index fields if not already set
840
             */
841
            foreach (BASIC_FIELDS as $field) {
842
                $value = get_option(ALGOWOO_DB_OPTION . BASIC_FIELD_PREFIX . $field);
843
                if (empty($value)) {
844
                    update_option(
845
                        ALGOWOO_DB_OPTION . BASIC_FIELD_PREFIX . $field,
846
                        '1'
847
                    );
848
                }
849
            }
850
851
852
853
            set_transient(self::PLUGIN_TRANSIENT, true);
854
        }
855
856
        /**
857
         * The actions to execute when the plugin is deactivated.
858
         *
859
         * @return void
860
         */
861
        public static function deactivate_plugin()
862
        {
863
            delete_transient(self::PLUGIN_TRANSIENT);
864
        }
865
    }
866
}
867