http://www.cnblogs.com/ruinet/archive/2009/11/10/1599984.html

asp.net自带的菜单控件采用的table和javascript,导致生成的大量的html,同时在很多浏览器中都无法显示出子菜单,也只能在IE中能显示出来。

本文介绍的菜单控件采用的css 和ul list来显示菜单,生成的html小,无需javascript支持,对大部分的浏览器都支持,除ie6要单独修改css也可以使其支持。

通过本文可以了解asp.net 控件的开发,及Composite设计模式的实际运用。

采用Composite设计模式设计菜单类:

MenuCompositeitem类


namespace Ruinet.Controls
{
  [Serializable()]
  public class MenuCompositeItem
  {
    private List<MenuCompositeItem> _children = new List<MenuCompositeItem>();
    private string _text;
    private string _link;
    private string _target;     /// <summary>
    /// 菜单项
    /// </summary>
    /// <param name="text">菜单名</param>
    /// <param name="link">链接</param>
    public MenuCompositeItem(string text, string link)
    {
      this._text = text;
      this._link = link;
    }
    /// <summary>
    /// 菜单项
    /// </summary>
    /// <param name="text">菜单名</param>
    /// <param name="link">链接</param>
    /// <param name="target">跳转目标</param>
    public MenuCompositeItem(string text, string link, string target)
    {
      this._text = text;
      this._link = link;
      this._target = target;
    }     /// <summary>
    /// 设置或获取菜单名
    /// </summary>
    public string Text
    {
      get { return _text; }
      set { _text = value; }
    }     /// <summary>
    /// 设置或获取链接
    /// </summary>
    public string Link
    {
      get { return _link; }
      set { _link = value; }
    }
    /// <summary>
    /// 跳转目标
    /// </summary>
    public string Target
    {
      get { return _target; }
      set { _target=value; }
    }     /// <summary>
    /// 设置或获取子菜单
    /// </summary>
    public List<MenuCompositeItem> Children
    {
      get { return _children; }
      set { _children = value; }
    }
  }

MenuComposite类


namespace Ruinet.Controls
{
  [DefaultProperty("Menu")]
  [ToolboxData("<{0}:MenuComposite runat=server></{0}:MenuComposite>")]
  public class MenuComposite : WebControl
  {
    /// <summary>
    /// 设置获取选择的菜单
    /// </summary>
    [Bindable(true)]
    [DefaultValue("")]
    [Localizable(true)]
    public string SelectedMenuText
    {
      get
      {
        String s = (String)ViewState["SelectedMenuText"];
        return ((s == null) ? String.Empty : s);
      }       set
      {
        ViewState["SelectedMenuText"] = value;
      }
    }     /// <summary>
    /// 获取和设置菜单项从ViewState
    /// </summary>
    [Bindable(true)]
    [DefaultValue(null)]
    [Localizable(true)]
    public MenuCompositeItem MenuItems
    {
      get
      {
        return ViewState["MenuItems"] as MenuCompositeItem;
      }       set
      {
        ViewState["MenuItems"] = value;
      }
    }     /// <summary>
    /// 呈现菜单结构
    /// </summary>
    /// <param name="output">HTML输出流</param>
    protected override void RenderContents(HtmlTextWriter output)
    {
      MenuCompositeItem root = this.MenuItems;       output.Write(@"<div class=""navmenu"">");
      output.Write(@"    <ul>");       for (int i = 0; i < root.Children.Count; i++)
      {
        RecursiveRender(output, root.Children[i]);
      }
      output.Write(@"    </ul>");
      output.Write(@"</div>");
    }     /// <summary>
    /// 递归输出菜单项
    /// </summary>
    /// <param name="output">HTML输出流</param>
    /// <param name="item">菜单项.</param>
    /// <param name="depth">Indentation depth.</param>
    private void RecursiveRender(HtmlTextWriter output, MenuCompositeItem item)
    {
      output.Write("<li>");
      if (string.IsNullOrEmpty(item.Target))//为空不设置跳转目标
      {
        output.Write(@"<a href=""" + item.Link + @""">");
      }
      else
      {
        output.Write(@"<a href=""" + item.Link + @""" target= """ + item.Target + @""">");
      }
      if (item.Text == SelectedMenuText)  //选中的菜单
      {
        output.Write(@"<span class=""selected"">");
        output.WriteLine(item.Text);
        output.WriteLine("</span>");
      }
      else
      {
        output.Write(item.Text);
      }       output.Write("</a>");       if (item.Children.Count > 0)
      {
        output.WriteLine();
        output.Write("<ul>");
        for (int i = 0; i < item.Children.Count; i++)
        {
          RecursiveRender(output, item.Children[i]);
        }
        output.Write("</ul>");
      }
      output.Write("</li>");
    }
  }
}

在页面中使用

添加对控件的引用后就可以直接在“工具箱”-》Controls组件中 看到MenuComposite组件

再就可以像其他asp.net 控件一样使用

使用:


 MenuCompositeItem root = new MenuCompositeItem("root", null);
    MenuCompositeItem menu01 = new MenuCompositeItem("menu01", ResolveUrl("~/Default.aspx"));
    MenuCompositeItem menu02 = new MenuCompositeItem("menu02", ResolveUrl("~/Default.aspx"));
    MenuCompositeItem menu03 = new MenuCompositeItem("menu03", ResolveUrl("~/Default.aspx"));
    MenuCompositeItem menu04 = new MenuCompositeItem("menu04", ResolveUrl("~/Page04.aspx"));
    MenuCompositeItem menu05 = new MenuCompositeItem("menu05", ResolveUrl("~/Default.aspx"));     MenuCompositeItem menu01_01 = new MenuCompositeItem("menu01-01", ResolveUrl("~/Default.aspx"));
    MenuCompositeItem menu01_02 = new MenuCompositeItem("menu01-02", ResolveUrl("~/Page01-02.aspx"));
    MenuCompositeItem menu01_03 = new MenuCompositeItem("menu01-03", ResolveUrl("~/Default.aspx"));
    MenuCompositeItem menu01_04 = new MenuCompositeItem("menu01-04", ResolveUrl("~/Default.aspx"));
    menu01.Children.Add(menu01_01);
    menu01.Children.Add(menu01_02);
    menu01.Children.Add(menu01_03);
    menu01.Children.Add(menu01_04);     MenuCompositeItem menu02_01 = new MenuCompositeItem("menu02-01", ResolveUrl("~/Default.aspx"));
    MenuCompositeItem menu02_02 = new MenuCompositeItem("menu02-02", ResolveUrl("~/Default.aspx"), "menu02-02");
    menu02.Children.Add(menu02_01);
    menu02.Children.Add(menu02_02);     MenuCompositeItem menu04_01 = new MenuCompositeItem("menu04-01", ResolveUrl("~/Default.aspx"));
    MenuCompositeItem menu04_02 = new MenuCompositeItem("menu04-02", ResolveUrl("~/Page04-02.aspx"), "_blank");
    menu04.Children.Add(menu04_01);
    menu04.Children.Add(menu04_02);     root.Children.Add(menu01);
    root.Children.Add(menu02);
    root.Children.Add(menu03);
    root.Children.Add(menu04);
    root.Children.Add(menu05);     TheMenuComposite.MenuItems = root;

此时生成的编译运行后会生成一个没有样式Ul list ,效果如下:

因此要生成可显示和隐藏的菜单项,关键在css的设置上,开始时将二级子菜单设置为隐藏visibility: hidden;

同时定义li的hover事件,li:hover时:自菜单的 visibility要改为visible; 大致原理是这样,当然还有注意菜单项的位置

一级菜单float:left;使其能水平显示。

CSS定义如下:


.navmenu *
{
    margin: 0;
    padding: 0;
}
.navmenu
{
    border: #000 1px solid;
    height: 25px;
}
.navmenu li
{
    /*水平菜单*/
    float: left;
    list-style: none;
    position: relative;
}
.navmenu a
{
    display: block;
    font-size: 12px;
    height: 24px;
    width: 100px;
    line-height: 24px;
    background-color: #CDEB8B;
    color: #0000ff;
    text-decoration: none;
    text-align: center;
    border-left: #36393D 1px inset;
    border-right: #36393D 1px inset;
    border-bottom: #36393D 1px inset;
}
/*单独设置一级菜单样式*/
.navmenu > ul > li > a
{
    font-size: 11px;
    font-weight: bold;
}
.navmenu a:hover
{
    background: #369;
    color: #fff;
}
/*新增的二级菜单部分*/
.navmenu ul ul
{
    visibility: hidden; /*开始时是隐藏的*/
    position: absolute;
    left: 0px;
    top: 24px;
}
.navmenu ul li:hover ul, .navmenu ul a:hover ul
{
    visibility: visible;
}
.navmenu ul ul li
{
    clear: both; /*垂直显示*/
    text-align: left;
}
/*选中菜单项*/
.navmenu .selected
{
    padding-left:15px;
    background-position-x:0px;
    background-image: url(./res/selected.gif);
    background-repeat: no-repeat;
    text-decoration:underline;
}

定义CSS后的效果如下:

到此菜单控件已完成。已测试过可以在IE7,IE8,Chrome,Firefox中正常显示,在IE6显示可能会有问题,可以参考纯CSS多级菜单进行修改,

本文的CSS显示部分参考了此文的介绍。

附上完整代码,如需要可自行下载修改:/Files/ruinet/WebMenu.zip

简洁的Asp.net菜单控件的更多相关文章

  1. Asp.net 菜单控件

    本文介绍的菜单控件采用的css 和ul list来显示菜单,生成的html小,无需javascript支持,对大部分的浏览器都支持,除ie6要单独修改css也可以使其支持. 通过本文可以了解asp.n ...

  2. ASP.NET的面包屑导航控件、树形导航控件、菜单控件

    原文:http://blog.csdn.net/pan_junbiao/article/details/8579293 ASP.NET的面包屑导航控件.树形导航控件.菜单控件. 1. 面包屑导航控件— ...

  3. ASP.NET控件<ASP:Button /> html控件<input type="button">区别联系

    ASP.NET控件<ASP:Button />-------html控件<input type="button">杨中科是这么说的:asp和input是一样 ...

  4. asp.net Login控件基本属性及事件说明

    原文:asp.net Login控件基本属性及事件说明 Login系列控件是微软为了简化我们的开发过程,为我们进行常规的安全开发提供块捷途径. Login系列控件包含下列控件: Login 登录控件 ...

  5. asp.net分页控件

    一.说明 AspNetPager.dll这个分页控件主要用于asp.net webform网站,现将整理代码如下 二.代码 1.首先在测试页面Default.aspx页面添加引用 <%@ Reg ...

  6. asp.net ajax控件tab扩展,极品啊,秒杀其它插件

    说明:asp.net ajax控件tab要设置width和height,而且在线文本编辑器放能够放入tab中,也必须是asp.net的控件型在线文本,例如fckeditor,下面是我设置好的配置. & ...

  7. javascript获取asp.net服务器端控件的值

    代码如下: <%@ Page Language="C#" CodeFile="A.aspx.cs" Inherits="OrderManage_ ...

  8. [ASP.NET]asp.net Repeater控件的使用方法

    asp.net Repeater控件的使用方法 -- : 4770人阅读 评论() 收藏 举报 asp.netserveraspdatasetdeletexhtml 今天学习了,Repeater控件 ...

  9. 关于ASP.net TextBox控件的失去焦点后触发其它事件

    编写人:CC阿爸 2015-2-02 今天在这里,我想与大家一起分享如何处理的ASP.net TextBox控件的失去焦点后触发其它事件的问题,在此做个小结,以供参考.有兴趣的同学,可以一同探讨与学习 ...

随机推荐

  1. 使用 XSLT 显示 XML

    通过使用 XSLT,您可以向 XML 文档添加显示信息. 使用 XSLT 显示 XML XSLT 是首选的 XML 样式表语言. XSLT (eXtensible Stylesheet Languag ...

  2. SQL Server 2012 sa 用户登录 18456 错误 (转)

    转自:http://blog.csdn.net/waterxcfg304/article/details/40617475 最近想研究下SQL SERVER2012 Enterprise版本的数据库, ...

  3. Vim 命令、操作、快捷键(收藏大全)

    ------ 命令历史 以:和/开头的命令都有历史纪录,可以首先键入:或/然后按上下箭头来选择某个历史命令. 启动vim 在命令行窗口中输入以下命令即可 vim 直接启动vim vim filenam ...

  4. wannalfy 挑战赛7 F Masha与老鼠(贪心+dp)

    链接:https://www.nowcoder.net/acm/contest/56/F   时间限制:C/C++ 1秒,其他语言2秒 空间限制:C/C++ 262144K,其他语言524288K 6 ...

  5. keystonejs富文本问题及思考过程

    上一篇讲了keystonejs的环境搭建,helloworld跑起来之后,实际运用中会发现各种问题,今天就说下富文本编辑器的问题(针对后端不熟的同学). 不太熟悉网页嵌入富文本编辑器的同学可能和我一样 ...

  6. jsp四种属性范围

    在JSP提供了四种属性的保存范围.所谓的属性保存范围,指的就是一个设置的对象,可以在多个页面中保存并可以继续使用.它们分别是:page.request.session.appliction. 1.pa ...

  7. Promethus

    https://blog.csdn.net/zl1zl2zl3/article/details/74332437

  8. JS 替换所有的空格

    在JS中替换掉输入框内的空格,是在处理表单需求的时候极为常用的一项操作,以防止用户的操作习惯引起数据异常,保证传参的安全性. NO.1 name.replace(" "," ...

  9. 洛谷P5022&P5049 旅行(及其数据加强版)

    旅行(不是加强版) 加强版 加强版数据范围: 我们注意到 也就是说要么是个树,要么是个基环树 60pts 这60分是个树,可以简单的贪心想到每次都走子树中编号最小的那个,并且把1作为根 dfs练手题 ...

  10. leetcode 374. 猜数字大小(python)

    我们正在玩一个猜数字游戏. 游戏规则如下:我从 1 到 n 选择一个数字. 你需要猜我选择了哪个数字.每次你猜错了,我会告诉你这个数字是大了还是小了.你调用一个预先定义好的接口 guess(int n ...