useRichTextEditor ) ? false : true; } public function get_field_input( $form, $value = '', $entry = null ) { global $current_screen; $form_id = absint( $form['id'] ); $is_entry_detail = $this->is_entry_detail(); $is_form_editor = $this->is_form_editor(); $is_admin = $is_entry_detail || $is_form_editor; $id = intval( $this->id ); $field_id = $is_entry_detail || $is_form_editor || $form_id == 0 ? "input_$id" : 'input_' . $form_id . "_$id"; $size = $this->size; $class_suffix = $is_entry_detail ? '_admin' : ''; $class = $size . $class_suffix; $class = esc_attr( $class ); $disabled_text = $is_form_editor ? 'disabled="disabled"' : ''; $maxlength_attribute = is_numeric( $this->maxLength ) ? "maxlength='{$this->maxLength}'" : ''; $placeholder_attribute = $this->get_field_placeholder_attribute(); $required_attribute = $this->isRequired ? 'aria-required="true"' : ''; $invalid_attribute = $this->failed_validation ? 'aria-invalid="true"' : 'aria-invalid="false"'; $aria_describedby = $this->get_aria_describedby(); $tabindex = $this->get_tabindex(); if ( $this->get_allowable_tags() === false ) { $value = esc_textarea( $value ); } else { $value = wp_kses_post( $value ); } //see if the field is set to use the rich text editor if ( ! $is_admin && $this->is_rich_edit_enabled() && ( ! $current_screen || ( $current_screen && ! rgobj( $current_screen, 'is_block_editor' ) ) ) ) { //placeholders cannot be used with the rte; message displayed in admin when this occurs //field cannot be used in conditional logic by another field; message displayed in admin and field removed from conditional logic drop down $tabindex = GFCommon::$tab_index > 0 ? GFCommon::$tab_index ++ : ''; add_filter( 'mce_buttons', array( $this, 'filter_mce_buttons' ), 10, 2 ); add_filter( 'mce_buttons_2', array( $this, 'filter_mce_buttons' ), 10, 2 ); add_filter( 'mce_buttons_3', array( $this, 'filter_mce_buttons' ), 10, 2 ); add_filter( 'mce_buttons_4', array( $this, 'filter_mce_buttons' ), 10, 2 ); /** * Filters the field options for the rich text editor. * * @since 2.0.0 * * @param array $editor_settings Array of settings that can be changed. * @param object $this The field object * @param array $form Current form object * @param array $entry Current entry object, if available * * Additional filters for specific form and fields IDs. */ $editor_settings = apply_filters( 'gform_rich_text_editor_options', array( 'textarea_name' => 'input_' . $id, 'wpautop' => true, 'editor_class' => $class, 'editor_height' => rgar( array( 'small' => 110, 'medium' => 180, 'large' => 280 ), $this->size ? $this->size : 'medium' ), 'tabindex' => $tabindex, 'media_buttons' => false, 'quicktags' => false, 'tinymce' => array( 'init_instance_callback' => "function (editor) { editor.on( 'keyup paste mouseover', function (e) { var content = editor.getContent( { format: 'text' } ).trim(); var textarea = jQuery( '#' + editor.id ); textarea.val( content ).trigger( 'keyup' ).trigger( 'paste' ).trigger( 'mouseover' ); });}" ), ), $this, $form, $entry ); $editor_settings = apply_filters( sprintf( 'gform_rich_text_editor_options_%d', $form['id'] ), $editor_settings, $this, $form, $entry ); $editor_settings = apply_filters( sprintf( 'gform_rich_text_editor_options_%d_%d', $form['id'], $this->id ), $editor_settings, $this, $form, $entry ); if ( ! has_action( 'wp_tiny_mce_init', array( __class__, 'start_wp_tiny_mce_init_buffer' ) ) ) { add_action( 'wp_tiny_mce_init', array( __class__, 'start_wp_tiny_mce_init_buffer' ) ); } ob_start(); wp_editor( $value, $field_id, $editor_settings ); $input = ob_get_clean(); remove_filter( 'mce_buttons', array( $this, 'filter_mce_buttons' ), 10 ); remove_filter( 'mce_buttons_2', array( $this, 'filter_mce_buttons' ), 10 ); remove_filter( 'mce_buttons_3', array( $this, 'filter_mce_buttons' ), 10 ); remove_filter( 'mce_buttons_4', array( $this, 'filter_mce_buttons' ), 10 ); } else { $input = ''; $input_style = ''; // RTE preview if ( $this->is_form_editor() ) { $display = $this->useRichTextEditor ? 'block' : 'none'; $input_style = $this->useRichTextEditor ? 'style="display:none;"' : ''; $size = $this->size ? $this->size : 'medium'; $input = "
"; } $input .= ""; } return sprintf( "
%s
", $input ); } public function validate( $value, $form ) { if ( ! is_numeric( $this->maxLength ) ) { return; } if ( $this->useRichTextEditor ) { $value = wp_specialchars_decode( $value ); } // Clean the string of characters not counted by the textareaCounter plugin. $value = strip_tags( $value ); $value = str_replace( "\r", '', $value ); $value = trim( $value ); if ( GFCommon::safe_strlen( $value ) > $this->maxLength ) { $this->failed_validation = true; $this->validation_message = empty( $this->errorMessage ) ? esc_html__( 'The text entered exceeds the maximum number of characters.', 'gravityforms' ) : $this->errorMessage; } } public static function start_wp_tiny_mce_init_buffer() { ob_start(); add_action( 'after_wp_tiny_mce', array( __class__, 'end_wp_tiny_mce_init_buffer' ), 1 ); } public static function end_wp_tiny_mce_init_buffer() { $script = ob_get_clean(); $pattern = '/()([\s\S]+)(<\/script>)/'; preg_match_all( $pattern, $script, $matches, PREG_SET_ORDER ); // Fix editor height issue: https://core.trac.wordpress.org/ticket/45461. $wp_version = get_bloginfo( 'version' ); $height_issue_fix = version_compare( $wp_version, '5.0', '>=' ) && version_compare( $wp_version, '5.2', '<' ) ? ' gform_post_conditional_logic' : ''; foreach ( $matches as $match ) { list( $search, $open_tag, $guts, $close_tag ) = $match; $custom = "\tif ( typeof current_page === 'undefined' ) { return; }\n\twindow.gformInitTinymce = function(){\n\tfor( var id in tinymce.editors ) { tinymce.EditorManager.remove( tinymce.editors[id] ); }"; $replace = sprintf( "%s\nfunction gformInitMCEInstances() { jQuery( document ).on( 'gform_post_render%s', function( event, form_id, current_page ) { \n%s\n%s\n\t}\n\tgformInitTinymce();\n} );}; gform.initializeOnLoaded( gformInitMCEInstances );\n%s", $open_tag, $height_issue_fix, $custom, $guts, $close_tag ); $script = str_replace( $search, $replace, $script ); } echo $script; } public function filter_mce_buttons( $mce_buttons, $editor_id ) { $remove_key = array_search( 'wp_more', $mce_buttons ); if ( $remove_key !== false ) { unset( $mce_buttons[ $remove_key ] ); } // Get current filter to detect which mce_buttons core filter is running. $current_filter = current_filter(); // Depending on the current mce_buttons filter, set variable to support filtering all potential rows. switch ( $current_filter ) { case 'mce_buttons_2': $mce_filter = '_row_two'; break; case 'mce_buttons_3': $mce_filter = '_row_three'; break; case 'mce_buttons_4': $mce_filter = '_row_four'; break; default: $mce_filter = ''; break; } /** * Filters the buttons within the TinyMCE editor * * @since 2.0.0 * * @param array $mce_buttons Buttons to be included. * @param string $editor_id HTML ID of the field. * @param object $this The field object * * Additional filters for specific form and fields IDs. */ $mce_buttons = gf_apply_filters( array( 'gform_rich_text_editor_buttons' . $mce_filter, $this->formId, $this->id ), $mce_buttons, $editor_id, $this ); return $mce_buttons; } /** * Format the entry value for display on the entry detail page and for the {all_fields} merge tag. * Return a value that's safe to display for the context of the given $format. * * @param string|array $value The field value. * @param string $currency The entry currency code. * @param bool|false $use_text When processing choice based fields should the choice text be returned instead of the value. * @param string $format The format requested for the location the merge is being used. Possible values: html, text or url. * @param string $media The location where the value will be displayed. Possible values: screen or email. * * @return string */ public function get_value_entry_detail( $value, $currency = '', $use_text = false, $format = 'html', $media = 'screen' ) { if ( $format === 'html' ) { $allowable_tags = $this->get_allowable_tags(); if ( $allowable_tags === false ) { // The value is unsafe so encode the value. $value = esc_html( $value ); $return = nl2br( $value ); } else { // The value contains HTML but the value was sanitized before saving. $return = wpautop( $value ); } } else { $return = $value; } return $return; } /** * Format the entry value for when the field/input merge tag is processed. Not called for the {all_fields} merge tag. * * Return a value that is safe for the context specified by $format. * * @since Unknown * @access public * * @param string|array $value The field value. Depending on the location the merge tag is being used the following functions may have already been applied to the value: esc_html, nl2br, and urlencode. * @param string $input_id The field or input ID from the merge tag currently being processed. * @param array $entry The Entry Object currently being processed. * @param array $form The Form Object currently being processed. * @param string $modifier The merge tag modifier. e.g. value * @param string|array $raw_value The raw field value from before any formatting was applied to $value. * @param bool $url_encode Indicates if the urlencode function may have been applied to the $value. * @param bool $esc_html Indicates if the esc_html function may have been applied to the $value. * @param string $format The format requested for the location the merge is being used. Possible values: html, text or url. * @param bool $nl2br Indicates if the nl2br function may have been applied to the $value. * * @return string */ public function get_value_merge_tag( $value, $input_id, $entry, $form, $modifier, $raw_value, $url_encode, $esc_html, $format, $nl2br ) { if ( $format === 'html' ) { $form_id = absint( $form['id'] ); $allowable_tags = $this->get_allowable_tags( $form_id ); if ( $allowable_tags === false ) { // The raw value is unsafe so escape it. $return = esc_html( $raw_value ); // Run nl2br() to preserve line breaks when auto-formatting is disabled on notifications/confirmations. $return = nl2br( $return ); } else { // The value contains HTML but the value was sanitized before saving. $return = wpautop( $raw_value ); } } else { $return = $value; } return $return; } /** * Determines if the RTE can be enabled for the current field and user. * * @since 2.2.5.14 * * @return bool */ public function is_rich_edit_enabled() { if ( ! $this->useRichTextEditor ) { return false; } global $wp_rich_edit; $wp_rich_edit = null; add_filter( 'get_user_option_rich_editing', array( $this, 'filter_user_option_rich_editing' ) ); $user_can_rich_edit = user_can_richedit(); remove_filter( 'get_user_option_rich_editing', array( $this, 'filter_user_option_rich_editing' ) ); return $user_can_rich_edit; } /** * Filter the rich_editing option for the current user. * * @since 2.2.5.14 * * @param string $value The value of the rich_editing option for the current user. * * @return string */ public function filter_user_option_rich_editing( $value ) { return 'true'; } // # FIELD FILTER UI HELPERS --------------------------------------------------------------------------------------- /** * Returns the filter operators for the current field. * * @since 2.4 * * @return array */ public function get_filter_operators() { $operators = parent::get_filter_operators(); $operators[] = 'contains'; return $operators; } } GF_Fields::register( new GF_Field_Textarea() );