c# - 无法为 TreeView 的节点设置图标

标签 c# winforms

我有一个 TreeView 和一个包含1个图标(folder.ico)的imageList,我只想为根节点设置图标,子节点将没有图标,所以我尝试为子节点设置图像索引,但是有问题,看图: enter image description here

我的代码:

        ImageList imageList = new ImageList();
        imageList.Images.Add(Image.FromFile(System.AppDomain.CurrentDomain.BaseDirectory.Replace("\\bin\\Debug\\","") + "\\Images\\folder.ico"), Color.Transparent);
        treeView1.ImageList = imageList;
        foreach (TreeNode node in treeView1.Nodes)
        {
            foreach (TreeNode node2 in node.Nodes)
            {
                node2.ImageIndex = 100;
                node2.SelectedImageIndex = 100;
            }
        }

谢谢

编辑 我在@Killercam的回答中创建了一个自定义TreeView:

class CustomTreeView : TreeView
{
public const int NOIMAGE = -1;

public CustomTreeView()
    : base()
{
    // .NET Bug: Unless LineColor is set, Win32 treeview returns -1 (default), .NET returns Color.Black!
    base.LineColor = SystemColors.GrayText;
    base.DrawMode = TreeViewDrawMode.OwnerDrawAll;
}

protected override void OnDrawNode(DrawTreeNodeEventArgs e)
{
    // Space between Image and Label.
    const int SPACE_IL = 3;  

    // We only do additional drawing.
    e.DrawDefault = true;
    base.OnDrawNode(e);
    if (base.ShowLines && base.ImageList != null && e.Node.ImageIndex == NOIMAGE
        // exclude root nodes, if root lines are disabled
        //&& (base.ShowRootLines || e.Node.Level > 0))
            )
    {
        // Using lines & images, but this node has none: fill up missing treelines

        // Image size
        int imgW = base.ImageList.ImageSize.Width;
        int imgH = base.ImageList.ImageSize.Height;

        // Image center
        int xPos = e.Node.Bounds.Left - SPACE_IL - imgW / 2;
        int yPos = (e.Node.Bounds.Top + e.Node.Bounds.Bottom) / 2;

        // Image rect
        Rectangle imgRect = new Rectangle(xPos, yPos, 0, 0);
        imgRect.Inflate(imgW / 2, imgH / 2);

        using (Pen p = new Pen(base.LineColor, 1))
        {
            p.DashStyle = DashStyle.Dot;

            // Account uneven Indent for both lines.
            p.DashOffset = base.Indent % 2;

            // Horizontal treeline across width of image
            // account uneven half of delta ItemHeight & ImageHeight.
            int yHor = yPos + ((base.ItemHeight - imgRect.Height) / 2) % 2;

            //if (base.ShowRootLines || e.Node.Level > 0)
            //{
            //    e.Graphics.DrawLine(p, imgRect.Left, yHor, imgRect.Right, yHor);
            //}
            //else
            //{
            //    // for root nodes, if root lines are disabled, start at center
            //    e.Graphics.DrawLine(p, xPos - (int)p.DashOffset, yHor, imgRect.Right, yHor);
            //}

            e.Graphics.DrawLine(p,
                    (base.ShowRootLines || e.Node.Level > 0) ? imgRect.Left : xPos - (int)p.DashOffset,
                    yHor, imgRect.Right, yHor);
            if (!base.CheckBoxes && e.Node.IsExpanded)
            {
                // Vertical treeline , offspring from NodeImage center to e.Node.Bounds.Bottom
                // yStartPos: account uneven Indent and uneven half of delta ItemHeight & ImageHeight
                int yVer = yHor + (int)p.DashOffset;
                e.Graphics.DrawLine(p, xPos, yVer, xPos, e.Node.Bounds.Bottom);
            }
        }
    }
}

protected override void OnAfterCollapse(TreeViewEventArgs e)
{
    base.OnAfterCollapse(e);
    if (!base.CheckBoxes && base.ImageList != null && e.Node.ImageIndex == NOIMAGE)
    {
        // DrawNode event not raised: redraw node with collapsed treeline
        base.Invalidate(e.Node.Bounds);
    }
}
}

然后在我的代码中使用它:

private void TestCustomTreeView_Load(object sender, EventArgs e)
    {
        // @Killercam EDIT: Set the default Image to one that is not used.
        valForm.siteTreeView.ImageIndex = 100;
        valForm.siteTreeView.SelectedImageIndex = 100;

        TemplateCustomTreeView myTree = new TemplateCustomTreeView();
        myTree.Font = new System.Drawing.Font("Tahoma", 8.25F, System.Drawing.FontStyle.Regular, System.Drawing.GraphicsUnit.Point, ((System.Byte)(0)));

        myTree.Location = new System.Drawing.Point(6, 26);
        myTree.Name = "tree";
        myTree.Scrollable = true;
        myTree.Size = new System.Drawing.Size(320, 296);
        myTree.ImageList = imageList1;
        /*Add item*/
        TreeNode node = new TreeNode();
        node.Name = "abc1";
        node.Text = "abc1";
        node.ImageIndex = 0;
        myTree.Nodes.Add(node);
        TreeNode node3 = new TreeNode();
        node3.Name = "abc2";
        node3.Text = "abc2";
        node3.ImageIndex = -1;            
        node.Nodes.Add(node3);
        ////
        TreeNode node2 = new TreeNode();
        node2.Name = "abc3";
        node2.Text = "abc3";
        node2.ImageIndex = 0;
        myTree.Nodes.Add(node2);
        this.Controls.AddRange(new System.Windows.Forms.Control[] { myTree });

    }
}

