File: MS\Internal\Data\CollectionViewGroupInternal.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: A CollectionViewGroupInternal, as created by a CollectionView according to a GroupDescription.
//
// See spec at Grouping.mht
//
 
using System.Collections;       // IEnumerator
using System.ComponentModel;    // PropertyChangedEventArgs, GroupDescription
using System.Windows;           // DependencyProperty.UnsetValue
using System.Windows.Data;      // CollectionViewGroup
using System.Windows.Threading; // Dispatcher
using System.Collections.Generic;
 
namespace MS.Internal.Data
{
    internal class CollectionViewGroupInternal : CollectionViewGroup
    {
        #region Constructors
 
        //------------------------------------------------------
        //
        //  Constructors
        //
        //------------------------------------------------------
 
        internal CollectionViewGroupInternal(object name, CollectionViewGroupInternal parent, bool isExplicit = false) : base(name)
        {
            _parentGroup = parent;
            _isExplicit = isExplicit;
        }
 
        #endregion Constructors
 
        #region Public Properties
 
        //------------------------------------------------------
        //
        //  Public Properties
        //
        //------------------------------------------------------
 
        /// <summary>
        /// Is this group at the bottom level (not further subgrouped).
        /// </summary>
        public override bool IsBottomLevel
        {
            get { return (_groupBy == null); }
        }
 
        #endregion  Public Properties
 
        #region Internal Properties
 
        //------------------------------------------------------
        //
        //  Internal Properties
        //
        //------------------------------------------------------
 
        // how this group divides into subgroups
        internal GroupDescription GroupBy
        {
            get { return _groupBy; }
            set
            {
                bool oldIsBottomLevel = IsBottomLevel;
 
                if (_groupBy != null)
                {
                    PropertyChangedEventManager.RemoveHandler(_groupBy, OnGroupByChanged, String.Empty);
                }
 
                _groupBy = value;
 
                if (_groupBy != null)
                {
                    PropertyChangedEventManager.AddHandler(_groupBy, OnGroupByChanged, String.Empty);
                }
 
                // choose a comparer based on info in the GroupDescription and the owning collection view
                _groupComparer = (_groupBy == null) ? null :
                    ListCollectionView.PrepareComparer(
                        _groupBy.CustomSort,
                        _groupBy.SortDescriptionsInternal,
                        static state =>
                        {
                            for (CollectionViewGroupInternal group = (CollectionViewGroupInternal)state;
                                 group != null;
                                 group = group.Parent)
                            {
                                CollectionViewGroupRoot root = group as CollectionViewGroupRoot;
                                if (root != null)
                                {
                                    return root.View;
                                }
                            }
                            return null;    // this should never happen - root should always be present
                        }, this);
 
                if (oldIsBottomLevel != IsBottomLevel)
                {
                    OnPropertyChanged(new PropertyChangedEventArgs("IsBottomLevel"));
                }
            }
        }
 
        // the number of items and groups in the subtree under this group
        internal int FullCount
        {
            get { return _fullCount; }
            set { _fullCount = value; }
        }
 
        // the most recent index where actvity took place
        internal int LastIndex
        {
            get { return _lastIndex; }
            set { _lastIndex = value; }
        }
 
        // the first item (leaf) added to this group.  If this can't be determined,
        // DependencyProperty.UnsetValue.
        internal object SeedItem
        {
            get
            {
                if (ItemCount > 0 && (GroupBy == null || GroupBy.GroupNames.Count == 0))
                {
                    // look for first item, child by child
                    for (int k = 0, n = Items.Count; k < n; ++k)
                    {
                        CollectionViewGroupInternal subgroup = Items[k] as CollectionViewGroupInternal;
                        if (subgroup == null)
                        {
                            // child is an item - return it
                            return Items[k];
                        }
                        else if (subgroup.ItemCount > 0)
                        {
                            // child is a nonempty subgroup - ask it
                            return subgroup.SeedItem;
                        }
                        // otherwise child is an empty subgroup - go to next child
                    }
 
                    // we shouldn't get here, but just in case...
                    return DependencyProperty.UnsetValue;
                }
                else
                {
                    // the group is empty, or it has explicit subgroups.
                    // In either case, we cannot determine the first item -
                    // it could have gone into any of the subgroups.
                    return DependencyProperty.UnsetValue;
                }
            }
        }
 
