Пользуемся AJAX-объектом

AJAX (Asynchronous JavaScript and XML) - термин, который не так давно вошел в лексикон создателей WEB-приложений. Он означает целую совокупность технологий, объединяемых в рамках одного интерфейса и позволяющих отправлять запросы без перезагрузки страницы.

В этой статье мы рассмотрим одну из составляющих AJAX-приложений, а именно JavaScript-объект, генерирующий запросы. Вникать в принцип его работы необязательно, достаточно скопировать код в отдельный файл и при необходимости использовать. Для тех кто решил разобраться в работе самого объекта оставлены некоторые комментарии в коде.

AJAX-объект


// Базовый объект
var net = new Object();
// Состояния запроса
net.READY_STATE_UNINITIALIZED = 0;
net.READY_STATE_LOADING = 1;
net.READY_STATE_LOADED = 2;
net.READY_STATE_INTERACTIVE = 3;
net.READY_STATE_COMPLETE = 4;
// Конструктор
// key - Ключ запроса в массиве requestsHash
// method - метод отправки запроса (POST, GET)
// url - URL скрипта на сервере, который обработает запрос
// params - параметры запроса (test=1&flag=2)
// onload - callback-функция, обработчик после загрузки
// onerror - callback-функция, обработчик в случае ошибки
// contentType - тип контента запроса (text/html)
// headers - прочие заголовки, кроме content-type, например, Expires
net.ContentLoader = function(key, method, url, params, onload, onerror, contentType, headers) {
this.hashKey = key;
// Свойство для рапознавания браузеров, не поддерживающих request
this.unrequestBrowser = false;
// request - объект
this.req = null;
this.onload = onload;
// Если её нет, вызывается дефолтный обработчик
this.onerror = (onerror) ? onerror : this.defaultError;
// Вызываем метод для генерации запроса
this.loadXMLDoc(method, url, params, contentType, headers);
}
// Все методы записаны в прототип
net.ContentLoader.prototype = { // Methods
// loadXMLDoc - метод, для генерации request-запроса
loadXMLDoc : function(method, url, params, contentType, headers) {
if (!
method) method="GET";
if (!
contentType && method=="POST") contentType='application/x-www-form-urlencoded';
if (
window.XMLHttpRequest) {
this.req=new XMLHttpRequest();
} else if (
window.ActiveXObject) {
this.req=new ActiveXObject("Microsoft.XMLHTTP");
} else {
this.unrequestBrowser = true;
return;
}
if (
this.req) {
try {
this.req.open(method,url,true);
if (
contentType){
this.req.setRequestHeader('Content-Type', contentType);
}
if (
headers) {
for (var
h in headers) {
this.req.setRequestHeader(h,headers[h]);
}
}
var
loader=this;
this.req.onreadystatechange=function() {
loader.onReadyState.call(loader);
}
this.req.send(params);
} catch (
err){
this.onerror.call(this);
}
}
},
// onReadyState - метод для обработки ответов сервера
onReadyState : function() {
var
req=this.req;
var
ready=req.readyState;
if (
ready==net.READY_STATE_COMPLETE) {
var
httpStatus=req.status;
if (
httpStatus==200 || httpStatus==0) {
this.onload.call(this);
} else {
this.onerror.call(this);
}
}
},
// defaultError - метод обработки ошибок по умолчанию
defaultError : function() {
alert("error fetching data!"+"nnreadyState:"+this.req.readyState +"nstatus: "+this.req.status+"nheaders: "+this.req.getAllResponseHeaders());
}
}
// Массив для хранения нескольких запросов
var requestsHash = [];
// Функция, создающая новый экземпляр объекта net.ContentLoader
// Записывает запросы в requestsHash
// Возвращает свойство unrequestBrowser
function setAjaxRequest(method, url, params, onload, onerror, contentType, headers) {
// Check of necessary parameters
if (!url) {
alert("Necessary parameters are not specified");
return;
}
requestsHash.push(new net.ContentLoader(requestsHash.length, method, url, params, onload, onerror, contentType, headers));
return
requestsHash[requestsHash.length - 1].unrequestBrowser;
}


Подготовка

Для создания так называемого асинхронного запроса и получения ответа потребуются некоторые условия:

1) Серверный скрипт, который примет запрос, обработает его и выдаст ответ. В качестве примера здесь будет использован PHP-скрипт http://fastcoder.org/demo/ajaxDemo.php, который на выходе отдает XML-дерево такого вида:

<xml>
<result>Success!</result>
<error></error>
</xml>


2) Функция, которая отправит запрос. Она уже есть (setAjaxRequest) и встроена в приведенный выше AJAX-объект, ей и воспользуемся.

3) Функции-обработчики, которые примут ответ (или возможную ошибку) и надлежащим образом отреагируют. Они будут вызваны как callback-функции. Это значит что this для них вернет ссылку на экземпляр объекта net.ContentLoader.

// Функция обработки ответа
function ansHandler() {
// Для обработки ответа можно изпользовать два основных свойства request-объекта
// responseText - вернет ответ в качестве текстовой переменной
// responseXML - вернет XML-ответ
// Последовательно воспользуемся обоими свойствами
alert(this.req.responseText);
// XML-структурой можно управлять более эффективно
// Например, получать значения отдельных узлов
var xml = this.req.responseXML.firstChild;
if (
xml.nodeType != 1) xml = xml.nextSibling; // fix for Opera
var result = xml.getElementsByTagName("result")[0].firstChild.nodeValue;
alert("result = " + result);
}
// Функция обработки ошибок
function errorHandler() {
alert("При обработке запроса произошла ошибка, повторите попытку снова");
}


Формирование и отправка запроса

function doit() {
var
method = "GET";
var
url = "http://fastcoder.org/demo/ajaxDemo.php";
var
params = false; // Без параметров
var onload = ansHandler;
var
onerror = errorHandler;
var
contentType = headers = false; // Без заголовков
return setAjaxRequest(method, url, params, onload, onerror, contentType, headers);
}


И какой-нибудь HTML-элемент, например ссылку, для отправки запроса:

<a onclick="return doit();" href="#">Отправить запрос</a>


Замечания


1) Браузер Opera ниже версии 8.0 не поддерживает request. Собственно для них и создавалось свойство unrequestBrowser, которое возвращается всеми функциями. Если вернется true, то произойдет простой переход по ссылке.
2) Скрипт можно улучшить и расширить, но необходимый минимум (и даже больше) в этой статье имеется. Если вы нашли ошибки или пришли к выводу, что чего-то существенно не хватает - пишите.




Рекомендуем почитать

 

Добавить комментарий


Ваше имя:


Комментарий:


Введите: Картинка