File: System\Windows\Controls\RichTextBox.cs
Web Access
Project: src\src\Microsoft.DotNet.Wpf\src\PresentationFramework\PresentationFramework.csproj (PresentationFramework)
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
 
using MS.Internal;
using MS.Internal.Documents;
using System.Windows.Input; // KeyboardNavigation
using System.ComponentModel; // DefaultValue
using System.Windows.Controls.Primitives; // TextBoxBase
using System.Windows.Documents; // TextEditor
using System.Windows.Automation.Peers; // AutomationPattern
using System.Windows.Markup; // IAddChild
using System.Collections; // IEnumerator
using MS.Internal.Controls; // EmptyEnumerator
using MS.Internal.Telemetry.PresentationFramework;
 
//
// Description: The stock rich text editing control.
//
 
namespace System.Windows.Controls
{
    /// <summary>
    /// RichTextBox control
    /// </summary>
    [Localizability(LocalizationCategory.Inherit)]
    [ContentProperty("Document")]
    public class RichTextBox : TextBoxBase, IAddChild
    {
        // -----------------------------------------------------------
        //
        // Constructors
        //
        // -----------------------------------------------------------
 
        #region Constructors
 
        /// <summary>
        /// Static constructor for RichTextBox.
        /// </summary>
        static RichTextBox()
        {
            DefaultStyleKeyProperty.OverrideMetadata(typeof(RichTextBox), new FrameworkPropertyMetadata(typeof(RichTextBox)));
            _dType = DependencyObjectType.FromSystemTypeInternal(typeof(RichTextBox));
 
            // Default value for AcceptsReturn is true
            KeyboardNavigation.AcceptsReturnProperty.OverrideMetadata(typeof(RichTextBox), new FrameworkPropertyMetadata(true));
 
            // Default value for AutoWordSelection is false.  We want true.
            TextBoxBase.AutoWordSelectionProperty.OverrideMetadata(typeof(RichTextBox), new FrameworkPropertyMetadata(true));
 
            if (!FrameworkAppContextSwitches.UseAdornerForTextboxSelectionRendering)
            {
                
                // Override the default selection opacity so if FrameworkAppContextSwitches.UseAdornerForTextboxSelectionRendering
                // is false, we still get the appropriate value.
                TextBoxBase.SelectionOpacityProperty.OverrideMetadata(typeof(RichTextBox), new FrameworkPropertyMetadata(TextBoxBase.AdornerSelectionOpacityDefaultValue));
            }
 
            // We need to transfer all character formatting properties and some behavioral inheriting properties
            // from RichTextBox level into its FlowDocument.
            // For this purpose we set listeners for all these properties:
            HookupInheritablePropertyListeners();
 
            ControlsTraceLogger.AddControl(TelemetryControls.RichTextBox);
        }
 
        /// <summary>
        /// Initializes a new instance of RichTextBox control.
        /// </summary>
        /// <remarks>
        /// Creates implicit instance of a FlowDocument as its initial content.
        /// The initial document will contain one Paragraph with an empty Run in it.
        /// </remarks>
        public RichTextBox() 
            : this(null)
        {
        }
 
