前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >发布一款层次下拉列表控件

发布一款层次下拉列表控件

作者头像
深蓝studyzy
发布2022-06-16 13:34:50
4720
发布2022-06-16 13:34:50
举报
文章被收录于专栏:深蓝居

在项目中经常遇到树状结构的对象比如产品分类、部门结构、地区……对于这类对象的呈现,一般都使用树控件(比如VS2005自带的TreeView控件)。但是树控件的使用和操作都比较复杂,对于一些比较简单的操作,比如单选其中的一个节点的情况则可用使用下拉列表框来代替。要在DropDownList中展示出树结构的层次,那就必须在每个节点的Text前加入一定的占位符,以实现层次的效果,比如:

中国--四川----成都----乐山----绵阳--北京--广东----深圳----东莞----珠江

由于这种下拉列表控件在项目中经常使用,于是决定写一个通用的服务器控件出来。该控件继承自DropDownList,在使用中只需要为该控件设置用于数据绑定的DataTextField和DataValueField,以及新增的属性ChildProperty(string,对象的Child属性的名字)和DeepChar(string,在表示层次中使用的占位符,默认是“--”),设置了这4个属性后,在后台就只需要将树结构对象的Root节点作为DataSource,然后执行DataBind()即可。

代码语言:javascript
复制
控件Code
using System;
using System.Collections.Generic;
using System.Collections;
using System.ComponentModel;
using System.Linq;
using System.Text;
using System.Web;
using System.Web.UI;
using System.Web.UI.WebControls; 

namespace ServerControls
{
    [ToolboxData("<{0}:DropDownTreeList runat=server></{0}:DropDownTreeList>")]
    public class DropDownTreeList : DropDownList
    {
        private object dataSource;
        private int deep = 0; 

        ----重写----#region ----重写----
        public override object DataSource
        {
            get
            {
                return this.dataSource;
            }
            set
            {
                this.dataSource = value;
            }
        }
        public override void DataBind()
        {
            if (ChildProperty == null)
            {
                throw new Exception("ChildProperty参数必须设置");
            }
            this.Items.Clear();
            ListItemCollection items=ConvertTreeToList(dataSource);
            foreach (ListItem item in items)
            {
                this.Items.Add(item);
            } 

        }
        #endregion 

        ----私有方法----#region ----私有方法----
        private ListItemCollection ConvertTreeToList(object root)
        {
            deep = 0;
            ListItemCollection list =  new ListItemCollection();
            list.Add(GetListItem(root));
            ConvertTree(list, root);
            return list;
        }
        /**//// <summary>
        /// 将对象转换为ListItem
        /// </summary>
        /// <param name="root"></param>
        /// <returns></returns>
        private ListItem GetListItem(object root)
        {
            ListItem item = new ListItem();
            item.Text =GetDeepChar()+ root.GetType().GetProperty(this.DataTextField).GetValue(root, null).ToString();
            item.Value = root.GetType().GetProperty(this.DataValueField).GetValue(root, null).ToString();
            return item;
        }
        private void ConvertTree(ListItemCollection list, object root)
        {
            object childs= root.GetType().GetProperty(this.ChildProperty).GetValue(root,null);//获得Child的集合
            if(childs==null)
            {
                return;
            }
            if(!(childs is ICollection))
            {
                throw new Exception("数据源的"+ChildProperty+"属性必须实现ICollection接口");
            }
            deep++;
            foreach(object child in (ICollection)childs)
            { 

                list.Add(GetListItem(child));
                ConvertTree(list, child);//递归转换下一层节点
            }
            deep--;
        }
        /**//// <summary>
        /// 根据节点的深度返回节点前的占位字符
        /// </summary>
        /// <returns></returns>
        private string GetDeepChar()
        {
            string str = "";
            for (int i = 0; i < deep; i++)
            {
                str += DeepChar;
            }
            return str; 

        }
        #endregion
        ----公开的属性----#region ----公开的属性----
        [Description("表示深度增加的字符")]
        public string DeepChar
        {
            get
            {
                if (ViewState["DeepChar"] == null || ViewState["DeepChar"].ToString()=="")
                {
                    return "--";
                }
                return ViewState["DeepChar"].ToString();
            }
            set { ViewState["DeepChar"] = value; }
        }
        [Description("对象的子节点集合属性名")]
        public string ChildProperty
        {
            get
            {
                if (ViewState["ChildProperty"] == null)
                {
                    return null;
                }
                return ViewState["ChildProperty"].ToString();
            }
            set { ViewState["ChildProperty"] = value; }
        }
        #endregion
    }
}

具体调用示例:

1,将该控件添加到aspx页面中并设置必要的属性。

代码语言:javascript
复制
<cc1:DropDownTreeList ID="DropDownTreeList1" runat="server" 
                ChildProperty="ChildArea" DataTextField="Name" DataValueField="Code" 
                DeepChar="--">

2,在页面的后台代码中添加数据源并进行数据绑定。

代码语言:javascript
复制
后台数据绑定
public partial class _Default : System.Web.UI.Page 
{
    protected void Page_Load(object sender, EventArgs e)
    {
        this.DropDownTreeList1.DataSource = InitArea();
        this.DropDownTreeList1.DataBind();
    } 

    private Area InitArea()
    {
        Area area1 = new Area { Code = 1, Name = "中国" };
        Area area2 = new Area { Code = 2, Name = "四川" };
        Area area3 = new Area { Code = 3, Name = "北京" };
        Area area4 = new Area { Code = 4, Name = "广东" };
        Area area5 = new Area { Code = 5, Name = "成都" };
        Area area6 = new Area { Code = 6, Name = "乐山" };
        Area area7 = new Area { Code = 7, Name = "绵阳" };
        Area area8 = new Area { Code = 8, Name = "广州" };
        Area area9 = new Area { Code = 9, Name = "深圳" };
        Area area10 = new Area { Code = 10, Name = "东莞" };
        Area area11 = new Area { Code = 11, Name = "珠江" }; 

        area1.ChildArea = new List<Area> { area2, area3, area4 };
        area2.ChildArea = new List<Area> { area5, area6, area7 };
        area4.ChildArea = new List<Area> { area9, area10, area11 };
        return area1;
    }
}
public class Area
{
    public string Name { get; set; }
    public int Code { get; set; }
    public List<Area> ChildArea { get; set; }
}

运行后的效果就是: 中国 --四川 ----成都 ----乐山 ----绵阳 --北京 --广东 ----深圳 ----东莞 ----珠江

由于本人接触服务器控件开发的时间并不长,对服务器控件开发经验还不足,该控件有需要改进或有错误的地方希望大家指正。

另外这个控件目前我发现的一个问题就是我将DataBind方法完全重写了,所以控件的DataTextFormatString失效了,由于很少使用这个属性,所以一般不影响使用。不知道有什么办法能够解决这个问题?

本文参与 腾讯云自媒体同步曝光计划,分享自作者个人站点/博客。
原始发表:2008-02-25,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档