PluginProbe ʕ •ᴥ•ʔ
Conditional Fields for Contact Form 7 / 2.3.8
Conditional Fields for Contact Form 7 v2.3.8
2.7.8 2.7.7 2.7.6 2.7.5 2.7.4 2.7.3 2.7.2 0.2.4 0.2.5 0.2.6 0.2.7 0.2.8 0.2.9 1.0 1.1 1.2 1.2.1 1.2.2 1.2.3 1.3 1.3.1 1.3.2 1.3.3 1.3.4 1.4 1.4.1 1.4.2 1.4.3 1.5 1.5.1 1.5.2 1.5.3 1.5.4 1.5.5 1.6.1 1.6.2 1.6.3 1.6.5 1.7 1.7.1 1.7.2 1.7.3 1.7.4 1.7.5 1.7.6 1.7.8 1.7.9 1.8 1.8.1 1.8.2 1.8.3 1.8.5 1.8.6 1.8.7 1.9 1.9.1 1.9.10 1.9.11 1.9.12 1.9.13 1.9.14 1.9.15 1.9.16 1.9.2 1.9.3 1.9.4 1.9.5 1.9.6 1.9.7 1.9.8 1.9.9 2.0 2.0.1 2.0.2 2.0.3 2.0.4 2.0.5 2.0.6 2.0.7 2.0.8 2.0.9 2.1 2.1.1 2.1.2 2.1.3 2.1.4 2.1.5 2.1.6 2.2 2.2.1 2.2.10 2.2.11 2.2.2 2.2.3 2.2.4 2.2.5 2.2.6 2.2.7 2.2.8 2.2.9 2.3 2.3.1 2.3.10 2.3.11 2.3.12 2.3.2 2.3.3 2.3.4 2.3.5 2.3.6 2.3.7 2.3.8 2.3.9 2.4 2.4.1 2.4.10 2.4.11 2.4.12 2.4.13 2.4.14 2.4.15 2.4.2 2.4.3 2.4.4 2.4.5 2.4.6 2.4.7 2.4.8 2.4.9 2.5 2.5.1 2.5.10 2.5.11 2.5.12 2.5.13 2.5.14 2.5.2 2.5.3 2.5.4 2.5.5 2.5.6 2.5.7 2.5.8 2.5.9 2.6 2.6.1 2.6.2 2.6.3 2.6.4 2.6.5 2.6.6 2.6.7 2.6.8 2.7 2.7.1 trunk 0.1 0.1.1 0.1.2 0.1.3 0.1.4 0.1.5 0.1.6 0.1.7 0.2 0.2.1 0.2.2 0.2.3
cf7-conditional-fields / cf7cf.php
cf7-conditional-fields Last commit date
js 3 years ago jsdoc-out 4 years ago Wpcf7cfMailParser.php 4 years ago admin-style.css 4 years ago admin-style.css.map 6 years ago admin-style.scss 4 years ago admin.php 3 years ago cf7cf.php 3 years ago contact-form-7-conditional-fields.php 3 years ago init.php 3 years ago readme.txt 3 years ago style.css 3 years ago tg_pane_group.php 5 years ago wpcf7cf-options.php 4 years ago
cf7cf.php
549 lines
1 <?php
2
3 class CF7CF {
4 private $hidden_fields = array();
5 private $visible_groups = array();
6 private $hidden_groups = array();
7 private $repeaters = array();
8
9 function __construct() {
10
11 // Register shortcodes
12 add_action('wpcf7_init', array(__CLASS__, 'add_shortcodes'));
13
14 // Tag generator
15 add_action('admin_init', array(__CLASS__, 'tag_generator'), 590);
16
17 // compatibility with CF7 multi-step forms by Webhead LLC.
18 add_filter( 'wpcf7_posted_data', array($this,'cf7msm_merge_post_with_cookie'), 8, 1 );
19
20 // compatibility with CF7 Multi Step by NinjaTeam https://wordpress.org/plugins/cf7-multi-step/
21 add_action('wp_ajax_cf7mls_validation', array($this,'cf7mls_validation_callback'),9);
22 add_action('wp_ajax_nopriv_cf7mls_validation', array($this,'cf7mls_validation_callback'),9);
23
24 add_filter( 'wpcf7_validate', array($this, 'skip_validation_for_hidden_fields'), 2, 2 );
25
26 add_filter( 'wpcf7_validate_file*', array($this, 'skip_validation_for_hidden_file_field'), 30, 3);
27 add_filter( 'wpcf7_validate_multifile*', array($this, 'skip_validation_for_hidden_file_field'), 30, 3);
28
29 // If acceptance_as_validation is on, then Acceptance fields inside hidden groups should not trigger an error
30 add_filter( 'wpcf7_acceptance', function($accepted, $submission) {
31 $acceptance_as_validation = $submission->get_contact_form()->additional_setting('acceptance_as_validation');
32 return $accepted || (is_array($acceptance_as_validation) && in_array('on', $acceptance_as_validation));
33 }, 20, 2 );
34
35 // validation messages
36 add_action('wpcf7_config_validator_validate', array($this,'wpcf7cf_config_validator_validate'));
37
38 add_action("wpcf7_before_send_mail", [$this, 'hide_hidden_mail_fields'], 10, 3);
39
40 register_activation_hook(__FILE__, array($this, 'activate'));
41
42 if (is_admin()) {
43 require_once dirname(__FILE__) . '/admin.php';
44 }
45 }
46
47
48
49 /**
50 * Suppress invalid mailbox syntax errors on fields that contain existing conditional
51 */
52 function wpcf7cf_config_validator_validate(WPCF7_ConfigValidator $wpcf7_config_validator) {
53
54 // TODO: For now we kill every syntax error once a [groupname] tag is detected.
55 // Ideally, this function should check each string inside the group for invalid syntax.
56 // TODO 2: ajax validation not working yet, because $cf->scan_form_tags() does not seem to contain group tags if it's an ajax request. Need to investigate.
57
58 $cf = $wpcf7_config_validator->contact_form();
59 $all_group_tags = $cf->scan_form_tags();
60
61 foreach ($wpcf7_config_validator->collect_error_messages() as $err_type => $err) {
62
63 // print_r($err_type);
64
65 $parts = explode('.',$err_type);
66
67 $property = $parts[0];
68
69 if ($property == 'form') continue; // the 'form' field can be safely validated by CF7. No need to suppress it.
70
71 $sub_prop = $parts[1];
72 $prop_val = $cf->prop($property)[$sub_prop];
73
74
75 // TODO 2: Dirty hack. Because of TODO 2 we are just going to kill the error message if we detect the string '[/'
76 // Start removing here.
77 if (strpos($prop_val, '[/') !== false) {
78 $wpcf7_config_validator->remove_error($err_type, WPCF7_ConfigValidator::error_invalid_mailbox_syntax);
79 continue;
80 }
81 // TODO 2: Stop removing here. and uncomment code below.
82
83 // foreach ($all_group_tags as $form_tag) {
84 // if (strpos($prop_val, '['.$form_tag->name.']') !== false) {
85 // $wpcf7_config_validator->remove_error($err_type, WPCF7_ConfigValidator::error_invalid_mailbox_syntax);
86 // }
87 // }
88
89 }
90
91 return new WPCF7_ConfigValidator($wpcf7_config_validator->contact_form());
92 }
93
94 function activate() {
95 //add options with add_option and stuff
96 }
97
98 public static function add_shortcodes() {
99 if (function_exists('wpcf7_add_form_tag'))
100 wpcf7_add_form_tag('group', array(__CLASS__, 'shortcode_handler'), true);
101 else if (function_exists('wpcf7_add_shortcode')) {
102 wpcf7_add_shortcode('group', array(__CLASS__, 'shortcode_handler'), true);
103 } else {
104 throw new Exception('functions wpcf7_add_form_tag and wpcf7_add_shortcode not found.');
105 }
106 }
107
108 // TODO: check if we can remove this function. Doesn't seem to be called.
109 function group_shortcode_handler( $atts, $content = "" ) {
110 return $content;
111 }
112
113 // TODO: check if we can remove this function. Doesn't seem to be called.
114 public static function shortcode_handler($tag) {
115 //$tag = new WPCF7_Shortcode($tag);
116 $tag = new WPCF7_FormTag($tag);
117 //ob_start();
118 //print_r($tag);
119 //return print_r($tag, true);
120 return $tag->content;
121 }
122
123
124 public static function tag_generator() {
125 if (! function_exists( 'wpcf7_add_tag_generator'))
126 return;
127
128 wpcf7_add_tag_generator('group',
129 __('Conditional Fields Group', 'cf7-conditional-fields'),
130 'wpcf7-tg-pane-group',
131 array(__CLASS__, 'tg_pane')
132 );
133
134 do_action('wpcf7cf_tag_generator');
135 }
136
137 static function tg_pane( $contact_form, $args = '' ) {
138 $args = wp_parse_args( $args, array() );
139 $type = 'group';
140
141 $description = __( "Generate a group tag to group form elements that can be shown conditionally.", 'cf7-conditional-fields' );
142
143 include 'tg_pane_group.php';
144 }
145
146 /**
147 * Remove validation requirements for fields that are hidden at the time of form submission.
148 * Required/invalid fields should never trigger validation errors if they are inside a hidden group during submission.
149 * Called using add_filter( 'wpcf7_validate', array($this, 'skip_validation_for_hidden_fields'), 2, 2 );
150 * where the priority of 2 causes this to kill any validations with a priority higher than 2
151 *
152 * NOTE: CF7 is weirdly designed when it comes to validating a form with files.
153 * Only the non-file fields are considered during the wpcf7_validate filter.
154 * When validation passes for all fields (except the file fields), the files fields are validated individually.
155 * ( see skip_validation_for_hidden_file_field )
156 *
157 * @param $result
158 * @param $tag
159 *
160 * @return mixed
161 */
162 function skip_validation_for_hidden_fields($result, $tags, $args = []) {
163
164 if(isset($_POST)) {
165 $this->set_hidden_fields_arrays($_POST);
166 }
167
168 $invalid_fields = $result->get_invalid_fields();
169 $return_result = new WPCF7_Validation();
170
171 if (count($this->hidden_fields) == 0 || !is_array($invalid_fields) || count($invalid_fields) == 0) {
172 $return_result = $result;
173 } else {
174 foreach ($invalid_fields as $invalid_field_key => $invalid_field_data) {
175 if (!in_array($invalid_field_key, $this->hidden_fields)) {
176 foreach ($tags as $tag) {
177 if ($tag['name'] === $invalid_field_key) {
178 $return_result->invalidate($tag, $invalid_field_data['reason']);
179 }
180 }
181 }
182 }
183 }
184
185 return apply_filters('wpcf7cf_validate', $return_result, $tags);
186
187 }
188
189 /**
190 * Does the same thing as skip_validation_for_hidden_fields, but CF7 will check files again later
191 * via the wpcf7_unship_uploaded_files function
192 * so we need to skip validation a second time for individual file fields
193 */
194 function skip_validation_for_hidden_file_field($result, $tag, $args=[]) {
195
196 if (!count($result->get_invalid_fields())) {
197 return $result;
198 }
199 if(isset($_POST)) {
200 $this->set_hidden_fields_arrays($_POST);
201 }
202
203 $invalid_field_keys = array_keys($result->get_invalid_fields());
204
205 // if the current file is the only invalid tag in the result AND if the file is hidden: return a valid (blank) object
206 if (isset($this->hidden_fields) && is_array($this->hidden_fields) && in_array($tag->name, $this->hidden_fields) && count($invalid_field_keys) == 1) {
207 return new WPCF7_Validation();
208 }
209
210 // if the current file is not hidden, we'll just return the result (keep it invalid).
211 // (Note that this might also return the hidden files as invalid, but that shouldn't matter because the form is invalid, and the notification will be inside a hidden group)
212 return $result;
213 }
214
215 function cf7msm_merge_post_with_cookie($posted_data) {
216
217 if (!function_exists('cf7msm_get') || !key_exists('cf7msm_posted_data',$_COOKIE)) return $posted_data;
218
219 if (!$posted_data) {
220 $posted_data = WPCF7_Submission::get_instance()->get_posted_data();
221 }
222
223 // this will temporarily set the hidden fields data to the posted_data.
224 // later this function will be called again with the updated posted_data
225 $this->set_hidden_fields_arrays($_POST);
226
227 // get cookie data
228 $cookie_data = cf7msm_get('cf7msm_posted_data');
229 $cookie_data_hidden_group_fields = json_decode(stripslashes($cookie_data['_wpcf7cf_hidden_group_fields']));
230 $cookie_data_hidden_groups = json_decode(stripslashes($cookie_data['_wpcf7cf_hidden_groups']));
231 $cookie_data_visible_groups = json_decode(stripslashes($cookie_data['_wpcf7cf_visible_groups']));
232
233 // remove all the currently posted data from the cookie data (we don't wanna add it twice)
234 $cookie_data_hidden_group_fields = array_diff($cookie_data_hidden_group_fields, array_keys($posted_data));
235 $cookie_data_hidden_groups = array_diff((array) $cookie_data_hidden_groups, $this->hidden_groups, $this->visible_groups);
236 $cookie_data_visible_groups = array_diff((array) $cookie_data_visible_groups, $this->hidden_groups, $this->visible_groups);
237
238 // update current post data with cookie data
239 $posted_data['_wpcf7cf_hidden_group_fields'] = addslashes(json_encode(array_merge((array) $cookie_data_hidden_group_fields, $this->hidden_fields)));
240 $posted_data['_wpcf7cf_hidden_groups'] = addslashes(json_encode(array_merge((array) $cookie_data_hidden_groups, $this->hidden_groups)));
241 $posted_data['_wpcf7cf_visible_groups'] = addslashes(json_encode(array_merge((array) $cookie_data_visible_groups, $this->visible_groups)));
242
243 return $posted_data;
244 }
245
246 // compatibility with CF7 Multi Step by NinjaTeam https://wordpress.org/plugins/cf7-multi-step/
247 function cf7mls_validation_callback() {
248 $this->set_hidden_fields_arrays($_POST);
249 }
250
251 /**
252 * Finds the currently submitted form and set the hidden_fields variables accoringly
253 *
254 * @param bool|array $posted_data
255 */
256 function set_hidden_fields_arrays($posted_data = false) {
257
258 if (!$posted_data) $posted_data = $_POST;
259
260 $hidden_fields = json_decode(stripslashes($posted_data['_wpcf7cf_hidden_group_fields']));
261 if (is_array($hidden_fields) && count($hidden_fields) > 0) {
262 foreach ($hidden_fields as $field) {
263 $this->hidden_fields[] = $field;
264 if (wpcf7cf_endswith($field, '[]')) {
265 $this->hidden_fields[] = substr($field,0,strlen($field)-2);
266 }
267 }
268 }
269 $this->hidden_groups = json_decode(stripslashes($posted_data['_wpcf7cf_hidden_groups']));
270 $this->visible_groups = json_decode(stripslashes($posted_data['_wpcf7cf_visible_groups']));
271 $this->repeaters = json_decode(stripslashes($posted_data['_wpcf7cf_repeaters']));
272 $this->steps = json_decode(stripslashes($posted_data['_wpcf7cf_steps']));
273 }
274
275 function hide_hidden_mail_fields($form,$abort,$submission) {
276 $props = $form->get_properties();
277 $mails = ['mail','mail_2','messages'];
278 foreach ($mails as $mail) {
279 if (!is_array($props[$mail])) { continue; }
280 foreach ($props[$mail] as $key=>$val) {
281
282 // remove unwanted whitespace between closing and opening groups from email
283 $count = 1;
284 while ($count) {
285 $val = preg_replace(WPCF7CF_REGEX_MAIL_UNWANTED_WHITESPACE, '$1$2', $val, -1, $count);
286 }
287
288 // remove hiddden groups from email
289 $parser = new Wpcf7cfMailParser($val, $this->visible_groups, $this->hidden_groups, $this->repeaters, $_POST);
290 $props[$mail][$key] = $parser->getParsedMail();
291 }
292 }
293
294
295 //$props['mail']['body'] = 'xxx';
296 $form->set_properties($props);
297 }
298
299 function hide_hidden_mail_fields_regex_callback ( $matches ) {
300 $name = $matches[1];
301 $content = $matches[2];
302 if ( in_array( $name, $this->hidden_groups ) ) {
303 // The tag name represents a hidden group, so replace everything from [tagname] to [/tagname] with nothing
304 return '';
305 } elseif ( in_array( $name, $this->visible_groups ) ) {
306 // The tag name represents a visible group, so remove the tags themselves, but return everything else
307 // instead of just returning the $content, return the preg_replaced content :)
308 return preg_replace_callback(WPCF7CF_REGEX_MAIL_GROUP, array($this, 'hide_hidden_mail_fields_regex_callback'), $content );
309 } else {
310 // The tag name doesn't represent a group that was used in the form. Leave it alone (return the entire match).
311 return $matches[0];
312 }
313 }
314
315 public static function parse_conditions($string, $format='array') {
316 // Parse stuff like "show [g1] if [field] equals 2" to Array
317
318 preg_match_all(WPCF7CF_REGEX_CONDITIONS, $string, $matches);
319
320 $conditions = [];
321
322 $prev_then_field = '';
323 foreach ($matches[0] as $i=>$line) {
324 $then_field = $matches[1][$i];
325 $if_field = $matches[2][$i];
326 $operator = $matches[3][$i];
327 $if_value = $matches[4][$i];
328
329 $index = count($conditions);
330
331 if ($then_field == '') {
332 $index = $index -1;
333 $then_field = $prev_then_field;
334 } else {
335 $conditions[$index]['then_field'] = $then_field;
336 }
337
338 $conditions[$index]['and_rules'][] = [
339 'if_field' => $if_field,
340 'operator' => $operator,
341 'if_value' => $if_value,
342 ];
343
344 $prev_then_field = $then_field;
345
346 }
347
348 $conditions = array_values($conditions);
349
350 if ($format == 'array') {
351 return $conditions;
352 } else if ($format == 'json') {
353 return json_encode($conditions);
354 }
355 }
356
357 /**
358 * load the conditions from the form's post_meta
359 *
360 * @param string $form_id
361 * @return array
362 */
363 public static function getConditions($form_id) {
364 // make sure conditions are an array.
365 $options = get_post_meta($form_id,'wpcf7cf_options',true);
366 return is_array($options) ? $options : array(); // the meta key 'wpcf7cf_options' is a bit misleading at this point, because it only holds the form's conditions, no other options/settings
367 }
368
369 /**
370 * load the conditions from the form's post_meta as plain text
371 *
372 * @param string $form_id
373 * @return void
374 */
375 public static function getConditionsPlainText($form_id) {
376 return CF7CF::serializeConditions(CF7CF::getConditions($form_id));
377 }
378
379 public static function serializeConditions($array) {
380
381 $lines = [];
382
383 foreach ($array as $entry) {
384 $then_field = $entry['then_field'];
385 $and_rules = $entry['and_rules'];
386 $indent = strlen($then_field) + 4;
387 foreach ($and_rules as $i => $rule) {
388 $if_field = $rule['if_field'];
389 $operator = $rule['operator'];
390 $if_value = $rule['if_value'];
391
392 if ($i == 0) {
393 $lines[] = "show [$then_field] if [$if_field] $operator \"$if_value\"";
394 } else {
395 $lines[] = str_repeat(' ',$indent)."and if [$if_field] $operator \"$if_value\"";
396 }
397 }
398 }
399
400 return implode("\n", $lines);
401 }
402
403 /**
404 * save the conditions to the form's post_meta
405 *
406 * @param string $form_id
407 * @return void
408 */
409 public static function setConditions($form_id, $conditions) {
410 return update_post_meta($form_id,'wpcf7cf_options',$conditions); // the meta key 'wpcf7cf_options' is a bit misleading at this point, because it only holds the form's conditions, no other options/settings
411 }
412 }
413
414 new CF7CF;
415
416 add_filter( 'wpcf7_contact_form_properties', 'wpcf7cf_properties', 10, 2 );
417
418 function wpcf7cf_properties($properties, $wpcf7form) {
419 // Before CF7 5.5.3, this function was called each time we call get_properties() on a contact form. Since CF7 5.5.3 this function is called only once in the WPCF7_ContactForm
420 if (!is_admin() || (defined('DOING_AJAX') && DOING_AJAX)) { // TODO: kind of hacky. maybe find a better solution. Needed because otherwise the group tags will be replaced in the editor as well.
421 $form = $properties['form'];
422
423 $form_parts = preg_split('/(\[\/?group(?:\]|\s.*?\]))/',$form, -1,PREG_SPLIT_NO_EMPTY | PREG_SPLIT_DELIM_CAPTURE);
424
425 ob_start();
426
427 $stack = array();
428
429 foreach ($form_parts as $form_part) {
430 if (substr($form_part,0,7) == '[group ') {
431 $tag_parts = explode(' ',rtrim($form_part,']'));
432
433 array_shift($tag_parts);
434
435 $tag_id = $tag_parts[0];
436 $tag_html_type = 'div';
437 $tag_html_data = array();
438
439 foreach ($tag_parts as $i => $tag_part) {
440 if ($i==0) continue;
441 $tag_part = explode(':',$tag_part);
442 if ($tag_part[0] == 'inline') $tag_html_type = 'span';
443 else if ($tag_part[0] == 'clear_on_hide') $tag_html_data[] = 'data-clear_on_hide';
444 else if ($tag_part[0] == 'disable_on_hide' && WPCF7CF_IS_PRO) $tag_html_data[] = 'data-disable_on_hide';
445 else if ($tag_part[0] == 'class') $tag_html_data[] = 'class="'.($tag_part[1]??'').'"';
446 }
447
448 array_push($stack,$tag_html_type);
449
450 echo '<'.$tag_html_type.' data-id="'.$tag_id.'" data-orig_data_id="'.$tag_id.'" '.implode(' ',$tag_html_data).' data-class="wpcf7cf_group">';
451 } else if ($form_part == '[/group]') {
452 echo '</'.array_pop($stack).'>';
453 } else {
454 echo $form_part;
455 }
456 }
457
458 $properties['form'] = ob_get_clean();
459 }
460 return $properties;
461 }
462
463 add_filter('wpcf7_form_hidden_fields', 'wpcf7cf_form_hidden_fields',10,1);
464
465 function wpcf7cf_form_hidden_fields($hidden_fields) {
466
467 $current_form = wpcf7_get_current_contact_form();
468 $current_form_id = $current_form->id();
469
470 $options = array(
471 'form_id' => $current_form_id,
472 'conditions' => CF7CF::getConditions($current_form_id),
473 'settings' => wpcf7cf_get_settings()
474 );
475
476 unset($options['settings']['license_key']); // don't show license key in the source code duh.
477
478 return array_merge($hidden_fields, array(
479 '_wpcf7cf_hidden_group_fields' => '[]',
480 '_wpcf7cf_hidden_groups' => '[]',
481 '_wpcf7cf_visible_groups' => '[]',
482 '_wpcf7cf_repeaters' => '[]',
483 '_wpcf7cf_steps' => '{}',
484 '_wpcf7cf_options' => ''.json_encode($options),
485 ));
486 }
487
488 function wpcf7cf_endswith($string, $test) {
489 $strlen = strlen($string);
490 $testlen = strlen($test);
491 if ($testlen > $strlen) return false;
492 return substr_compare($string, $test, $strlen - $testlen, $testlen) === 0;
493 }
494
495 add_filter( 'wpcf7_form_tag_data_option', 'wpcf7cf_form_tag_data_option', 10, 3 );
496
497 function wpcf7cf_form_tag_data_option($output, $args, $nog) {
498 $data = array();
499 return $data;
500 }
501
502 /* Scripts & Styles */
503
504 function wpcf7cf_load_js() {
505 return apply_filters( 'wpcf7cf_load_js', WPCF7CF_LOAD_JS );
506 }
507
508 function wpcf7cf_load_css() {
509 return apply_filters( 'wpcf7cf_load_css', WPCF7CF_LOAD_CSS );
510 }
511
512 add_action( 'wp_enqueue_scripts', 'wpcf7cf_do_enqueue_scripts', 20, 0 );
513
514 function wpcf7cf_do_enqueue_scripts() {
515 if ( wpcf7cf_load_js() ) {
516 wpcf7cf_enqueue_scripts();
517 }
518
519 if ( wpcf7cf_load_css() ) {
520 wpcf7cf_enqueue_styles();
521 }
522 }
523
524 function wpcf7cf_enqueue_scripts() {
525 if (is_admin()) return;
526 wp_enqueue_script('wpcf7cf-scripts', plugins_url('js/scripts.js', __FILE__), array('jquery'), WPCF7CF_VERSION, true);
527 wp_localize_script('wpcf7cf-scripts', 'wpcf7cf_global_settings',
528 array(
529 'ajaxurl' => admin_url('admin-ajax.php'),
530 )
531 );
532
533 }
534
535 function wpcf7cf_enqueue_styles() {
536 if (is_admin()) return;
537 wp_enqueue_style('cf7cf-style', plugins_url('style.css', __FILE__), array(), WPCF7CF_VERSION);
538 }
539
540 // Make sure CF7 doesn't target any disabled fields for validation
541 // (HTML standard: "disabled fields don't get submitted", so no need to validate them)
542 add_filter( 'wpcf7_feedback_response', function($response, $result) {
543 foreach ($response['invalid_fields'] as $i => $inv) {
544 if (isset($response['invalid_fields'][$i]['into'])) {
545 $response['invalid_fields'][$i]['into'] .= ':not(.wpcf7cf-disabled)';
546 }
547 }
548 return $response;
549 }, 2, 10 );