        /// <summary>
        /// Initializes a new instance of RichTextBox control and specifies a FlowDocument as its content
        /// </summary>
        /// <param name="document">
        /// A FlowDocument specified as a content for this instance of RichTextBox.
        /// </param>
        public RichTextBox(FlowDocument document)
            : base()
        {
            // Register static editing command handlers.
            // This only has an effect that first time we make the call.
            // We don't use the static ctor because there are cases
            // where another control will want to alias our properties
            // but doesn't need this overhead.
            TextEditor.RegisterCommandHandlers(typeof(RichTextBox), /*acceptsRichContent:*/true, /*readOnly*/false, /*registerEventListeners*/false);
 
            // Create TextContainer and TextEditor associated with it
            if (document == null)
            {
                document = new FlowDocument();
                document.Blocks.Add(new Paragraph());
 
                // Mark the document as implicit.
                // This flag will affect these behaviors:
                //  a) RichTextBox serialization will not output FlowDocument child if it is implicit and still empty
                //  b) IAddChild will allow adding the first child (which will become non-implicit, and would not allow the subsecuent additions)
                //  c) Property inheritance from RichTextBox to its FlowDocument will work only for implicit document
                // This call must be done after Document assignment, as it always clears the _implicitDocument field.
                _implicitDocument = true;
            }
 
            // Initialize the Document property.
            // Note that _implicitDocument flag was set to true/false just before that.
            // The peemptive flag setting has its effect only when _document is null (this case only),
            // otherwise the new document would be considered as explicit (so flag would be cleared by the Document property assignment).
            this.Document = document;
 
            // Values that must be set as a side effect of Document assignment,
            // that are required for the RichTextBox instance functioning.
            Invariant.Assert(this.TextContainer != null);
            Invariant.Assert(this.TextEditor != null);
            Invariant.Assert(this.TextEditor.TextContainer == this.TextContainer);
        }
 
        #endregion Constructors
 
        // -----------------------------------------------------------
        //
        // Public Methods
        //
        // -----------------------------------------------------------
 
        #region Public Methods
 
        // -----------------------------------------------------------
        //
        // IAddChild interface
        //
        // -----------------------------------------------------------
 
        ///<summary>
        /// This method is called to Add the object as a child of the RichTextBox.  This method is used primarily
        /// by the parser; a more direct way of adding a child to a RichTextBox is to use the <see cref="Document" />
        /// property.
        ///</summary>
        ///<param name="value">
        /// The object to add as a child; it must be a UIElement.
        ///</param>
        void IAddChild.AddChild(Object value)
        {
            ArgumentNullException.ThrowIfNull(value);
 
            if (!(value is FlowDocument))
            {
                throw new ArgumentException(SR.Format(SR.UnexpectedParameterType, value.GetType(), typeof(FlowDocument)), "value");
            }
 
            if (!_implicitDocument)
            {
                throw new ArgumentException(SR.Format(SR.CanOnlyHaveOneChild, this.GetType(), value.GetType()));
            }
 
            this.Document = (FlowDocument)value;
        }
 
        ///<summary>
        /// This method is called by the parser when text appears under the tag in markup.
        /// As RichTextBox do not support text, calling this method has no effect if the text
        /// is all whitespace.  For non-whitespace text, throw an exception.
        ///</summary>
        ///<param name="text">
        /// Text to add as a child.
        ///</param> 
        void IAddChild.AddText(string text)
        {
            ArgumentNullException.ThrowIfNull(text);
 
            XamlSerializerUtil.ThrowIfNonWhiteSpaceInAddText(text, this);
        }
 
 
        ///<summary>
        /// Returns the TextPointer located closest to a supplied Point.
        /// </summary>
        /// <param name="point">
        /// Point to query, in the coordinate space of the RichTextBox.
        /// </param>
        /// <param name="snapToText">
        /// If true, this method will always return a TextPointer --
        /// the closest position as calculated by the control's heuristics. 
        /// If false, this method will return a null position if the test 
        /// point does not fall within any character bounding box.
        /// </param>
        /// <returns>
        /// The closest TextPointer to the supplied Point, or null if
        /// snapToText is false and the supplied Point is not contained
        /// within any character bounding box, or if no content element yet exists.
        /// </returns>
        /// <exception cref="System.InvalidOperationException">
        /// Throws InvalidOperationException if layout is dirty.
        /// </exception>
        public TextPointer GetPositionFromPoint(Point point, bool snapToText)
        {
            if (this.RenderScope == null)
            {
                return null;
            }
 
            return (TextPointer)GetTextPositionFromPointInternal(point, snapToText);
        }
 
        /// <summary>
        /// Returns the associated SpellingError at a specified position.
        /// </summary>
        /// <param name="position">
        /// Position of text to query.
        /// </param>
        /// <remarks>
        /// The position and its LogicalDirection specify a character to query.
        /// If the specificed character is not part of a misspelled word then
        /// this method will return null.
        /// </remarks>
        public SpellingError GetSpellingError(TextPointer position)
        {
            ValidationHelper.VerifyPosition(this.TextContainer, position);
 
            return this.TextEditor.GetSpellingErrorAtPosition(position, position.LogicalDirection);
        }
 
