WebComponent魔法堂:深究Custom Element 之 标准构建

前言

 通过《WebComponent魔法堂:深究Custom Element 之 面向痛点编程》,我们明白到其实Custom Element并不是什么新东西,我们甚至可以在IE5.5上定义自己的alert元素。但这种简单粗暴的自定义元素并不是我们需要的,我们需要的是具有以下特点的自定义元素:

  1. 自定义元素可通过原有的方式实例化(<custom-element></custom-element>,new CustomElement()document.createElement('CUSTOM-ELEMENT'))
  2. 可通过原有的方法操作自定义元素实例(如document.body.appendChild,可被CSS样式所修饰等)
  3. 能监听元素的生命周期
     而Google为首提出的H5 Custom Element让我们可以在原有标准元素的基础上向浏览器注入各种抽象层次更高的自定义元素,并且在元素CRUD操作上与原生API无缝对接,编程体验更平滑。下面我们一起来通过H5 Custom Element来重新定义alert元素吧!

命名这件“小事”

 在正式撸代码前我想让各位最头痛的事应该就是如何命名元素了,下面3个因素将影响我们的命名:

  1. 命名冲突。自定义组件如同各种第三方类库一样存在命名冲突的问题,那么很自然地会想到引入命名空间来解决,但由于组件的名称并不涉及组件资源加载的问题,因此我们这里简化一下——为元素命名添加前缀即可,譬如采用很JAVA的com-cnblogs-fsjohnhuang-alert
  2. 语义化。语义化我们理解就是元素名称达到望文生义的境界,譬如x-alert一看上去就是知道x是前缀而已跟元素的功能无关,alert才是元素的功能。
  3. 足够的吊:)高大上的名称总让人赏心悦目,就像我们项目组之前开玩笑说要把预警系统改名为"超级无敌全球定位来料品质不间断跟踪预警综合平台",呵呵!
     除了上述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">&times;</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重写上面两个示例

  1. 从头定义
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">&times;</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)
  1. 渐进增强
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语句要么没有,要么就只能是returnreturn this
1.3. 不能调用document.writedocument.open方法
1.4. 不要访问元素的特性(attribute)和子元素,因为元素可能处于undefined状态并没有特性和子元素可访问
1.5. 不要设置元素的特性和子元素,因为即使元素处于defined状态,通过document.createElementnew方式创建元素实例时,本应该是没有特性和子元素的
2.打造focusable元素 by tabindex特性
 默认情况下自定义元素是无法获取焦点的,因此需要显式添加tabindex特性来让其focusable。另外还要注意的是若元素disabledtrue时,必须移除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类:

  1. Void elements
     格式为<tag-name>,包含以下元素area,base,br,col,embed,hr,img,keygen,link,meta,param,source,track,wbr
  2. Raw text elements
     格式为<tag-name></tag-name>,包含以下元素script,style
  3. escapable raw text elements
     格式为<tag-name></tag-name>,包含以下元素textarea,title
  4. Foreign elements
     格式为<tag-name/>,MathML和SVG命名空间下的元素
  5. 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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 213,752评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,100评论 3 387
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,244评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,099评论 1 286
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,210评论 6 385
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,307评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,346评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,133评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,546评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,849评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,019评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,702评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,331评论 3 319
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,030评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,260评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,871评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,898评论 2 351

推荐阅读更多精彩内容

  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,644评论 18 139
  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,917评论 25 707
  • 在听这首歌,突然听歌词里有 "china bule "这个词,以前都没注意过。 作为中国人,听到China就敏感。...
    WoodSage阅读 525评论 0 0
  • 爸爸 爸爸 已经好久未这样称呼您啦 只喊:老爸 老爸 忽然 好想近在咫尺的爸爸 在每个清晨黄昏里 在每顿早餐晚饭后...
    塵光阅读 529评论 0 2
  • 又是一年圣诞节,朋友圈也犹如一场没有硝烟的战争:有人晒着其乐融融的美照,洋溢着幸福的笑脸。然而,还有一些人却唱着反...
    韧青阅读 390评论 2 0