很久都没有写点什么出来分享了,最近在做多级树的时候,发现来来回回写过很多遍,于是封装成用户控件,以方便日后重复使用.

首先上效果:

我们看到以上2种效果,都是支持任意级的,这里源码中使用的是递归,以便高效的完成HTML的渲染.

下面上代码,代码中解释的都很详细了,我就不再细说.下面将有示例调用演示:

 public partial class  UC_MultiLevelTree : System.Web.UI.UserControl
{
#region 数据相关属性 /// <summary>
/// 要绑定的数据源
/// </summary>
public DataTable DataSource { get; set; } /// <summary>
/// 多级树显示文本所在列列名
/// </summary>
public string TextFeild { get; set; } /// <summary>
/// 多级树单条数据识别列列名(即选择项的值)
/// </summary>
public string ValueFeild { get; set; } /// <summary>
/// 多级树层级区别列列名(仅限单个列区分层级)
/// </summary>
public string LevelFeild { get; set; } /// <summary>
/// 多级树顶级的父项值
/// </summary>
public string TopLevelFeildValue { get; set; } #endregion #region 显示相关属性 /// <summary>
/// 是否显示多选框,默认为显示
/// </summary>
public bool ShowCheckBox { get; set; } /// <summary>
/// 是否显示自定义根节点
/// </summary>
public bool ShowCustomerRoot { get; set; } /// <summary>
/// 自定义根节点文本
/// </summary>
public string CustomerRootText { get; set; } /// <summary>
/// 多级树宽度,可为像素或者百分比
/// </summary>
public string Width { get; set; } /// <summary>
/// 多级树高度,可为像素或者百分比
/// </summary>
public string Height { get; set; } /// <summary>
/// 展开符号(可为HTML代码)
/// </summary>
public string ExtendSign { get; set; } /// <summary>
/// 收缩符号(可为HTML代码) /// </summary>
public string ShrinkSign { get; set; } /// <summary>
/// 每级与上级空格个数
/// </summary>
public int LevelSeparatorCount { get; set; } /// <summary>
/// 默认展开级别
/// </summary>
public int ExtendLevelNum { get; set; } #endregion #region 私有变量 /// <summary>
/// 扩展标记的HTML
/// </summary>
private string StrExtendSign; /// <summary>
/// 收缩标记的HTML
/// </summary>
private string StrShrinkSign; /// <summary>
/// 多选框的HTML
/// </summary>
private string StrCheckbox; /// <summary>
/// 子层级开始符号的HTML
/// </summary>
private string LevelSeparator = "&nbsp;"; #endregion protected override void OnInit(EventArgs e)
{
base.OnInit(e);
this.ShowCheckBox = true;
this.Width = "100%";
this.Height = "100%";
this.ExtendSign = "[+]";
this.ShrinkSign = "[-]";
this.TopLevelFeildValue = CRMCommon.strNullGuid;
this.LevelSeparatorCount = ;
this.ExtendLevelNum = ;
} protected void Page_Load(object sender, EventArgs e)
{ } public void DataBind()
{
this.StrCheckbox = this.ShowCheckBox ? "<input type='checkbox' class='MLT_Checkbox'/>" : "";
this.StrExtendSign = "<span class='MLT_ExtendSign' {0}>" + this.ExtendSign + "</span>";
this.StrShrinkSign = "<span class='MLT_ShrinkSign' {0}>" + this.ShrinkSign + "</span>";
this.ltMultiLevelTreeHtml.Text = RenderTree(this.TopLevelFeildValue, );
} private string RenderTree(string parentValue, int level)
{
StringBuilder sb = new StringBuilder(); string extendSignHtml = "";
string shrinkSignHtml = ""; //收缩,展开按钮的显示控制
if (level < this.ExtendLevelNum)
{
extendSignHtml = string.Format(this.StrExtendSign, "style='display:none;'");
shrinkSignHtml = string.Format(this.StrShrinkSign, "");
}
else
{
extendSignHtml = string.Format(this.StrExtendSign, "");
shrinkSignHtml = string.Format(this.StrShrinkSign, "style='display:none;'");
} //自定义根节点
if (level == )
{
sb.AppendFormat("<div class='MLT_Panel' style='width:{0};height:{1}'>", this.Width, this.Height);
if (this.ShowCustomerRoot)
{
sb.AppendFormat("<div class='MLT_Item' level='{0}' rel=''>{1}<span class='MLT_Item_Text'>{2}</span></div>", level, extendSignHtml + shrinkSignHtml + this.StrCheckbox, this.CustomerRootText); level += ;
}
sb.Append(RenderTree(parentValue, level));
sb.Append("</div>");
} else if (level != )
{
//数据项绑定
if (this.DataSource != null && this.DataSource.Rows.Count > )
{
string levelSeparator = ""; if (level > )
{
levelSeparator += "<span class='MLT_LevelSeparator'>";
for (int i = ; i <= (level - ) * this.LevelSeparatorCount; i++)
{
levelSeparator += this.LevelSeparator;
}
levelSeparator += "</span>";
} DataRow[] drList = this.DataSource.Select(string.Format("{0}='{1}'", this.LevelFeild, parentValue)); if (drList != null && drList.Length > )
{
level += ;
foreach (DataRow dr in drList)
{
string childHtml = RenderTree(dr[ValueFeild].ToString(), level); string signs = string.IsNullOrWhiteSpace(childHtml) ? "<span class='MLT_ExtendSignPlaceholder'></span>" : extendSignHtml + shrinkSignHtml; sb.AppendFormat("<div class='MLT_Item' level='{0}' rel='{1}' parent='{2}' {3}>{4}<span class='MLT_Item_Text'>{5}</span></div>", level - , dr[ValueFeild], dr[LevelFeild], level - > this.ExtendLevelNum ? "style='display:none;'" : "", levelSeparator + signs + this.StrCheckbox, dr[TextFeild]); if (!string.IsNullOrWhiteSpace(childHtml))
{
sb.Append(childHtml);
}
}
}
}
}
return sb.ToString(); }
}
 <%@ Control Language="C#" AutoEventWireup="true" CodeFile="UC_MultiLevelTree.ascx.cs" Inherits="UC_MultiLevelTree" %>
