$form_id ), 'field_order', array( 'order_by' => 'field_order DESC' ) ); $values['field_key'] = FrmAppHelper::get_unique_key( '', $wpdb->prefix . 'frm_fields', 'field_key' ); $values['form_id'] = $form_id; $values['field_order'] = $field_count + 1; $values['field_options']['custom_html'] = self::get_default_html( $type ); if ( ! empty( $setting ) ) { if ( in_array( $type, array( 'data', 'lookup' ), true ) ) { $values['field_options']['data_type'] = $setting; } else { $values['field_options'][ $setting ] = 1; } } // Increase the field order of submit field and fields in the same row. $last_row_field_ids = FrmAppHelper::get_post_param( 'last_row_field_ids', array() ); if ( $last_row_field_ids ) { foreach ( $last_row_field_ids as $index => $last_row_field_id ) { FrmField::update( $last_row_field_id, array( 'field_order' => $field_count + $index + 2 ) ); } } return $values; } public static function get_html_id( $field, $plus = '' ) { return apply_filters( 'frm_field_html_id', 'field_' . $field['field_key'] . $plus, $field ); } public static function setup_edit_vars( $field, $doing_ajax = false ) { $values = self::field_object_to_array( $field ); return apply_filters( 'frm_setup_edit_field_vars', $values, array( 'doing_ajax' => $doing_ajax ) ); } public static function field_object_to_array( $field ) { $values = (array) $field; self::fill_field_array( $field, $values ); $values['custom_html'] = isset( $field->field_options['custom_html'] ) ? $field->field_options['custom_html'] : self::get_default_html( $field->type ); return $values; } private static function fill_field_array( $field, array &$field_array ) { $field_array['options'] = $field->options; $field_array['value'] = $field->default_value; self::prepare_edit_front_field( $field_array, $field ); $field_array = array_merge( (array) $field->field_options, $field_array ); } /** * Prepare field while creating a new entry * * @since 3.0 */ public static function prepare_new_front_field( &$field_array, $field, $args = array() ) { $args['action'] = 'new'; self::prepare_front_field( $field_array, $field, $args ); } /** * Prepare field while editing an entry * * @since 3.0 */ public static function prepare_edit_front_field( &$field_array, $field, $entry_id = 0, $args = array() ) { $args['entry_id'] = $entry_id; $args['action'] = 'edit'; self::prepare_front_field( $field_array, $field, $args ); } /** * Prepare field while creating a new entry * * @since 3.0 * * @param array $field_array * @param stdClass $field * @param array $args * @return void */ private static function prepare_front_field( &$field_array, $field, $args ) { self::fill_default_field_opts( $field, $field_array ); self::fill_cleared_strings( $field, $field_array ); // Track the original field's type $field_array['original_type'] = isset( $field->field_options['original_type'] ) ? $field->field_options['original_type'] : $field->type; self::prepare_field_options_for_display( $field_array, $field, $args ); if ( $args['action'] === 'edit' ) { /** * @param array $field_array * @param stdClass $field * @param int|string $entry_id * @param array $args */ $field_array = apply_filters( 'frm_setup_edit_fields_vars', $field_array, $field, $args['entry_id'], $args ); } else { /** * @param array $field_array * @param stdClass $field * @param array $args */ $field_array = apply_filters( 'frm_setup_new_fields_vars', $field_array, $field, $args ); } } /** * @since 3.0 * * @param string $type * * @return array */ public static function get_default_field_options( $type ) { $field_type = FrmFieldFactory::get_field_type( $type ); return $field_type->get_default_field_options(); } /** * @since 3.0 * * @param object $field * @param array $values */ private static function fill_default_field_opts( $field, array &$values ) { $check_post = FrmAppHelper::is_admin_page() && $_POST && isset( $_POST['field_options'] ); // phpcs:ignore WordPress.Security.NonceVerification.Missing $defaults = self::get_default_field_options_from_field( $field, $values ); if ( ! $check_post ) { $defaults['required_indicator'] = ''; $defaults['original_type'] = $field->type; } foreach ( $defaults as $opt => $default ) { $values[ $opt ] = isset( $field->field_options[ $opt ] ) ? $field->field_options[ $opt ] : $default; if ( $check_post ) { self::get_posted_field_setting( $opt . '_' . $field->id, $values[ $opt ] ); } unset( $opt, $default ); } } /** * Fill the required message, invalid message, * and refill the HTML when cleared * * @since 3.0 * * @param object $field * @param array $field_array */ private static function fill_cleared_strings( $field, array &$field_array ) { if ( '' == $field_array['blank'] && '1' === $field_array['required'] ) { $field_array['blank'] = self::default_blank_msg(); } if ( '' === $field_array['invalid'] ) { if ( 'captcha' === $field->type ) { $frm_settings = FrmAppHelper::get_settings(); $field_array['invalid'] = $frm_settings->re_msg; } else { $field_array['invalid'] = self::default_invalid_msg(); } } if ( '' == $field_array['custom_html'] ) { $field_array['custom_html'] = self::get_default_html( $field->type ); } } /** * @since 6.8.3 * * @return string */ public static function default_invalid_msg() { /* translators: %s: [field_name] shortcode (Which gets replaced by a Field Name) */ return sprintf( __( '%s is invalid', 'formidable' ), '[field_name]' ); } /** * @since 6.8.3 * * @return string */ public static function default_unique_msg() { $frm_settings = FrmAppHelper::get_settings(); $unique_message = $frm_settings->unique_msg; $unique_message = str_replace( 'This value', '[field_name]', $unique_message ); return $unique_message; } /** * @since 6.8.3 * * @return string */ public static function default_blank_msg() { $frm_settings = FrmAppHelper::get_settings(); $blank_message = $frm_settings->blank_msg; $blank_message = str_replace( 'This field', '[field_name]', $blank_message ); return $blank_message; } /** * @since 3.0 * * @param string $setting * @param mixed $value */ private static function get_posted_field_setting( $setting, &$value ) { if ( ! isset( $_POST['field_options'][ $setting ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing return; } if ( strpos( $setting, 'html' ) !== false ) { // Strip slashes from HTML but not regex or script tags. $value = wp_unslash( $_POST['field_options'][ $setting ] ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized, WordPress.Security.NonceVerification.Missing } elseif ( strpos( $setting, 'format_' ) === 0 ) { // TODO: Remove stripslashes on output, and use on input only. $value = sanitize_text_field( $_POST['field_options'][ $setting ] ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.MissingUnslash, WordPress.Security.NonceVerification.Missing } else { $value = wp_unslash( $_POST['field_options'][ $setting ] ); // phpcs:ignore WordPress.Security.ValidatedSanitizedInput.InputNotSanitized, WordPress.Security.NonceVerification.Missing FrmAppHelper::sanitize_value( 'wp_kses_post', $value ); } } /** * @since 3.0 * * @param object $field * @param array $values The field array is needed for hooks. * * @return array */ public static function get_default_field_options_from_field( $field, $values = array() ) { $field_type = self::get_original_field( $field ); $opts = $field_type->get_default_field_options(); $opts = apply_filters( 'frm_default_field_opts', $opts, $values, $field ); $opts = apply_filters( 'frm_default_' . $field->type . '_field_opts', $opts, $values, $field ); return $opts; } /** * @since 3.0 * * @param object $field * * @return FrmFieldType */ private static function get_original_field( $field ) { $original_type = FrmField::get_option( $field, 'original_type' ); if ( ! empty( $original_type ) && $field->type != $original_type ) { $field->type = $original_type; } return FrmFieldFactory::get_field_object( $field ); } /** * @since 3.0 * * @param array $field_array * @param object $field * @param array $atts */ private static function prepare_field_options_for_display( &$field_array, $field, $atts ) { $field_obj = FrmFieldFactory::get_field_object( $field ); $field_array = $field_obj->prepare_front_field( $field_array, $atts ); } /** * @since 3.0 * * @param string $type * * @return array */ public static function get_default_field( $type ) { $field_type = FrmFieldFactory::get_field_type( $type ); return $field_type->get_new_field_defaults(); } public static function fill_field( &$values, $field, $form_id, $new_key = '' ) { global $wpdb; $values['field_key'] = FrmAppHelper::get_unique_key( $new_key, $wpdb->prefix . 'frm_fields', 'field_key' ); $values['form_id'] = $form_id; $values['options'] = maybe_serialize( $field->options ); $values['default_value'] = FrmAppHelper::maybe_json_encode( $field->default_value ); foreach ( array( 'name', 'description', 'type', 'field_order', 'field_options', 'required' ) as $col ) { $values[ $col ] = $field->{$col}; } } /** * @since 2.0 * * @param array|object $field * @param string $error * * @return string */ public static function get_error_msg( $field, $error ) { $frm_settings = FrmAppHelper::get_settings(); $conf_msg = __( 'The entered values do not match', 'formidable' ); $defaults = array( 'unique_msg' => array( 'full' => self::default_unique_msg(), /* translators: %s: Field name */ 'part' => sprintf( __( '%s must be unique', 'formidable' ), '[field_name]' ), ), 'invalid' => array( 'full' => __( 'This field is invalid', 'formidable' ), /* translators: %s: Field name */ 'part' => sprintf( __( '%s is invalid', 'formidable' ), '[field_name]' ), ), 'blank' => array( 'full' => $frm_settings->blank_msg, 'part' => $frm_settings->blank_msg, ), 'conf_msg' => array( 'full' => $conf_msg, 'part' => $conf_msg, ), ); $msg = FrmField::get_option( $field, $error ); $msg = empty( $msg ) ? $defaults[ $error ]['part'] : $msg; $msg = do_shortcode( $msg ); $msg = self::maybe_replace_substrings_with_field_name( $msg, $error, $field ); return $msg; } /** * @since 6.8.3 * * @param string $msg * @param string $error * @param array|object $field */ private static function maybe_replace_substrings_with_field_name( $msg, $error, $field ) { $field_name = is_array( $field ) ? $field['name'] : $field->name; $field_name = FrmAppHelper::maybe_kses( $field_name ); $substrings = self::get_substrings_to_replace_with_field_name( $field_name, compact( 'msg', 'error', 'field' ) ); // Use the "This value"/"This field" placeholder strings if field name is empty. if ( ! $field_name ) { if ( 'unique_msg' === $error ) { $field_name = __( 'This value', 'formidable' ); } else { $field_name = __( 'This field', 'formidable' ); } } $msg = str_replace( $substrings, $field_name, $msg ); return $msg; } /** * @since 6.8.3 * * @param string $field_name * @param array $filter_args { * Filter arguments. * * @type string $msg The current error message before the substrings are replaced. * @type string $error A key including 'unique_msg', 'invalid', 'blank', or 'conf_msg'. * @type array|object $field The field with the error. * } * @return array */ private static function get_substrings_to_replace_with_field_name( $field_name, $filter_args ) { $substrings = array( '[field_name]' ); if ( $field_name ) { array_push( $substrings, 'This value', 'This field' ); } /** * @since 6.8.3 * * @param array $substrings * @param array $filter_args */ $filtered_substrings = apply_filters( 'frm_error_substrings_to_replace_with_field_name', $substrings, $filter_args ); if ( is_array( $filtered_substrings ) ) { $substrings = $filtered_substrings; } else { _doing_it_wrong( __FUNCTION__, 'Only arrays should be returned when using the frm_error_substrings_to_replace_with_field_name filter.', '6.8.3' ); } return $substrings; } public static function get_form_fields( $form_id, $error = array() ) { $fields = FrmField::get_all_for_form( $form_id ); return apply_filters( 'frm_get_paged_fields', $fields, $form_id, $error ); } public static function get_default_html( $type = 'text' ) { $field = FrmFieldFactory::get_field_type( $type ); $default_html = $field->default_html(); // these hooks are here for reverse compatibility since 3.0 if ( ! apply_filters( 'frm_normal_field_type_html', true, $type ) ) { $default_html = apply_filters( 'frm_other_custom_html', '', $type ); } return apply_filters( 'frm_custom_html', $default_html, $type ); } /** * @param array $fields * @param array $errors * @param object $form * @param object $form_action */ public static function show_fields( $fields, $errors, $form, $form_action ) { foreach ( $fields as $field ) { $field_obj = FrmFieldFactory::get_field_type( $field['type'], $field ); $field_obj->show_field( compact( 'errors', 'form', 'form_action' ) ); } } /** * @since 3.0 * * @param array $atts * @param array|string $value */ public static function run_wpautop( $atts, &$value ) { $autop = isset( $atts['wpautop'] ) ? $atts['wpautop'] : true; if ( apply_filters( 'frm_use_wpautop', $autop ) ) { if ( is_array( $value ) ) { $value = implode( "\n", $value ); } $value = wpautop( $value ); } } /** * Get the class to use for the label position * * @since 2.05 */ public static function &label_position( $position, $field, $form ) { if ( $position && $position != '' ) { if ( $position === 'inside' && ! self::is_placeholder_field_type( $field['type'] ) ) { $position = 'top'; } return $position; } $position = FrmStylesController::get_style_val( 'position', $form ); if ( $position === 'none' ) { $position = 'top'; } elseif ( $position === 'no_label' ) { $position = 'none'; } elseif ( $position === 'inside' && ! self::is_placeholder_field_type( $field['type'] ) ) { $position = 'top'; } $position = apply_filters( 'frm_html_label_position', $position, $field, $form ); $position = ! empty( $position ) ? $position : 'top'; return $position; } /** * Check if this field type allows placeholders * * @since 2.05 * @param string $type * @return bool */ public static function is_placeholder_field_type( $type ) { return ! in_array( $type, array( 'radio', 'checkbox', 'hidden', 'file' ), true ); } public static function get_checkbox_id( $field, $opt_key, $type = 'checkbox' ) { $id = $field['id']; if ( isset( $field['in_section'] ) && $field['in_section'] && ! FrmAppHelper::is_admin_page( 'formidable' ) ) { $id .= '-' . $field['in_section']; } return 'frm_' . $type . '_' . $id . '-' . $opt_key; } public static function show_single_option( $field ) { self::hidden_field_option( $field ); if ( ! is_array( $field['options'] ) ) { return; } $base_name = 'default_value_' . $field['id']; $html_id = isset( $field['html_id'] ) ? $field['html_id'] : self::get_html_id( $field ); $default_type = self::get_default_value_type( $field ); foreach ( $field['options'] as $opt_key => $opt ) { $field_val = self::get_value_from_array( $opt, $opt_key, $field ); $opt = self::get_label_from_array( $opt, $opt_key, $field ); $field_name = $base_name . ( $default_type === 'checkbox' ? '[' . $opt_key . ']' : '' ); $checked = ( isset( $field['default_value'] ) && ( ( ! is_array( $field['default_value'] ) && $field['default_value'] == $field_val ) || ( is_array( $field['default_value'] ) && in_array( $field_val, $field['default_value'] ) ) ) ); // If this is an "Other" option, get the HTML for it. if ( self::is_other_opt( $opt_key ) ) { if ( FrmAppHelper::pro_is_installed() ) { require FrmProAppHelper::plugin_path() . '/classes/views/frmpro-fields/other-option.php'; } } else { require FrmAppHelper::plugin_path() . '/classes/views/frm-fields/single-option.php'; } unset( $checked ); } } /** * Include hidden row for javascript to duplicate. * * @since 4.0 * @param array $field */ private static function hidden_field_option( $field ) { // Don't duplicate during an ajax add option. $ajax_action = FrmAppHelper::get_param( 'action', '', 'post', 'sanitize_text_field' ); if ( $ajax_action === 'frm_add_field_option' ) { return; } $opt_key = '000'; $field_val = __( 'New Option', 'formidable' ); $opt = __( 'New Option', 'formidable' ); $checked = false; $field_name = 'default_value_' . $field['id']; $html_id = isset( $field['html_id'] ) ? $field['html_id'] : self::get_html_id( $field ); $default_type = self::get_default_value_type( $field ); $field_name .= ( $default_type === 'checkbox' ? '[' . $opt_key . ']' : '' ); require FrmAppHelper::plugin_path() . '/classes/views/frm-fields/single-option.php'; } /** * @since 4.0 * * @param array $field * @return string radio or checkbox */ private static function get_default_value_type( $field ) { $default_type = $field['type']; if ( $field['type'] === 'select' ) { $default_type = FrmField::is_multiple_select( $field ) ? 'checkbox' : 'radio'; } return $default_type; } public static function get_value_from_array( $opt, $opt_key, $field ) { $opt = apply_filters( 'frm_field_value_saved', $opt, $opt_key, $field ); return FrmFieldsController::check_value( $opt, $opt_key, $field ); } public static function get_label_from_array( $opt, $opt_key, $field ) { $opt = apply_filters( 'frm_field_label_seen', $opt, $opt_key, $field ); return FrmFieldsController::check_label( $opt ); } /** * Shows the inline modal. * * @since 4.0 * @since 6.4.1 Added `inside_class` in the arguments. * * @param array $args The arguments. */ public static function inline_modal( $args ) { $defaults = array( 'id' => '', 'class' => '', 'show' => 0, 'callback' => array(), 'args' => array(), 'title' => '', 'inside_class' => 'inside', ); $args = array_merge( $defaults, $args ); include FrmAppHelper::plugin_path() . '/classes/views/frm-fields/back-end/inline-modal.php'; } /** * @since 4.0 */ public static function smart_values() { $continue = apply_filters( 'frm_smart_values_box', true ); if ( $continue === true ) { $upgrade_link = array( 'medium' => 'builder', 'content' => 'smart-tags', ); include FrmAppHelper::plugin_path() . '/classes/views/frm-fields/back-end/smart-values.php'; } } /** * @since 4.0 */ public static function input_mask() { include FrmAppHelper::plugin_path() . '/classes/views/frm-fields/back-end/input-mask-info.php'; } /** * @since 4.0 */ public static function layout_classes() { include FrmAppHelper::plugin_path() . '/classes/views/frm-fields/back-end/layout-classes.php'; } /** * @param int $tax_id * * @return string */ public static function get_term_link( $tax_id ) { $tax = get_taxonomy( $tax_id ); if ( ! $tax ) { return ''; } $link = sprintf( /* translators: %1$s: Start HTML link, %2$s: Content type label, %3$s: Content type, %4$s: End HTML link */ esc_html__( 'Options are dynamically created from your %1$s%2$s: %3$s%4$s', 'formidable' ), '', esc_html__( 'taxonomy', 'formidable' ), empty( $tax->labels->name ) ? esc_html__( 'Categories', 'formidable' ) : $tax->labels->name, '' ); unset( $tax ); return $link; } public static function value_meets_condition( $observed_value, $cond, $hide_opt ) { $hide_opt = self::get_value_for_comparison( $hide_opt ); $observed_value = self::get_value_for_comparison( $observed_value ); if ( is_array( $observed_value ) ) { return self::array_value_condition( $observed_value, $cond, $hide_opt ); } $m = false; if ( $cond === '==' ) { $m = $observed_value == $hide_opt; } elseif ( $cond === '!=' ) { $m = $observed_value != $hide_opt; } elseif ( $cond === '>' ) { $m = $observed_value > $hide_opt; } elseif ( $cond === '>=' ) { $m = $observed_value >= $hide_opt; } elseif ( $cond === '<' ) { $m = $observed_value < $hide_opt; } elseif ( $cond === '<=' ) { $m = $observed_value <= $hide_opt; } elseif ( $cond === 'LIKE' || $cond === 'not LIKE' ) { $m = stripos( $observed_value, $hide_opt ); if ( $cond === 'not LIKE' ) { $m = $m === false; } else { $m = $m !== false; } } elseif ( $cond === '%LIKE' ) { // ends with $length = strlen( $hide_opt ); $substr = substr( $observed_value, strlen( $observed_value ) - $length ); $m = 0 === strcasecmp( $substr, $hide_opt ); } elseif ( 'LIKE%' === $cond ) { // starts with $length = strlen( $hide_opt ); $substr = substr( $observed_value, 0, $length ); $m = 0 === strcasecmp( $substr, $hide_opt ); }//end if return $m; } /** * Trim and sanitize the values * * @since 2.05 */ private static function get_value_for_comparison( $value ) { // Remove white space from hide_opt if ( ! is_array( $value ) ) { $value = trim( $value ); } FrmAppHelper::sanitize_value( 'wp_kses_post', $value ); return $value; } public static function array_value_condition( $observed_value, $cond, $hide_opt ) { $m = false; if ( $cond === '==' ) { if ( is_array( $hide_opt ) ) { $m = array_intersect( $hide_opt, $observed_value ); $m = ! empty( $m ); } else { $m = in_array( $hide_opt, $observed_value ); } } elseif ( $cond === '!=' ) { $m = ! in_array( $hide_opt, $observed_value ); } elseif ( $cond === '>' ) { $min = min( $observed_value ); $m = $min > $hide_opt; } elseif ( $cond === '<' ) { $max = max( $observed_value ); $m = $max < $hide_opt; } elseif ( $cond === 'LIKE' || $cond === 'not LIKE' ) { foreach ( $observed_value as $ob ) { $m = strpos( $ob, $hide_opt ); if ( $m !== false ) { $m = true; break; } } if ( $cond === 'not LIKE' ) { $m = $m === false; } } elseif ( $cond === '%LIKE' ) { // ends with foreach ( $observed_value as $ob ) { if ( $hide_opt === substr( $ob, strlen( $ob ) - strlen( $hide_opt ) ) ) { $m = true; break; } } } elseif ( $cond === 'LIKE%' ) { // starts with foreach ( $observed_value as $ob ) { if ( $hide_opt === substr( $ob, 0, strlen( $hide_opt ) ) ) { $m = true; break; } } }//end if return $m; } /** * Replace a few basic shortcodes and field ids * * @since 2.0 * @return string */ public static function basic_replace_shortcodes( $value, $form, $entry ) { if ( strpos( $value, '[sitename]' ) !== false ) { $new_value = wp_specialchars_decode( FrmAppHelper::site_name(), ENT_QUOTES ); $value = str_replace( '[sitename]', $new_value, $value ); } $value = apply_filters( 'frm_content', $value, $form, $entry ); $value = do_shortcode( $value ); return $value; } public static function get_shortcodes( $content, $form_id ) { if ( FrmAppHelper::pro_is_installed() ) { return FrmProDisplaysHelper::get_shortcodes( $content, $form_id ); } $fields = FrmField::getAll( array( 'fi.form_id' => (int) $form_id, 'fi.type not' => FrmField::no_save_fields(), ) ); $tagregexp = self::allowed_shortcodes( $fields ); preg_match_all( "/\[(if )?($tagregexp)\b(.*?)(?:(\/))?\](?:(.+?)\[\/\2\])?/s", $content, $matches, PREG_PATTERN_ORDER ); return $matches; } public static function allowed_shortcodes( $fields = array() ) { $tagregexp = array( 'editlink', 'id', 'key', 'ip', 'siteurl', 'sitename', 'admin_email', 'post[-|_]id', 'created[-|_]at', 'updated[-|_]at', 'updated[-|_]by', 'parent[-|_]id', ); foreach ( $fields as $field ) { $tagregexp[] = $field->id; $tagregexp[] = $field->field_key; } $tagregexp = implode( '|', $tagregexp ); return $tagregexp; } public static function replace_content_shortcodes( $content, $entry, $shortcodes ) { foreach ( $shortcodes[0] as $short_key => $tag ) { if ( empty( $tag ) ) { continue; } $atts = FrmShortcodeHelper::get_shortcode_attribute_array( $shortcodes[3][ $short_key ] ); $tag = FrmShortcodeHelper::get_shortcode_tag( $shortcodes, $short_key ); $atts['entry'] = $entry; $atts['tag'] = $tag; $replace_with = self::get_value_for_shortcode( $atts ); if ( $replace_with !== null ) { $replace_with = self::trigger_shortcode_atts( $replace_with, $atts ); self::sanitize_embedded_shortcodes( compact( 'entry' ), $replace_with ); $content = str_replace( $shortcodes[0][ $short_key ], $replace_with, $content ); } unset( $atts, $replace_with ); } return $content; } /** * @param string $replace_with * @param array $atts * @return string */ private static function trigger_shortcode_atts( $replace_with, $atts ) { $supported_atts = array( 'remove_accents', 'sanitize', 'sanitize_url' ); $included_atts = array_intersect( $supported_atts, array_keys( $atts ) ); foreach ( $included_atts as $included_att ) { if ( '0' === $atts[ $included_att ] ) { // Skip any option that uses 0 so sanitize_url=0 does not encode. continue; } $function = 'atts_' . $included_att; $replace_with = self::$function( $replace_with, $atts ); } return $replace_with; } /** * Converts all accent characters to ASCII characters. * * @since 6.3.1 * * @param string $replace_with The text to remove accents from. * * @return string */ public static function atts_remove_accents( $replace_with ) { return remove_accents( $replace_with ); } /** * @param string $replace_with * @return string */ private static function atts_sanitize( $replace_with ) { return sanitize_title_with_dashes( $replace_with ); } /** * @param string $replace_with * @return string */ private static function atts_sanitize_url( $replace_with ) { return urlencode( $replace_with ); } /** * Prevent shortcodes in fields from being processed * * @since 3.01.02 * * @param array $atts Includes entry object. * @param string $value */ public static function sanitize_embedded_shortcodes( $atts, &$value ) { $atts['value'] = $value; $should_sanitize = apply_filters( 'frm_sanitize_shortcodes', true, $atts ); if ( $should_sanitize ) { $value = str_replace( '[', '[', $value ); } } /** * @since 3.0 * * @param array $atts * * @return string */ private static function get_value_for_shortcode( $atts ) { $clean_tag = str_replace( '-', '_', $atts['tag'] ); $shortcode_values = array( 'id' => $atts['entry']->id, 'key' => $atts['entry']->item_key, 'ip' => $atts['entry']->ip, ); $dynamic_default = array( 'admin_email', 'siteurl', 'frmurl', 'sitename', 'get' ); if ( isset( $shortcode_values[ $atts['tag'] ] ) ) { $replace_with = $shortcode_values[ $atts['tag'] ]; } elseif ( in_array( $atts['tag'], $dynamic_default, true ) ) { $replace_with = self::dynamic_default_values( $atts['tag'], $atts ); } elseif ( $clean_tag === 'user_agent' ) { $description = $atts['entry']->description; $replace_with = FrmEntriesHelper::get_browser( $description['browser'] ); } elseif ( $clean_tag === 'created_at' || $clean_tag === 'updated_at' ) { $atts['tag'] = $clean_tag; $replace_with = self::get_entry_timestamp( $atts ); } elseif ( $clean_tag === 'created_by' || $clean_tag === 'updated_by' ) { $replace_with = self::get_display_value( $atts['entry']->{$clean_tag}, (object) array( 'type' => 'user_id' ), $atts ); } else { $replace_with = self::get_field_shortcode_value( $atts ); } return $replace_with; } /** * @since 3.0 * * @param array $atts * * @return string */ private static function get_entry_timestamp( $atts ) { if ( isset( $atts['format'] ) ) { $time_format = ' '; } else { $atts['format'] = get_option( 'date_format' ); $time_format = ''; } return FrmAppHelper::get_formatted_time( $atts['entry']->{$atts['tag']}, $atts['format'], $time_format ); } /** * @since 3.0 * * @param array $atts * * @return string|null */ private static function get_field_shortcode_value( $atts ) { $field = FrmField::getOne( $atts['tag'] ); if ( empty( $field ) ) { return null; } if ( isset( $atts['show'] ) && $atts['show'] === 'field_label' ) { $replace_with = $field->name; } elseif ( isset( $atts['show'] ) && $atts['show'] === 'description' ) { $replace_with = $field->description; } else { $replace_with = FrmEntryMeta::get_meta_value( $atts['entry'], $field->id ); $string_value = $replace_with; if ( is_array( $replace_with ) ) { $sep = isset( $atts['sep'] ) ? $atts['sep'] : ', '; $string_value = implode( $sep, $replace_with ); } if ( empty( $string_value ) && $string_value != '0' ) { $replace_with = ''; } else { $atts['entry_id'] = $atts['entry']->id; $atts['entry_key'] = $atts['entry']->item_key; $replace_with = self::get_display_value( $replace_with, $field, $atts ); } } return $replace_with; } /** * Get the value to replace a few standard shortcodes * * @since 2.0 * @return string */ public static function dynamic_default_values( $tag, $atts = array(), $return_array = false ) { $new_value = ''; switch ( $tag ) { case 'admin_email': $new_value = get_option( 'admin_email' ); break; case 'siteurl': $new_value = FrmAppHelper::site_url(); break; case 'frmurl': $new_value = FrmAppHelper::plugin_url(); break; case 'sitename': $new_value = FrmAppHelper::site_name(); break; case 'get': $new_value = self::process_get_shortcode( $atts, $return_array ); } return $new_value; } /** * Process the [get] shortcode * * @since 2.0 * @return array|string */ public static function process_get_shortcode( $atts, $return_array = false ) { if ( ! isset( $atts['param'] ) ) { return ''; } if ( strpos( $atts['param'], '[' ) ) { $atts['param'] = str_replace( '[', '[', $atts['param'] ); $atts['param'] = str_replace( ']', ']', $atts['param'] ); } $new_value = FrmAppHelper::get_param( $atts['param'], '', 'get', 'sanitize_text_field' ); $new_value = FrmAppHelper::get_query_var( $new_value, $atts['param'] ); if ( $new_value == '' ) { if ( ! isset( $atts['prev_val'] ) ) { $atts['prev_val'] = ''; } $new_value = isset( $atts['default'] ) ? $atts['default'] : $atts['prev_val']; } if ( is_array( $new_value ) && ! $return_array ) { $new_value = implode( ', ', $new_value ); } return $new_value; } public static function get_display_value( $value, $field, $atts = array() ) { $value = apply_filters( 'frm_get_' . $field->type . '_display_value', $value, $field, $atts ); $value = apply_filters( 'frm_get_display_value', $value, $field, $atts ); $value = self::get_unfiltered_display_value( compact( 'value', 'field', 'atts' ) ); return apply_filters( 'frm_display_value', $value, $field, $atts ); } /** * @param array $atts Includes value, field, and atts. */ public static function get_unfiltered_display_value( $atts ) { $value = $atts['value']; $field = $atts['field']; $atts = isset( $atts['atts'] ) ? $atts['atts'] : $atts; if ( is_array( $field ) ) { $field = $field['id']; } $field_obj = FrmFieldFactory::get_field_object( $field ); return $field_obj->get_display_value( $value, $atts ); } /** * Get a value from the user profile from the user ID * * @since 3.0 * * @return string */ public static function get_user_display_name( $user_id, $user_info = 'display_name', $args = array() ) { $defaults = array( 'blank' => false, 'link' => false, 'size' => 96, ); $args = wp_parse_args( $args, $defaults ); $user = get_userdata( $user_id ); $info = ''; if ( $user ) { if ( $user_info === 'avatar' ) { $info = get_avatar( $user_id, $args['size'] ); } elseif ( $user_info === 'author_link' ) { $info = get_author_posts_url( $user_id ); } else { $info = isset( $user->$user_info ) ? $user->$user_info : ''; } if ( 'display_name' === $user_info && empty( $info ) && ! $args['blank'] ) { $info = $user->user_login; } } if ( $args['link'] ) { $info = '' . $info . ''; } return $info; } /** * @param string $type * @return array */ public static function get_field_types( $type ) { $single_input = self::single_input_fields(); $multiple_input = array( 'radio', 'checkbox', 'select', 'scale', 'star', 'lookup' ); $field_selection = FrmField::all_field_selection(); $field_types = array(); if ( in_array( $type, $single_input, true ) ) { self::field_types_for_input( $single_input, $field_selection, $field_types ); } elseif ( in_array( $type, $multiple_input, true ) ) { self::field_types_for_input( $multiple_input, $field_selection, $field_types ); } elseif ( isset( $field_selection[ $type ] ) ) { $field_types[ $type ] = $field_selection[ $type ]; } $field_types = apply_filters( 'frm_switch_field_types', $field_types, compact( 'type', 'field_selection' ) ); return $field_types; } /** * Get a list of all fields that use a single value input. * * @since 4.0 */ public static function single_input_fields() { $fields = array( 'text', 'textarea', 'rte', 'number', 'email', 'url', 'date', 'phone', 'hidden', 'time', 'tag', 'password', ); return apply_filters( 'frm_single_input_fields', $fields ); } /** * @param string[] $inputs * @param array $fields * @param array $field_types * @return void */ private static function field_types_for_input( $inputs, $fields, &$field_types ) { foreach ( $inputs as $input ) { // This may not be set if a field type was removed using the frm_available_fields or frm_pro_available_fields filters. if ( array_key_exists( $input, $fields ) ) { $field_types[ $input ] = $fields[ $input ]; } unset( $input ); } } /** * Check if current field option is an "other" option * * @since 2.0.6 * * @param string $opt_key * * @return bool Returns true if current field option is an "Other" option */ public static function is_other_opt( $opt_key ) { return $opt_key && strpos( $opt_key, 'other_' ) === 0; } /** * Get value that belongs in "Other" text box * * @since 2.0.6 * * @param array $args */ public static function get_other_val( $args ) { $defaults = array( 'opt_key' => 0, 'field' => array(), 'parent' => false, 'pointer' => false, ); $args = wp_parse_args( $args, $defaults ); $opt_key = $args['opt_key']; $field = $args['field']; $parent = $args['parent']; $pointer = $args['pointer']; $other_val = ''; // If option is an "other" option and there is a value set for this field, // check if the value belongs in the current "Other" option text field if ( ! self::is_other_opt( $opt_key ) || ! FrmField::is_option_true( $field, 'value' ) ) { return $other_val; } // Check posted vals before checking saved values // For fields inside repeating sections - note, don't check if $pointer is true because it will often be zero if ( $parent && isset( $_POST['item_meta'][ $parent ][ $pointer ]['other'][ $field['id'] ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing if ( FrmField::is_field_with_multiple_values( $field ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing $other_val = isset( $_POST['item_meta'][ $parent ][ $pointer ]['other'][ $field['id'] ][ $opt_key ] ) ? sanitize_text_field( wp_unslash( $_POST['item_meta'][ $parent ][ $pointer ]['other'][ $field['id'] ][ $opt_key ] ) ) : ''; } else { $other_val = sanitize_text_field( wp_unslash( $_POST['item_meta'][ $parent ][ $pointer ]['other'][ $field['id'] ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing } return $other_val; } if ( isset( $field['id'] ) && isset( $_POST['item_meta']['other'][ $field['id'] ] ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing // For normal fields if ( FrmField::is_field_with_multiple_values( $field ) ) { // phpcs:ignore WordPress.Security.NonceVerification.Missing $other_val = isset( $_POST['item_meta']['other'][ $field['id'] ][ $opt_key ] ) ? sanitize_text_field( wp_unslash( $_POST['item_meta']['other'][ $field['id'] ][ $opt_key ] ) ) : ''; } else { $other_val = sanitize_text_field( wp_unslash( $_POST['item_meta']['other'][ $field['id'] ] ) ); // phpcs:ignore WordPress.Security.NonceVerification.Missing } return $other_val; }//end if // For checkboxes if ( $field['type'] === 'checkbox' && is_array( $field['value'] ) ) { // Check if there is an "other" val in saved value and make sure the // "other" val is not equal to the Other checkbox option if ( isset( $field['value'][ $opt_key ] ) && $field['options'][ $opt_key ] != $field['value'][ $opt_key ] ) { $other_val = $field['value'][ $opt_key ]; } } else { /** * For radio buttons and dropdowns * Check if saved value equals any of the options. If not, set it as the other value. */ foreach ( $field['options'] as $opt_key => $opt_val ) { $temp_val = is_array( $opt_val ) ? $opt_val['value'] : $opt_val; // Multi-select dropdowns - key is not preserved if ( is_array( $field['value'] ) ) { $o_key = array_search( $temp_val, $field['value'] ); if ( isset( $field['value'][ $o_key ] ) ) { unset( $field['value'][ $o_key ], $o_key ); } } elseif ( $temp_val == $field['value'] ) { // For radio and regular dropdowns return ''; } else { $other_val = $field['value']; } unset( $opt_key, $opt_val, $temp_val ); } // For multi-select dropdowns only if ( is_array( $field['value'] ) && ! empty( $field['value'] ) ) { $other_val = reset( $field['value'] ); } }//end if return $other_val; } /** * Check if there is a saved value for the "Other" text field. If so, set it as the $other_val. * Intended for front-end use * * @since 2.0.6 * * @param array $args Should include field, opt_key and field name. * @param bool $other_opt * @param string $checked * * @return array $other_args */ public static function prepare_other_input( $args, &$other_opt, &$checked ) { $other_args = array( 'name' => '', 'value' => '', ); // Check if this is an "Other" option. if ( ! self::is_other_opt( $args['opt_key'] ) ) { return $other_args; } $other_opt = true; self::set_other_name( $args, $other_args ); self::set_other_value( $args, $other_args ); if ( '' !== $other_args['value'] ) { $checked = 'checked="checked" '; } // If 'other' is selected as one of the default values for a checkbox field, 'checked' attribute should be on. if ( ! $checked && $args['field']['type'] === 'checkbox' && is_array( $args['field']['value'] ) && isset( $args['field_val'] ) && in_array( $args['field_val'], $args['field']['value'], true ) ) { $checked = 'checked="checked" '; } return $other_args; } /** * @since 2.0.6 * @param array $args * @param array $other_args */ private static function set_other_name( $args, &$other_args ) { // Set up name for other field $other_args['name'] = str_replace( '[]', '', $args['field_name'] ); $other_args['name'] = preg_replace( '/\[' . $args['field']['id'] . '\]$/', '', $other_args['name'] ); $other_args['name'] = $other_args['name'] . '[other][' . $args['field']['id'] . ']'; // Converts item_meta[field_id] => item_meta[other][field_id] and // item_meta[parent][0][field_id] => item_meta[parent][0][other][field_id] if ( FrmField::is_field_with_multiple_values( $args['field'] ) ) { $other_args['name'] .= '[' . $args['opt_key'] . ']'; } } /** * Find the parent and pointer, and get text for "other" text field * * @since 2.0.6 * @param array $args * @param array $other_args */ private static function set_other_value( $args, &$other_args ) { $parent = ''; $pointer = ''; // Check for parent ID and pointer $temp_array = explode( '[', $args['field_name'] ); // Count should only be greater than 3 if inside of a repeating section if ( count( $temp_array ) > 3 ) { $parent = str_replace( ']', '', $temp_array[1] ); $pointer = str_replace( ']', '', $temp_array[2] ); } // Get text for "other" text field $other_args['value'] = self::get_other_val( array( 'opt_key' => $args['opt_key'], 'field' => $args['field'], 'parent' => $parent, 'pointer' => $pointer, ) ); } /** * If this field includes an other option, show it * * @since 2.0.6 * @param array $args */ public static function include_other_input( $args ) { if ( ! $args['other_opt'] ) { return; } $classes = array( 'frm_other_input' ); if ( ! $args['checked'] || trim( $args['checked'] ) == '' ) { // hide the field if the other option is not selected $classes[] = 'frm_pos_none'; } if ( $args['field']['type'] === 'select' && $args['field']['multiple'] ) { $classes[] = 'frm_other_full'; } // Set up HTML ID for Other field $other_id = self::get_other_field_html_id( $args['field']['type'], $args['html_id'], $args['opt_key'] ); $label = isset( $args['opt_label'] ) ? $args['opt_label'] : $args['field']['name']; echo '' . ''; } /** * Get the HTML id for an "Other" text field * Note: This does not affect fields in repeating sections * * @since 2.0.08 * * @param string $type Field type. * @param string $html_id * @param bool|string $opt_key * * @return string $other_id */ public static function get_other_field_html_id( $type, $html_id, $opt_key = false ) { $other_id = $html_id; // If hidden radio field, add an opt key of 0 if ( $type === 'radio' && $opt_key === false ) { $opt_key = 0; } if ( $opt_key !== false ) { $other_id .= '-' . $opt_key; } $other_id .= '-otext'; return $other_id; } public static function switch_field_ids( $val ) { global $frm_duplicate_ids; $replace = array(); $replace_with = array(); foreach ( (array) $frm_duplicate_ids as $old => $new ) { $replace[] = '[if ' . $old . ']'; $replace_with[] = '[if ' . $new . ']'; $replace[] = '[if ' . $old . ' '; $replace_with[] = '[if ' . $new . ' '; $replace[] = '[/if ' . $old . ']'; $replace_with[] = '[/if ' . $new . ']'; $replace[] = '[\/if ' . $old . ']'; $replace_with[] = '[\/if ' . $new . ']'; $replace[] = '[foreach ' . $old . ']'; $replace_with[] = '[foreach ' . $new . ']'; $replace[] = '[/foreach ' . $old . ']'; $replace_with[] = '[/foreach ' . $new . ']'; $replace[] = '[\/foreach ' . $old . ']'; $replace_with[] = '[\/foreach ' . $new . ']'; $replace[] = '[' . $old . ']'; $replace_with[] = '[' . $new . ']'; $replace[] = '[' . $old . ' '; $replace_with[] = '[' . $new . ' '; $replace[] = 'field_id="' . $old . '"'; $replace_with[] = 'field_id="' . $new . '"'; $replace[] = 'field_id=\"' . $old . '\"'; $replace_with[] = 'field_id=\"' . $new . '\"'; unset( $old, $new ); }//end foreach if ( is_array( $val ) ) { foreach ( $val as $k => $v ) { if ( is_string( $v ) ) { if ( 'custom_html' === $k ) { $val[ $k ] = self::switch_ids_except_strings( $replace, $replace_with, array( '[if description]', '[description]', '[/if description]' ), $v ); unset( $k, $v ); continue; } $val[ $k ] = str_replace( $replace, $replace_with, $v ); unset( $k, $v ); } } } else { $val = str_replace( $replace, $replace_with, $val ); } return $val; } /** * Removes exception strings from replacement arrays and replaces the rest in the provided value string. * * @since 6.14 * * @param array $replace Values to be replaced. * @param array $replace_with Replacement values. * @param array $exceptions Array of strings to skip. * @param string $value Value to be updated. * * @return string */ private static function switch_ids_except_strings( $replace, $replace_with, $exceptions, $value ) { foreach ( $exceptions as $exception ) { $index = array_search( $exception, $replace, true ); if ( false === $index ) { continue; } unset( $replace[ $index ] ); unset( $replace_with[ $index ] ); } $value = str_replace( $replace, $replace_with, $value ); return $value; } /** * @since 4.0 */ public static function bulk_options_overlay() { $prepop = array(); self::get_bulk_prefilled_opts( $prepop, true ); include FrmAppHelper::plugin_path() . '/classes/views/frm-fields/back-end/bulk-options-overlay.php'; } public static function get_us_states() { $states = array( 'AL' => 'Alabama', 'AK' => 'Alaska', 'AR' => 'Arkansas', 'AZ' => 'Arizona', 'CA' => 'California', 'CO' => 'Colorado', 'CT' => 'Connecticut', 'DE' => 'Delaware', 'DC' => 'District of Columbia', 'FL' => 'Florida', 'GA' => 'Georgia', 'HI' => 'Hawaii', 'ID' => 'Idaho', 'IL' => 'Illinois', 'IN' => 'Indiana', 'IA' => 'Iowa', 'KS' => 'Kansas', 'KY' => 'Kentucky', 'LA' => 'Louisiana', 'ME' => 'Maine', 'MD' => 'Maryland', 'MA' => 'Massachusetts', 'MI' => 'Michigan', 'MN' => 'Minnesota', 'MS' => 'Mississippi', 'MO' => 'Missouri', 'MT' => 'Montana', 'NE' => 'Nebraska', 'NV' => 'Nevada', 'NH' => 'New Hampshire', 'NJ' => 'New Jersey', 'NM' => 'New Mexico', 'NY' => 'New York', 'NC' => 'North Carolina', 'ND' => 'North Dakota', 'OH' => 'Ohio', 'OK' => 'Oklahoma', 'OR' => 'Oregon', 'PA' => 'Pennsylvania', 'RI' => 'Rhode Island', 'SC' => 'South Carolina', 'SD' => 'South Dakota', 'TN' => 'Tennessee', 'TX' => 'Texas', 'UT' => 'Utah', 'VT' => 'Vermont', 'VA' => 'Virginia', 'WA' => 'Washington', 'WV' => 'West Virginia', 'WI' => 'Wisconsin', 'WY' => 'Wyoming', ); return apply_filters( 'frm_us_states', $states ); } public static function get_countries() { $countries = array( __( 'Afghanistan', 'formidable' ), __( 'Aland Islands', 'formidable' ), __( 'Albania', 'formidable' ), __( 'Algeria', 'formidable' ), __( 'American Samoa', 'formidable' ), __( 'Andorra', 'formidable' ), __( 'Angola', 'formidable' ), __( 'Anguilla', 'formidable' ), __( 'Antarctica', 'formidable' ), __( 'Antigua and Barbuda', 'formidable' ), __( 'Argentina', 'formidable' ), __( 'Armenia', 'formidable' ), __( 'Aruba', 'formidable' ), __( 'Australia', 'formidable' ), __( 'Austria', 'formidable' ), __( 'Azerbaijan', 'formidable' ), __( 'Bahamas', 'formidable' ), __( 'Bahrain', 'formidable' ), __( 'Bangladesh', 'formidable' ), __( 'Barbados', 'formidable' ), __( 'Belarus', 'formidable' ), __( 'Belgium', 'formidable' ), __( 'Belize', 'formidable' ), __( 'Benin', 'formidable' ), __( 'Bermuda', 'formidable' ), __( 'Bhutan', 'formidable' ), __( 'Bolivia', 'formidable' ), __( 'Bonaire, Sint Eustatius and Saba', 'formidable' ), __( 'Bosnia and Herzegovina', 'formidable' ), __( 'Botswana', 'formidable' ), __( 'Bouvet Island', 'formidable' ), __( 'Brazil', 'formidable' ), __( 'British Indian Ocean Territory', 'formidable' ), __( 'Brunei', 'formidable' ), __( 'Bulgaria', 'formidable' ), __( 'Burkina Faso', 'formidable' ), __( 'Burundi', 'formidable' ), __( 'Cambodia', 'formidable' ), __( 'Cameroon', 'formidable' ), __( 'Canada', 'formidable' ), __( 'Cape Verde', 'formidable' ), __( 'Cayman Islands', 'formidable' ), __( 'Central African Republic', 'formidable' ), __( 'Chad', 'formidable' ), __( 'Chile', 'formidable' ), __( 'China', 'formidable' ), __( 'Christmas Island', 'formidable' ), __( 'Cocos (Keeling) Islands', 'formidable' ), __( 'Colombia', 'formidable' ), __( 'Comoros', 'formidable' ), __( 'Congo', 'formidable' ), __( 'Cook Islands', 'formidable' ), __( 'Costa Rica', 'formidable' ), __( 'Côte d\'Ivoire', 'formidable' ), __( 'Croatia', 'formidable' ), __( 'Cuba', 'formidable' ), __( 'Curacao', 'formidable' ), __( 'Cyprus', 'formidable' ), __( 'Czech Republic', 'formidable' ), __( 'Denmark', 'formidable' ), __( 'Djibouti', 'formidable' ), __( 'Dominica', 'formidable' ), __( 'Dominican Republic', 'formidable' ), __( 'East Timor', 'formidable' ), __( 'Ecuador', 'formidable' ), __( 'Egypt', 'formidable' ), __( 'El Salvador', 'formidable' ), __( 'Equatorial Guinea', 'formidable' ), __( 'Eritrea', 'formidable' ), __( 'Estonia', 'formidable' ), __( 'Ethiopia', 'formidable' ), __( 'Falkland Islands (Malvinas)', 'formidable' ), __( 'Faroe Islands', 'formidable' ), __( 'Fiji', 'formidable' ), __( 'Finland', 'formidable' ), __( 'France', 'formidable' ), __( 'French Guiana', 'formidable' ), __( 'French Polynesia', 'formidable' ), __( 'French Southern Territories', 'formidable' ), __( 'Gabon', 'formidable' ), __( 'Gambia', 'formidable' ), __( 'Georgia', 'formidable' ), __( 'Germany', 'formidable' ), __( 'Ghana', 'formidable' ), __( 'Gibraltar', 'formidable' ), __( 'Greece', 'formidable' ), __( 'Greenland', 'formidable' ), __( 'Grenada', 'formidable' ), __( 'Guadeloupe', 'formidable' ), __( 'Guam', 'formidable' ), __( 'Guatemala', 'formidable' ), __( 'Guernsey', 'formidable' ), __( 'Guinea', 'formidable' ), __( 'Guinea-Bissau', 'formidable' ), __( 'Guyana', 'formidable' ), __( 'Haiti', 'formidable' ), __( 'Heard Island and McDonald Islands', 'formidable' ), __( 'Holy See', 'formidable' ), __( 'Honduras', 'formidable' ), __( 'Hong Kong', 'formidable' ), __( 'Hungary', 'formidable' ), __( 'Iceland', 'formidable' ), __( 'India', 'formidable' ), __( 'Indonesia', 'formidable' ), __( 'Iran', 'formidable' ), __( 'Iraq', 'formidable' ), __( 'Ireland', 'formidable' ), __( 'Israel', 'formidable' ), __( 'Isle of Man', 'formidable' ), __( 'Italy', 'formidable' ), __( 'Jamaica', 'formidable' ), __( 'Japan', 'formidable' ), __( 'Jersey', 'formidable' ), __( 'Jordan', 'formidable' ), __( 'Kazakhstan', 'formidable' ), __( 'Kenya', 'formidable' ), __( 'Kiribati', 'formidable' ), __( 'North Korea', 'formidable' ), __( 'South Korea', 'formidable' ), __( 'Kosovo', 'formidable' ), __( 'Kuwait', 'formidable' ), __( 'Kyrgyzstan', 'formidable' ), __( 'Laos', 'formidable' ), __( 'Latvia', 'formidable' ), __( 'Lebanon', 'formidable' ), __( 'Lesotho', 'formidable' ), __( 'Liberia', 'formidable' ), __( 'Libya', 'formidable' ), __( 'Liechtenstein', 'formidable' ), __( 'Lithuania', 'formidable' ), __( 'Luxembourg', 'formidable' ), __( 'Macao', 'formidable' ), __( 'Macedonia', 'formidable' ), __( 'Madagascar', 'formidable' ), __( 'Malawi', 'formidable' ), __( 'Malaysia', 'formidable' ), __( 'Maldives', 'formidable' ), __( 'Mali', 'formidable' ), __( 'Malta', 'formidable' ), __( 'Marshall Islands', 'formidable' ), __( 'Martinique', 'formidable' ), __( 'Mauritania', 'formidable' ), __( 'Mauritius', 'formidable' ), __( 'Mayotte', 'formidable' ), __( 'Mexico', 'formidable' ), __( 'Micronesia', 'formidable' ), __( 'Moldova', 'formidable' ), __( 'Monaco', 'formidable' ), __( 'Mongolia', 'formidable' ), __( 'Montenegro', 'formidable' ), __( 'Montserrat', 'formidable' ), __( 'Morocco', 'formidable' ), __( 'Mozambique', 'formidable' ), __( 'Myanmar', 'formidable' ), __( 'Namibia', 'formidable' ), __( 'Nauru', 'formidable' ), __( 'Nepal', 'formidable' ), __( 'Netherlands', 'formidable' ), __( 'New Caledonia', 'formidable' ), __( 'New Zealand', 'formidable' ), __( 'Nicaragua', 'formidable' ), __( 'Niger', 'formidable' ), __( 'Nigeria', 'formidable' ), __( 'Niue', 'formidable' ), __( 'Norfolk Island', 'formidable' ), __( 'Northern Mariana Islands', 'formidable' ), __( 'Norway', 'formidable' ), __( 'Oman', 'formidable' ), __( 'Pakistan', 'formidable' ), __( 'Palau', 'formidable' ), __( 'Palestine', 'formidable' ), __( 'Panama', 'formidable' ), __( 'Papua New Guinea', 'formidable' ), __( 'Paraguay', 'formidable' ), __( 'Peru', 'formidable' ), __( 'Philippines', 'formidable' ), __( 'Pitcairn', 'formidable' ), __( 'Poland', 'formidable' ), __( 'Portugal', 'formidable' ), __( 'Puerto Rico', 'formidable' ), __( 'Qatar', 'formidable' ), __( 'Reunion', 'formidable' ), __( 'Romania', 'formidable' ), __( 'Russia', 'formidable' ), __( 'Rwanda', 'formidable' ), __( 'Saint Barthelemy', 'formidable' ), __( 'Saint Helena, Ascension and Tristan da Cunha', 'formidable' ), __( 'Saint Kitts and Nevis', 'formidable' ), __( 'Saint Lucia', 'formidable' ), __( 'Saint Martin (French part)', 'formidable' ), __( 'Saint Pierre and Miquelon', 'formidable' ), __( 'Saint Vincent and the Grenadines', 'formidable' ), __( 'Samoa', 'formidable' ), __( 'San Marino', 'formidable' ), __( 'Sao Tome and Principe', 'formidable' ), __( 'Saudi Arabia', 'formidable' ), __( 'Senegal', 'formidable' ), __( 'Serbia', 'formidable' ), __( 'Seychelles', 'formidable' ), __( 'Sierra Leone', 'formidable' ), __( 'Singapore', 'formidable' ), __( 'Sint Maarten (Dutch part)', 'formidable' ), __( 'Slovakia', 'formidable' ), __( 'Slovenia', 'formidable' ), __( 'Solomon Islands', 'formidable' ), __( 'Somalia', 'formidable' ), __( 'South Africa', 'formidable' ), __( 'South Georgia and the South Sandwich Islands', 'formidable' ), __( 'South Sudan', 'formidable' ), __( 'Spain', 'formidable' ), __( 'Sri Lanka', 'formidable' ), __( 'Sudan', 'formidable' ), __( 'Suriname', 'formidable' ), __( 'Svalbard and Jan Mayen', 'formidable' ), __( 'Swaziland', 'formidable' ), __( 'Sweden', 'formidable' ), __( 'Switzerland', 'formidable' ), __( 'Syria', 'formidable' ), __( 'Taiwan', 'formidable' ), __( 'Tajikistan', 'formidable' ), __( 'Tanzania', 'formidable' ), __( 'Thailand', 'formidable' ), __( 'Timor-Leste', 'formidable' ), __( 'Togo', 'formidable' ), __( 'Tokelau', 'formidable' ), __( 'Tonga', 'formidable' ), __( 'Trinidad and Tobago', 'formidable' ), __( 'Tunisia', 'formidable' ), __( 'Turkey', 'formidable' ), __( 'Turkmenistan', 'formidable' ), __( 'Turks and Caicos Islands', 'formidable' ), __( 'Tuvalu', 'formidable' ), __( 'Uganda', 'formidable' ), __( 'Ukraine', 'formidable' ), __( 'United Arab Emirates', 'formidable' ), __( 'United Kingdom', 'formidable' ), __( 'United States', 'formidable' ), __( 'United States Minor Outlying Islands', 'formidable' ), __( 'Uruguay', 'formidable' ), __( 'Uzbekistan', 'formidable' ), __( 'Vanuatu', 'formidable' ), __( 'Vatican City', 'formidable' ), __( 'Venezuela', 'formidable' ), __( 'Vietnam', 'formidable' ), __( 'Virgin Islands, British', 'formidable' ), __( 'Virgin Islands, U.S.', 'formidable' ), __( 'Wallis and Futuna', 'formidable' ), __( 'Western Sahara', 'formidable' ), __( 'Yemen', 'formidable' ), __( 'Zambia', 'formidable' ), __( 'Zimbabwe', 'formidable' ), ); sort( $countries, SORT_LOCALE_STRING ); return apply_filters( 'frm_countries', $countries ); } /** * Gets bulk prefilled options. * * @since 5.0.04 Add `$include_class` param. * * @param array $prepop Bulk options. * @param array|false $include_class Include the class in the bulk options. */ public static function get_bulk_prefilled_opts( array &$prepop, $include_class = false ) { // Countries. $countries = self::get_countries(); if ( $include_class ) { $countries['class'] = 'frm-countries-opts'; } $prepop[ __( 'Countries', 'formidable' ) ] = $countries; // State abv. $states = self::get_us_states(); $state_abv = array_keys( $states ); sort( $state_abv ); if ( $include_class ) { $state_abv['class'] = 'frm-state-abv-opts'; } $prepop[ __( 'U.S. State Abbreviations', 'formidable' ) ] = $state_abv; // States. $states = array_values( $states ); sort( $states ); if ( $include_class ) { $states['class'] = 'frm-states-opts'; } $prepop[ __( 'U.S. States', 'formidable' ) ] = $states; unset( $state_abv, $states ); // Age. $ages = array( __( 'Under 18', 'formidable' ), __( '18-24', 'formidable' ), __( '25-34', 'formidable' ), __( '35-44', 'formidable' ), __( '45-54', 'formidable' ), __( '55-64', 'formidable' ), __( '65 or Above', 'formidable' ), __( 'Prefer Not to Answer', 'formidable' ), ); if ( $include_class ) { $ages['class'] = 'frm-age-opts'; } $prepop[ __( 'Age', 'formidable' ) ] = $ages; // Satisfaction. $satisfaction = array( __( 'Very Unsatisfied', 'formidable' ), __( 'Unsatisfied', 'formidable' ), __( 'Neutral', 'formidable' ), __( 'Satisfied', 'formidable' ), __( 'Very Satisfied', 'formidable' ), __( 'N/A', 'formidable' ), ); if ( $include_class ) { $satisfaction['class'] = 'frm-satisfaction-opts'; } $prepop[ __( 'Satisfaction', 'formidable' ) ] = $satisfaction; // Importance. $importance = array( __( 'Not at all Important', 'formidable' ), __( 'Somewhat Important', 'formidable' ), __( 'Neutral', 'formidable' ), __( 'Important', 'formidable' ), __( 'Very Important', 'formidable' ), __( 'N/A', 'formidable' ), ); if ( $include_class ) { $importance['class'] = 'frm-importance-opts'; } $prepop[ __( 'Importance', 'formidable' ) ] = $importance; // Agreement. $agreement = array( __( 'Strongly Disagree', 'formidable' ), __( 'Disagree', 'formidable' ), __( 'Neutral', 'formidable' ), __( 'Agree', 'formidable' ), __( 'Strongly Agree', 'formidable' ), __( 'N/A', 'formidable' ), ); if ( $include_class ) { $agreement['class'] = 'frm-agreement-opts'; } $prepop[ __( 'Agreement', 'formidable' ) ] = $agreement; // Likely. $likely = array( __( 'Extremely Unlikely', 'formidable' ), __( 'Unlikely', 'formidable' ), __( 'Neutral', 'formidable' ), __( 'Likely', 'formidable' ), __( 'Extremely Likely', 'formidable' ), __( 'N/A', 'formidable' ), ); if ( $include_class ) { $likely['class'] = 'frm-likely-opts'; } $prepop[ __( 'Likely', 'formidable' ) ] = $likely; $prepop = apply_filters( 'frm_bulk_field_choices', $prepop ); } /** * Display a field value selector * * @since 2.03.05 * * @param int $selector_field_id * @param array $selector_args */ public static function display_field_value_selector( $selector_field_id, $selector_args ) { $field_value_selector = FrmFieldFactory::create_field_value_selector( $selector_field_id, $selector_args ); $field_value_selector->display(); } /** * Convert a field object to a flat array * * @since 2.03.05 * * @param object $field * * @return array */ public static function convert_field_object_to_flat_array( $field ) { $field_options = $field->field_options; $field_array = get_object_vars( $field ); unset( $field_array['field_options'] ); return $field_array + $field_options; } /** * @since 4.04 * * @param array $args * @return void */ public static function show_add_field_buttons( $args ) { $field_key = $args['field_key']; $field_type = $args['field_type']; $field_label = FrmAppHelper::icon_by_class( FrmFormsHelper::get_field_link_icon( $field_type ), array( 'echo' => false ) ); $field_name = FrmFormsHelper::get_field_link_name( $field_type ); $field_label .= ' ' . $field_name . ''; if ( ! empty( $field_type['is_new'] ) ) { ob_start(); FrmAppHelper::show_pill_text(); $field_label .= ob_get_clean(); } // If the individual field isn't allowed, disable it. $run_filter = true; $single_no_allow = ' '; $install_data = ''; $requires = ''; $link = isset( $field_type['link'] ) ? esc_url_raw( $field_type['link'] ) : ''; $has_show_upgrade_class = strpos( $field_type['icon'], ' frm_show_upgrade' ); $show_upgrade = $has_show_upgrade_class || false !== strpos( $args['no_allow_class'], 'frm_show_upgrade' ); if ( $has_show_upgrade_class ) { $single_no_allow .= 'frm_show_upgrade'; $field_type['icon'] = str_replace( ' frm_show_upgrade', '', $field_type['icon'] ); $run_filter = false; if ( isset( $field_type['addon'] ) ) { $upgrading = FrmAddonsController::install_link( $field_type['addon'] ); if ( isset( $upgrading['url'] ) ) { $install_data = json_encode( $upgrading ); } $requires = FrmFormsHelper::get_plan_required( $upgrading ); } elseif ( isset( $field_type['require'] ) ) { $requires = $field_type['require']; } } $upgrade_label = ''; $upgrade_message = ''; if ( $show_upgrade ) { /* translators: %s: Field name */ $upgrade_label = sprintf( esc_html__( '%s fields', 'formidable' ), $field_name ); if ( isset( $field_type['message'] ) ) { $upgrade_message = FrmAppHelper::kses( $field_type['message'], array( 'a', 'img' ) ); } } $li_params = array( 'class' => 'frmbutton frm6 ' . $args['no_allow_class'] . $single_no_allow . ' frm_t' . str_replace( '|', '-', $field_key ), 'id' => $field_key, 'data-upgrade' => $upgrade_label, 'data-link' => $link, 'data-medium' => 'builder', 'data-oneclick' => $install_data, 'data-content' => $field_key, 'data-requires' => $requires, ); if ( ! empty( $field_type['hide'] ) ) { $li_params['class'] .= ' frm_hidden'; } if ( ! empty( $field_type['upsell_image'] ) ) { $li_params['data-upsell-image'] = $field_type['upsell_image']; } if ( $upgrade_message ) { $li_params['data-message'] = $upgrade_message; } ?>
  • >
  • array( 'text' => __( 'Simple', 'formidable' ), 'svg' => 'frm_simple_radio', ), '1' => array( 'text' => __( 'Images', 'formidable' ), 'svg' => 'frm_image_as_option', 'addon' => 'pro', 'upgrade' => __( 'Image Options', 'formidable' ), 'message' => __( 'Show images instead of radio buttons or check boxes. This is ideal for polls, surveys, segmenting questionnaires and more.', 'formidable' ) . '', 'content' => 'image-options', ), 'buttons' => array( 'text' => __( 'Buttons', 'formidable' ), 'svg' => 'frm_button_as_option', 'addon' => 'surveys', 'upgrade' => __( 'Button Options', 'formidable' ), 'message' => __( 'Show buttons for radio buttons or check boxes. This is ideal for polls, surveys, segmenting questionnaires and more.', 'formidable' ), 'content' => 'button-options', ), ); /** * Allows modifying the options of Display format setting of Radio field. * * @since 5.0.04 * * @param array $options Options. * @param array $field */ $options = apply_filters( 'frm_' . $field['type'] . '_display_format_options', $options, $field ); return $options; } /** * Gets display format arguments to pass to the images_dropdown() method. * * @since 5.0.04 * * @param array $field Field data. * @param array $options Options array. * @return array */ public static function get_display_format_args( $field, $options ) { $args = array( 'selected' => '0', 'options' => array(), 'name' => 'field_options[image_options_' . $field['id'] . ']', 'input_attrs' => array( 'class' => 'frm_toggle_image_options', ), ); self::fill_image_setting_options( $options, $args ); /** * Allows modifying the arguments of Display format setting of Radio field. * * @since 5.0.04 * * @param array $args Arguments. * @param array $method_args The arguments from the method. Contains `field`, `options`. */ return apply_filters( 'frm_' . $field['type'] . '_display_format_args', $args, compact( 'field', 'options' ) ); } /** * @since 5.0.04 */ private static function fill_image_setting_options( $options, &$args ) { foreach ( $options as $key => $option ) { $args['options'][ $key ] = $option; if ( ! empty( $option['addon'] ) ) { $args['options'][ $key ]['custom_attrs'] = self::fill_image_setting_addon_link( $option ); } unset( $args['options'][ $key ]['addon'] ); $fill = array( 'upgrade', 'message', 'content' ); foreach ( $fill as $f ) { unset( $args['options'][ $key ][ $f ], $f ); } } } /** * @since 5.0.04 * * @return array */ private static function fill_image_setting_addon_link( $option ) { $custom_attrs = array( 'class' => 'frm_noallow frm_show_upgrade', 'data-medium' => 'builder', ); // translators: Add-on name. $custom_attrs['data-upgrade'] = sprintf( __( 'Formidable %s', 'formidable' ), ucwords( $option['addon'] ) ); $fill = array( 'upgrade', 'message', 'content' ); foreach ( $fill as $f ) { if ( isset( $option[ $f ] ) ) { $custom_attrs[ 'data-' . $f ] = $option[ $f ]; } } if ( 'pro' === $option['addon'] ) { return $custom_attrs; } $upgrading = FrmAddonsController::install_link( $option['addon'] ); if ( isset( $upgrading['url'] ) ) { $install_data = wp_json_encode( $upgrading ); } else { $install_data = ''; } $custom_attrs['data-oneclick'] = $install_data; $custom_attrs['data-requires'] = FrmFormsHelper::get_plan_required( $upgrading ); return $custom_attrs; } /** * Maybe adjust a field value based on type. * Some types require unserializing an array. * These types are defined by a array_allowed property on their field model class. * * @since 6.2 * * @param mixed $value * @param string $field_type * @return void */ public static function prepare_field_value( &$value, $field_type ) { $field_object = FrmFieldFactory::get_field_type( $field_type ); $value = $field_object->maybe_decode_value( $value ); } /** * @since 6.8 * * @param int|string $form_id * @param array $field_ids If this is not empty, the results will be filtered by field id. * @return array */ public static function get_draft_field_results( $form_id, $field_ids = array() ) { if ( FrmAppHelper::pro_is_installed() ) { $child_form_ids = FrmDb::get_col( 'frm_forms', array( 'parent_form_id' => $form_id ) ); $form_ids = array_merge( array( $form_id ), $child_form_ids ); } else { $form_ids = array( $form_id ); } $where = array( 'form_id' => $form_ids, // Do a soft check for fields that look like drafts only. 'field_options LIKE' => 's:5:"draft";i:1;', ); if ( $field_ids ) { $where['id'] = $field_ids; } $rows = FrmDb::get_results( 'frm_fields', $where, 'id, field_options' ); return array_filter( $rows, function ( $row ) { FrmAppHelper::unserialize_or_decode( $row->field_options ); return is_array( $row->field_options ) && ! empty( $row->field_options['draft'] ); } ); } /** * This is called when loading the form builder. * Any unsaved draft fields get added to a hidden draft_fields input on load. * * @since 6.8 * * @param int|string $form_id * @return array */ public static function get_all_draft_field_ids( $form_id ) { $draft_field_rows = self::get_draft_field_results( $form_id ); return wp_list_pluck( $draft_field_rows, 'id' ); } }