You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			457 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			457 lines
		
	
	
		
			14 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| 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<TreeNode> _list;
 | |
|         private TreeNode _owner;
 | |
|         private bool _updateParent;
 | |
|         private int _version;
 | |
| 
 | |
|         private bool _isTrackingViewState;
 | |
| 
 | |
|         private List<LogItem> _log;
 | |
| 
 | |
| 
 | |
|         public TreeNodeCollection() : this(null, true) {
 | |
|         }
 | |
| 
 | |
| 
 | |
|         public TreeNodeCollection(TreeNode owner) : this(owner, true) {
 | |
|         }
 | |
| 
 | |
|         internal TreeNodeCollection(TreeNode owner, bool updateParent) {
 | |
|             _owner = owner;
 | |
|             _list = new List<TreeNode>();
 | |
|             _updateParent = updateParent;
 | |
|         }
 | |
| 
 | |
| 
 | |
|         public int Count {
 | |
|             get {
 | |
|                 return _list.Count;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         public bool IsSynchronized {
 | |
|             get {
 | |
|                 return ((ICollection)_list).IsSynchronized;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private List<LogItem> Log {
 | |
|             get {
 | |
|                 if (_log == null) {
 | |
|                     _log = new List<LogItem>();
 | |
|                 }
 | |
|                 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
 | |
| 
 | |
|         /// <internalonly/>
 | |
|         bool IStateManager.IsTrackingViewState {
 | |
|             get {
 | |
|                 return _isTrackingViewState;
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         /// <internalonly/>
 | |
|         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]);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
| 
 | |
|         /// <internalonly/>
 | |
|         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);
 | |
|         }
 | |
| 
 | |
| 
 | |
|         /// <internalonly/>
 | |
|         void IStateManager.TrackViewState() {
 | |
|             _isTrackingViewState = true;
 | |
|             for (int i = 0; i < Count; i++) {
 | |
|                 ((IStateManager)this[i]).TrackViewState();
 | |
|             }
 | |
|         }
 | |
|         #endregion
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     Convenience class for storing and using log entries.
 | |
|         /// </devdoc>
 | |
|         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;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|         }
 | |
| 
 | |
|         /// <devdoc>
 | |
|         ///     Convenience enumeration for identifying log commands
 | |
|         /// </devdoc>
 | |
|         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;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| }
 |