[精品]基于 Ajax 的无限级菜单
大家先到<a href="http://www.start.com/" target="_blank" >http://www.start.com/</A>看一下效果<br><br><a href="http://www.start.com/2/" target="_blank" >http://www.start.com/2/</A><br><br><a href="http://www.start.com/3/" target="_blank" >http://www.start.com/3/</A>[align=right][color=#000066][此贴子已经被作者于2006-4-17 14:55:13编辑过][/color][/align]
<STRONG>基于 Ajax 的无限级菜单<BR></STRONG>作者:<a href="http://www.blueidea.com/common/contact.asp?type=作者&username=ycoe12" target="_blank" >ycoe12</A> 时间: 2006-01-12 文档类型:原创 来自:<a href="http://www.blueidea.com/" target="_blank" >蓝色理想</A><BR>
<P>现在到处都有这方面的教程,我重点说一下我自己搞的一个框架。</P>
<P><STRONG>特点:<BR></STRONG>支持Form的无闪提交(方法有点笨)<BR>支持MVC框架,即支持传统网页架构<BR>多线程并发请求(要语言支持线程)<BR>动态加载文件,只加载有用的!处理了Ajax框架臃肿的JS文件问题。<BR>采用no table的全div + css布局</P>
<P><STRONG>a. </STRONG>获得XMLHTTPRequest对象,网上到处都找得到了,不多说:</P>
<P>function newXMLHttpRequest() {<BR>var xmlreq = false;<BR>if (window.XMLHttpRequest) {<BR>xmlreq = new XMLHttpRequest();<BR>} else if (window.ActiveXObject) {<BR>try {<BR>xmlreq = new ActiveXObject("Msxml2.XMLHTTP");<BR>} catch (e1) {<BR>try {<BR>xmlreq = new ActiveXObject("Microsoft.XMLHTTP");<BR>} catch (e2) {<BR>}<BR>}<BR>}<BR>return xmlreq;<BR>}</P>
<P>这里提供一个通用的支持多浏览器的方法。</P>
<P><STRONG>b.</STRONG>提出异步请求</P>
<P>
<P></P>
<P>//这里用Bcandy作为方法名是为了感谢一个对我来说很重要的人,她一直在支持我<BR>function Bcandy(Tid,url,parm,js) {<BR>if(url == ""){<BR>return;<BR>}<BR>//这是一个加载信息提示框,也可以不要!<BR>document.getElementById("load").style.visibility = "visible";<BR>//加载相应页面的JS文件<BR>if(js != null){<BR>//加载JS文件<BR>LoadJS(js);<BR>}<BR>// 获取一个XMLHttpRequest实例<BR>var req = newXMLHttpRequest();<BR>// 设置用来从请求对象接收回调通知的句柄函数<BR>var handlerFunction = getReadyStateHandler(req,Tid);<BR>req.onreadystatechange = handlerFunction;<BR>// 第三个参数表示请求是异步的<BR>req.open("POST", url, true);<BR>// 指示请求体包含form数据<BR>req.setRequestHeader("Content-Type",<BR>"application/x-www-form-urlencoded");<BR>// 发送参数<BR>req.send(parm);<BR>}</P>
<P>function getReadyStateHandler(req,Tid) {<BR>// 返回一个监听XMLHttpRequest实例的匿名函数<BR>return function () {<BR>// 如果请求的状态是“完成”<BR>if (req.readyState == 4) {<BR>// 成功接收了服务器响应<BR>if (req.status == 200) {<BR>//下面一句是重点,这里显示了返回信息的内容部分,也可以加以修改。进行其它处理<BR>document.getElementById(Tid).innerHTML = req.responseText;<BR>document.getElementById(Tid).style.visibility = "visible";<BR>//这一句是实现加载信息提示框的隐藏,也可以不要。<BR>document.getElementById("load").style.visibility = "hidden";<BR>} else {<BR>// 有HTTP问题发生<BR>document.getElementById("load").style.visibility = "hidden";<BR>alert("HTTP error: "+req.status);<BR>}<BR>}<BR>}<BR>}</P>
<p>
<P>//动态加载JS文件<BR>function LoadJS(file){<BR>var head = document.getElementsByTagName('HEAD').item(0);<BR>var script = document.createElement('SCRIPT');<BR>script.src = file;<BR>script.type = "text/javascript";<BR>head.appendChild(script);<BR>}</P>
<P>这就是基本的框架了,因为使用了request.responseText;所以,可以直接请求一个页面jsp,servlet但在使用Struts框架的请求时要进行特殊处理,因为Form不支持异步请求。建议在这些页面上不要加入<html><body>标签,就像.net里的asxm文件!而且在使用Struts框架时有点要注意的是,Mapping对象直接返回null就可以了,因为我们会在下面讲到并发多线程。来处理这个问题的。<BR>总的来看,有点像是积木搭建起来的。这样方便文件的修改和扩展,互相之间并不影响,而且,实现了代码和标签分离。在进行传统页面改版时,也不用重新编写全部代码。只要修改一小部分就可以完美实现Ajax带来的无闪刷新快感。</P>
<P>以上代码均在<STRONG>IE,FireFox</STRONG>下测试过!</P> <P><STRONG>首先建立一个数据表menu</STRONG></P>
<P>mId 菜单主键<BR>name 菜单名称<BR>url 菜单链接<BR>father 低级菜单ID<BR>sub 是否最底层菜单(用于判断是否还可以继续展开)<BR>target 菜单链接目标(用ajax方式打开时作为显示id)<BR>pa 菜单参数(这项用于ajax方式打开菜单)<BR><BR>制作一个<STRONG>菜单对象类</STRONG></P>
<P>
<P>class Menu{<BR>private int mId;<BR>private String name;<BR>...//其它成员</P>
<P>public getMid(){<BR>return mId;<BR>}<BR>public setMid(int mId){<BR>this.mId = mId;<BR>}<BR>....//其它成员的get set方法,<BR>}</P>
<P></P>
<p>
<P>另一个<STRONG>是操作类</STRONG></P>
<P>class MenuOpt(){<BR>public Vector getMenus(int father){<BR>Vector vector = new Vector();<BR>//这里是取得父级菜单ID为father的全部菜单<BR>//并封装进Vector的一个对象中。。<BR>return vector;<BR>}<BR>}</P>
<P>其次就是一般的jsp文件了。但要注意以前说过的,不要包含<html><body>标签!<BR><STRONG>menu.jsp:</STRONG></P>
<P><%@page contentType="text/html; charset=GB2312"%><BR><%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%><BR><!--前面用到了JSTL的标签定义(学JSP的强烈推荐!)--><BR><jsp:useBean id="menu" scope="page" class="ycoe.basic.MenuOpt"/><BR><jsp:setProperty name="menu" property="father" value="${param.father}"/><BR><div><BR><c:forEach var="m" items="${menu.vector}" varStatus = "c"><BR><c:choose><BR><c:when test="${m.sub eq 'Y'}"><BR><div onClick="showMenu('${m.mid}','${m.url}','${m.target}','father=${m.mid}')"><BR><img src="pic/menu0.gif" id="img${m.mid}" alt="" style=" cursor:hand;"><BR><a href="#" class="text1">${m.name}</a><BR></div><BR><div style="display:none;" id="tr${m.mid}"><BR><div style="padding-left:12pt" id="${m.mid}"></div><BR></div><BR></c:when><BR><c:otherwise><BR><div onclick="openMenu('${m.url}','${m.target}','${m.pa}');"><BR><img src="pic/menu1.gif" id="img${m.mid}" alt=""><BR><a href="#" class="text1">${m.name}</a><BR></div><BR></c:otherwise><BR></c:choose><BR></c:forEach><BR></div></P>
<P><STRONG>menu.js:</STRONG></P>
<P>//operMenu(打开下拉菜单的ID,打开的地址,链接打开的目标,参数)。<BR>//这是用在menu.jsp的方法<BR>function showMenu(id,url,target,param){<BR>var trObj = document.getElementById("tr"+id);<BR>var tdObj = document.getElementById(id);<BR>//try{<BR>if(document.getElementById("tr"+id).style.display == "none"){<BR>//显示菜单<BR>if(tdObj.innerHTML == null || tdObj.innerHTML == ""){<BR>//提取数据<BR>document.getElementById("tr"+id).style.display = "";<BR>document.getElementById("img"+id).src = "pic/menu2.gif"<BR>Bcandy(id,"page/menu.jsp",param,"");<BR>openMenu(url,target,param);<BR>}else{<BR>//如果里面有内容,直接显示<BR>document.getElementById("tr"+id).style.display = "";<BR>document.getElementById("img"+id).src = "pic/menu2.gif"<BR>openMenu(url,target,param);<BR>}<BR>//Bcandy(target,url,param,"");//打开菜单链接<BR>}else{<BR>//隐藏菜单<BR>document.getElementById("tr"+id).style.display = "none";<BR>document.getElementById("img"+id).src = "pic/menu0.gif"<BR>}<BR>//}catch(e){}<BR>}<BR><BR>//打开菜单<BR>function openMenu(url,target,param){<BR>//这里不用我写了吧。有好几种实现方法,建议使用ajax实现!<BR>}</P> <P>最后是<STRONG>显示页面</STRONG>:</P>
<P>
<P></P>
<P><%@ page contentType="text/html; charset=GB2312" %><BR><meta http-equiv=Content-Type content="text/html; charset=gb2312"><BR><style><BR>.text1:hover { border: 1px #999999 solid; background-color: #CCCCCC; height: 12px;}<BR>.text1{border: 1px #FFFFFF solid; height: 12px;}<BR></style><BR><script type="" src="js/Function.js"></script><BR>function ini(){<BR>Bcandy("0","menu.jsp","id=0&father=0","menu.js");<BR>}<BR></script><BR><body onload="ini();"><BR><div id="load" style="z-index:1; color:#FF0000; visibility:hidden; filter: Alpha(opacity=85); background-color:#FFFFFF; left: 48%; top: 48%;BORDER-RIGHT: #000000 1px solid; PADDING-RIGHT: 12px; BORDER-TOP: #000000 1px solid; PADDING-LEFT: 12px;PADDING-BOTTOM: 12px; BORDER-LEFT: #000000 1px solid; LINE-HEIGHT: 22px; PADDING-TOP: 12px; BORDER-BOTTOM: #000000 1px solid; POSITION: absolute;"><BR><img src='pic/loop.gif' alt=""><br><BR>数据处理中,请稍候...<BR><br><BR></div><BR><div id="0" align="center"><BR></div><BR></body><BR></html></P>
<p>
<P>可以看到,无论在哪个层面,都和传统的没什么分别,只有jsp部分除去文件头而已(其实不去掉也行的,呵呵),而且,还可以看到,一个页面,已经分成了好几部分。就像之前说的那样,积木式的(这是网上看到一篇关于.net框架的结构时作者提出的一种结构,觉得不错,被我应用到JSP来了)。</P>
<P>在一些细节方面,我作了一些保留,请理解。但大致框架都是经过IE和FireFox测试。一些功能方面的扩展,自己想想了。</P>
<P><STRONG>原理:</STRONG>其实就是应用了页面递归!就和一般的递归方法一下,不过用在页面上而已</P>
<P><div id="tr${m.id}"><BR>循环,将从封装进vector的对象逐一显示出来<BR>for{<BR> if(如果是最上层菜单sub=N){<BR> <div id="t${m.id}" onClick="ShowMenu(${m.father....})"><BR> 显示菜单内容<BR> </div><BR> <!--这里不显示内容,仅作为下一次的容器--><BR> <div style="display:none" id="td${m.id....}"></div><BR> }else{<BR> <div onClick="OpenMenu(${m.id})">显示菜单内容</div><BR> }<BR>}<BR></div></P>
<P>showMenu(father,id....)方法,将根据传入的father去服务器里取得数据后,再次调用这个页面。而这时,是将页面的内容显示在新的ID里面。这样,看起来就有和MSDN里的树菜单一样的效果了。</P>
<P>优点:多级菜单多次获取,加快了反应速度,同时应用了ajax请求,让人感觉不到页面的闪烁,亲和力强。再者,可以JS里加入了代码,让用户不用每次点击都去获取服务器数据,而是先判断有没有内容,没有再取。。。同时,实现了菜单与页面的同步,在每打开一级菜单,都可以在相应的地方打开页面。同样,这个operMenu()也可以采用ajax方式。<BR></P>
<P><STRONG>效果</STRONG>可以上 <a href="http://www.start.com/" target="_blank" >http://www.start.com/</A> 看看:</P> 给予支持!
页:
[1]
