TreeNodeCollection.cs source code in C# .NET

Source code for the .NET framework in C#

                        

Code:

/ 4.0 / 4.0 / untmp / DEVDIV_TFS / Dev10 / Releases / RTMRel / ndp / fx / src / xsp / System / Web / UI / WebControls / TreeNodeCollection.cs / 1305376 / TreeNodeCollection.cs

                            namespace System.Web.UI.WebControls { 
    using System;
    using System.Collections;
    using System.Collections.Generic;
    using System.ComponentModel; 
    using System.Globalization;
    using System.Text; 
    using System.Web.UI; 

    public sealed class TreeNodeCollection : ICollection, IStateManager { 
        private List _list;
        private TreeNode _owner;
        private bool _updateParent;
        private int _version; 

        private bool _isTrackingViewState; 
 
        private List _log;
 

        public TreeNodeCollection() : this(null, true) {
        }
 

        public TreeNodeCollection(TreeNode owner) : this(owner, true) { 
        } 

        internal TreeNodeCollection(TreeNode owner, bool updateParent) { 
            _owner = owner;
            _list = new List();
            _updateParent = updateParent;
        } 

 
        public int Count { 
            get {
                return _list.Count; 
            }
        }

 
        public bool IsSynchronized {
            get { 
                return ((ICollection)_list).IsSynchronized; 
            }
        } 

        private List Log {
            get {
                if (_log == null) { 
                    _log = new List();
                } 
                return _log; 
            }
        } 


        public object SyncRoot {
            get { 
                return ((ICollection)_list).SyncRoot;
            } 
        } 

 
        public TreeNode this[int index] {
            get {
                return _list[index];
            } 
        }
 
 
        public void Add(TreeNode child) {
            AddAt(Count, child); 
        }


        public void AddAt(int index, TreeNode child) { 
            if (child == null) {
                throw new ArgumentNullException("child"); 
            } 

            if (_updateParent) { 
                if (child.Owner != null && child.Parent == null) {
                    child.Owner.Nodes.Remove(child);
                }
                if (child.Parent != null) { 
                    child.Parent.ChildNodes.Remove(child);
                } 
                if (_owner != null) { 
                    child.SetParent(_owner);
                    child.SetOwner(_owner.Owner); 
                }
            }

            _list.Insert(index, child); 
            _version++;
 
            if (_isTrackingViewState) { 
                ((IStateManager)child).TrackViewState();
                child.SetDirty(); 
            }
            Log.Add(new LogItem(LogItemType.Insert, index, _isTrackingViewState));
        }
 

        public void Clear() { 
            if (this.Count == 0) return; 
            if (_owner != null) {
                TreeView owner = _owner.Owner; 
                if (owner != null) {
                    // Clear checked nodes if necessary
                    if (owner.CheckedNodes.Count != 0) {
                        owner.CheckedNodes.Clear(); 
                    }
                    TreeNode current = owner.SelectedNode; 
                    // Check if the selected item is under this collection 
                    while (current != null) {
                        if (this.Contains(current)) { 
                            owner.SetSelectedNode(null);
                            break;
                        }
                        current = current.Parent; 
                    }
                } 
            } 
            foreach (TreeNode node in _list) {
                node.SetParent(null); 
            }

            _list.Clear();
            _version++; 
            if (_isTrackingViewState) {
                // Clearing invalidates all previous log entries, so we can just clear them out and save some space 
                Log.Clear(); 
            }
            Log.Add(new LogItem(LogItemType.Clear, 0, _isTrackingViewState)); 
        }


        public void CopyTo(TreeNode[] nodeArray, int index) { 
            ((ICollection)this).CopyTo(nodeArray, index);
        } 
 

        public bool Contains(TreeNode c) { 
            return _list.Contains(c);
        }

        internal TreeNode FindNode(string[] path, int pos) { 
            if (pos == path.Length) {
                return _owner; 
            } 

            string pathPart = TreeView.UnEscape(path[pos]); 
            for (int i = 0; i < Count; i++) {
                TreeNode node = this[i];
                if (node.Value == pathPart) {
                    return node.ChildNodes.FindNode(path, pos + 1); 
                }
            } 
 
            return null;
        } 


        public IEnumerator GetEnumerator() {
            return new TreeNodeCollectionEnumerator(this); 
        }
 
 
        public int IndexOf(TreeNode value) {
            return _list.IndexOf(value); 
        }


        public void Remove(TreeNode value) { 
            if (value == null) {
                throw new ArgumentNullException("value"); 
            } 

            int index = _list.IndexOf(value); 
            if (index != -1) {
                RemoveAt(index);
            }
        } 

 
        public void RemoveAt(int index) { 
            TreeNode node = _list[index];
            if (_updateParent) { 
                TreeView owner = node.Owner;
                if (owner != null) {
                    if (owner.CheckedNodes.Count != 0) {
                        // We have to scan the whole tree of subnodes to remove any checked nodes 
                        // (and unselect the selected node if it is a descendant).
                        // That could badly hurt performance, except that removing a node is a pretty 
                        // exceptional event. 
                        UnCheckUnSelectRecursive(node);
                    } 
                    else {
                        // otherwise, we can just climb the tree up from the selected node
                        // to see if it is a descendant of the removed node.
                        TreeNode current = owner.SelectedNode; 
                        // Check if the selected item is under this collection
                        while (current != null) { 
                            if (current == node) { 
                                owner.SetSelectedNode(null);
                                break; 
                            }
                            current = current.Parent;
                        }
                    } 
                }
                node.SetParent(null); 
            } 

            _list.RemoveAt(index); 
            _version++;
            Log.Add(new LogItem(LogItemType.Remove, index, _isTrackingViewState));
        }
 
        internal void SetDirty() {
            foreach (LogItem item in Log) { 
                item.Tracked = true; 
            }
            for (int i = 0; i < Count; i++) { 
                this[i].SetDirty();
            }
        }
 
        private static void UnCheckUnSelectRecursive(TreeNode node) {
            TreeNodeCollection checkedNodes = node.Owner.CheckedNodes; 
            if (node.Checked) { 
                checkedNodes.Remove(node);
            } 
            TreeNode selectedNode = node.Owner.SelectedNode;
            if (node == selectedNode) {
                node.Owner.SetSelectedNode(null);
                selectedNode = null; 
            }
            // Only recurse if there could be some more work to do 
            if (selectedNode != null || checkedNodes.Count != 0) { 
                foreach (TreeNode child in node.ChildNodes) {
                    UnCheckUnSelectRecursive(child); 
                }
            }
        }
 
        #region ICollection implementation
 
        void ICollection.CopyTo(Array array, int index) { 
            if (!(array is TreeNode[])) {
                throw new ArgumentException(SR.GetString(SR.TreeNodeCollection_InvalidArrayType), "array"); 
            }
            _list.CopyTo((TreeNode[])array, index);
        }
        #endregion 

        #region IStateManager implementation 
 
        /// 
        bool IStateManager.IsTrackingViewState { 
            get {
                return _isTrackingViewState;
            }
        } 

 
        ///  
        void IStateManager.LoadViewState(object state) {
            object[] nodeState = (object[])state; 
            if (nodeState != null) {
                if (nodeState[0] != null) {
                    string logString = (string)nodeState[0];
                    // Process each log entry 
                    string[] items = logString.Split(',');
                    for (int i = 0; i < items.Length; i++) { 
                        string[] parts = items[i].Split(':'); 
                        LogItemType type = (LogItemType)Int32.Parse(parts[0], CultureInfo.InvariantCulture);
                        int index = Int32.Parse(parts[1], CultureInfo.InvariantCulture); 

                        if (type == LogItemType.Insert) {
                            if (_owner != null && _owner.Owner != null) {
                                AddAt(index, _owner.Owner.CreateNode()); 
                            }
                            else { 
                                AddAt(index, new TreeNode()); 
                            }
                        } 
                        else if (type == LogItemType.Remove) {
                            RemoveAt(index);
                        }
                        else if (type == LogItemType.Clear) { 
                            Clear();
                        } 
                    } 
                }
 
                for (int i = 0; i < nodeState.Length - 1; i++) {
                    if ((nodeState[i + 1] != null) && (this[i] != null)) {
                        ((IStateManager)this[i]).LoadViewState(nodeState[i + 1]);
                    } 
                }
            } 
        } 

 
        /// 
        object IStateManager.SaveViewState() {
            object[] nodes = new object[Count + 1];
 
            bool hasViewState = false;
 
            if ((_log != null) && (_log.Count > 0)) { 
                // Construct a string representation of the log, delimiting entries with commas
                // and seperator command and index with a colon 
                StringBuilder builder = new StringBuilder();
                int realLogCount = 0;
                for (int i = 0; i < _log.Count; i++) {
                    LogItem item = _log[i]; 
                    if (item.Tracked) {
                        builder.Append((int)item.Type); 
                        builder.Append(":"); 
                        builder.Append(item.Index);
                        if (i < (_log.Count - 1)) { 
                            builder.Append(",");
                        }

                        realLogCount++; 
                    }
                } 
 
                if (realLogCount > 0) {
                    nodes[0] = builder.ToString(); 
                    hasViewState = true;
                }
            }
 
            for (int i = 0; i < Count; i++) {
                nodes[i + 1] = ((IStateManager)this[i]).SaveViewState(); 
                if (nodes[i + 1] != null) { 
                    hasViewState = true;
                } 
            }

            return (hasViewState ? nodes : null);
        } 

 
        ///  
        void IStateManager.TrackViewState() {
            _isTrackingViewState = true; 
            for (int i = 0; i < Count; i++) {
                ((IStateManager)this[i]).TrackViewState();
            }
        } 
        #endregion
 
        ///  
        ///     Convenience class for storing and using log entries.
        ///  
        private class LogItem {
            private LogItemType _type;
            private int _index;
            private bool _tracked; 

            public LogItem(LogItemType type, int index, bool tracked) { 
                _type = type; 
                _index = index;
                _tracked = tracked; 
            }

            public int Index {
                get { 
                    return _index;
                } 
            } 

            public bool Tracked { 
                get {
                    return _tracked;
                }
                set { 
                    _tracked = value;
                } 
            } 

            public LogItemType Type { 
                get {
                    return _type;
                }
            } 

        } 
 
        /// 
        ///     Convenience enumeration for identifying log commands 
        /// 
        private enum LogItemType {
            Insert = 0,
            Remove = 1, 
            Clear = 2
        } 
 
        // This is a copy of the ArrayListEnumeratorSimple in ArrayList.cs
        private class TreeNodeCollectionEnumerator : IEnumerator { 
            private TreeNodeCollection list;
            private int index;
            private int version;
            private TreeNode currentElement; 

            internal TreeNodeCollectionEnumerator(TreeNodeCollection list) { 
                this.list = list; 
                this.index = -1;
                version = list._version; 
            }

            public bool MoveNext() {
                if (version != list._version) 
                    throw new InvalidOperationException(SR.GetString(SR.ListEnumVersionMismatch));
 
                if (index < (list.Count - 1)) { 
                    index++;
                    currentElement = list[index]; 
                    return true;
                }
                else
                    index = list.Count; 
                return false;
            } 
 
            object IEnumerator.Current {
                get { 
                    return Current;
                }
            }
 
            public TreeNode Current {
                get { 
                    if (index == -1) 
                        throw new InvalidOperationException(SR.GetString(SR.ListEnumCurrentOutOfRange));
                    if (index >= list.Count) 
                        throw new InvalidOperationException(SR.GetString(SR.ListEnumCurrentOutOfRange));
                    return currentElement;
                }
            } 

            public void Reset() { 
                if (version != list._version) 
                    throw new InvalidOperationException(SR.GetString(SR.ListEnumVersionMismatch));
                currentElement = null; 
                index = -1;
            }
        }
    } 
}

// File provided for Reference Use Only by Microsoft Corporation (c) 2007.
// Copyright (c) Microsoft Corporation. All rights reserved.


                        

Link Menu

Network programming in C#, Network Programming in VB.NET, Network Programming in .NET
This book is available now!
Buy at Amazon US or
Buy at Amazon UK