        /// <summary>
        /// Returns the TextRange covering a misspelled word at a specified
        /// position.
        /// </summary>
        /// <param name="position">
        /// Position of text to query.
        /// </param>
        /// <remarks>
        /// The position and its LogicalDirection specify a character to query.
        /// If the specificed character is not part of a misspelled word then
        /// this method will return null.
        /// </remarks>
        public TextRange GetSpellingErrorRange(TextPointer position)
        {
            ValidationHelper.VerifyPosition(this.TextContainer, position);
 
            SpellingError spellingError = this.TextEditor.GetSpellingErrorAtPosition(position, position.LogicalDirection);
 
            return (spellingError == null) ? null : new TextRange(spellingError.Start, spellingError.End);
        }
 
        /// <summary>
        /// Returns the position of the next character in a specificed direction
        /// that is the start of a misspelled word.
        /// </summary>
        /// <param name="position">
        /// Position of text to query.
        /// </param>
        /// <param name="direction">
        /// Direction to query.
        /// </param>
        /// <remarks>
        /// The position and its LogicalDirection specify a character to query.
        /// The search includes the spelling error containing the character
        /// specified by position/direction (if any).
        /// 
        /// If no misspelled word is encountered, the method returns null.
        /// </remarks>
        public TextPointer GetNextSpellingErrorPosition(TextPointer position, LogicalDirection direction)
        {
            ValidationHelper.VerifyPosition(this.TextContainer, position);
 
            return (TextPointer)this.TextEditor.GetNextSpellingErrorPosition(position, direction);
        }
 
        #endregion Public Methods
 
        //------------------------------------------------------
        //
        //  Protected Methods
        //
        //------------------------------------------------------
 
        #region Protected Methods
 
        /// <summary>
        /// Creates AutomationPeer (<see cref="UIElement.OnCreateAutomationPeer"/>)
        /// </summary>
        protected override AutomationPeer OnCreateAutomationPeer() 
        {
            return new RichTextBoxAutomationPeer(this);
        }
 
        protected override void OnDpiChanged(DpiScale oldDpiScaleInfo, DpiScale newDpiScaleInfo)
        {
            Document?.SetDpi(newDpiScaleInfo);
        }
 
        /// <summary>
        /// Measurement override. Implement your size-to-content logic here.
        /// </summary>
        /// <param name="constraint">
        /// Sizing constraint.
        /// </param>
        protected override Size MeasureOverride(Size constraint)
        {
            if (constraint.Width == Double.PositiveInfinity)
            {
                // If we're sized to infinity, we won't behave the same way TextBox does under
                // the same conditions.  So, we fake it.
                constraint.Width = this.MinWidth;
            }
            return base.MeasureOverride(constraint);
        }
 
        // Allocates the initial render scope for this control.
        internal override FrameworkElement CreateRenderScope()
        {
            FlowDocumentView renderScope = new FlowDocumentView
            {
                Document = this.Document
            };
 
            // Set a margin so that the BiDi Or Italic caret has room to render at the edges of content.
            // Otherwise, anti-aliasing or italic causes the caret to be partially clipped.
            renderScope.Document.PagePadding = new Thickness(CaretElement.CaretPaddingWidth, 0, CaretElement.CaretPaddingWidth, 0);
 
            // We want current style to ignore all properties from theme style for renderScope.
            renderScope.OverridesDefaultStyle = true;
 
            return renderScope;
        }
 
        #endregion Protected Methods
 
        // -----------------------------------------------------------
        //
        // Public Properties
        //
        // -----------------------------------------------------------
 
        #region Public Properties
 