<asp:Literal runat="server" ID="ltMultiLevelTreeHtml"></asp:Literal>
<script>
$(function () {
$(".MLT_Item").click(function (e) {
e.stopPropagation();
$(".MLT_Item").removeClass("MLT_Item_hover");
$(this).addClass("MLT_Item_hover");
extendItem(this);
}); $(".MLT_ExtendSign").click(function (e) {
e.stopPropagation();
var event = e.currentTarget;
var item = $(event).parent();
extendItem(item);
}) $(".MLT_ShrinkSign").click(function (e) {
e.stopPropagation();
var event = e.currentTarget;
var item = $(event).parent();
shrinkItem(item);
}) $(".MLT_Checkbox").click(function (e) {
e.stopPropagation();
var event = e.currentTarget;
var item = $(event).parent();
var checked = $(event).attr("checked");
checkItems(item, checked);
});
}); //展开项
function extendItem(obj) {
var rel = $(obj).attr("rel");
if (rel != undefined && rel.length > 0) {
$(obj).siblings("div[parent=" + rel + "]").each(function () {
$(this).show();
});
}
else {
$(obj).siblings("div[level=2]").show();
}
$(obj).find(".MLT_ExtendSign").hide();
$(obj).find(".MLT_ShrinkSign").show();
} //收缩项
function shrinkItem(obj) {
var rel = $(obj).attr("rel");
if (rel != undefined && rel.length > 0) {
$(obj).siblings("div[parent=" + rel + "]").each(function () {
$(this).hide();
$(this).find(".MLT_ExtendSign").show();
$(this).find(".MLT_ShrinkSign").hide();
shrinkItem(this);
});
}
else {
$(obj).siblings("div[level!=1]").hide();
$(obj).siblings("div[level!=1]").find(".MLT_ExtendSign").show();
$(obj).siblings("div[level!=1]").find(".MLT_ShrinkSign").hide();
}
$(obj).find(".MLT_ExtendSign").show();
$(obj).find(".MLT_ShrinkSign").hide();
} //选择项
function checkItems(obj, checked) {
var rel = $(obj).attr("rel");
if (rel != undefined && rel.length > 0) {
if (checked) {
$(obj).siblings("div[parent=" + rel + "]").each(function () {
$(this).find("input[type=checkbox]").attr("checked", "checked");
checkItems(this, checked);
});
}
else {
$(obj).siblings("div[parent=" + rel + "]").each(function () {
$(this).find("input[type=checkbox]").removeAttr("checked");
checkItems(this, checked);
});
}
}
else {
if (checked) {
$(obj).parent().find("input[type=checkbox]").attr("checked", "checked");
}
else {
$(obj).parent().find("input[type=checkbox]").removeAttr("checked");
}
}
}
</script>
<style type="text/css">
.MLT_Panel
{
white-space: nowrap;
overflow: auto;
} .MLT_Item
{
font-size: 12px;
line-height: 20px;
cursor: pointer;
} .MLT_Item_hover
{
background-color: rgb(167, 205, 240);
} .MLT_Item span
{
line-height: 20px;
display: inline-block;
} .MLT_Checkbox
{
position: relative;
width: 12px;
height: 12px;
margin: 0 2px;
bottom: 2px;
} .MLT_ExtendSign, .MLT_ShrinkSign, .MLT_ExtendSignPlaceholder
{
font-family: "宋体";
width: 18px;
text-align: center;
}
</style>

