Completed
Pull Request — master (#1454)
by Aristeides
06:15 queued 03:29
created

Kirki_Field::set_partial_refresh()   B

Complexity

Conditions 8
Paths 12

Size

Total Lines 17
Code Lines 10

Duplication

Lines 0
Ratio 0 %

Importance

Changes 0
Metric Value
cc 8
eloc 10
nc 12
nop 0
dl 0
loc 17
rs 7.7777
c 0
b 0
f 0
1
<?php
2
/**
3
 * Creates and validates field parameters.
4
 *
5
 * @package     Kirki
6
 * @category    Core
7
 * @author      Aristeides Stathopoulos
8
 * @copyright   Copyright (c) 2017, Aristeides Stathopoulos
9
 * @license     http://opensource.org/licenses/https://opensource.org/licenses/MIT
10
 * @since       1.0
11
 */
12
13
/**
14
 * Please do not use this class directly.
15
 * You should instead extend it per-field-type.
16
 */
17
class Kirki_Field {
18
19
	/**
20
	 * An array of the field arguments.
21
	 *
22
	 * @access protected
23
	 * @var array
24
	 */
25
	protected $args = array();
26
27
	/**
28
	 * The ID of the kirki_config we're using.
29
	 *
30
	 * @see Kirki_Config
31
	 * @access protected
32
	 * @var string
33
	 */
34
	protected $kirki_config = 'global';
35
36
	/**
37
	 * Thje capability required so that users can edit this field.
38
	 *
39
	 * @access protected
40
	 * @var string
41
	 */
42
	protected $capability = 'edit_theme_options';
43
44
	/**
45
	 * If we're using options instead of theme_mods
46
	 * and we want them serialized, this is the option that
47
	 * will saved in the db.
48
	 *
49
	 * @access protected
50
	 * @var string
51
	 */
52
	protected $option_name = '';
53
54
	/**
55
	 * Custom input attributes (defined as an array).
56
	 *
57
	 * @access protected
58
	 * @var array
59
	 */
60
	protected $input_attrs = array();
61
62
	/**
63
	 * Use "theme_mod" or "option".
64
	 *
65
	 * @access protected
66
	 * @var string
67
	 */
68
	protected $option_type = 'theme_mod';
69
70
	/**
71
	 * The name of this setting (id for the db).
72
	 *
73
	 * @access protected
74
	 * @var string|array
75
	 */
76
	protected $settings = '';
77
78
	/**
79
	 * Set to true if you want to disable all CSS output for this field.
80
	 *
81
	 * @access protected
82
	 * @var bool
83
	 */
84
	protected $disable_output = false;
85
86
	/**
87
	 * The field type.
88
	 *
89
	 * @access protected
90
	 * @var string
91
	 */
92
	protected $type = 'kirki-generic';
93
94
	/**
95
	 * Some fields require options to be set.
96
	 * We're whitelisting the property here
97
	 * and suggest you validate this in a child class.
98
	 *
99
	 * @access protected
100
	 * @var array
101
	 */
102
	protected $choices = array();
103
104
	/**
105
	 * Assign this field to a section.
106
	 * Fields not assigned to a section will not be displayed in the customizer.
107
	 *
108
	 * @access protected
109
	 * @var string
110
	 */
111
	protected $section = '';
112
113
	/**
114
	 * The default value for this field.
115
	 *
116
	 * @access protected
117
	 * @var string|array|bool
118
	 */
119
	protected $default = '';
120
121
	/**
122
	 * Priority determines the position of a control inside a section.
123
	 * Lower priority numbers move the control to the top.
124
	 *
125
	 * @access protected
126
	 * @var int
127
	 */
128
	protected $priority = 10;
129
130
	/**
131
	 * Unique ID for this field.
132
	 * This is auto-calculated from the $settings argument.
133
	 *
134
	 * @access protected
135
	 * @var string
136
	 */
137
	protected $id = '';
138
139
	/**
140
	 * Use if you want to automatically generate CSS from this field's value.
141
	 *
142
	 * @see https://kirki.org/docs/arguments/output
143
	 * @access protected
144
	 * @var array
145
	 */
146
	protected $output = array();
147
148
	/**
149
	 * Use to automatically generate postMessage scripts.
150
	 * Not necessary to use if you use 'transport' => 'auto'
151
	 * and have already set an array for the 'output' argument.
152
	 *
153
	 * @see https://kirki.org/docs/arguments/js_vars
154
	 * @access protected
155
	 * @var array
156
	 */
157
	protected $js_vars = array();
158
159
	/**
160
	 * If you want to use a CSS compiler, then use this to set the variable names.
161
	 *
162
	 * @see https://kirki.org/docs/arguments/variables
163
	 * @access protected
164
	 * @var array
165
	 */
166
	protected $variables = array();
167
168
	/**
169
	 * Text that will be used in a tooltip to provide extra info for this field.
170
	 *
171
	 * @access protected
172
	 * @var string
173
	 */
174
	protected $tooltip = '';
175
176
	/**
177
	 * A custom callback to determine if the field should be visible or not.
178
	 *
179
	 * @access protected
180
	 * @var string|array
181
	 */
182
	protected $active_callback = '__return_true';
183
184
	/**
185
	 * A custom sanitize callback that will be used to properly save the values.
186
	 *
187
	 * @access protected
188
	 * @var string|array
189
	 */
190
	protected $sanitize_callback = '';
191
192
	/**
193
	 * Use 'refresh', 'postMessage' or 'auto'.
194
	 * 'auto' will automatically geberate any 'js_vars' from the 'output' argument.
195
	 *
196
	 * @access protected
197
	 * @var string
198
	 */
199
	protected $transport = 'refresh';
200
201
	/**
202
	 * Define dependencies to show/hide this field based on the values of other fields.
203
	 *
204
	 * @access protected
205
	 * @var array
206
	 */
207
	protected $required = array();
208
209
	/**
210
	 * Partial Refreshes array.
211
	 *
212
	 * @access protected
213
	 * @var array
214
	 */
215
	protected $partial_refresh = array();
216
217
	/**
218
	 * The class constructor.
219
	 * Parses and sanitizes all field arguments.
220
	 * Then it adds the field to Kirki::$fields.
221
	 *
222
	 * @access public
223
	 * @param string $config_id    The ID of the config we want to use.
224
	 *                             Defaults to "global".
225
	 *                             Configs are handled by the Kirki_Config class.
226
	 * @param array  $args         The arguments of the field.
227
	 */
228
	public function __construct( $config_id = 'global', $args = array() ) {
229
230
		if ( isset( $args['setting'] ) && ! empty( $args['setting'] ) && ( ! isset( $args['settings'] ) || empty( $args['settings'] ) ) ) {
231
			/* translators: %s represents the field ID where the error occurs. */
232
			_doing_it_wrong( __METHOD__, sprintf( esc_attr__( 'Typo found in field %s - setting instead of settings.', 'kirki' ), esc_attr( $args['settings'] ) ), '3.0.10' );
233
			$args['settings'] = $args['setting'];
234
			unset( $args['setting'] );
235
		}
236
237
		// In case the user only provides 1 argument,
238
		// assume that the provided argument is $args and set $config_id = 'global'.
239
		if ( is_array( $config_id ) && empty( $args ) ) {
240
			/* translators: %1$s represents the field ID where the error occurs. %2$s is the URL in the documentation site. */
241
			_doing_it_wrong( __METHOD__, sprintf( esc_attr__( 'Config not defined for field %1$s - See %2$s for details on how to properly add fields.', 'kirki' ), esc_attr( $args['settings'] ), 'https://aristath.github.io/kirki/docs/getting-started/fields.html' ), '3.0.10' );
242
			$args = $config_id;
243
			$config_id = 'global';
244
		}
245
246
		$args['kirki_config'] = $config_id;
247
248
		$this->kirki_config = trim( esc_attr( $config_id ) );
249
		if ( '' === $config_id ) {
250
			/* translators: %1$s represents the field ID where the error occurs. %2$s is the URL in the documentation site. */
251
			_doing_it_wrong( __METHOD__, sprintf( esc_attr__( 'Config not defined for field %1$s - See %2$s for details on how to properly add fields.', 'kirki' ), esc_attr( $args['settings'] ), 'https://aristath.github.io/kirki/docs/getting-started/fields.html' ), '3.0.10' );
252
			$this->kirki_config = 'global';
253
		}
254
255
		// Get defaults from the class.
256
		$defaults = get_class_vars( __CLASS__ );
257
258
		// Get the config arguments, and merge them with the defaults.
259
		$config_defaults = ( isset( Kirki::$config['global'] ) ) ? Kirki::$config['global'] : array();
260
		if ( 'global' !== $this->kirki_config && isset( Kirki::$config[ $this->kirki_config ] ) ) {
261
			$config_defaults = Kirki::$config[ $this->kirki_config ];
262
		}
263
		$config_defaults = ( is_array( $config_defaults ) ) ? $config_defaults : array();
264
		foreach ( $config_defaults as $key => $value ) {
265
			if ( isset( $defaults[ $key ] ) && ! empty( $value ) && $value !== $defaults[ $key ] ) {
266
				$defaults[ $key ] = $value;
267
			}
268
		}
269
270
		// Merge our args with the defaults.
271
		$args = wp_parse_args( $args, $defaults );
272
273
		// Set the class properties using the parsed args.
274
		foreach ( $args as $key => $value ) {
275
			$this->$key = $value;
276
		}
277
278
		$this->args = $args;
279
280
		$this->set_field();
281
	}
282
283
	/**
284
	 * Processes the field arguments
285
	 *
286
	 * @access protected
287
	 */
288
	protected function set_field() {
289
290
		$properties = get_class_vars( __CLASS__ );
291
292
		// Some things must run before the others.
293
		$this->set_option_name();
294
		$this->set_option_type();
295
		$this->set_settings();
296
297
		// Sanitize the properties, skipping the ones that have already run above.
298
		foreach ( $properties as $property => $value ) {
299
			if ( in_array( $property, array( 'option_name', 'option_type', 'settings' ), true ) ) {
300
				continue;
301
			}
302
			if ( method_exists( $this, 'set_' . $property ) ) {
303
				$method_name = 'set_' . $property;
304
				$this->$method_name();
305
			}
306
		}
307
308
		// Get all arguments with their values.
309
		$args = get_object_vars( $this );
310
		foreach ( array_keys( $args ) as $key ) {
311
			$args[ $key ] = $this->$key;
312
		}
313
314
		// Add the field to the static $fields variable properly indexed.
315
		Kirki::$fields[ $this->settings ] = $args;
316
317
	}
318
319
	/**
320
	 * Escape $kirki_config.
321
	 *
322
	 * @access protected
323
	 */
324
	protected function set_kirki_config() {
325
326
		$this->kirki_config = esc_attr( $this->kirki_config );
327
	}
328
329
	/**
330
	 * Escape $option_name.
331
	 *
332
	 * @access protected
333
	 */
334
	protected function set_option_name() {
335
336
		$this->option_name = esc_attr( $this->option_name );
337
	}
338
339
	/**
340
	 * Escape the $section.
341
	 *
342
	 * @access protected
343
	 */
344
	protected function set_section() {
345
346
		$this->section = sanitize_key( $this->section );
347
	}
348
349
	/**
350
	 * Escape the $section.
351
	 *
352
	 * @access protected
353
	 */
354
	protected function set_input_attrs() {
355
356
		if ( ! is_array( $this->input_attrs ) ) {
357
			$this->input_attrs = array();
358
		}
359
	}
360
361
	/**
362
	 * Checks the capability chosen is valid.
363
	 * If not, then falls back to 'edit_theme_options'
364
	 *
365
	 * @access protected
366
	 */
367
	protected function set_capability() {
368
369
		// Early exit if we're using 'edit_theme_options'.
370
		if ( 'edit_theme_options' === $this->capability ) {
371
			return;
372
		}
373
		// Escape & trim the capability.
374
		$this->capability = trim( esc_attr( $this->capability ) );
375
	}
376
377
	/**
378
	 * Make sure we're using the correct option_type
379
	 *
380
	 * @access protected
381
	 */
382
	protected function set_option_type() {
383
384
		// Take care of common typos.
385
		if ( 'options' === $this->option_type ) {
386
			$this->option_type = 'option';
387
		}
388
		// Take care of common typos.
389
		if ( 'theme_mods' === $this->option_type ) {
390
			/* translators: %1$s represents the field ID where the error occurs. */
391
			_doing_it_wrong( __METHOD__, sprintf( esc_attr__( 'Typo found in field %s - "theme_mods" vs "theme_mod"', 'kirki' ), esc_attr( $this->settings ) ), '3.0.10' );
392
			$this->option_type = 'theme_mod';
393
		}
394
	}
395
396
	/**
397
	 * Modifications for partial refreshes.
398
	 *
399
	 * @access protected
400
	 */
401
	protected function set_partial_refresh() {
402
403
		if ( ! is_array( $this->partial_refresh ) ) {
404
			$this->partial_refresh = array();
405
		}
406
		foreach ( $this->partial_refresh as $id => $args ) {
407
			if ( ! is_array( $args ) || ! isset( $args['selector'] ) || ! isset( $args['render_callback'] ) || ! is_callable( $args['render_callback'] ) ) {
408
				/* translators: %1$s represents the field ID where the error occurs. */
409
				_doing_it_wrong( __METHOD__, sprintf( esc_attr__( '"partial_refresh" invalid entry in field %s', 'kirki' ), esc_attr( $this->settings ) ), '3.0.10' );
410
				unset( $this->partial_refresh[ $id ] );
411
				continue;
412
			}
413
		}
414
		if ( ! empty( $this->partial_refresh ) ) {
415
			$this->transport = 'postMessage';
416
		}
417
	}
418
419
	/**
420
	 * Sets the settings.
421
	 * If we're using serialized options it makes sure that settings are properly formatted.
422
	 * We'll also be escaping all setting names here for consistency.
423
	 *
424
	 * @access protected
425
	 */
426
	protected function set_settings() {
427
428
		// If settings is not an array, temporarily convert it to an array.
429
		// This is just to allow us to process everything the same way and avoid code duplication.
430
		// if settings is not an array then it will not be set as an array in the end.
431
		if ( ! is_array( $this->settings ) ) {
432
			$this->settings = array(
433
				'kirki_placeholder_setting' => $this->settings,
434
			);
435
		}
436
		$settings = array();
437
		foreach ( $this->settings as $setting_key => $setting_value ) {
438
			$settings[ $setting_key ] = $setting_value;
439
			// If we're using serialized options then we need to spice this up.
440
			if ( 'option' === $this->option_type && '' !== $this->option_name && ( false === strpos( $setting_key, '[' ) ) ) {
441
				$settings[ $setting_key ] = "{$this->option_name}[{$setting_value}]";
442
			}
443
		}
444
		$this->settings = $settings;
445
		if ( isset( $this->settings['kirki_placeholder_setting'] ) ) {
446
			$this->settings = $this->settings['kirki_placeholder_setting'];
447
		}
448
	}
449
450
	/**
451
	 * Escapes the tooltip messages.
452
	 *
453
	 * @access protected
454
	 */
455
	protected function set_tooltip() {
456
457
		if ( '' !== $this->tooltip ) {
458
			$this->tooltip = wp_strip_all_tags( $this->tooltip );
459
			return;
460
		}
461
	}
462
463
	/**
464
	 * Sets the active_callback
465
	 * If we're using the $required argument,
466
	 * Then this is where the switch is made to our evaluation method.
467
	 *
468
	 * @access protected
469
	 */
470
	protected function set_active_callback() {
471
472
		if ( is_array( $this->active_callback ) && ! is_callable( $this->active_callback ) ) {
473
			if ( isset( $this->active_callback[0] ) ) {
474
				$this->required = $this->active_callback;
475
			}
476
		}
477
478
		if ( ! empty( $this->required ) ) {
479
			$this->active_callback = array( 'Kirki_Active_Callback', 'evaluate' );
480
			return;
481
		}
482
		// No need to proceed any further if we're using the default value.
483
		if ( '__return_true' === $this->active_callback ) {
484
			return;
485
		}
486
		// Make sure the function is callable, otherwise fallback to __return_true.
487
		if ( ! is_callable( $this->active_callback ) ) {
488
			$this->active_callback = '__return_true';
489
		}
490
	}
491
492
	/**
493
	 * Sets the control type.
494
	 *
495
	 * @access protected
496
	 */
497
	protected function set_type() {
498
499
		// Escape the control type (it doesn't hurt to be sure).
500
		$this->type = esc_attr( $this->type );
501
	}
502
503
	/**
504
	 * Sets the $id.
505
	 * Setting the ID should happen after the 'settings' sanitization.
506
	 * This way we can also properly handle cases where the option_type is set to 'option'
507
	 * and we're using an array instead of individual options.
508
	 *
509
	 * @access protected
510
	 */
511
	protected function set_id() {
512
513
		$this->id = sanitize_key( str_replace( '[', '-', str_replace( ']', '', $this->settings ) ) );
514
	}
515
516
	/**
517
	 * Sets the $choices.
518
	 *
519
	 * @access protected
520
	 */
521
	protected function set_choices() {
522
523
		if ( ! is_array( $this->choices ) ) {
524
			$this->choices = array();
525
		}
526
	}
527
528
	/**
529
	 * Escapes the $disable_output.
530
	 *
531
	 * @access protected
532
	 */
533
	protected function set_disable_output() {
534
535
		$this->disable_output = (bool) $this->disable_output;
536
537
	}
538
539
	/**
540
	 * Sets the $sanitize_callback
541
	 *
542
	 * @access protected
543
	 */
544
	protected function set_output() {
545
546
		if ( empty( $this->output ) ) {
547
			return;
548
		}
549
		if ( ! is_array( $this->output ) ) {
550
			/* translators: The field ID where the error occurs. */
551
			_doing_it_wrong( __METHOD__, sprintf( esc_attr__( '"output" invalid format in field %s. The "output" argument should be defined as an array of arrays.', 'kirki' ), esc_attr( $this->settings ) ), '3.0.10' );
552
			$this->output = array(
553
				array(
554
					'element' => $this->output,
555
				),
556
			);
557
		}
558
		// Convert to array of arrays if needed.
559
		if ( isset( $this->output['element'] ) ) {
560
			/* translators: The field ID where the error occurs. */
561
			_doing_it_wrong( __METHOD__, sprintf( esc_attr__( '"output" invalid format in field %s. The "output" argument should be defined as an array of arrays.', 'kirki' ), esc_attr( $this->settings ) ), '3.0.10' );
562
			$this->output = array( $this->output );
563
		}
564
		foreach ( $this->output as $key => $output ) {
565
			if ( empty( $output ) || ! isset( $output['element'] ) ) {
566
				unset( $this->output[ $key ] );
567
				continue;
568
			}
569
			if ( ! isset( $output['sanitize_callback'] ) && isset( $output['callback'] ) ) {
570
				$this->output[ $key ]['sanitize_callback'] = $output['callback'];
571
			}
572
			// Convert element arrays to strings.
573
			if ( isset( $output['element'] ) && is_array( $output['element'] ) ) {
574
				$this->output[ $key ]['element'] = array_unique( $this->output[ $key ]['element'] );
575
				sort( $this->output[ $key ]['element'] );
576
				$this->output[ $key ]['element'] = implode( ',', $this->output[ $key ]['element'] );
577
			}
578
		}
579
	}
580
581
	/**
582
	 * Sets the $js_vars
583
	 *
584
	 * @access protected
585
	 */
586
	protected function set_js_vars() {
587
588
		if ( ! is_array( $this->js_vars ) ) {
589
			$this->js_vars = array();
590
		}
591
592
		// Check if transport is set to auto.
593
		// If not, then skip the auto-calculations and exit early.
594
		if ( 'auto' !== $this->transport ) {
595
			return;
596
		}
597
598
		// Set transport to refresh initially.
599
		// Serves as a fallback in case we failt to auto-calculate js_vars.
600
		$this->transport = 'refresh';
601
602
		$js_vars = array();
603
604
		// Try to auto-generate js_vars.
605
		// First we need to check if js_vars are empty, and that output is not empty.
606
		if ( empty( $this->js_vars ) && ! empty( $this->output ) ) {
607
608
			// Start going through each item in the $output array.
609
			foreach ( $this->output as $output ) {
610
				$output['function'] = ( isset( $output['function'] ) ) ? $output['function'] : 'style';
611
612
				// If 'element' or 'property' are not defined, skip this.
613
				if ( ! isset( $output['element'] ) || ! isset( $output['property'] ) ) {
614
					continue;
615
				}
616
				if ( is_array( $output['element'] ) ) {
617
					$output['element'] = implode( ',', $output['element'] );
618
				}
619
620
				// If there's a sanitize_callback defined skip this, unless we also have a js_callback defined.
621
				if ( isset( $output['sanitize_callback'] ) && ! empty( $output['sanitize_callback'] ) && ! isset( $output['js_callback'] ) ) {
622
					continue;
623
				}
624
625
				// If we got this far, it's safe to add this.
626
				$js_vars[] = $output;
627
			}
628
629
			// Did we manage to get all the items from 'output'?
630
			// If not, then we're missing something so don't add this.
631
			if ( count( $js_vars ) !== count( $this->output ) ) {
632
				return;
633
			}
634
			$this->js_vars   = $js_vars;
635
			$this->transport = 'postMessage';
636
637
		}
638
	}
639
640
	/**
641
	 * Sets the $variables
642
	 *
643
	 * @access protected
644
	 */
645
	protected function set_variables() {
646
647
		if ( ! is_array( $this->variables ) ) {
648
			$variable = ( is_string( $this->variables ) && ! empty( $this->variables ) ) ? $this->variables : false;
649
			$this->variables = array();
650
			if ( $variable && empty( $this->variables ) ) {
651
				$this->variables[0]['name'] = $variable;
652
			}
653
		}
654
	}
655
656
	/**
657
	 * Sets the $transport
658
	 *
659
	 * @access protected
660
	 */
661
	protected function set_transport() {
662
663
		if ( 'postmessage' === trim( strtolower( $this->transport ) ) ) {
664
			$this->transport = 'postMessage';
665
		}
666
	}
667
668
	/**
669
	 * Sets the $required
670
	 *
671
	 * @access protected
672
	 */
673
	protected function set_required() {
674
675
		if ( ! is_array( $this->required ) ) {
676
			$this->required = array();
677
		}
678
	}
679
680
	/**
681
	 * Sets the $priority
682
	 *
683
	 * @access protected
684
	 */
685
	protected function set_priority() {
686
687
		$this->priority = absint( $this->priority );
688
	}
689
}
690