现在到处都有这方面的教程,我重点说一下我自己搞的一个框架。
特点:
支持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方式。
分享到:
相关推荐
### 实现基于Ajax的无限级菜单源代码解析与关键技术点 #### 一、引言 在Web开发领域,Ajax技术的应用极大地提升了用户体验,使得页面无需整体刷新即可完成数据交互。本文将详细介绍一种基于Ajax实现的无限级菜单...
总结起来,实现基于Ajax的无限级菜单主要涉及以下几个步骤: 1. 创建XMLHttpRequest对象。 2. 发起Ajax请求,包括设置请求类型、URL、数据及异步标志。 3. 定义回调函数处理响应,根据服务器返回更新页面内容。 4. ...
**AJAX无限级联动菜单**是一种常见的Web交互技术,它基于ASP.NET框架,结合AJAX(Asynchronous JavaScript and XML)和ACCESS数据库实现。这个技术的主要目的是提供用户友好的界面,通过异步方式加载数据,无需刷新...
本项目提供了一个完整的解决方案,基于Ajax技术实现了一个无限级的树形菜单,并且与数据库相结合,实现了动态的数据加载和交互。 Ajax(Asynchronous JavaScript and XML)是一种在不刷新整个页面的情况下,通过...
基于ajax、jquery 通用无限级联菜单; 引用场景: 国家省市级联菜单、任何二级级联菜单、三级菜单、四级菜单、五.... 使用范例:详细参阅District.Selector.js文件 以国家地理信息为例: $(function () { var ...
Ajax无限级树是一种基于Web的用户界面技术,用于在不刷新整个页面的情况下动态加载和显示层级数据。这种技术的核心是利用Ajax(异步JavaScript和XML)进行后台数据交互,配合前端JavaScript库或框架来实现树形结构的...
【程序老媛出品,必属精品,亲测校正,质量保证】 ...源码说明: 基于asp的ajax无限级联动菜单程序 可自定义添加联动菜单层数 包含完整代码和注释 很适合借鉴学习 适合人群:新手及有一定经验的开发人员
处理了Ajax框架臃肿的JS文件问题。采用no table的全div + css布局 a. 获得XMLHTTPRequest对象,网上到处都找得到了,不多说: function newXMLHttpRequest() {var xmlreq = false;if (window.XMLHttpRequest) {...
无限级菜单的展开与折叠效果可以通过CSS的`display`属性控制,通过JavaScript动态改变`display`值来实现展开和收起。 3. JavaScript交互: layui的核心库`layui.js`包含了丰富的组件和API,其中的`layui.nav`就是...
Asp+Ajax无限级联动下拉框菜单Access版 ASP仿Google输入框提示_自动完成功能 AJAX+ASP多级无限制级联菜单(地市版) ASP下结合AJAX实现输入框提示(自动完成) ASP 树形菜单TreeView 多样式版 jQuery实例_飞飞ajax带...
在本案例中,我们讨论的是一段实现这种功能的源码,它基于ASP.NET和AJAX技术栈。以下是关于这个主题的详细知识点: 1. **无限级级联菜单**:这是一种菜单系统,能够展示多层级的数据,且层级数量不受限制。通常在...
【艾恩JS无限级菜单树】是一个基于JavaScript的ASP应用程序,专为构建具有数据库支持的无限层级菜单设计。这个解决方案提供了完整的菜单管理功能,允许用户动态地创建、编辑和删除菜单项,使得网站的导航结构更加...
以下是对这个菜单实现的关键知识点的详细说明: 1. **无限级菜单**:无限级意味着菜单可以包含任意数量的子菜单,每级菜单都可以有其子菜单,形成深度无限的树状结构。这种设计适应了内容分类复杂的情况,如大型...
此外,对于更复杂的无限级菜单,可以使用递归函数来处理每一级子菜单的展开和收起,确保无论菜单层级多深都能正常工作。 在提供的压缩包中,`index.html`文件包含了HTML结构和内联CSS,展示了如何构建一个多级导航...
总之,这个压缩包提供了一个基于jQuery的无限级树形菜单的实现,通过解析HTML结构、应用CSS样式并利用jQuery的事件处理和DOM操作,实现了优雅的展开和折叠效果。对于学习和理解jQuery以及前端交互设计的开发者来说,...
在ASP.NET中,实现无限级树型菜单通常涉及以下几个步骤: 1. **数据库设计**:Sql2000数据库用于存储菜单信息,通常包括ID(主键)、父ID(标识父级菜单)、菜单名、链接地址等字段。这里使用DBHelperSql工具类进行...
最后,`实现一个基于Ajax的调查程序`展示了Ajax如何用于实时提交表单数据,无需页面刷新,提高用户参与度。这可以应用于各种互动功能,如投票、评论或搜索建议。 总结来说,这个Ajax实例涵盖了从基础的Ajax请求到...
综合以上,"坏孩子无限级树型菜单.rar"提供了一个实例,展示了如何在Asp.net环境中构建动态、高效的无限级菜单系统,同时利用JavaScript和数据库技术实现数据的实时交互。对于想要学习或使用类似功能的开发者来说,...