结果还是不行,文字前面还是有一个文件夹图标! enter image description here

最佳答案

你无法完成你所要求的事情

"All I want is don't show icon at child node."

不覆盖控件。我还发现您无法使用标准 WinForms TreeView 显示不同节点的不同图像。下面是一些可以使 TreeView 看起来更好的代码;即为子节点绘制树线的一小部分。

class CustomTreeView : TreeView
{
    public const int NOIMAGE = -1;

    public CustomTreeView()
        : base()
    {
        // .NET Bug: Unless LineColor is set, Win32 treeview returns -1 (default), .NET returns Color.Black!
        base.LineColor = SystemColors.GrayText;
        base.DrawMode = TreeViewDrawMode.OwnerDrawAll;
    }

    protected override void OnDrawNode(DrawTreeNodeEventArgs e)
    {
        // Space between Image and Label.
        const int SPACE_IL = 3;  

        // We only do additional drawing.
        e.DrawDefault = true;
        base.OnDrawNode(e);
        if (base.ShowLines && base.ImageList != null && e.Node.ImageIndex == NOIMAGE
            // exclude root nodes, if root lines are disabled
            //&& (base.ShowRootLines || e.Node.Level > 0))
                )
        {
            // Using lines & images, but this node has none: fill up missing treelines

            // Image size
            int imgW = base.ImageList.ImageSize.Width;
            int imgH = base.ImageList.ImageSize.Height;

            // Image center
            int xPos = e.Node.Bounds.Left - SPACE_IL - imgW / 2;
            int yPos = (e.Node.Bounds.Top + e.Node.Bounds.Bottom) / 2;

            // Image rect
            Rectangle imgRect = new Rectangle(xPos, yPos, 0, 0);
            imgRect.Inflate(imgW / 2, imgH / 2);

            using (Pen p = new Pen(base.LineColor, 1))
            {
                p.DashStyle = DashStyle.Dot;

                // Account uneven Indent for both lines.
                p.DashOffset = base.Indent % 2;

                // Horizontal treeline across width of image
                // account uneven half of delta ItemHeight & ImageHeight.
                int yHor = yPos + ((base.ItemHeight - imgRect.Height) / 2) % 2;

                //if (base.ShowRootLines || e.Node.Level > 0)
                //{
                //    e.Graphics.DrawLine(p, imgRect.Left, yHor, imgRect.Right, yHor);
                //}
                //else
                //{
                //    // for root nodes, if root lines are disabled, start at center
                //    e.Graphics.DrawLine(p, xPos - (int)p.DashOffset, yHor, imgRect.Right, yHor);
                //}

                e.Graphics.DrawLine(p,
                        (base.ShowRootLines || e.Node.Level > 0) ? imgRect.Left : xPos - (int)p.DashOffset,
                        yHor, imgRect.Right, yHor);
                if (!base.CheckBoxes && e.Node.IsExpanded)
                {
                    // Vertical treeline , offspring from NodeImage center to e.Node.Bounds.Bottom
                    // yStartPos: account uneven Indent and uneven half of delta ItemHeight & ImageHeight
                    int yVer = yHor + (int)p.DashOffset;
                    e.Graphics.DrawLine(p, xPos, yVer, xPos, e.Node.Bounds.Bottom);
                }
            }
        }
    }

    protected override void OnAfterCollapse(TreeViewEventArgs e)
    {
        base.OnAfterCollapse(e);
        if (!base.CheckBoxes && base.ImageList != null && e.Node.ImageIndex == NOIMAGE)
        {
            // DrawNode event not raised: redraw node with collapsed treeline
            base.Invalidate(e.Node.Bounds);
        }
    }
}

这将为您提供一个如下所示的 TreeView:

CustomTreeView

这里我的主节点(Node[0])是没有指定Image的节点,并且是您的File1/所需的节点File2 节点。

我希望这会有所帮助。

关于c# - 无法为 TreeView 的节点设置图标,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/11500917/

相关文章:

vb.net - 如何卸载 VB.NET 中所有打开的表单?

.net - 自动调整 TableLayoutPanel 的大小

c# - 如何将表单添加到控制台应用程序以便用户可以选择文件?

c# - 将 gZipStream 与一个或两个内存流一起使用会产生很大的不同

c# - 删除调整大小的能力

c# - XNA Keyboard.GetState() 不读取数字/字符键,而是读取其他键

c# - 错误 "Sanitizer provider is not configured in the web.config file. "

c# - 如果 picturebox.image == Properties.Resources.Image

c# - 富文本框 CtrlI

c# - 如何使用 View 切换导航在 Prism 6 WPF 应用程序中注册模块?