<input type="week">
Елементи <input>
типу week
(тиждень) створюють поля введення, що дають змогу легко ввести рік та номер тижня ISO 8601 протягом цього року (тобто тиждень від 1 до 52 або 53).
Спробуйте його в дії
Користувацький інтерфейс цього контрольного елемента у різних браузерах відрізняється; кросбраузерна підтримка наразі трохи обмежена, і лише Chrome, Opera та Microsoft Edge зараз його підтримують. У браузерах, що його не підтримують, цей контрольний елемент доладно відступає, працюючи ідентично до <input type="text">
.
Значення
Рядок, що представляє значення тижня-року, введеного в поле. Формат дати та часу, що використовується цим типом введення, описано в розділі Рядки тижнів.
Задати усталене значення поля можна шляхом додання цього значення в атрибуті value
, наприклад:
<label for="week">У якому тижні ви хотіли б розпочати?</label>
<input id="week" type="week" name="week" value="2017-W01" />
Річ, котру слід зауважити: відображений формат може відрізнятися від фактичного value
, яке завжди форматується як yyyy-Www
. Коли вищезазначене значення подається на сервер, наприклад, браузери можуть відображати його як Week 01, 2017
, але надіслане значення завжди матиме вигляд week=2017-W01
.
Також отримати та задати значення можна в JavaScript – за допомогою властивості елемента поля value
, наприклад:
const weekControl = document.querySelector('input[type="week"]');
weekControl.value = "2017-W45";
Додаткові атрибути
На додачу до атрибутів, що спільні для елементів <input>
, поля тижня мають наступні атрибути.
max
Найпізніші (за часом) рік та номер тижня, у форматі рядка, що описаний в розділі Значення вище, що будуть прийняті. Якщо value
, введене в елемент, перевищує це значення, елемент не пройде перевірку обмежень. Якщо значення атрибута max
не є дійсним рядком тижня, то такий елемент не має максимального значення.
Це значення має бути більшим або рівним року та тижню, вказаним в атрибуті min
.
min
Найраніші рік та номер тижня, що будуть прийняті. Якщо value
, введене в елемент, менше цього значення, елемент не пройде перевірку обмежень. Якщо значення атрибута min
не є дійсним рядком тижня, то такий елемент не має мінімального значення.
Це значення має бути меншим або рівним року та тижню, вказаним в атрибуті max
.
readonly
Булів атрибут, котрий, якщо є, означає, що це поле не може бути змінено користувачем. Проте його value
може бути змінено за допомогою JavaScript коду, що безпосередньо задає властивість value
HTMLInputElement
.
[!NOTE] Оскільки поле, що має атрибут
readonly
, не може мати значення,required
не має жодного ефекту на поля з атрибутомreadonly
.
step
Атрибут step
– це число, що вказує на дискретність, котрій повинно відповідати значення, або спеціальне значення any
, що описане нижче. Лише значення, котрі дорівнюють основі для кроку (min
, якщо вказано, інакше – value
, і ще інакше – відповідне усталене значення) є дійсними.
Рядкове значення any
означає, що крокування не накладається, і дозволяється будь-яке значення (з урахуванням інших обмежень, як то min
та max
).
[!NOTE] Коли дані, введені користувачем, не відповідають налаштуванням крокування, користувацький агент може округлити до найближчого дійсного значення, віддаючи перевагу числам в додатному напрямку, коли є два рівновіддалені варіанти.
Для полів week
значення step
дається в тижнях, з масштабним фактором 604.800.000 (оскільки за лаштунками часове значення зберігається в мілісекундах). Усталене значення step
– 1, що позначає 1 тиждень. Усталене значення основи крокування дорівнює -259.200.000, що є початком першого тижня 1970 року ("1970-W01"
).
Наразі неясно, що означає для step
значення "any"
, коли воно використовується на полях week
. Цей розділ буде оновлено, як тільки ця інформація буде визначена.
Застосування полів тижнів
На перший погляд, поля тижнів – зручні, адже вони надають легкий користувацький інтерфейс для вибору тижнів, а також нормалізують формат даних, що надсилається на сервер, незалежно від браузера чи локалі користувача. Проте з полями <input type="week">
є проблеми, оскільки підтримка не гарантована у всіх браузерах.
Розгляньмо базове та складніше застосування <input type="week">
, а потім запропонуємо поради щодо пом'якшення проблеми підтримки браузерами (дивіться Роботу з підтримкою браузерами).
Базове застосування week
Найбазовіше застосування <input type="week">
включає базову комбінацію з елементів <input>
і <label>
, як видно нижче:
<form>
<label for="week">У якому тижні ви хотіли б розпочати?</label>
<input id="week" type="week" name="week" />
</form>
Контроль розміру поля
<input type="week">
не підтримує атрибути розміру форми, наприклад, size
. Для потреб задання розміру доведеться обмежитись CSS.
Застосування атрибута step
Повинна була б бути змога використовувати атрибут step
, щоб змінити кількість тижнів, які перескакуються при інкременті або декременті значення, проте, схоже, що він ніяк не діє в браузерах, що підтримують тип поля week
.
Валідація
Усталено <input type="week">
не застосовує до введених значень жодної валідації. Реалізації інтерфейсу користувача, як правило, не дозволяють задати нічого, що не є дійсними тижнем-роком, що допомагає, але все одно можна відправити поле пустим, і ви, можливо, захочете обмежити діапазон доступних тижнів.
Задання максимального та мінімального тижнів
Можна скористатися атрибутами min
і max
, щоб обмежити дійсні тижні, які може вибрати користувач. У наступному прикладі задано мінімальне значення Тиждень 01, 2017
та максимальне значення Тиждень 52, 2017
:
<form>
<label for="week">У якому тижні ви хотіли б розпочати?</label>
<input id="week" type="week" name="week" min="2017-W01" max="2017-W52" />
<span class="validity"></span>
</form>
Ось CSS, використаний у прикладі вище. Тут використовуються властивості CSS :valid
та :invalid
, щоб стилізувати поле на основі того, чи є поточне значення дійсним. Довелося помістити піктограми на <span>
поруч з полем, а не на саме поле, оскільки в Chrome всередині елементу керування формою розміщується згенерований вміст, і він не може бути ефективно стилізований або показаний.
div {
margin-bottom: 10px;
position: relative;
}
input[type="number"] {
width: 100px;
}
input + span {
padding-right: 30px;
}
input:invalid + span::after {
position: absolute;
content: "✖";
padding-left: 5px;
}
input:valid + span::after {
position: absolute;
content: "✓";
padding-left: 5px;
}
Результатом є те, що лише тижні між W01 і W52 2017 року вважаються дійсними та можуть бути вибрані в браузерах, що підтримують цей тип поля.
Обов'язковість значень тижнів
На додачу, можна скористатися атрибутом required
, щоб зробити заповнення поля обов'язковим. В результаті браузери, що підтримують цей тип поля, показуватимуть помилку, якщо спробувати подати пусте поле тижня.
Погляньмо на приклад; тут задано мінімальний та максимальний тижні, а також поле зроблено обов'язковим:
<form>
<div>
<label for="week">У якому тижні ви хотіли б розпочати?</label>
<input
id="week"
type="week"
name="week"
min="2017-W01"
max="2017-W52"
required />
<span class="validity"></span>
</div>
<div>
<input type="submit" value="Подати форму" />
</div>
</form>
Якщо спробувати подати форму без значення, то браузер покаже помилку. Тепер спробуйте погратися з цим прикладом:
Ось знімок екрана для тих, хто користується браузером без підтримки полів week
:
[!WARNING] Валідація форм HTML не є заміною серверних сценаріїв, що пересвідчуються в тому, що введені дані мають коректний формат. Занадто легко підлаштувати HTML, щоб обійти валідацію, або навіть геть її прибрати. Також можна обійти HTML взагалі, й подавати дані безпосередньо на сервер. Якщо ваш код на серверному боці не може валідувати дані, котрі отримує, то може статись лихо, коли в вашу базу даних введуть некоректно відформатовані дані (або завеликі дані, дані не того типу, і так далі).
Робота з підтримкою браузерами
Як зазначалося вище, наразі основна проблема використання полів week
— це підтримка браузерами: їх не підтримують Safari та Firefox на настільних комп'ютерах, а також старі версії IE.
Мобільні платформи, як от Android та iOS, використовують такий тип полів ідеально, надаючи спеціалізовані елементи керування із користувальницьким інтерфейсом, що дає змогу легко вибирати значення в середовищі з дотиковим екраном. Наприклад, вибір тижня в Chrome для Android має такий вигляд:
Браузери без підтримки week
доладно відступають до текстового поля, але це утворює проблеми як щодо сталості користувацького інтерфейсу (представлений елемент керування буде іншим), так і обробки даних.
Друга проблема з двох – серйозніша. Як зазначалося вище, з полем week
фактичне значення завжди нормалізується до формату yyyy-Www
. Коли браузер відступає до звичайного текстового поля, то немає нічого, що б допомогло користувачеві правильно відформатувати введене значення (і це точно не інтуїтивно). Є чимало способів, у які люди можуть вводити значення тижня; наприклад:
Тиждень 1 2017
Січ 2-8 2017
2017-W01
- тощо
Наразі найкращий спосіб використання тижнів-років у формах у кросбраузерний спосіб – це дозволити користувачеві вводити номер тижня та рік в окремих елементах керування (популярними є елементи <select>
; див. приклад нижче), або використовувати бібліотеки JavaScript, наприклад, віконце вибору дати jQuery.
Приклади
У цьому прикладі створюються два набори елементів користувацького інтерфейсу для вибору тижнів: нативний інтерфейс вибору, створений за допомогою <input type="week">
, та набір з двох елементів <select>
для вибору тижнів-років у старих браузерах, які не підтримують тип поля week
.
HTML має такий вигляд:
<form>
<div class="nativeWeekPicker">
<label for="week">У якому тижні ви хотіли б розпочати?</label>
<input
id="week"
type="week"
name="week"
min="2017-W01"
max="2018-W52"
required />
<span class="validity"></span>
</div>
<p class="fallbackLabel">У якому тижні ви хотіли б розпочати?</p>
<div class="fallbackWeekPicker">
<div>
<span>
<label for="week">Тиждень:</label>
<select id="fallbackWeek" name="week"></select>
</span>
<span>
<label for="year">Рік:</label>
<select id="year" name="year">
<option value="2017" selected>2017</option>
<option value="2018">2018</option>
</select>
</span>
</div>
</div>
</form>
Значення тижнів генеруються динамічно за допомогою коду JavaScript, наведеного нижче.
Іще одна частина коду, що може бути цікавою, – код зі з'ясування можливостей браузера. Щоб з'ясувати, чи підтримує браузер <input type="week">
, створюється новий елемент <input>
, робиться спроба задати його тип як week
, а потім одразу перевірка, який тип задано фактично. Браузери, які не підтримують цей тип, повертають text
, тому що тип week
відступає до типу text
. Якщо <input type="week">
не підтримується, нативний інтерфейс вибору приховується, а замість нього виводиться запасний (елементи <select>
).
// Отримати елементи користувацького інтерфейсу
const nativePicker = document.querySelector(".nativeWeekPicker");
const fallbackPicker = document.querySelector(".fallbackWeekPicker");
const fallbackLabel = document.querySelector(".fallbackLabel");
const yearSelect = document.querySelector("#year");
const weekSelect = document.querySelector("#fallbackWeek");
// Спершу приховати запасні
fallbackPicker.style.display = "none";
fallbackLabel.style.display = "none";
// Перевірити, чи нове поле дати відступає до текстового поля, чи ні
const test = document.createElement("input");
try {
test.type = "week";
} catch (e) {
console.log(e.description);
}
// Якщо так, то виконати код всередині блоку if () {}
if (test.type === "text") {
// Приховати нативний інтерфейс вибору і показати запасний
nativePicker.style.display = "none";
fallbackPicker.style.display = "block";
fallbackLabel.style.display = "block";
// динамічно заповнити тижні
populateWeeks();
}
function populateWeeks() {
// заповнити select тижнів 52 тижнями
for (let i = 1; i <= 52; i++) {
const option = document.createElement("option");
option.textContent = i < 10 ? `0${i}` : i;
weekSelect.appendChild(option);
}
}
[!NOTE] Пам'ятайте, що деякі роки мають 53 тижні (див. Тижні на рік)! Це необхідно враховувати при розробці промислових застосунків.
Технічний підсумок
Значення | Рядок, що представляє тиждень і рік, або є порожнім |
Події |
change та
input
|
Доступні спільні атрибути |
autocomplete ,
list ,
readonly ,
step
|
Атрибути IDL |
list ,
value ,
valueAsDate ,
valueAsNumber
|
Інтерфейс DOM | |
Методи |
select() ,
stepDown()
і stepUp()
|
Неявна роль ARIA | немає відповідної ролі |
Специфікації
Специфікація |
---|
HTML Standard (HTML) # week-state-(type=week) |
Сумісність із браузерами
desktop | mobile | |||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|
type="week"
|
Chrome Full support 20 | Edge Full support 12 | Firefox No support Ні | Internet Explorer No support Ні | Opera Full support 11 | Safari No support Ні | WebView Android Full support Так | Chrome Android Full support Так | Firefox for Android Full support 18 | Opera Android Full support Так | Safari on iOS No support Ні | Samsung Internet Full support Так |
Дивіться також
- Узагальнений елемент
<input>
та інтерфейс, що використовується для роботи з ним,HTMLInputElement
- Формати дати та часу, що вживаються в HTML
<input type="datetime-local">
,<input type="date">
,<input type="time">
і<input type="month">