介紹
Angular Components、ReactJS Components、VueJS Components 通通都有,就是沒有屬於 VanillaJS 的 Components,所以今後只好自己創造 (大誤)。
其實早在 2011 年, Web Components 的概念就已經被提出,除了希望可以建立重複使用的 Web UI 元件外,也帶入了封裝的概念,如透過 Shadow DOM 的引入以封裝 Custom Element 的 CSS 樣式,避免互相污染,另外透過 Custom Element 中自定義的行為來達到功能上的封裝與重用性。
上面提到 Shadow DOM
、 Custom Elements
,另外還有 HTML Template
為 Web Components 主要的三個核心內容,以下我們就分別介紹這三項核心內容。
Custom elements
Custom elements 透過原生 JavaScript APIs 來自定義客製化 HTML 標籤,包含它的 attributes、properties與 events 等,方便依照網站設計的客製化需求來使用。
|
|
上面的範例示範了如何透過 JavaScript 來宣告一個 Custom Element,並定義一個 custom-element
的 HTML 標籤,定義完成後就可以透過下列的範例來實用這個 Custom Element。
|
|
Shadow DOM
在講 Shadow DOM
之前我們先介紹一下什麼是 DOM tree
, HTML 是一個結構化的標籤語言,所以每一個標籤在 DOM tree 上可以看作一個節點 (Node)。
DOM Tree 範例 (圖片來源: 維基百科)
而 Shadow DOM 則是允許創建隱藏的 DOM trees 並附加在任意一個的 DOM 節點上。
Shadow DOM 渲染 (圖片來源: MDN)
由於 Shadow DOM 是屬於隱藏的 DOM trees,所以可以起到封裝的效果,您可以在 Shadow DOM 中加上 <style></style>
元件,並在其中撰寫關於這個元件的 CSS 樣式,而這裡面的 CSS 樣式並不會去影響到原生 DOM trees 中的節點,這屬於樣式的封裝。
另一個封裝的效果是關於 id
與 class
,在 HTML 中擁有同樣 id
的 element 只能同時存在一個,對於 UI 元件設計人員來說這非常不方便,而不同的 elements 雖然可以使用相同的 class
屬性,但命名衝突時還是可能造成下列的問題:
- CSS 選擇器問題: 在使用 JavaScript 取用特定 element 時使用 CSS selector 可能造成元件選擇錯誤,如:
1 2 3 4 5 6 7
<script> // 使用 class 取用元件時,可能因命名衝突選到錯誤的元件 const sectionElement = document.querySelector('.section'); </script> <div class="section">Another element</div> <div class="section">Target element</div>
- 樣式套用問題: 不同的 element 因為設計錯誤而使用了相同的
id
或class
也可能造成樣式套用錯誤,如下範例,.right
在第一個div
元件中代表著文字應該靠右對齊,而在第二個div
元件中代表著靠右浮動,但由於使用了相同的 class 名稱造成命名衝突,所以導致樣式套用的問題。1 2 3 4 5 6 7 8 9 10 11 12 13
<style> /* style for element A */ .right { text-align: right; } /* style for element B */ .right { float: right; } </style> <div class="right">Element A</div> <div class="right">Element B</div>
Shadow DOM 能有效封裝頁面元素的 id
和 class
,防止不同元件之間的命名衝突。
舉例來說,元件 A 定義了一個class為 .header
的元素,元件 B 也定義了一個同名 class。
使用Shadow DOM後,這兩個 class 不會互相影響:
|
|
HTML templates
HTML templates 提供了 <template>
以及 <slot>
兩個標籤提供一個方式來注入 HTML 到 Custom Elements 中。
template 標籤
使用 <template>
標籤,我們可以定義一個在元件實體中會重覆出現的 html 結構,例如在一個輸入信用卡號的 Custom Element 中,會重複出現四個卡號的 input 元件
、一個後三碼 input 元件
,還有兩個有效期限的輸入欄位
:
|
|
slot 標籤
<slot>
標籤則方便我們使用在需要動態注入 html 元件或 text node 的情況,例如我們實作了一個紅色邊框元件 <red-frame>
,可以在任意 html 元件上都加上紅色框框,範例如下:
|
|
複習
今天我們提到了關於 Web Components 的介紹,並提到了關於 Web Components 的三個核心功能: Shadow DOM
、 Custom Elements
,另外還有 HTML Template
。
除了可以封裝我們辛苦寫好的 UI 元件方便後續可以重複使用外,也介紹了使用 Shadow DOM
可以避免不同元間之間樣式污染還有因為命名衝突所帶來的問題。
最後介紹了透過 template
來實作 html 樣版設計,以及使用 slot
來動態注入 html 元件到 custom element 中,增加 custom element 的靈活度,後續我們也會再透過實際的一個設計案例來帶大家從無到有實作一個 Web Components,希望透過 Web Components 的封裝,可以讓大家在前端設計與開發上都更輕鬆 😊