在wenform中使用treecombobox

Winform中如何实现下拉树效果

简介:Winform中如何实现下拉树效果(类似于ComboBox下拉时显示的是树状结构),

如果是BS的WebForm就有很多解决方案了,Devexpress就更方便,利用DropdownList和TreeView去组合控件实现

但是CS中的Winform的传统窗体控件和Devexpress里面去实现此功能就比较麻烦了

而此文,我们要讲解的就是如何利用Winform的传统用户控件去实现下拉树的效果

主要利用TextBox,TreeView,Button等控件事件组合而成类似于ComboBox下拉出现树状结构

下面就看看我们要实现的下拉树的效果图:

说明:上图所示的就是我们要实现的效果当点击下拉控件时,出现一个树状结构,

当我们选择树节点时(双击),把相应的节点值,放入上面的文本框中

好了,下面我们就马上去看看,这样的功能是如何实现的呢?


首先我们需要建立一个用户控件,如图:

说明:不需要再进行任何操作


接下来,我们需要放一张图片文件,这张图片文件是为了模拟下拉框右边的三角下拉按钮用的,如图:


接下来就是完全的代码模块了,就这样的界面操作,是如何让我们做到下拉树效果的呢?

在用户控件窗体CS文件中,我们需要添加如下代码:

一.自定义成员变量

二.内部辅助方法

三.对TextBox的事件处理

四.对TreeView的事件处理

五.对Button的事件处理

六.对自身的事件处理

七.外部属性封装

八.构造函数


别看上面说的步骤很多,其实加起来也就几百行代码,也没有什么难点,接下来我们就依次展示我们八大步骤模块的代码:

一.自定义成员变量

#region 自定义成员变量
 
        private TextBox m_TextBox;
        private TreeView m_TreeView;
        private Button m_DropdownButton;
        private int m_MaxDropDownItems = 8;
        //是否正在显示下拉列表
        private bool b_Dropdown = false;
        //使能
        private bool b_Enabled = true;
        //事件
        public event EventHandler DropDown;
        public event EventHandler DropDownClosed;
        public event EventHandler EnableChanged;
        public event TreeViewCancelEventHandler BeforeExpand;
        public event TreeViewCancelEventHandler BeforeCollapse;
        public event TreeViewEventHandler AfterExpand;
        public event TreeViewEventHandler AfterCollapse;
 
        #endregion

二.内部辅助方法