        #endregion Properties
 
        #region Internal methods
 
        //------------------------------------------------------
        //
        //  Internal methods
        //
        //------------------------------------------------------
 
        internal void Add(object item)
        {
            if (_groupComparer == null)
            {
                ChangeCounts(item, +1);
                ProtectedItems.Add(item);
            }
            else
            {
                Insert(item, null, null);
            }
        }
 
        internal int Remove(object item, bool returnLeafIndex)
        {
            int index = -1;
            int localIndex = ProtectedItems.IndexOf(item);
 
            if (localIndex >= 0)
            {
                if (returnLeafIndex)
                {
                    index = LeafIndexFromItem(null, localIndex);
                }
 
                CollectionViewGroupInternal subGroup = item as CollectionViewGroupInternal;
                if (subGroup != null)
                {
                    subGroup.Clear();
 
                    // Remove from the name to group map.
                    RemoveSubgroupFromMap(subGroup);
                }
 
                ChangeCounts(item, -1);
 
                // ChangeCounts may clear this group, if it is now empty.
                // In that case, don't use localIndex - it's now out of range.
                if (ProtectedItems.Count > 0)
                {
                    ProtectedItems.RemoveAt(localIndex);
                }
            }
 
            return index;
        }
 
        internal void Clear()
        {
            // reset the counts before delving into subgroups.   The subgroup
            // changes can incur re-entrant calls to LeafAt(index) which will
            // get out-of-range exceptions, unless we fend them
            // off by ensuring that count<=index.
            FullCount = 1;
            ProtectedItemCount = 0;
 
            if (_groupBy != null)
            {
                // This group has subgroups.  Disconnect from GroupDescription events
                PropertyChangedEventManager.RemoveHandler(_groupBy, OnGroupByChanged, String.Empty);
                _groupBy = null;
 
                // recursively clear subgroups
                for (int i = 0, n = ProtectedItems.Count; i < n; ++i)
                {
                    CollectionViewGroupInternal subGroup = ProtectedItems[i] as CollectionViewGroupInternal;
                    if (subGroup != null)
                    {
                        subGroup.Clear();
                    }
                }
            }
 
            ProtectedItems.Clear();
            if (_nameToGroupMap != null)
            {
                _nameToGroupMap.Clear();
            }
        }
 
        // return the index of the given item within the list of leaves governed
        // by this group
        internal int LeafIndexOf(object item)
        {
            int leaves = 0;         // number of leaves we've passed over so far
            for (int k = 0, n = Items.Count; k < n; ++k)
            {
                CollectionViewGroupInternal subgroup = Items[k] as CollectionViewGroupInternal;
                if (subgroup != null)
                {
                    int subgroupIndex = subgroup.LeafIndexOf(item);
                    if (subgroupIndex < 0)
                    {
                        leaves += subgroup.ItemCount;       // item not in this subgroup
                    }
                    else
                    {
                        return (leaves + subgroupIndex);    // item is in this subgroup
                    }
                }
                else
                {
                    // current item is a leaf - compare it directly
                    if (System.Windows.Controls.ItemsControl.EqualsEx(item, Items[k]))
                    {
                        return leaves;
                    }
                    else
                    {
                        leaves += 1;
                    }
                }
            }
 
            // item not found
            return -1;
        }
 
