一组动态菜单的自定义标签

原创 2004年09月21日 16:47:00
一组动态菜单的自定义标签

作者:Prabu Arumugam
2003/04/09

  菜单是软件应用的重要组件。实现静态菜单相对来说比较简单――仅需一次性的自定义分类和功能即可。对于一个动态菜单,由于每个用户的资料和参数都不尽相同,实现起来既具有挑战性,又显得十分棘手。
  Java程序语言内置了创建基本菜单的架构,而JSP缺少此项支持。Web应用必须使用Java applets或javascript来实现菜单结构。许多web应用开发人员偏爱于javascript(它比applets更易开发)。本文将讲述一个可简化生成动态javascript过程的自定义标签库,包括标签库的设计和实现等内容。

设计方案
  我们的设计目标是一个可以实现等级分类的对象模型,包括产生javascript的方法。
在实现等级分类的菜单系统中,每个菜单都包括一个或多个菜单项或子菜单,子菜单又可以展开包括菜单项和子菜单。菜单提供了实现程序详细功能的入口。不管菜单是何种类型,菜单对象都应该可以产生适当的javascript。
  在基于复合物的设计模式上,simple menu在叶层次上实现了菜单,composite menu在菜单架构上实现了子菜单组和simple menu。换句话说:Composite可以包含SimpleMenu对象和其他CompositeMenu对象的列表。Menu抽象类声明了menu,定义了render方法供子类提供各自的实现。

    下图描述了此复合物的设计模式:
 
菜单类的UML图
  SimpleMenu实现了带有指向各自应用功能URL菜单项。CompositeMenu通过循环访问子菜单列表重写了render方法,分别调用各个的render方法。记住,列表可能既有菜单项,又有子菜单,所以菜单可能有多层。
  下面的列表显示了CompositeMenu和SimpleMenu的render方法。这两个方法都产生了javascript,调用dynamicmenu.js存在的方法。这个文件的方法负责在浏览器种生成菜单。这些功能的设计与实现不在本文探讨的范围之内。

  SimpleMenu的render方法
StringBuffer sb = new StringBuffer();
sb.append("addmenuitem(");
sb.append("/"" + getLevelCoord() + "/",");
sb.append("/"" + getMenuName() + "/",");
sb.append("/"" + getUrl() + "/",");
sb.append("/"black/",/"FAEBD7/",/"white/", /"3366CC/",/"white/",
    /"3366CC/",/"font-family:Tahoma, Verdana, Arial; 
    font-size:12px;font-weight:normal,text-decoration:none;padding: 4px/");");
sb.append("/n");
return sb.toString();

CompositeMenu的render方法

StringBuffer sb = new StringBuffer();
sb.append("addmenuitem(");
sb.append("/"" + getLevelCoord() + "/",");
sb.append("/"" + getMenuName() + "/",");
if (null == getUrl())
    sb.append("null" + ",");
else
    sb.append("/"" + getUrl() + "/",");
sb.append("/"black/",/"FAEBD7/",/"white/",/"3366CC/",/"white/",/"3366CC/",
        /"font-family:Tahoma, Verdana, Arial; font-size:12px;
        font-weight:normal,text-decoration:none;padding: 4px/");");

sb.append("/n");
Iterator it = list.iterator();
int i=1;
while(it.hasNext())
{
    Menu menu = (Menu)it.next();
    menu.setLevelCoord(getLevelCoord() + "," + i);
    sb.append(menu.render());
    i++;
}
return sb.toString();

菜单数据源
  菜单创建者(menu builder)负责根据数据源创建菜单。菜单数据根据种类分级。它可以来自任何可支持表单等级数据的源。我们实现了创建者,分别使用两种通用的数据源:XML和数据库。当然,你也可以很方便的根据此设计方法开发出其他的创建者。

XMLMenuBuilder
  既然菜单结构和XML结构都由等级属性,XML可方便的表达菜单数据。XMLMenuBuilder接受包含菜单定义的XML文件,解析XML文档来创建动态菜单。XML文档的<menu>元素可以看作是菜单项或子菜单,子菜单又可以展开包括菜单项或子菜单。

JDBCMenuBuilder
  既然数据库表格可以表示相关联的数据,表也可以用来表达等级数据。HIERARCHYMENU表表示等级化的菜单数据。JDBCMenuBuilder接受从数据库的信息,按照次序从HIERARCHYMENU表中选择菜单数据。

