`
mars李
  • 浏览: 48683 次
  • 性别: Icon_minigender_1
  • 来自: 柳州
社区版块
存档分类
最新评论

实现基于 Ajax 的无限级菜单

    博客分类:
  • AJAX
阅读更多
现在到处都有这方面的教程,我重点说一下我自己搞的一个框架。


特点:

支持Form的无闪提交(方法有点笨)

支持MVC框架,即支持传统网页架构

多线程并发请求(要语言支持线程)

动态加载文件,只加载有用的!处理了Ajax框架臃肿的JS文件问题。

采用no table的全div + css布局


a. 获得XMLHTTPRequest对象,网上到处都找得到了,不多说:


function newXMLHttpRequest() {

var xmlreq = false;

if (window.XMLHttpRequest) {

xmlreq = new XMLHttpRequest();

} else if (window.ActiveXObject) {

try {

xmlreq = new ActiveXObject("Msxml2.XMLHTTP");

} catch (e1) {

try {

xmlreq = new ActiveXObject("Microsoft.XMLHTTP");

} catch (e2) {

}

}

}

return xmlreq;

}

这里提供一个通用的支持多浏览器的方法。


b.提出异步请求




//这里用Bcandy作为方法名是为了感谢一个对我来说很重要的人,她一直在支持我

function Bcandy(Tid,url,parm,js) {

if(url == ""){

return;

}

//这是一个加载信息提示框,也可以不要!

document.getElementById("load").style.visibility = "visible";

//加载相应页面的JS文件

if(js != null){

//加载JS文件

LoadJS(js);

}

// 获取一个XMLHttpRequest实例

var req = newXMLHttpRequest();

// 设置用来从请求对象接收回调通知的句柄函数

var handlerFunction = getReadyStateHandler(req,Tid);

req.onreadystatechange = handlerFunction;

// 第三个参数表示请求是异步的

req.open("POST", url, true);

// 指示请求体包含form数据

req.setRequestHeader("Content-Type",

"application/x-www-form-urlencoded");

// 发送参数

req.send(parm);

}


function getReadyStateHandler(req,Tid) {

// 返回一个监听XMLHttpRequest实例的匿名函数

return function () {

// 如果请求的状态是“完成”

if (req.readyState == 4) {

// 成功接收了服务器响应

if (req.status == 200) {

//下面一句是重点,这里显示了返回信息的内容部分,也可以加以修改。进行其它处理

document.getElementById(Tid).innerHTML = req.responseText;

document.getElementById(Tid).style.visibility = "visible";

//这一句是实现加载信息提示框的隐藏,也可以不要。

document.getElementById("load").style.visibility = "hidden";

} else {

// 有HTTP问题发生

document.getElementById("load").style.visibility = "hidden";

alert("HTTP error: "+req.status);

}

}

}

}



//动态加载JS文件

function LoadJS(file){

var head = document.getElementsByTagName('HEAD').item(0);

var script = document.createElement('SCRIPT');

script.src = file;

script.type = "text/javascript";

head.appendChild(script);

}

这就是基本的框架了,因为使用了request.responseText;所以,可以直接请求一个页面jsp,servlet但在使用Struts框架的请求时要进行特殊处理,因为Form不支持异步请求。建议在这些页面上不要加入标签,就像.net里的asxm文件!而且在使用Struts框架时有点要注意的是,Mapping对象直接返回null就可以了,因为我们会在下面讲到并发多线程。来处理这个问题的。

总的来看,有点像是积木搭建起来的。这样方便文件的修改和扩展,互相之间并不影响,而且,实现了代码和标签分离。在进行传统页面改版时,也不用重新编写全部代码。只要修改一小部分就可以完美实现Ajax带来的无闪刷新快感。


以上代码均在IE,FireFox下测试过!

首先建立一个数据表menu


mId  菜单主键

name 菜单名称

url  菜单链接

father 低级菜单ID

sub  是否最底层菜单(用于判断是否还可以继续展开)

target 菜单链接目标(用ajax方式打开时作为显示id)

pa   菜单参数(这项用于ajax方式打开菜单)


制作一个菜单对象类



class Menu{

private int mId;

private String name;

...//其它成员


public getMid(){

return mId;

}

public setMid(int mId){

this.mId = mId;

}

....//其它成员的get set方法,

}



另一个是操作类


class MenuOpt(){

public Vector getMenus(int father){

Vector vector = new Vector();

//这里是取得父级菜单ID为father的全部菜单

//并封装进Vector的一个对象中。。

return vector;

}

}

其次就是一般的jsp文件了。但要注意以前说过的,不要包含标签!

menu.jsp:


%@page contentType="text/html; charset=GB2312"%>

%@taglib uri="http://java.sun.com/jsp/jstl/core" prefix="c"%>


jsp:useBean id="menu" scope="page" class="ycoe.basic.MenuOpt"/>

jsp:setProperty name="menu" property="father" value="${param.father}"/>

div>

c:forEach var="m" items="${menu.vector}" varStatus = "c">

c:choose>

c:when test="${m.sub eq 'Y'}">

div onClick="showMenu('${m.mid}','${m.url}','${m.target}','father=${m.mid}')">

img src="pic/menu0.gif" id="img${m.mid}" alt="" style=" cursor:hand;">

a href="#" class="text1">${m.name}

/div>

div style="display:none;" id="tr${m.mid}">

div style="padding-left:12pt" id="${m.mid}">

/div>

/c:when>

c:otherwise>

div onclick="openMenu('${m.url}','${m.target}','${m.pa}');">

img src="pic/menu1.gif" id="img${m.mid}" alt="">

a href="#" class="text1">${m.name}

/div>





menu.js:


//operMenu(打开下拉菜单的ID,打开的地址,链接打开的目标,参数)。

//这是用在menu.jsp的方法

function showMenu(id,url,target,param){

var trObj = document.getElementById("tr"+id);

var tdObj = document.getElementById(id);

//try{

if(document.getElementById("tr"+id).style.display == "none"){

//显示菜单

if(tdObj.innerHTML == null || tdObj.innerHTML == ""){

//提取数据

document.getElementById("tr"+id).style.display = "";

document.getElementById("img"+id).src = "pic/menu2.gif"

Bcandy(id,"page/menu.jsp",param,"");

openMenu(url,target,param);

}else{

//如果里面有内容,直接显示

document.getElementById("tr"+id).style.display = "";

document.getElementById("img"+id).src = "pic/menu2.gif"

openMenu(url,target,param);

}

//Bcandy(target,url,param,"");//打开菜单链接

}else{

//隐藏菜单

document.getElementById("tr"+id).style.display = "none";

document.getElementById("img"+id).src = "pic/menu0.gif"

}

//}catch(e){}

}


//打开菜单

function openMenu(url,target,param){

//这里不用我写了吧。有好几种实现方法,建议使用ajax实现!

}

最后是显示页面:




%@ page contentType="text/html; charset=GB2312" %>

meta http-equiv=Content-Type content="text/html; charset=gb2312">

style>

.text1:hover { border: 1px #999999 solid; background-color: #CCCCCC; height: 12px;}

.text1{border: 1px #FFFFFF solid; height: 12px;}



function ini(){

Bcandy("0","menu.jsp","id=0&father=0","menu.js");

}


body onload="ini();">

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;">

img src='pic/loop.gif' alt="">


数据处理中,请稍候...

br>


div id="0" align="center">





可以看到,无论在哪个层面,都和传统的没什么分别,只有jsp部分除去文件头而已(其实不去掉也行的,呵呵),而且,还可以看到,一个页面,已经分成了好几部分。就像之前说的那样,积木式的(这是网上看到一篇关于.net框架的结构时作者提出的一种结构,觉得不错,被我应用到JSP来了)。


在一些细节方面,我作了一些保留,请理解。但大致框架都是经过IE和FireFox测试。一些功能方面的扩展,自己想想了。


原理:其实就是应用了页面递归!就和一般的递归方法一下,不过用在页面上而已


div id="tr${m.id}">

循环,将从封装进vector的对象逐一显示出来

for{

 if(如果是最上层菜单sub=N){

 div id="t${m.id}" onClick="ShowMenu(${m.father....})">

  显示菜单内容

 

 

 div style="display:none" id="td${m.id....}">

 }else{

  div onClick="OpenMenu(${m.id})">显示菜单内容

 }

}


showMenu(father,id....)方法,将根据传入的father去服务器里取得数据后,再次调用这个页面。而这时,是将页面的内容显示在新的ID里面。这样,看起来就有和MSDN里的树菜单一样的效果了。


优点:多级菜单多次获取,加快了反应速度,同时应用了ajax请求,让人感觉不到页面的闪烁,亲和力强。再者,可以JS里加入了代码,让用户不用每次点击都去获取服务器数据,而是先判断有没有内容,没有再取。。。同时,实现了菜单与页面的同步,在每打开一级菜单,都可以在相应的地方打开页面。同样,这个operMenu()也可以采用ajax方式。

分享到:
评论
1 楼 NicholasBugs 2008-10-06  
要是配上效果图就完美了。

相关推荐

    实现基于 Ajax 的无限级菜单源代码

    实现基于 Ajax 的无限级菜单源代码 支持Form的无闪提交(方法有点笨) 支持MVC框架,即支持传统网页架构 多线程并发请求(要语言支持线程) 动态加载文件,只加载有用的!处理了Ajax框架臃肿的JS文件问题...

    实现基于Ajax的无限级菜单

    实现基于Ajax的无限级菜单

    基于ajax、jquery 通用无限级联菜单

    基于ajax、jquery 通用无限级联菜单; 引用场景: 国家省市级联菜单、任何二级级联菜单、三级菜单、四级菜单、五.... 使用范例:详细参阅District.Selector.js文件 以国家地理信息为例: $(function () { var ...

    ajax版本 asp无限级联动菜单程序源码.zip

    【程序老媛出品,必属精品,亲测校正,质量保证】 ...源码说明: 基于asp的ajax无限级联动菜单程序 可自定义添加联动菜单层数 包含完整代码和注释 很适合借鉴学习 适合人群:新手及有一定经验的开发人员

    基于 Ajax 的无限级菜单

    处理了Ajax框架臃肿的JS文件问题。采用no table的全div + css布局 a. 获得XMLHTTPRequest对象,网上到处都找得到了,不多说: function newXMLHttpRequest() {var xmlreq = false;if (window.XMLHttpRequest) {...

    Ajax 四级导航菜单ASP+Access动态版

    Asp+Ajax无限级联动下拉框菜单Access版 ASP仿Google输入框提示_自动完成功能 AJAX+ASP多级无限制级联菜单(地市版) ASP下结合AJAX实现输入框提示(自动完成) ASP 树形菜单TreeView 多样式版 jQuery实例_飞飞ajax带...

    ajax实例

    实现基于 Ajax 的无限级菜单,在 AJAX 开发中集成数据库技术, AJAX指南(1)---- Hello,world

    JQuery&CSS;&CSS;+DIV实例大全.rar

    实用jQuery无限级导航菜单源码下载 23.适合于网站注册的jQuery用户注册条款插件下载(带特效) 24.提升用户体验jquery Ajax表单输入检测验证示例代码 25.推荐jQuery美化Select下拉单选框模拟插件V1.3.6版本下载 ...

    asp飞飞无限级分类v1.0 Asp+sql+存储过程+ajax提供下载

    飞飞Asp乐园Asp Sql存储过程版无限级分类 开发说明: 本程序基于Asp环境开发 使用的数据库:SQLServer2000(存储过程) js框架:Jquery/ajax 数据库名称:archives

    Myth Tree无限级Tree,Json数据支持、排序拖拽

    摘要:脚本资源,Ajax/JavaScript,Tree,树形菜单,无限级树 一个基于jQuery的无限级树代码,MythTree v1.0的代码及实例,涉及到的函数讲解(JQuery方式):  $(自定义标签).mythTree(); 默认可不传值。插件会根据自己...

    史上最好传智播客就业班.net培训教程60G 不下会后悔

    大型互联网开发技术:代码生成、网页静态化、基于JQuery的Web2.0页面开发、AJAX、SEO、网站调优、采集器、RSS/XML、网站防黑(防XSS攻击、防注入漏洞攻击、防CC攻击、防挂马、防盗链、敏感词过滤、广告帖智能过滤)...

    最好的asp CMS系统科讯CMSV7.0全功能SQL商业版,KesionCMS V7.0最新商业全能版-免费下载

    科汛cms实现了“网站模板与程序完全分离”的新概念,独创ajax输出和JS标签跨站调用,支持不同频道、栏目、专题、内容页应用不同的模板,支持批量绑定模板,模板换肤更是一键之谈。 模板制作也非常方便,用户可以...

    迈思网站内容管理系统(MyStepCMS) 1.0.5.zip

     内容分类系统支持无限级栏目,栏目类型可为内部栏目或外部链接,并有独立标签支持现实任意级别下、任意级别深度的菜单。 7、多语言支持  系统原生支持多语言,只要设定相关参数,网站语种立即变化。 8、多域名...

    MyStepCMS 迈思网站内容管理系统 v1.0.5.zip

    内容分类系统支持无限级栏目,栏目类型可为内部栏目或外部链接,并有独立标签支持现实任意级别下、任意级别深度的菜单。   7、多语言支持 系统原生支持多语言,只要设定相关参数,网站语种立即变化。   8、多...

    买车网整站源码 v12.rar

    3、后台缺少基于角色的权限菜单管理; 4、缺少专题功能 本次升级,完美的解决了前两项不足,后两项列入下一版本首要解决的任务!   主要特性: 1、系统采用模块化开发,各功能模块清晰,是一款100%开源的程序...

    Too Naughty网址导航主题-PHP

    Too Naughty网址导航主题是一个基于WordPress进行开发的网址导航类主题源码。特点:纯净的导航界面(去繁求简)响应式设计的前端设计聚合式搜索框(一键切换百度,搜狗,360,必应搜索平台)用户可自行申请收录,由...

    骑士PHP人才系统 4.2.66.zip

    骑士PHP人才系统是一项基于PHP MYSQL为核心开发的一套免费 开源专业人才系统,软件具执行效率高、模板自由切换、后台管理功能方便等诸多优秀特点,凭借骑士网络的不断创新精神和认真的工作态度,相信骑士能够为您...

Global site tag (gtag.js) - Google Analytics