form_id = (int) $form_id; $this->default_label_position_field_ids = array(); } /** * Modify form behaviours for the styler preview. * * @since 6.0 * * @return void */ public function adjust_form_for_preview() { // Don't bother including the antispam token in the preview as the form isn't submitted. add_filter( 'frm_run_antispam', '__return_false', 99 ); // We don't need the honeypot in the preview so leave it out. add_filter( 'frm_run_honeypot', '__return_false' ); $this->hide_captcha_fields(); $this->disable_javascript_validation(); $this->add_a_div_class_for_default_label_positions(); } /** * Add a frm-default-label-position class to any field with a default label position. * A field with a custom label position shouldn't ever change in the styler preview. * * @since 6.0.1 * @return void */ private function add_a_div_class_for_default_label_positions() { // Only fields with no label position set hit this filter, so track those for the frm_field_div_classes filter. add_filter( 'frm_html_label_position', /** * @param string $position * @param array|object $field * @return string */ function ( $position, $field ) { if ( is_array( $field ) ) { $this->default_label_position_field_ids[] = (int) $field['id']; } return $position; }, 10, 2 ); add_filter( 'frm_field_div_classes', /** * @param string $classes * @param array|object $field * @return string */ function ( $classes, $field ) { if ( is_array( $field ) && in_array( (int) $field['id'], $this->default_label_position_field_ids, true ) ) { $classes .= ' frm-default-label-position'; } return $classes; }, 10, 2 ); } /** * Captcha does not initialize in the preview. Hide it so we don't see the label either. * * @since 6.0 * * @return void */ private function hide_captcha_fields() { add_filter( 'frm_show_normal_field_type', /** * @param bool $show * @param string $field_type * @param string $target_field_type * @return bool */ function ( $show, $field_type ) { if ( 'captcha' === $field_type ) { $show = false; } return $show; }, 10, 2 ); } /** * Turn off JavaScript validation for the preview. * Without this we hit a "PHP Fatal error: Uncaught Error: Maximum function nesting level of '256' reached" error. * This only happened with specific Look up fields when FrmProLookupFieldsController::get_independent_lookup_field_options is called in Pro. * * @since 6.0 * * @return void */ private function disable_javascript_validation() { add_filter( 'frm_form_object', /** * @param stdClass|null $form * @param int $form_id * @return stdClass|null */ function ( $form ) { if ( is_object( $form ) && is_array( $form->options ) ) { $form->options['js_validate'] = false; } return $form; } ); } /** * @since 6.0 * * @return string */ public function get_html_for_form_preview() { // Force is_admin to false so the "Entry Key" field doesn't render in the preview. add_filter( 'frm_is_admin', '__return_false' ); $target_form_preview_html = FrmFormsController::show_form( $this->form_id, '', 'auto', 'auto' ); $this->form_includes_captcha = wp_script_is( 'captcha-api', 'enqueued' ); if ( $this->form_includes_captcha ) { // If a form includes a CAPTCHA field, don't try to load the CAPTCHA scripts for the visual styler preview. wp_dequeue_script( 'captcha-api' ); } // Return the is_admin status. // Otherwise success messages won't use the proper mark up and will appear without the green background and padding. remove_filter( 'frm_is_admin', '__return_false' ); return $target_form_preview_html; } /** * @since 6.0 * * @todo Only show the note once for a form per user per month or something. * * @return array */ public function get_notes_for_styler_preview() { $notes = array(); $fallback_form_note = $this->get_fallback_form_note(); if ( is_string( $fallback_form_note ) ) { $notes[] = $fallback_form_note; } $frm_settings = FrmAppHelper::get_settings(); if ( 'none' === $frm_settings->load_style ) { $notes[] = function () { printf( // translators: %1$s: Anchor tag open, %2$s: Anchor tag close. esc_html__( 'Formidable styles are disabled. This needs to be enabled in %1$sGlobal Settings%2$s.', 'formidable' ), '', '' ); }; } if ( class_exists( 'FrmProStylesController' ) && ! class_exists( 'FrmProStylesPreviewHelper' ) ) { $notes[] = __( 'You are using an outdated version of Formidable Pro. Please update to version 6.0 to get access to all styler features.', 'formidable' ); } if ( is_callable( 'FrmProStylesController::get_notes_for_styler_preview' ) ) { $notes = array_merge( $notes, FrmProStylesController::get_notes_for_styler_preview() ); } $disabled_features_note = $this->get_disabled_features_note(); if ( is_string( $disabled_features_note ) ) { $notes[] = $disabled_features_note; } return $notes; } /** * @since 6.0 * * @return string */ private function get_disabled_features_note() { return __( 'Not all JavaScript is loaded in this preview. Some features will appear differently on the front end.', 'formidable' ); } /** * @since 6.0 * * @return false|string */ private function get_fallback_form_note() { if ( ! FrmAppHelper::simple_get( 'form', 'absint' ) ) { return __( 'This form is being previewed because no form was selected. Use the form dropdown to select a new preview target.', 'formidable' ); } return false; } /** * Get all warnings to display above the visual styler preview. * * @since 6.0 * * @param stdClass|WP_Post $style A new style is not a WP_Post object. * @param WP_Post $default_style * @param string $view Either 'list' or 'edit'. * @return array */ public function get_warnings_for_styler_preview( $style, $default_style, $view ) { $warnings = array(); if ( 'edit' === $view && $this->should_show_multiple_forms_warning( $style->ID, $default_style->ID ) ) { $warnings[] = __( 'Changes that you will make to this style will apply to every form using this style.', 'formidable' ); } return $warnings; } /** * @since 6.0 * * @param int $style_id * @param int $default_style_id * @return bool */ private function should_show_multiple_forms_warning( $style_id, $default_style_id ) { $is_default_style = $style_id === $default_style_id; if ( ! $is_default_style ) { // Also check for the conversational default. $conversational_style_id = FrmDb::get_var( 'posts', array( 'post_name' => 'lines-no-boxes' ), 'ID' ); if ( $conversational_style_id && (int) $conversational_style_id === $style_id ) { $is_default_style = true; } } $form_count = FrmStylesHelper::get_form_count_for_style( $style_id, $is_default_style ); if ( $form_count <= 1 ) { return false; } // Only show the warning once per user per style. $user_id = get_current_user_id(); $meta_key = 'frm_dismiss_multiple_forms_warning_' . $style_id; $meta = get_user_meta( $user_id, $meta_key, true ); if ( $meta ) { return false; } add_user_meta( $user_id, $meta_key, 1 ); return true; } /** * @since 6.0 * * @param WP_Styles $styles * @return void */ public static function disable_conflicting_wp_admin_css( $styles ) { if ( ! is_callable( array( $styles, 'remove' ) ) || ! array_key_exists( 'wp-admin', $styles->registered ) ) { return; } $styles->remove( 'edit' ); $wp_admin_dependencies = $styles->registered['wp-admin']->deps; $edit_key = array_search( 'edit', $wp_admin_dependencies ); if ( false === $edit_key ) { return; } // Remove the edit dependency from wp-admin so it still loads, just without edit.css. self::remove_wp_admin_dependency( $styles, 'edit' ); } /** * Provides few fixes for style preview. * Fix the "Width" from Fields Settings to get reflected on each style preview ajax update. * Fix the Radio & Checkbox "Single Row" or "Multiple Row" to get reflected in style preview ajax update. * * @since 6.14 * * @param array $settings The style options. * @param bool $is_preview * * @return void */ public static function get_additional_preview_style( $settings, $is_preview = false ) { if ( ! $is_preview ) { return; } $radio_display_type = 'inline' === $settings['radio_align'] ? 'inline-block' : 'block'; $check_display_type = 'inline' === $settings['check_align'] ? 'inline-block' : 'block'; $style = '#frm_style_preview .with_frm_style input[type=text], #frm_style_preview .with_frm_style input[type=password], #frm_style_preview .with_frm_style input[type=email], #frm_style_preview .with_frm_style input[type=number], #frm_style_preview .with_frm_style input[type=url], #frm_style_preview .with_frm_style input[type=tel], #frm_style_preview .with_frm_style input[type=phone], #frm_style_preview .with_frm_style input[type=search], #frm_style_preview .with_frm_style textarea, #frm_style_preview .frm_form_fields_style, #frm_style_preview .with_frm_style .frm_scroll_box .frm_opt_container, #frm_style_preview .frm_form_fields_active_style, #frm_style_preview .frm_form_fields_error_style, #frm_style_preview .with_frm_style .frm-card-element.StripeElement, #frm_style_preview .with_frm_style .frm_slimselect.ss-main { width: var(--field-width); } #frm_style_preview .with_frm_style .frm_radio { display:' . $radio_display_type . '; } #frm_style_preview .with_frm_style .frm_checkbox { display:' . $check_display_type . '; }'; echo esc_html( $style ); } /** * @since 6.0 * * @param WP_Styles $styles * @param string $key * @return void */ private static function remove_wp_admin_dependency( $styles, $key ) { $dependencies = $styles->registered['wp-admin']->deps; $index = array_search( $key, $dependencies ); if ( false === $index ) { return; } unset( $dependencies[ $index ] ); $dependencies = array_values( $dependencies ); $styles->registered['wp-admin']->deps = $dependencies; } }