示例:

使用原数据:

生成HTML中,每一级每一条都包含在使用相同class的DIV当中,不同之外在于自定义的level,parent,rel等属性,请参见代码.

下列为调用代码方法,各参数可自行设定,说见用户控件CS代码:

由于时间问题,可能很多地方不便细说.如有更多疑问,请加QQ群:ASP.NET高级群,群号: 261882616

不再为无限级树结构烦恼,且看此篇s的更多相关文章

  1. 集合类--最详细的面试宝典--看这篇就够用了(java 1.8)

    看了一个星期源码,搜索上百篇博文,终于总结出了集合类的所有基础知识点,学集合,看这篇就够用了!!! 篇幅有点长, 如果你能全部理解,java最重要的集合就不怕了,秒过面试!!!(本篇素材来自网络,如有 ...

  2. React入门看这篇就够了

    摘要: 很多值得了解的细节. 原文:React入门看这篇就够了 作者:Random Fundebug经授权转载,版权归原作者所有. React 背景介绍 React 入门实例教程 React 起源于 ...

  3. 任务21 :了解ASP.NET Core 依赖注入,看这篇就够了

    DI在.NET Core里面被提到了一个非常重要的位置, 这篇文章主要再给大家普及一下关于依赖注入的概念,身边有工作六七年的同事还个东西搞不清楚.另外再介绍一下.NET  Core的DI实现以及对实例 ...

  4. [转]React入门看这篇就够了

    摘要: 很多值得了解的细节. 原文:React入门看这篇就够了 作者:Random Fundebug经授权转载,版权归原作者所有. React 背景介绍 React 入门实例教程 React 起源于 ...

  5. vs2010如何安装mvc3,怎样安装,详细的步骤,从哪下载?请看这篇文章。

    vs2010如何安装mvc3,怎样安装,详细的步骤,从哪下载?请看这篇文章. 安装步骤:vs2010 -> vs2010sp1 -> AspNetMVC3Setup -> AspNe ...

  6. Java设计模式(十三) 别人再问你设计模式,叫他看这篇文章

    原创文章,转载请务注明出处 OOP三大基本特性 封装 封装,也就是把客观事物封装成抽象的类,并且类可以把自己的属性和方法只让可信的类操作,对不可信的进行信息隐藏. 继承 继承是指这样一种能力,它可以使 ...

  7. [ZZ]如果有人问你数据库的原理,叫他看这篇文章

    如果有人问你数据库的原理,叫他看这篇文章 http://blog.jobbole.com/100349/ 文章把知识链都给串起来,对数据库做一个概述. 合并排序 阵列.树和哈希表 B+树索引概述 数据 ...

  8. ASP.NET Core WebApi使用Swagger生成api说明文档看这篇就够了

    引言 在使用asp.net core 进行api开发完成后,书写api说明文档对于程序员来说想必是件很痛苦的事情吧,但文档又必须写,而且文档的格式如果没有具体要求的话,最终完成的文档则完全取决于开发者 ...

  9. .NET Core实战项目之CMS 第二章 入门篇-快速入门ASP.NET Core看这篇就够了

    作者:依乐祝 原文链接:https://www.cnblogs.com/yilezhu/p/9985451.html 本来这篇只是想简单介绍下ASP.NET Core MVC项目的(毕竟要照顾到很多新 ...

随机推荐

  1. Java高级架构师(一)第16节:Mybatis动态查询和Json自动拼装

    <%@ page language="java" contentType="text/html; charset=utf-8" pageEncoding= ...

  2. Asp.Net MVC part3 路由Route

    路由Route路由规则Route:可以查看源代码了解一下构造方法,需要指定路由格式.默认值.处理器三个值路由数据RouteData:当前请求上下文匹配路由规则而得到的一个对象,可以在Action中通过 ...

  3. WebLogic Server 12.1.2后的字符型安装模式

    weblogic Server 12.1.1全部都可以用原来方式. WebLogic Server 12.1.2后已经取消了console安装模式,目前只有gui和静默安装模式.并且安装方式下也有很大 ...

  4. http://jingyan.baidu.com/article/dca1fa6fa07000f1a44052f6.html

    http://jingyan.baidu.com/article/dca1fa6fa07000f1a44052f6.html

  5. git fetch 的简单用法:更新远程代码到本地仓库及冲突处理

    Git中从远程的分支获取最新的版本到本地方式如下,如何更新下载到代码到本地,请参阅ice的博客基于Github参与eoe的开源项目指南方式一1. 查看远程仓库 1 2 3 4 5 6 $ git re ...

  6. [Javascript] Use a Pure RNG with the State ADT to Select an Element from State

    The resultant in a State ADT instance can be used as a means of communication between different stat ...

  7. Photoshop - 描边

    描边在后期的UI制作的时候会比较少用,因为有一些缺陷,可以用选取收缩 1.快捷键E + S  (菜单栏-编辑-描边.photoshop cc) 2.使用图层样式进行描边(双击图层列表区域,图层缩略图的 ...

  8. 【DB2】报错:-30090 25000 指定的操作对远程执行失败

    场景描述: 数据库:DB_1,DB_2 现在在DB_1中建立NICKNAME为CST_INFO_NICK,并且该别名指向数据库DB_2的CST_INFO表,在DB_1中建立存储过程,该存储过程需要  ...

  9. 详解CSS的相对定位和绝对定位(讲得很详细)

    详解CSS的相对定位和绝对定位 CSS的相对定位和绝对定位通常情况下,我们元素的position属性的值默认为static 就是没有定位,元素出现在正常的文档流中,,这个时候你给这个元素设置的left ...

  10. Keep-Alive简介及在Tomcat中配置

      Keep-Alive功能使客户端到服务器端的连接持续有效,当出现对服务器的后继请求时,Keep-Alive功能避免了建立或者重新建立连接.市场上 的大部分Web服务器,包括iPlanet.IIS和 ...