34

我正在制作一个界面来编辑游戏的场景。基本上它由具有嵌套条件和动作的事件组成。因此,我计划使用两个树视图 - 一个用于选择事件,另一个用于选择要编辑的事件内的条件/操作。

现在,你看,如果我选择一个事件(在左侧树视图中)然后尝试在右侧树视图中选择某些内容,左侧树视图将停止显示蓝色选择矩形。这显然很糟糕,因为现在用户不知道他正在编辑哪个事件!

我发现保留有关当前选择的某种信息的唯一方法是使用 SelectedImageIndex,但这只是一个不同的小图像。

在没有关注树视图的情况下,还有其他方法可以突出显示树节点吗?我知道我可以只使用 Graphics.DrawRectangle 什么的,但我听说绘图应该在 Paint 事件中完成,而 treeview 没有绘图事件......所以我想如果我在失去焦点的情况下绘制它,然后拖动表格掉出屏幕什么的,会被“抹去”吗?

无论如何,如果您有想法,请告诉我(除了为选定的和未选定的树节点使用单独的图标)。

4

6 回答 6

71

您正在寻找的HideSelectionTreeView.

来自 MSDN:

获取或设置一个值,该值指示即使在树视图失去焦点时,选定的树节点是否仍保持突出显示。

链接:http: //msdn.microsoft.com/en-us/library/system.windows.forms.treeview.hideselection.aspx

代码:

TreeView.HideSelection = false;
于 2012-04-05T19:26:06.883 回答
17

它仍然显示,但仅以浅灰色显示,这取决于您的屏幕和当前设置,可以接近可见!

覆盖 OnDrawNode 事件。因此,您创建和新类(称为“SpecialTreeView”)继承自 Microsoft,TreeView例如class SpecialTreeView : TreeView. 然后添加以下事件覆盖:

protected override void OnDrawNode(DrawTreeNodeEventArgs e)
{
    TreeNodeStates treeState = e.State;
    Font treeFont = e.Node.NodeFont ?? e.Node.TreeView.Font;

    // Colors.
    Color foreColor = e.Node.ForeColor;
    string strDeselectedColor = @"#6B6E77", strSelectedColor = @"#94C7FC";
    Color selectedColor = System.Drawing.ColorTranslator.FromHtml(strSelectedColor);
    Color deselectedColor = System.Drawing.ColorTranslator.FromHtml(strDeselectedColor);

    // New brush.
    SolidBrush selectedTreeBrush = new SolidBrush(selectedColor);
    SolidBrush deselectedTreeBrush = new SolidBrush(deselectedColor);

    // Set default font color.
    if (foreColor == Color.Empty)
        foreColor = e.Node.TreeView.ForeColor;

    // Draw bounding box and fill.
    if (e.Node == e.Node.TreeView.SelectedNode)
    {
        // Use appropriate brush depending on if the tree has focus.
        if (this.Focused)
        {
            foreColor = SystemColors.HighlightText;
            e.Graphics.FillRectangle(selectedTreeBrush, e.Bounds);
            ControlPaint.DrawFocusRectangle(e.Graphics, e.Bounds, foreColor, SystemColors.Highlight);
            TextRenderer.DrawText(e.Graphics, e.Node.Text, treeFont, e.Bounds,
                                         foreColor, TextFormatFlags.GlyphOverhangPadding);
        }
        else
        {
            foreColor = SystemColors.HighlightText;
            e.Graphics.FillRectangle(deselectedTreeBrush, e.Bounds);
            ControlPaint.DrawFocusRectangle(e.Graphics, e.Bounds, foreColor, SystemColors.Highlight);
            TextRenderer.DrawText(e.Graphics, e.Node.Text, treeFont, e.Bounds,
                                         foreColor, TextFormatFlags.GlyphOverhangPadding);
        }
    }
    else
    {
        if ((e.State & TreeNodeStates.Hot) == TreeNodeStates.Hot)
        {
            e.Graphics.FillRectangle(SystemBrushes.Window, e.Bounds);
            TextRenderer.DrawText(e.Graphics, e.Node.Text, hotFont, e.Bounds,
                                         System.Drawing.Color.Black, TextFormatFlags.GlyphOverhangPadding);
        }
        else
        {
            e.Graphics.FillRectangle(SystemBrushes.Window, e.Bounds);
            TextRenderer.DrawText(e.Graphics, e.Node.Text, treeFont, e.Bounds,
                                         foreColor, TextFormatFlags.GlyphOverhangPadding);
        }
    }
}

