File: MS\Internal\Documents\DocumentGrid.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.
 
//
// Description: DocumentGrid displays DocumentPaginator content in a grid-like
//              arrangement and is used by DocumentViewer to display documents.
//
 
 
using MS.Utility;
using System.Windows;
using System.Windows.Controls;
using System.Windows.Controls.Primitives;
using System.Windows.Documents;
using System.Windows.Input;
using System.Windows.Media;
using System.Windows.Threading;
using System.Collections;
using System.Collections.ObjectModel;
 
namespace MS.Internal.Documents
{
    /// <summary>
    /// DocumentGrid is an internal Avalon FrameworkElement that executes all the
    /// "heavy lifting" involved in loading and displaying an DocumentPaginator-based
    /// document inside of a DocumentViewer control.
    /// </summary>
    /// <speclink>http://d2/DRX/default.aspx</speclink>
    internal class DocumentGrid : FrameworkElement, IDocumentScrollInfo
    {
        //------------------------------------------------------
        //
        //  Constructors
        //
        //------------------------------------------------------
        #region Constructors
 
        /// <summary>
        /// Static constructor
        /// </summary>
        static DocumentGrid()
        {
            //Register for the RequestBringIntoView event so we can get BIV events for
            //TextEditor IP movements.
            EventManager.RegisterClassHandler(typeof(DocumentGrid),
                RequestBringIntoViewEvent,
                new RequestBringIntoViewEventHandler(OnRequestBringIntoView));
            //Register the default ContextMenu
            DocumentGridContextMenu.RegisterClassHandler();
        }
 
        /// <summary>
        /// The constructor
        /// </summary>
        public DocumentGrid() : base()
        {
            Initialize();
        }
 
        #endregion Constructors
 
        //------------------------------------------------------
        //
        //  Internal Methods
        //
        //------------------------------------------------------
        #region Internal Methods
        /// <summary>
        /// Hit-Test on the multi-page UI scope to return a DocumentPage
        /// that contains this point
        /// </summary>
        /// <param name="point">Point in pixel unit, relative to the UI Scope's coordinates</param>
        /// <returns>A DocumentPage that is hit or null if no page is hit</returns>
        internal DocumentPage GetDocumentPageFromPoint(Point point)
        {
            DocumentPageView dp = GetDocumentPageViewFromPoint(point);
 
            // if we hit a DocumentPageView we can return its DocumentPage.
            if (dp != null)
            {
                return dp.DocumentPage;
            }
 
            //Nothing hit, return null.
            return null;
        }
 
        #endregion Internal Methods
 
 
        //------------------------------------------------------
        //
        //  Public Interfaces
        //
        //------------------------------------------------------
        #region Interface Implementations
 
 
        #region IDocumentScrollInfo
        //------------------------------------------------------
        //
        //  IDocumentScrollInfo Methods
        //
        //------------------------------------------------------
 
        /// <summary>
        /// Scroll content by one line to the top.
        /// </summary>
        public void LineUp()
        {
            if (_canVerticallyScroll)
            {
                SetVerticalOffsetInternal(VerticalOffset - _verticalLineScrollAmount);
            }
        }
 
        /// <summary>
        /// Scroll content by one line to the bottom.
        /// </summary>
        public void LineDown()
        {
            if (_canVerticallyScroll)
            {
                SetVerticalOffsetInternal(VerticalOffset + _verticalLineScrollAmount);
 
                //Perf Tracing - Mark LineDown Start
                EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXLineDown);
            }
        }
 
        /// <summary>
        /// Scroll content by one line to the left.
        /// </summary>
        public void LineLeft()
        {
            if (_canHorizontallyScroll)
            {
                SetHorizontalOffsetInternal(HorizontalOffset - _horizontalLineScrollAmount);
            }
        }
 
        /// <summary>
        /// Scroll content by one line to the right.
        /// </summary>
        public void LineRight()
        {
            if (_canHorizontallyScroll)
            {
                SetHorizontalOffsetInternal(HorizontalOffset + _horizontalLineScrollAmount);
            }
        }
 
        /// <summary>
        /// Scroll content by one viewport to the top.
        /// </summary>
        public void PageUp()
        {
            SetVerticalOffsetInternal(VerticalOffset - ViewportHeight);
        }
 
