AJAX - группа технологий, которая используется в веб разработке для создания интерактивных приложений. AJAX позволяет передавать данные с сервера без перезагрузки страницы. Таким образом можно получать очень впечатляющие результаты. А библиотека jQuery существенно облегчает реализацию AJAX с помощью встроенных методов.
Для реализации технологии используется метод $.ajax или jQuery.ajax :
$.ajax(свойства) или $.ajax(url [, свойства])
Второй параметр был добавлен в версии 1.5 jQuery.
url - адрес запрашиваемой страницы;
properties - свойства запроса.
Полный список параметров приведен в документации jQuery.
В уроке мы используем несколько наиболее часто используемых параметров.
success (функция) - данная функция вызывается после успешного завершения запроса. Функция получает от 1 до 3 параметров (в зависимости от используемой версии библиотеки). Но первый параметр всегда содержит возвращаемые с сервера данные.
data (объект/строка) - пользовательские данные, которые передаются на запрашиваемую страницу.
dataType (строка) - возможные значения: xml, json, script или html. Описание типа данных, которые ожидаются в ответе сервера.
type (строка) - тип запроса. Возможные значения: GET или POST. По умолчанию: GET.
url (строка) - адрес URL для запроса.
Пример 1Простая передача текста.
$.ajax({ url: "response.php?action=sample1", success: function(data) { $(".results").html(data); } });
Для ответа имеется элемент div .result .
Ждем ответа
Сервер просто возвращает строку:
Echo "Пример 1 - передача завершилась успешно";
Пример 2Передаем пользовательские данные PHP скрипту.
$.ajax({ type: "POST", url: "response.php?action=sample2", data: "name=Andrew&nickname=Aramis", success: function(data){ $(".results").html(data); } });
Сервер возвращает строку со вставленными в нее переданными данными:
Echo "Пример 2 - передача завершилась успешно. Параметры: name = " . $_POST["name"] . ", nickname= " . $_POST["nickname"];
Пример 3Передача и выполнение кода JavaScript
$.ajax({ dataType: "script", url: "response.php?action=sample3", })
Сервер выполняет код:
Echo "$(".results").html("Пример 3 - Выполнение JavaScript");";
Пример 4Используем XML. Пример можно использовать для работы с внешними XML, например, RSS фидом.
$.ajax({ dataType: "xml", url: "response.php?action=sample4", success: function(xmldata){ $(".results").html(""); $(xmldata).find("item").each(function(){ $(" ").html($(this).text()).appendTo(".results"); }); } });
Сервер должен возвращать XML код:
Header ("Content-Type: application/xml; charset=UTF-8"); echo
Здесь все гораздо проще. Сначала устанавливаем кодировку выходящих данных, с помощью header. Устанавливаем запрет на кеширование данных. sleep(2) – приостанавливает работу скрипта на 2 секунды, это для того, что бы увидеть анимацию ожидания wait.gif. Выводим полученные данные, при этом читая все элементы массива $_POST и преобразуя их в нужную кодировку (для кириллицы).
Для запуска нашего ajax php приложения загружаем в браузер страничку ajax_page.html
Вот что у меня получилось при нажатии кнопки TEST AJAX:
Это ответ, полученный от файла get_ajax.php:
Остались еще вопросы по поводу ajax + php? Задайте их в комментариях ниже…
Набор пар ключ/значение, которые настраивают запрос AJAX . Все параметры являются необязательными . Допускается, но не рекомендовано установить значение по умолчанию для любого параметра с использованием метода $.ajaxSetup() .accepts (по умолчанию: зависит от dataType ).
Тип: PlainObject
.
Набор пар ключ/значение, которые отправляется в Accept
заголовка запроса. Этот заголовок сообщает серверу, какой ответ запрос будет принимать в ответ. Обратите внимание, что значение параметра, указанного в dataType
(тип данных, которые мы ожидаем от сервера) сопоставляется с указанным в параметре. Кроме того, для корректной обработки ответа от сервера необходимо в параметре converters
указать функцию, которая возвращает преобразованное значение ответа. Например:
$.ajax({
accepts
: {
mycustomtype: "application/x-some-custom-type
"
}
,
// указываем как обрабатывать ответ
converters
: {
"text mycustomtype
": function
(result
) {
// возвращаем преобразованное значение ответа
return
newresult;
}
}
,
// Ожидаемый тип данных ("mycustomtype")
dataType
: "mycustomtype
"
}
);
async (по умолчанию: true ).
Тип: Boolean
.
По умолчанию, все запросы отправляются асинхронно, если вам необходимо организовать синхронные запросы, то установите этот параметр в false
. Обратите внимание, что кроссдоменные запросы и элемент, параметр dataType
которого имеет значение "jsonp"
не поддерживают запросы в синхронном режиме. Учтите, что используя синхронные запросы вы можете временно заблокировать браузер отключив какие-либо действия пока запрос будет активен.
cache (по умолчанию: true , для dataType "script" и "jsonp" false ).
Тип: Boolean
.
Если задано значение false
, то это заставит запрашиваемые страницы не кэшироваться браузером. Обратите внимание, что значение false
будет правильно работать только с HEAD
и GET
запросами.
complete .
Тип: Function
(jqXHR
jqXHR
, String
textStatus
).
Функция, которая вызывается, когда запрос заканчивается (функция выполняется после AJAX событий "success"
или "error"
). В функцию передаются два параметра: jqXHR
(в jQuery 1.4.х объект XMLHTTPRequest
) и строка соответствующая статусу запроса ("success"
, "notmodified"
, "nocontent"
, "error"
, "timeout"
, "abort"
, или "parsererror"
). Начиная с версии jQuery 1.5 параметр complete
может принимать массив из функций, которые будут вызываться по очереди.
contents .
Тип: PlainObject
.
Объект состоящий из пар строка/регулярное выражение, определяющих, как jQuery будет обрабатывать (парсить) ответ в зависимости от типа содержимого. Добавлен в версии jQuery 1.5.
contentType (по умолчанию: "application/x-www-form-urlencoded; charset=UTF-8" ).
Тип: Boolean
, или String
.
Определяет тип содержимого, которое указывается в запросе при передаче данных на сервер. С версии с jQuery 1.6 допускается указать значение false
, в этом случае jQuery не передает в заголовке поле Content-Type
совсем.
context .
Тип: PlainObject
.
При выполнении AJAX функций обратного вызова контекстом их выполнения является объект window
. Параметр context
позволяет настроить контекст исполнения функции таким образом, что $(this
) будет ссылаться на определенный DOM
элемент, или объект. Например:
$.ajax({
url
: "test.html
",
context
: $(".myClass
"), // новый контекст исполнения функции
success
: function
(){
// если запрос успешен вызываем функцию
$(this
).html("Всё ок
"); // добавлем текстовое содержимое в элемент с классом.myClass
}
}
);
converters
Значения по умолчанию:
{
"* text
": window.String, // любой тип в текст
"text html
": true
, // текст в html
"text json
": jQuery.parseJSON, // текст в JSON
"text xml
": jQuery.parseXML // текст в XML
}
Тип: PlainObject
.
Объект, содержащий тип данных для конвертации и способ его преобразования. Значение каждого преобразователя является функцией, которая возвращает преобразованное значение ответа. Добавлен в версии jQuery 1.5.
crossDomain (по умолчанию: false для запросов внутри того же домена, true для кроссдоменных запросов).
Тип: Boolean
.
Если вы хотите сделать кроссдоменный запрос находясь на том же домене (например jsonp-запрос), то установите этот параметр в true
. Это позволит, к примеру, сделать перенаправление запроса на другой домен с вашего сервера. Добавлен в версии jQuery 1.5.
Тип: PlainObject
, или String
, или Array
.
Данные, которые будут отправлены на сервер. Если они не является строкой, то преобразуются в строку запроса. Для GET
запросов строка будет добавлена к URL. Для того, чтобы предотвратить автоматическую обработку вы можете воспользоваться параметром processData
со значением false
. Если данные передаются в составе объекта, то он должен состоять из пар ключ/значение. Если значение является массивом, то jQuery сериализует несколько значений с одним и тем же ключом (в зависимости от значения параметра traditional
, который позволяет задействовать традиционный тип сериализации основанный на методе $.param).
dataFilter .
Тип: Function
(String
data
, String
type
) => Anything
.
Функция вызывается после успешного выполнения AJAX запроса и позволяет обработать "сырые" данные, полученные из ответа сервера. Возврат данных должен происходить сразу после их обработки. Функция принимает два аргумента: data
- данные полученные от сервера в виде строки и type
- тип этих данных (значение параметра dataType
).
dataType (по умолчанию: xml , json , script , или html ).
Тип: String
.
Определяет тип данных, который вы ожидаете получить от сервера. Если тип данных не указан, то jQuery будет пытаться определить его на основе типа MIME из ответа (XML
тип MIME
приведет к получению XML , с версии jQuery 1.4 json
будет давать объект JavaScript
, script
будет выполнять скрипт, а все остальное будет возвращено в виде строки).
Основные типы (результат передается в качестве первого аргумента в функцию обратного вызова success ):
Тип: Function
(jqXHR
jqXHR
, String
textStatus
, String
errorThrown
).
Функция обратного вызова, которая вызывается если AJAX запрос не был выполнен. Функция получает три аргумента:
global (по умолчанию: true ).
Тип: Boolean
.
Логический параметр, который определяет допускается ли вызвать глобальные обработчики событий AJAX для этого запроса. Значением по умолчанию является true
. Если Вам необходимо предотвратить вызов глобальных обработчиков событий, таких как.ajaxStart() , или.ajaxStop() , то используйте значение false
.
headers (по умолчанию: { } ).
Тип: PlainObject
.
Объект, который содержит пары ключ/значение дополнительных заголовков запроса, предназначенные для отправки вместе с запросом с использованием объекта XMLHttpRequest
. Обращаю Ваше внимание, что заголовок X-Requested-With: XMLHttpRequest
добавляется всегда, но значение XMLHttpRequest по умоланию допускается изменить с использованием этого параметра. Значения headers
также могут быть переопределены параметром beforeSend
. Добавлен в версии jQuery 1.5.
ifModified (по умолчанию: false ).
Тип: Boolean
.
По умолчанию значение false
, игнорирует поля заголовка HTTP запроса, а при значении true
AJAX запрос переводится в статус успешно (success
), только в том случае, если ответ от сервера изменился с момента последнего запроса. Проверка производится путем проверки поля заголовка Last-Modified . Начиная с версии jQuery 1.4
, помимо заголовка Last-Modified производится проверка и "etag" (entity tag
) - это закрытый идентификатор, присвоенный веб-сервером на определенную версию ресурса, найденного на URL. Если содержание ресурса для этого адреса меняется на новое, назначается и новый etag.
isLocal (по умолчанию: зависит от текущего местоположения).
Тип: Boolean
.
Используйте значение true
для определения текущего окружения как "локального" (например, file:///url), даже если jQuery не распознает его таким по умоланию. Следующие протоколы в настоящее время признаются как локальные: file
, *-extension
и widget
. Если Вам необходимо изменить параметр isLocal
, то рекомендуется сделать это один раз при помощи функции $.ajaxSetup() . Добавлен в версии jQuery 1.5.1.
Тип: Boolean
, или String
.
Переопределяет имя функции обратного вызова в JSONP
запросе. Это значение будет использоваться вместо "callback
" ("http://domain.ru/test.php?callback=?"
) в составе части строки запроса в URL адресе. Например, значение {
jsonp
: "onLoad
"}
передастся на сервер в виде следующей строки запроса "http://domain/test.php?onLoad=?"
.
Начиная с версии jQuery 1.5 при установке значения параметра jsonp
в значение false
предотвращает добавление строки "?callback
" к URL адресу, или попытки использовать "=?
" для преобразования ответа. В этом случае Вы дополнительно должны указать значение параметра jsonpCallback
, например:
{
jsonp
: false
,
jsonpCallback
: "callbackName
"
}
По соображениям безопасности, если Вы не доверяете цели ваших AJAX запросов, то рекомендуется установить значение параметра jsonp
в значение false
.
jsonpCallback .
Тип: String
, или Function
.
Задает имя функции обратного вызова для JSONP
запроса. Это значение будет использоваться вместо случайного имени, которое автоматически генерируется и присваивается библиотекой jQuery. Рекомендуется, чтобы jQuery самостоятелно генерировало уникальное имя, это позволит легче управлять запросами и обрабатывать возможные ошибки. В некоторых случаях установка собственного имени функции позволит улучшить браузерное кеширование GET
запросов.
Начиная с версии jQuery 1.5, вы можете в качестве значения параметра jsonpCallback
указать функцию. В этом случае, в значение параметра jsonpCallback
должно быть установлено возвращаемое значение этой функцией.
method (по умолчанию: "GET" ).
Тип: String
.
Метод HTTP
, используемый для запроса (например, "POST"
, "GET"
, "PUT"
). Добавлен в версии jQuery 1.9.0.
mimeType .
Тип: String
.
MIME
тип, который переопределяет MIME тип, указанынй в объекте XHR
по умолчанию. Добавлен в версии jQuery 1.5.1.
password .
Тип: String
.
Пароль, который будет использован с XMLHttpRequest
в ответе на запрос проверки подлинности доступа HTTP
.
processData (по умолчанию: true ).
Тип: Boolean
.
По умолчанию данные, передаваемые в параметр data
в качестве объекта будут обработаны и преобразованы в строку запроса, подходящую для типа данных по умолчанию "application/x-www-form-urlencoded"
. Если Вам необходимо отправить DOMDocument , или другие не обработанные данные, то установите значение этого параметра в false
.
scriptCharset .
Тип: String
.
Устанавливает атрибут charset (кодировка символов) на HTML тег , используемый в запросе. Используется, когда кодировка на странице отличается от кодировки удаленного скрипта. Обратите внимание, что параметр scriptCharset
применяется только в кроссдоменных запросах с параметром type
со значением "GET"
(по умолчанию) и параметром dataType
со значением "jsonp"
, или "script"
.
statusCode (по умолчанию: { } ).
Тип: PlainObject
.
Объект числовых кодов HTTP
и функции, которые будут вызываться, когда код ответа сервера имеет соотвествующее значение (определенный код HTTP
). Например, следующая функция будет вызвана, если от сервера получен код ответа 404
, или "Not found"
(стандартный код ответа HTTP о том, что клиент был в состоянии общаться с сервером, но сервер не может найти данные согласно запросу.):
$.ajax({
statusCode
: {
404: function
(){
// выполнить функцию если код ответа HTTP 404
alert("страница не найдена
");
}
,
403: function
(){
// выполнить функцию если код ответа HTTP 403
alert("доступ запрещен
");
}
}
}
);
success .
Тип: Function
(Anything
data
, String
textStatus
, jqXHR
jqXHR
).
Функция обратного вызова, которая вызывается если AJAX запрос выполнится успешно.
Функции передаются три аргумента:
timeout .
Тип: Number
.
Устанавливает в миллисекундах
таймаут для запроса. Значение 0
означает, что таймаут не установлен. Обращаю Ваше внимание, что этот параметр переопределяет значение таймаута, установленного с помощью функции $.ajaxSetup() . Таймаут ожидания начинается в момент вызова метода $.ajax() .
traditional .
Тип: Boolean
.
Если вы планируете использовать традиционные параметры сериализации (подходит для использования в строке URL
запроса или запроса AJAX
), то установите значение этого параметра в true
.
type (по умолчанию: "GET" ).
Тип: String
.
Псевдоним (алиас) для параметра method
. Вы должны использовать type
, если вы используете версии jQuery до 1.9.0
.
url (по умолчанию: текущая страница ).
Тип: String
.
Строка, содержащая URL
адрес, на который отправляется запрос.
username .
Тип: String
.
Имя пользователя, которое будет использовано с XMLHttpRequest
в ответе на запрос проверки подлинности доступа HTTP
.
xhr (по умолчанию: ActiveXObject , когда доступен (Internet Explorer ), в других случаях XMLHttpRequest .
Тип: Function()
.
Обратный вызов для создания объекта XMLHttpRequest
. С помощью этого параметра Вы можете переопределить объект XMLHttpRequest
, чтобы обеспечить свою собственную реализацию.
xhrFields .
Тип: PlainObject
.
Объект, содержащий пары имя_поля: значение_поля, которые будут установлены на объект XHR
. Например, вы можете определить, должны ли создаваться кроссдоменные запросы с использованием таких идентификационных данных как cookie
, авторизационные заголовки или TLS сертификаты:
$.ajax({
url
: "cross_domain_url
", // адрес, на который будет отправлен запрос
xhrFields
: {
withCredentials: true
// поддерживается в jQuery 1.5.1 +
}
}
);
Сейчас мы с вами разберем работу с AJAX с помощью библиотеки jQuery. Мы будем разбирать jQuery версии 3. В ней работа с AJAX отличается от предыдущих версий. Будьте внимательны.
Внимание: для тех, кто на менторстве: этот урок желательно провести очно через скайп. Напишите мне, мы с вами назначим время занятия.
Что такое AJAXТехнология AJAX - это способ обновления части страницы без ее полной перезагрузки.
Как это происходит: в определенный момент, например, по какому-либо действию пользователя, JavaScript шлет запрос на сервер по определенному url . Через некоторое время сервер присылает ответ в виде кусочка HTML кода . Этот кусочек HTML можно затем вставить в любое место страницы.
Запросы, которые шлет JavaScript, бывают синхронными и асинхронными .
При синхронном запросе страница замирает в ожидании ответа сервера, а при асинхронном пользователь может спокойно продолжать работу со страницей, а когда придет ответ сервера - страница обновит какой-либо кусочек.
Чаще всего используются асинхронные запросы.
Использовать AJAX можно на чистом JavaScript, а можно на jQuery. Мы разберем второй вариант, так как он проще.
Простые запросы через jQueryС помощью jQuery отправить AJAX запрос очень просто - вся работа делается через метод $.ajax() . Этот метод принимает параметром объект с настройками.
Самая главная настройка - url : адрес страницы, на который отправляется запрос.
Давайте, например, отправим запрос на страницу ajax.html :
$.ajax({ url: "ajax.html", });
Запрос отправится, но результата мы не дождемся - мы просто не сказали, что делать, когда страница ajax.html ответит нам.
Для этого существует метод .done - параметром он принимает функцию, которая выполнится после ответа страницы (в нашем случае страницы ajax.html ).
Эта функция в свою очередь ожидает параметром переменную - в нее положится ответ страницы.
Итак, давайте пошлем AJAX запрос на страницу ajax.html , полученный ответ положим в переменную text и запишем его в блок #result :
$.ajax({ url: "ajax.html", }) .done(function(text) { $("#result").html(text); });
По умолчанию метод $.ajax() шлет запросы асинхронно. Это значит, что окно браузера не замирает в ожидании ответа страницы, а продолжает свою работу. А когда через некоторое время страница ответит - тогда и выполнится метод done .
Если же произошла какая-то ошибка и страница по этому url нам не отвечает - мы можем отловить это событие с помощью метода fail :
$.ajax({ url: "ajax.html", }) .done(function(text) { $("#result").html(text); }) .fail(function() { $("#result").html("Ошибка загрузки"); });
При запросе выполнится или метод done при успехе или метод fail при ошибке.
Существует также метод always , который выполнится в любом случае:
$.ajax({ url: "ajax.html", }) .always(function(html) { $("#result").html(html); });
Отправка формПредыдущие запросы отправлялись методом GET . Это значит, что мы просто обращались на определенный адрес и ожидали ответа.
Можно также отправлять запросы методом POST . В этом случае мы кроме обращения на определенный url можем еще отправить туда некоторые данные, например, данные из формы, которую заполнил пользователь.
Как отправить такой запрос: для этого необходимо добавить еще две настройки в наш объект: свойство method указать "POST" и тогда в data можно положить данные, которые отправятся на сервер методом POST:
$.ajax({ url: "ajax.html", method: "POST", data: {name: "Вася", age: 25} }) .done(function(text) { $("#result").html(text); });
Больше информацииБольше информации вы можете найти в официальной документации jQuery по AJAX: $.ajax , все методы по AJAX .
Чистый JS Что вам делать дальше:Приступайте к решению задач по следующей ссылке: задачи к уроку .
Когда все решите - переходите к изучению новой темы.
Некоторые видео могут забегать вперед, тк к этому месту учебника мы прошли еще не весь ES6. Просто пропускайте такие видео, посмотрите потом.
В эпоху современного веба, большинство сайтов становятся все более интерактивными. Если ранее для получения обновленных данных нам нужно было обновить полностью страницу, то сейчас появились технологии которые позволяют полностью страницу не загружать, а только лишь отдельную ее часть. В свою очередь это предоставляет удобство как пользователям так и владельцам серверов, ведь для пользователя загрузка страницы будет быстрее, так как загружается только отдельная часть страницы, а серверу не нужно каждый раз генерировать страницу и отдавать ее пользователю. Эти возможности просто реализовать при помощи php и ajax.
Сегодня мы разберем небольшой пример для лучшего понимания работы концепции AJAX . Иногда новичкам бывает трудно понять каким же образом взаимодействует между собой php и ajax, много людей ищут примеры того как валидировать формы на лету без перезагрузки всей страницы. Я вам вкратце покажу как это делается, для того, чтобы вы могли понять основы и принципы которые позволят вам в будущем более быстро освоить другие инструменты и писать свои собственные скрипты.
Придумаем небольшое задание себе, будем проверять наличие email адреса в базе данных без перезагрузки страницы используя php и ajax. Такой пример хорошо продемонстрирует как мы можем взаимодействовать с сервером без перезагрузки страницы в браузере, а также, это часто используется при различного рода валидациях пользовательских форм. В корневом каталоге создадим 3 файла с именами index.php , email.php , validate.js .
Создание страницыСоздадим простую страницу с одной формой, которая содержит только одно поле для ввода email.
Синтаксис файла index.php
AJAX Tutorial
Самый простой способ работать с AJAX — это подключить фреймворк jQuery , что собственно я и сделал. jQuery предоставляет нам простой в понимании и работе синтаксис для отправки AJAX запросов, почему бы не использовать это преимущество?
Создание js скриптаСинтаксис файла validate.js
$(document).ready(function(){ var email = ""; $("#email").keyup(function(){ var value = $(this).val(); $.ajax({ type:"POST", url:"email.php", data:"email="+value, success:function(msg){ if(msg == "valid"){ $("#message").html("Этот Email можно использовать.Этот Email уже занят."); } } }); }); $("#submit").click(function(){ if(email == ""){ alert("Please, put data to all email"); }else{ $.ajax({ type: "POST", url:"email.php", data:"add_email="+email, success:function(msg){ $("#message").html(msg); } }); } }); });
Обработчик на phpЭтот скрипт будет получать POST
запрос от клиента, обрабатывать его и возвращать результат. AJAX
считывает результат и на его основе принимает решение.
Синтаксис файла email.php
$connection = mysqli_connect("localhost","email","email","email"); if(isset($_POST["email"]) && $_POST["email"] != ""){ $email = $_POST["email"]; $email = mysqli_real_escape_string($connection,$email); if(!filter_var($email, FILTER_VALIDATE_EMAIL)){ echo "invalid"; }else{ $sql = "SELECT id FROM email WHERE email="$email""; $result = mysqli_query($connection,$sql); if(mysqli_num_rows($result) == 1){ echo "invalid"; }else{ echo "valid"; } } } if(isset($_POST["add_email"]) && $_POST["add_email"] != ""){ $email = mysqli_real_escape_string($connection,$_POST["add_email"]); $sql = "INSERT INTO email(email) VALUES("$email")"; if(mysqli_query($connection,$sql)){ echo Success"; }else{ echo "Error"; } }
В нашем php скрипте, самый обычный код, который обрабатывает post запрос и печатает на странице определенный текст. В результате AJAX отправляет запрос php скрипту, скрипт его обрабатывает и выдает результат, AJAX считывает результат и изменяет страницу в реальном времени.
AJAX передает POST запрос скрипту посредством этого участка кода:
$.ajax({ type:"POST", url:"email.php", data:"email="+value, success:function(msg){ if(msg == "valid"){ $("#message").html("Этот Email можно использовать. "); email = value; }else{ $("#message").html("Этот Email уже занят. "); } } });
type - Тип запроса, POST или GET. В нашем случае POST;
url - адрес скрипта которому отправляют запрос;
data - данные которые передаются в запросе;
success - что делать в результате успешного выполнения запроса. В нашем случае вызывается функция;
В самом скрипте, проверка наличия email в базе выполняется при каждом вводе символа в поле email. В скрипте за обработку ввода отвечает участок $("#email").keyup(function(){}); , который проверяет нажатие клавиши в поле с id = "email" .
Как видите, код довольно простой и не требует особо больших навыков для понимания, все завязано на обработке событий keyup() - нажатие клавиши, click() - клик мышкой по элементу. Далее следует AJAX
запрос и ответ от скрипта. Таким образом используя php и ajax можно можно получить практически безграничные возможности для создания интерактивных страниц.
Данный код не претендует на звание высококачественного, но если развить, добавить правильных валидаций на уровне клиента и сервера, ввести css, то вполне можно использовать в своих проектах.
Если у вас возникли вопросы, не стесняйтесь, пишите комментарии.
Желаю вам хорошего дня и до скорых встреч 🙂