#region 内部辅助方法
 
        ///
        /// 创建并初始化所有控件,包括添加事件处理函数
        ///
        /// PengZhen
        ///
        private void InitControls()
        {
            //TextBox
            this.m_TextBox = new TextBox();
            this.m_TextBox.KeyDown += new KeyEventHandler(m_TextBox_KeyDown);
            this.m_TextBox.Parent = this;
 
            //Button
            this.m_DropdownButton = new Button();
            //Assembly asm = Assembly.GetExecutingAssembly();
            //System.IO.Stream stream = asm.GetManifestResourceStream(asm.GetName().Name + “.BM_dropdown.bmp”);
            Bitmap bm = new Bitmap(this.GetType(), “BM_dropdown.bmp”);
            this.m_DropdownButton.Image = bm;
            this.m_DropdownButton.Width = 16;
            this.m_DropdownButton.Height = this.m_TextBox.Height - 2;
            this.m_DropdownButton.Location = new Point(this.m_TextBox.Right - 18, 1);
            this.m_DropdownButton.Click += new EventHandler(m_DropdownButton_Click);
            this.m_DropdownButton.FlatStyle = FlatStyle.Flat;
            this.m_DropdownButton.Parent = this;
            this.m_DropdownButton.BringToFront();
 
            //TreeView
            this.m_TreeView = new TreeView();
            this.m_TreeView.Visible = false;
            this.m_TreeView.DoubleClick += new EventHandler(m_TreeView_DoubleClick);
            this.m_TreeView.KeyDown += new KeyEventHandler(m_TreeView_KeyDown);
            this.m_TreeView.LostFocus += new EventHandler(m_TreeView_LostFocus);
            this.m_TreeView.BeforeExpand += new TreeViewCancelEventHandler(m_TreeView_BeforeExpand);
            this.m_TreeView.BeforeCollapse += new TreeViewCancelEventHandler(TreeComboBox_BeforeCollapse);
            this.m_TreeView.AfterExpand += new TreeViewEventHandler(m_TreeView_AfterExpand);
            this.m_TreeView.AfterCollapse += new TreeViewEventHandler(TreeComboBox_AfterCollapse);
            this.m_TreeView.Location = new Point(0, 0);
            this.m_TreeView.Parent = null;
 
            this.LostFocus += new EventHandler(TreeComboBox_LostFocus);
        }
 
        ///
        /// 布局所有控件,让TextBox尺寸适应容器尺寸
        ///
        /// PengZhen
        ///
        private void LayoutControls()
        {
            this.m_TextBox.Width = this.Width;
            this.Height = this.m_TextBox.Height;
            this.m_DropdownButton.Left = this.m_TextBox.Right - 18;
            this.m_DropdownButton.Height = this.m_TextBox.Height - 2;
            this.m_TreeView.Width = this.Width;
            this.m_TreeView.Height = (int) ((this.Font.Height + 3)*this.m_MaxDropDownItems);
        }
 
        ///
        /// 显示下拉列表
        ///
        /// PengZhen
        ///
        private void ShowDropDown()
        {
            if (this.Parent == null)
                return;
 
            // 智能计算显示的位置,尝试在下方显示,如果没有足够空间,则在上方显示
            // 尝试在下方的位置(现在只是相对父窗口的相对位置)
            Point pos = new Point(this.Left, this.Bottom - 1);
 
            // 把位置映射到顶层窗口,获取父窗口的屏幕坐标
            Point parentPos = this.Parent.PointToScreen(this.Parent.Location);
            // 获取顶层窗口的屏幕坐标
            Point topParentPos = this.TopLevelControl.PointToScreen(this.Parent.Location);
            // 把相对父窗口的位置变换为相对顶级窗口的位置,因为popup的父是顶级窗口
            pos.Offset(parentPos.X - topParentPos.X, parentPos.Y - topParentPos.Y);
 
            // 检查是否有足够空间用于在label下方显示day picker
            if ((pos.Y + this.m_TreeView.Height) > this.TopLevelControl.ClientRectangle.Height)
            {
                // 没有足够的空间(超出了顶级窗口客户区),尝试在上方显示将Y方向,向上平移
                pos.Y -= (this.Height + this.m_TreeView.Height);
                if (pos.Y < 0)
                {
                    // 如果上方仍然没有空间显示,则显示在顶级窗口的底部
                    pos.Y = (this.TopLevelControl.ClientRectangle.Height - this.m_TreeView.Height);
                }
            }
 
            // 尝试停靠,如果右边超过顶级窗口的客户区,则将控件向左移动,并紧靠在顶级窗口右侧
            if ((pos.X + this.m_TreeView.Width) > this.TopLevelControl.ClientRectangle.Width)
                pos.X = (this.TopLevelControl.ClientRectangle.Width - this.m_TreeView.Width);
 
            this.m_TreeView.Location = pos; // this.Parent.PointToScreen(pt);
            this.m_TreeView.Visible = true;
            this.m_TreeView.Parent = this.TopLevelControl;
            this.m_TreeView.BringToFront();
            this.b_Dropdown = true;
            //raise event
            if (this.DropDown != null)
                this.DropDown(this, EventArgs.Empty);
            this.m_TreeView.Focus();
        }
 
        ///
        /// 隐藏下拉列表
        ///
        /// PengZhen
        ///
        private void HideDropDown()
        {
            if (this.DropDownClosed != null)
                this.DropDownClosed(this, EventArgs.Empty);
            this.m_TreeView.Parent = null;
            this.m_TreeView.Visible = false;
            this.b_Dropdown = false;
        }
 
        #endregion

三.对TextBox的事件处理