        /// <summary>
        /// Scroll content by one viewport to the bottom.
        /// </summary>
        public void PageDown()
        {
            SetVerticalOffsetInternal(VerticalOffset + ViewportHeight);
 
            //Perf Tracing - Mark PageDown Start
            EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXPageDown, (int)VerticalOffset);
        }
 
        /// <summary>
        /// Scroll content by one viewport to the left.
        /// </summary>
        public void PageLeft()
        {
            SetHorizontalOffsetInternal(HorizontalOffset - ViewportWidth);
        }
 
        /// <summary>
        /// Scroll content by one viewport to the right.
        /// </summary>
        public void PageRight()
        {
            SetHorizontalOffsetInternal(HorizontalOffset + ViewportWidth);
        }
 
        /// <summary>
        /// Scroll content up via the mousewheel.
        /// </summary>
        public void MouseWheelUp()
        {
            if (CanMouseWheelVerticallyScroll)
            {
                SetVerticalOffsetInternal(VerticalOffset - MouseWheelVerticalScrollAmount);
            }
            else
            {
                PageUp();
            }
        }
 
        /// <summary>
        /// Scroll content down via the mousewheel.
        /// </summary>
        public void MouseWheelDown()
        {
            if (CanMouseWheelVerticallyScroll)
            {
                SetVerticalOffsetInternal(VerticalOffset + MouseWheelVerticalScrollAmount);
            }
            else
            {
                PageDown();
            }
        }
 
        /// <summary>
        /// Scroll content left via the mousewheel.
        /// </summary>
        public void MouseWheelLeft()
        {
            if (CanMouseWheelHorizontallyScroll)
            {
                SetHorizontalOffsetInternal(HorizontalOffset - MouseWheelHorizontalScrollAmount);
            }
            else
            {
                PageLeft();
            }
        }
 
        /// <summary>
        /// Scroll content right via the mousewheel.
        /// </summary>
        public void MouseWheelRight()
        {
            if (CanMouseWheelHorizontallyScroll)
            {
                SetHorizontalOffsetInternal(HorizontalOffset + MouseWheelHorizontalScrollAmount);
            }
            else
            {
                PageRight();
            }
        }
 
        /// <summary>
        /// Ensures that the specified visual is made visible.
        /// </summary>
        /// <returns>
        /// A rectangle in the IScrollInfo's coordinate space that has been made visible.
        /// Other ancestors to in turn make this new rectangle visible.
        /// The rectangle should generally be a transformed version of the input rectangle.  In some cases, like
        /// when the input rectangle cannot entirely fit in the viewport, the return value might be smaller.
        /// </returns>
        public Rect MakeVisible(Visual v, Rect r)
        {
            if (Content != null && v != null)
            {
                ContentPosition cp = Content.GetObjectPosition(v);
                MakeContentPositionVisibleAsync(new MakeVisibleData(v, cp, r));
            }
 
            return r;
        }
 
        /// <summary>
        /// Ensures that the specified object is made visible, given that the page it lives on is already known.
        /// </summary>
        /// <returns>
        /// A rectangle in the IScrollInfo's coordinate space that has been made visible.
        /// Other ancestors to in turn make this new rectangle visible.
        /// The rectangle should generally be a transformed version of the input rectangle.  In some cases, like
        /// when the input rectangle cannot entirely fit in the viewport, the return value might be smaller.
        /// </returns>
        public Rect MakeVisible(object o, Rect r, int pageNumber)
        {
            ContentPosition cp = Content.GetObjectPosition(o);
            MakeVisibleAsync(new MakeVisibleData(o as Visual, cp, r), pageNumber);
            return r;
        }
 
        /// <summary>
        /// Scrolls the current selection into view.  Requests for empty or
        /// invalid selections will do nothing.
        /// </summary>
        public void MakeSelectionVisible()
        {
            //We can only continue if we have a TextEditor attached...
            if (TextEditor != null && TextEditor.Selection != null)
            {
                //Get the TextPointer for the start of our selection.
                ITextPointer tp = TextEditor.Selection.Start;
 
                //Ensure that the TextPointer we use has gravity set to forwards (or into
                //the selection) so that the selected text is always displayed.
                tp = tp.CreatePointer(LogicalDirection.Forward);
 
                //If the TextPointer is also a ContentPosition, we can
                //make that ContentPosition visible.
                ContentPosition cp = tp as ContentPosition;
                MakeContentPositionVisibleAsync(new MakeVisibleData(null, cp, Rect.Empty));
            }
        }
 
        /// <summary>
        /// Scrolls the requested page into view.
        /// </summary>
        /// <param name="pageNumber">The page to make visible.</param>
        public void MakePageVisible(int pageNumber)
        {
            //If we're moving more than one page then this is a "page jump"
            //and we should log the perf event.
            if (Math.Abs(pageNumber - _firstVisiblePageNumber) > 1)
            {
                //Perf Tracing - Mark Page Jump Start
                EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXPageJump, _firstVisiblePageNumber, pageNumber);
            }
 
            //Clip the offset into range for out-of-range page numbers
            if (pageNumber < 0)
            {
                //Clip to the top-left of the document
                SetVerticalOffsetInternal(0.0d);
                SetHorizontalOffsetInternal(0.0d);
            }
            else if (pageNumber >= _pageCache.PageCount || _rowCache.RowCount == 0)
            {
                //If the doc is done loading, then this page is out of range.
                if (_pageCache.IsPaginationCompleted && _rowCache.HasValidLayout)
                {
                    //Clip to the bottom-right of the document
                    SetVerticalOffsetInternal(ExtentHeight);
                    SetHorizontalOffsetInternal(ExtentWidth);
                }
                else
                {
                    //The doc is not done loading.
                    //Wait for the page to be laid out and try again.
                    _pageJumpAfterLayout = true;
                    _pageJumpAfterLayoutPageNumber = pageNumber;
                }
            }
            else
            {
                //This page is valid, so scroll to it now.
                RowInfo scrolledRow = _rowCache.GetRowForPageNumber(pageNumber);
                SetVerticalOffsetInternal(scrolledRow.VerticalOffset);
 
                //Calculate the Horizontal offset of the page we're bringing into view:
                double horizontalOffset = GetHorizontalOffsetForPage(scrolledRow, pageNumber);
                SetHorizontalOffsetInternal(horizontalOffset);
            }
        }
 
        /// <summary>
        /// Scrolls the next row of pages into view.  This differs from
        /// IScrollInfo?s "PageDown" in that PageDown pages by Viewports
        /// which may not coincide with page dimensions, whereas
        /// ScrollToNextRow takes these dimensions into account so that
        /// precisely the next row of pages is displayed.
        /// </summary>
        public void ScrollToNextRow()
        {
            //We change our vertical offset to be the offset of the next row (if there is one).
            //If there isn't, we do nothing.
            int nextRow = _firstVisibleRow + 1;
 
            if (nextRow < _rowCache.RowCount)
            {
                //Get the next row.
                RowInfo row = _rowCache.GetRow(nextRow);
                SetVerticalOffsetInternal(row.VerticalOffset);
            }
        }
 
        /// Scrolls the previous row of pages into view.  This differs from
        /// IScrollInfo?s "PageUp" in that PageUp pages by Viewports
        /// which may not coincide with page dimensions, whereas
        /// ScrollToPreviousRow takes these dimensions into account so that
        /// precisely the previously row of pages is displayed.
        public void ScrollToPreviousRow()
        {
            //We change our vertical offset to be the offset of the previous row (if there is one).
            int previousRow = _firstVisibleRow - 1;
 
            if (previousRow >= 0 && previousRow < _rowCache.RowCount)
            {
                //Get the previous row.
                RowInfo row = _rowCache.GetRow(previousRow);
                SetVerticalOffsetInternal(row.VerticalOffset);
            }
        }
 
        /// <summary>
        /// Scrolls to the top of the document.
        /// </summary>
        public void ScrollToHome()
        {
            //We just set the VerticalOffset to 0.
            SetVerticalOffsetInternal(0);
        }
 
        /// <summary>
        /// Scrolls to the bottom of the document.
        /// </summary>
        public void ScrollToEnd()
        {
            //We just set the VerticalOffset to our document's extent.
            SetVerticalOffsetInternal(ExtentHeight);
        }
 
        /// <summary>
        /// Sets the scale factor applied to pages in the document, while
        /// keeping the "Active Focus" centered.
        /// </summary>
        /// <param name="scale"></param>
        public void SetScale(double scale)
        {
            if (!DoubleUtil.AreClose(scale, Scale))
            {
                ArgumentOutOfRangeException.ThrowIfLessThanOrEqual(scale, 0.0);
 
                if (!Helper.IsDoubleValid(scale))
                {
                    throw new ArgumentOutOfRangeException("scale");
                }
 
                QueueSetScale(scale);
            }
        }
 
        /// <summary>
        /// Changes the view to the specified number of columns.
        /// </summary>
        /// <param name="columns"></param>
        public void SetColumns(int columns)
        {
            ArgumentOutOfRangeException.ThrowIfLessThan(columns, 1);
 
            //Perf Tracing - Mark Layout Change Start
            EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXLayoutBegin);
 
            QueueUpdateDocumentLayout(new DocumentLayout(columns, ViewMode.SetColumns));
        }
 
        /// <summary>
        /// Changes the view to the specified number of columns.
        /// </summary>
        /// <param name="columns"></param>
        public void FitColumns(int columns)
        {
            ArgumentOutOfRangeException.ThrowIfLessThan(columns, 1);
 
            //Perf Tracing - Mark Layout Change Start
            EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXLayoutBegin);
 
            QueueUpdateDocumentLayout(new DocumentLayout(columns, ViewMode.FitColumns));
        }
 
        /// <summary>
        /// Changes the view to a single page, scaled such that it is as wide as the Viewport.
        /// </summary>
        public void FitToPageWidth()
        {
            //Perf Tracing - Mark Layout Change Start
            EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXLayoutBegin);
 
            QueueUpdateDocumentLayout(
                new DocumentLayout(1 /* one column */, ViewMode.PageWidth));
        }
 
        /// <summary>
        /// Changes the view to a single page, scaled such that it is as tall as the Viewport.
        /// </summary>
        public void FitToPageHeight()
        {
            //Perf Tracing - Mark Layout Change Start
            EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXLayoutBegin);
 
            QueueUpdateDocumentLayout(
                new DocumentLayout(1 /* one column */, ViewMode.PageHeight));
        }
 
        /// <summary>
        /// Changes the view to ?thumbnail view? which will scale the document
        /// such that as many pages are visible at once as is possible.
        /// </summary>
        public void ViewThumbnails()
        {
            //Perf Tracing - Mark Layout Change Start
            EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXLayoutBegin);
 
            QueueUpdateDocumentLayout(
                new DocumentLayout(1 /* one column, arbitrary */, ViewMode.Thumbnails));
        }
 
        //------------------------------------------------------
        //
        //  IDocumentScrollInfo Properties
        //
        //------------------------------------------------------
 
        /// <summary>
        /// DocumentGrid always scrolls in both dimensions.
        /// </summary>
        public bool CanHorizontallyScroll
        {
            get { return _canHorizontallyScroll; }
            set { _canHorizontallyScroll = value; }
        }
 
        /// <summary>
        /// DocumentGrid always scrolls in both dimensions.
        /// </summary>
        public bool CanVerticallyScroll
        {
            get { return _canVerticallyScroll; }
            set { _canVerticallyScroll = value; }
        }
 
        /// <summary>
        /// ExtentWidth contains the full horizontal range of the scrolled content.
        /// </summary>
        public double ExtentWidth
        {
            get
            {
                return _rowCache.ExtentWidth;
            }
        }
 
        /// <summary>
        /// ExtentHeight contains the full vertical range of the scrolled content.
        /// </summary>
        public double ExtentHeight
        {
            get
            {
                return _rowCache.ExtentHeight;
            }
        }
 
        /// <summary>
        /// ViewportWidth contains the currently visible horizontal range of the scrolled content.
        /// </summary>
        public double ViewportWidth
        {
            get
            {
                return _viewportWidth;
            }
        }
 
        /// <summary>
        /// ViewportHeight contains the currently visible vertical range of the scrolled content.
        /// </summary>
        public double ViewportHeight
        {
            get
            {
                return _viewportHeight;
            }
        }
 
        /// <summary>
        /// HorizontalOffset is the horizontal offset into the scrolled content that represents the first unit visible.
        /// Valid values are inclusively between 0 and <see cref="ExtentWidth" /> less <see cref="ViewportWidth" />.
        /// </summary>
        public double HorizontalOffset
        {
            get
            {
                //Clip HorizontalOffset into range.
                double clippedHorizontalOffset = Math.Min(_horizontalOffset, ExtentWidth - ViewportWidth);
                clippedHorizontalOffset = Math.Max(clippedHorizontalOffset, 0.0);
 
                return clippedHorizontalOffset;
            }
        }
 
        /// <summary>
        /// Set the HorizontalOffset.  If there are pending layout delegates, then
        /// this will be processed by a delegate.
        /// </summary>
        /// <param name="offset"></param>
        public void SetHorizontalOffset(double offset)
        {
            if (!DoubleUtil.AreClose(_horizontalOffset, offset))
            {
                if (Double.IsNaN(offset))
                {
                    throw new ArgumentOutOfRangeException("offset");
                }
 
                // If there aren't any pending document layout delegates, then change
                // the HorizontalOffset immediately, otherwise schedule a delegate for it.
                if (_documentLayoutsPending == 0)
                {
                    SetHorizontalOffsetInternal(offset);
                }
                else
                {
                    QueueUpdateDocumentLayout(new DocumentLayout(offset, ViewMode.SetHorizontalOffset));
                }
            }
        }
 
        /// <summary>
        /// VerticalOffset is the vertical offset into the scrolled content that represents the first unit visible.
        /// Valid values are inclusively between 0 and <see cref="ExtentHeight" /> less <see cref="ViewportHeight" />.
        /// </summary>
        public double VerticalOffset
        {
            get
            {
                //Clip VerticalOffset into range.
                double clippedVerticalOffset = Math.Min(_verticalOffset, ExtentHeight - ViewportHeight);
                clippedVerticalOffset = Math.Max(clippedVerticalOffset, 0.0);
 
                return clippedVerticalOffset;
            }
        }
 
        /// <summary>
        /// Set the VerticalOffset.  If there are pending layout delegates, then
        /// this will be processed by a delegate.
        /// </summary>
        /// <param name="offset"></param>
        public void SetVerticalOffset(double offset)
        {
            if (!DoubleUtil.AreClose(_verticalOffset, offset))
            {
                if (Double.IsNaN(offset))
                {
                    throw new ArgumentOutOfRangeException("offset");
                }
 
                // If there aren't any pending document layout delegates, then change
                // the VerticalOffset immediately, otherwise schedule a delegate for it.
                if (_documentLayoutsPending == 0)
                {
                    SetVerticalOffsetInternal(offset);
                }
                else
                {
                    QueueUpdateDocumentLayout(new DocumentLayout(offset, ViewMode.SetVerticalOffset));
                }
            }
        }
 
        /// <summary>
        /// Provides the IDocumentScrollInfo implementer with a content
        /// tree to be paginated.  Developers are free to modify this
        /// Content at any time (remove, add, modify pages, etc?)
        /// and the IDocumentScrollInfo implementer is responsible for
        /// noting the changes and updating as necessary.
        /// </summary>
        /// <value>The DocumentPaginator to be assigned as the content</value>
        public DynamicDocumentPaginator Content
        {
            get
            {
                //_pageCache is guaranteed to be non-null as it's created in the
                //Constructor.
                return _pageCache.Content;
            }
            set
            {
                //_pageCache is guaranteed to be non-null as it's created in the
                //Constructor.
                if (value != _pageCache.Content)
                {
                    //Null out our TextContainer.  It will be created as needed.
                    _textContainer = null;
 
                    //Remove our old events from the content
                    if (_pageCache.Content != null)
                    {
                        _pageCache.Content.GetPageNumberCompleted -= new GetPageNumberCompletedEventHandler(OnGetPageNumberCompleted);
                    }
 
                    //Remove our ScrollChanged events from our ScrollViewer
                    if (ScrollOwner != null)
                    {
                        ScrollOwner.ScrollChanged -= new ScrollChangedEventHandler(OnScrollChanged);
                        _scrollChangedEventAttached = false;
                    }
 
                    //Assign the new content
                    _pageCache.Content = value;
 
                    if (_pageCache.Content != null)
                    {
                        //Add our new events to the content
                        _pageCache.Content.GetPageNumberCompleted += new GetPageNumberCompletedEventHandler(OnGetPageNumberCompleted);
                    }
 
                    //Clear out our visual collection so that the old pages (pointing to old content)
                    //will be replaced with new ones on the next Measure/Arrange pass.
                    ResetVisualTree(false /*pruneOnly*/);
                    ResetPageViewCollection();
 
                    //Reset our visible pages.
                    _firstVisiblePageNumber = 0;
                    _lastVisiblePageNumber = 0;
 
                    // Perf Tracing - PageVisible Changed
                    EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXPageVisible, _firstVisiblePageNumber, _lastVisiblePageNumber);
 
                    _lastRowChangeExtentWidth = 0.0;
                    _lastRowChangeVerticalOffset = 0.0;
 
                    //Cause the new content to be laid out in the same fashion as
                    //the previous content.
                    //If the view is Thumbnails we'll change it to SetColumns
                    //so that the Column count will be maintained.  This is done
                    //because we're getting new content which initially has 0
                    //pages and a Thumbnail view only gives decent results after
                    //the entire content has been loaded; since we don't want to provide a
                    //"jarring" situation (where layout suddenly changes after the content's loaded)
                    //we use SetColumns to maintain the same exact layout as the old content.
                    //(This is consistent with DocumentViewer's overall behavior -- any view setting is
                    //a "one time thing" and isn't recomputed if the content changes.)
                    if (_documentLayout.ViewMode == ViewMode.Thumbnails)
                    {
                        _documentLayout.ViewMode = ViewMode.SetColumns;
                    }
                    QueueUpdateDocumentLayout(_documentLayout);
 
                    //Invalidate Measure and our IDSI so that properties changed
                    //by the content assignment will be properly updated.
                    InvalidateMeasure();
                    InvalidateDocumentScrollInfo();
                }
            }
        }
 
        /// <summary>
        /// Indicates the number of pages currently in the document.
        /// </summary>
        /// <value></value>
        public int PageCount
        {
            get
            {
                return _pageCache.PageCount;
            }
        }
 
        /// <summary>
        /// When queried, FirstVisiblePageNumber returns the first page visible onscreen.
        /// </summary>
        /// <value></value>
        public int FirstVisiblePageNumber
        {
            get
            {
                return _firstVisiblePageNumber;
            }
        }
 
        /// <summary>
        /// Returns the current Scale factor applied to the pages given the current settings.
        /// </summary>
        /// <value></value>
        public double Scale
        {
            get
            {
                return _rowCache.Scale;
            }
        }
 
        /// <summary>
        /// Returns the current number of Columns of pages displayed given the current settings.
        /// </summary>
        /// <value></value>
        public int MaxPagesAcross
        {
            get
            {
                return _maxPagesAcross;
            }
        }
 
        /// <summary>
        /// Specifies the vertical gap between Pages when laid out, in pixel (1/96?) units.
        /// </summary>
        /// <value></value>
        public double VerticalPageSpacing
        {
            get
            {
                return _rowCache.VerticalPageSpacing;
            }
 
            set
            {
                if (!Helper.IsDoubleValid(value))
                {
                    throw new ArgumentOutOfRangeException("value");
                }
 
                _rowCache.VerticalPageSpacing = value;
            }
        }
 
        /// <summary>
        /// Specifies the horizontal gap between Pages when laid out, in pixel (1/96?) units.
        /// </summary>
        /// <value></value>
        public double HorizontalPageSpacing
        {
            get
            {
                return _rowCache.HorizontalPageSpacing;
            }
 
            set
            {
                if (!Helper.IsDoubleValid(value))
                {
                    throw new ArgumentOutOfRangeException("value");
                }
 
                _rowCache.HorizontalPageSpacing = value;
            }
        }
 
        /// <summary>
        /// Specifies whether each displayed page should be adorned with a ?Drop Shadow? border or not.
        /// </summary>
        /// <value></value>
        public bool ShowPageBorders
        {
            get
            {
                return _showPageBorders;
            }
 
            set
            {
                if (_showPageBorders != value)
                {
                    _showPageBorders = value;
 
                    //Update our pages' ShowPageBorder properties.
                    //Get the current Visual Collection, which contains our pages.
                    int count = _childrenCollection.Count;
                    for (int i = 0; i < count; i++)
                    {
                        DocumentGridPage dp = _childrenCollection[i] as DocumentGridPage;
 
                        if (dp != null)
                        {
                            dp.ShowPageBorders = _showPageBorders;
                        }
                    }
                }
            }
        }
 
        /// <summary>
        /// Specifies whether the last "view mode" related property change should be locked
        /// for resizing.
        /// </summary>
        /// <value></value>
        public bool LockViewModes
        {
            get
            {
                return _lockViewModes;
            }
 
            set
            {
                _lockViewModes = value;
            }
        }
 
        /// <summary>
        /// Returns a TextContainer for current content
        /// </summary>
        /// <returns>The content's TextContainer, or null if there is none.</returns>
        public ITextContainer TextContainer
        {
            get
            {
                if (_textContainer == null)
                {
                    if (Content != null)
                    {
                        IServiceProvider isp = Content as IServiceProvider;
                        if (isp != null)
                        {
                            _textContainer = (ITextContainer)isp.GetService(typeof(ITextContainer));
                        }
                    }
                }
 
                return _textContainer;
            }
        }
 
        /// <summary>
        /// Returns the MultiPageTextView for the current content.
        /// </summary>
        /// <value></value>
        public ITextView TextView
        {
            get
            {
                if (TextEditor != null)
                {
                    return TextEditor.TextView;
                }
                else
                {
                    return null;
                }
            }
        }
 
        /// <summary>
        /// The collection of currently-visible DocumentPageViews.
        /// </summary>
        public ReadOnlyCollection<DocumentPageView> PageViews
        {
            get
            {
                return _pageViews;
            }
        }
 
        /// <summary>
        /// ScrollOwner is the container that controls any scrollbars, headers, etc... that are dependent
        /// on this IScrollInfo's properties.  Implementers of IScrollInfo should call InvalidateScrollInfo()
        /// on this object when related properties change.
        /// </summary>
        public ScrollViewer ScrollOwner
        {
            get
            {
                return _scrollOwner;
            }
 
            set
            {
                _scrollOwner = value;
                InvalidateDocumentScrollInfo();
            }
        }
 
        /// <summary>
        /// DocumentViewerOwner is the DocumentViewer Control and UI that hosts the IDocumentScrollInfo object.
        /// This control is dependent on this IDSI?s properties, so implementers of IDSI should call
        /// InvalidateDocumentScrollInfo() on this object when related properties change so that
        /// DocumentViewer?s UI will be kept in sync.  This property is analogous to IScrollInfo?s ScrollOwner
        /// property.
        /// </summary>
        /// <value></value>
        public DocumentViewer DocumentViewerOwner
        {
            get
            {
                return _documentViewerOwner;
            }
 
            set
            {
                _documentViewerOwner = value;
            }
        }
 
        #endregion IDocumentScrollInfo
 
        #endregion Interface Implementations
 
        //------------------------------------------------------
        //
        //  Protected Methods
        //
        //------------------------------------------------------
        #region Protected Methods
 
        /// <summary>
        ///   Derived class must implement to support Visual children. The method must return
        ///    the child at the specified index. Index must be between 0 and GetVisualChildrenCount-1.
        ///
        ///    By default a Visual does not have any children.
        ///
        ///  Remark:
        ///       During this virtual call it is not valid to modify the Visual tree.
        /// </summary>
        protected override Visual GetVisualChild(int index)
        {
            if (_childrenCollection == null || index < 0 || index >= _childrenCollection.Count)
            {
                throw new ArgumentOutOfRangeException("index", index, SR.Visual_ArgumentOutOfRange);
            }
 
            return _childrenCollection[index];
        }
 
        /// <summary>
        ///  Derived classes override this property to enable the Visual code to enumerate
        ///  the Visual children. Derived classes need to return the number of children
        ///  from this method.
        ///
        ///    By default a Visual does not have any children.
        ///
        ///  Remark: During this virtual method the Visual tree must not be modified.
        /// </summary>
        protected override int VisualChildrenCount
        {
            get
            {
                // _childrenCollection cannot be null since its initialized in the constructor
                return _childrenCollection.Count;
            }
        }
 
        /// <summary>
        /// MeasureOverride is repsonsible for measuring any visible pages to their correct sizes.
        /// </summary>
        /// <param name="constraint">The upper bound for child sizes</param>
        /// <returns></returns>
        protected override Size MeasureOverride(Size constraint)
        {
            // If layoutSize is infinity, we need to return our absolute smallest size.
            // This might happen if we are inside an element which sizes-to-content.
            // For DocumentGrid, we use a hard coded constraint.
            if (double.IsInfinity(constraint.Width) || double.IsInfinity(constraint.Height))
            {
                constraint = _defaultConstraint;
            }
 
            //Determine which pages are visible at the current offset given the current constraint.
            RecalculateVisualPages(VerticalOffset, constraint);
 
            //Get our visual children count...
            int count = _childrenCollection.Count;
 
            //Now go through our child collection and measure all the pages to their sizes.
            for (int i = 0; i < count; i++)
            {
                //This should be our background.
                if (i == _backgroundVisualIndex)
                {
                    Border background = _childrenCollection[i] as Border;
 
                    if (background == null)
                    {
                        throw new InvalidOperationException(SR.DocumentGridVisualTreeContainsNonBorderAsFirstElement);
                    }
 
                    //We measure this to the size of our constraint.
                    background.Measure(constraint);
                }
                //Otherwise it's a page.
                else
                {
                    //Ensure that this is actually a DocumentGridPage.  If it is not,
                    //Then someone's been mucking with our VisualTree, so we'll throw.
                    DocumentGridPage page = _childrenCollection[i] as DocumentGridPage;
 
                    if (page == null)
                    {
                        throw new InvalidOperationException(SR.DocumentGridVisualTreeContainsNonDocumentGridPage);
                    }
 
                    //Get the cached size of this page and scale it to our current scale factor.
                    Size pageSize = _pageCache.GetPageSize(page.PageNumber);
                    pageSize.Width *= Scale;
                    pageSize.Height *= Scale;
 
                    //Measure the page if necessary.
                    if (!page.IsMeasureValid)
                    {
                        page.Measure(pageSize);
 
                        //See if the cached size has changed since we Measured.
                        //This can happen if in the course of Measuring the page
                        //a GetPageAsync() calls back immediately with the real page size.
                        //If this happens we need to re-measure the page before we finish here,
                        //otherwise we'll end up with a page that's Measured to one size
                        //and Arranged to another, which looks bad.
                        Size newPageSize = _pageCache.GetPageSize(page.PageNumber);
                        if (newPageSize != Size.Empty)
                        {
                            newPageSize.Width *= Scale;
                            newPageSize.Height *= Scale;
                            if (newPageSize.Width != pageSize.Width ||
                            newPageSize.Height != pageSize.Height)
                            {
                                //Measure again.
                                page.Measure(newPageSize);
                            }
                        }
                    }
                }
            }
 
            return constraint;
        }
 
        /// <summary>
        /// ArrangeOverride is responsible for arranging the previously measured pages in the right order.
        /// </summary>
        /// <param name="arrangeSize">The final constraint, inside of which everything must live.</param>
        /// <returns></returns>
        protected override Size ArrangeOverride(Size arrangeSize)
        {
            if (_viewportHeight != arrangeSize.Height ||
                _viewportWidth != arrangeSize.Width)
            {
                //Update our Viewport sizes
                _viewportWidth = arrangeSize.Width;
                _viewportHeight = arrangeSize.Height;
 
                if (LockViewModes && IsViewLoaded())
                {
                    if (_firstVisiblePageNumber < _pageCache.PageCount && _rowCache.HasValidLayout)
                    {
                        //If we're locking the view modes and we have loaded content, then we need to re-apply
                        //the last mode setting since our constraint has changed
                        ApplyViewParameters(_rowCache.GetRowForPageNumber(_firstVisiblePageNumber));
                        MeasureOverride(arrangeSize);
                    }
                }
 
                UpdateTextView();
            }
 
            //If we have a non-zero viewport size, we should execute any
            //requests for layout that may have been made but were unable
            //to complete due to a zero viewport size.
            if (IsViewportNonzero)
            {
                if (ExecutePendingLayoutRequests())
                {
                    //We need to re-do layout (RowCache has changed), so call measure here
                    //to ensure everything's updated accordingly.
                    MeasureOverride(arrangeSize);
                }
            }
 
            //If our constraint size has changed then we need to
            //alert our parents so they can update their ViewportWidth/Height properties.
            if (_previousConstraint != arrangeSize)
            {
                _previousConstraint = arrangeSize;
                InvalidateDocumentScrollInfo();
            }
 
            //Now we go through the visible rows and arrange the pages within them.
            //Get our visual collection count
            int count = _childrenCollection.Count;
 
            //If we have no visual children, there's nothing to arrange
            //so quit now.
            if (count == 0)
            {
                return arrangeSize;
            }
 
            //Arrange the background first.  This is always child 0.
            //The background takes up the entire constraint.
            UIElement background = _childrenCollection[_backgroundVisualIndex] as UIElement;
            background.Arrange(new Rect(new Point(0, 0), arrangeSize));
 
            //The offsets for the current page being arranged.
            double xOffset;
            double yOffset;
 
            //The current visual child (aka DocumentGridPage) we're arranging.
            //The first child in our tree is always the background so we start at
            //1 which is our first page.
            int visualChild = _firstPageVisualIndex;
 
            //Now walk through the visible rows and arrange the pages therein.
            for (int row = _firstVisibleRow; row < _firstVisibleRow + _visibleRowCount; row++)
            {
                //Calculate the position for this row.
                CalculateRowOffsets(row, out xOffset, out yOffset);
 
                //Get the current row.
                RowInfo currentRow = _rowCache.GetRow(row);
 
                //Now we can lay out this row.
                for (int page = currentRow.FirstPage; page < currentRow.FirstPage + currentRow.PageCount; page++)
                {
                    //This should never, ever happen so we'll throw if it does.
                    if (visualChild > _childrenCollection.Count - 1)
                    {
                        throw new InvalidOperationException(SR.DocumentGridVisualTreeOutOfSync);
                    }
 
                    //Get the cached size of this page.
                    Size pageSize = _pageCache.GetPageSize(page);
 
                    //Scale it by our scale factor
                    pageSize.Width *= Scale;
                    pageSize.Height *= Scale;
 
                    //Arrange the page if necessary
                    UIElement uiPage = _childrenCollection[visualChild] as UIElement;
                    if (uiPage != null)
                    {
                        Point pageOffset;
                        //Move the page to the right place based on the FlowDirection of the content.
                        if (_pageCache.IsContentRightToLeft)
                        {
                            pageOffset = new Point(Math.Max(ViewportWidth, ExtentWidth) - (xOffset + pageSize.Width), yOffset);
                        }
                        else
                        {
                            pageOffset = new Point(xOffset, yOffset);
                        }
                        uiPage.Arrange(new Rect(pageOffset, pageSize));
                    }
                    else
                    {
                        throw new InvalidOperationException(SR.DocumentGridVisualTreeContainsNonUIElement);
                    }
 
                    //Increment our horizontal offset to point to where the next page should go.
                    xOffset += (pageSize.Width + HorizontalPageSpacing);
 
                    //Move to the next page.
                    visualChild++;
                }
            }
 
            // As we scroll we need to keep the AdornerLayer up-to-date.
            // This ensures that annotation components scroll with the content.
            AdornerLayer layer = AdornerLayer.GetAdornerLayer(this);
            if (layer != null && layer.GetAdorners(this) != null)
                layer.Update(this);
 
            return arrangeSize;
        }
 
        /// <summary>
        /// Override the OnPreviewMouseLeftButtonDown method so that we can trap the
        /// keyboard+mouse events needed for Rubberband selection.
        /// Clicking the Left mouse button while holding Alt will enable the Rubberband
        /// selection "mode" until the Left mouse button is again pressed without the
        /// Alt key held.
        /// </summary>
        /// <param name="e">The MouseButtonEventArgs associated with this mouse event.</param>
        protected override void OnPreviewMouseLeftButtonDown(MouseButtonEventArgs e)
        {
            //Determine whether either Alt key is being held at this moment.
            bool altKeyDown = Keyboard.IsKeyDown(Key.LeftAlt) || Keyboard.IsKeyDown(Key.RightAlt);
 
            //If the Alt key is held and we aren't currently in RubberBandSelection mode,
            //we can create and attach our RubberBandSelector now.
            //We'll stay in this mode until the mouse is clicked without the Alt key held.
            if (altKeyDown && _rubberBandSelector == null)
            {
                //See if our content implements IServiceProvider.
                IServiceProvider serviceProvider = Content as IServiceProvider;
                if (serviceProvider != null)
                {
                    //See if our content supports rubber band selection.
                    _rubberBandSelector = serviceProvider.GetService(typeof(RubberbandSelector)) as RubberbandSelector;
 
                    if (_rubberBandSelector != null)
                    {
                        DocumentViewerOwner.Focus(); // text editor needs to be focused when cleared
                        ITextRange textRange = TextEditor.Selection;
                        textRange.Select(textRange.Start, textRange.Start); //clear selection
                        DocumentViewerOwner.IsSelectionEnabled = false;
 
                        _rubberBandSelector.AttachRubberbandSelector((FrameworkElement)this); //attach the Rubber band selector.
                    }
                }
            }
            //We got a mouse-down event and the Alt key is not being held, so we revert back
            //to normal selection mode now.
            else if (!altKeyDown && _rubberBandSelector != null)
            {
                //Detach the Rubberband Selector
                if (_rubberBandSelector != null)
                {
                    _rubberBandSelector.DetachRubberbandSelector();
                    _rubberBandSelector = null;
                }
 
                DocumentViewerOwner.IsSelectionEnabled = true;
            }
        }
 
        /// <summary>
        ///
        /// Reset the entire visual tree when the visual parent changes in order to ensure that
        /// the HighlightVisuals associated with the FixedPages in the document are re-created
        /// and added to the new AdornerLayer.
        /// </summary>
        /// <param name="oldParent">The old visual parent (not used)</param>
        protected internal override void OnVisualParentChanged(DependencyObject oldParent)
        {
            base.OnVisualParentChanged(oldParent);
 
            // No need for a reset if we don't have a parent since there is no AdornerLayer
            // to add HighlightVisuals back to.
            if (VisualTreeHelper.GetParent(this) != null)
            {
                // Do a full reset, we want to ensure even visible pages are reset.
                ResetVisualTree(pruneOnly: false);
            }
        }
 
        #endregion Protected Methods
 
        //------------------------------------------------------
        //
        //  Private Methods
        //
        //------------------------------------------------------
        #region Private Methods
 
        /// <summary>
        /// Recalculates the set of pages that are currently visible and updates
        /// DocumentGrid's VisualCollection so it contains them.
        /// </summary>
        /// <param name="constraint">The viewport to search for visible pages in.</param>
        /// <param name="offset">The offset in the document to start the search.</param>
        private void RecalculateVisualPages(double offset, Size constraint)
        {
            //Do we actually have any rows in the cache?
            //If not, we can just clear our visual collection and return.
            if (_rowCache.RowCount == 0)
            {
                ResetVisualTree(false /*pruneOnly*/);
                ResetPageViewCollection();
                _firstVisibleRow = 0;
                _visibleRowCount = 0;
                _firstVisiblePageNumber = 0;
                _lastVisiblePageNumber = 0;
 
                // Perf Tracing - PageVisible Changed
                EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXPageVisible, _firstVisiblePageNumber, _lastVisiblePageNumber);
 
                return;
            }
 
            int newFirstVisibleRow = 0;
            int newVisibleRowCount = 0;
 
            //Ask the RowCache for the currently visible rows.
            _rowCache.GetVisibleRowIndices(offset,
                                            offset + constraint.Height,
                                            out newFirstVisibleRow,
                                            out newVisibleRowCount);
 
            //Do we have no visible rows at all?  Then clear the Visual collection and return.
            if (newVisibleRowCount == 0)
            {
                ResetVisualTree(false /*pruneOnly*/);
                ResetPageViewCollection();
                _firstVisibleRow = 0;
                _visibleRowCount = 0;
                _firstVisiblePageNumber = 0;
                _lastVisiblePageNumber = 0;
 
                // Perf Tracing - PageVisible Changed
                EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXPageVisible, _firstVisiblePageNumber, _lastVisiblePageNumber);
 
                return;
            }
 
            //Now walk through each visible row and compare the pages therein
            //with the current set of pages in our Visual Collection.
            //New pages are inserted into the collection, unused pages are removed.
 
            //Get the current first and last pages visible.
            int firstPage = -1;
            int lastPage = -1;
 
            //If we have more visuals than just the background (element 0)
            //then we have pages, so get the page numbers from them.
            if (_childrenCollection.Count > _firstPageVisualIndex)
            {
                DocumentGridPage firstDp = _childrenCollection[1] as DocumentGridPage;
                firstPage = firstDp != null ? firstDp.PageNumber : -1;
 
                DocumentGridPage lastDp = _childrenCollection[_childrenCollection.Count - 1] as DocumentGridPage;
                lastPage = lastDp != null ? lastDp.PageNumber : -1;
            }
 
 
            //Update our First & LastVisiblePage properties
            RowInfo firstRow = _rowCache.GetRow(newFirstVisibleRow);
            _firstVisiblePageNumber = firstRow.FirstPage;
            RowInfo lastRow = _rowCache.GetRow(newFirstVisibleRow + newVisibleRowCount - 1);
            _lastVisiblePageNumber = lastRow.FirstPage + lastRow.PageCount - 1;
 
            // Perf Tracing - PageVisible Changed
            EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXPageVisible, _firstVisiblePageNumber, _lastVisiblePageNumber);
 
            //Update our cached visible row info (used by Measure/Arrange)
            _firstVisibleRow = newFirstVisibleRow;
            _visibleRowCount = newVisibleRowCount;
 
 
            //If IDSI properties have changed (namely the First/LastVisiblePage properties) we invalidate them now.
            if (_firstVisiblePageNumber != firstPage ||
                _lastVisiblePageNumber != lastPage)
            {
                //Create our temporary VisualCollection, which will hold the new list of
                //visible pages.
                ArrayList visiblePages = new ArrayList();
 
                //Now walk through the visible rows and add the pages to our temporary list.
                for (int i = _firstVisibleRow; i < _firstVisibleRow + _visibleRowCount; i++)
                {
                    //Get the row
                    RowInfo currentRow = _rowCache.GetRow(i);
 
                    //Walk through the row
                    for (int j = currentRow.FirstPage; j < currentRow.FirstPage + currentRow.PageCount; j++)
                    {
                        //Is this page new?
                        if (j < firstPage || j > lastPage || _childrenCollection.Count <= _firstPageVisualIndex)
                        {
                            //Create a new page and add it to our temporary visual collection.
                            DocumentGridPage dp = new DocumentGridPage(Content)
                            {
                                ShowPageBorders = ShowPageBorders,
                                PageNumber = j
                            };
 
                            //Attach the Loaded event handler
                            dp.PageLoaded += new EventHandler(OnPageLoaded);
                            visiblePages.Add(dp);
                        }
                        else
                        {
                            //This page already exists in our visual collection, so we copy that entry over
                            //from the visual collection instead of creating a new page.
                            //(We start at 1 to skip over the background visual.)
                            visiblePages.Add(_childrenCollection[_firstPageVisualIndex + j - Math.Max(0, firstPage)]);
                        }
                    }
                }
 
                //Copy our new visible page collection over to the VisualCollection and update
                //the MultiPageTextView's list of visible DocumentPageViews.
                //First, prune our visual tree so it only contains the set of pages that are visible
                //before and after the layout change.
                ResetVisualTree(true /*pruneOnly*/);
 
                Collection<DocumentPageView> documentPageViews =
                    new Collection<DocumentPageView>();
 
                //State machine for updating visual collection without removing still-visible pages
                //from the collection:
                //We walk through the set of visible pages that we computed above.
                //We insert new pages before existing pages, and add pages after existing pages.
                //
                //We take advantage of the fact that both the current set of pages in the Visual Collection
                //and the set of to-be-made visible pages in visiblePages are in strictly increasing order
                //with no gaps between pages.
                //To better understand how the below works, refer to Fig. A below:
                //
                //         +-----------------------------+
                //  +------+                             +------+
                //  | new  |  Pruned Visual Collection   |  new |
                //  +------+   (Existing Page Visuals)   +------+
                //         +-----------------------------+
                //  ^      ^                             ^
                //  |      |                             |
                //  A      B                             C
                //
                //                                 [Fig. A: Diagram of states]
                //
                //- The routine starts off in state A (BeforeExisting).
                //  At this point we insert any new pages in the visiblePages collection until we
                //  find a page in the visiblePages collection that is also in the Pruned Visual Collection.
                //  This indicates that the set of common unchanged pages has been reached.
                //  The state machine then transitions to state B (DuringExisting).
                //- The routine stays in B merely iterating through visiblePages until it finds a page
                //  in visiblePages that does not correspond to a page in the Visual Tree.  This indicates
                //  that the end of the set of common unchanged pages has been reached; at this point we
                //  add the new page and transition to state C (AfterExisting).
                //- State C ends when no more pages are left in visiblePages.
                VisualTreeModificationState state = VisualTreeModificationState.BeforeExisting;
 
                //The index pointing to the first common page still in the visual tree after the above pruning.
                int vcIndex = _firstPageVisualIndex;
 
                for (int i = 0; i < visiblePages.Count; i++)
                {
                    Visual current = (Visual)visiblePages[i];
 
                    switch (state)
                    {
                        case VisualTreeModificationState.BeforeExisting:
                            //Keep inserting until we find a page that already exists
                            if (vcIndex < _childrenCollection.Count && _childrenCollection[vcIndex] == current)
                            {
                                //Move to "During" state
                                state = VisualTreeModificationState.DuringExisting;
                            }
                            else
                            {
                                //Insert this page at the current index.
                                _childrenCollection.Insert(vcIndex, current);
                            }
                            //Increment the index into the Visual collection to ensure that it continues
                            //to point to the first common page.
                            vcIndex++;
                            break;
 
                        case VisualTreeModificationState.DuringExisting:
                            //Leave the visual collection alone until we find a page that isn't in the collection
                            //or run out of pages in the collection.
                            if (vcIndex >= _childrenCollection.Count || _childrenCollection[vcIndex] != current)
                            {
                                //Move to "After" state
                                state = VisualTreeModificationState.AfterExisting;
                                //Append this page to the end.
                                _childrenCollection.Add(current);
                            }
                            //Keep moving through the Visual collection...
                            vcIndex++;
                            break;
 
                        case VisualTreeModificationState.AfterExisting:
                            //Keep going until the end.
                            _childrenCollection.Add(current);
                            break;
                    }
 
                    //Add this to the collection of PageViews.
                    documentPageViews.Add(((DocumentGridPage)visiblePages[i]).DocumentPageView);
                }
 
                //Update our collection of PageViews with the current set.
                _pageViews = new ReadOnlyCollection<DocumentPageView>(documentPageViews);
 
                //Tell our parent DocumentViewer that we've updated our PageView collection.
                InvalidatePageViews();
                InvalidateDocumentScrollInfo();
            }
        }
 
        /// <summary>
        /// Handles the PageLoaded event for a given page, and kicks off
        /// BringIntoView actions where necessary.
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="args"></param>
        private void OnPageLoaded(object sender, EventArgs args)
        {
            DocumentGridPage page = sender as DocumentGridPage;
            Invariant.Assert(page != null, "Invalid sender for OnPageLoaded event.");
 
            //Detach the event handler, we don't need this event any longer.
            page.PageLoaded -= new EventHandler(OnPageLoaded);
 
            //Is there a MakeVisible operation waiting for this page to be loaded?
            //If so, invoke its dispatcher in the background.
            if (_makeVisiblePageNeeded == page.PageNumber)
            {
                _makeVisiblePageNeeded = -1;
                _makeVisibleDispatcher.Priority = DispatcherPriority.Background;
            }
 
            // Perf Tracing - PageLoaded
            if (EventTrace.IsEnabled(EventTrace.Keyword.KeywordXPS, EventTrace.Level.Info))
            {
                EventTrace.EventProvider.TraceEvent(EventTrace.Event.WClientDRXPageLoaded, EventTrace.Keyword.KeywordXPS, EventTrace.Level.Info, page.PageNumber);
            }
        }
 
 
        /// <summary>
        /// Calculates the X and Y offsets of the given row based on the current
        /// Viewport dimensions.
        /// </summary>
        /// <param name="row">The row to calculate the offsets of</param>
        /// <param name="xOffset">The X offset of the row</param>
        /// <param name="yOffset">The Y offset of the row</param>
        private void CalculateRowOffsets(int row, out double xOffset, out double yOffset)
        {
            xOffset = 0.0;
            yOffset = 0.0;
 
            //Get the current row.
            RowInfo currentRow = _rowCache.GetRow(row);
 
            //Figure out the width we'll use to center the pages.
            //If the ViewportWidth is wider than the document, we use that.  Otherwise we center
            //the content based on the width of the document.
            double centerWidth = Math.Max(ViewportWidth, ExtentWidth);
 
            //Figure out the offset of the upper left corner of this row.
 
            //X Coordinate:
            //If this is the last row in the document and we're viewing
            //uniformly-sized pages then this row is
            //always left-aligned (not centered).
            if (row == _rowCache.RowCount - 1 && !_pageCache.DynamicPageSizes)
            {
                //This is the last row, so we arrange it such that the left edge of the
                //page is flush with the left edge of the document.
                xOffset = (centerWidth - ExtentWidth) / 2.0 +
                    (HorizontalPageSpacing / 2.0) - HorizontalOffset;
            }
            else
            {
                //Otherwise we center this page inside the document.
                xOffset = (centerWidth - currentRow.RowSize.Width) / 2.0 +
                    (HorizontalPageSpacing / 2.0) - HorizontalOffset;
            }
 
            //Y Coordinate:
            if (ExtentHeight > ViewportHeight)
            {
                //The document is taller than the viewport, so we just display
                //the content at the current offset.
                yOffset = currentRow.VerticalOffset +
                    (VerticalPageSpacing / 2.0) - VerticalOffset;
            }
            else
            {
                //If the document is shorter than the Viewport we're showing it in,
                //we center it vertically within the viewport.  We do not need to factor in
                //VerticalOffset as it is always 0.0 in this scenario.
                yOffset = currentRow.VerticalOffset +
                    (ViewportHeight - ExtentHeight) / 2.0 + (VerticalPageSpacing / 2.0);
            }
        }
 
        /// <summary>
        /// Resets DocumentGrid's visual tree to its initial state or prunes non-visible pages.
        /// This is empty except for a border which acts as a background.
        /// </summary>
        /// <param name="pruneOnly">Whether to clear all pages, or only those that are not visible.</param>
        private void ResetVisualTree(bool pruneOnly)
        {
            //We need to dispose and remove any pages that will no longer be in the visual tree.
            for (int i = _childrenCollection.Count - 1; i >= _firstPageVisualIndex; i--)
            {
                DocumentGridPage dp = _childrenCollection[i] as DocumentGridPage;
                if (dp != null &&
                        (!pruneOnly ||
                        _rowCache.RowCount == 0 ||
                        dp.PageNumber < _firstVisiblePageNumber ||
                        dp.PageNumber > _lastVisiblePageNumber))
                {
                    //This page will not be visible any longer, so get rid of it.
                    //Remove this page from the Visual tree.
                    _childrenCollection.Remove(dp);
 
                    //Remove any PageLoaded event handlers
                    dp.PageLoaded -= new EventHandler(OnPageLoaded);
 
                    //Dispose of the page.
                    ((IDisposable)dp).Dispose();
                }
            }
 
            //Create the background if it does not exist.
            if (_documentGridBackground == null)
            {
                //We create a Border with a transparent background so that it can
                //participate in Hit-Testing (which allows click events like those
                //for our Context Menu to work).
                _documentGridBackground = new Border
                {
                    Background = Brushes.Transparent
                };
 
                //Add the background in.
                _childrenCollection.Add(_documentGridBackground);
            }
        }
 
        /// <summary>
        /// Nulls out the PageViews collection and notifies DocumentViewer of the change.
        /// </summary>
        private void ResetPageViewCollection()
        {
            //Null out our collection of PageViews.
            _pageViews = null;
 
            //Tell our parent DocumentViewer that we've updated our PageView collection.
            InvalidatePageViews();
        }
 
        #region MakeVisible Helpers
 
        /// <summary>
        /// Handles the GetPageNumberCompleted event fired as a result of a MakeContentVisibleAsync
        /// call.  At this point we know the page number corresponding to the ContentPosition we need
        /// to make visible, so we invoke MakeVisibleAsync() to bring it into view.
        /// </summary>
        /// <param name="sender">The sender of this event</param>
        /// <param name="e">The args associated with this event.
        /// We expect e.UserState to be a MakeVisibleData.</param>
        private void OnGetPageNumberCompleted(object sender, GetPageNumberCompletedEventArgs e)
        {
            ArgumentNullException.ThrowIfNull(e);
            //Ensure that the UserState passed with this event contains an
            //MakeVisibleData object. If not, we ignore it as this event
            //could have originated from someone else calling GetPageNumberAsync.
            if (e.UserState is MakeVisibleData data)
            {
                MakeVisibleAsync(data, e.PageNumber);
            }
        }
 
 
        /// <summary>
        /// Makes the specified object on the specified page visible, which may be an
        /// asynchronous operation if the page is not already in view.
        /// </summary>
        /// <param name="data">Data corresponding to the object to be made visible.</param>
        /// <param name="pageNumber">The page number the object is on.</param>
        private void MakeVisibleAsync(MakeVisibleData data, int pageNumber)
        {
            //This page may not be currently visible.
            //First we need to make the page visible, if necessary.
            //This will be done at background priority to allow currently-loading pages time to
            //finish.  If we do not do this, then in the corner case of:
            // 1) Document has just been loaded (for example, just after a hyperlink navigation to the doc)
            // 2) Document has non 8.5x11-sized pages at the beginning of the document
            // 3) Navigation is to a page past the initially visible pages.
            //In this case, the order of operations is something like this:
            // 1) Initially visible pages start loading
            // 2) MakeVisible is invoked, and MakePageVisible is called, which uses cached
            //    page info to decide where to scroll to.  (8.5x11 is assumed until a page is loaded)
            // 3) Document is scrolled to position computed in #2
            // 4) Pages loading in #1 finish loading, GetPageCompleted is called, PageCache is updated,
            //    and the document layout changes.  This will shift the page we actually want to see up or down,
            //    potentially by a substantial amount.
            // 5) User is now looking at the wrong page, and if the page that the hyperlink target is on isn't
            //    visible at this point then the MakeVisible operation fails in MakeVisibleImpl since the target
            //    isn't in the visual tree.
            // Everyone got that?  So, doing the initial "MakePageVisible" operation at Background priority
            // allows step #4 above to finish _before_ we do steps 2 and 3, so the right page will be visible
            // in 5 and the MakeVisible operation will succeed.
            Dispatcher.BeginInvoke(DispatcherPriority.Background,
                   new BringPageIntoViewCallback(BringPageIntoViewDelegate), data, pageNumber);
        }
 
        /// <summary>
        /// Delegate method used to bring a specified page into view.
        /// </summary>
        /// <param name="data"></param>
        /// <param name="pageNumber"></param>
        private void BringPageIntoViewDelegate(MakeVisibleData data, int pageNumber)
        {
            //Make the page visible if necessary:
            // - If our layout is not yet valid
            // - If the visual being made visible is a FixedPage and
            //   the bring-into-view rect is the entire page
            //   (in which case we always want to move it as close to the top of the
            //   viewport as possible even if it is already partially visible)
            // - If the page isn't currently visible.
            if (!_rowCache.HasValidLayout ||
                (data.Visual is FixedPage &&
                 data.Visual.VisualContentBounds == data.Rect) ||
                pageNumber < _firstVisiblePageNumber ||
                pageNumber > _lastVisiblePageNumber)
            {
                MakePageVisible(pageNumber);
            }
 
            //The page's contents have already been loaded, we can bring the object into view immediately.
            if (IsPageLoaded(pageNumber))
            {
                MakeVisibleImpl(data);
            }
            else
            {
                //Now we have to wait for the page to be loaded so that we can
                //ensure that the object itself is visible.
                //As pages are loaded, this page will be checked for, and the dispatcher below
                //executed as appropriate.
                _makeVisiblePageNeeded = pageNumber;
                _makeVisibleDispatcher = Dispatcher.BeginInvoke(DispatcherPriority.Inactive,
                    (DispatcherOperationCallback)delegate (object arg)
                    {
                        MakeVisibleImpl((MakeVisibleData)arg);
                        return null;
                    }, data);
            }
        }
 
        /// <summary>
        /// Implementation of MakeVisible logic, the final step in a MakeVisible operation.
        /// </summary>
        /// <param name="data"></param>
        private void MakeVisibleImpl(MakeVisibleData data)
        {
            if (data.Visual != null)
            {
                //Ensure that the passed-in visual is a descendant of DocumentGrid.
                if (((Visual)this).IsAncestorOf(data.Visual))
                {
                    //Now we can determine where this visual is relative to the upper left
                    //corner of the DocumentGrid and thus make it visible.
                    GeneralTransform transform = data.Visual.TransformToAncestor(this);
                    Rect boundingRect = (data.Rect != Rect.Empty) ? data.Rect : data.Visual.VisualContentBounds;
 
                    Rect offsetRect = transform.TransformBounds(boundingRect);
                    MakeRectVisible(offsetRect, false /* alwaysCenter */);
                }
            }
            else if (data.ContentPosition != null)
            {
                ITextPointer tp = data.ContentPosition as ITextPointer;
 
                //If we have a valid TextView and the TextPointer is in that TextView
                //we can make the TextPointer's Rect visible...
                if (TextViewContains(tp))
                {
                    MakeRectVisible(TextView.GetRectangleFromTextPosition(tp), false /* alwaysCenter */);
                }
            }
            else
            {
                Invariant.Assert(false, "Invalid object brought into view.");
            }
        }
 
 
        /// <summary>
        /// Moves the specified rectangle into view, if it isn't already visible.
        /// </summary>
        /// <param name="r">A rectangle relative to the upper-left corner of the Viewport</param>
        /// <param name="alwaysCenter">Whether to center the rect at all times or only when necessary.</param>
        private void MakeRectVisible(Rect r, bool alwaysCenter)
        {
            if (r != Rect.Empty)
            {
                //Calculate the real position of the rectangle in the document.
                Rect translatedRect = new Rect(HorizontalOffset + r.X, VerticalOffset + r.Y,
                                               r.Width, r.Height);
 
                Rect viewportRect = new Rect(HorizontalOffset, VerticalOffset,
                                             ViewportWidth,
                                             ViewportHeight);
 
                //Unless the alwaysCenter flag is set, if the new position is already
                //visible we don't need to shift the viewport. Otherwise we shift
                //the offsets so the rect is visible, centering if possible.
                if (alwaysCenter || !translatedRect.IntersectsWith(viewportRect))
                {
                    SetHorizontalOffsetInternal(translatedRect.X - (ViewportWidth / 2.0));
                    SetVerticalOffsetInternal(translatedRect.Y - (ViewportHeight / 2.0));
                }
            }
        }
 
        /// <summary>
        /// Moves the specified IP into view, if it isn't already visible.
        /// </summary>
        /// <param name="r">A rectangle relative to the upper-left corner of the Viewport which represents
        /// an IP (Insertion Point)</param>
        private void MakeIPVisible(Rect r)
        {
            if (r != Rect.Empty && TextEditor != null)
            {
                Rect viewportRect = new Rect(HorizontalOffset, VerticalOffset,
                                             ViewportWidth,
                                             ViewportHeight);
 
                //If the new position is already fully visible, we don't need to shift the viewport,
                //otherwise we shift the offsets so the rect is visible, moving as minimally as possible.
                if (!viewportRect.Contains(r))
                {
                    //Scroll left/right if the IP is off the screen Horizontally.
                    if (r.X < HorizontalOffset)
                    {
                        SetHorizontalOffsetInternal(HorizontalOffset - (HorizontalOffset - r.X));
                    }
                    else if (r.X > HorizontalOffset + ViewportWidth)
                    {
                        SetHorizontalOffsetInternal(HorizontalOffset + (r.X - (HorizontalOffset + ViewportWidth)));
                    }
 
                    //Scroll up/down if part of the IP is off the screen Vertically.
                    if (r.Y < VerticalOffset)
                    {
                        SetVerticalOffsetInternal(VerticalOffset - (VerticalOffset - r.Y));
                    }
                    else if (r.Y + r.Height > VerticalOffset + ViewportHeight)
                    {
                        SetVerticalOffsetInternal(VerticalOffset + ((r.Y + r.Height) - (VerticalOffset + ViewportHeight)));
                    }
                }
            }
        }
 
        /// <summary>
        /// Invokes GetPageNumberAsync on the passed in ContentPosition.
        /// The handler for GetPageNumberAsync will bring that ContentPosition into view.
        /// </summary>
        /// <param name="data">The MakeVisibleData to be made visible</param>
        private void MakeContentPositionVisibleAsync(MakeVisibleData data)
        {
            //If the ContentPosition is valid, we can make it visible now.
            if (data.ContentPosition != null && data.ContentPosition != ContentPosition.Missing)
            {
                Content.GetPageNumberAsync(data.ContentPosition, data);
            }
        }
 
        #endregion MakeVisible Helpers
 
        /// <summary>
        /// Places a delegate for an SetScale call on the queue.  We do this for
        /// performance reasons, as changing the document scale takes significant time.
        /// </summary>
        /// <param name="scale"></param>
        private void QueueSetScale(double scale)
        {
            //If there's a SetScale operation in the Pending state, then we'll
            //abort it (we only care that the last operation invoked completes.)
            if (_setScaleOperation != null &&
                _setScaleOperation.Status == DispatcherOperationStatus.Pending)
            {
                _setScaleOperation.Abort();
            }
            _setScaleOperation = Dispatcher.CurrentDispatcher.BeginInvoke(DispatcherPriority.Input,
                   new DispatcherOperationCallback(SetScaleDelegate),
                   scale);
        }
 
        private object SetScaleDelegate(object scale)
        {
            if (!(scale is double))
            {
                return null;
            }
 
            double newScale = (double)scale;
            _documentLayout.ViewMode = ViewMode.Zoom;
 
            //Get the current visible selection, if any.
            //The results of this will determine how we handle the
            //zoom operation.
            ITextPointer selection = GetVisibleSelection();
 
            if (selection != null)
            {
                //The visible-IP case:
                //First, we find out what page the IP is on:
                int selectionPage = GetPageNumberForVisibleSelection(selection);
 
                //Then we scale the document:
                UpdateLayoutScale(newScale);
 
                //Now we ensure that the selection page is still
                //visible:
                MakePageVisible(selectionPage);
 
                //The rest of this process is done asynchronously --
                //We wait for LayoutUpdated (which happens after layout
                //but before rendering) and then make the IP visible.
                //This will cause the IP to be centered without any
                //visible flicker.
 
                //Attach a LayoutUpdated handler.
                LayoutUpdated += new EventHandler(OnZoomLayoutUpdated);
            }
            else
            {
                //The non-visible-IP case:
                //This is considerably easier.  The expected behavior is that
                //we zoom in on the upper-left corner of the currently-visible
                //content.  This is accomplished by scaling the Vertical and
                //Horizontal offsets in tandem with the document scale which will
                //put us approximately where we were before.
 
                //Scale the document:
                UpdateLayoutScale(newScale);
            }
 
            return null;
        }
 
        /// <summary>
        /// Updates the Scale applied to our RowCache.
        /// </summary>
        /// <param name="scale"></param>
        private void UpdateLayoutScale(double scale)
        {
            if (!DoubleUtil.AreClose(scale, Scale))
            {
                double oldExtentHeight = ExtentHeight;
                double oldExtentWidth = ExtentWidth;
 
                //Tell our RowCache to rescale the layout.
                _rowCache.Scale = scale;
 
                //Rescale our offsets
                //Divide the old extents by the new to determine the amount to
                //scale the offsets
                double verticalScale = oldExtentHeight == 0.0 ? 1.0 : ExtentHeight / oldExtentHeight;
                double horizontalScale = oldExtentWidth == 0.0 ? 1.0 : ExtentWidth / oldExtentWidth;
 
                //Now we scale the offsets.
                SetVerticalOffsetInternal(_verticalOffset * verticalScale);
                SetHorizontalOffsetInternal(_horizontalOffset * horizontalScale);
 
                InvalidateMeasure();
 
                //Invalidate the measure of our visual children so that they can
                //be resized.
                InvalidateChildMeasure();
 
                //Invalidate our parents' properties
                InvalidateDocumentScrollInfo();
            }
        }
 
        /// <summary>
        /// Places a delegate for an UpdateDocumentLayout call on the queue.  We do this for
        /// performance reasons, as changing the document layout takes significant time.
        /// </summary>
        /// <param name="layout"></param>
        private void QueueUpdateDocumentLayout(DocumentLayout layout)
        {
            // Increase the count of pending DocumentLayout delegates
            _documentLayoutsPending++;
            Dispatcher.CurrentDispatcher.BeginInvoke(DispatcherPriority.Input,
                   new DispatcherOperationCallback(UpdateDocumentLayoutDelegate),
                   layout);
        }
 
        /// <summary>
        /// Asynchronously invokes UpdateDocumentLayout.
        /// </summary>
        /// <param name="layout"></param>
        /// <returns></returns>
        private object UpdateDocumentLayoutDelegate(object layout)
        {
            if (layout is DocumentLayout)
            {
                UpdateDocumentLayout((DocumentLayout)layout);
            }
            // Decrease the count of pending DocumentLayout delegates
            _documentLayoutsPending--;
 
            return null;
        }
 
        /// <summary>
        /// Updates the current layout of our RowCache to the specified number of
        /// columns.
        /// </summary>
        /// <param name="layout"></param>
        private void UpdateDocumentLayout(DocumentLayout layout)
        {
            // Check if the layout is for a Vertical or Horizontal offset update,
            // in which case the value can be changed immediately.
            if (layout.ViewMode == ViewMode.SetHorizontalOffset)
            {
                SetHorizontalOffsetInternal(layout.Offset);
                return;
            }
            else if (layout.ViewMode == ViewMode.SetVerticalOffset)
            {
                SetVerticalOffsetInternal(layout.Offset);
                return;
            }
 
            //Store off the layout in case we need it later.
            //(For example, if our viewport is (0,0).
            _documentLayout = layout;
 
            //Update MaxPagesAcross
            _maxPagesAcross = _documentLayout.Columns;
 
            //If we have a non (0,0) Viewport then we can calculate a new layout.
            //Otherwise we set our "Layout Requested" flag so that when we get
            //a non-zero Viewport size we'll compute the requested layout.
            if (IsViewportNonzero)
            {
                //If this is a Thumbnails layout, we need to calculate how many
                //columns we should fit on the pivotRow.
                if (_documentLayout.ViewMode == ViewMode.Thumbnails)
                {
                    _maxPagesAcross = _documentLayout.Columns = CalculateThumbnailColumns();
                }
 
                //We need to determine the page that has the active focus so we know what page
                //to keep visible in the new layout.
                int pivotPage = GetActiveFocusPage();
 
                //Ask the RowCache to recalculate the layout based
                //on the specified pivot page and the number of columns
                //requested.
                //The RowCache will call us back with a
                //RowLayoutCompleted event when the layout is complete
                //and we'll update the scale and invalidate our layout there.
                _rowCache.RecalcRows(pivotPage, _documentLayout.Columns);
 
                _isLayoutRequested = false;
            }
            else
            {
                _isLayoutRequested = true;
            }
        }
 
        /// <summary>
        /// Calls UpdateLayout with the saved column and view mode parameters,
        /// if there's a previously requested layout to perform.
        /// Used when we get a non-zero Viewport size and we've previously requested
        /// a new Row layout.
        /// </summary>
        /// <returns>A bool indicating whether a new layout was calculated.</returns>
        private bool ExecutePendingLayoutRequests()
        {
            if (_isLayoutRequested)
            {
                UpdateDocumentLayout(_documentLayout);
                return true;
            }
 
            return false;
        }
 
        /// <summary>
        /// Set the HorizontalOffset to the value provided, the value will be set immediately.
        /// </summary>
        /// <param name="offset"></param>
        private void SetHorizontalOffsetInternal(double offset)
        {
            if (!DoubleUtil.AreClose(_horizontalOffset, offset))
            {
                if (Double.IsNaN(offset))
                {
                    throw new ArgumentOutOfRangeException("offset");
                }
 
                _horizontalOffset = offset;
                InvalidateMeasure();
                InvalidateDocumentScrollInfo();
                UpdateTextView();
            }
        }
 
        /// <summary>
        /// Set the VerticalOffset to the value provided, the value will be set immediately.
        /// </summary>
        /// <param name="offset"></param>
        private void SetVerticalOffsetInternal(double offset)
        {
            if (!DoubleUtil.AreClose(_verticalOffset, offset))
            {
                if (Double.IsNaN(offset))
                {
                    throw new ArgumentOutOfRangeException("offset");
                }
 
                _verticalOffset = offset;
                InvalidateMeasure();
                InvalidateDocumentScrollInfo();
                UpdateTextView();
            }
        }
 
        /// <summary>
        /// Updates the TextView so that it knows about size and position changes.
        /// </summary>
        private void UpdateTextView()
        {
            MultiPageTextView tv = TextView as MultiPageTextView;
            if (tv != null)
            {
                tv.OnPageLayoutChanged();
            }
        }
 
        /// <summary>
        /// Calculates the number of columns to fit on one row so that the resultant
        /// view will approximate a "thumbnail" view.
        /// The basic idea is that we attempt to fit (and scale) a number of pages on the row
        /// such that the resultant view given the current Viewport will show as many
        /// pages as possible, with a lower bound of a 5% zoom.
        /// We attempt to optimize to minimize wasted space, where possible.  This may mean
        /// that not all pages will be completely displayed, but we favor a better looking
        /// layout (less dead space) over being able to see every page.
        /// </summary>
        /// <returns>The number of pages to show on the first row.</returns>
        private int CalculateThumbnailColumns()
        {
            //If our current Viewport size is zero, we'll just return 1.
            //(because there always needs to be at least 1 page on a row regardless.)
            if (!IsViewportNonzero)
            {
                return 1;
            }
 
            //If we have no pages, we'll just return 1
            //(because there always needs to be at least 1 page on a row regardless.)
            if (_pageCache.PageCount == 0)
            {
                return 1;
            }
 
            //We use the first page of the document as our basis for our calculations.
            //This means that documents with varying page sizes can potentially have
            //sub-optimal thumbnail views.
            Size pageSize = _pageCache.GetPageSize(0);
 
            //Calculate the viewport's aspect ratio.
            double viewportAspect = ViewportWidth / ViewportHeight;
 
            //Calculate the maximum number of columns we can lay out on a single row
            //without needing to scale below our floor of 12.5%.
            int maxColumns =
                (int)Math.Floor(ViewportWidth /
                    (CurrentMinimumScale * pageSize.Width + HorizontalPageSpacing));
 
            //Ensure this value isn't greater than the number of pages in the document,
            //since we can't possibly lay out a row with more than that number of pages in it.
            maxColumns = Math.Min(maxColumns, _pageCache.PageCount);
            maxColumns = Math.Min(maxColumns, DocumentViewerConstants.MaximumMaxPagesAcross);
 
            //Now we do the following:
            //We iterate through the possible permutations of row and column
            //combinations and choose the arrangement of columns that best fits the current
            //viewport's aspect ratio.
            int minAspectColumns = 1;   //The current optimal number of columns found.
            double minAspectDiff = Double.MaxValue; //The current optimal aspect ratio match
            for (int columns = 1; columns <= maxColumns; columns++)
            {
                //Calculate the number of rows for this arrangment given the current column count
                int rows = (int)Math.Floor((double)(_pageCache.PageCount / columns));
 
                //Calculate the approximate dimensions that this layout would
                //have.
                double width = pageSize.Width * columns;
                double height = pageSize.Height * rows;
 
                //Determine the aspect ratio of this layout.
                double layoutAspect = width / height;
 
                //See if the aspect ratio of this layout is a closer match for our Viewport
                //than previous attempts.
                double aspectDiff = Math.Abs(layoutAspect - viewportAspect);
                if (aspectDiff < minAspectDiff)
                {
                    //It is, so save it.
                    minAspectDiff = aspectDiff;
                    minAspectColumns = columns;
                }
            }
 
            return minAspectColumns;
        }
 
        /// <summary>
        /// Calls InvalidateMeasure() on our visual children in order to force them
        /// to be re-measured and arranged on the next layout pass.  This is called
        /// whenever the Scale is changed, as it is only then when re-measuring is
        /// required.  We do this to avoid unnecessary layout/measure passes on our
        /// pages.
        /// </summary>
        private void InvalidateChildMeasure()
        {
            //Get the current Visual Collection, which contains our pages.
            int count = _childrenCollection.Count;
 
            for (int i = 0; i < count; i++)
            {
                UIElement page = _childrenCollection[i] as UIElement;
 
                if (page != null)
                {
                    page.InvalidateMeasure();
                }
            }
        }
 
        /// <summary>
        /// Helper function that indicates whether all the pages on the specified
        /// row point to clean cache entries.
        /// </summary>
        /// <param name="row"></param>
        /// <returns></returns>
        private bool RowIsClean(RowInfo row)
        {
            bool clean = true;
 
            for (int i = row.FirstPage; i < row.FirstPage + row.PageCount; i++)
            {
                if (_pageCache.IsPageDirty(i))
                {
                    clean = false;
                    break;
                }
            }
 
            return clean;
        }
 
        /// <summary>
        /// Checks that the current scale factor is optimal for the passed in row.
        /// </summary>
        /// <param name="pivotRow">The Row to pass to the Delegate</param>
        private void EnsureFit(RowInfo pivotRow)
        {
            //Get the scale factor necessary to fit this row into view.
            //If the result is not 1.0 (within a certain margin of error)
            //then we need to re-layout, alas.
            double neededScaleFactor = CalculateScaleFactor(pivotRow);
            double newScale = neededScaleFactor * _rowCache.Scale;
 
            //If the neededScaleFactor would require DocumentGrid scale the pages
            //below the minimum allowed zoom, or above the maximum, then we won't
            //do anything here.
            if (newScale < CurrentMinimumScale ||
                newScale > DocumentViewerConstants.MaximumScale)
            {
                return;
            }
 
            if (!DoubleUtil.AreClose(1.0, neededScaleFactor))
            {
                //Rescale the row.
                ApplyViewParameters(pivotRow);
 
                //Make the row visible again -- the offsets may have
                //changed due to the above rescaling.
                SetVerticalOffsetInternal(pivotRow.VerticalOffset);
            }
        }
 
        /// <summary>
        /// Given a pivot row and a previously set ViewMode, the scale is adjusted so as
        /// to cause the pivot row to be fit based on the specified ViewMode.
        /// </summary>
        /// <param name="pivotRow"></param>
        private void ApplyViewParameters(RowInfo pivotRow)
        {
            //Update our MaxPagesAcross property to the number of rows on the pivot row
            //if page sizes vary.  (If page sizes are uniform, this value will not change as a result of
            //a layout change)
            if (_pageCache.DynamicPageSizes)
            {
                _maxPagesAcross = pivotRow.PageCount;
            }
 
            //Get the scale factor necessary to fit the given row into the Viewport.
            double scaleFactor = CalculateScaleFactor(pivotRow);
 
            //Calculate the new scale. We multiply our scale factor by the old scale factor to cancel out any
            //previously applied scale.
            double newScale = scaleFactor * _rowCache.Scale;
 
            //Clip the value into the acceptable range
            newScale = Math.Max(newScale, CurrentMinimumScale);
            newScale = Math.Min(newScale, DocumentViewerConstants.MaximumScale);
 
            //Update the Row Layout's scale.
            UpdateLayoutScale(newScale);
        }
 
        private double CalculateScaleFactor(RowInfo pivotRow)
        {
            //Determine the dimensions of this row minus any spacing between the pages.
            //We use this as the baseline for our scale factor as page spacing does not scale.
            double rowWidth;
 
            //If the page sizes vary, we use the width of the pivot row,
            //otherwise we use the overall width of the document (ExtentWidth).
            //(For uniform page sizes, we always use the width of the document, even
            //for the last row which may not have the same width as the rest of the document).
            if (_pageCache.DynamicPageSizes)
            {
                rowWidth = pivotRow.RowSize.Width - pivotRow.PageCount * HorizontalPageSpacing;
            }
            else
            {
                rowWidth = ExtentWidth - MaxPagesAcross * HorizontalPageSpacing;
            }
 
            double rowHeight = pivotRow.RowSize.Height - VerticalPageSpacing;
 
            //If we have row dimensions of zero or less, there's no reason to scale anything.
            //So just return 1.0 to indicate no change.
            if (rowWidth <= 0.0 || rowHeight <= 0.0)
            {
                return 1.0;
            }
 
            //The dimensions of our Viewport minus any spacing.  We use this as the baseline for our
            //scale factor as page spacing does not scale.
            double compensatedViewportWidth;
 
            if (_pageCache.DynamicPageSizes)
            {
                compensatedViewportWidth = ViewportWidth - pivotRow.PageCount * HorizontalPageSpacing;
            }
            else
            {
                compensatedViewportWidth = ViewportWidth - MaxPagesAcross * HorizontalPageSpacing;
            }
 
            double compensatedViewportHeight = ViewportHeight - VerticalPageSpacing;
 
            //If we have no space to display pages, there's nothing to scale.
            //So just return 1.0 to indicate no change.
            if (compensatedViewportWidth <= 0.0 ||
                compensatedViewportHeight <= 0.0)
            {
                return 1.0;
            }
 
            double scaleFactor = 1.0;
 
            //Based on the previously determined ViewMode (set in SetColumns(), FitToWidth(), etc..
            //scale the pages appropriately.
            switch (_documentLayout.ViewMode)
            {
                case ViewMode.SetColumns:
                    //We leave the scale factor as is -- this is not a "page-fit" mode.
                    break;
 
                case ViewMode.FitColumns:
                    //Update the scale factor so that the pivot row is completely visible.
                    scaleFactor = Math.Min(compensatedViewportWidth / rowWidth, compensatedViewportHeight / rowHeight);
                    break;
 
                case ViewMode.PageWidth:
                    //Update the scale factor so that the pivot row is as wide as the viewport.
                    scaleFactor = compensatedViewportWidth / rowWidth;
                    break;
 
                case ViewMode.PageHeight:
                    //Update the scale factor so that the pivot row is as tall as the viewport.
                    scaleFactor = compensatedViewportHeight / rowHeight;
                    break;
 
                case ViewMode.Thumbnails:
                    //Update the scale factor so that the _entire layout_ is completely visible.  As in previous
                    //cases we must compensate for the fact that the spacing between pages does not scale.
                    //However, unlike in previous cases, we must exclude the space between all rows rather
                    //merely one space, so we must recalculate the compensated values.  Furthermore we must
                    //also compensate for the ExtentHeight as well since it includes the spaces.
                    double thumbnailCompensatedExtentHeight = ExtentHeight - VerticalPageSpacing * _rowCache.RowCount;
                    double thumbnailCompensatedViewportHeight = ViewportHeight - VerticalPageSpacing * _rowCache.RowCount;
                    //If we have no space to display pages, there's nothing to scale.
                    //So just return 1.0 to indicate no change.
                    if (thumbnailCompensatedViewportHeight <= 0.0)
                    {
                        scaleFactor = 1.0;
                    }
                    else
                    {
                        scaleFactor = Math.Min(compensatedViewportWidth / rowWidth,
                            thumbnailCompensatedViewportHeight / thumbnailCompensatedExtentHeight);
                    }
                    break;
 
                case ViewMode.Zoom:
                    //We will not change the scale here, as this is not a "page-fit" mode.
                    break;
 
                default:
                    throw new InvalidOperationException(SR.DocumentGridInvalidViewMode);
            }
 
            return scaleFactor;
        }
 
        /// <summary>
        /// Creates the caches used by DocumentGrid, and sets default property values.
        /// </summary>
        private void Initialize()
        {
            //Create our caches
            _pageCache = new PageCache();
            _childrenCollection = new VisualCollection(this);
 
            _rowCache = new RowCache
            {
                PageCache = _pageCache
            };
            _rowCache.RowCacheChanged += new RowCacheChangedEventHandler(OnRowCacheChanged);
            _rowCache.RowLayoutCompleted += new RowLayoutCompletedEventHandler(OnRowLayoutCompleted);
        }
 
        /// <summary>
        /// Updates Scrolling-related properties and calls
        /// InvalidateScrollInfo and InvalidateDocumentScrollInfo on the
        /// ScrollOwner and DocumentScrollOwner, respectively.
        /// </summary>
        private void InvalidateDocumentScrollInfo()
        {
            if (ScrollOwner != null)
            {
                ScrollOwner.InvalidateScrollInfo();
            }
 
            if (DocumentViewerOwner != null)
            {
                DocumentViewerOwner.InvalidateDocumentScrollInfo();
            }
        }
 
        /// <summary>
        /// Calls InvalidatePageViews and ApplyTemplate on our DocumentViewer owner
        /// so that the base implementation can keep its collection up to date.
        /// </summary>
        private void InvalidatePageViews()
        {
            Invariant.Assert(DocumentViewerOwner != null, "DocumentViewerOwner cannot be null.");
 
            if (DocumentViewerOwner != null)
            {
                DocumentViewerOwner.InvalidatePageViewsInternal();
                DocumentViewerOwner.ApplyTemplate();
            }
 
            //Perf Tracing - InvalidatePageViews
            EventTrace.EasyTraceEvent(EventTrace.Keyword.KeywordXPS, EventTrace.Event.WClientDRXInvalidateView);
        }
 
        /// <summary>
        /// Returns an ITextPointer to the current visible selection, if there is one.
        /// </summary>
        /// <returns>An ITextPointer to the current selection, or null if none exists.</returns>
        private ITextPointer GetVisibleSelection()
        {
            ITextPointer selection = null;
 
            if (HasSelection())
            {
                ITextPointer tp = TextEditor.Selection.Start;
 
                //If the TextView contains the selection
                //then the selection is on a visible page.
                if (TextViewContains(tp))
                {
                    selection = tp;
                }
            }
 
            return selection;
        }
 
        /// <summary>
        /// Indicates whether a selection (visible or not) has been made.
        /// </summary>
        /// <returns>true if a selection has been made, false otherwise.</returns>
        private bool HasSelection()
        {
            return (TextEditor != null && TextEditor.Selection != null);
        }
 
        /// <summary>
        /// Gets the page number that the specified ITextPointer to a visible selection
        /// is on.
        /// </summary>
        /// <param name="selection">The TextPointer to find the page number for.</param>
        /// <returns></returns>
        private int GetPageNumberForVisibleSelection(ITextPointer selection)
        {
            Invariant.Assert(TextViewContains(selection));
 
            //Walk through the current DocumentPageViews and see which one contains the selection.
            foreach (DocumentPageView pageView in _pageViews)
            {
                //Get the TextView for this page.
                DocumentPageTextView textView =
                    ((IServiceProvider)pageView).GetService(typeof(ITextView)) as DocumentPageTextView;
 
                //If this TextView contains the selection, return the page's number.
                if (textView != null &&
                    textView.IsValid &&
                    textView.Contains(selection))
                {
                    return pageView.PageNumber;
                }
            }
 
            Invariant.Assert(false, "Selection was in TextView, but not found in any visible page!");
            return 0;
        }
 
        /// <summary>
        /// Finds the "Active Focus" point:
        /// Either the page that has a Selection/Insertion Point on it,
        /// or lacking that, the center of the viewport.
        /// </summary>
        /// <returns></returns>
        private Point GetActiveFocusPoint()
        {
            ITextPointer tp = GetVisibleSelection();
 
            if (tp != null && tp.HasValidLayout)
            {
                Rect selectionRect = TextView.GetRectangleFromTextPosition(tp);
 
                //If the selection rectangle is not empty, then we have a selection or an IP
                if (selectionRect != Rect.Empty)
                {
                    //Return the upper-left corner of the selection.
                    return new Point(selectionRect.Left, selectionRect.Top);
                }
            }
 
            //No selection, so we default to the upper-left of the viewport.
            return new Point(0.0, 0.0);
        }
 
        /// <summary>
        /// Returns the page that has "Active Focus," or
        /// the first visible page if there is none.
        /// </summary>
        /// <returns></returns>
        private int GetActiveFocusPage()
        {
            DocumentPageView dp = GetDocumentPageViewFromPoint(GetActiveFocusPoint());
 
            if (dp != null)
            {
                return dp.PageNumber;
            }
 
            //No selection, we default to the first visible page.
            return _firstVisiblePageNumber;
        }
 
        /// <summary>
        /// Given a point onscreen, returns a DocumentPageView that occupies that point,
        /// if any.
        /// </summary>
        /// <param name="point">The point at which to search for a DocumentPageView.</param>
        /// <returns></returns>
        private DocumentPageView GetDocumentPageViewFromPoint(Point point)
        {
            //Hit test to find the DocumentPageView
            HitTestResult result = VisualTreeHelper.HitTest(this, point);
            DependencyObject currentVisual = (result != null) ? result.VisualHit : null;
 
            DocumentPageView page = null;
 
            // Traverse the visual parent chain until we encounter a DocumentPageView.
            while (currentVisual != null)
            {
                page = currentVisual as DocumentPageView;
                if (page != null)
                {
                    //We found the DocumentPageView.
                    return page;
                }
                currentVisual = VisualTreeHelper.GetParent(currentVisual);
            }
 
            //Didn't find one at this point.
            return null;
        }
 
        /// <summary>
        /// Helper function to safely verify that the TextView contains a given TextPointer.
        /// </summary>
        /// <param name="tp">The TextPointer to check</param>
        /// <returns></returns>
        private bool TextViewContains(ITextPointer tp)
        {
            return (TextView != null &&
                TextView.IsValid &&
                TextView.Contains(tp));
        }
 
        /// <summary>
        /// Helper function that calculates the Horizontal offset of the given page.
        /// </summary>
        /// <param name="row">The row which the desired page lives on</param>
        /// <param name="pageNumber">The page to find the offset for</param>
        /// <returns>The Horizontal offset of the page in the document.</returns>
        private double GetHorizontalOffsetForPage(RowInfo row, int pageNumber)
        {
            ArgumentNullException.ThrowIfNull(row);
            ArgumentOutOfRangeException.ThrowIfLessThan(pageNumber, row.FirstPage);
            ArgumentOutOfRangeException.ThrowIfGreaterThan(pageNumber, row.FirstPage + row.PageCount);
 
            //Rows are centered if the content has varying page sizes,
            //Left-aligned otherwise.
            double horizontalOffset = _pageCache.DynamicPageSizes ?
                Math.Max(0.0, (ExtentWidth - row.RowSize.Width) / 2.0) : 0.0;
 
            //Add the widths of the pages (and spacing) prior to this one on the row
            for (int i = row.FirstPage; i < pageNumber; i++)
            {
                Size pageSize = _pageCache.GetPageSize(i);
                horizontalOffset += pageSize.Width * Scale + HorizontalPageSpacing;
            }
 
            return horizontalOffset;
        }
 
        /// <summary>
        /// Helper method that determines whether a given RowCacheChange will have
        /// an impact on currently-visible rows.
        /// </summary>
        /// <param name="change"></param>
        /// <returns></returns>
        private bool RowCacheChangeIsVisible(RowCacheChange change)
        {
            int firstVisibleRow = _firstVisibleRow;
            int lastVisibleRow = _firstVisibleRow + _visibleRowCount;
 
            int firstChangedRow = change.Start;
            int lastChangedRow = change.Start + change.Count;
 
            //If the first changed row (and hence following changes) are visible OR
            //The last changed row (and hence prior changes) are visible OR
            //if the changes are a super-set of the visible range, then the change is visible.
            if ((firstChangedRow >= firstVisibleRow && firstChangedRow <= lastVisibleRow) ||
                (lastChangedRow >= firstVisibleRow && lastChangedRow <= lastVisibleRow) ||
                (firstChangedRow < firstVisibleRow && lastChangedRow > lastVisibleRow))
            {
                return true;
            }
 
            return false;
        }
 
        /// <summary>
        /// Determines whether the given page's contents have been loaded into the visual tree.
        /// </summary>
        /// <param name="pageNumber">The number of the page to check</param>
        /// <returns></returns>
        private bool IsPageLoaded(int pageNumber)
        {
            DocumentGridPage page = GetDocumentGridPageForPageNumber(pageNumber);
 
            if (page != null)
            {
                return page.IsPageLoaded;
            }
            else
            {
                return false;
            }
        }
 
        /// <summary>
        /// Determines whether every page currently visible has been loaded into the visual tree.
        /// </summary>
        /// <returns></returns>
        private bool IsViewLoaded()
        {
            bool viewIsLoaded = true;
 
            for (int i = _firstPageVisualIndex; i < _childrenCollection.Count; i++)
            {
                DocumentGridPage dp = _childrenCollection[i] as DocumentGridPage;
 
                // Check that this page has been loaded; break if not.
                if (dp != null && !dp.IsPageLoaded)
                {
                    viewIsLoaded = false;
                    break;
                }
            }
 
            return viewIsLoaded;
        }
 
        /// <summary>
        /// Retrieves a DocumentGridPage from our Visual Tree that has the given page number (if one exists).
        /// </summary>
        /// <param name="pageNumber">The number of the page to get</param>
        /// <returns></returns>
        private DocumentGridPage GetDocumentGridPageForPageNumber(int pageNumber)
        {
            for (int i = _firstPageVisualIndex; i < _childrenCollection.Count; i++)
            {
                DocumentGridPage dp = _childrenCollection[i] as DocumentGridPage;
 
                if (dp != null && dp.PageNumber == pageNumber)
                {
                    return dp;
                }
            }
 
            return null;
        }
 
        #region Event Handlers
 
        /// <summary>
        /// Handles the RequestBringIntoView routed event in the case where the element to be
        /// brought into view is DocumentGrid itself, as is the case when the TextEditor's IP moves.
        /// In this case we use the incoming target rectangle
        /// and ensure that rect is made visible inside of DocumentGrid.
        /// </summary>
        /// <param name="sender">The sender of this routed event, expected to be a DocumentGrid.</param>
        /// <param name="args">The RequestBringIntoView event args associated with this event.</param>
        private static void OnRequestBringIntoView(object sender, RequestBringIntoViewEventArgs args)
        {
            //We only handle this here if the sender and the target of this event are both the same
            //DocumentGrid.
            DocumentGrid senderGrid = sender as DocumentGrid;
            DocumentGrid targetGrid = args.TargetObject as DocumentGrid;
            if (senderGrid != null && targetGrid != null && senderGrid == targetGrid)
            {
                //Bring the IP into view and mark the event as handled.
                args.Handled = true;
                targetGrid.MakeIPVisible(args.TargetRect);
            }
            else
            {
                args.Handled = false;
            }
        }
 
        /// <summary>
        /// This event is fired when our parent ScrollViewer's layout has changed(before render).
        /// If we need to ensure that a given row has been properly fit -- if ScrollBars have been hidden/
        /// made visible due to this change then we may need to resize.  We call EnsureFit from here
        /// to make sure that's done.
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="args"></param>
        private void OnScrollChanged(object sender, EventArgs args)
        {
            //Remove our handler.
            if (ScrollOwner != null)
            {
                _scrollChangedEventAttached = false;
                ScrollOwner.ScrollChanged -= new ScrollChangedEventHandler(OnScrollChanged);
            }
 
            //Ensure that our fit is good for the currently displayed row if we have any.
            if (_rowCache.HasValidLayout)
            {
                EnsureFit(_rowCache.GetRowForPageNumber(FirstVisiblePageNumber));
            }
        }
 
        /// <summary>
        /// This event is fired after a Zoom change when Layout has completed (but before render).
        /// We make sure the current visible selection is centered onscreen.
        /// </summary>
        /// <param name="sender"></param>
        /// <param name="args"></param>
        private void OnZoomLayoutUpdated(object sender, EventArgs args)
        {
            //Remove the event handler so we don't get called again.
            LayoutUpdated -= new EventHandler(OnZoomLayoutUpdated);
 
            ITextPointer selection = GetVisibleSelection();
 
            if (selection != null)
            {
                //Now we make the selection visible.
                MakeRectVisible(TextView.GetRectangleFromTextPosition(
                    selection), true /* alwaysCenter */);
            }
        }
 
        /// <summary>
        /// When the RowCache is changed for any reason (due to a new layout, or if pages change, etc...)
        /// then we need to invalidate our Measure (so we can pick up any changes to visible pages)
        /// and invalidate our IDocumentScrollInfo parents so they know that something's changed.
        /// </summary>
        /// <param name="source"></param>
        /// <param name="args"></param>
        private void OnRowCacheChanged(object source, RowCacheChangedEventArgs args)
        {
            //If:
            //1) We have a saved pivot row from a previous RowCacheCompleted event,
            //and
            //2) We've been told to do a "Page-Fit" operation (that is, a non-zoom viewing preference)
            //and
            //3) The pivot row is now "clean" (that is, we know the actual dimensions of all the pages
            //   on the row and we aren't just guessing)
            //Then we can now officially calculate the scale needed in order to fit the given row in the manner
            //chosen.
            if (_savedPivotRow != null &&
                RowIsClean(_savedPivotRow))
            {
                if (_documentLayout.ViewMode != ViewMode.Zoom &&
                    _documentLayout.ViewMode != ViewMode.SetColumns
                    )
                {
                    if (_savedPivotRow.FirstPage < _rowCache.RowCount)
                    {
                        RowInfo newRow = _rowCache.GetRowForPageNumber(_savedPivotRow.FirstPage);
 
                        //If the new row's dimensions differ, then we need to rescale, otherwise we do nothing.
                        if (newRow.RowSize.Width != _savedPivotRow.RowSize.Width ||
                            newRow.RowSize.Height != _savedPivotRow.RowSize.Height)
                        {
                            //Rescale.
                            ApplyViewParameters(newRow);
                        }
 
                        //Null out the saved Pivot Row -- we've scaled this row properly now
                        //so we don't need to be concerned with it any longer.
                        _savedPivotRow = null;
                    }
                }
                else
                {
                    // The view is already correct; null out the saved Pivot Row.
                    _savedPivotRow = null;
                }
            }
 
            //If we're viewing a document with varying page size, we've scrolled since the last layout change
            //and the Width of the document has increased
            //then this means that new, wider pages have just been scrolled into view.
            //If we do nothing here, then the content prior to these new pages will appear to "jump"
            //to the right (because we center the pages within the width of the document.)
            //This jump is jarring and not a good user experience.
            //To prevent this, we adjust the HorizontalOffset such that the content that was previously visible
            //appears at the same position when it is rendered.
            if (_pageCache.DynamicPageSizes &&
                _lastRowChangeVerticalOffset != VerticalOffset &&
                _lastRowChangeExtentWidth < ExtentWidth)
            {
                if (_lastRowChangeExtentWidth != 0.0)
                {
                    //Tweak the HorizontalOffset so that the content does not appear to move.
                    SetHorizontalOffsetInternal(HorizontalOffset + (ExtentWidth - _lastRowChangeExtentWidth) / 2.0);
                }
 
                _lastRowChangeExtentWidth = ExtentWidth;
            }
 
            _lastRowChangeVerticalOffset = VerticalOffset;
 
            //The row cache has been changed.
            //If we're displaying rows that were affected,
            //we need to invalidate our measure so they'll be
            //redrawn.
            for (int i = 0; i < args.Changes.Count; i++)
            {
                RowCacheChange change = args.Changes[i];
 
                if (RowCacheChangeIsVisible(change))
                {
                    InvalidateMeasure();
                    InvalidateChildMeasure();
                }
            }
 
            InvalidateDocumentScrollInfo();
        }
 
        /// <summary>
        /// When a new RowLayout has finished being computed we scale the layout such that it
        /// fits within our window.
        /// </summary>
        /// <param name="source"></param>
        /// <param name="args"></param>
        private void OnRowLayoutCompleted(object source, RowLayoutCompletedEventArgs args)
        {
            if (args == null)
            {
                return;
            }
            if (args.PivotRowIndex >= _rowCache.RowCount)
            {
                throw new ArgumentOutOfRangeException("args");
            }
 
            //Get the pivot row
            RowInfo pivotRow = _rowCache.GetRow(args.PivotRowIndex);
 
            //If this row is not clean, and we're not applying a
            //Zoom to the content then we need to rescale the layout when the
            //pages on this row are retrieved.
            if (!RowIsClean(pivotRow) && _documentLayout.ViewMode != ViewMode.Zoom)
            {
                //Save off this row in case we need to rescale due to
                //dirty cache entries becoming clean (i.e. page sizes changing
                //due to the cached size being an inaccurate guess.)
                //OnRowCacheChanged will check this row to ensure that it gets scaled
                //properly when all the pages on the row become available.
                _savedPivotRow = pivotRow;
            }
            else
            {
                _savedPivotRow = null;
            }
 
            //Now rescale.  We do this after checking the cleanliness of the row
            //so that _savedPivotRow is properly set before we apply our view parameters.
            //Otherwise the code that relies on it in OnRowCacheChanged (which may be called
            //as a result of calling ApplyViewParameters) may use the wrong row.
            ApplyViewParameters(pivotRow);
 
            //Now that we've recalculated the row layout, it's time to make the previously-visible
            //content visible again.
            //We do not do this the first time the content is assigned, for two reasons,
            //(similar to the ones described in DocumentViewer.OnDocumentChanged()):
            //  1) If this is the first assignment, then we're already there by default.
            //  2) The user may have specified vertical or horizontal offsets in markup or
            //     otherwise (<DocumentViewer VerticalOffset="1000">) and we need to honor
            //     those settings.
            if (!_firstRowLayout && !_pageJumpAfterLayout)
            {
                MakePageVisible(pivotRow.FirstPage);
            }
            else if (_pageJumpAfterLayout)
            {
                MakePageVisible(_pageJumpAfterLayoutPageNumber);
                _pageJumpAfterLayout = false;
            }
 
            _firstRowLayout = false;
 
            //If our view was of a "Fit" type, we need to ensure that the fit is
            //correct -- if the status of Vertical/Horizontal Scrollbars has changed
            //as a result of our view selection then the Viewport size may have changed.
            //If so, our current fit is probably wrong.  We'll attach a ScrollChanged handler
            //to our ScrollOwner and when the event is invoked (after layout, but before rendering)
            //we'll check.
            //This is "Step 2" of the two-pass layout necessary to do fit properly inside of a
            //ScrollViewer.
            if (!_scrollChangedEventAttached &&
                ScrollOwner != null &&
                _documentLayout.ViewMode != ViewMode.Zoom &&
                _documentLayout.ViewMode != ViewMode.SetColumns)
            {
                _scrollChangedEventAttached = true;
                ScrollOwner.ScrollChanged += new ScrollChangedEventHandler(OnScrollChanged);
            }
        }
 
        #endregion Event Handlers
 
        #endregion Private Methods
 
        //------------------------------------------------------
        //
        //  Private Properties
        //
        //------------------------------------------------------
        #region Private Properties
 
        /// <summary>
        /// Indicates that our Viewport is or is not exactly (0,0).
        /// </summary>
        /// <value></value>
        private bool IsViewportNonzero
        {
            get
            {
                return (ViewportWidth != 0.0 && ViewportHeight != 0.0);
            }
        }
 
        /// <summary>
        /// Provides access to DocumentViewer's TextEditor.
        /// </summary>
        private TextEditor TextEditor
        {
            get
            {
                if (DocumentViewerOwner != null)
                {
                    return DocumentViewerOwner.TextEditor;
                }
                else
                {
                    return null;
                }
            }
        }
 
        /// <summary>
        /// Represents the number of pixels to scroll by when using the
        /// Mouse Wheel; based on System.Parameters.WheelScrollLines.
        /// </summary>
        private double MouseWheelVerticalScrollAmount
        {
            get
            {
                //SystemParameters.WheelScrollLines indicates the number of lines to
                //scroll when the wheel is moved one "click," we multiply this by
                //our scroll amount to get the number of pixels to move.
                return _verticalLineScrollAmount * SystemParameters.WheelScrollLines;
            }
        }
 
        private bool CanMouseWheelVerticallyScroll
        {
            get { return _canVerticallyScroll && SystemParameters.WheelScrollLines > 0; }
        }
 
        /// <summary>
        /// Represents the number of pixels to scroll by when using the
        /// Mouse Wheel; based on System.Parameters.WheelScrollLines.
        /// </summary>
        private double MouseWheelHorizontalScrollAmount
        {
            get
            {
                //SystemParameters.WheelScrollLines indicates the number of lines to
                //scroll when the wheel is moved one "click," we multiply this by
                //our scroll amount to get the number of pixels to move.
                return _horizontalLineScrollAmount * SystemParameters.WheelScrollLines;
            }
        }
 
        private bool CanMouseWheelHorizontallyScroll
        {
            get { return _canHorizontallyScroll && SystemParameters.WheelScrollLines > 0; }
        }
 
        /// <summary>
        /// Returns the minimum allowed scale based on the current view mode.
        /// Thumbnails mode has a higher minimum than other views.
        /// </summary>
        private double CurrentMinimumScale
        {
            get
            {
                return _documentLayout.ViewMode == ViewMode.Thumbnails ?
                  DocumentViewerConstants.MinimumThumbnailsScale :
                  DocumentViewerConstants.MinimumScale;
            }
        }
 
        #endregion Private Properties
 
        //------------------------------------------------------
        //
        //  Private Fields
        //
        //------------------------------------------------------
 
        #region Private Fields
 
        // Our Caches
        private PageCache _pageCache;
        private RowCache _rowCache;
 
        // Our collection of currently-displayed pages.
        private ReadOnlyCollection<DocumentPageView> _pageViews;
 
        // Data for Properties
        private bool _canHorizontallyScroll;
        private bool _canVerticallyScroll;
        private double _verticalOffset;
        private double _horizontalOffset;
        private double _viewportHeight;
        private double _viewportWidth;
        private int _firstVisibleRow;
        private int _visibleRowCount;
        private int _firstVisiblePageNumber;
        private int _lastVisiblePageNumber;
        private ScrollViewer _scrollOwner;
        private DocumentViewer _documentViewerOwner;
        private bool _showPageBorders = true;
        private bool _lockViewModes;
        private int _maxPagesAcross = 1;
 
        // The previous constraint passed to ArrangeCore.
        private Size _previousConstraint;
 
        // The viewing mode (columns, fit, etc...) last used to request a layout.
        private DocumentLayout _documentLayout =
            new DocumentLayout(1, ViewMode.SetColumns);
        private int _documentLayoutsPending;
 
        // The pivot rows last used to form the basis of a layout.
        private RowInfo _savedPivotRow;
 
        // The last ExtentWidth and VerticalOffsets encountered in
        // OnRowCacheChanged, used to determine whether to tweak the HorizontalOffset.
        private double _lastRowChangeExtentWidth;
        private double _lastRowChangeVerticalOffset;
 
        // Editing
        private ITextContainer _textContainer;
 
        // RubberBand selector used for rubberband selection.
        private RubberbandSelector _rubberBandSelector;
 
        // Flags
        private bool _isLayoutRequested;  //Whether we have requested a layout from the RowCache.
        private bool _pageJumpAfterLayout;   //Whether we need to bring a page into view after layout
        private int _pageJumpAfterLayoutPageNumber; //The page to jump to after layout
        private bool _firstRowLayout = true;
        private bool _scrollChangedEventAttached; //Whether or not we've attached a ScrollChanged event to our ScrollViewer.
 
        // We create a Border with a transparent background so that it can
        // participate in Hit-Testing (which allows click events like those
        // for our Context Menu to work).  This border is displayed behind
        // the displayed pages so that "dead space" surrounding the pages can
        // be clicked on.
        private Border _documentGridBackground;
        private const int _backgroundVisualIndex = 0;
        private const int _firstPageVisualIndex = 1;
 
        //Constants for MeasureCore constraints
        //We use this size if we're placed inside a "Size-To-Parent" container like
        //ScrollViewer or StackPanel and are given Infinite constraints.
        private readonly Size _defaultConstraint = new Size(250.0, 250.0);
 
        //Store all our visual children (pages) here
        private VisualCollection _childrenCollection;
 
        //Information for MakeVisible operations involving pages that are not
        //yet visible.
        private int _makeVisiblePageNeeded = -1;
        private DispatcherOperation _makeVisibleDispatcher;
 
        //DispatcherOperations used for executing time-consuming property changes in the background.
        private DispatcherOperation _setScaleOperation;
 
        //Delegate used for BringPageIntoView.
        private delegate void BringPageIntoViewCallback(MakeVisibleData data, int pageNumber);
 
        /// <summary>
        /// Represents a state in the Visual tree merging state machine
        /// used in RecalcVisiblePages.
        /// </summary>
        private enum VisualTreeModificationState
        {
            /// <summary>
            /// Inserting pages before existing
            /// </summary>
            BeforeExisting = 0,
 
            /// <summary>
            /// Scanning through existing pages
            /// </summary>
            DuringExisting,
 
            /// <summary>
            /// Adding pages after existing
            /// </summary>
            AfterExisting
        }
 
        /// <summary>
        /// Represents a layout mode specified by SetColumns,
        /// FitToPage, FitToWidth, etc...
        /// </summary>
        private enum ViewMode
        {
            /// <summary>
            /// A request to lay out a specified number of columns was made.
            /// </summary>
            SetColumns = 0,
 
            /// <summary>
            /// A request to make the specified number of columns visible was made.
            /// </summary>
            FitColumns,
 
            /// <summary>
            /// A request for a fit-to-page-width view was made.
            /// </summary>
            PageWidth,
 
            /// <summary>
            /// A request for a fit-to-page-height view was made.
            /// </summary>
            PageHeight,
 
            /// <summary>
            /// A request for a thumbnail view was made.
            /// </summary>
            Thumbnails,
 
            /// <summary>
            /// A request for a non "page-fit" view was made.
            /// </summary>
            Zoom,
 
            /// <summary>
            /// A request for the HorizontalOffset to be updated.
            /// </summary>
            SetHorizontalOffset,
 
            /// <summary>
            /// A request for the VerticalOffset to be updated.
            /// </summary>
            SetVerticalOffset
        }
 
        /// <summary>
        /// Represents a particular document layout --
        /// includes the number of Columns to view and the
        /// mode to view them in.
        /// </summary>
        private class DocumentLayout
        {
            public DocumentLayout(int columns, ViewMode viewMode)
                : this(columns, 0.0 /* default */, viewMode) { }
 
            public DocumentLayout(double offset, ViewMode viewMode)
                : this(1 /* default */, offset, viewMode) { }
 
            public DocumentLayout(int columns, double offset, ViewMode viewMode)
            {
                _columns = columns;
                _offset = offset;
                _viewMode = viewMode;
            }
 
            /// <summary>
            /// The ViewMode to apply to the layout.
            /// </summary>
            public ViewMode ViewMode
            {
                set { _viewMode = value; }
                get { return _viewMode; }
            }
 
            /// <summary>
            /// The number of columns for the layout.
            /// </summary>
            public int Columns
            {
                set { _columns = value; }
                get { return _columns; }
            }
 
            /// <summary>
            /// The offset (horizontal of vertical) for the layout.
            /// </summary>
            public double Offset
            {
                // Set not currently used.
                // set { _offset = value; }
                get { return _offset; }
            }
 
            private ViewMode _viewMode;
            private int _columns;
            private double _offset;
        }
 
        /// <summary>
        /// An MakeVisibleData object contains data and operation information
        /// related to asynchronous MakeVisible operations.
        /// </summary>
        private struct MakeVisibleData
        {
            /// <summary>
            /// Constructs a new MakeVisibleData object
            /// </summary>
            /// <param name="visual">A visual to be made visible.</param>
            /// <param name="contentPosition">A ContentPosition to be made visible.</param>
            /// <param name="rect">Any bounding rect to be made visible.</param>
            public MakeVisibleData(Visual visual, ContentPosition contentPosition, Rect rect)
            {
                _visual = visual;
                _contentPosition = contentPosition;
                _rect = rect;
            }
 
            /// <summary>
            /// The Visual to be made visible
            /// </summary>
            public Visual Visual
            {
                get { return _visual; }
            }
 
            /// <summary>
            /// The ContentPosition to be made Visible
            /// </summary>
            public ContentPosition ContentPosition
            {
                get { return _contentPosition; }
            }
 
            /// <summary>
            /// The bounding rectangle to be made visible
            /// </summary>
            public Rect Rect
            {
                get { return _rect; }
            }
 
            private Visual _visual;
            private ContentPosition _contentPosition;
            private Rect _rect;
        }
 
        //Constants for line scrolling amounts
        private const double _verticalLineScrollAmount = 16.0;
        private const double _horizontalLineScrollAmount = 16.0;
 
        #endregion Private Fields
    }
}