Эта глава посвящена IFRAME
– самому древнему и кросс-браузерному способу AJAX-запросов.
Сейчас он используется, разве что, для поддержки кросс-доменных запросов в IE7- и, что чуть более актуально, для реализации COMET в IE9-.
Для общения с сервером создаётся невидимый IFRAME
. В него отправляются данные, и в него же сервер пишет ответ.
Введение
Сначала – немного вспомогательных функций и особенности работы с IFRAME
.
Двуличность IFRAME: окно+документ
Что такое IFRAME? На этот вопрос у браузера два ответа
-
IFRAME – это HTML-тег:
<iframe>
со стандартным набором свойств.- Тег можно создавать в JavaScript
- У тега есть стили, можно менять.
- К тегу можно обратиться через
document.getElementById
и другие методы.
-
IFRAME – это окно браузера, вложенное в основное
- IFRAME – такое же по функциональности окно браузера, как и основное, с адресом и т.п.
- Если документ в
IFRAME
и внешнее окно находятся на разных доменах, то прямой вызов методов друг друга невозможен.
- Если документ в
- Ссылку на это окно можно получить через
window.frames['имя фрейма']
.
- IFRAME – такое же по функциональности окно браузера, как и основное, с адресом и т.п.
Для достижения цели мы будем работать как с тегом, так и с окном. Они, конечно же, взаимосвязаны.
В теге <iframe>
свойство contentWindow
хранит ссылку на окно.
Окна также содержатся в коллекции window.frames
.
Например:
// Окно из ифрейма
var iframeWin = iframe.contentWindow;
// Можно получить и через frames, если мы знаем имя ифрейма (и оно у него есть)
var iframeWin = window.frames[iframe.name];
iframeWin.parent == window; // parent из iframe указывает на родительское окно
// Документ не будет доступен, если iframe с другого домена
var iframeDoc = iframe.contentWindow.document;
Больше информации об ифреймах вы можете получить в главе Общение между окнами и фреймами.
IFRAME и история посещений
IFRAME
– полноценное окно, поэтому навигация в нём попадает в историю посещений.
Это означает, что при нажатии кнопки «Назад» браузер вернёт посетителя назад не в основном окне, а в ифрейме. В лучшем случае – браузер возьмёт предыдущее состояние ифрейма из кэша и посетитель просто подумает, что кнопка не сработала. В худшем – в ифрейм будет сделан предыдущий запрос, а это уже точно ни к чему.
Наши запросы в ифрейм – служебные и для истории не предназначены. К счастью, есть ряд техник, которые позволяют обойти проблему.
-
Ифрейм нужно создавать динамически, через JavaScript.
-
Когда ифрейм уже создан, то единственный способ поменять его
src
без попадания запроса в историю посещений:// newSrc - новый адрес iframeDoc.location.replace(newSrc);
Вы можете возразить: «но ведь
iframeDoc
не всегда доступен!iframe
может быть с другого домена – как быть тогда?». Ответ: вместо сменыsrc
этого ифрейма – создать новый, с новымsrc
. -
POST-запросы в
iframe
всегда попадают в историю посещений. -
… Но если
iframe
удалить, то лишняя история тоже исчезнет :). Сделать это можно по окончании запроса.
Таким образом, общий принцип использования IFRAME
: динамически создать, сделать запрос, удалить.
Бывает так, что удалить по каким-то причинам нельзя, тогда возможны проблемы с историей, описанные выше.
Функция createIframe
Приведённая ниже функция createIframe(name, src, debug)
кросс-браузерно создаёт ифрейм с данным именем и src
.
Аргументы:
name
- Имя и
id
ифрейма src
- Исходный адрес ифрейма. Необязательный параметр.
debug
- Если параметр задан, то ифрейм после создания не прячется.
function createIframe(name, src, debug) {
src = src || 'javascript:false'; // пустой src
var tmpElem = document.createElement('div');
// в старых IE нельзя присвоить name после создания iframe
// поэтому создаём через innerHTML
tmpElem.innerHTML = '<iframe name="' + name + '" id="' + name + '" src="' + src + '">';
var iframe = tmpElem.firstChild;
if (!debug) {
iframe.style.display = 'none';
}
document.body.appendChild(iframe);
return iframe;
}
Ифрейм здесь добавляется к document.body
. Конечно, вы можете исправить этот код и добавлять его в любое другое место документа.
Кстати, при вставке, если не указан src
, тут же произойдёт событие iframe.onload
. Пока обработчиков нет, поэтому оно будет проигнорировано.
Функция postToIframe
Функция postToIframe(url, data, target)
отправляет POST-запрос в ифрейм с именем target
, на адрес url
с данными data
.
Аргументы:
url
- URL, на который отправлять запрос.
data
- Объект содержит пары
ключ:значение
для полей формы. Значение будет приведено к строке. target
- Имя ифрейма, в который отправлять данные.
// Например: postToIframe('/vote', {mark:5}, 'frame1')
function postToIframe(url, data, target) {
var phonyForm = document.getElementById('phonyForm');
if (!phonyForm) {
// временную форму создаём, если нет
phonyForm = document.createElement("form");
phonyForm.id = 'phonyForm';
phonyForm.style.display = "none";
phonyForm.method = "POST";
document.body.appendChild(phonyForm);
}
phonyForm.action = url;
phonyForm.target = target;
// заполнить форму данными из объекта
var html = [];
for (var key in data) {
var value = String(data[key]).replace(/"/g, """);
// в старых IE нельзя указать name после создания input
// поэтому используем innerHTML вместо DOM-методов
html.push("<input type='hidden' name=\"" + key + "\" value=\"" + value + "\">");
}
phonyForm.innerHTML = html.join('');
phonyForm.submit();
}
Эта функция формирует форму динамически, но, конечно, это лишь один из возможных сценариев использования.
В IFRAME
можно отправлять и существующую форму, включающую файловые и другие поля.
Запросы GET и POST
Общий алгоритм обращения к серверу через ифрейм:
-
Создаём
iframe
со случайным именемiframeName
. -
Создаём в основном окне объект
CallbackRegistry
, в котором вCallbackRegistry[iframeName]
сохраняем функцию, которая будет обрабатывать результат. -
Отправляем GET или POST-запрос в него.
-
Сервер отвечает как-то так:
<script> parent.CallbackRegistry[window.name]({данные}); </script>
…То есть, вызывает из основного окна функцию обработки (
window.name
в ифрейме – его имя). -
Дополнительно нужен обработчик
iframe.onload
– он сработает и проверит, выполнилась ли функцияCallbackRegistry[window.name]
. Если нет, значит какая-то ошибка. Сервер при нормальном потоке выполнения всегда отвечает её вызовом.
Подробнее можно понять процесс, взглянув на код.
Мы будем использовать в нём две функции – одну для GET, другую – для POST:
iframeGet(url, onSuccess, onError)
– для GET-запросов наurl
. При успешном запросе вызываетсяonSuccess(result)
, при неуспешном:onError()
.iframePost(url, data, onSuccess, onError)
– для POST-запросов наurl
. Значениемdata
должен быть объектключ:значение
для пересылаемых данных, он конвертируется в поля формы.
Пример в действии, возвращающий дату сервера при GET и разницу между датами клиента и сервера при POST:
function createIframe(name, src, debug) {
src = src || 'javascript:false'; // пустой src
var tmpElem = document.createElement('div');
// в старых IE нельзя присвоить name после создания iframe, поэтому создаём через innerHTML
tmpElem.innerHTML = '<iframe name="' + name + '" id="' + name + '" src="' + src + '">';
var iframe = tmpElem.firstChild;
if (!debug) {
iframe.style.display = 'none';
}
document.body.appendChild(iframe);
return iframe;
}
// функция постит объект-хэш content в виде формы с нужным url , target
// напр. postToIframe('/count.php', {a:5,b:6}, 'frame1')
function postToIframe(url, data, target) {
var phonyForm = document.getElementById('phonyForm');
if (!phonyForm) {
// временную форму создаем, если нет
phonyForm = document.createElement("form");
phonyForm.id = 'phonyForm';
phonyForm.style.display = "none";
phonyForm.method = "POST";
phonyForm.enctype = "multipart/form-data";
document.body.appendChild(phonyForm);
}
phonyForm.action = url;
phonyForm.target = target;
// заполнить форму данными из объекта
var html = [];
for (var key in data) {
var value = String(data[key]).replace(/"/g, """);
html.push("<input type='hidden' name=\"" + key + "\" value=\"" + value + "\">");
}
phonyForm.innerHTML = html.join('');
phonyForm.submit();
}
var CallbackRegistry = {}; // реестр
function iframeGet(url, onSuccess, onError) {
var iframeOk = false; // флаг успешного ответа сервера
var iframeName = Math.random(); // случайное имя для ифрейма
var iframe = createIframe(iframeName, url);
CallbackRegistry[iframeName] = function(data) {
iframeOk = true; // сервер ответил успешно
onSuccess(data);
}
iframe.onload = function() {
iframe.parentNode.removeChild(iframe); // очистка
delete CallbackRegistry[iframeName];
if (!iframeOk) onError(); // если сервер не ответил как надо - что-то не так
}
}
// аналогично iframeGet, но в postToIframe передаются данные data
function iframePost(url, data, onSuccess, onError) {
var iframeOk = false;
var iframeName = Math.random();
var iframe = createIframe(iframeName);
CallbackRegistry[iframeName] = function(data) {
iframeOk = true;
onSuccess(data);
}
iframe.onload = function() {
iframe.parentNode.removeChild(iframe); // очистка
delete CallbackRegistry[iframeName];
if (!iframeOk) onError(); // если коллбэк не вызвался - что-то не так
}
postToIframe(url, data, iframeName);
}
var http = require('http');
var url = require('url');
var static = require('node-static');
var file = new static.Server('.', {
cache: 0
});
var multiparty = require('multiparty');
function accept(req, res) {
var urlParsed = url.parse(req.url, true);
res.setHeader('Cache-Control', 'no-cache');
if (urlParsed.pathname == '/server') {
res.end(wrap(new Date()));
return;
} else if (urlParsed.pathname == '/diff') {
var form = new multiparty.Form();
form.parse(req, function(err, fields, files) {
var diff = new Date() - fields.clientDate[0];
res.end(wrap(diff));
});
} else {
file.serve(req, res);
}
}
function wrap(data) {
return '<script>parent.CallbackRegistry[window.name](' + JSON.stringify(data) + ')</script>';
}
// ------ запустить сервер -------
if (!module.parent) {
http.createServer(accept).listen(8080);
} else {
exports.accept = accept;
}
<!DOCTYPE html>
<html>
<body>
<script src="iframe.js"></script>
<script>
function ok(result) {
alert('result: ' + result);
}
function fail() {
alert('fail');
}
</script>
<button onclick="iframeGet('server', ok, fail)">iframeGet server</button>
<button onclick="iframePost('diff', {clientDate: +new Date}, ok, fail)">iframePost diff</button>
</body>
</html>
Прямой вызов функции внешнего окна из ифрейма отлично работает, потому что они с одного домена. Если с разных, то нужны дополнительные действия, например:
-
В IE8+ есть интерфейс postMessage для общения между окнами с разных доменов.
-
В любых, даже самых старых IE, можно обмениваться данными через
window.name
. Эта переменная хранит «имя» окна или фрейма, которое не меняется при перезагрузке страницы.Поэтому если мы сделали
POST
в<iframe>
на другой домен и он поставилwindow.name = "Вася"
, а затем сделал редирект на основной домен, то эти данные станут доступны внешней странице. -
Также в совсем старых IE можно обмениваться данными через хеш, то есть фрагмент URL после
#
. Его изменение доступно между ифреймами с разных доменов и не приводит к перезагрузке страницы. Таким образом они могут передавать данные друг другу. Есть готовые библиотеки, которые реализуют этот подход, например Porthole.
IFRAME для COMET
Бесконечный IFRAME – самый старый способ организации COMET. Когда-то он был основой AJAX-приложений, а сейчас – используется лишь в случаях, когда браузер не поддерживает современный стандарт WebSocket, то есть для IE9-.
Этот способ основан на том, что браузер читает страницу последовательно и обрабатывает все новые теги по мере того, как сервер их присылает.
Классическая реализация – это когда клиент создаёт невидимый IFRAME, ведущий на служебный URL. Сервер, получив соединение на этот URL, не закрывает его, а
время от времени присылает блоки сообщений <script>…javascript…</script>
. Появившийся в IFRAME’е javascript тут же выполняется браузером, передавая информацию на основную страницу.
Таким образом, для передачи данных используется «бесконечный» ифрейм, через который сервер присылает все новые данные.
Схема работы:
- Создаётся
<iframe src="COMET_URL">
, по адресуCOMET_URL
расположен сервер. - Сервер выдаёт начало («шапку») документа и останавливается, оставляя соединение активным.
- Когда сервер хочет что-то отправить – он пишет в соединение
<script>parent.onMessage(сообщение)</script>
Браузер тут же выполняет этот скрипт – так сообщение приходит на клиент. - Ифрейм, в теории, грузится бесконечно. Его завершение означает обрыв канала связи. Его можно поймать по
iframe.onload
и заново открыть соединение (создать новыйiframe
).
Также ифрейм можно пересоздавать время от времени, для очистки памяти от старых сообщений.
Ифрейм при этом работает только на получение данных с сервера, как альтернатива Server Sent Events. Для запросов используется обычный XMLHttpRequest
.
Обход проблем с IE
Такое использование ифреймов является хаком. Поэтому есть ряд проблем:
- Показывается индикатор загрузки, «курсор-часики».
- При POST в
<iframe>
раздаётся звук «клика». - Браузер буферизует начало страницы.
Мы должны эти проблемы решить, прежде всего, в IE, поскольку в других браузерах есть WebSocket и Server Sent Events .
Проще всего решить последнюю – IE не начинает обработку страницы, пока она не загрузится до определённого размера.
Поэтому в таком IFRAME
первые несколько сообщений задержатся:
<!DOCTYPE HTML>
<html>
<body>
<script>parent.onMessage("привет");</script>
<script>parent.onMessage("от сервера");</script>
...
Решение – забить начало ифрейма чем-нибудь, поставить, например, килобайт пробелов в начале:
<!DOCTYPE HTML>
<html>
<body>
******* 1 килобайт пробелов, а потом уже сообщения ******
<script>
parent.onMessage("привет");
</script>
<script>
parent.onMessage("от сервера");
</script>
...
Для решения проблемы с индикацией загрузки и клика мы можем использовать безопасный ActiveX-объект htmlfile
. IE не требует разрешений на его создание. Фактически, это независимый HTML-документ.
Оказывается, если iframe
создать в нём, то никакой анимации и звуков не будет.
Итак, схема:
- Основное окно
main
создаёт вспомогательный объект:new ActiveXObject("htmlfile")
. Это HTML-документ со своимwindow
, похоже на встроенныйiframe
. - В
htmlfile
записываетсяiframe
. - Цепочка общения: основное окно –
htmlfile
– ифрейм.
iframeActiveXGet
На самом деле всё ещё проще, если посмотреть на код:
Метод iframeActiveXGet
по существу идентичен обычному iframeGet
, которое мы рассмотрели. Единственное отличие – вместо createIframe
используется особый метод createActiveXFrame
:
function iframeActiveXGet(url, onSuccess, onError) {
var iframeOk = false;
var iframeName = Math.random();
var iframe = createActiveXFrame(iframeName, url);
CallbackRegistry[iframeName] = function(data) {
iframeOk = true;
onSuccess(data);
}
iframe.onload = function() {
iframe.parentNode.removeChild(iframe); // очистка
delete CallbackRegistry[iframeName];
if (!iframeOk) onError(); // если колбэк не вызвался - что-то не так
}
}
createActiveXFrame
В этой функции творится вся IE-магия:
function createActiveXFrame(name, src) {
// (1)
var htmlfile = window.htmlfile;
if (!htmlfile) {
htmlfile = window.htmlfile = new ActiveXObject("htmlfile");
htmlfile.open();
// (2)
htmlfile.write("<html><body></body></html>");
htmlfile.close();
// (3)
htmlfile.parentWindow.CallbackRegistry = CallbackRegistry;
}
// (4)
src = src || 'javascript:false';
htmlfile.body.insertAdjacentHTML('beforeEnd',
"<iframe name='" + name + "' src='" + src + "'></iframe>");
return htmlfile.body.lastChild;
}
-
Вспомогательный объект
htmlfile
будет один и он будет глобальным. Можно и спрятать переменную в замыкании. Смысл в том, что в одинhtmlfile
можно записать много ифреймов, так что не будем множить сущности и занимать ими лишнюю память. -
В
htmlfile
можно записать любой текст и, при необходимости, черезdocument.write('<script>...<\/script>)
. Здесь мы делаем пустой документ. -
Когда загрузится
iframe
, он сделает вызов:<script> parent.CallbackRegistry[window.name](объект с данными); </script>
Здесь
parent'ом
дляiframe'а
будетhtmlfile
, т.е.CallbackRegistry
будет искаться среди переменных соответствующего ему окна, а вовсе не верхнегоwindow
.Окно для
htmlfile
доступно какhtmlfile.parentWindow
, копируем в него ссылку на реестр колбэковCallbackRegistry
. Теперь ифрейм его найдёт. -
Далее вставляем ифрейм в документ. В старых
IE
нельзя поменятьname
ифрейму через DOM, поэтому вставляем строкой черезinsertAdjacentHTML
.
Пример в действии (только IE):
var CallbackRegistry = {}; // реестр
function iframeActiveXGet(url, onSuccess, onError) {
var iframeOk = false;
var iframeName = Math.random();
var iframe = createActiveXFrame(iframeName, url);
CallbackRegistry[iframeName] = function(data) {
iframeOk = true;
onSuccess(data);
}
iframe.onload = function() {
iframe.parentNode.removeChild(iframe); // очистка
delete CallbackRegistry[iframeName];
if (!iframeOk) onError(); // если коллбэк не вызвался - что-то не так
}
}
function createActiveXFrame(name, src) {
var htmlfile = window.htmlfile;
if (!htmlfile) {
htmlfile = window.htmlfile = new ActiveXObject("htmlfile");
htmlfile.open();
htmlfile.write("<html><body></body></html>");
htmlfile.close();
htmlfile.parentWindow.CallbackRegistry = CallbackRegistry;
}
src = src || 'javascript:false'; // пустой src
htmlfile.body.insertAdjacentHTML('beforeEnd', "<iframe name='" + name + "' src='" + src + "'></iframe>");
return htmlfile.body.lastChild; // window in .document.parentWindow
}
var http = require('http');
var url = require('url');
var static = require('node-static');
var file = new static.Server('.', {
cache: 0
});
var multiparty = require('multiparty');
function accept(req, res) {
var urlParsed = url.parse(req.url, true);
res.setHeader('Cache-Control', 'no-cache');
if (urlParsed.pathname == '/server') {
res.end(wrap(new Date()));
return;
} else if (urlParsed.pathname == '/diff') {
var form = new multiparty.Form();
form.parse(req, function(err, fields, files) {
var diff = new Date() - fields.clientDate[0];
res.end(wrap(diff));
});
} else {
file.serve(req, res);
}
}
function wrap(data) {
return '<script>parent.CallbackRegistry[window.name](' + JSON.stringify(data) + ')</script>';
}
// ------ запустить сервер -------
if (!module.parent) {
http.createServer(accept).listen(8080);
} else {
exports.accept = accept;
}
<!DOCTYPE html>
<html>
<body>
<script src="activex.js"></script>
<script>
function ok(result) {
alert('result: ' + result);
}
function fail() {
alert('fail');
}
function go() {
if (!("ActiveXObject" in window)) {
alert("Только Internet Explorer");
return;
}
iframeActiveXGet("server", ok, fail);
}
</script>
<button onclick="go()">iframeActiveXGet server date</button>
</body>
</html>
Запрос, который происходит, полностью незаметен.
Метод POST делается аналогично, только форму нужно добавлять не в основное окно, а в htmlfile
, через вызов htmlfile.appendChild
. В остальном – всё так же, как и при обычной отправке через ифрейм.
Впрочем, для COMET нужен именно GET.
Можно и сочетать эти способы: если есть ActiveX: if ("ActiveXObject" in window)
– используем методы для IE, описанные выше, а иначе – обычные методы.
Вот мини-приложение с сервером на Node.JS, непрерывно получающее текущее время с сервера через <iframe>
, сочетающее эти подходы:
var IframeComet = new function() {
var self = this;
var connectTries = 0,
reconnectTimer;
var htmlfile; // for ie only
var iframe;
this.onConnected = function() {
connectTries = 0;
clearTimeout(reconnectTimer);
};
this.onMessage = function(message) { /* ... */ };
this.onError = function(err) { /* ... */ };
this.open = function(url) {
connectTries++;
if (connectTries > 3) {
self.onError("Unable to connect");
}
if ("ActiveXObject" in window) { // IE
createActiveXFrame(url);
} else {
createIframe(url);
}
reconnectTimer = setTimeout(function() {
if (!self.isConnected()) {
self.open(url);
}
}, connectTries * 2000);
// в Chrome не срабатывает при обрыве соединения,
// так что используем там другой =) транспорт
iframe.onload = function() {
self.open(url);
};
};
this.isConnected = function() {
return connectTries == 0; // onConnect обнуляет connectTries
}
function cleanIframe() {
if (iframe) {
iframe.src = "javascript:false";
iframe.parentNode.removeChild(iframe); // очистка
}
}
function createIframe(src) {
cleanIframe();
iframe = document.createElement('iframe');
iframe.src = src || 'javascript:false';
iframe.style.display = 'none';
document.body.appendChild(iframe);
}
function createActiveXFrame(src) {
cleanIframe();
if (!htmlfile) {
htmlfile = new ActiveXObject("htmlfile");
htmlfile.open();
htmlfile.write("<html><body></body></html>");
htmlfile.close();
htmlfile.parentWindow.IframeComet = self;
}
src = src || 'javascript:false'; // пустой src
htmlfile.body.insertAdjacentHTML('beforeEnd', "<iframe src='" + src + "'></iframe>");
iframe = htmlfile.body.lastChild; // window in .document.parentWindow
}
}
var http = require('http');
var url = require('url');
var static = require('node-static');
var file = new static.Server('.', {
cache: 0
});
function accept(req, res) {
if (req.url == '/comet') {
res.writeHead(200, {
'Content-Type': 'text/html; charset=utf-8'
});
res.write('<!DOCTYPE HTML><html> \
<head><meta junk="' + new Array(2000).join('*') + '"/> \
<script> \
var i = parent.IframeComet; \
i.onConnected()</script> \
</head><body>');
setInterval(function() {
var now = new Date();
var timeStr = now.getHours() + ':' + now.getMinutes() + ':' + now.getSeconds();
res.write('<script>i.onMessage("' + timeStr + '")</script>');
}, 1000);
return;
} else {
file.serve(req, res);
}
}
// ------ запустить сервер -------
if (!module.parent) {
http.createServer(accept).listen(8080);
} else {
exports.accept = accept;
}
<!DOCTYPE html>
<html>
<body>
<script src="iframeComet.js"></script>
<button onclick="go()">IframeComet.open("comet");</button>
<div id="showElem"></div>
<script>
IframeComet.onMessage = IframeComet.onError = show;
function show(msg) {
showElem.innerHTML = msg;
}
function go() {
IframeComet.open("comet");
}
</script>
</body>
</html>
Ещё раз заметим, что обычно такое сочетание не нужно, так как если не IE9-, то можно использовать более современные средства для COMET.
Итого
- Iframe позволяет делать «AJAX»-запросы и хитро обходить кросс-доменные ограничения в IE7-. Обычно для этого используют либо
window.name
с редиректом, либо хеш с библиотекой типа Porthole. - В IE9- iframe можно использовать для COMET. В IE10 уже есть WebSocket.
Существует ряд уже готовых клиент-серверных библиотек, которые реализуют AJAX/COMET, в том числе и через iframe, мы рассмотрим их позже. Поэтому совсем не обязательно делать «с нуля». Хотя, как можно видеть из главы, это совсем несложно.
Комментарии
<code>
, для нескольких строк кода — тег<pre>
, если больше 10 строк — ссылку на песочницу (plnkr, JSBin, codepen…)