#region 事件处理 - TextBox
 
        ///
        /// 在编辑框中按下按键
        ///
        /// PengZhen
        ///
        ///
        ///
        private void m_TextBox_KeyDown(object sender, KeyEventArgs e)
        {
            if (e.KeyCode == Keys.Enter)
            {
                if (this.b_Dropdown)
                    this.HideDropDown();
                else
                    this.ShowDropDown();
            }
            else if (e.KeyCode == Keys.Down)
            {
                this.ShowDropDown();
                this.m_TreeView.Focus();
            }
        }
 
        #endregion

四.对TreeView的事件处理

#region 事件处理 - TreeView
 
        ///
        /// 在下拉列表中选择了一个节点的事件处理!
        ///
        /// PengZhen
        ///
        ///
        ///
        private void m_TreeView_KeyDown(object sender, KeyEventArgs e)
        {
            //如果按下回车表示要选中当前节点
            if (e.KeyCode == Keys.Enter)
            {
                this.m_TreeView_DoubleClick(sender, EventArgs.Empty);
            }
        }
 
        ///
        /// 失去焦点时,如果不是被下拉按钮夺取的焦点,则隐藏它!
        ///
        /// PengZhen
        ///
        ///
        ///
        private void m_TreeView_LostFocus(object sender, EventArgs e)
        {
            if (!this.m_DropdownButton.Focused)
                this.HideDropDown();
        }
 
        ///
        /// 在下拉列表中双击
        ///
        /// PengZhen
        ///
        ///
        ///
        private void m_TreeView_DoubleClick(object sender, EventArgs e)
        {
            TreeNode node = this.m_TreeView.SelectedNode;
            if (node != null)
                this.m_TextBox.Text = node.Text;
 
            if (this.b_Dropdown)
            {
                this.HideDropDown();
            }
        }
 
        ///
        /// 折叠后事件
        ///
        /// PengZhen
        ///
        ///
        ///
        private void TreeComboBox_AfterCollapse(object sender, TreeViewEventArgs e)
        {
            if (this.AfterCollapse != null)
                this.AfterCollapse(this, e);
        }
 
        ///
        /// 折叠前事件
        ///
        /// PengZhen
        ///
        ///
        ///
        private void TreeComboBox_BeforeCollapse(object sender, TreeViewCancelEventArgs e)
        {
            if (this.BeforeCollapse != null)
                this.BeforeCollapse(this, e);
        }
 
        ///
        /// 展开后事件
        ///
        /// PengZhen
        ///
        ///
        ///
        private void m_TreeView_AfterExpand(object sender, TreeViewEventArgs e)
        {
            if (this.AfterExpand != null)
                this.AfterExpand(this, e);
        }
 
        ///
        /// 展开前事件
        ///
        /// PengZhen
        ///
        ///
        ///
        private void m_TreeView_BeforeExpand(object sender, TreeViewCancelEventArgs e)
        {
            if (this.BeforeExpand != null)
                this.BeforeExpand(this, e);
        }
 
 
        #endregion

五.对Button的事件处理

#region 事件处理 - Button
 
        ///
        /// 下拉按钮点击事件
        ///
        /// PengZhen
        ///
        ///
        ///
        private void m_DropdownButton_Click(object sender, EventArgs e)
        {
            //throw new Exception(“The method or operation is not implemented.”);
            if (this.b_Dropdown)
                this.HideDropDown();
            else
                this.ShowDropDown();
        }
 
        #endregion

六.对自身的事件处理

#region 事件处理 - 自身
 
        ///
        /// 失去焦点
        ///
        /// PengZhen
        ///
        ///
        ///
        private void TreeComboBox_LostFocus(object sender, EventArgs e)
        {
            if (this.b_Dropdown)
                this.HideDropDown();
        }
 
        ///
        /// 重设尺寸
        ///
        /// PengZhen
        ///
        ///
        protected override void OnResize(EventArgs e)
        {
            base.OnResize(e);
            this.LayoutControls();
        }
 
        ///
        /// 尺寸改变
        ///
        /// PengZhen
        ///
        ///
        protected override void OnSizeChanged(EventArgs e)
        {
            base.OnSizeChanged(e);
            this.LayoutControls();
        }
 
        #endregion