        // return the index of the given item within the list of leaves governed
        // by the full group structure.  The item must be a (direct) child of this
        // group.  The caller provides the index of the item within this group,
        // if known, or -1 if not.
        internal int LeafIndexFromItem(object item, int index)
        {
            int result = 0;
 
            // accumulate the number of predecessors at each level
            for (CollectionViewGroupInternal group = this;
                    group != null;
                    item = group, group = group.Parent, index = -1)
            {
                // accumulate the number of predecessors at the level of item
                for (int k = 0, n = group.Items.Count; k < n; ++k)
                {
                    // if we've reached the item, move up to the next level
                    if ((index < 0 && System.Windows.Controls.ItemsControl.EqualsEx(item, group.Items[k])) ||
                        index == k)
                    {
                        break;
                    }
 
                    // accumulate leaf count
                    CollectionViewGroupInternal subgroup = group.Items[k] as CollectionViewGroupInternal;
                    result += (subgroup == null) ? 1 : subgroup.ItemCount;
                }
            }
 
            return result;
        }
 
 
        // return the item at the given index within the list of leaves governed
        // by this group
        internal object LeafAt(int index)
        {
            for (int k = 0, n = Items.Count; k < n; ++k)
            {
                CollectionViewGroupInternal subgroup = Items[k] as CollectionViewGroupInternal;
                if (subgroup != null)
                {
                    // current item is a group - either drill in, or skip over
                    if (index < subgroup.ItemCount)
                    {
                        return subgroup.LeafAt(index);
                    }
                    else
                    {
                        index -= subgroup.ItemCount;
                    }
                }
                else
                {
                    // current item is a leaf - see if we're done
                    if (index == 0)
                    {
                        return Items[k];
                    }
                    else
                    {
                        index -= 1;
                    }
                }
            }
 
            // the loop should have found the index.  We shouldn't get here.
            throw new ArgumentOutOfRangeException("index");
        }
 
        // return an enumerator over the leaves governed by this group
        internal IEnumerator GetLeafEnumerator()
        {
            return new LeafEnumerator(this);
        }
 
        // insert a new item or subgroup and return its index.  Seed is a
        // representative from the subgroup (or the item itself) that
        // is used to position the new item/subgroup w.r.t. the order given
        // by the comparer.  (If comparer is null, just add at the end).
        internal int Insert(object item, object seed, IComparer comparer)
        {
            // when group sorting is not declared,
            // never insert the new item/group before the explicit subgroups
            int low = 0;
            if (_groupComparer == null && GroupBy != null)
            {
                low = GroupBy.GroupNames.Count;
            }
 
            int index = FindIndex(item, seed, comparer, low, ProtectedItems.Count);
 
            // now insert the item
            ChangeCounts(item, +1);
            ProtectedItems.Insert(index, item);
 
            return index;
        }
 
        protected virtual int FindIndex(object item, object seed, IComparer comparer, int low, int high)
        {
            int index;
 
            if (_groupComparer == null)
            {
                // group sorting is not declared - find the position using the seed
                if (comparer != null)
                {
                    IListComparer ilc = comparer as IListComparer;
                    if (ilc != null)
                    {
                        // reset the IListComparer before each search.  This cannot be done
                        // any less frequently (e.g. in Root.AddToSubgroups), due to the
                        // possibility that the item may appear in more than one subgroup.
                        ilc.Reset();
                    }
 
                    for (index = low; index < high; ++index)
                    {
                        CollectionViewGroupInternal subgroup = ProtectedItems[index] as CollectionViewGroupInternal;
                        object seed1 = (subgroup != null) ? subgroup.SeedItem : ProtectedItems[index];
                        if (seed1 == DependencyProperty.UnsetValue)
                            continue;
                        if (comparer.Compare(seed, seed1) < 0)
                            break;
                    }
                }
                else
                {
                    index = high;
                }
            }
            else
            {
                // group sorting is declared - find the position using the local comparer
                for (index = low; index < high; ++index)
                {
                    if (_groupComparer.Compare(item, ProtectedItems[index]) < 0)
                        break;
                }
            }
 
            return index;
        }
 
