File: Microsoft\Windows\Controls\Ribbon\Primitives\RibbonGalleryItemsPanel.cs
Web Access
Project: src\src\Microsoft.DotNet.Wpf\src\System.Windows.Controls.Ribbon\System.Windows.Controls.Ribbon_h51tyt4o_wpftmp.csproj (System.Windows.Controls.Ribbon)
// 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.
 
 
 
#region Using Declarations
 
using System.Diagnostics;
#if RIBBON_IN_FRAMEWORK
 
#if RIBBON_IN_FRAMEWORK
namespace System.Windows.Controls.Ribbon.Primitives
#else
namespace Microsoft.Windows.Controls.Ribbon.Primitives
#endif
{
#else
    using Microsoft.Windows.Controls.Ribbon;
#endif
 
    #endregion
 
    /// <summary>
    ///   The default Items panel for the RibbonGalleryCategory class.
    /// </summary>
    public class RibbonGalleryItemsPanel : Panel
    {
 
        #region Private Methods
 
        // It just determines if any ancestor supports StarLayout and is not in StarLayoutPass mode.
        private bool IsAutoLayoutPass()
        {
            RibbonGallery gallery = Gallery;
            if (gallery != null)
            {
                RibbonGalleryCategoriesPanel categoriesPanel = gallery.ItemsHostSite as RibbonGalleryCategoriesPanel;
                if (categoriesPanel != null)
                {
                    IContainsStarLayoutManager iContainsStarLayoutManager = (IContainsStarLayoutManager)categoriesPanel;
                    if (iContainsStarLayoutManager.StarLayoutManager != null)
                        return !iContainsStarLayoutManager.StarLayoutManager.IsStarLayoutPass;
                }
            }
 
            return false;
        }
 
        private void AddScrollDeltaInfo(double sumOfHeight, int childrenCount)
        {
            RibbonGalleryCategory category = Category;
            if (category != null)
            {
                // Adding virtual count of items and cumulative height to RGC for the purpose of calcualting 
                // avg height as scrolling delta in RibbonGalleryCategoriesPanel.
                category.averageItemHeightInfo.Count = childrenCount;
                category.averageItemHeightInfo.CumulativeHeight = sumOfHeight;
            }
        }
 
        // Minimum number of Col must be shown is determined by this method
        private int GetMinColumnCount()
        {
            RibbonGalleryCategory category = Category;
            if (category != null)
            {
                return (int)category.MinColumnCount;
            }
            return 1;
        }
 
        // Maximum number of Col must be shown is determined by this method
        private int GetMaxColumnCount()
        {
            RibbonGalleryCategory category = Category;
            if (category != null)
            {
                return (int)category.MaxColumnCount;
            }
 
            return int.MaxValue;
        }
 
        // Gets the MaxColumnWidth if the shared scope is Gallery
        // then MaxColumnWidth is used which is defined on Gallery to lay out all Items as Uniform width columns in scope of Gallery
        // otherwise they layout as Uniform width column within the scope of current Category only.
        private double GetMaxColumnWidth()
        {
            RibbonGallery gallery = Gallery;
            if (gallery != null && SharedColumnSizeScopeAtGalleryLevel)
            {
                return gallery.MaxColumnWidth;
            }
 
            return _maxColumnWidth;
        }
 
#if IN_RIBBON_GALLERY
        // MaxItemHeight is the maximum height of an Item among all the Items of all categories in the gallery this panel is part of.
        // MaximumItemHeight is used by InRibbonGallery to find the Item Box for uniformity.
        private double GetMaxItemHeight()
        {
            RibbonGallery gallery = Gallery;
            if (gallery != null)
            {
                return gallery.MaxItemHeight;
            }
 
            return _maxItemHeight;
        }
#endif
 
        // Sets the MaxColumnWidth on Gallery if the shared scope is Gallery
        private void SetMaxColumnWidth(double value)
        {
            MaxColumnWidth = value;
 
            RibbonGallery gallery = Gallery;
            if (gallery != null)
            {
                if (SharedColumnSizeScopeAtGalleryLevel)
                {
                    if (gallery.MaxColumnWidth < value || !gallery.IsMaxColumnWidthValid)
                    {
                        gallery.MaxColumnWidth = value;
                        gallery.IsMaxColumnWidthValid = true;
                    }
                }
            }
        }
 
#if IN_RIBBON_GALLERY
        private void SetMaxColumnWidthAndHeight(double maxWidth, double maxHeight)
        {
            SetMaxColumnWidth(maxWidth);
 
            RibbonGallery gallery = Gallery;
            if (gallery != null)
            {            
                if (gallery.MaxItemHeight < maxHeight)
                {
                    gallery.MaxItemHeight = maxHeight;
                }
            }
        }
#endif
 
        private double GetArrangeWidth()
        {
            RibbonGallery gallery = Gallery;
            if (gallery != null && SharedColumnSizeScopeAtGalleryLevel)
            {
                return gallery.ArrangeWidth;
            }
 
            return _arrangeWidth;
        }
 
        // Sets the ArrangeWidth on Gallery if the shared scope is Gallery
        private void SetArrangeWidth(double value)
        {
            _arrangeWidth = value;
 
            RibbonGallery gallery = Gallery;
            if (gallery != null && SharedColumnSizeScopeAtGalleryLevel)
            {
                gallery.ArrangeWidth = value;
                gallery.IsArrangeWidthValid = true;
            }
        }
 
        #endregion Private Methods
 
        #region Protected Overrides
        
        // There are three different scenarios:
        // 1. In InRibbonGalleryMode where the gallery is shown within Ribbon via InRibbonGallery and Measure becomes responsibility 
        //    of InRibbonGalleryModeMeasureOverride.
        // 2. Popup mode : it's the common and RealMeasureOverride which is the common case of RibbonGallery hosted normally in Popup
        //    Weather it belongs to InRibbonGallery or any other control containing Gallery like RibbonComboBox/RibbonMenuButton etc.
        // 3. Popup mode also utilizes AutoPass to be understood by hosting Control like RibbonComboBox/RibbonMenuButton etc.
        protected override Size MeasureOverride(Size availableSize)
        {
            RibbonGallery gallery = Gallery;
#if IN_RIBBON_GALLERY
            if (gallery != null && gallery.IsInInRibbonGalleryMode())
            {
                return InRibbonGalleryModeMeasureOverride(availableSize);
            }
            else
            {
#endif
                return RealMeasureOverride(availableSize);
#if IN_RIBBON_GALLERY
            }
#endif
        }
 
#if IN_RIBBON_GALLERY
        // InRibbonGalleryMode where the gallery is shown within Ribbon via InRibbonGallery and Measure becomes responsibility 
        // of InRibbonGalleryModeMeasureOverride.
        private Size InRibbonGalleryModeMeasureOverride(Size availableSize)
        {
            UIElementCollection children = InternalChildren;
            Size panelSize = new Size();
            Size childConstraint = new Size(double.PositiveInfinity, double.PositiveInfinity);
            double maxItemHeight = 0;
            double maxColumnWidth = 0;
            double sumItemHeight = 0;
            int columnCount = 0;
            int childrenCount = children.Count;
            int minColumnCount = GetMinColumnCount();
            int maxColumnCount = GetMaxColumnCount();
 
            Debug.Assert(maxColumnCount >= minColumnCount);
 
            // Determine the maximum column width so that all items 
            // can be hosted in equispaced columns. row height is auto
            // and depends on the maximum height of the items in that row
            for (int i = 0; i < childrenCount; i++)
            {
                UIElement child = children[i] as UIElement;
 
                // It has been already measure once directly in RibbonCategoriesPanel and hence will get short circuit already.
                // The call still needed as to Measure in case if the Parent panel is being changed from 
                // RibbonGalleryCatgoriesPanel to something else.
                child.Measure(childConstraint);
 
                Size childSize = child.DesiredSize;
                maxColumnWidth = Math.Max(maxColumnWidth, childSize.Width);
                maxItemHeight = Math.Max(maxItemHeight, childSize.Height);
                sumItemHeight += childSize.Height;
            }
 
            // It has been already calculated once directly in RibbonCategoriesPanel and hence will get short circuit already.
            // The call still needs to be made in case if the Parent panel is being changed from RibbonGalleryCatgoriesPanel 
            // to something else.
            SetMaxColumnWidthAndHeight(maxColumnWidth, maxItemHeight);
 
            // Gets the final MaxColumnWidth for this panel.
            maxColumnWidth = GetMaxColumnWidth();
            maxItemHeight = GetMaxItemHeight();
 
            AddScrollDeltaInfo(sumItemHeight, childrenCount);
            
            if (!double.IsInfinity(availableSize.Width) && maxColumnWidth != 0)
            {
                columnCount = (int)(availableSize.Width / maxColumnWidth);
            }
            else
            {
                columnCount = childrenCount;
            }
 
            // Initialize Width and Height, specially in case of height it's more off leaving the empty space in panel where Items in other categories
            // are to be rendered.
            panelSize.Width = columnCount * maxColumnWidth;
 
            RibbonGalleryCategory category = Category;
            if (category != null)
            {
                panelSize.Height = category.RowOffset * maxItemHeight;
 
                if (columnCount != 0)
                {
                    // There could be space left in the last row of the previous category to render some of the items in this category. 
                    Debug.Assert(category.ColumnOffset < columnCount);
                    int spotsLeftInPreviousRow = category.ColumnOffset == 0 ? 0 : columnCount - category.ColumnOffset;
 
                    if (childrenCount > spotsLeftInPreviousRow)
                    {
                        int remainingChildren = childrenCount - spotsLeftInPreviousRow;
                        category.ColumnEndOffSet = remainingChildren % columnCount;
 
                        category.RowCount = remainingChildren / columnCount;  // Initialize RowCount as the number of full rows.
 
                        // Increment RowCount if we filled the previous row.
                        if (spotsLeftInPreviousRow > 0)
                            category.RowCount++;
 
                        // Increment RowCount if we started a new row.
                        if (category.ColumnEndOffSet > 0)
                            category.RowCount++;
                    }
                    else
                    {
                        // All children can fit in the previous row.
 
                        category.ColumnEndOffSet = (category.ColumnOffset + childrenCount) % columnCount;
                        category.RowCount = 1;
                    }
                }
                else
                {
                    category.ColumnEndOffSet = 0;
                    category.RowCount = 0;
                }
 
                panelSize.Height += category.RowCount * maxItemHeight;
            }
 
            return panelSize;
        }
#endif
 
        private Size RealMeasureOverride(Size availableSize)
        {
            // Iterate through all of the children. For each row first measure # of children
            // to infinity and gather their Max of DesiredWidths. Also, calculate the 
            // columnCount. Then space permitting measure as many more 
            // children that will fit into that row till columnCount is acheived and get 
            // MaxHeight for that Row. Cumulative height of such Rows gives you DesiredHeight
            // for Panel.
            // Return desired size for this Panel as the 
            // new Size(ColumnCount * MaxColumnWidth, cumulative RowHeights).
 
            UIElementCollection children = InternalChildren;
            Size panelSize = new Size();
            Size childConstraint = new Size(double.PositiveInfinity, double.PositiveInfinity);
            double maxRowHeight = 0;
            double maxItemHeight = 0;
            double sumItemHeight = 0;
            int columnCount = 0;
            int childrenCount = children.Count;
            RibbonGallery parentGallery = Gallery;
            double maxColumnWidth = (parentGallery != null && parentGallery.IsMaxColumnWidthValid) ? GetMaxColumnWidth() : 0.0;
            int minColumnCount = GetMinColumnCount();
            int maxColumnCount = GetMaxColumnCount();
 
            Debug.Assert(maxColumnCount >= minColumnCount);
            
            // Determine the maximum column width so that all items 
            // can be hosted in equispaced columns. row height is auto
            // and depends on the maximum height of the items in that row
            for (int i = 0; i < childrenCount; i++)
            {
                UIElement child = children[i] as UIElement;
                child.Measure(childConstraint);
                Size childSize = child.DesiredSize;
                maxColumnWidth = Math.Max(maxColumnWidth, childSize.Width);
                maxItemHeight = Math.Max(maxItemHeight, childSize.Height);
                sumItemHeight += childSize.Height;
            }
 
            // if none of the children has substantial width, panelsize would be equivalent to zero.
            if (maxColumnWidth == 0.0)
            {
                return panelSize;
            }
 
            // Updates the MaxColumnWidth of this Category as well as the of the parent Gallery if suffices all conditions.
            SetMaxColumnWidth(maxColumnWidth);
 
            // Gets the final MaxColumnWidth for this panel.
            maxColumnWidth = GetMaxColumnWidth();
            AddScrollDeltaInfo(sumItemHeight, childrenCount);
 
            if (!IsAutoLayoutPass())
            {
                if (!double.IsInfinity(availableSize.Width))
                {
                    columnCount = Math.Min(Math.Max(minColumnCount, Math.Min((int)(availableSize.Width / maxColumnWidth), childrenCount)), maxColumnCount);
                    
                    RibbonGalleryCategory category = Category;
                    if (parentGallery != null && category != null)
                    {
                        if (SharedColumnSizeScopeAtGalleryLevel && parentGallery.ColumnsStretchToFill)
                        {
                            // Since Gallery is a SharedColumnScope, store ArrangeWidth to be shared by the entire Gallery
                            double arrangeWidth = GetArrangeWidth();
                            if (!parentGallery.IsArrangeWidthValid)
                            {
                                // Calculate ArrangeWidth such that columnCount no. of columns occupy all of availableSize.Width
                                columnCount = Math.Min(Math.Max(minColumnCount, Math.Min((int)(availableSize.Width / maxColumnWidth), childrenCount)), maxColumnCount);
                                arrangeWidth = Math.Max(availableSize.Width / columnCount, maxColumnWidth);
                                SetArrangeWidth(arrangeWidth);
                            }
                            else
                            {
                                // Once a valid arrangeWidth has been computed, we use arrangeWidth to determine number of columns. 
                                columnCount = Math.Min(Math.Max(minColumnCount, Math.Min((int)(availableSize.Width / arrangeWidth), childrenCount)), maxColumnCount);
                            }
                        }
                        else if (!SharedColumnSizeScopeAtGalleryLevel && category.ColumnsStretchToFill)
                        {
                            // Since category is a sharedColumnScope. 
                            // Calculate and store _arrangeWidth locally for just this category
                            if (!_isArrangeWidthValid)
                            {
                                columnCount = Math.Min(Math.Max(minColumnCount, Math.Min((int)(availableSize.Width / maxColumnWidth), childrenCount)), maxColumnCount);
                                _arrangeWidth = Math.Max(availableSize.Width / columnCount, maxColumnWidth);
                                _isArrangeWidthValid = true;
                            }
                            else
                            {
                                // Once a valid arrangeWidth has been computed, we use arrangeWidth to determine number of columns. 
                                columnCount = Math.Min(Math.Max(minColumnCount, Math.Min((int)(availableSize.Width / _arrangeWidth), childrenCount)), maxColumnCount);
                            }
                        }
                    }
                }
                else
                {
                    columnCount = Math.Max(minColumnCount, Math.Min(childrenCount, maxColumnCount));
                }
 
                // Finds row Items once ColumnWidth is determined to fetch MaxHeight of a particular row.
                // Also adds these height to acheive cumulative height which is desired height of the panel.
                for (int i = 0; i < childrenCount; i++)
                {
                    UIElement child = children[i] as UIElement;
                    Size childSize = child.DesiredSize;
                    maxRowHeight = Math.Max(maxRowHeight, childSize.Height);
                    if ((i + 1) % columnCount == 0 || i == childrenCount - 1)
                    {
                        panelSize.Height += maxRowHeight;
 
                        // Save the maxRowHeight so it can be used for PageDown operations
                        _maxRowHeight = maxRowHeight;
 
                        maxRowHeight = 0;
                    }
                }
            }
            else
            {
                columnCount = minColumnCount;
                panelSize.Height = maxItemHeight;
            }
 
            panelSize.Width = columnCount * maxColumnWidth;
            return panelSize;
        }
 
        // There are two different scenarios:
        // 1. In InRibbonGalleryMode where the gallery is shown within Ribbon via InRibbonGallery and Arrange becomes responsibility 
        //    of InRibbonGalleryModeMeasureOverride.
        // 2. Popup mode : it's the common and RealArrangeOverride which is the common case of RibbonGallery hosted normally in Popup
        //    weather it belongs to InRibbonGallery or any other control containing Gallery like RibbonComboBox/RibbonMenuButton etc.
        protected override Size ArrangeOverride(Size finalSize)
        {
            RibbonGallery gallery = Gallery;
#if IN_RIBBON_GALLERY
            if (gallery != null && gallery.IsInInRibbonGalleryMode())
            {
                return InRibbonGalleryModeArrangeOverride(finalSize);
            }
            else
            {
#endif
                return RealArrangeOverride(finalSize);
#if IN_RIBBON_GALLERY
            }
#endif
        }
 
#if IN_RIBBON_GALLERY
        private Size InRibbonGalleryModeArrangeOverride(Size finalSize)
        {
            // Get final coumn count by finalsizw.width , MaxColumnWidth
            // Iterate through children one row at a time. 
            // Arrange the first row at offset 0 and the next 
            // row just below that and so on. Besure to arrange 
            // the children within each row uniformly based on 
            // the MaxColumnWidth that was computed during the 
            // Measure pass.
 
            UIElementCollection children = InternalChildren;
            double rowStartHeight = 0;
            double rowStartWidth = 0;
            int finalColumnCount = 0;
            int childrenCount = children.Count;
            double maxColumnWidth = GetMaxColumnWidth();
            double maxItemHeight = GetMaxItemHeight();
 
            //Calculate the available column count based on final space
            //keeping the same column width. 
            if (maxColumnWidth == 0.0)
                return finalSize;
 
            if (!double.IsInfinity(finalSize.Width) && maxColumnWidth != 0)
            {
                finalColumnCount = (int)(finalSize.Width / maxColumnWidth);
            }
            else
            {
                finalColumnCount = childrenCount;
            }
            
            if (finalColumnCount == 0)
            {
                return finalSize;
            }
 
            RibbonGalleryCategory category = Category;
 
            // Calculate the starting offsets in pixels from actual Row and Coulmn offsets.
            if (category != null)
            {
                rowStartHeight = category.RowOffset * maxItemHeight;
                rowStartWidth = category.ColumnOffset * maxColumnWidth;
            }
 
            for (int i = 0; i < childrenCount; i++)
            {
                children[i].Arrange(new Rect(rowStartWidth, rowStartHeight, maxColumnWidth, maxItemHeight));
                rowStartWidth += maxColumnWidth;
                if ((i + category.ColumnOffset + 1) % finalColumnCount == 0)
                {
                    rowStartHeight += maxItemHeight;
                    rowStartWidth = 0;
                }
            }
 
            return finalSize;
        }
#endif
 
        private Size RealArrangeOverride(Size finalSize)
        {
            // Get final coumn count by finalsizw.width , MaxColumnWidth
            // Iterate through children one row at a time. 
            // Arrange the first row at offset 0 and the next 
            // row just below that and so on. Besure to arrange 
            // the children within each row uniformly based on 
            // the MaxColumnWidth that was computed during the 
            // Measure pass.
 
            UIElementCollection children = InternalChildren;
            double rowStartHeight = 0;
            double rowStartWidth = 0; 
            double maxRowHeight = 0.0;
            int finalColumnCount = 0;
            int rowStartIndex = 0;
            int childrenCount = children.Count;
            int minColumnCount = GetMinColumnCount();
            int maxColumnCount = GetMaxColumnCount();
            RibbonGallery parentGallery = Gallery;
            RibbonGalleryCategory category = Category;
            double arrangeWidth = 0.0;
 
            if (parentGallery != null && category != null)
            {
                if (SharedColumnSizeScopeAtGalleryLevel && parentGallery.ColumnsStretchToFill)
                {
                    // If sharedScope is Gallery, fetch global ArrangeWidth
                    arrangeWidth = GetArrangeWidth();
                }
                else if (category.IsSharedColumnSizeScope && category.ColumnsStretchToFill)
                {
                    // SharedScope is Category, use local arrangeWidth.
                    arrangeWidth = _arrangeWidth;
                }
                else
                {
                    // ColumnStretchToFill is false. 
                    arrangeWidth = GetMaxColumnWidth();
                }
            }
 
            //Calculate the available column count based on final space
            //keeping the same column width. 
            if (arrangeWidth == 0.0)
                return finalSize;
 
            finalColumnCount = Math.Max(minColumnCount, Math.Min((int)(finalSize.Width / arrangeWidth), maxColumnCount));
 
            if (finalColumnCount == 0)
            {
                return finalSize;
            }
 
            for (int i = 0; i < childrenCount; i++)
            {
                maxRowHeight = Math.Max(maxRowHeight, children[i].DesiredSize.Height);
                if ((i + 1) % finalColumnCount == 0 || i == childrenCount-1)
                {
                    //Arrange the row
                    for (int j = rowStartIndex; j <= i; j++)
                    {
                        children[j].Arrange(new Rect(rowStartWidth, rowStartHeight, arrangeWidth, maxRowHeight));
                        rowStartWidth += arrangeWidth;
                    }
                    rowStartHeight += maxRowHeight;
                    maxRowHeight = 0;
                    rowStartIndex = i + 1;
                    rowStartWidth = 0;
                }
            }
 
            return finalSize;
        }
 
        #endregion Protected Overrides
 
        #region Data
 
        private RibbonGalleryCategory Category
        {
            get
            {
                return (RibbonGalleryCategory)ItemsControl.GetItemsOwner(this);
            }
        }
		
        private RibbonGallery Gallery
        {
            get
            {
                RibbonGalleryCategory category = this.Category;
                if (category != null)
                {
                    return category.RibbonGallery;
                }
                return null;
            }
        }
 
        // Shared column size scope can be at either the gallery or category level.
        // Gallery level is the default.
        private bool SharedColumnSizeScopeAtGalleryLevel
        {
            get
            {
                RibbonGalleryCategory category = Category;
                if (category != null &&
                    category.IsSharedColumnSizeScope)
                {
                    return false;
                }
 
                return true;
            }
        }
 
        internal double MaxColumnWidth
        {
            get { return _maxColumnWidth; }
            private set
            {
                if (_maxColumnWidth != value)
                {
                    _maxColumnWidth = value;
                    _isArrangeWidthValid = false;
                }
            }
        }
 
        internal double MaxRowHeight
        {
            get { return _maxRowHeight; }
        }
 
        // this is local value of maxColumnWidth per category
        private double _maxColumnWidth = 0, _arrangeWidth = 0.0;
        private double _maxRowHeight = 0;
#if IN_RIBBON_GALLERY
        private double _maxItemHeight = 0;
#endif
        private bool _isArrangeWidthValid = false;
 
        #endregion
    }
}