        /// <summary>
        /// A Property representing a content of this RichTextBox
        /// </summary>
        public FlowDocument Document
        {
            get
            {
                Invariant.Assert(_document != null);
                return _document;
            }
 
            set
            {
                ArgumentNullException.ThrowIfNull(value);
 
                if (value != _document &&
                    value.StructuralCache != null && value.StructuralCache.TextContainer != null && 
                    value.StructuralCache.TextContainer.TextSelection != null)
                {
                    throw new ArgumentException(SR.RichTextBox_DocumentBelongsToAnotherRichTextBoxAlready);
                }
 
                if (_document != null && this.TextSelectionInternal.ChangeBlockLevel > 0)
                {
                    throw new InvalidOperationException(SR.RichTextBox_CantSetDocumentInsideChangeBlock);
                }
 
                if (value == _document)
                {
                    // Same document nothing to do.
                    return;
                }
 
                // Identify the case for the _document initialization
                bool initialSetting = _document == null;
                
                // Detach existing FlowDocument
                if (_document != null)
                {
                    // Detach PageSize change listener
                    _document.PageSizeChanged -= new EventHandler(this.OnPageSizeChangedHandler);
 
                    // Remove the document from the logical tree
                    this.RemoveLogicalChild(_document);
 
                    // Stop collecting text changes
                    _document.TextContainer.CollectTextChanges = false;
 
                    // Clear thereference to a document
                    _document = null;
                }
 
                // Clear the implicitDocument flag.
                // Any assignment to the Document property clears the _implicitDocument flag
                // expect for the initial one, which may happen from a RichTextBox constructor which
                // can create an empolicit document and sets _implicitDocument flag before
                // the Document property assignment.
                if (!initialSetting)
                {
                    _implicitDocument = false;
                }
 
                // Store the document for future use - just for comparing when it changes and detaching from it
                _document = value;
                _document.SetDpi(this.GetDpi());
 
                // Save existing renderScope before calling TextBoxBase.InitializeTextContainer,
                // because it will discard it.
                UIElement renderScope = this.RenderScope;
 
                // Start collecting text changes
                _document.TextContainer.CollectTextChanges = true;
 
                // Attach to new text container and re-create TextEditor instance
                this.InitializeTextContainer(_document.TextContainer);
 
                // Add listener for PageSize - to redirect it to inner renderScope.Width
                _document.PageSizeChanged += new EventHandler(this.OnPageSizeChangedHandler);
 
                // Add the document as a child to the logical tree
                this.AddLogicalChild(_document);
 
                // Re-attach to visual tree
                if (renderScope != null)
                {
                    // Re-atach to visual tree if we have a new TextContainer.
                    this.AttachToVisualTree();
                }
 
                // Make sure that all inherited properties properly transferred
                // to the new document according to its Standalone/Inherited status.
                TransferInheritedPropertiesToFlowDocument();
 
                // Raise a TextChanged event for all assignments except initializing one.
                if (!initialSetting)
                {
                    //Re-apply the cached undo properties
                    this.ChangeUndoLimit(this.UndoLimit);
                    this.ChangeUndoEnabled(this.IsUndoEnabled);
 
                    Invariant.Assert(this.PendingUndoAction == UndoAction.None);
                    this.PendingUndoAction = UndoAction.Clear;
                    try
                    {
                        this.OnTextChanged(new TextChangedEventArgs(TextChangedEvent, UndoAction.Clear));
                    }
                    finally
                    {
                        this.PendingUndoAction = UndoAction.None;
                    }
                }
            }
        }
 
        /// <summary>
        /// This method is used by TypeDescriptor to determine if this property should
        /// be serialized.
        /// </summary>
        // According to the contract with the serializer the (private) method named as
        // ShouldSerialize<PropertyName> tells the serializer whether it should serialize the
        // property for this instance or not.
        // We want to avoid serializing implicitly created document if it does not have any meaningful content.
        [EditorBrowsable(EditorBrowsableState.Never)]
        public bool ShouldSerializeDocument()
        {
            Block firstBlock = _document.Blocks.FirstBlock;
 
            if (_implicitDocument &&
                (firstBlock == null
                    ||
                    firstBlock == _document.Blocks.LastBlock &&
                    firstBlock is Paragraph))
            {
                Inline firstInline = (firstBlock == null) ? null : ((Paragraph)firstBlock).Inlines.FirstInline;
                if (firstInline == null
                        ||
                        firstInline == ((Paragraph)firstBlock).Inlines.LastInline &&
                        firstInline is Run &&
                        firstInline.ContentStart.CompareTo(firstInline.ContentEnd) == 0)
                {
                    // We have implicit document without any text content.
                    return false;
                }
            }
 
            // In all other cases we should serialize the FlowDocument child.
            return true;
        }
 