        // move an item and return true if it really moved.
        // Also translate the indices to "leaf" coordinates
        internal bool Move(object item, IList list, ref int oldIndex, ref int newIndex)
        {
            int oldIndexLocal = -1, newIndexLocal = -1;
            int localIndex = 0;
            int n = ProtectedItems.Count;
 
            // the input is in "full" coordinates.  Find the corresponding local coordinates
            for (int fullIndex = 0; ; ++fullIndex)
            {
                if (fullIndex == oldIndex)
                {
                    oldIndexLocal = localIndex;
                    if (newIndexLocal >= 0)
                        break;
                    ++localIndex;
                }
 
                if (fullIndex == newIndex)
                {
                    newIndexLocal = localIndex;
                    if (oldIndexLocal >= 0)
                    {
                        --newIndexLocal;
                        break;
                    }
                    ++fullIndex;
                    ++oldIndex;
                }
 
                if (localIndex < n && System.Windows.Controls.ItemsControl.EqualsEx(ProtectedItems[localIndex], list[fullIndex]))
                    ++localIndex;
            }
 
            // the move may be a no-op w.r.t. this group
            if (oldIndexLocal == newIndexLocal)
                return false;
 
            // translate to "leaf" coordinates
            int low, high, lowLeafIndex, delta = 0;
            if (oldIndexLocal < newIndexLocal)
            {
                low = oldIndexLocal + 1;
                high = newIndexLocal + 1;
                lowLeafIndex = LeafIndexFromItem(null, oldIndexLocal);
            }
            else
            {
                low = newIndexLocal;
                high = oldIndexLocal;
                lowLeafIndex = LeafIndexFromItem(null, newIndexLocal);
            }
 
            for (int i = low; i < high; ++i)
            {
                CollectionViewGroupInternal subgroup = Items[i] as CollectionViewGroupInternal;
                delta += (subgroup == null) ? 1 : subgroup.ItemCount;
            }
 
            if (oldIndexLocal < newIndexLocal)
            {
                oldIndex = lowLeafIndex;
                newIndex = oldIndex + delta;
            }
            else
            {
                newIndex = lowLeafIndex;
                oldIndex = newIndex + delta;
            }
 
            // do the actual move
            ProtectedItems.Move(oldIndexLocal, newIndexLocal);
            return true;
        }
 
        // the group's description has changed - notify parent
        protected virtual void OnGroupByChanged()
        {
            if (Parent != null)
                Parent.OnGroupByChanged();
        }
 
        /// <summary>
        ///     Maps the given name with the given subgroup
        /// </summary>
        internal void AddSubgroupToMap(object nameKey, CollectionViewGroupInternal subgroup)
        {
            Debug.Assert(subgroup != null);
 
            // Use null name place holder.
            nameKey ??= s_nullGroupNameKey;
 
            // The dictionary is not initialized until first addition 
            _nameToGroupMap ??= new Dictionary<object, WeakReference>();
 
            // Add to the map. Use WeakReference to avoid memory leaks
            // in case some one calls ProtectedItems.Remove instead of
            // CollectionViewGroupInternal.Remove
            _nameToGroupMap[nameKey] = new WeakReference(subgroup);
 
            ScheduleMapCleanup();
        }
 
        /// <summary>
        ///     Removes the given subgroup from the name to group map.
        /// </summary>
        private void RemoveSubgroupFromMap(CollectionViewGroupInternal subgroup)
        {
            Debug.Assert(subgroup != null);
 
            if (_nameToGroupMap == null)
                return;
 
            // Search for the subgroup in the map.
            foreach (KeyValuePair<object, WeakReference> item in _nameToGroupMap)
            {
                if (item.Value.Target == subgroup)
                {
                    _nameToGroupMap.Remove(item.Key);
                    break;
                }
            }
 
            ScheduleMapCleanup();
        }
 
        /// <summary>
        ///     Tries to find the subgroup for the name from the map.
        /// </summary>
        internal CollectionViewGroupInternal GetSubgroupFromMap(object nameKey)
        {
            if (_nameToGroupMap != null)
            {
                // Use null name place holder.
                nameKey ??= s_nullGroupNameKey;
 
                // Find and return the subgroup
                if (_nameToGroupMap.TryGetValue(nameKey, out WeakReference weakRef))
                {
                    return weakRef.Target as CollectionViewGroupInternal;
                }
            }
 
            return null;
        }
 
