Create New Item
Item Type
File
Folder
Item Name
Search file in folder and subfolders...
Are you sure want to rename?
forbidals
/
wp-content
/
plugins
/
formidable
/
classes
/
controllers
:
FrmFieldsController.php
Advanced Search
Upload
New Item
Settings
Back
Back Up
Advanced Editor
Save
<?php if ( ! defined( 'ABSPATH' ) ) { die( 'You are not allowed to call this page directly.' ); } class FrmFieldsController { /** * This is stored statically so we can re-use this data for every field. * * @var FrmFieldSelectionData|null */ private static $field_selection_data; public static function load_field() { FrmAppHelper::permission_check( 'frm_edit_forms' ); check_ajax_referer( 'frm_ajax', 'nonce' ); // Javascript may be included in some field settings. // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized $fields = isset( $_POST['field'] ) ? wp_unslash( $_POST['field'] ) : array(); if ( ! $fields ) { wp_die(); } $_GET['page'] = 'formidable'; $values = array( 'id' => FrmAppHelper::get_post_param( 'form_id', '', 'absint' ), 'doing_ajax' => true, ); $field_html = array(); foreach ( $fields as $field ) { $field = htmlspecialchars_decode( nl2br( $field ) ); $field = json_decode( $field ); if ( ! isset( $field->id ) || ! is_numeric( $field->id ) ) { // this field may have already been loaded continue; } if ( ! isset( $field->value ) ) { $field->value = ''; } $field->field_options = json_decode( json_encode( $field->field_options ), true ); $field->options = json_decode( json_encode( $field->options ), true ); $field->default_value = json_decode( json_encode( $field->default_value ), true ); ob_start(); self::load_single_field( $field, $values ); $field_html[ absint( $field->id ) ] = ob_get_clean(); }//end foreach echo json_encode( $field_html ); wp_die(); } /** * Create a new field with ajax */ public static function create() { FrmAppHelper::permission_check( 'frm_edit_forms' ); check_ajax_referer( 'frm_ajax', 'nonce' ); $field_type = FrmAppHelper::get_post_param( 'field_type', '', 'sanitize_text_field' ); $form_id = FrmAppHelper::get_post_param( 'form_id', 0, 'absint' ); $field_options = FrmAppHelper::get_post_param( 'field_options', array(), 'wp_kses_post' ); do_action( 'frm_before_create_field', $field_type, $form_id ); $field = self::include_new_field( $field_type, $form_id, $field_options ); // this hook will allow for multiple fields to be added at once do_action( 'frm_after_field_created', $field, $form_id ); wp_die(); } /** * Set up and create a new field * * @param string $field_type * @param int $form_id * @param array $field_options * * @return array|false */ public static function include_new_field( $field_type, $form_id, $field_options = array() ) { $field_values = FrmFieldsHelper::setup_new_vars( $field_type, $form_id ); if ( $field_options ) { $field_values['field_options'] = array_merge( $field_values['field_options'], $field_options ); } // When a new field is added to the form, flag it as draft and hide it from the front-end. $field_values['field_options']['draft'] = 1; /** * @param array $field_values */ $field_values = apply_filters( 'frm_before_field_created', $field_values ); $field_id = FrmField::create( $field_values ); if ( ! $field_id ) { return false; } $field = self::get_field_array_from_id( $field_id ); $values = array(); if ( FrmAppHelper::pro_is_installed() ) { $values['post_type'] = FrmProFormsHelper::post_type( $form_id ); $parent_form_id = FrmDb::get_var( 'frm_forms', array( 'id' => $form_id ), 'parent_form_id' ); if ( $parent_form_id ) { $field['parent_form_id'] = $parent_form_id; } } self::load_single_field( $field, $values, $form_id ); return $field; } public static function duplicate() { FrmAppHelper::permission_check( 'frm_edit_forms' ); check_ajax_referer( 'frm_ajax', 'nonce' ); $field_id = FrmAppHelper::get_post_param( 'field_id', 0, 'absint' ); $form_id = FrmAppHelper::get_post_param( 'form_id', 0, 'absint' ); $new_field = FrmField::duplicate_single_field( $field_id, $form_id ); if ( is_array( $new_field ) && ! empty( $new_field['field_id'] ) ) { self::load_single_field( $new_field['field_id'], $new_field['values'] ); } wp_die(); } /** * @since 3.0 * * @param int $field_id * * @return array */ public static function get_field_array_from_id( $field_id ) { $field = FrmField::getOne( $field_id ); return FrmFieldsHelper::setup_edit_vars( $field ); } /** * @since 3.0 * * @param array|int|object $field_object * @param array $values * @param int $form_id * * @return void */ public static function load_single_field( $field_object, $values, $form_id = 0 ) { global $frm_vars; $frm_vars['is_admin'] = true; if ( is_numeric( $field_object ) ) { $field_object = FrmField::getOne( $field_object ); } elseif ( is_array( $field_object ) ) { $field = $field_object; $field_object = FrmField::getOne( $field['id'] ); } $field_obj = FrmFieldFactory::get_field_factory( $field_object ); $display = self::display_field_options( array(), $field_obj ); $ajax_loading = ! empty( $values['ajax_load'] ); $ajax_this_field = isset( $values['count'] ) && $values['count'] > 10 && ! in_array( $field_object->type, array( 'divider', 'end_divider' ), true ); if ( $ajax_loading && $ajax_this_field ) { $li_classes = self::get_classes_for_builder_field( array(), $display, $field_obj ); include FrmAppHelper::plugin_path() . '/classes/views/frm-fields/back-end/ajax-field-placeholder.php'; return; } if ( ! isset( $field ) && is_object( $field_object ) ) { $field_object->parent_form_id = $values['id'] ?? $field_object->form_id; $field = FrmFieldsHelper::setup_edit_vars( $field_object ); } /** * Filter for adding extra attributes to the field container. * * @since 6.23 * * @param array $extra_field_attributes * @param array $field * @param array $display */ $extra_field_attributes = apply_filters( 'frm_field_container_extra_attributes', array(), $field, $display ); $li_classes = self::get_classes_for_builder_field( $field, $display, $field_obj ); $li_classes .= ' ui-state-default widgets-holder-wrap'; require FrmAppHelper::plugin_path() . '/classes/views/frm-forms/add_field.php'; } /** * @since 3.0 * * @param array $field * @param array $display * @param FrmFieldType $field_info * * @return string */ private static function get_classes_for_builder_field( $field, $display, $field_info ) { $li_classes = $field_info->form_builder_classes( $display['type'] ); $li_classes .= ' frm_form_field frmstart '; if ( isset( $field['classes'] ) ) { $li_classes .= trim( $field['classes'] ) . ' '; } $li_classes .= 'frmend'; if ( $field ) { $li_classes = apply_filters( 'frm_build_field_class', $li_classes, $field ); } return $li_classes; } public static function destroy() { FrmAppHelper::permission_check( 'frm_edit_forms' ); check_ajax_referer( 'frm_ajax', 'nonce' ); $field_id = FrmAppHelper::get_post_param( 'field_id', 0, 'absint' ); FrmField::destroy( $field_id ); wp_die(); } /** * Field Options. */ public static function import_options() { FrmAppHelper::permission_check( 'frm_edit_forms' ); check_ajax_referer( 'frm_ajax', 'nonce' ); if ( ! is_admin() || ! current_user_can( 'frm_edit_forms' ) ) { return; } $field_id = FrmAppHelper::get_param( 'field_id', '', 'post', 'absint' ); $field = FrmField::getOne( $field_id ); $bulk_edit_types = array( 'radio', 'checkbox', 'select' ); /** * Filter to add new fields that will support import_options/Bulk Edit Options. * * @since 6.8.4 * * @param array $bulk_edit_types * * @return array */ $bulk_edit_types = apply_filters( 'frm_bulk_edit_field_types', $bulk_edit_types ); if ( ! in_array( $field->type, $bulk_edit_types, true ) ) { return; } $field = FrmFieldsHelper::setup_edit_vars( $field ); $opts = FrmAppHelper::get_param( 'opts', '', 'post', 'wp_kses_post' ); $opts = explode( "\n", rtrim( $opts, "\n" ) ); $opts = array_map( 'trim', $opts ); $separate = FrmAppHelper::get_param( 'separate', '', 'post', 'sanitize_text_field' ); $field['separate_value'] = $separate === 'true'; if ( $field['separate_value'] ) { foreach ( $opts as $opt_key => $opt ) { if ( str_contains( $opt, '|' ) ) { $vals = explode( '|', $opt ); $opts[ $opt_key ] = array( 'label' => trim( $vals[0] ), 'value' => trim( $vals[1] ), ); unset( $vals ); } unset( $opt_key, $opt ); } } // Keep other options after bulk update. // phpcs:ignore Universal.Operators.StrictComparisons if ( ! empty( $field['field_options']['other'] ) ) { $other_array = array(); foreach ( $field['options'] as $opt_key => $opt ) { if ( FrmFieldsHelper::is_other_opt( $opt_key ) ) { $other_array[ $opt_key ] = $opt; } unset( $opt_key, $opt ); } if ( $other_array ) { $opts = array_merge( $opts, $other_array ); } } $field['options'] = $opts; FrmFieldsHelper::show_single_option( $field ); wp_die(); } /** * @since 4.0 * * @param array $atts - Includes field array, field_obj, display array, values array. * * @return void */ public static function load_single_field_settings( $atts ) { $field = $atts['field']; $field_obj = $atts['field_obj']; $values = $atts['values']; $display = $atts['display']; unset( $atts ); if ( ! isset( $field['unique'] ) ) { $field['unique'] = false; } if ( ! isset( $field['read_only'] ) ) { $field['read_only'] = false; } $field_selection_data = self::maybe_define_field_selection_data(); $all_field_types = $field_selection_data->all_field_types; $disabled_fields = $field_selection_data->disabled_fields; $frm_settings = FrmAppHelper::get_settings(); $field_types = FrmFieldTypeOptionData::get_field_types( $field['type'] ); if ( ! isset( $all_field_types[ $field['type'] ] ) ) { // Add fallback for an add-on field type that has been deactivated. $all_field_types[ $field['type'] ] = array( 'name' => ucfirst( $field['type'] ), 'icon' => 'frmfont frm_pencil_icon', ); } elseif ( ! is_array( $all_field_types[ $field['type'] ] ) ) { // Fallback for fields added in a more basic way. FrmFormsHelper::prepare_field_type( $all_field_types[ $field['type'] ] ); } $type_name = $all_field_types[ $field['type'] ]['name']; if ( $field['type'] === 'divider' && FrmField::is_option_true( $field, 'repeat' ) ) { $type_name = $all_field_types['divider|repeat']['name']; } if ( $display['default'] ) { $default_value_types = self::default_value_types( $field, compact( 'display' ) ); } if ( $display['clear_on_focus'] && is_array( $field['placeholder'] ) ) { $field['placeholder'] = implode( ', ', $field['placeholder'] ); } include FrmAppHelper::plugin_path() . '/classes/views/frm-fields/back-end/settings.php'; } /** * @since 6.9.1 * * @return FrmFieldSelectionData */ private static function maybe_define_field_selection_data() { if ( ! isset( self::$field_selection_data ) ) { self::$field_selection_data = new FrmFieldSelectionData(); } return self::$field_selection_data; } /** * Get the list of default value types that can be toggled in the builder. * * @since 4.0 * * @param array $field * @param array $atts * * @return array */ private static function default_value_types( $field, $atts ) { $types = array( 'default_value' => array( 'class' => '', 'icon' => 'frmfont frm_text2_icon', 'title' => __( 'Default Value', 'formidable' ), 'data' => array( 'frmshow' => '#default-value-for-', ), ), 'calc' => array( 'class' => 'frm_show_upgrade frm_noallow', 'title' => __( 'Calculate Value', 'formidable' ), 'icon' => 'frmfont frm_calculator_icon', 'data' => array( 'medium' => 'calculations', 'upgrade' => __( 'Calculator forms', 'formidable' ), ), 'tooltip' => __( 'Automatically calculate the value of this field based on values from other fields.', 'formidable' ), ), 'get_values_field' => array( 'class' => 'frm_show_upgrade frm_noallow', 'title' => __( 'Lookup', 'formidable' ), 'icon' => 'frmfont frm_search_icon', 'data' => array( 'medium' => 'lookup', 'upgrade' => __( 'Lookup fields', 'formidable' ), ), 'tooltip' => __( 'Dynamically retrieve the value of this field from a lookup field.', 'formidable' ), ), ); $types = apply_filters( 'frm_default_value_types', $types, $atts ); // Set active class. $settings = array_keys( $types ); $active = 'default_value'; if ( FrmAppHelper::pro_is_connected() ) { foreach ( $settings as $type ) { if ( ! empty( $field[ $type ] ) ) { $active = $type; } } } $types[ $active ]['class'] .= ' current'; $types[ $active ]['current'] = true; return $types; } /** * @param string $type * * @return string */ public static function change_type( $type ) { $type_switch = array( 'scale' => 'radio', 'star' => 'radio', '10radio' => 'radio', 'rte' => 'textarea', 'website' => 'url', 'image' => 'url', ); if ( isset( $type_switch[ $type ] ) ) { $type = $type_switch[ $type ]; } $pro_fields = FrmField::pro_field_selection(); // We want to keep credit_card types as credit card types for Stripe Lite. // The credit_card key is set for backward compatibility. unset( $pro_fields['credit_card'] ); if ( array_key_exists( $type, $pro_fields ) ) { $type = 'text'; } return $type; } /** * @param array $settings * @param FrmFieldType|null $field_info * * @return array */ public static function display_field_options( $settings, $field_info = null ) { if ( $field_info ) { $settings = $field_info->display_field_settings(); // Field appears to be protected but there is a __get function in FrmFieldType that makes this public. $settings['field_data'] = $field_info->field; } return apply_filters( 'frm_display_field_options', $settings ); } /** * Display the format option * * @since 3.0 * * @param array $field Field data. * @param bool $is_hidden Whether the format option should be hidden. * * @return void */ public static function show_format_option( $field, $is_hidden = false ) { $attributes = array( 'class' => 'frm-has-modal', 'id' => 'frm-field-format-custom-' . $field['id'], ); if ( $is_hidden ) { $attributes['class'] .= ' frm_hidden'; } include FrmAppHelper::plugin_path() . '/classes/views/frm-fields/back-end/value-format.php'; } /** * @param array $field * @param bool $echo * * @return string */ public static function input_html( $field, $echo = true ) { $class = array(); self::add_input_classes( $field, $class ); $add_html = array(); self::add_html_size( $field, $add_html ); self::add_html_length( $field, $add_html ); self::add_html_placeholder( $field, $add_html, $class ); self::add_validation_messages( $field, $add_html ); $class = apply_filters( 'frm_field_classes', implode( ' ', $class ), $field ); FrmFormsHelper::add_html_attr( $class, 'class', $add_html ); self::add_shortcodes_to_html( $field, $add_html ); self::add_pattern_attribute( $field, $add_html ); $add_html = apply_filters( 'frm_field_extra_html', $add_html, $field ); $add_html = ' ' . implode( ' ', $add_html ) . ' '; if ( $echo ) { echo $add_html; // phpcs:ignore WordPress.Security.EscapeOutput.OutputNotEscaped } return $add_html; } /** * @param array $field * @param array $class * * @return void */ private static function add_input_classes( $field, array &$class ) { if ( ! empty( $field['input_class'] ) ) { $class[] = $field['input_class']; } if ( $field['type'] === 'hidden' || $field['type'] === 'user_id' ) { return; } if ( isset( $field['size'] ) && $field['size'] > 0 ) { $class[] = 'auto_width'; } } /** * @param array $field * @param array $add_html * * @return void */ private static function add_html_size( $field, array &$add_html ) { $size_fields = array( 'select', 'data', 'time', 'hidden', 'file', 'lookup', ); if ( ! isset( $field['size'] ) || $field['size'] <= 0 || in_array( $field['type'], $size_fields, true ) ) { return; } if ( FrmAppHelper::is_admin_page( 'formidable' ) ) { return; } if ( is_numeric( $field['size'] ) ) { $field['size'] .= 'px'; } $important = apply_filters( 'frm_use_important_width', 1, $field ); // Note: This inline styling must stay since we cannot realistically set a class for every possible field size. $add_html['style'] = 'style="width:' . esc_attr( $field['size'] ) . ( $important ? ' !important' : '' ) . '"'; self::add_html_cols( $field, $add_html ); } /** * @param array $field * @param array $add_html * * @return void */ private static function add_html_cols( $field, array &$add_html ) { if ( ! in_array( $field['type'], array( 'textarea', 'rte' ), true ) ) { return; } // Convert to cols for textareas. $calc = array( '' => 9, 'px' => 9, 'rem' => 0.444, 'em' => 0.544, ); // include "col" for valid html $unit = trim( preg_replace( '/[0-9]+/', '', $field['size'] ) ); if ( ! isset( $calc[ $unit ] ) ) { return; } $size = (float) str_replace( $unit, '', $field['size'] ) / $calc[ $unit ]; $add_html['cols'] = 'cols="' . absint( $size ) . '"'; } /** * @param array $field * @param array $add_html * * @return void */ private static function add_html_length( $field, array &$add_html ) { // Check for max setting and if this field accepts maxlength. $fields = array( 'textarea', 'rte', 'hidden', 'file', ); if ( FrmField::is_option_empty( $field, 'max' ) || in_array( $field['type'], $fields, true ) ) { return; } if ( FrmAppHelper::is_admin_page( 'formidable' ) ) { // Don't load on form builder page. return; } $add_html['maxlength'] = 'maxlength="' . esc_attr( $field['max'] ) . '"'; } /** * @param array $field * @param array $add_html * @param array $class * * @return void */ private static function add_html_placeholder( $field, array &$add_html, array &$class ) { // phpcs:ignore Universal.Operators.StrictComparisons if ( $field['default_value'] != '' ) { if ( is_array( $field['default_value'] ) ) { $add_html['data-frmval'] = 'data-frmval="' . esc_attr( json_encode( $field['default_value'] ) ) . '"'; } else { self::add_frmval_to_input( $field, $add_html ); } } $field['placeholder'] = self::prepare_placeholder( $field ); // phpcs:ignore Universal.Operators.StrictComparisons if ( $field['placeholder'] == '' || is_array( $field['placeholder'] ) ) { // don't include a json placeholder return; } self::add_placeholder_to_input( $field, $add_html ); } /** * Prepares field placeholder. * * @since 5.4 This doesn't call `FrmFieldsController::get_default_value_from_name()` anymore. * * @param array $field Field array. * * @return string */ private static function prepare_placeholder( $field ) { return $field['placeholder'] ?? ''; } /** * If the label position is "inside", * get the label to use as the placeholder * * @since 2.05 * @since 5.4 Remove the logic code for "inside" label position. * * @param array $field * * @return string */ public static function get_default_value_from_name( $field ) { return ''; } /** * Maybe add a blank placeholder option before any options * in a dropdown. * * @since 4.04 * * @param array|object $field * * @return bool True if placeholder was added. */ public static function add_placeholder_to_select( $field ) { $placeholder = FrmField::get_option( $field, 'placeholder' ); if ( ! $placeholder ) { $placeholder = self::get_default_value_from_name( $field ); } $use_placeholder = $placeholder; $autocomplete = FrmField::get_option( $field, 'autocom' ); if ( $autocomplete ) { $use_chosen = ! is_callable( 'FrmProAppHelper::use_chosen_js' ) || FrmProAppHelper::use_chosen_js(); if ( $use_chosen ) { $use_placeholder = ''; } } if ( $placeholder !== '' ) { $placeholder_attributes = array( 'class' => 'frm-select-placeholder', 'value' => '', 'data-placeholder' => 'true', ); FrmHtmlHelper::echo_dropdown_option( $use_placeholder, false, $placeholder_attributes ); return true; } return false; } /** * Use HTML5 placeholder with js fallback. * * @param array $field * @param array $add_html * * @return void */ private static function add_placeholder_to_input( $field, &$add_html ) { if ( FrmFieldsHelper::is_placeholder_field_type( $field['type'] ) ) { $add_html['placeholder'] = 'placeholder="' . esc_attr( $field['placeholder'] ) . '"'; } } /** * @param array $field * @param array $add_html * * @return void */ private static function add_frmval_to_input( $field, &$add_html ) { // phpcs:ignore Universal.Operators.StrictComparisons if ( $field['placeholder'] != '' ) { $add_html['data-frmval'] = 'data-frmval="' . esc_attr( $field['placeholder'] ) . '"'; if ( 'select' === $field['type'] ) { $add_html['data-frmplaceholder'] = 'data-frmplaceholder="' . esc_attr( $field['placeholder'] ) . '"'; } } // phpcs:ignore Universal.Operators.StrictComparisons if ( $field['default_value'] != '' ) { $add_html['data-frmval'] = 'data-frmval="' . esc_attr( $field['default_value'] ) . '"'; } } /** * @param array $field * @param array $add_html * * @return void */ private static function add_validation_messages( $field, array &$add_html ) { $field_validation_messages_status = self::get_validation_data_attribute_visibility_info( $field ); if ( FrmField::is_required( $field ) && ! empty( $field_validation_messages_status['data-reqmsg'] ) ) { $required_message = FrmFieldsHelper::get_error_msg( $field, 'blank' ); $add_html['data-reqmsg'] = 'data-reqmsg="' . esc_attr( $required_message ) . '"'; self::maybe_add_html_required( $field, $add_html ); } if ( ! FrmField::is_option_empty( $field, 'invalid' ) && ! empty( $field_validation_messages_status['data-invmsg'] ) ) { $invalid_message = FrmFieldsHelper::get_error_msg( $field, 'invalid' ); $add_html['data-invmsg'] = 'data-invmsg="' . esc_attr( $invalid_message ) . '"'; } if ( ! empty( $add_html['data-reqmsg'] ) || ! empty( $add_html['data-invmsg'] ) ) { self::maybe_add_error_html_for_js_validation( $field, $add_html ); } } /** * Returns an array that contains field validation messages status. * * @since 6.9 * * @param array|object $field * * @return array */ private static function get_validation_data_attribute_visibility_info( $field ) { if ( FrmField::get_field_type( $field ) === 'hidden' ) { $field_validation_data_attributes = array( 'data-invmsg' => false, 'data-reqmsg' => false, ); } else { $field_validation_data_attributes = array( 'data-invmsg' => true, 'data-reqmsg' => true, ); } /** * Allows controlling which field validation messages would be included in the field html. * * @since 6.9 * * @param array $field_validation_messages_status * @param array|object $field */ return apply_filters( 'frm_field_validation_include_data_attributes', $field_validation_data_attributes, $field ); } /** * @since 5.0.03 * * @param array $field * @param array $add_html * * @return void */ private static function maybe_add_error_html_for_js_validation( $field, array &$add_html ) { $form = self::get_form_for_js_validation( $field ); if ( false === $form ) { return; } $error_body = self::pull_custom_error_body_from_custom_html( $form, $field ); if ( false !== $error_body ) { $error_body = urlencode( $error_body ); $add_html['data-error-html'] = 'data-error-html="' . esc_attr( $error_body ) . '"'; } } /** * @since 5.0.03 * * @param array $field * * @return false|stdClass false if there is no form object found with JS validation active. */ private static function get_form_for_js_validation( $field ) { global $frm_vars; if ( ! empty( $frm_vars['js_validate_forms'] ) ) { if ( isset( $frm_vars['js_validate_forms'][ $field['form_id'] ] ) ) { return $frm_vars['js_validate_forms'][ $field['form_id'] ]; } if ( ! empty( $field['parent_form_id'] ) && isset( $frm_vars['js_validate_forms'][ $field['parent_form_id'] ] ) ) { return $frm_vars['js_validate_forms'][ $field['parent_form_id'] ]; } } return false; } /** * @param stdClass $form * @param array $field * @param array $errors * * @return false|string */ public static function pull_custom_error_body_from_custom_html( $form, $field, $errors = array() ) { if ( empty( $field['custom_html'] ) ) { return false; } $custom_html = $field['custom_html']; $custom_html = apply_filters( 'frm_before_replace_shortcodes', $custom_html, $field, $errors, $form ); $start = strpos( $custom_html, '[if error]' ); if ( false === $start ) { return false; } $end = strpos( $custom_html, '[/if error]' ); if ( false === $end || $end < $start ) { return false; } $error_body = substr( $custom_html, $start + 10, $end - $start - 10 ); $default_html = array( '<div class="frm_error" id="frm_error_field_[key]">[error]</div>', '<div class="frm_error" role="alert" id="frm_error_field_[key]">[error]</div>', '<div class="frm_error">[error]</div>', '<div class="frm_error" role="alert">[error]</div>', ); return in_array( $error_body, $default_html, true ) ? false : $error_body; } /** * If 'required' is added to a conditionally hidden field, the form won't * submit in many browsers. Check to make sure the javascript to conditionally * remove it is present if needed. * * @since 3.06.01 * * @param array $field * @param array $add_html * * @return void */ private static function maybe_add_html_required( $field, array &$add_html ) { $excluded_field_types = FrmField::is_radio( $field ) || FrmField::is_checkbox( $field ) || FrmField::is_field_type( $field, 'file' ) || FrmField::is_field_type( $field, 'nps' ) || FrmField::is_field_type( $field, 'scale' ); if ( $excluded_field_types ) { return; } $add_html['aria-required'] = 'aria-required="true"'; } /** * @param array $field * @param array $add_html * * @return void */ private static function add_shortcodes_to_html( $field, array &$add_html ) { if ( FrmField::is_option_empty( $field, 'shortcodes' ) ) { return; } if ( ! empty( $field['autocomplete'] ) ) { unset( $field['shortcodes']['autocomplete'] ); } foreach ( $field['shortcodes'] as $k => $v ) { if ( isset( $field['subfield_name'] ) && str_starts_with( $k, 'aria-invalid' ) ) { $subfield_name = $field['subfield_name']; if ( ! isset( $field['shortcodes'][ 'aria-invalid-' . $subfield_name ] ) ) { continue; } // Change the key to the correct aria-invalid value so that $add_html is set correctly for the current subfield of a combo field. $k = 'aria-invalid'; $v = $field['shortcodes'][ 'aria-invalid-' . $subfield_name ]; unset( $field['shortcodes'][ 'aria-invalid-' . $subfield_name ] ); } if ( 'opt' === $k || ! self::should_allow_input_attribute( $k ) ) { continue; } if ( is_numeric( $k ) && str_contains( $v, '=' ) ) { $add_html[] = $v; } elseif ( ! empty( $k ) && isset( $add_html[ $k ] ) ) { $add_html[ $k ] = str_replace( $k . '="', $k . '="' . $v, $add_html[ $k ] ); } else { $add_html[ $k ] = $k . '="' . esc_attr( $v ) . '"'; } unset( $k, $v ); }//end foreach } /** * Disallow possibly unsafe attributees (that trigger JavaScript) when unasfe HTML is not allowed. * * @since 6.11.2 * * @param string $key The option key. * * @return bool */ private static function should_allow_input_attribute( $key ) { if ( ! FrmAppHelper::should_never_allow_unfiltered_html() ) { return true; } return FrmAppHelper::input_key_is_safe( $key ); } /** * Add pattern attribute. * * @since 3.0 * * @param array $field * @param array $add_html * * @return void */ private static function add_pattern_attribute( $field, array &$add_html ) { $format_value = FrmField::get_option( $field, 'format' ); $has_format = $format_value && ! FrmCurrencyHelper::is_currency_format( $format_value ); $format_field = FrmField::is_field_type( $field, 'text' ); if ( $field['type'] === 'phone' || ( $has_format && $format_field ) ) { $format = FrmEntryValidate::phone_format( $field ); $format = substr( $format, 2, - 1 ); $add_html['pattern'] = 'pattern="' . esc_attr( $format ) . '"'; } } /** * @param mixed $opt * @param mixed $opt_key * @param array|object $field * * @return mixed */ public static function check_value( $opt, $opt_key, $field ) { if ( is_array( $opt ) ) { if ( FrmField::is_option_true( $field, 'separate_value' ) ) { $opt = $opt['value'] ?? $opt['label'] ?? reset( $opt ); } else { $opt = $opt['label'] ?? reset( $opt ); } } return $opt; } /** * @param mixed $opt * * @return mixed */ public static function check_label( $opt ) { if ( is_array( $opt ) ) { $opt = $opt['label'] ?? reset( $opt ); } return $opt; } }