`
cuisuqiang
  • 浏览: 3962977 次
  • 性别: Icon_minigender_1
  • 来自: 北京
博客专栏
3feb66c0-2fb6-35ff-968a-5f5ec10ada43
Java研发技术指南
浏览量:3672326
社区版块
存档分类
最新评论

一款二级菜单和使用自定义标签实现简单权限控制

    博客分类:
  • J2EE
阅读更多

因为要搞一个简单的权限系统,所以最近我进行了一些设计和实现。经过研究,根据业务需求,决定使用一个二级菜单和自定义标签来实现权限的控制。

 

首先来解决这款二级菜单,当然实现自己也肯定能实现,但是别人做好了自己就用吧。

其他技术你可以访问我的博客:http://cuisuqiang.iteye.com/

这个控件叫 chromemenu,官方网站是http://www.dynamicdrive.com/ ,当然我的附件里面已经带了一个,你可以直接下载看一下。

 

使用很简单,就是几个层和超链接,正好我可以控制层和超链接的显示来实现权限控制。

我们来定义一个标签web-html.tld:

<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE taglib PUBLIC "-//Sun Microsystems, Inc.//DTD JSP Tag Library 1.2//EN"  
	"http://java.sun.com/dtd/web-jsptaglibrary_1_2.dtd">
<taglib>
	<tlib-version>1.0</tlib-version>
	<jsp-version>1.2</jsp-version>
	<short-name>html</short-name>
	<tag>
		<name>resourceUrl</name>
		<tag-class>com.nms.taglib.ResourceUrl</tag-class>
		<body-content>JSP</body-content>
		<attribute>
			<name>key</name>
			<rtexprvalue>true</rtexprvalue>
		</attribute>
		<attribute>
			<name>href</name>
			<rtexprvalue>true</rtexprvalue>
		</attribute>
		<attribute>
			<name>rel</name>
			<rtexprvalue>true</rtexprvalue>
		</attribute>
		<attribute>
			<name>note</name>
			<rtexprvalue>true</rtexprvalue>
		</attribute>
		<attribute>
			<name>isModule</name>
			<rtexprvalue>true</rtexprvalue>
		</attribute>
	</tag>
</taglib>

 

在web.xml中配置一下:

<?xml version="1.0" encoding="UTF-8"?>
<web-app version="2.5" xmlns="http://java.sun.com/xml/ns/javaee"
	xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xsi:schemaLocation="http://java.sun.com/xml/ns/javaee 
	http://java.sun.com/xml/ns/javaee/web-app_2_5.xsd">
	<!-- 自定义标签 开始 -->
	<jsp-config>
		<taglib>
			<taglib-uri>/tld/web-html</taglib-uri>
			<taglib-location>
				/WEB-INF/tlds/web-html.tld
			</taglib-location>
		</taglib>
	</jsp-config>
	<!-- 自定义标签 结束 -->
	<welcome-file-list>
		<welcome-file>index.jsp</welcome-file>
	</welcome-file-list>
</web-app>

 

看一下实现类:

package com.nms.taglib;

import java.io.IOException;

import javax.servlet.jsp.JspException;
import javax.servlet.jsp.JspTagException;
import javax.servlet.jsp.tagext.BodyTagSupport;

/**
 * @说明 菜单生成
 * @author 崔素强
 */
@SuppressWarnings("serial")
public class ResourceUrl extends BodyTagSupport {
	@Override
	public int doStartTag() throws JspException {
		try {
			// 从开放的接口取得是否允许Key资源的输出
			boolean isCheck = true;
			if(isCheck){
				StringBuffer results = new StringBuffer("");
				if("true".equals(isModule)){
					results.append("<li>");
				}
				results.append("<a ");
				if (href != null) {
					results.append(" href=\"");
					results.append(href);
					results.append("\"");
				}
				if (rel != null) {
					results.append(" rel=\"");
					results.append(rel);
					results.append("\"");
				}
				results.append(">");
				results.append(note);
				results.append("</a>");
				if("true".equals(isModule)){
					results.append("</li>");
				}
				pageContext.getOut().write(results.toString());
			}
		} catch (IOException ex) {
			throw new JspTagException("错误");
		}
		return EVAL_BODY_INCLUDE;
	}

	@Override
	public int doEndTag() throws JspException {
		return EVAL_PAGE;
	}
	//权限验证标记
	protected String key;
	// 实际连接地址
	protected String href;
	// 对应的下拉板块
	protected String rel;
	// 是否是总标记
	protected String isModule;
	// 文字
	protected String note;
	
	public String getKey() {
		return key;
	}

	public void setKey(String key) {
		this.key = key;
	}

	public String getHref() {
		return href;
	}

	public void setHref(String href) {
		this.href = href;
	}

	public String getRel() {
		return rel;
	}

	public void setRel(String rel) {
		this.rel = rel;
	}

	public String getIsModule() {
		return isModule;
	}

	public void setIsModule(String isModule) {
		this.isModule = isModule;
	}

	public String getNote() {
		return note;
	}

	public void setNote(String note) {
		this.note = note;
	}
	
	
	/**
	 * doStartTag()方法是遇到标签开始时会呼叫的方法,其合法的返回值是EVAL_BODY_INCLUDE与SKIP_BODY,前者表示将显示标签间的文字,后者表示不显示标签间的文字
	 * doEndTag()方法是在遇到标签结束时呼叫的方法,其合法的返回值是EVAL_PAGE与SKIP_PAGE,前者表示处理完标签后继续执行以下的JSP网页,后者是表示不处理接下来的JSP网页
	 * doAfterBody(),这个方法是在显示完标签间文字之后呼叫的,其返回值有EVAL_BODY_AGAIN与SKIP_BODY,前者会再显示一次标签间的文字,后者则继续执行标签处理的下一步
	 * EVAL_BODY_INCLUDE:把Body读入存在的输出流中,doStartTag()函数可用
	 * EVAL_PAGE:继续处理页面,doEndTag()函数可用
	 * SKIP_BODY:忽略对Body的处理,doStartTag()和doAfterBody()函数可用
	 * SKIP_PAGE:忽略对余下页面的处理,doEndTag()函数可用
	 * EVAL_BODY_BUFFERED:申请缓冲区,由setBodyContent()函数得到的BodyContent对象来处理tag的body,如果类实现了BodyTag,那么doStartTag()可用,否则非法
	 * EVAL_BODY_AGAIN:请求继续处理body,返回自doAfterBody(),这个返回值在你制作循环tag的时候是很有用的
	 * 预定的处理顺序是:doStartTag()返回SKIP_BODY,doAfterBodyTag()返回SKIP_BODY,doEndTag()返回EVAL_PAGE
	 * 如果继承了TagSupport之后,如果没有改写任何的方法,标签处理的执行顺序是:doStartTag() ->不显示文字
	 * ->doEndTag()->执行接下来的网页 如果您改写了doStartTag(),则必须指定返回值,
	 * 如果指定了EVAL_BODY_INCLUDE,则执行顺序是:doStartTag()->显示文字->doAfterBodyTag()->doEndTag()->执行下面的网页
	 */
}

 

你要关注这行代码:

boolean isCheck = true;

 在使用时,你要根据 KEY 去判断是否显示某个菜单,具体实现就看你的了。

 

然后我们在JSP页面中进行使用:

<%@ page language="java" pageEncoding="UTF-8"%>
<%@ taglib uri="/tld/web-html" prefix="html"%>
<%
String path = request.getContextPath();
String basePath = request.getScheme()+"://"+request.getServerName()+":"+request.getServerPort()+path+"/";
%>
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN">
<html>
  <head>
    <base href="<%=basePath%>">    
    <title>菜单示例</title>
	<link rel="stylesheet" type="text/css" href="chromestyle.css" />
	<script type="text/javascript" src="chrome.js"></script>
  </head>  
  <body>
<div class="chromestyle" id="chromemenu">
<ul>
<html:resourceUrl href="#" key="" note="Model001" isModule="true"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="Model002" isModule="true"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="Model003" isModule="true" rel="dropmenu1"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="Model004" isModule="true" rel="dropmenu2"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="Model005" isModule="true" rel="dropmenu3"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="Model006" isModule="true" rel="dropmenu4"></html:resourceUrl>
</ul>
</div>
<!--1st drop down menu -->                                                   
<div id="dropmenu1" class="dropmenudiv">
<html:resourceUrl href="#" key="" note="a1"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="a2"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="a3"></html:resourceUrl>
</div>

<!--2nd drop down menu -->                                                
<div id="dropmenu2" class="dropmenudiv" style="width: 150px;">
<html:resourceUrl href="#" key="" note="b1"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="b2"></html:resourceUrl>
</div>
<!--3rd drop down menu -->                                                   
<div id="dropmenu3" class="dropmenudiv" style="width: 150px;">
<html:resourceUrl href="#" key="" note="c1"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="c2"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="c3"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="c4"></html:resourceUrl>
</div>
<!--4rd drop down menu -->                                                   
<div id="dropmenu4" class="dropmenudiv" style="width: 150px;">
<html:resourceUrl href="#" key="" note="d1"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="d2"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="d3"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="d4"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="d5"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="d6"></html:resourceUrl>
<html:resourceUrl href="#" key="" note="d7"></html:resourceUrl>
</div>
<script type="text/javascript">
cssdropdown.startchrome("chromemenu");
</script>
  </body>
</html>

 

如果是一级菜单,那么要设置:

isModule="true"

 

运行你看到了所有的菜单,你可以自己控制显示那个菜单。

 

这个权限就是很简单,就是根据某个Key去判断能否访问那个资源。当然在实际中应该是你请求的连接,这样再定义一个Filter去过滤所有请求,就能实现不能通过地址栏直接访问该资源。

 

请您到ITEYE看我的原创:http://cuisuqiang.iteye.com

或支持我的个人博客,地址:http://www.javacui.com

 

2
1
分享到:
评论
2 楼 yangzi010 2012-06-29  
感谢分享啊,楼主辛苦
1 楼 cuisuqiang 2012-04-18  
在标签处理时可以通过请求对象获得一些信息来判断是否有权限访问:
private String userId;

@Override
public void setPageContext(PageContext pageContext) {
	this.userId = (String) pageContext.getSession().getAttribute("CurrentUserId");
	super.setPageContext(pageContext);
}

相关推荐

    自定义菜单表

    自定义菜单表是数据库设计中的一个重要概念,尤其在企业级应用和网站开发中,它扮演着用户界面个性化和用户体验优化的角色。自定义菜单表通常用于存储用户或角色所能访问的菜单项及其排列顺序,以此来构建个性化的...

    二级树形菜单

    例如,我们可以添加一个`click`事件监听器来控制二级菜单的显示和隐藏: ```javascript document.querySelectorAll('.tree li').forEach(function(item) { item.addEventListener('click', function(e) { e....

    Web应用系统菜单及权限控制

    例如,一级菜单下可以有多个二级菜单,二级菜单下还可以有三级菜单,以此类推。 在构建完树结构后,我们利用树的先根遍历来遍历整个菜单树。先根遍历是指首先访问根节点,然后依次访问左子树和右子树。在这个场景中...

    GMS基础修改菜单和权限管理完整版

    综上所述,"GMS基础修改菜单和权限管理完整版"是一个深度定制的软件系统,旨在提供强大的用户界面定制能力和细致的权限控制,以满足不同用户和组织的安全及使用需求。通过理解和利用这个解决方案,开发者和管理员...

    Spinner二级联动下拉菜单

    对于简单的静态数据,可以使用`ArrayAdapter`,但在这个案例中,因为需要二级联动,我们需要自定义`BaseAdapter`,以便在省份选择变化时更新城市列表。 3. **事件监听**: 使用`OnItemSelectedListener`监听...

    使用layui+ajax实现简单的菜单权限管理及排序的方法

    对于一级菜单,创建`&lt;li&gt;`标签并添加`layui-nav-item`类,二级菜单则嵌套在`&lt;dl class="layui-nav-child"&gt;`和`&lt;/dl&gt;`中,最后通过`layui.element.init()`初始化element模块,实现菜单的折叠和展开效果。 在实现排序...

    三级树形+用户控件分配权限

    在三级树形结构中,数据被分为三个层次:顶级节点(一级)、次级节点(二级)和叶子节点(三级)。这种结构允许用户通过逐级展开节点来探索和操作数据,使得复杂的权限配置变得清晰和易于操作。 接下来是“用户控件...

    C#通用权限

    总结,C#通用权限系统的设计和实现涉及到多个层面,包括UI的菜单权限和业务逻辑的方法权限。通过合理地运用.NET框架提供的工具和设计模式,开发者可以构建出安全、灵活且易于维护的权限管理系统。在实际项目中,应...

    三个动态菜单.net值得学习

    在.NET中,可以使用内置的身份验证和授权机制(如ASP.NET Identity或自定义实现)来管理用户角色和权限。通过将菜单项与特定的角色或权限关联,可以确保只有具有相应权限的用户才能看到并访问相应的菜单项。例如,...

    二级路由开启远程访问的设置

    标题中的“二级路由开启远程访问的设置”是指在已经有一个主要路由器的情况下,如何配置第二个路由器以允许外部网络(外网)用户能够通过互联网访问到二级路由器的Web控制界面。这通常用于多级网络架构,例如在企业...

    ofbiz权限(全)

    系统中的权限种类繁多,包括但不限于系统预设权限、用户自定义权限、资源权限、操作权限等。这些权限与安全组之间建立多对多的关系,同样地,用户也通过多对多的方式与安全组关联。 系统预设权限通常是通过XML配置...

    C# WinForm中MenuStrip动态菜单使用总结

    本文将深入探讨如何在C# WinForm中利用MenuStrip实现动态菜单,特别是结合数据库进行权限控制的情况。 ### 一、理解MenuStrip控件 MenuStrip控件允许开发者在Windows Forms应用程序中创建传统的菜单条,其功能包括...

    discuz2.5二级导航代码

    3. **JavaScript交互**:为了让二级菜单在鼠标悬停时显示和隐藏,可以使用JavaScript或者jQuery库。通过事件监听(如`onmouseover`和`onmouseout`),你可以动态改变二级菜单的可见性。 4. **PHP编程**:在某些情况...

    repeater控件实现的模块权限管理

    例如,一级模块可能显示为导航菜单,二级模块作为下拉菜单,而更深层次的模块可能以折叠/展开的形式展示。模板中可以嵌套其他控件,如`LinkButton`或`TreeView`,来处理模块的点击事件。 4. **角色权限管理**: ...

    多级菜单的图标添加及路由配置.rar

    - **权限控制**:结合用户角色和权限进行菜单显示和路由跳转的过滤,确保不同用户看到的菜单符合其权限范围。 - **性能优化**:对于大型应用,考虑懒加载策略,只在需要时加载子路由对应的组件,减少初始加载量。 ...

    一级导航菜单.rar

    在IT行业中,导航菜单是构建用户界面(UI)的关键元素,尤其在网页和应用程序设计中。一级导航菜单通常指的是网站或应用的主要导航部分,它位于...同时,这也为实现更复杂的功能,如二级或多级下拉菜单,提供了基础。

    winform左侧菜单

    2. **自定义控件**:由于.NET Framework默认控件库中没有内置的Outlook风格的菜单控件,开发者可能需要创建自定义控件来实现这一功能。这涉及到继承自`UserControl`,并在其中添加布局控件,如`FlowLayoutPanel`或`...

    asp菜单配置

    ASP(Active Server Pages)是一种微软开发的服务器端脚本...通过以上知识点,我们可以理解ASP菜单配置的核心原理和实现方法。在实际项目中,根据需求选择合适的技术和工具,能够创建出灵活、易维护的网站导航系统。

    配合 springboot-shiro 一起使用的demo,基于ant-design-pro的简易权限系统.zip

    Ant Design Pro是蚂蚁金服开源的一款基于Ant Design的中台前端解决方案,提供了丰富的UI组件和预设的企业级模版,便于快速构建企业级的后台管理系统。 四、Ant Design Pro与权限系统集成 1. 页面路由控制:利用Ant...

Global site tag (gtag.js) - Google Analytics