菜单标签库
菜单标签库管理创建javascript菜单的复杂过程。菜单标签自身是一个继承TagSupport类的抽象类,并重写了doStartTag和doEndTag方法。getMenu方法是一个抽象方法应该被子类重写用来提供javascript,它可以增加在doStartTag方法中创建的菜单结构的菜单项。菜单标签的子类重写了getMenu方法,使用了可从数据源提取菜单数据的菜单创建者。
其中动态菜单标签库包含两个标签:JDBCMenuTag和XMLMenuTag。JDBCMenuTag使用JDBCBuilder根据数据库中的数据创建菜单结构,而XMLMenuTag使用XMLBuilder根据XML数据源创建菜单结构。文档提供了详细的使用标签库的说明。
这个版本的标签库也包含一些局限性。它没有使用户角色和菜单发生相关联,每个应用对此都不尽相同。标签库也没有提供对菜单项特定颜色和字体的支持。
菜单标签库应用例子
Menu-example.war文件包含了使用标签库的JSP页面。在Tomcat下以一下步骤部署应用:
(1)拷贝menu-example.war到tomcat安装文件夹/webapps.
(2)重启Tomcat来展开应用。
(3)根据环境调整xmlmenu.jsp和jdbcmenu.jsp
(4)创建一个新的数据库表结构,导入menu.sql文件中的数据。
(5)访问根据XML创建的菜单,访问localhost:8080/menu-example/xmlmenu.jsp.
(6)访问根据JDBC创建大额菜单,访问localhost:8080/menu-example/jdbcmenu.jsp.

结论
动态菜单标签库管理由javascript创建菜单的复杂过程,并允许web应用通过数据库或xml源创建动态等级菜单。

[b]资源/b]
关于菜单标签库的更多信息,请访问menutag.html文档
•《设计模式》,Eric Gamma,Richard Helm,Ralph Johnson,John Vlissides(Addison-Wesley Publishing Co., 1995; ISBN: 0201633612) 
•JSP Tag Libraries :url.gifhttp://java.sun.com/products/jsp/taglibraries.html



相关下载:
• menu.zip:url.gifhttp://www.onjava.com/onjava/2003/04/09/examples/menu.zip
• menu-example.war:url.gifhttp://www.onjava.com/onjava/2003/04/09/examples/menu-example.war

Menu标签库使用文档
url.gifhttp://www.onjava.com/onjava/2003/04/09/examples/menutag.html

Prabu Arumugam 是一名软件架构师和高级JAVA开发人员,就职于Forest Express,LLC.

Spring源码之动态AOP自定义标签

相信读者用过Spring的AOP自定义标签,也就是在bean.xml的文件中添加,注解了这句话后,Spring就会支持注解AOP。那么Spring是如何去处理呢?接下来笔者就带领读者去了解Spring...
  • owen_william
  • owen_william
  • 2016年03月20日 18:12
  • 325

freemarker如何编写自定义标签

编写自定义标签需要实现 freemarker.template.TemplateDirectiveModel 接口 demo如下freemarker模板 freemarker自定义标签.ft...
  • liang_love_java
  • liang_love_java
  • 2016年09月06日 10:39
  • 5045

自定义ViewGroup实现标签换行(动态创建标签

具体实现步骤:  1.继承ViewGroup,实现三个构造方法  2.通过generateLayoutParams给自定义的控件指定参数  3.实现onMeasure方法         a.在这个方...
  • u010419467
  • u010419467
  • 2015年08月03日 14:43
  • 668

Bootstrap快速上手(一)----右键菜单

前言 具体使用过程 引用 初始化 右键菜单js文件前言    一款软件,除了稳定,功能强大,用户体验也很重要。界面风格和布局的一致性让人在视觉上感觉到这是一个统一的系统,协调的系统,至于采用什么风格及...
  • u010853701
  • u010853701
  • 2016年02月03日 15:14
  • 9929

Android 实时动态刷新更改菜单

点击Button,做如上q
  • luohaowang320
  • luohaowang320
  • 2014年08月14日 11:44
  • 5258

JSP自定义标签由浅到深详细讲解(全)

一、基本概念 1.标签(Tag):   标签是一种XML元素,通过标签可以使JSP网页变得简洁并且易于维护,还可以方便地实现同一个JSP文件支持多种语言版本。由于标签是XML元素,所以它的...
  • sinat_29581293
  • sinat_29581293
  • 2016年07月18日 10:40
  • 4099

Android动态修改ToolBar的Menu菜单

Android动态修改ToolBar的Menu菜单效果图实现 实现很简单,就是一个具有3个Action的Menu,在我们滑动到不同状态的时候,把对应的Action隐藏了。 开始上货 Menu...
  • q4878802
  • q4878802
  • 2016年04月15日 11:42
  • 22625

微信公众账号服务号自定义菜单配置与实现

微信自定义菜单 实现其实非常简单。只不过很多人不知道怎么写代码 怎么去做。 下面po上我的实现方法,以备后用。 1、拿到access_token [php] view p...
  • zxq1474477147
  • zxq1474477147
  • 2016年11月17日 15:08
  • 280

微信公众平台动态创建菜单----前端页面

菜单设置 先填写你的账号信息: AppId value="" class="px" style="width: 200px"/> AppSecret id="appSecret"...
  • qs9816
  • qs9816
  • 2014年04月25日 20:08
  • 2025

WPF 托盘菜单 自定义菜单风格 的做法

背景:我已经自定义标题栏中菜单的风格了, 现在想添加托盘菜单一样的菜单(包括风格也一样)。。  在网上找了好多资料都是用 WinForm 的代码如下         void InitTrayMenu...
  • W_SX12553
  • W_SX12553
  • 2016年05月23日 10:17
  • 1483
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:一组动态菜单的自定义标签
举报原因:
原因补充:

(最多只允许输入30个字)