编译代码,您应该在设计器的工具箱中看到“SpecialTreeView”。将您的 TreeView 替换为使用相同名称的新视图,唯一不同的是选择颜色。选中的时候会selectedColor,不选中的时候会deselectedColor

我希望这有帮助。

于 2012-04-05T19:33:09.980 回答
12

快速解决方案:

设置属性:

  • HideSelection = false;
  • DrawMode = TreeViewDrawMode.OwnerDrawText;

然后在 DrawNode 事件处理程序中简单地做:

private void treeView1_DrawNode(object sender, DrawTreeNodeEventArgs e) {
  e.DrawDefault = true;
}

在 Windwos 7 上,这会恢复旧的渲染,包括选择周围的虚线框(实际上看起来有点过时)。文本将是有焦点的白色,没有焦点的黑色。背景保持蓝色且可见。

这个答案并不新鲜,其他的也包含这些步骤,但这是最少需要的(至少在 Windows 7 中,没有测试其他操作系统)。

于 2015-12-22T00:50:28.553 回答
8

不是绝对完美的解决方案,但非常接近:

treeView.HideSelection = false;
treeView.DrawMode = TreeViewDrawMode.OwnerDrawText;
treeView.DrawNode += (o, e) =>
{
    if (!e.Node.TreeView.Focused && e.Node == e.Node.TreeView.SelectedNode)
    {
        Font treeFont = e.Node.NodeFont ?? e.Node.TreeView.Font;
        e.Graphics.FillRectangle(Brushes.Gray, e.Bounds);
        ControlPaint.DrawFocusRectangle(e.Graphics, e.Bounds, SystemColors.HighlightText, SystemColors.Highlight);
        TextRenderer.DrawText(e.Graphics, e.Node.Text, treeFont, e.Bounds, SystemColors.HighlightText, TextFormatFlags.GlyphOverhangPadding);
    }
    else
        e.DrawDefault = true;
};
treeView.MouseDown += (o, e) =>
{
    TreeNode node = treeView.GetNodeAt(e.X, e.Y);
    if (node != null && node.Bounds.Contains(e.X, e.Y))
        treeView.SelectedNode = node;
};
于 2014-10-09T12:50:24.713 回答
3

找到了一个更简单的方法:

  1. 设置 TreeView.HideSelection = True
  2. 将以下内容添加到 TreeView.AfterSelect-Callback:
private void treeViewBenutzerverwaltung_AfterSelect(object sender, TreeViewEventArgs e)
{
    // Select new node
    e.Node.BackColor = SystemColors.Highlight;
    e.Node.ForeColor = SystemColors.HighlightText;
    if (_lastSelectedNode != null)
    {
        // Deselect old node
        _lastSelectedNode.BackColor = SystemColors.Window;
        _lastSelectedNode.ForeColor = SystemColors.WindowText;
    }
    _lastSelectedNode = e.Node;
}
于 2019-10-10T12:57:24.787 回答
1

与上一个类似,但外观更类似于 Win10 标准:

treeView.HideSelection = false;
treeView.DrawMode = TreeViewDrawMode.OwnerDrawText;

treeView.DrawNode += (o, e) =>
{
    if (e.Node == e.Node.TreeView.SelectedNode)
    {
        Font font = e.Node.NodeFont ?? e.Node.TreeView.Font;
        Rectangle r = e.Bounds;
        r.Offset(0, 1);
        Brush brush = e.Node.TreeView.Focused ? SystemBrushes.Highlight : Brushes.Gray;
        e.Graphics.FillRectangle(brush, e.Bounds);
        TextRenderer.DrawText(e.Graphics, e.Node.Text, font, r, SystemColors.HighlightText, TextFormatFlags.GlyphOverhangPadding);
    }
    else
        e.DrawDefault = true;
};

treeView.MouseDown += (o, e) =>
{
    TreeNode node = treeView.GetNodeAt(e.Location);
    if (node != null && node.Bounds.Contains(e.Location)) treeView.SelectedNode = node;
};
于 2019-08-12T00:10:52.397 回答