惯性聚合 高效追踪和阅读你感兴趣的博客、新闻、科技资讯
阅读原文 在惯性聚合中打开

推荐订阅源

酷 壳 – CoolShell
酷 壳 – CoolShell
H
Hacker News: Front Page
P
Palo Alto Networks Blog
T
ThreatConnect
Apple Machine Learning Research
Apple Machine Learning Research
博客园_首页
T
True Tiger Recordings
P
Privacy & Cybersecurity Law Blog
B
Blog
IT之家
IT之家
Last Week in AI
Last Week in AI
F
Full Disclosure
Hacker News: Ask HN
Hacker News: Ask HN
C
Comments on: Blog
Microsoft Azure Blog
Microsoft Azure Blog
C
Cybersecurity and Infrastructure Security Agency CISA
Microsoft Security Blog
Microsoft Security Blog
博客园 - 【当耐特】
N
News and Events Feed by Topic
NISL@THU
NISL@THU
腾讯CDC
雷峰网
雷峰网
Security Latest
Security Latest
李成银的技术随笔
M
Microsoft Research Blog - Microsoft Research
L
LangChain Blog
L
Lohrmann on Cybersecurity
cs.CL updates on arXiv.org
cs.CL updates on arXiv.org
C
Check Point Blog
Y
Y Combinator Blog
Recent Announcements
Recent Announcements
博客园 - Franky
N
News | PayPal Newsroom
V
V2EX
A
About on SuperTechFans
The Register - Security
The Register - Security
月光博客
月光博客
奇客Solidot–传递最新科技情报
奇客Solidot–传递最新科技情报
Google Online Security Blog
Google Online Security Blog
MyScale Blog
MyScale Blog
Cisco Talos Blog
Cisco Talos Blog
Vercel News
Vercel News
WordPress大学
WordPress大学
C
Cyber Attacks, Cyber Crime and Cyber Security
The Hacker News
The Hacker News
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
IntelliJ IDEA : IntelliJ IDEA – the Leading IDE for Professional Development in Java and Kotlin | The JetBrains Blog
爱范儿
爱范儿
A
Arctic Wolf
L
LINUX DO - 最新话题
freeCodeCamp Programming Tutorials: Python, JavaScript, Git & More

博客园 - ruinet

WCF通用服务请求类 使用MVP模式实现B/S和C/S平台的功能通用 WCF中使用扩展行为来验证连接的用户 Microsoft.Practices.Unity依赖注入使用实例 Windows Mobile无线打印的实现 - ruinet 使用HTML,CSS快速导出数据到Excel - ruinet 软件技术网站精选 - ruinet CakePHP架构入门 - ruinet 升级Sql Server 2000到Sql Server 2005中要注意的问题 - ruinet asp.net web开发综合技能 编写第一个Silverlight程序 Saas学习 - ruinet 在Windows Mobile上控制输入法 - ruinet - 博客园 在Windows Mobile创建桌面快捷方式 在仿真设备中使用主机网络 CSS,JavaSript,Html实用小代码 重启PocketPC移动设备 使用Ajax控件引发性能问题 智能移动项目打包发布经验交流
简洁的Asp.net菜单控件
ruinet · 2009-11-10 · via 博客园 - ruinet

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