You've already forked linux-packaging-mono
							
							
		
			
				
	
	
		
			1096 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			1096 lines
		
	
	
		
			27 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| // Permission is hereby granted, free of charge, to any person obtaining
 | |
| // a copy of this software and associated documentation files (the
 | |
| // "Software"), to deal in the Software without restriction, including
 | |
| // without limitation the rights to use, copy, modify, merge, publish,
 | |
| // distribute, sublicense, and/or sell copies of the Software, and to
 | |
| // permit persons to whom the Software is furnished to do so, subject to
 | |
| // the following conditions:
 | |
| //
 | |
| // The above copyright notice and this permission notice shall be
 | |
| // included in all copies or substantial portions of the Software.
 | |
| //
 | |
| // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
 | |
| // EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
 | |
| // MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
 | |
| // NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
 | |
| // LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
 | |
| // OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
 | |
| // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 | |
| //
 | |
| // Copyright (c) 2004-2005 Novell, Inc.
 | |
| //
 | |
| // Authors:
 | |
| //	Jackson Harper (jackson@ximian.com)
 | |
| //	Kazuki Oikawa (kazuki@panicode.com)
 | |
| 
 | |
| using System;
 | |
| using System.ComponentModel;
 | |
| using System.Drawing;
 | |
| using System.Runtime.Serialization;
 | |
| using System.Text;
 | |
| 
 | |
| namespace System.Windows.Forms
 | |
