/* Copyright (c) Citrix Systems, Inc.
* All rights reserved.
*
* Redistribution and use in source and binary forms,
* with or without modification, are permitted provided
* that the following conditions are met:
*
* * Redistributions of source code must retain the above
* copyright notice, this list of conditions and the
* following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the
* following disclaimer in the documentation and/or other
* materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND
* CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES,
* INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
* MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR
* CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING,
* BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY,
* WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
* NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF
* SUCH DAMAGE.
*/
using System.Collections.Generic;
using System.Windows.Forms;
using System.Drawing;
using System.Reflection;
namespace XenAdmin.Controls
{
///
/// This interface is used to hide these members from general use.
///
internal interface IMultiSelectTreeNode
{
void SetSelected(bool selected);
void SetEnabled(bool enabled);
}
public class MultiSelectTreeNode : TreeNode, IMultiSelectTreeNodeCollectionOwner, IMultiSelectTreeNode
{
private readonly MultiSelectTreeNodeCollection _nodes;
private Color _backColor;
private Color _foreColor;
private bool _selected;
private bool _enabled;
private Color _highlightForeColor = SystemColors.HighlightText;
private Color _highlightBackColor = SystemColors.Highlight;
private bool _showCheckBox;
public MultiSelectTreeNode()
: this(string.Empty)
{
}
public MultiSelectTreeNode(string text)
: base(text)
{
_nodes = new MultiSelectTreeNodeCollection(this);
_backColor = BackColor;
_foreColor = ForeColor;
}
///
/// Gets or sets the back-color of the treeNode from the base class. If this is done by setting the BackColor of the base class, then
/// the entire treeview is invalidated which can cause flicker. It is therefore achieved by setting the internal propbag and then invalidating
/// just this node.
///
private Color BaseBackColor
{
get
{
return base.BackColor;
}
set
{
if (value != base.BackColor)
{
if (TreeView != null && PropBag != null && TreeView.IsHandleCreated)
{
PropBag.BackColor = value;
TreeView.Invalidate(Bounds);
}
else
{
base.BackColor = value;
}
}
}
}
///
/// Gets or sets the fore-color of the treeNode from the base class. If this is done by setting the BackColor of the base class, then
/// the entire treeview is invalidated which can cause flicker. It is therefore achieved by setting the internal propbag and then invalidating
/// just this node.
///
private Color BaseForeColor
{
get
{
return base.ForeColor;
}
set
{
if (value != base.ForeColor)
{
if (TreeView != null && PropBag != null && TreeView.IsHandleCreated)
{
PropBag.ForeColor = value;
TreeView.Invalidate(Bounds);
}
else
{
base.ForeColor = value;
}
}
}
}
///
/// Gets the internal prop bag for this node. Users of this property should always check for null after calling this property.
///
private OwnerDrawPropertyBag PropBag
{
get
{
FieldInfo fieldInfo = GetType().GetField("propBag", BindingFlags.NonPublic | BindingFlags.Instance);
if (fieldInfo != null)
{
return (OwnerDrawPropertyBag)fieldInfo.GetValue(this);
}
return null;
}
}
private void UpdateColors()
{
if (_enabled && _selected)
{
BaseBackColor = _highlightBackColor;
BaseForeColor = _highlightForeColor;
}
else
{
BaseBackColor = _backColor;
BaseForeColor = _foreColor;
}
}
public new Color BackColor
{
get
{
return _backColor;
}
set
{
_backColor = value;
if (!_selected)
{
BaseBackColor = value;
}
}
}
public new Color ForeColor
{
get
{
return _foreColor;
}
set
{
_foreColor = value;
if (!_selected)
{
BaseForeColor = value;
}
}
}
public Color HighlightForeColor
{
get
{
return _highlightForeColor;
}
set
{
_highlightForeColor = value;
UpdateColors();
}
}
public Color HighlightBackColor
{
get
{
return _highlightBackColor;
}
set
{
_highlightBackColor = value;
UpdateColors();
}
}
public new MultiSelectTreeNode NextVisibleNode
{
get
{
return (MultiSelectTreeNode)base.NextVisibleNode;
}
}
public new MultiSelectTreeNode PrevVisibleNode
{
get
{
return (MultiSelectTreeNode)base.PrevVisibleNode;
}
}
public new MultiSelectTreeNode Parent
{
get
{
return (MultiSelectTreeNode)base.Parent;
}
}
private bool IsTreeViewEnabled()
{
// iterate through ancestors to find treeview as this node might have only just been
// added and therefore the TreeView property will still return null.
MultiSelectTreeNode node = this;
while (node != null)
{
if (node.TreeView != null)
{
return node.TreeView.Enabled;
}
node = node.Parent;
}
return true;
}
///
/// Gets a value indicating whether the tree node is in the selected state.
///
///
/// true if the tree node is in the selected state; otherwise, false.
///
public new bool IsSelected
{
get
{
return _selected;
}
}
///
/// Gets or sets a value indicating whether a CheckBox should be shown on this node. This
/// only takes effect if TreeView.ShowCheckBoxes is set to true.
///
public bool ShowCheckBox
{
get
{
return _showCheckBox;
}
set
{
if (_showCheckBox != value)
{
_showCheckBox = value;
if (TreeView != null)
{
TreeView.UpdateCheckboxVisibility(this);
}
}
}
}
///
/// Iterates through all the descendants of this .
///
public IEnumerable Descendants
{
get
{
foreach (MultiSelectTreeNode node in Nodes)
{
yield return node;
foreach (MultiSelectTreeNode n in node.Descendants)
{
yield return n;
}
}
}
}
///
/// Gets the parent tree view that the tree node is assigned to.
///
///
///
/// A that represents the parent tree view that the tree node is assigned to, or null if the node has not been assigned to a tree view.
///
public new MultiSelectTreeView TreeView
{
get
{
return (MultiSelectTreeView)base.TreeView;
}
}
#region IMultiSelectTreeNodeCollectionOwner Members
public new MultiSelectTreeNodeCollection Nodes
{
get { return _nodes; }
}
#endregion
#region IMultiSelectTreeNode Members
void IMultiSelectTreeNode.SetSelected(bool selected)
{
_enabled = IsTreeViewEnabled();
_selected = selected;
UpdateColors();
}
void IMultiSelectTreeNode.SetEnabled(bool enabled)
{
_enabled = enabled;
UpdateColors();
}
#endregion
}
}