Вбудований елемент <template> використовується як обгортка для шаблонної розмітки HTML. Браузер не бере до уваги його вміст, лише перевіряє на правильність синтаксису. Ми можемо доступитися та використовувати його у JavaScript, щоб створювати інші елементи.
Теоретично, ми могли б створити будь-який невидимий елемент де-небудь у HTML для зберігання у ньому розмітки HTML. Тоді що ж такого особливого у <template>?
По-перше, він може містити у собі будь-який коректний HTML, навіть якщо за звичайних умов він би потребував якогось додаткового тега.
Наприклад, ми можемо помістити тег рядка таблиці <tr>:
<template> <tr> <td>Вміст</td> </tr> </template> Зазвичай, якщо ми пробуємо помістити тег <tr> всередину, скажімо, тега <div>, браузер помічає некоректну структуру DOM та “виправляє” її, додаючи навколо <table>. Але це не те, що ми хочемо. На противагу цьому, <template> зберігає дані у тому вигляді, у якому ми їх туди помістили.
Ми також можемо помістити всередину <template> стилі чи скрипт:
<template> <style> p { font-weight: bold; } </style> <script> alert("Привіт"); </script> </template> Браузер розглядає вміст тега <template> як такий, який існує “за межами документа”: до нього не застосовуються стилі, не виконуються скрипти, <video autoplay> не запускається і тому подібне.
Вміст починає опрацьовуватися (застосовуються стилі, виконуються скрипти і т. ін.), коли ми вставляємо його у документ.
Вставка шаблону
Вміст template доступний у його властивості content як DocumentFragment – спеціальний тип вузла DOM.
Ми можемо працювати з ним як зі всіма іншими вузлами DOM, за винятком однієї особливості: коли ми його кудись вставляємо, вставляються його дочірні елементи, а не він сам.
Наприклад:
<template id="tmpl"> <script> alert("Привіт"); </script> <div class="message">Привіт, світ!</div> </template> <script> let elem = document.createElement('div'); // Клонувати вміст шаблону, для його подальшого використання elem.append(tmpl.content.cloneNode(true)); document.body.append(elem); // Тепер запускається скрипт з <template> </script> Перепишемо приклад Shadow DOM з попереднього розділу, використовуючи <template>:
<template id="tmpl"> <style> p { font-weight: bold; } </style> <p id="message"></p> </template> <div id="elem">Клікни мене</div> <script> elem.onclick = function() { elem.attachShadow({mode: 'open'}); elem.shadowRoot.append(tmpl.content.cloneNode(true)); // (*) elem.shadowRoot.getElementById('message').innerHTML = "Привіт зі світу тіней!"; }; </script> У рядку (*), коли ми клонуємо та вставляємо tmpl.content, як його DocumentFragment, натомість отримуємо його дочірні (<style>, <p>).
Вони формують Shadow DOM:
<div id="elem"> #shadow-root <style> p { font-weight: bold; } </style> <p id="message"></p> </div> Підсумки
Узагальнимо:
<template>його вмістом може бути будь-який синтаксично правильний HTML.<template>вміст вважається “за межами документа”, тому він ні на що не впливає.- Ми можемо доступитися до
template.contentз JavaScript, клонувати його, щоб потім знову використати у новому компоненті.
Тег <template> є унікальним, оскільки:
- Браузер перевіряє HTML синтаксис всередині нього (на відміну від рядка всередині скрипта з тією ж розміткою).
- …Але все одно дозволяє використання будь-якого тега HTML, навіть тих, використання яких не має сенсу без відповідної обгортки (напр.
<tr>). - Вміст стає інтерактивним: виконуються скрипти, запускається
<video autoplay>і т. ін., коли ми переміщаємо його у документ.
Елемент <template> не має жодних механізмів ітерації, зв’язування даних чи заміни змінних, але ми можемо реалізувати їх поверх нього.
Коментарі
<code>, для кількох рядків – обгорніть їх тегом<pre>, для понад 10 рядків – використовуйте пісочницю (plnkr, jsbin, codepen…)