Ajax自Prototype框架把它普及以来,已经成为前端必须用到的组件,以致不少库把它放在核心代码里,例如jquery、prototype等。
QWrap库中,还是把它还原到组件级别。(由于QWrap有灵活的多应用Apps机制,所以,实际中也可以将Ajax包进核心库)
这里介绍的是WaGang团队基于QWrap开发出的Ajax组件。。
先通过这个Demo页现认识一下吧:Ajax-Demo.html
在项目中使用Ajax是ajax/ajax_retouched.js,它是一个复合文件,内容包括:
document.write('<script type="text/javascript" src="'+srcPath+'wagang/ajax/ajax.js"></script>');
document.write('<script type="text/javascript" src="'+srcPath+'wagang/ajax/ajax_retouch.js"></script>');
其中,/ajax.js是基本实现,/ajax_retouch.js是为在项目订做的易用性retouch。
先看一下ajax.js
View Code
按Ajax的基本实现,可以通过Ajax的构造器与它的一些方法,来实现所有的XMLHttpRequest相关的需求,/*
* @fileoverview Encapsulates common operations of Ajax
* @author JK ,绝大部分代码来自BBLib/util/BBAjax(1.0版),其作者为:chen.minliang@gmail.com。致谢
* @version 0.1
* @create-date : 2009-02-20
* @last-modified : 2009-02-20
*/
(function(){
var mix=QW.ObjectH.mix,
encodeURIJson=QW.ObjectH.encodeURIJson,
encodeURIForm=QW.NodeH.encodeURIForm,
CustEvent=QW.CustEvent;
/**
* @class Ajax Ajax类的构造函数
* @param {json} options 构造参数
*----------------------------------------------------------------------------------------
*| options属性 | 说明 | 默认值 |
*----------------------------------------------------------------------------------------
*| url: | 请求的路径 | 空字符串 |
*| method: | 请求的方法 | get |
*| async: | 是否异步请求 | true |
*| user: | 用户名 | 空字符串 |
*| pwd: | 密码 | 空字符串 |
*| requestHeaders: | 请求头属性 | {} |
*| data: | 发送的数据 | 空字符串 |
*| useLock: | 是否使用锁机制 | 0 |
*| timeout: | 设置请求超时的时间(ms) | 30000 |
*| onsucceed: | 请求成功监控 (成功指:200-300以及304) |
*| onerror: | 请求失败监控 |
*| oncancel: | 请求取消监控 |
*| oncomplete: | 请求结束监控 (success与error都算complete) |
*----------------------------------------------------------------------------------------
* @return {Ajax}
*/
function Ajax( options ){
this.options = options;
this._initialize();
};
mix(Ajax,{
/*
* 请求状态
*/
STATE_INIT: 0,
STATE_REQUEST: 1,
STATE_SUCCESS: 2,
STATE_ERROR: 3,
STATE_TIMEOUT: 4,
STATE_CANCEL: 5,
/**
* defaultHeaders: 默认的requestHeader信息
*/
defaultHeaders:{
'Content-type':'application/x-www-form-urlencoded UTF-8',//最常用配置
'com-info-1':'QW'//根具体应用有关的header信息
},
/**
* EVENTS: Ajax的CustEvents:'succeed','error','cancel','complete'
*/
EVENTS:['succeed','error','cancel','complete'],
/**
*XHRVersions: IE下XMLHttpRequest的版本
*/
XHRVersions:['MSXML2.XMLHttp.6.0', 'MSXML2.XMLHttp.3.0', 'MSXML2.XMLHttp.5.0', 'MSXML2.XMLHttp.4.0', 'Msxml2.XMLHTTP','MSXML.XMLHttp','Microsoft.XMLHTTP'],
/*
* getXHR(): 得到一个XMLHttpRequest对象
* @returns {XMLHttpRequest} : 返回一个XMLHttpRequest对象。
*/
getXHR: function () {
var versions=Ajax.XHRVersions;
if(window.ActiveXObject){
while( versions.length > 0 ){
try{
return new ActiveXObject(versions[0]);
}
catch(ex){
versions.shift();
}
}
}
else if(window.XMLHttpRequest){
return new XMLHttpRequest();
}
return null;
},
/**
* 静态request方法
* @method request
* @static
* @param {String|Form} url 请求的地址。(也可以是Json,当为Json时,则只有此参数有效,会当作Ajax的构造参数)。
* @param {String|Json|Form} data (Optional)发送的数据
* @param {Function} callback 请求完成后的回调
* @param {String} method (Optional) 请求方式,get或post
* @returns {Ajax}
* @example
QW.Ajax.get('http://demo.com',{key: 'value'},function(data){});
*/
request: function( url, data, callback, method){
if (url.constructor == Object) {
var a=new Ajax(url);
}
else {
if(typeof data == 'function'){
method = callback;
callback = data;
if(url && url.tagName == 'FORM'){
method = method || url.method;
data = url;
url = url.action;
}
else {
data='';
}
}
a = new Ajax({
url: url,
method: method,
data: data,
oncomplete:callback
});
}
a.send();
return a;
},
/**
* 静态get方法
* @method get
* @static
* @param {String|Form} url 请求的地址
* @param {String|Json|Form} data (Optional)发送的数据
* @param {Function} callback 请求完成后的回调
* @returns {Ajax}
* @example
QW.Ajax.get('http://demo.com',{key: 'value'},function(e){alert(this.requester.responseText);});
*/
get: function( url, data, callback){
var args=[].slice.call(arguments,0);
args.push('get');
return Ajax.request.apply(null,args);
},
/**
* 静态post方法
* @method post
* @static
* @param {String|Form} url 请求的地址
* @param {String|Json|Form} data (Optional)发送的数据
* @param {Function} callback 请求完成后的回调
* @returns {Ajax}
* @example
QW.Ajax.post('http://demo.com',{key: 'value'},function(e){alert(this.requester.responseText);});
*/
post: function( url, data, callback){
var args=[].slice.call(arguments,0);
args.push('post');
return Ajax.request.apply(null,args);
}
});
mix(Ajax.prototype,{
//参数
url: '',
method: 'get',
async: true,
user: '',
pwd: '',
requestHeaders: null,//是一个json对象
data: '',
/*
* 是否给请求加锁,如果加锁则必须在之前的请求完成后才能进行下一次请求。
* 默认不加锁。
*/
useLock: 0,
timeout: 30000, //超时时间
//私有变量|readOnly变量
isLocked : 0, //处于锁定状态
state : Ajax.STATE_INIT, //还未开始请求
/**
* send( url, method, data ): 发送请求
* @param {string} url 请求的url
* @param {string} method 传送方法,get/post
* @param {string|jason|FormElement} data 可以是字符串,也可以是Json对象,也可以是FormElement
* @returns {void}
*/
send : function( url, method, data ){
var me=this;
if( me.isLocked ) throw new Error('Locked.');
else if(me.isProcessing()) {me.cancel();}
var requester=me.requester;
if(!requester){
requester=me.requester = Ajax.getXHR();
if( !requester ){
throw new Error('Fail to get HTTPRequester.');
}
}
url = url || me.url;
method = (method || me.method || '').toLowerCase();
if(method != 'post') method='get';
data = data || me.data;
if( typeof data == 'object' ) {
if(data.tagName='FORM') data=encodeURIForm(data); //data是Form HTMLElement
else data = encodeURIJson(data); //data是Json数据
}
//如果是get方式请求,则传入的数据必须是'key1=value1&key2=value2'格式的。
if(data && method != 'post')
url += (url.indexOf('?') != -1?'&':'?') + data ;
if( me.user)
requester.open(method,url,me.async,me.user,me.pwd );
else
requester.open(method,url,me.async);
//设置请求头
for(var i in me.requestHeaders){
requester.setRequestHeader( i, me.requestHeaders[i] );
}
//重置
me.isLocked=0;
me.state=Ajax.STATE_INIT;
//send事件
if(me.async) {
me._sendTime=new Date();
if(me.useLock) me.isLocked=1;
this.requester.onreadystatechange = function(){
var state = me.requester.readyState;
if( state == 4 ) {
me._execComplete();
}
};
me._checkTimeout();
}
if( method == 'post' ){
if( !data ) data = ' ';
requester.send(data);
}
else{
requester.send(null);
}
if(!me.async){
me._execComplete('timeout');
}
},
/**
* isSuccess(): 判断现在的状态是否是“已请求成功”
* @returns {boolean} : 返回XMLHttpRequest是否成功请求。
*/
isSuccess: function(){
var status = this.requester.status;
return !status || (status >= 200 && status < 300) || status == 304;
},
/**
* isProcessing(): 判断现在的状态是否是“正在请求中”
* @returns {boolean} : 返回XMLHttpRequest是否正在请求。
*/
isProcessing: function(){
var state = this.requester?this.requester.readyState:0;
return state > 0 && state < 4;
},
/**
* get(url,data): 用get方式发送请求
* @param {string} url: 请求的url
* @param {string|jason|FormElement} data: 可以是字符串,也可以是Json对象,也可以是FormElement
* @returns {void} : 。
* @see : send 。
*/
get: function(url,data){
this.send(url,'get',data);
},
/**
* get(url,data): 用post方式发送请求
* @param {string} url: 请求的url
* @param {string|jason|FormElement} data: 可以是字符串,也可以是Json对象,也可以是FormElement
* @returns {void} : 。
* @see : send 。
*/
post: function(url,data){
this.send(url,'post',data);
},
/**
* cancel(): 取消请求
* @returns {boolean}: 是否有取消动作发生(因为有时请求已经发出,或请求已经成功)
*/
cancel: function(){
var me=this;
if( me.requester && me.isProcessing() ) {
me.state = Ajax.STATE_CANCEL;
me.requester.abort();
me._execComplete();
me.fire('cancel');
return true;
}
return false;
},
/**
* _initialize(): 对一个Ajax进行初始化
* @returns {void}:
*/
_initialize: function(){
var me=this;
CustEvent.createEvents(me,Ajax.EVENTS);
mix(me,me.options,1);
me.requestHeaders=mix(me.requestHeaders||{},Ajax.defaultHeaders);
},
/**
* _checkTimeout(): 监控是否请求超时
* @returns {void}:
*/
_checkTimeout: function(){
var me = this;
if(me.async){
clearTimeout(me._timer);
this._timer = setTimeout(function(){
// Check to see if the request is still happening
if (me.requester && !me.isProcessing() ) {
// Cancel the request
me.state=Ajax.STATE_TIMEOUT;
me.requester.abort();//Firefox执行该方法后会触发onreadystatechange事件,并且state=4;所以会触发oncomplete事件。而IE、Safari不会
me._execComplete('timeout');
}
}, me.timeout);
}
},
/**
* _execComplete(): 执行请求完成的操作
* @returns {void}:
*/
_execComplete: function(){
var me=this;
var requester = me.requester;
requester.onreadystatechange=new Function;//防止IE6下的内存泄漏
me.isLocked = 0;//解锁
clearTimeout( this._timer );
if( me.state==Ajax.STATE_CANCEL || me.state==Ajax.STATE_TIMEOUT ){
//do nothing. 如果是被取消掉的则不执行回调
}
else if( me.isSuccess()) {
me.state=Ajax.STATE_SUCCESS;
me.fire('succeed');
}
else {
me.state=Ajax.STATE_ERROR;
me.fire('error');
}
me.fire('complete');
}
});
QW.provide('Ajax',Ajax);
})();
但是,构造器加调用,形如“var ajax=new Ajax(opts);ajax.send();”,这种用法很不方便,所以,会在Ajax上添加几个静态方法:
Ajax.request ( url, data, callback, method) //data可省,data是form时url可省;
Ajax.get ( url, data, callback) //data可省,data是form时url可省;
Ajax.post ( url, data, callback) //data可省,data是form时url可省;
在以上文件中对应的代码是这段:
View Code
/**
* 静态request方法
* @method request
* @static
* @param {String|Form} url 请求的地址。(也可以是Json,当为Json时,则只有此参数有效,会当作Ajax的构造参数)。
* @param {String|Json|Form} data (Optional)发送的数据
* @param {Function} callback 请求完成后的回调
* @param {String} method (Optional) 请求方式,get或post
* @returns {Ajax}
* @example
QW.Ajax.get('http://demo.com',{key: 'value'},function(data){});
*/
request: function( url, data, callback, method){
if (url.constructor == Object) {
var a=new Ajax(url);
}
else {
if(typeof data == 'function'){
method = callback;
callback = data;
if(url && url.tagName == 'FORM'){
method = method || url.method;
data = url;
url = url.action;
}
else {
data='';
}
}
a = new Ajax({
url: url,
method: method,
data: data,
oncomplete:callback
});
}
a.send();
return a;
},
/**
* 静态get方法
* @method get
* @static
* @param {String|Form} url 请求的地址
* @param {String|Json|Form} data (Optional)发送的数据
* @param {Function} callback 请求完成后的回调
* @returns {Ajax}
* @example
QW.Ajax.get('http://demo.com',{key: 'value'},function(e){alert(this.requester.responseText);});
*/
get: function( url, data, callback){
var args=[].slice.call(arguments,0);
args.push('get');
return Ajax.request.apply(null,args);
},
/**
* 静态post方法
* @method post
* @static
* @param {String|Form} url 请求的地址
* @param {String|Json|Form} data (Optional)发送的数据
* @param {Function} callback 请求完成后的回调
* @returns {Ajax}
* @example
QW.Ajax.post('http://demo.com',{key: 'value'},function(e){alert(this.requester.responseText);});
*/
post: function( url, data, callback){
var args=[].slice.call(arguments,0);
args.push('post');
return Ajax.request.apply(null,args);
}
/ajax.js事实上是一个已经健全的Ajax组件了。但是,在项目中,我们可能还有一些问题需要处理,例如:
前端与后端约定了一些通用的错误码(例如登录超时、脏数据、黄反词),处理这些通用的错误码也应该有通用的处理方式。
----于是,我们可以添加一个通用处理:Ajax.prototype.opResults()
----问题来了,这个方法是Ajax组件应该提供的吗?这些其实是业务需求,不应该出现在标准的Ajax组件里。但是对于一个项目的人来说,这个也应该提供。
----这个问题,在之前介绍QWrap时已经用Retouch机制解决过很多这类需求了。那我们继续发扬Retouch精神吧。ajax.js是标准的,我们不变。但是我们再加一个/ajax_retouch.js,来为标准的ajax添加一些更适合本项目的功能。
看下这个ajax_retouch.js吧:
View Code
/*
* Copyright (c) 2011, Baidu Inc. All rights reserved.
* version: $version$ $release$ released
* author: JK
* description: ajax推荐retouch....
*/
(function (){
var Ajax = QW.Ajax,
NodeW = QW.NodeW,
g=QW.NodeH.g;
Ajax.Delay = 1000;
/*
* Youa项目中处理json格式response的推荐方法
* @method opResults
* 例如,以下这句会处通一些通用的错误:Ajax.post(oForm,function(e){var status=this.opResults();})
*/
Ajax.prototype.opResults = function(url) {
var ajax=this;
if(ajax.isSuccess()){
var responseText=ajax.requester.responseText;
try {
var status=new Function( 'return (' + responseText +');')();
} catch(ex) {
alert("系统错误,请稍后再试。");
return {isop:true,err:"inter"};
}
}
else{
alert("系统错误,请稍后再试。");
return {isop:true,err:"inter"};
}
status.isop = true;//记录是否已经作过处理
var Valid_fail=window.Valid && Valid.fail || function (el,msg,needFocus){
alert(msg);
el.focus();
}
switch(status.err){
case "mcphp.ok":
if(url != false) {
if(url){
if(url === true) {
setTimeout(function(){location.reload(true);}, Ajax.Delay);
} else {
setTimeout(function(){location.href=url;}, Ajax.Delay);
}
}
else{
if(status.data && status.data.url){
setTimeout(function(){location.href=status.data.url;}, Ajax.Delay);
}
else{
setTimeout(function(){location.reload(true);}, Ajax.Delay);
}
}
} else {
status.isop = false;
}
break;
case "gmap.internal":
alert("验证码过期,请刷新页面后重试!");
break;
case "mcphp.u_vcode":
var els = document.getElementsByName('_vcode'), elKeys = document.getElementsByName('_vcode_key');
if(els.length > 0 && elKeys.length > 0) {
var el = els[0], elKey = elKeys[0];
Valid_fail(el, "请检查您输入的验证码。", true);
if(g('_vcode_img')) {
g('_vcode_img').src = '/vcode?k=' + elKey.value + '&random='+Math.random();
}
}
break;
case "mcphp.u_input":
for(var i in status.data) {
var tempEl=document.getElementsByName(i);
try{
if(tempEl.length>0){
if (tempEl.length == 1) {
Valid_fail(tempEl[0], "您输入的内容格式不对或超长,请检查是否包含特殊字符。", true);
}
else{
for(var x=0;x<tempEl.length;x++){
if(status.fields[i]==tempEl[x].value.encode4Html()){
Valid_fail(tempEl[x],"您输入的内容格式不对或超长,请检查是否包含特殊字符。", true);
break;
};
}
}
}
}
catch(e){
alert("您输入的内容格式不对或超长,请检查是否包含特殊字符。");
}
}
break;
case "mcphp.u_notfound":
alert("您请求的页面不存在!");
break;
case "mcphp.u_antispam":
alert("您提交的内容包含敏感词汇,请检查后重新提交!");
break;
case "mcphp.u_deny":
alert("您的操作不允许!");
break;
case "mcphp.u_bfilter":
alert("您的操作太频繁!");
break;
case "login":
case "mcphp.u_login":
try {
User.Login.show();
User.Login.hint("您需要登录后才能继续刚才的操作");
} catch(e) {
try {
top.User.Login.show();
top.User.Login.hint("您需要登录后才能继续刚才的操作");
} catch(e1) {
alert('您需要登录后才能继续刚才的操作!');
}
}
break;
case "mcphp.u_power":
window.location ="http://co.youa.baidu.com/content/misc/err/nopower/index.html";
break;
case "mcphp.fatal":
case "mar.sys_inter":
alert("系统错误,请稍后再试。");
break;
default:
status.isop = false;
}
return status;
};
/*
* Youa项目中处理javascript格式response的推荐方法
* @method execJs
* 例如,以下这句会执行返回结果:Ajax.post(oForm,function(e){var status=this.execJs();})
*/
Ajax.prototype.execJs = function() {
QW.StringH.execJs(this.requester.responseText);
};
/*
* 扩展NodeW
* @method ajaxForm
* @example W('#formId').ajaxForm();
*/
var FormH = {
ajaxForm : function(oForm,opts) {
if (typeof opts == 'function') {
var o = {
data: oForm,
oncomplete: opts
};
}
else{
var o = {
data: oForm,
oncomplete: Ajax.opResults
};
}
QW.ObjectH.mix(o, opts || {},true);
new Ajax(o).require();
}
};
NodeW.pluginHelper(FormH,'operator');
})();
它除了提供刚才的opResults之外,还提供其它的一些retouch:
把responseText当作js来运行的Ajax.prototype.evalJs方法。
为NodeW作Retouch,提供一个W(oForm).ajaxForm()的方法。
更多Ajax相关的信息,参见本文件夹:
http://dev.qwrap.com/resource/js/wagang/ajax
另:QWrap网址:http://www.qwrap.com