        /// <summary>
        /// Enables or disables TextElements and UIElements contained in this RichTextBox's FlowDocument.
        /// </summary>
        /// <remarks>
        /// By default child elements have their IsEnabled property coerced
        /// false when hosted by RichTextBox.  Use this property to enable
        /// contained elements.
        /// </remarks>
        public static readonly DependencyProperty IsDocumentEnabledProperty =
            DependencyProperty.Register("IsDocumentEnabled", typeof(bool), typeof(RichTextBox),
            new FrameworkPropertyMetadata(false, new PropertyChangedCallback(OnIsDocumentEnabledChanged)));
 
        /// <summary>
        /// Enables or disables TextElements and UIElements contained in this RichTextBox's FlowDocument.
        /// </summary>
        /// <remarks>
        /// <see cref="IsDocumentEnabledProperty" />
        /// </remarks>
        public bool IsDocumentEnabled
        {
            get
            {
                return (bool)GetValue(IsDocumentEnabledProperty);
            }
 
            set
            {
                SetValue(IsDocumentEnabledProperty, value);
            }
        }
 
        // ...........................................................
        //
        // Content Accessing Properties
        //
        // ..........................................................
 
        #region Content Accessing Properties
 
        /// <summary>
        /// Returns enumerator to logical children
        /// </summary>
        protected internal override IEnumerator LogicalChildren
        {
            get
            {
                if (this._document == null)
                {
                    // Using _document (not .Document),as it can be null on destruction scenarios 
                    // (even though .Document cannot be null) - it is called for property invalidation reasons...
                    return EmptyEnumerator.Instance;
                }
                else
                {
                    return new SingleChildEnumerator(this._document);
                }
            }
        }
 
        /// <summary>
        /// Text Selection (readonly)
        /// </summary>
        public TextSelection Selection
        {
            get
            {
                return (TextSelection)TextSelectionInternal;
            }
        }
 
        /// <summary>
        /// Position of the caret.
        /// </summary>
        [DesignerSerializationVisibility(DesignerSerializationVisibility.Hidden)]
        public TextPointer CaretPosition
        {
            get
            {
                return Selection.MovingPosition;
            }
 
            set
            {
                ArgumentNullException.ThrowIfNull(value);
                if (!Selection.Start.IsInSameDocument(value))
                {
                    throw new ArgumentException(SR.RichTextBox_PointerNotInSameDocument, "value");
                }
                Selection.SetCaretToPosition(value, value.LogicalDirection, /*allowStopAtLineEnd:*/true, /*allowStopNearSpace:*/false);
            }
        }
        
        #endregion Content Accessing Properties
 
        #endregion Public Properties
 
        //------------------------------------------------------
        //
        //  Internal Methods
        //
        //------------------------------------------------------
 
        #region Internal Methods
 
        // Returns the DependencyObjectType for the registered ThemeStyleKey's default 
        // value. Controls will override this method to return approriate types.
        internal override DependencyObjectType DTypeThemeStyleKey
        {
            get
            {
                return _dType;
            }
        }
 
        #endregion Internal Methods
 
        // -----------------------------------------------------------
        //
        // Private Methods
        //
        // -----------------------------------------------------------
 
        #region Private Methods
 
        // ...........................................................
        //
        // Transferring Properties from RichTextBox down to FlowDocument
        //
        // ...........................................................
 