        /// <summary>
        ///     Schedules a dispatcher operation to clean up the map
        ///     of garbage collected weak references.
        /// </summary>
        private void ScheduleMapCleanup()
        {
            if (!_mapCleanupScheduled)
            {
                _mapCleanupScheduled = true;
                Dispatcher.CurrentDispatcher.BeginInvoke(() =>
                {
                    _mapCleanupScheduled = false;
                    if (_nameToGroupMap != null)
                    {
                        foreach (KeyValuePair<object, WeakReference> item in _nameToGroupMap)
                        {
                            if (!item.Value.IsAlive)
                            {
                                _nameToGroupMap.Remove(item.Key);
                            }
                        }
                    }
                }, DispatcherPriority.ContextIdle);
            }
        }
 
        #endregion Internal methods
 
        #region Internal Types
 
        // this comparer is used to insert an item into a group in a position consistent
        // with a given IList.  It only works when used in the pattern that FindIndex
        // uses, namely first call Reset(), then call Compare(item, x) any number of
        // times with the same item (the new item) as the first argument, and a sequence
        // of x's as the second argument that appear in the IList in the same sequence.
        // This makes the total search time linear in the size of the IList.  (To give
        // the correct answer regardless of the sequence of arguments would involve
        // calling IndexOf and leads to O(N^2) total search time.)
 
        internal class IListComparer : IComparer
        {
            internal IListComparer(IList list)
            {
                ResetList(list);
            }
 
            internal void Reset()
            {
                _index = 0;
            }
 
            internal void ResetList(IList list)
            {
                _list = list;
                _index = 0;
            }
 
            public int Compare(object x, object y)
            {
                if (System.Windows.Controls.ItemsControl.EqualsEx(x, y))
                    return 0;
 
                // advance the index until seeing one x or y
                int n = (_list != null) ? _list.Count : 0;
                for (; _index < n; ++_index)
                {
                    object z = _list[_index];
                    if (System.Windows.Controls.ItemsControl.EqualsEx(x, z))
                    {
                        return -1;  // x occurs first, so x < y
                    }
                    else if (System.Windows.Controls.ItemsControl.EqualsEx(y, z))
                    {
                        return +1;  // y occurs first, so x > y
                    }
                }
 
                // if we don't see either x or y, declare x > y.
                // This has the effect of putting x at the end of the list.
                return +1;
            }
 
            int _index;
            IList _list;
        }
 
        #endregion Internal Types
 
        #region Private Properties
 
        //------------------------------------------------------
        //
        //  Private Properties
        //
        //------------------------------------------------------
 
        internal CollectionViewGroupInternal Parent
        {
            get { return _parentGroup; }
        }
 
        private bool IsExplicit
        {
            get { return _isExplicit; }
        }
 
        #endregion Private Properties
 
        #region Private methods
 
        //------------------------------------------------------
        //
        //  Private methods
        //
        //------------------------------------------------------
 
        protected void ChangeCounts(object item, int delta)
        {
            bool changeLeafCount = !(item is CollectionViewGroup);
 
            using (EmptyGroupRemover remover = EmptyGroupRemover.Create(isNeeded: changeLeafCount && delta < 0))
            {
                for (CollectionViewGroupInternal group = this;
                        group != null;
                        group = group._parentGroup)
                {
                    group.FullCount += delta;
                    if (changeLeafCount)
                    {
                        group.ProtectedItemCount += delta;
 
                        if (group.ProtectedItemCount == 0)
                        {
                            remover.RemoveEmptyGroup(group);
                        }
                    }
                }
            }
 
            unchecked { ++_version; }     // this invalidates enumerators
        }
 
        void OnGroupByChanged(object sender, System.ComponentModel.PropertyChangedEventArgs e)
        {
            OnGroupByChanged();
        }
 
        #endregion Private methods
 