| {
 | |
| 	[DefaultProperty ("Text")]
 | |
| 	[TypeConverter(typeof(TreeNodeConverter))]
 | |
| 	[Serializable]
 | |
| 	public class TreeNode : MarshalByRefObject, ICloneable, ISerializable
 | |
| 	{
 | |
| 		#region Fields
 | |
| 		private TreeView tree_view;
 | |
| 		internal TreeNode parent;
 | |
| 
 | |
| 		private string text;
 | |
| 		private int image_index = -1;
 | |
| 		private int selected_image_index = -1;
 | |
| 		private ContextMenu context_menu;
 | |
| 		private ContextMenuStrip context_menu_strip;
 | |
| 		private string image_key = String.Empty;
 | |
| 		private string selected_image_key = String.Empty;
 | |
| 		private int state_image_index = -1;
 | |
| 		private string state_image_key = String.Empty;
 | |
| 		private string tool_tip_text = String.Empty;
 | |
| 		internal TreeNodeCollection nodes;
 | |
| 		internal TreeViewAction check_reason = TreeViewAction.Unknown;
 | |
| 
 | |
| 		internal int visible_order = 0;
 | |
| 		internal int width = -1;
 | |
| 		
 | |
| 		internal bool is_expanded = false;
 | |
| 		private bool check;
 | |
| 		internal OwnerDrawPropertyBag prop_bag;
 | |
| 
 | |
| 		private object tag;
 | |
| 
 | |
| 		internal IntPtr handle;
 | |
| 		
 | |
| 		private string name = string.Empty;
 | |
| 		#endregion	// Fields
 | |
| 
 | |
| 		#region Internal Constructors
 | |
| 		internal TreeNode (TreeView tree_view) : this ()
 | |
| 		{
 | |
| 			this.tree_view = tree_view;
 | |
| 			is_expanded = true;
 | |
| 		}
 | |
| 
 | |
| 		protected TreeNode (SerializationInfo serializationInfo, StreamingContext context) : this ()
 | |
| 		{
 | |
| 			SerializationInfoEnumerator	en;
 | |
| 			SerializationEntry		e;
 | |
| 			int				children;
 | |
| 
 | |
| 			en = serializationInfo.GetEnumerator();
 | |
| 			children = 0;
 | |
| 			while (en.MoveNext()) {
 | |
| 				e = en.Current;
 | |
| 				switch(e.Name) {
 | |
| 					case "Text": Text = (string)e.Value; break;
 | |
| 					case "PropBag": prop_bag = (OwnerDrawPropertyBag)e.Value; break;
 | |
| 					case "ImageIndex": image_index = (int)e.Value; break;
 | |
| 					case "SelectedImageIndex": selected_image_index = (int)e.Value; break;
 | |
| 					case "Tag": tag = e.Value; break;
 | |
| 					case "IsChecked": check = (bool)e.Value; break;
 | |
| 					case "ChildCount": children = (int)e.Value; break;
 | |
| 				}
 | |
| 			}
 | |
| 			if (children > 0) {
 | |
| 				for (int i = 0; i < children; i++) {
 | |
| 					TreeNode node = (TreeNode) serializationInfo.GetValue ("children" + i, typeof (TreeNode));
 | |
| 					Nodes.Add (node);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 		#endregion	// Internal Constructors
 | |
| 
 | |
| 		#region Public Constructors
 | |
| 		public TreeNode ()
 | |
| 		{
 | |
| 			nodes = new TreeNodeCollection (this);
 | |
| 		}
 | |
| 
 | |
| 		public TreeNode (string text) : this ()
 | |
| 		{
 | |
| 			Text = text;
 | |
| 		}
 | |
| 
 | |
| 		public TreeNode (string text, TreeNode [] children) : this (text)
 | |
| 		{
 | |
| 			Nodes.AddRange (children);
 | |
| 		}
 | |
| 
 | |
| 		public TreeNode (string text, int imageIndex, int selectedImageIndex) : this (text)
 | |
| 		{
 | |
| 			this.image_index = imageIndex;
 | |
| 			this.selected_image_index = selectedImageIndex;
 | |
| 		}
 | |
| 
 | |
| 		public TreeNode (string text, int imageIndex, int selectedImageIndex,
 | |
| 				TreeNode[] children)
 | |
| 			: this (text, imageIndex, selectedImageIndex)
 | |
| 		{
 | |
| 			Nodes.AddRange (children);
 | |
| 		}
 | |
| 
 | |
| 		#endregion	// Public Constructors
 | |
| 
 | |
| 		#region ICloneable Members
 | |
| 		public virtual object Clone ()
 | |
| 		{
 | |
| 			TreeNode tn = (TreeNode)Activator.CreateInstance (GetType ());
 | |
| 			tn.name = name;
 | |
| 			tn.text = text;
 | |
| 			tn.image_key = image_key;
 | |
| 			tn.image_index = image_index;
 | |
| 			tn.selected_image_index = selected_image_index;
 | |
| 			tn.selected_image_key = selected_image_key;
 | |
| 			tn.state_image_index = state_image_index;
 | |
| 			tn.state_image_key = state_image_key;
 | |
| 			tn.tag = tag;
 | |
| 			tn.check = check;
 | |
| 			tn.tool_tip_text = tool_tip_text;
 | |
| 			tn.context_menu = context_menu;
 | |
| 			tn.context_menu_strip = context_menu_strip;
 | |
| 			if (nodes != null) {
 | |
| 				foreach (TreeNode child in nodes)
 | |
| 					tn.nodes.Add ((TreeNode)child.Clone ());
 | |
| 			}
 | |
| 			if (prop_bag != null)
 | |
| 				tn.prop_bag = OwnerDrawPropertyBag.Copy (prop_bag);
 | |
| 			return tn;
 | |
| 		}
 | |
| 
 | |
| 		#endregion	// ICloneable Members
 | |
| 
 | |
| 		#region ISerializable Members
 | |
| 		void ISerializable.GetObjectData (SerializationInfo si, StreamingContext context)
 | |
| 		{
 | |
| 			si.AddValue ("Text", Text);
 | |
| 			si.AddValue ("prop_bag", prop_bag, typeof (OwnerDrawPropertyBag));
 | |
| 			si.AddValue ("ImageIndex", ImageIndex);
 | |
| 			si.AddValue ("SelectedImageIndex", SelectedImageIndex);
 | |
| 			si.AddValue ("Tag", Tag);
 | |
| 			si.AddValue ("Checked", Checked);
 | |
| 
 | |
| 			si.AddValue ("NumberOfChildren", Nodes.Count);
 | |
| 			for (int i = 0; i < Nodes.Count; i++)
 | |
| 				si.AddValue ("Child-" + i, Nodes [i], typeof (TreeNode));
 | |
| 		}
 | |
| 
 | |
| 		protected virtual void Deserialize (SerializationInfo serializationInfo, StreamingContext context)
 | |
| 		{
 | |
| 			Text = serializationInfo.GetString ("Text");
 | |
| 			prop_bag = (OwnerDrawPropertyBag)serializationInfo.GetValue ("prop_bag", typeof (OwnerDrawPropertyBag));
 | |
| 			ImageIndex = serializationInfo.GetInt32 ("ImageIndex");
 | |
| 			SelectedImageIndex = serializationInfo.GetInt32 ("SelectedImageIndex");
 | |
| 			Tag = serializationInfo.GetValue ("Tag", typeof (Object));
 | |
| 			Checked = serializationInfo.GetBoolean ("Checked");
 | |
| 			
 | |
| 			int count = serializationInfo.GetInt32 ("NumberOfChildren");
 | |
| 			
 | |
| 			for (int i = 0; i < count; i++)
 | |
| 				Nodes.Add ((TreeNode)serializationInfo.GetValue ("Child-" + i, typeof (TreeNode)));
 | |
| 		}
 | |
| 		
 | |
| 		protected virtual void Serialize (SerializationInfo si,  StreamingContext context)
 | |
| 		{
 | |
| 			si.AddValue ("Text", Text);
 | |
| 			si.AddValue ("prop_bag", prop_bag, typeof (OwnerDrawPropertyBag));
 | |
| 			si.AddValue ("ImageIndex", ImageIndex);
 | |
| 			si.AddValue ("SelectedImageIndex", SelectedImageIndex);
 | |
| 			si.AddValue ("Tag", Tag);
 | |
| 			si.AddValue ("Checked", Checked);
 | |
| 
 | |
| 			si.AddValue ("NumberOfChildren", Nodes.Count);
 | |
| 			for (int i = 0; i < Nodes.Count; i++)
 | |
| 				si.AddValue ("Child-" + i, Nodes[i], typeof (TreeNode));
 | |
| 		}
 | |
| 		#endregion	// ISerializable Members
 | |
| 
 | |
| 		#region Public Instance Properties
 | |
| 		public Color BackColor {
 | |
| 			get {
 | |
| 				if (prop_bag != null)
 | |
| 					return prop_bag.BackColor;
 | |
| 				return Color.Empty;
 | |
| 			}
 | |
| 			set {
 | |
| 				if (prop_bag == null)
 | |
| 					prop_bag = new OwnerDrawPropertyBag ();
 | |
| 				prop_bag.BackColor = value;
 | |
| 
 | |
| 				TreeView tree_view = TreeView;
 | |
| 				if (tree_view != null)
 | |
| 					tree_view.UpdateNode (this);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public Rectangle Bounds {
 | |
| 			get {
 | |
| 				if (TreeView == null)
 | |
| 					return Rectangle.Empty;
 | |
| 
 | |
| 				int x = GetX ();
 | |
| 				int y = GetY ();
 | |
| 				
 | |
| 				if (width == -1)
 | |
| 					width = TreeView.GetNodeWidth (this);
 | |
| 
 | |
| 				Rectangle res = new Rectangle (x, y, width, TreeView.ActualItemHeight);
 | |
| 				return res;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		internal int GetY ()
 | |
| 		{
 | |
| 			if (TreeView == null)
 | |
| 				return 0;
 | |
| 			return (visible_order - 1) * TreeView.ActualItemHeight - (TreeView.skipped_nodes * TreeView.ActualItemHeight);
 | |
| 		}
 | |
| 
 | |
| 		internal int GetX ()
 | |
| 		{
 | |
| 			if (TreeView == null)
 | |
| 				return 0;
 | |
| 			int indent_level = IndentLevel;
 | |
| 			int roots = (TreeView.ShowRootLines ? 1 : 0);
 | |
| 			int cb = (TreeView.CheckBoxes ? 19 : 0);
 | |
| 			if (!TreeView.CheckBoxes && StateImage != null)
 | |
| 				cb = 19;
 | |
| 			int imgs = (TreeView.ImageList != null ?  TreeView.ImageList.ImageSize.Width + 3 : 0);
 | |
| 			return ((indent_level + roots) * TreeView.Indent) + cb + imgs - TreeView.hbar_offset;
 | |
| 		}
 | |
| 
 | |
| 		internal int GetLinesX ()
 | |
| 		{
 | |
| 			int roots = (TreeView.ShowRootLines ? 1 : 0);
 | |
| 			return (IndentLevel + roots) * TreeView.Indent - TreeView.hbar_offset;
 | |
| 		}
 | |
| 
 | |
| 		internal int GetImageX ()
 | |
| 		{
 | |
| 			return GetLinesX () + (TreeView.CheckBoxes || StateImage != null ? 19 : 0);
 | |
| 		}
 | |
| 
 | |
| 		// In theory we should be able to track this instead of computing
 | |
| 		// every single time we need it, however for now I am going to
 | |
| 		// do it this way to reduce bugs in my new bounds computing code
 | |
| 		internal int IndentLevel {
 | |
| 			get {
 | |
| 				TreeNode walk = this;
 | |
| 				int res = 0;
 | |
| 				while (walk.Parent != null) {
 | |
| 					walk = walk.Parent;
 | |
| 					res++;
 | |
| 				}
 | |
| 
 | |
| 				return res;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[DefaultValue (false)]
 | |
| 		public bool Checked {
 | |
| 			get { return check; }
 | |
| 			set {
 | |
| 				if (check == value)
 | |
| 					return;
 | |
| 				TreeViewCancelEventArgs args = new TreeViewCancelEventArgs (this, false, check_reason);
 | |
| 				if (TreeView != null)
 | |
| 					TreeView.OnBeforeCheck (args);
 | |
| 				if (!args.Cancel) {
 | |
| 					check = value;
 | |
| 
 | |
| 					// TreeView can become null after OnAfterCheck, this the double null check
 | |
| 					if (TreeView != null)
 | |
| 						TreeView.OnAfterCheck (new TreeViewEventArgs (this, check_reason));
 | |
| 					if (TreeView != null)
 | |
| 						TreeView.UpdateNode (this);
 | |
| 				}
 | |
| 				check_reason = TreeViewAction.Unknown;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[DefaultValue (null)]
 | |
| 		public virtual ContextMenu ContextMenu {
 | |
| 			get { return context_menu; }
 | |
| 			set { context_menu = value; }
 | |
| 		}
 | |
| 		
 | |
| 		[DefaultValue (null)]
 | |
| 		public virtual ContextMenuStrip ContextMenuStrip {
 | |
| 			get { return context_menu_strip; }
 | |
| 			set { context_menu_strip = value; }
 | |
| 		}
 | |
| 		
 | |
| 		[Browsable (false)]
 | |
| 		public TreeNode FirstNode {
 | |
| 			get {
 | |
| 				if (nodes.Count > 0)
 | |
| 					return nodes [0];
 | |
| 				return null;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public Color ForeColor {
 | |
| 			get {
 | |
| 				if (prop_bag != null)
 | |
| 					return prop_bag.ForeColor;
 | |
| 				if (TreeView != null)
 | |
| 					return TreeView.ForeColor;
 | |
| 				return Color.Empty;
 | |
| 			}
 | |
| 			set {
 | |
| 				if (prop_bag == null)
 | |
| 					prop_bag = new OwnerDrawPropertyBag ();
 | |
| 				prop_bag.ForeColor = value;
 | |
| 
 | |
| 				TreeView tree_view = TreeView;
 | |
| 				if (tree_view != null)
 | |
| 					tree_view.UpdateNode (this);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public string FullPath {
 | |
| 			get {
 | |
| 				if (TreeView == null)
 | |
| 					throw new InvalidOperationException ("No TreeView associated");
 | |
| 
 | |
| 				StringBuilder builder = new StringBuilder ();
 | |
| 				BuildFullPath (builder);
 | |
| 				return builder.ToString ();
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[DefaultValue (-1)]
 | |
| 		[RelatedImageList ("TreeView.ImageList")]
 | |
| 		[TypeConverter (typeof (TreeViewImageIndexConverter))]
 | |
| 		[RefreshProperties (RefreshProperties.Repaint)]
 | |
| 		[Editor ("System.Windows.Forms.Design.ImageIndexEditor, " + Consts.AssemblySystem_Design, typeof (System.Drawing.Design.UITypeEditor))]
 | |
| 		[Localizable(true)]
 | |
| 		public int ImageIndex {
 | |
| 			get { return image_index; }
 | |
| 			set {
 | |
| 				if (image_index == value)
 | |
| 					return;
 | |
| 				image_index = value;
 | |
| 				image_key = string.Empty;
 | |
| 				TreeView tree = TreeView;
 | |
| 				if (tree != null)
 | |
| 					tree.UpdateNode (this);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Localizable(true)]
 | |
| 		[DefaultValue ("")]
 | |
| 		[RelatedImageList ("TreeView.ImageList")]
 | |
| 		[TypeConverter (typeof (TreeViewImageKeyConverter))]
 | |
| 		[RefreshProperties (RefreshProperties.Repaint)]
 | |
| 		[Editor ("System.Windows.Forms.Design.ImageIndexEditor, " + Consts.AssemblySystem_Design, typeof (System.Drawing.Design.UITypeEditor))]
 | |
| 		public string ImageKey {
 | |
| 			get { return image_key; }
 | |
| 			set {
 | |
| 				if (image_key == value)
 | |
| 					return;
 | |
| 				image_key = value;
 | |
| 				image_index = -1;
 | |
| 
 | |
| 				TreeView tree = TreeView;
 | |
| 				if (tree != null)
 | |
| 				tree.UpdateNode(this);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public bool IsEditing {
 | |
| 			get {
 | |
| 				TreeView tv = TreeView;
 | |
| 				if (tv == null)
 | |
| 					return false;
 | |
| 				return tv.edit_node == this;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public bool IsExpanded {
 | |
| 			get {
 | |
| 				TreeView tv = TreeView;
 | |
| 
 | |
| 				if (tv != null && tv.IsHandleCreated) {
 | |
| 					// This is ridiculous
 | |
| 					bool found = false;
 | |
| 					foreach (TreeNode walk in TreeView.Nodes) {
 | |
| 						if (walk.Nodes.Count > 0)
 | |
| 							found = true;
 | |
| 					}
 | |
| 
 | |
| 					if (!found)
 | |
| 						return false;
 | |
| 				}
 | |
| 
 | |
| 				return is_expanded;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public bool IsSelected {
 | |
| 			get {
 | |
| 				if (TreeView == null || !TreeView.IsHandleCreated)
 | |
| 					return false;
 | |
| 				return TreeView.SelectedNode == this;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public bool IsVisible {
 | |
| 			get {
 | |
| 				if (TreeView == null || !TreeView.IsHandleCreated || !TreeView.Visible)
 | |
| 					return false;
 | |
| 
 | |
| 				if (visible_order <= TreeView.skipped_nodes || visible_order - TreeView.skipped_nodes > TreeView.VisibleCount)
 | |
| 					return false;
 | |
| 
 | |
| 				return ArePreviousNodesExpanded;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public TreeNode LastNode {
 | |
| 			get {
 | |
| 				return (nodes == null || nodes.Count == 0) ? null : nodes [nodes.Count - 1];
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public int Level {
 | |
| 			get { return IndentLevel; }
 | |
| 		}
 | |
| 		
 | |
| 		public string Name
 | |
| 		{
 | |
| 			get { return this.name; }
 | |
| 			set {
 | |
| 				// Value should never be null as per spec
 | |
| 				this.name = (value == null) ? string.Empty : value;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public TreeNode NextNode {
 | |
| 			get {
 | |
| 				if (parent == null)
 | |
| 					return null;
 | |
| 				int index = Index;
 | |
| 				if (parent.Nodes.Count > index + 1)
 | |
| 					return parent.Nodes [index + 1];
 | |
| 				return null;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public TreeNode NextVisibleNode {
 | |
| 			get {
 | |
| 				OpenTreeNodeEnumerator o = new OpenTreeNodeEnumerator (this);
 | |
| 				o.MoveNext (); // move to the node itself
 | |
| 
 | |
| 				if (!o.MoveNext ())
 | |
| 					return null;
 | |
| 				TreeNode c = o.CurrentNode;
 | |
| 				if (!c.IsInClippingRect)
 | |
| 					return null;
 | |
| 				return c;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[DefaultValue (null)]
 | |
| 		[Localizable (true)]
 | |
| 		public Font NodeFont {
 | |
| 			get {
 | |
| 				if (prop_bag != null)
 | |
| 					return prop_bag.Font;
 | |
| 				if (TreeView != null)
 | |
| 					return TreeView.Font;
 | |
| 				return null;
 | |
| 			}
 | |
| 			set {
 | |
| 				if (prop_bag == null)
 | |
| 					prop_bag = new OwnerDrawPropertyBag (); 
 | |
| 				prop_bag.Font = value;
 | |
| 				Invalidate ();
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		[ListBindable (false)]
 | |
| 		public TreeNodeCollection Nodes {
 | |
| 			get {
 | |
| 				if (nodes == null)
 | |
| 					nodes = new TreeNodeCollection (this);
 | |
| 				return nodes;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public TreeNode Parent {
 | |
| 			get {
 | |
| 				TreeView tree_view = TreeView;
 | |
| 				if (tree_view != null && tree_view.root_node == parent)
 | |
| 					return null;
 | |
| 				return parent;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public TreeNode PrevNode {
 | |
| 			get {
 | |
| 				if (parent == null)
 | |
| 					return null;
 | |
| 				int index = Index;
 | |
| 				if (index <= 0 || index > parent.Nodes.Count)
 | |
| 					return null;
 | |
| 				return parent.Nodes [index - 1];
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public TreeNode PrevVisibleNode {
 | |
| 			get {
 | |
| 				OpenTreeNodeEnumerator o = new OpenTreeNodeEnumerator (this);
 | |
| 				o.MovePrevious (); // move to the node itself
 | |
| 
 | |
| 				if (!o.MovePrevious ())
 | |
| 					return null;
 | |
| 				TreeNode c = o.CurrentNode;
 | |
| 				if (!c.IsInClippingRect)
 | |
| 					return null;
 | |
| 				return c;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[DefaultValue (-1)]
 | |
| 		[RelatedImageList ("TreeView.ImageList")]
 | |
| 		[TypeConverter (typeof (TreeViewImageIndexConverter))]
 | |
| 		[RefreshProperties (RefreshProperties.Repaint)]
 | |
| 		[Editor ("System.Windows.Forms.Design.ImageIndexEditor, " + Consts.AssemblySystem_Design, typeof (System.Drawing.Design.UITypeEditor))]
 | |
| 		[Localizable (true)]
 | |
| 		public int SelectedImageIndex {
 | |
| 			get { return selected_image_index; }
 | |
| 			set { selected_image_index = value; }
 | |
| 		}
 | |
| 
 | |
| 		[Localizable (true)]
 | |
| 		[DefaultValue ("")]
 | |
| 		[RelatedImageList ("TreeView.ImageList")]
 | |
| 		[TypeConverter (typeof (TreeViewImageKeyConverter))]
 | |
| 		[RefreshProperties (RefreshProperties.Repaint)]
 | |
| 		[Editor ("System.Windows.Forms.Design.ImageIndexEditor, " + Consts.AssemblySystem_Design, typeof (System.Drawing.Design.UITypeEditor))]
 | |
| 		public string SelectedImageKey {
 | |
| 			get { return selected_image_key; }
 | |
| 			set { selected_image_key = value; }
 | |
| 		}
 | |
| 
 | |
| 		[Localizable (true)]
 | |
| 		[DefaultValue (-1)]
 | |
| 		[RelatedImageList ("TreeView.StateImageList")]
 | |
| 		[TypeConverter (typeof (NoneExcludedImageIndexConverter))]
 | |
| 		[RefreshProperties (RefreshProperties.Repaint)]
 | |
| 		[Editor ("System.Windows.Forms.Design.ImageIndexEditor, " + Consts.AssemblySystem_Design, typeof (System.Drawing.Design.UITypeEditor))]
 | |
| 		public int StateImageIndex {
 | |
| 			get { return state_image_index; }
 | |
| 			set {
 | |
| 				if (state_image_index != value) {
 | |
| 					state_image_index = value;
 | |
| 					state_image_key = string.Empty;
 | |
| 					Invalidate ();
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Localizable (true)]
 | |
| 		[DefaultValue ("")]
 | |
| 		[RelatedImageList ("TreeView.StateImageList")]
 | |
| 		[TypeConverter (typeof (ImageKeyConverter))]
 | |
| 		[RefreshProperties (RefreshProperties.Repaint)]
 | |
| 		[Editor ("System.Windows.Forms.Design.ImageIndexEditor, " + Consts.AssemblySystem_Design, typeof (System.Drawing.Design.UITypeEditor))]
 | |
| 		public string StateImageKey {
 | |
| 			get { return state_image_key; }
 | |
| 			set {
 | |
| 				if (state_image_key != value) {
 | |
| 					state_image_key = value;
 | |
| 					state_image_index = -1;
 | |
| 					Invalidate ();
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Bindable(true)]
 | |
| 		[Localizable(false)]
 | |
| 		[TypeConverter(typeof(System.ComponentModel.StringConverter))]
 | |
| 		[DefaultValue(null)]
 | |
| 		public object Tag {
 | |
| 			get { return tag; }
 | |
| 			set { tag = value; }
 | |
| 		}
 | |
| 
 | |
| 		[Localizable(true)]
 | |
| 		public string Text {
 | |
| 			get {
 | |
| 				if (text == null)
 | |
| 					return String.Empty;
 | |
| 				return text;
 | |
| 			}
 | |
| 			set {
 | |
| 				if (text == value)
 | |
| 					return;
 | |
| 				text = value;
 | |
| 				Invalidate ();
 | |
| 				// UIA Framework Event: Text Changed
 | |
| 				TreeView view = TreeView;
 | |
| 				if (view != null)
 | |
| 					view.OnUIANodeTextChanged (new TreeViewEventArgs (this));
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[DefaultValue ("")]
 | |
| 		[Localizable (false)]
 | |
| 		public string ToolTipText {
 | |
| 			get { return tool_tip_text; }
 | |
| 			set { tool_tip_text = value; }
 | |
| 		}
 | |
| 		
 | |
| 		[Browsable (false)]
 | |
| 		public TreeView TreeView {
 | |
| 			get {
 | |
| 				if (tree_view != null)
 | |
| 					return tree_view;
 | |
| 				TreeNode walk = parent;
 | |
| 				while (walk != null) {
 | |
| 					if (walk.TreeView != null)
 | |
| 						break;
 | |
| 					walk = walk.parent;
 | |
| 				}
 | |
| 				if (walk == null)
 | |
| 					return null;
 | |
| 				return walk.TreeView;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		[Browsable (false)]
 | |
| 		public IntPtr Handle {
 | |
| 			get {
 | |
| 				// MS throws a NullReferenceException if the TreeView isn't set...
 | |
| 				if (handle == IntPtr.Zero && TreeView != null)
 | |
| 					handle = TreeView.CreateNodeHandle ();
 | |
| 				return handle;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		#endregion	// Public Instance Properties
 | |
| 
 | |
| 		
 | |
| 		public static TreeNode FromHandle (TreeView tree, IntPtr handle)
 | |
| 		{
 | |
| 			if (handle == IntPtr.Zero)
 | |
| 				return null;
 | |
| 			// No arg checking on MS it just throws a NullRef if treeview is null
 | |
| 			return tree.NodeFromHandle (handle);
 | |
| 		}
 | |
| 
 | |
| 		#region Public Instance Methods
 | |
| 		public void BeginEdit ()
 | |
| 		{
 | |
| 			TreeView tv = TreeView;
 | |
| 			if (tv != null)
 | |
| 				tv.BeginEdit (this);
 | |
| 		}
 | |
| 
 | |
| 		public void Collapse ()
 | |
| 		{
 | |
| 			CollapseInternal (false);
 | |
| 		}
 | |
| 
 | |
| 		public void Collapse (bool ignoreChildren)
 | |
| 		{
 | |
| 			if (ignoreChildren)
 | |
| 				Collapse ();
 | |
| 			else
 | |
| 				CollapseRecursive (this);
 | |
| 		}
 | |
| 
 | |
| 		public void EndEdit (bool cancel)
 | |
| 		{
 | |
| 			TreeView tv = TreeView;
 | |
| 			if (!cancel && tv != null)
 | |
| 				tv.EndEdit (this);
 | |
| 			else if (cancel && tv != null)
 | |
| 				tv.CancelEdit (this);
 | |
| 		}
 | |
| 
 | |
| 		public void Expand ()
 | |
| 		{
 | |
| 			Expand (false);
 | |
| 		}
 | |
| 
 | |
| 		public void ExpandAll ()
 | |
| 		{
 | |
| 			ExpandRecursive (this);
 | |
| 			if(TreeView != null)
 | |
| 				TreeView.UpdateNode (TreeView.root_node);
 | |
| 		}
 | |
| 
 | |
| 		public void EnsureVisible ()
 | |
| 		{
 | |
| 			if (TreeView == null)
 | |
| 				return;
 | |
| 
 | |
| 			if (this.Parent != null)
 | |
| 				ExpandParentRecursive (this.Parent);
 | |
| 
 | |
| 			Rectangle bounds = Bounds;
 | |
| 			if (bounds.Y < 0) {
 | |
| 				TreeView.SetTop (this);
 | |
| 			} else if (bounds.Bottom > TreeView.ViewportRectangle.Bottom) {
 | |
| 				TreeView.SetBottom (this);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		public int GetNodeCount (bool includeSubTrees)
 | |
| 		{
 | |
| 			if (!includeSubTrees)
 | |
| 				return Nodes.Count;
 | |
| 
 | |
| 			int count = 0;
 | |
| 			GetNodeCountRecursive (this, ref count);
 | |
| 
 | |
| 			return count;
 | |
| 		}
 | |
| 
 | |
| 		public void Remove ()
 | |
| 		{
 | |
| 			if (parent == null)
 | |
| 				return;
 | |
| 			int index = Index;
 | |
| 			parent.Nodes.RemoveAt (index);
 | |
| 		}
 | |
| 
 | |
| 		public void Toggle ()
 | |
| 		{
 | |
| 			if (is_expanded)
 | |
| 				Collapse ();
 | |
| 			else
 | |
| 				Expand ();
 | |
| 		}
 | |
| 
 | |
| 		public override String ToString ()
 | |
| 		{
 | |
| 			return String.Concat ("TreeNode: ", Text);
 | |
| 		}
 | |
| 
 | |
| 		#endregion	// Public Instance Methods
 | |
| 
 | |
| 		#region Internal & Private Methods and Properties
 | |
| 
 | |
| 		internal bool ArePreviousNodesExpanded {
 | |
| 			get {
 | |
| 				TreeNode parent = Parent;
 | |
| 				while (parent != null) {
 | |
| 					if (!parent.is_expanded)
 | |
| 						return false;
 | |
| 					parent = parent.Parent;
 | |
| 				}
 | |
| 
 | |
| 				return true;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		internal bool IsRoot {
 | |
| 			get {
 | |
| 				TreeView tree_view = TreeView;
 | |
| 				if (tree_view == null)
 | |
| 					return false;
 | |
| 				if (tree_view.root_node == this)
 | |
| 					return true;
 | |
| 				return false;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		bool BuildFullPath (StringBuilder path)
 | |
| 		{
 | |
| 			if (parent == null)
 | |
| 				return false;
 | |
| 
 | |
| 			if (parent.BuildFullPath (path))
 | |
| 				path.Append (TreeView.PathSeparator);
 | |
| 
 | |
| 			path.Append (text);
 | |
| 			return true;
 | |
| 		}
 | |
| 
 | |
| 		public int Index {
 | |
| 			get {
 | |
| 				if (parent == null)
 | |
| 					return 0;
 | |
| 				return parent.Nodes.IndexOf (this);
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		private void Expand (bool byInternal)
 | |
| 		{
 | |
| 			if (is_expanded || nodes.Count < 1) {
 | |
| 				is_expanded = true;
 | |
| 				return;
 | |
| 			}
 | |
| 
 | |
| 			bool cancel = false;
 | |
| 			TreeView tree_view = TreeView;
 | |
| 			if (tree_view != null) {
 | |
| 				TreeViewCancelEventArgs e = new TreeViewCancelEventArgs (this, false, TreeViewAction.Expand);
 | |
| 				tree_view.OnBeforeExpand (e);
 | |
| 				cancel = e.Cancel;
 | |
| 			}
 | |
| 
 | |
| 			if (!cancel) {
 | |
| 				is_expanded = true;
 | |
| 				int count_to_next = CountToNext ();
 | |
| 
 | |
| 				if (tree_view != null) {
 | |
| 					tree_view.OnAfterExpand (new TreeViewEventArgs (this));
 | |
| 
 | |
| 					tree_view.RecalculateVisibleOrder (this);
 | |
| 					tree_view.UpdateScrollBars (false);
 | |
| 
 | |
| 					// ExpandBelow if we affect the visible area
 | |
| 					if (visible_order < tree_view.skipped_nodes + tree_view.VisibleCount + 1 && ArePreviousNodesExpanded)
 | |
| 						tree_view.ExpandBelow (this, count_to_next);
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		private void CollapseInternal (bool byInternal)
 | |
| 		{
 | |
| 			if (!is_expanded || nodes.Count < 1)
 | |
| 				return;
 | |
| 
 | |
| 			if (IsRoot)
 | |
| 				return;
 | |
| 
 | |
| 			bool cancel = false;
 | |
| 			TreeView tree_view = TreeView;
 | |
| 			if (tree_view != null) {
 | |
| 				TreeViewCancelEventArgs e = new TreeViewCancelEventArgs (this, false, TreeViewAction.Collapse);
 | |
| 				tree_view.OnBeforeCollapse (e);
 | |
| 				cancel = e.Cancel;
 | |
| 			}
 | |
| 
 | |
| 			if (!cancel) {
 | |
| 				int count_to_next = CountToNext ();
 | |
| 
 | |
| 				is_expanded = false;
 | |
| 
 | |
| 				if (tree_view != null) {
 | |
| 					tree_view.OnAfterCollapse (new TreeViewEventArgs (this));
 | |
| 
 | |
| 					bool hbar_visible = tree_view.hbar.Visible;
 | |
| 					bool vbar_visible = tree_view.vbar.Visible;
 | |
| 
 | |
| 					tree_view.RecalculateVisibleOrder (this);
 | |
| 					tree_view.UpdateScrollBars (false);
 | |
| 
 | |
| 					// CollapseBelow if we affect the visible area
 | |
| 					if (visible_order < tree_view.skipped_nodes + tree_view.VisibleCount + 1 && ArePreviousNodesExpanded)
 | |
| 						tree_view.CollapseBelow (this, count_to_next);
 | |
| 					if(!byInternal && HasFocusInChildren ())
 | |
| 						tree_view.SelectedNode = this;
 | |
| 
 | |
| 					// If one or both of our scrollbars disappeared,
 | |
| 					// invalidate everything
 | |
| 					if ((hbar_visible & !tree_view.hbar.Visible) || (vbar_visible & !tree_view.vbar.Visible))
 | |
| 						tree_view.Invalidate ();
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		private int CountToNext ()
 | |
| 		{
 | |
| 			bool expanded = is_expanded;
 | |
| 			is_expanded = false;
 | |
| 			OpenTreeNodeEnumerator walk = new OpenTreeNodeEnumerator (this);
 | |
| 
 | |
| 			TreeNode next= null;
 | |
| 			if (walk.MoveNext () && walk.MoveNext ())
 | |
| 				next = walk.CurrentNode;
 | |
| 
 | |
| 			is_expanded = expanded;
 | |
| 			walk.Reset ();
 | |
| 			walk.MoveNext ();
 | |
| 
 | |
| 			int count = 0;
 | |
| 			while (walk.MoveNext () && walk.CurrentNode != next)
 | |
| 				count++;
 | |
| 
 | |
| 			return count;
 | |
| 		}
 | |
| 
 | |
| 		private bool HasFocusInChildren()
 | |
| 		{
 | |
| 			if (TreeView == null)
 | |
| 				return false;
 | |
| 			foreach (TreeNode node in nodes) {
 | |
| 				if(node == TreeView.SelectedNode)
 | |
| 					return true;
 | |
| 				if(node.HasFocusInChildren ())
 | |
| 					return true;
 | |
| 			}
 | |
| 			return false;
 | |
| 		}
 | |
| 
 | |
| 		private void ExpandRecursive (TreeNode node)
 | |
| 		{
 | |
| 			node.Expand (true);
 | |
| 			foreach (TreeNode child in node.Nodes)
 | |
| 				ExpandRecursive (child);
 | |
| 		}
 | |
| 
 | |
| 		private void ExpandParentRecursive (TreeNode node)
 | |
| 		{
 | |
| 			node.Expand (true);
 | |
| 			if (node.Parent != null)
 | |
| 				ExpandParentRecursive (node.Parent);
 | |
| 		}
 | |
| 
 | |
| 		internal void CollapseAll ()
 | |
| 		{
 | |
| 			CollapseRecursive (this);
 | |
| 		}
 | |
| 
 | |
| 		internal void CollapseAllUncheck ()
 | |
| 		{
 | |
| 			CollapseUncheckRecursive (this);
 | |
| 		}
 | |
| 
 | |
| 		private void CollapseRecursive (TreeNode node)
 | |
| 		{
 | |
| 			node.Collapse ();
 | |
| 			foreach (TreeNode child in node.Nodes)
 | |
| 				CollapseRecursive (child);
 | |
| 		}
 | |
| 
 | |
| 		private void CollapseUncheckRecursive (TreeNode node)
 | |
| 		{
 | |
| 			node.Collapse ();
 | |
| 			node.Checked = false;
 | |
| 			foreach (TreeNode child in node.Nodes)
 | |
| 				CollapseUncheckRecursive (child);
 | |
| 		}
 | |
| 
 | |
| 		internal void SetNodes (TreeNodeCollection nodes)
 | |
| 		{
 | |
| 			this.nodes = nodes;
 | |
| 		}
 | |
| 
 | |
| 		private void GetNodeCountRecursive (TreeNode node, ref int count)
 | |
| 		{
 | |
| 			count += node.Nodes.Count;
 | |
| 			foreach (TreeNode child in node.Nodes)
 | |
| 				GetNodeCountRecursive (child, ref count);
 | |
| 		}
 | |
| 
 | |
| 		internal bool NeedsWidth {
 | |
| 			get { return width == -1; }
 | |
| 		}
 | |
| 
 | |
| 		internal void Invalidate ()
 | |
| 		{
 | |
| 			// invalidate width first so Bounds retrieves 
 | |
| 			// the updated value (we don't use it here however)
 | |
| 			width = -1;
 | |
| 
 | |
| 			TreeView tv = TreeView;
 | |
| 			if (tv == null)
 | |
| 				return;
 | |
| 
 | |
| 			tv.UpdateNode (this);
 | |
| 		}
 | |
| 
 | |
| 		internal void InvalidateWidth ()
 | |
| 		{
 | |
| 			// bounds.Width = 0;
 | |
| 			width = -1;
 | |
| 		}
 | |
| 
 | |
| 		internal void SetWidth (int width)
 | |
| 		{
 | |
| 			this.width = width;
 | |
| 		}
 | |
| 
 | |
| 		internal void SetParent (TreeNode parent)
 | |
| 		{
 | |
| 			this.parent = parent;
 | |
| 		}
 | |
| 
 | |
| 		private bool IsInClippingRect {
 | |
| 			get {
 | |
| 				if (TreeView == null)
 | |
| 					return false;
 | |
| 				Rectangle bounds = Bounds;
 | |
| 				if (bounds.Y < 0 && bounds.Y > TreeView.ClientRectangle.Height)
 | |
| 					return false;
 | |
| 				return true;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		internal Image StateImage {
 | |
| 			get {
 | |
| 				if (TreeView != null) {
 | |
| 					if (TreeView.StateImageList == null)
 | |
| 						return null;
 | |
| 					if (state_image_index >= 0)
 | |
| 						return TreeView.StateImageList.Images[state_image_index];
 | |
| 					if (state_image_key != string.Empty)
 | |
| 						return TreeView.StateImageList.Images[state_image_key];
 | |
| 				}
 | |
| 
 | |
| 				return null;
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Order of operation:
 | |
| 		// 1) Node.Image[Key|Index]
 | |
| 		// 2) TreeView.Image[Key|Index]
 | |
| 		// 3) First image in TreeView.ImageList
 | |
| 		internal int Image {
 | |
| 			get {
 | |
| 				if (TreeView == null || TreeView.ImageList == null)
 | |
| 					return -1;
 | |
| 					
 | |
| 				if (IsSelected) {
 | |
| 					if (selected_image_index >= 0)
 | |
| 						return selected_image_index;
 | |
| 					if (!string.IsNullOrEmpty (selected_image_key))
 | |
| 						return TreeView.ImageList.Images.IndexOfKey (selected_image_key);
 | |
| 					if (!string.IsNullOrEmpty (TreeView.SelectedImageKey))
 | |
| 						return TreeView.ImageList.Images.IndexOfKey (TreeView.SelectedImageKey);
 | |
| 					if (TreeView.SelectedImageIndex >= 0)
 | |
| 						return TreeView.SelectedImageIndex;
 | |
| 				} else {
 | |
| 					if (image_index >= 0)
 | |
| 						return image_index;
 | |
| 					if (!string.IsNullOrEmpty (image_key))
 | |
| 						return TreeView.ImageList.Images.IndexOfKey (image_key);
 | |
| 					if (!string.IsNullOrEmpty (TreeView.ImageKey))
 | |
| 						return TreeView.ImageList.Images.IndexOfKey (TreeView.ImageKey);
 | |
| 					if (TreeView.ImageIndex >= 0)
 | |
| 						return TreeView.ImageIndex;
 | |
| 				}
 | |
| 
 | |
| 				if (TreeView.ImageList.Images.Count > 0)
 | |
| 					return 0;
 | |
| 					
 | |
| 				return -1;
 | |
| 			}
 | |
| 		}
 | |
| 		#endregion	// Internal & Private Methods and Properties
 | |
| 	}
 | |
| }
 |