        // We need to transfer all character formatting properties from RichTextBox level into its FlowDocument.
        //
        // Note that we have to set all properties explicitly - not even trying
        // to bypass some of them when using assumptions about their default values (for non-implicit document)
        // or a coincidence with the current context values (for implicit document) -
        // the FlowDocument mast have all properties explicitly set to keep them intact
        // after being moved to another contextual location (both in Save-Load and in Print scenarios).
 
        // For this purpose we set listeners for all these properties:
        private static void HookupInheritablePropertyListeners()
        {
            // All inhgeriting formatting properties need to be transferred over the implicit document boundary.
            // This is required for treating UIContext as default setting for implicit document.
            // Note that mechanism is not applied to explicit document.
            PropertyChangedCallback formattingPropertyCallback = new PropertyChangedCallback(OnFormattingPropertyChanged);
            DependencyProperty[] inheritableFormattingProperties = TextSchema.GetInheritableProperties(typeof(FlowDocument));
            for (int i = 0; i < inheritableFormattingProperties.Length; i++)
            {
                inheritableFormattingProperties[i].OverrideMetadata(typeof(RichTextBox), new FrameworkPropertyMetadata(formattingPropertyCallback));
            }
 
            // Inheriting behavioral properties need to be transferred over any Standalone document boundary.
            PropertyChangedCallback behavioralPropertyCallback = new PropertyChangedCallback(OnBehavioralPropertyChanged);
            DependencyProperty[] inheritableBehavioralProperties = TextSchema.BehavioralProperties;
            for (int i = 0; i < inheritableBehavioralProperties.Length; i++)
            {
                inheritableBehavioralProperties[i].OverrideMetadata(typeof(RichTextBox), new FrameworkPropertyMetadata(behavioralPropertyCallback));
            }
        }
 
        // Transfer all properties from the current context to this
        // implicit document - to simulate inheritance.
        // This is how implicit document gets its initial values -
        // from the context.
        // After that RichTextBox will maintain such inheritance simulation
        // (for implicit document only) by listening for property
        // change notifications.
        private void TransferInheritedPropertiesToFlowDocument()
        {
            // Implicit document needs all formatting properties be transferred to it
            if (_implicitDocument)
            {
                DependencyProperty[] inheritableFormattingProperties = TextSchema.GetInheritableProperties(typeof(FlowDocument));
                for (int i = 0; i < inheritableFormattingProperties.Length; i++)
                {
                    DependencyProperty property = inheritableFormattingProperties[i];
                    TransferFormattingProperty(property, this.GetValue(property));
                }
            }
 
            // Behavioral properties must be transferred to any document whether it has Standalone
            // or Inherited FormattingDefaults. All such values are set as local values even
            // in case when they are equal to the ones inherited from the UI context
            // (see TransferBehavioralProperty method implementation).
            // Such strong logic is needed to work correctly in any sequence of
            // setting FormattingDefaults property and adding children.
            DependencyProperty[] inheritableBehavioralProperties = TextSchema.BehavioralProperties;
            for (int i = 0; i < inheritableBehavioralProperties.Length; i++)
            {
                DependencyProperty property = inheritableBehavioralProperties[i];
                TransferBehavioralProperty(property, this.GetValue(property));
            }
        }
 
        /// <summary>
        /// Callback for changes to the any text formatting property.
        /// Transfers the new value to explisit setting on a FlowDocument.
        /// </summary>
        private static void OnFormattingPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            RichTextBox richTextBox = (RichTextBox)d;
 
            if (richTextBox._implicitDocument)
            {
                richTextBox.TransferFormattingProperty(e.Property, e.NewValue);
            }
        }
 
        // Transfers single formatting property from this RichTextBox to its implicit document
        private void TransferFormattingProperty(DependencyProperty property, object inheritedValue)
        {
            Invariant.Assert(_implicitDocument, "We only supposed to do this for implicit documents");
 
            object defaultValue = _document.GetValue(property);
            if (!TextSchema.ValuesAreEqual(inheritedValue, defaultValue))
            {
                _document.ClearValue(property);
                defaultValue = _document.GetValue(property);
                if (!TextSchema.ValuesAreEqual(inheritedValue, defaultValue))
                {
                    _document.SetValue(property, inheritedValue);
                }
            }
        }
 