        #region Private fields
 
        //------------------------------------------------------
        //
        //  Private fields
        //
        //------------------------------------------------------
 
        private readonly CollectionViewGroupInternal _parentGroup;
        private readonly bool _isExplicit;
 
        private GroupDescription _groupBy;
        private IComparer _groupComparer;
        private int _fullCount = 1;
        private int _lastIndex;
        private int _version;       // for detecting stale enumerators
 
        private static readonly NamedObject s_nullGroupNameKey = new("NullGroupNameKey");
        private Dictionary<object, WeakReference> _nameToGroupMap; // To cache the mapping between name and subgroup
        private bool _mapCleanupScheduled = false;
 
        #endregion Private fields
 
        #region Private classes
 
        //------------------------------------------------------
        //
        //  Private classes
        //
        //------------------------------------------------------
 
        private class LeafEnumerator : IEnumerator
        {
            public LeafEnumerator(CollectionViewGroupInternal group)
            {
                _group = group;
                DoReset();  // don't call virtual Reset in ctor
            }
 
            void IEnumerator.Reset()
            {
                DoReset();
            }
 
            void DoReset()
            {
                _version = _group._version;
                _index = -1;
                _subEnum = null;
            }
 
            bool IEnumerator.MoveNext()
            {
                // check for invalidated enumerator
                if (_group._version != _version)
                    throw new InvalidOperationException();
 
                // move forward to the next leaf
                while (_subEnum == null || !_subEnum.MoveNext())
                {
                    // done with the current top-level item.  Move to the next one.
                    ++_index;
                    if (_index >= _group.Items.Count)
                        return false;
 
                    CollectionViewGroupInternal subgroup = _group.Items[_index] as CollectionViewGroupInternal;
                    if (subgroup == null)
                    {
                        // current item is a leaf - it's the new Current
                        _current = _group.Items[_index];
                        _subEnum = null;
                        return true;
                    }
                    else
                    {
                        // current item is a subgroup - get its enumerator
                        _subEnum = subgroup.GetLeafEnumerator();
                    }
                }
 
                // the loop terminates only when we have a subgroup enumerator
                // positioned at the new Current item
                _current = _subEnum.Current;
                return true;
            }
 
            object IEnumerator.Current
            {
                get
                {
                    if (_index < 0 || _index >= _group.Items.Count)
                        throw new InvalidOperationException();
                    return _current;
                }
            }
 
            CollectionViewGroupInternal _group; // parent group
            int _version;   // parent group's version at ctor
            int _index;     // current index into Items
            IEnumerator _subEnum;   // enumerator over current subgroup
            object _current;   // current item
        }
 
        // When removing a leaf item, ChangeCounts removes groups that become empty.
        // It's important to propagate the changed item count all the way up
        // to the root before actually removing the empty groups, as the removal
        // raises an event whose handlers could call back into the group tree -
        // if the counts are inconsistent, crashes can occur.
        // This class remembers the empty groups, and removes them after the
        // counts have all been changed.
        private class EmptyGroupRemover : IDisposable
        {
            public static EmptyGroupRemover Create(bool isNeeded)
            {
                return isNeeded ? new EmptyGroupRemover() : null;
            }
 
            public void RemoveEmptyGroup(CollectionViewGroupInternal group)
            {
                if (_toRemove == null)
                {
                    _toRemove = new List<CollectionViewGroupInternal>();
                }
                _toRemove.Add(group);
            }
 
            public void Dispose()
            {
                if (_toRemove != null)
                {
                    foreach (CollectionViewGroupInternal group in _toRemove)
                    {
                        CollectionViewGroupInternal parent = group.Parent;
 
                        if (parent != null)
                        {
                            // remove the subgroup unless it is one of the explicit groups
                            if (!group.IsExplicit)
                            {
                                parent.Remove(group, false);
                            }
                        }
                    }
                }
            }
 
            private List<CollectionViewGroupInternal> _toRemove;
        }
 
        #endregion Private classes
    }
}