七.外部属性封装

#region 外部属性封装
 
        ///
        /// 获取节点集合
        ///
        /// PengZhen
        ///
        public TreeNodeCollection Nodes
        {
            get { return this.m_TreeView.Nodes; }
        }
 
        ///
        /// 设置或者获取节点的图片列表
        ///
        /// PengZhen
        ///
        public ImageList ImageList
        {
            get { return this.m_TreeView.ImageList; }
            set { this.m_TreeView.ImageList = value; }
        }
 
        ///
        /// 重写enabled属性
        ///
        /// PengZhen
        ///
        public new bool Enabled
        {
            get { return this.b_Enabled; }
            set
            {
                if (this.b_Enabled != value)
                {
                    this.b_Enabled = value;
                    this.m_DropdownButton.Enabled = value;
                    this.b_Enabled = value;
                    //当灰掉时,隐藏下拉列表
                    if (!this.b_Enabled && this.b_Dropdown)
                        this.HideDropDown();
                    if (this.EnableChanged != null)
                        this.EnableChanged(this, EventArgs.Empty);
                }
            }
        }
 
        ///
        /// 字体
        ///
        /// PengZhen
        ///
        public override Font Font
        {
            get { return base.Font; }
            set
            {
                base.Font = value;
                this.m_TextBox.Font = value;
                this.m_TreeView.Font = value;
                //调整布局
                this.LayoutControls();
            }
        }
 
        ///
        /// 重写Text
        ///
        /// PengZhen
        ///
        public override string Text
        {
            get { return this.m_TextBox.Text; }
            set
            {
                if (this.Text != value)
                {
                    this.m_TextBox.Text = value;
                }
            }
        }
 
        ///
        /// 是否显示lines
        ///
        /// PengZhen
        ///
        public bool ShowLines
        {
            get { return this.m_TreeView.ShowLines; }
            set { this.m_TreeView.ShowLines = value; }
        }
 
        ///
        /// 是否显示+ -按钮
        ///
        /// PengZhen
        ///
        public bool ShowPlusMinus
        {
            get { return this.m_TreeView.ShowPlusMinus; }
            set { this.m_TreeView.ShowPlusMinus = value; }
        }
 
        ///
        /// 是否显示root lines
        ///
        /// PengZhen
        ///
        public bool ShowRootLines
        {
            get { return this.m_TreeView.ShowRootLines; }
            set { this.m_TreeView.ShowRootLines = value; }
        }
 
        ///
        /// 是否显示root lines
        ///
        /// PengZhen
        ///
        public bool ShowNodeToolTips
        {
            get { return this.m_TreeView.ShowNodeToolTips; }
            set { this.m_TreeView.ShowNodeToolTips = value; }
        }
 
        ///
        /// 获取或者设置选中的节点!
        ///
        /// PengZhen
        ///
        public TreeNode SelectedNode
        {
            get { return this.m_TreeView.SelectedNode; }
            set
            {
                this.m_TreeView.SelectedNode = value;
                if (value != null)
                    this.Text = value.Text;
            }
        }
 
        ///
        /// 设置下拉最多项
        ///
        /// PengZhen
        ///
        public int MaxDropDownItems
        {
            get { return this.m_MaxDropDownItems; }
            set
            {
                if (this.m_MaxDropDownItems != value)
                {
                    this.m_MaxDropDownItems = value;
                    this.m_TreeView.Height = this.m_TextBox.Height*value;
                }
            }
        }
 
        #endregion

八.构造函数

///
        /// 构造函数
        ///
        public TreeComboBox()
        {
            this.InitControls();
            this.LayoutControls();
        }

OK,经过上面八个步骤,我们的下拉树自定义控件就完成了,接下来,就看下我们是如何使用它的

首先我们需要生成一下项目,成功后我们将在我们的工具栏中,看到如下控件:

说明:做到此处出现错误时,请参照文章尾部提出的解决方案…


接下来我们就需要把该控件拖到我们的窗体中


这些完成的之后,接下来我们就是我们真正的代码调用模块了,我们需要在调用下拉树控件窗体中添加如下代码:

首先是绑定数据:

///
        /// 数据分类数据绑定
        ///
        /// PengZhen
        ///
        private void BindDataType()
        {
            for (int i = 0; i < 4; i++)
            {
                tcbDataType.Nodes.Add(“key” + i, “Depart” + i, 0, 0);
                for (int j = 0; j < i + 1; j++)
                {
                    tcbDataType.Nodes[i].Nodes.Add(“key_child” + i + j, “User” + i + j, 2, 2);
                }
            }
        }
说明:此处的for循环代表树的级数,在此我们可以看出,此代码将会生成一个两级树,如果我们想出现多级,只需要加入多个循环,就可以了

如果您嫌麻烦,可以封装一下通用方法


其次是添加事件:

1.添加折叠事件

///
        /// 下拉树折叠后事件
        ///
        /// PengZhen
        ///
        ///
        ///
        private void tcbDataType_AfterCollapse(object sender, TreeViewEventArgs e)
        {
            if (e.Node.ImageIndex == 1)
                e.Node.ImageIndex = e.Node.SelectedImageIndex = 0;
        }
2.添加展开事件

///
        /// 下拉树展开后事件
        ///
        /// PengZhen
        ///
        ///
        ///
        private void tcbDataType_AfterExpand(object sender, TreeViewEventArgs e)
        {
          if (e.Node.ImageIndex == 0)
               e.Node.ImageIndex = e.Node.SelectedImageIndex = 1;
        }

最后,我们需要注册事件:

可以在调用窗体的构造函数中加入如下代码:

this.tcbDataType.AfterExpand += new TreeViewEventHandler(tcbDataType_AfterExpand);
            this.tcbDataType.AfterCollapse += new TreeViewEventHandler(tcbDataType_AfterCollapse);

或者查看下拉树控件的属性,在相应的事件中,选择我们编写的事件即可,如下图:


好了,到此,我们的下拉树就完成了,让我们看下效果,如图:


细心的人一定会发现,不对啊,这个数结构和文章的第一张图里面不一样啊,是的,因为少了节点图片,

我们做的这么久还没有给节点加图片的操作呢,好了,接下来,就是我最后的操作了

给树节点加图片操作,分三步

1.拖一个ImageList控件到窗体中

2.给ImageList添加图片列表,如图

3.将我们的下拉树控件中的属性ImageList设置成我们刚拖进来的ImageList控件,就可以了


好的,做到这,我们的下拉树控件的一个完整效果就完成了,运行之后,我们就会看见文章刚出现的效果图的画面了

不过,做到这,我们还要最后一个收尾操作,那就是获取下拉树中用户选择的节点键和值

代码如下:

///
        /// 查询
        ///
        /// PengZhen
        ///
        ///
        ///
        private void btSelect_Click(object sender, EventArgs e)
        {
            string value = tcbDataType.SelectedNode.Name;
 
            MessageBox.Show(value);
 
            string text = tcbDataType.Text.Trim();
 
            MessageBox.Show(text);
        }

OK,OK,终于,整个下拉树的完整操作就全部实现了,有木有发现很简单呢,嘿嘿


备注:此控件请放在窗体中使用,别放在用户控件中使用,至于为什么,您可以试试,就会出现意想不到的效果噢,

而出现此问题,就留给你们自己去思考,解决吧,多去试试,变通思想,会出现更奇妙的事噢,

好了,就写到这,让你们自己去见证奇迹的时刻把,嘿嘿…

写点题外话,当你们去完成以上操作时,可能会出现以下错误,去看看下面的链接,就知道怎么去解决了

System.ArgumentException无法在类中找到资源 
————————————————
版权声明:本文为CSDN博主「GlenPeng」的原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接及本声明。
原文链接:https://blog.csdn.net/pengzhen8805/article/details/14005181

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值