WebComponent魔法堂:深究Custom Element 之 标准构建
前言
通过《WebComponent魔法堂:深究Custom Element 之 面向痛点编程》,我们明白到其实Custom Element并不是什么新东西,我们甚至可以在IE5.5上定义自己的alert
元素。但这种简单粗暴的自定义元素并不是我们需要的,我们需要的是具有以下特点的自定义元素:
- 自定义元素可通过原有的方式实例化(
<custom-element></custom-element>
,new CustomElement()
和document.createElement('CUSTOM-ELEMENT')
) - 可通过原有的方法操作自定义元素实例(如
document.body.appendChild
,可被CSS样式所修饰等) - 能监听元素的生命周期
而Google为首提出的H5 Custom Element让我们可以在原有标准元素的基础上向浏览器注入各种抽象层次更高的自定义元素,并且在元素CRUD操作上与原生API无缝对接,编程体验更平滑。下面我们一起来通过H5 Custom Element来重新定义alert
元素吧!
命名这件“小事”
在正式撸代码前我想让各位最头痛的事应该就是如何命名元素了,下面3个因素将影响我们的命名:
- 命名冲突。自定义组件如同各种第三方类库一样存在命名冲突的问题,那么很自然地会想到引入命名空间来解决,但由于组件的名称并不涉及组件资源加载的问题,因此我们这里简化一下——为元素命名添加前缀即可,譬如采用很JAVA的
com-cnblogs-fsjohnhuang-alert
。 - 语义化。语义化我们理解就是元素名称达到望文生义的境界,譬如
x-alert
一看上去就是知道x
是前缀而已跟元素的功能无关,alert
才是元素的功能。 - 足够的吊:)高大上的名称总让人赏心悦目,就像我们项目组之前开玩笑说要把预警系统改名为"超级无敌全球定位来料品质不间断跟踪预警综合平台",呵呵!
除了上述3点外,H5规范中还有这条规定:自定义元素必须至少包含一个连字符,即最简形式也要这样a-b
。而不带连字符的名称均留作浏览器原生元素使用。换个说法就是名称带连字符的元素被识别为有效的自定义元素,而不带连字符的元素要么被识别为原生元素,要么被识别为无效元素。
const compose = (...fns) => {
const lastFn = fns.pop()
fns = fns.reverse()
return a => fns.reduce((p, fn) => fn(p), lastFn(a))
}
const info = msg => console.log(msg)
const type = o => Object.prototype.toString.call(o)
const printType = compose(info, type)
const newElem = tag => document.createElement(tag)
// 创建有效的自定义元素
const xAlert = newElem('x-alert')
infoType(xAlert) // [object HTMLElement]
// 创建无效的自定义元素
const alert = newElem('alert')
infoType(alert) // [object HTMLUnknownElement]
// 创建有效的原生元素
const div = newElem('div')
infoType(div) // [object HTMLDivElement]
那如果我偏要用alert
来自定义元素呢?浏览器自当会说一句“悟空,你又调皮了”
现在我们已经通过命名规范来有效区分自定义元素和原生元素,并且通过前缀解决了命名冲突问题。嘿稍等,添加前缀真的是解决命名冲突的好方法吗?这其实跟通过添加前缀解决id冲突一样,假如有两个元素发生命名冲突时,我们就再把前缀加长直至不再冲突为止,那就有可能出现很JAVA的com-cnblogs-fsjohnhuang-alert
的命名,噪音明显有点多,直接降低语义化的程度,重点还有每次引用该元素时都要敲这么多字符,打字的累看的也累。这一切的根源就是有且仅有一个Scope——Global Scope,因此像解决命名冲突的附加信息则无法通过上下文来隐式的提供,直接导致需要通过前缀的方式来硬加上去。
前缀的方式我算是认了,但能不能少打写字呢?像命名空间那样
木有命名冲突时
#!usr/bin/env python
# -*- coding: utf-8 -*-
from django.http import HttpResponse
def index(request):
return HttpResponse('Hello World!')
存在命名冲突时
#!usr/bin/env python
# -*- coding: utf-8 -*-
import django.db.models
import peewee
type(django.db.models.CharField)
type(peewee.CharField)
前缀也能有选择的省略就好了!
把玩Custome Element v0
对元素命名吐嘈一地后,是时候把玩API了。
从头到脚定义新元素
/** x-alert元素定义 **/
const xAlertProto = Object.create(HTMLElement.prototype, {
/* 元素生命周期的事件 */
// 实例化时触发
createdCallback: {
value: function(){
console.log('invoked createCallback!')
const raw = this.innerHTML
this.innerHTML = `<div class="alert alert-warning alert-dismissible fade in">
<button type="button" class="close" aria-label="Close">
<span aria-hidden="true">×</span>
</button>
<div class="content">${raw}</div>
</div>`
this.querySelector('button.close').addEventListener('click', _ => this.close())
}
},
// 元素添加到DOM树时触发
attachedCallback: {
value: function(){
console.log('invoked attachedCallback!')
}
},
// 元素DOM树上移除时触发
detachedCallback: {
value: function(){
console.log('invoked detachedCallback!')
}
},
// 元素的attribute发生变化时触发
attributeChangedCallback: {
value: function(attrName, oldVal, newVal){
console.log(`attributeChangedCallback-change ${attrName} from ${oldVal} to ${newVal}`)
}
},
/* 定义元素的公有方法和属性 */
// 重写textContent属性
textContent: {
get: function(){ return this.querySelector('.content').textContent },
set: function(val){ this.querySelector('.content').textContent = val }
},
close: {
value: function(){ this.style.display = 'none' }
},
show: {
value: function(){ this.style.display = 'block' }
}
})
// 向浏览器注册自定义元素
const XAlert = document.registerElement('x-alert', { prototype: xAlertProto })
/** 操作 **/
// 实例化
const xAlert1 = new XAlert() // invoked createCallback!
const xAlert2 = document.createElement('x-alert') // invoked createCallback!
// 添加到DOM树
document.body.appendChild(xAlert1) // invoked attachedCallback!
// 从DOM树中移除
xAlert1.remove() // invoked detachedCallback!
// 仅作为DIV的子元素,而不是DOM树成员不会触发attachedCallback和detachedCallback函数
const d = document.createElement('div')
d.appendChild(xAlert1)
xAlert1.remove()
// 访问元素实例方法和属性
xAlert1.textContent = 1
console.log(xAlert1.textContent) // 1
xAlert1.close()
// 修改元素实例特性
xAlert1.setAttribute('d', 1) // attributeChangedCallback-change d from null to 1
xAlert1.removeAttribute('d') // attributeChangedCallback-change d from 1 to null
// setAttributeNode和removeAttributeNode方法也会触发attributeChangedCallback
上面通过定义x-alert
元素展现了Custom Element的所有API,其实就是继承HTMLElement
接口,然后选择性地实现4个生命周期回调方法,而在createdCallback
中书写自定义元素内容展开的逻辑。另外可以定义元素公开属性和方法。最后通过document.registerElement
方法告知浏览器我们定义了全新的元素,你要好好对它哦!
那现在的问题在于假如<x-alert></x-alert>
这个HTML Markup出现在document.registerElement
调用之前,那会出现什么情况呢?这时的x-alert
元素处于unresolved状态,并且可以通过CSS Selector :unresolved
来捕获,当执行document.registerElement
后,x-alert
元素则处于resolved状态。于是可针对两种状态作样式调整,告知用户处于unresolved状态的元素暂不可用,敬请期待。
<style>
x-alert{
display: block;
}
x-alert:unresolved{
content: 'LOADING...';
}
</style>
渐进增强原生元素
有时候我们只是想在现有元素的基础上作些功能增强,倘若又要从头做起那也太折腾了,幸好Custom Element规范早已为我们想好了。下面我们来对input元素作增强
const xInputProto = Object.create(HTMLInputElement.prototype, {
createdCallback: {
value: function(){ this.value = 'x-input' }
},
isEmail: {
value: function(){
const val = this.value
return /[0-9a-zA-Z]+@\S+\.\S+/.test(val)
}
}
})
document.registerElement('x-input', {
prototype: xInputProto,
extends: 'input'
})
// 操作
const xInput1 = document.createElement('input', 'x-input') // <input is="x-input">
console.log(xInput1.value) // x-input
console.log(xInput1.isEmail()) // false
Custom Element v1 —— 换个装而已啦
Custom Element API现在已经升级到v1版本了,其实就是提供一个专门的window.customElements
作为入口来统一管理和操作自定义元素,并且以对ES6 class更友善的方式定义元素,其中的步骤和概念并没有什么变化。下面我们采用Custom Element v1的API重写上面两个示例
- 从头定义
class XAlert extends HTMLElement{
// 相当于v0中的createdCallback,但要注意的是v0中的createdCallback仅元素处于resolved状态时才触发,而v1中的constructor就是即使元素处于undefined状态也会触发,因此尽量将操作延迟到connectedCallback里执行
constructor(){
super() // 必须调用父类的构造函数
const raw = this.innerHTML
this.innerHTML = `<div class="alert alert-warning alert-dismissible fade in">
<button type="button" class="close" aria-label="Close">
<span aria-hidden="true">×</span>
</button>
<div class="content">${raw}</div>
</div>`
this.querySelector('button.close').addEventListener('click', _ => this.close())
}
// 相当于v0中的attachedCallback
connectedCallback(){
console.log('invoked connectedCallback!')
}
// 相当于v0中的detachedCallback
disconnectedCallback(){
console.log('invoked disconnectedCallback!')
}
// 相当于v0中的attributeChangedCallback,但新增一个可选的observedAttributes属性来约束所监听的属性数目
attributeChangedCallback(attrName, oldVal, newVal){
console.log(`attributeChangedCallback-change ${attrName} from ${oldVal} to ${newVal}`)
}
// 缺省时表示attributeChangedCallback将监听所有属性变化,若返回数组则仅监听数组中的属性变化
static get observedAttributes(){ return ['disabled'] }
// 新增事件回调,就是通过document.adoptNode方法修改元素ownerDocument属性时触发
adoptedCallback(){
console.log('invoked adoptedCallback!')
}
get textContent(){
return this.querySelector('.content').textContent
}
set textContent(val){
this.querySelector('.content').textContent = val
}
close(){
this.style.display = 'none'
}
show(){
this.style.display = 'block'
}
}
customElements.define('x-alert', XAlert)
- 渐进增强
class XInput extends HTMLInputElement{
constructor(){
super()
this.value = 'x-input'
}
isEmail(){
const val = this.value
return /[0-9a-zA-Z]+@\S+\.\S+/.test(val)
}
}
customElements.define('x-input', XInput, {extends: 'input'})
// 实例化方式
document.createElement('input', {is: 'x-input'})
new XInput()
<input is="x-input">
除此之外之前的unresolved状态改成defined和undefined状态,CSS对应的选择器为:defined
和:not(:defined)
。
还有就是新增一个customeElements.whenDefined({String} tagName):Promise
方法,让我们能监听自定义元素从undefined转换为defined的事件。
<share-buttons>
<social-button type="twitter"><a href="...">Twitter</a></social-button>
<social-button type="fb"><a href="...">Facebook</a></social-button>
<social-button type="plus"><a href="...">G+</a></social-button>
</share-buttons>
// Fetch all the children of <share-buttons> that are not defined yet.
let undefinedButtons = buttons.querySelectorAll(':not(:defined)');
let promises = [...undefinedButtons].map(socialButton => {
return customElements.whenDefined(socialButton.localName);
));
// Wait for all the social-buttons to be upgraded.
Promise.all(promises).then(() => {
// All social-button children are ready.
});
从头定义一个刚好可用的元素不容易啊!
到这里我想大家已经对Custom Element API有所认识了,下面我们尝试自定义一个完整的元素吧。不过再实操前,我们先看看一个刚好可用的元素应该注意哪些细节。
明确各阶段适合的操作
1.constructor
用于初始化元素的状态和设置事件监听,或者创建Shadow Dom。
2.connectedCallback
资源获取和元素渲染等操作适合在这里执行,但该方法可被调用多次,因此对于只执行一次的操作要自带检测方案。
3.disconnectedCallback
适合作资源清理等工作(如移除事件监听)
更细的细节
1.constructor中的细节
1.1. 第一句必须调用super()
保证父类实例创建
1.2. return
语句要么没有,要么就只能是return
或return this
1.3. 不能调用document.write
和document.open
方法
1.4. 不要访问元素的特性(attribute)和子元素,因为元素可能处于undefined状态并没有特性和子元素可访问
1.5. 不要设置元素的特性和子元素,因为即使元素处于defined状态,通过document.createElement
和new
方式创建元素实例时,本应该是没有特性和子元素的
2.打造focusable元素 by tabindex特性
默认情况下自定义元素是无法获取焦点的,因此需要显式添加tabindex
特性来让其focusable。另外还要注意的是若元素disabled
为true
时,必须移除tabindex
让元素unfocusable。
3.ARIA特性
通过ARIA特性让其他阅读器等其他访问工具可以识别我们的自定义元素。
4.事件类型转换
通过addEventListener
捕获事件,然后通过dispathEvent
发起事件来对事件类型进行转换,从而触发更符合元素特征的事件类型。
下面我们来撸个x-btn
吧
class XBtn extends HTMLElement{
static get observedAttributes(){ return ['disabled'] }
constructor(){
super()
this.addEventListener('keydown', e => {
if (!~[13, 32].indexOf(e.keyCode)) return
this.dispatchEvent(new MouseEvent('click', {
cancelable: true,
bubbles: true
}))
})
this.addEventListener('click', e => {
if (this.disabled){
e.stopPropagation()
e.preventDefault()
}
})
}
connectedCallback(){
this.setAttribute('tabindex', 0)
this.setAttribute('role', 'button')
}
get disabled(){
return this.hasAttribute('disabled')
}
set disabled(val){
if (val){
this.setAttribute('disabled','')
}
else{
this.removeAttribute('disabled')
}
}
attributeChangedCallback(attrName, oldVal, newVal){
this.setAttribute('aria-disabled', !!this.disabled)
if (this.disabled){
this.removeAttribute('tabindex')
}
else{
this.setAttribute('tabindex', '0')
}
}
}
customElements.define('x-btn', XBtn)
如何开始使用Custom Element v1?
Chrome54默认支持Custom Element v1,Chrome53则须要修改启动参数chrome --enable-blink-features=CustomElementsV1
。其他浏览器可使用webcomponets.js这个polyfill。
题外话一番
关于Custom Element我们就说到这里吧,不过我在此提一个有点怪但又确实应该被注意到的细节问题,那就是自定义元素是不是一定要采用<x-alert></x-alert>
来声明呢?能否采用<x-alert/>
或<x-alert>
的方式呢?
答案是不行的,由于自定义元素属于Normal Element,因此必须采用<x-alert></x-alert>
这种开始标签和闭合标签来声明。那么什么是Normal Element呢?
其实元素分为以下5类:
- Void elements
格式为<tag-name>
,包含以下元素area
,base
,br
,col
,embed
,hr
,img
,keygen
,link
,meta
,param
,source
,track
,wbr
- Raw text elements
格式为<tag-name></tag-name>
,包含以下元素script
,style
- escapable raw text elements
格式为<tag-name></tag-name>
,包含以下元素textarea
,title
- Foreign elements
格式为<tag-name/>
,MathML和SVG命名空间下的元素 - Normal elements
格式为<tag-name></tag-name>
,除上述4种元素外的其他元素。某些条件下可以省略结束标签,因为浏览器会自动为我们补全,但结果往往会很吊轨,所以还是自己写完整比较安全。
总结
当头一回听到Custom Element时我是那么的兴奋不已,犹如找到根救命稻草似的。但如同其他新技术的出现一样,利弊同行,如何判断和择优利用是让人头痛的事情,也许前人的经验能给我指明方向吧!下篇《WebComponent魔法堂:深究Custom Element 之 从过去看现在》,我们将穿越回18年前看看先驱HTML Component的黑历史,然后再次审视WebComponent吧!
尊重原创,转载请注明来自:http://www.cnblogs.com/fsjohnhuang/p/5938790.html _肥仔John
感谢
How to Create Custom HTML Elements
A vocabulary and associated APIs for HTML and XHTML
Custom Elements v1
custom-elements-customized-builtin-example
WebComponent魔法堂:深究Custom Element 之 标准构建的更多相关文章
- WebComponent魔法堂:深究Custom Element 之 面向痛点编程
前言 最近加入到新项目组负责前端技术预研和选型,一直偏向于以Polymer为代表的WebComponent技术线,于是查阅各类资料想说服老大向这方面靠,最后得到的结果是:"资料99%是英语 ...
- WebComponent魔法堂:深究Custom Element 之 从过去看现在
前言 说起Custom Element那必然会想起那个相似而又以失败告终的HTML Component.HTML Component是在IE5开始引入的新技术,用于对原生元素作功能"增强& ...
- JS魔法堂:属性、特性,傻傻分不清楚
一.前言 或许你和我一样都曾经被下面的代码所困扰 var el = document.getElementById('dummy'); el.hello = "test"; con ...
- CSS魔法堂:"那不是bug,是你不懂我!" by inline-block
前言 每当来个需要既要水平排版又要设置固定高宽时,我就会想起display:inline-block,还有为了支持IE5.5/6/7的hack*display:inline;*zoom:1;.然后发 ...
- HTML5魔法堂:全面理解Drag & Drop API
一.前言 在HTML4的时代,各前端工程师为了实现拖拽功能可说是煞费苦心,初听HTML5的DnD API觉得那些痛苦的日子将一去不复返,但事实又是怎样的呢?下面我们一起来看看DnD API的真面 ...
- JS魔法堂:判断节点位置关系
一.前言 在polyfill querySelectorAll 和写弹出窗时都需要判断两个节点间的位置关系,通过jQuery我们可以轻松搞定,但原生JS呢?下面我将整理各种判断方法,以供日后查阅. 二 ...
- JS魔法堂:那些困扰你的DOM集合类型
一.前言 大家先看看下面的js,猜猜结果会怎样吧! 可选答案: ①. 获取id属性值为id的节点元素 ②. 抛namedItem is undefined的异常 var nodes = documen ...
- JS魔法堂:doctype我们应该了解的基础知识
一.前言 什么是doctype?其实我们一直使用,却很少停下来看清楚它到底是什么,对网页有什么作用.本篇将和大家一起探讨那个默默无闻的doctype吧! 二.什么是doctype doctype或DT ...
- CSS魔法堂:重拾Border之——不仅仅是圆角
前言 当CSS3推出border-radius属性时我们是那么欣喜若狂啊,一想到终于不用再添加额外元素来模拟圆角了,但发现border-radius还分水平半径和垂直半径,然后又发现border-t ...
随机推荐
- .NET 对接JAVA 使用Modulus,Exponent RSA 加密
最近有一个工作是需要把数据用RSA发送给Java 虽然一开始标准公钥 net和Java RSA填充的一些算法不一样 但是后来这个坑也补的差不多了 具体可以参考 http://www.cnblogs. ...
- 虚拟dom与diff算法 分析
好文集合: 深入浅出React(四):虚拟DOM Diff算法解析 全面理解虚拟DOM,实现虚拟DOM
- ASP.NET Core应用的错误处理[1]:三种呈现错误页面的方式
由于ASP.NET Core应用是一个同时处理多个请求的服务器应用,所以在处理某个请求过程中抛出的异常并不会导致整个应用的终止.出于安全方面的考量,为了避免敏感信息的外泄,客户端在默认的情况下并不会得 ...
- Kotlin与Android SDK 集成(KAD 05)
作者:Antonio Leiva 时间:Dec 19, 2016 原文链接:https://antonioleiva.com/kotlin-integrations-android-sdk/ 使用Ko ...
- Android开发学习——动画
帧动画> 一张张图片不断的切换,形成动画效果* 在drawable目录下定义xml文件,子节点为animation-list,在这里定义要显示的图片和每张图片的显示时长 ...
- Oracle 列数据聚合方法汇总
网上流传众多列数据聚合方法,现将各方法整理汇总,以做备忘. wm_concat 该方法来自wmsys下的wm_concat函数,属于Oracle内部函数,返回值类型varchar2,最大字符数4000 ...
- IP报头
位字段的值设置为二进制的0100表示IP版本4(IPv4).设置为0110表示IP版本6(IPv6) 位,它表示32位字长的IP报头长度,设计报头长度的原因是数据包可选字段大小会发生变化.IP ...
- [转]ThinkPHP中实例化对象M()和D()的区别,select和find的区别
1.ThinkPHP中实例化对象M()和D()的区别 在实例化的过程中,经常使用D方法和M方法,这两个方法的区别在于M方法实例化模型无需用户为每个数据表定义模型类,如果D方法没有找到定义的模型类,则会 ...
- Ubuntu(Linux) + mono + xsp4 + nginx +asp.net MVC3 部署
折腾了一下,尝试用Linux,部署mvc3. 分别用过 centos 和 ubuntu ,用ubuntu是比较容易部署的. 操作步骤如下: 一.终端分别如下操作 sudo su ->输入密码 a ...
- Jexus服务器SSL二级证书安装指南
申请获得服务器证书有三张,一张服务器证书,二张中级CA证书.在Android微信中访问Https,如果服务器只有一张CA证书,就无法访问. 获取服务器证书中级CA证书: 为保障服务器证书在客户端的兼容 ...