        /// <summary>
        /// Callback for changes to the any behavioral property.
        /// Transfers the new value to a FlowDocument.
        /// </summary>
        /// <remarks>
        /// Behavioral properties must be transferred to any document whether it has Standalone
        /// or Inherited FormattingDefaults. All such values are set as local values even
        /// in case when they are equal to the ones inherited from the UI context.
        /// Such strong logic is needed to work correctly in any sequence of
        /// setting FormattingDefaults property and adding children.
        /// </remarks>
        private static void OnBehavioralPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            RichTextBox richTextBox = (RichTextBox)d;
 
            richTextBox.TransferBehavioralProperty(e.Property, e.NewValue);
        }
 
        /// <summary>
        /// Transfers single behavioral property from this RichTextBox to its implicit document
        /// </summary>
        /// <remarks>
        /// Behavioral properties must be transferred to any document whether it has Standalone
        /// or Inherited FormattingDefaults. All such values are set as local values even
        /// in case when they are equal to the ones inherited from the UI context.
        /// Such strong logic is needed to work correctly in any sequence of
        /// setting FormattingDefaults property and adding children.
        /// </remarks>
        private void TransferBehavioralProperty(DependencyProperty property, object inheritedValue)
        {
            // Set the value unconditionally as explicit local value
            _document.SetValue(property, inheritedValue);
        }
 
        // ...........................................................
        //
        // TextEditor Parameterization Properties Access
        //
        // ...........................................................
 
        #region TextEditor Parameterization Propeties Access
 
        private void OnPageSizeChangedHandler(object sender, EventArgs e)
        {
            if (this.RenderScope == null)
            {
                return;
            }
 
            // Make sure that the TextWrapping property is set correctly
            if (this.Document != null)
            {
                this.Document.TextWrapping = TextWrapping.Wrap;
            }
 
            // The Document does not have explicit PageWidth set OR Wrap/WrapWithOverflow is requested.
            // The RenderScope must occupy as much space as its content required (no wrapping)
            // We could set Width to positive infinity, but that would make horizontal scrollbar
            // look wrong. So we need to make sure that render scope has a size of
            // its actual content. For that we clear loal value of Width property
            // from RenderScope.
 
            // To let RenderScope occupy all space we clear Width local value on it.
            this.RenderScope.ClearValue(FlowDocumentView.WidthProperty);
 
            // Set alighment to Stretch - for RenderScope to occupy the whole viewport.
            this.RenderScope.ClearValue(FrameworkElement.HorizontalAlignmentProperty);
            // Normally TextBox style does not set any balue for HorizontalAlignment property,
            // so clearing would set it to a required default - Stretch.
            // However, if style author sets some other value - it will be set as a result of ClearValue,
            // so we need to enforce that.
            // Note, that trying to clear first saves a memory for inline property application.
            if (this.RenderScope.HorizontalAlignment != HorizontalAlignment.Stretch)
            {
                this.RenderScope.HorizontalAlignment = HorizontalAlignment.Stretch;
            }
        }
 
 
        #endregion TextEditor Parameterization Properties Access
 
        // Callback for IsDocumentEnabledProperty changes.
        // Forces a coercion of the child FlowDocument, with the new setting.
        private static void OnIsDocumentEnabledChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
        {
            RichTextBox richTextBox = (RichTextBox)d;
 
            if (richTextBox.Document != null)
            {
                richTextBox.Document.CoerceValue(IsEnabledProperty);
            }
        }
 
        #endregion Private Methods
 
        // -----------------------------------------------------------
        //
        // Private Fields
        //
        // -----------------------------------------------------------
 
        #region Private Fields
 
        private FlowDocument _document;
        private bool _implicitDocument; // true if Document property was set by AddChild or Document setter (not in constructor)
 
        private static DependencyObjectType _dType; // Needed for property system optimization
 
        #endregion Private Fields
    }
}