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