2023-02-21 22:34:08 +08:00

14 KiB
Raw Blame History

Exparser设计原理

本章主要包含以下内容:

  • WebComponent原理
  • Custom Element原理
  • ShadowDOM思想
  • Exparser原理

什么是WebComponent?

WebComponent 汉语直译过来第一感觉是web组件的意思但是它只是一套规则、一套API。你可以通过这些API创建自定义的新的组件并且组件是可以重复使用的封装好的组件可以在网页和Web应用程序中进行使用。

当前的前端开发环境Vue、React等都基于组件化开发的形式但是他们的组件生态并不互通如果你有过两个框架的开发经验的话你应该知道最烦恼的就是两个框架的UI组件表现不一致的问题。

我们抽离组件为了提高复用率,提升开发效率。但是脱离了像Vue、React这样的框架后你会发现原生JS难道就不能开发自定义组件吗WebComponent就是为了解决这个问题。

换一个角度来说,并不是所有的业务场景都需要Vue\React这样的框架进行开发、也并是都需要工程化。很多业务场景我们需要原生JS、HTML。

言归正传,WebComponent实现的组件可以和HTML原生标签一起使用有了这个概念之后我们看一下它的具体表现形式是怎样的。

<body>
  <custom-component></custom-component>
</body>

上面我们看到<body>标签还是我们熟悉的标签,但是<custom-component>标签就是自定义组件的标签了它不属于html语义化标签中的任何一个是自定义的。

接下来我们从这个简单的DEMO入手WebComponent进行了解。首先就是三大规范:

  • Custom Elements规范
  • Template规范
  • Shadow DOM规范

MDNhttps://developer.mozilla.org/zh-CN/docs/Web/Web_Components

Custom Element

所谓自定义元素,即当内置元素无法为问题提供解决方案时,自己动手来创建一个自定义元素来解决,上方的<custom-component>就是我们手动创建的自定义元素。

元素的状态是指定义该元素(或者叫做升级该元素)时元素状态的改变,升级过程是异步的。 元素内部的状态有:

  • undefined 未升级即自定义元素还未被define。
  • failed 升级失败即define过了也实例化过了但失败了。会自动按HTMLUnknownElement类来实例化。
  • uncustomized 未定制化没有define过但却被实例化了会自动按HTMLUnknownElement类来实例化。
  • custom 升级成功define过并且实例化成功了。

接下来我们来看一个示例:

<custom-component></custom-component>
.custom-style{
  display: inline-block;
  padding: 15px;
  border: 1px solid red;
  border-radius: 4px;
  color: red;
}
// 定义自定义组件
class CustomComponent extends HTMLElement {
  constructor() {
    super();

    const box = document.createElement("div");
    box.className = "custom-style";

    const text = document.createElement("p");
    text.innerText = "这是一个自定义组件";

    box.appendChild(text);

    this.appendChild(box);
  }
}
window.customElements.define("custom-component", CustomComponent);

效果如下:

image-20230215094800286

首先可以看出需要有个类的概念。自定义元素类必须继承自window内置的HTMLElement类。

然后在constructor中定义类一些标记模版,定义模板后,执行this.appendChild,其中this指向了当前类实例。

最后将自定义组件挂载到customElements上,通过window.customElements.define方法。这个时候注意了,需要给自定义组件起一个名字,可以看到上面例子中我起的名字为custom-component。起名字是有规则的,规则如下:

  • 自定义元素的名称,必须包含短横线(-。它可以确保html解析器能够区分常规元素和自定义元素还能确保html标记的兼容性。
  • 自定义元素只能一次定义一个,一旦定义无法撤回。
  • 自定义元素不能单标记封闭。比如<custom-component />,必须写一对开闭标记。比如 <custom-component></custom-component>

对于自定义组件挂载的相关API

  • window.customElement.define('custom-component', CustomComponent, extendsInit) // 定义一个自定义元素
  • window.customElement.get('custom-component') // 返回已定义的自定义元素的构造函数
  • window.customElement.whenDefined('custom-component') // 接收一个promise对象是当定义自定义元素时返回的可监听元素状态变化但无法捕捉内部状态值。

其中window.customElement.whenDefined方法监听的元素状态为上述讲解的四种元素状态中的: failed升级失败和custom升级成功。

这里有个问题我们demo里的dom结构比较简单所以我们通过document.createElementappendChild方法进行构建还不算复杂如果dom结构很复杂的组件怎么办呢一顿使用createElement也不是个办法。这个时候我们就要引入<template>标记了。

Template

Web Components API 提供了<template>标签可以在它里面使用HTML定义DOM结构。这样的话我们改版一下我们的自定义组件

<custom-component></custom-component>

<template id="constomCompinentTemplate">
  <div class="custom-style">
    <p>这是一个自定义组件</p>
  </div>
</template>
// 定义自定义组件
class CustomComponent extends HTMLElement {
  constructor() {
    super();

    const template = document.getElementById("constomCompinentTemplate");
    const content = template.content.cloneNode(true);
    this.appendChild(content);
  }
}
window.customElements.define("custom-component", CustomComponent);

这里有两个需要考虑的地方:

  1. 这里因为是demo演示所以把<template>标签写在了一起,其实可以用脚本把<template>注入网页。这样的话JavaScript 脚本跟<template>就能封装成一个 JS 文件,成为独立的组件文件。网页只要加载这个脚本,就能使用<custom-component>组件。
  2. <template>标签内的节点进行操作必须通过template.content返回的节点来操作。因为这里获取的template并不是一个正常的DOM结构在控制台打印一下template.content得到的结果是#document-fragment。它其实是DocumentFragment节点,里面才是真正的结构。而且这个模板还要留给其他实例使用,所以不能直接移动它的子元素

在 Vue 和 React 中使用组件时,我们经常涉及到 props 的传递,例如:

<custom-component></custom-component>
<custom-component text="显示这个文本"></custom-component>

传入自定义的文本text如果有text内容那么就展示text如果没有那么展示默认值。

<template id="constomCompinentTemplate">
  <style>
    .custom-style {
      display: inline-block;
      padding: 15px;
      border: 1px solid red;
      border-radius: 4px;
      color: red;
    }
  </style>
  <div class="custom-style">
    <p class="component-text">这是一个自定义组件</p>
  </div>
</template>
// 定义自定义组件
class CustomComponent extends HTMLElement {
  constructor() {
    super();

    const template = document.getElementById("constomCompinentTemplate");
    const content = template.content.cloneNode(true);

    // 从 this 上获取 text 属性,如果有值就赋值给 content
    const textValue = this.getAttribute("text");
    if(textValue){
      content.querySelector(".component-text").innerHTML = textValue;
    }

    this.appendChild(content);
  }
}
window.customElements.define("custom-component", CustomComponent);

你看这样之后就可以传入参数了但是我们平常使用组件的时候是可以嵌套的我们不仅仅需要参数注入的形式还需要嵌套的children形式。继续修改自定义组件。

slot

WebComponent有一个slot概念插槽提供了一个“缺口”让给需要嵌套的dom。用法和Vue是比较相似的例如

<custom-component>
  <p slot="my-text">这是插入的内容!</p>
</custom-component>
<custom-component text="显示这个文本"></custom-component>
<p class="custom-style">这是一个测试</p>

<template id="constomCompinentTemplate">
  <style>
    .custom-style {
      display: inline-block;
      padding: 15px;
      border: 1px solid red;
      border-radius: 4px;
      color: red;
    }
  </style>
  <div class="custom-style">
    <p class="component-text">这是一个自定义组件</p>
    <slot name="my-text">插槽默认内容</slot>
  </div>
</template>
// 定义自定义组件
class CustomComponent extends HTMLElement {
  constructor() {
    super();
    const shadow = this.attachShadow({ mode: "closed" });

    const template = document.getElementById("constomCompinentTemplate");
    const content = template.content.cloneNode(true);

    // 从 this 上获取 text 属性,如果有值就赋值给 content
    const textValue = this.getAttribute("text");
    if (textValue) {
      content.querySelector(".component-text").innerHTML = textValue;
    }

    shadow.appendChild(content);
  }
}
window.customElements.define("custom-component", CustomComponent);

在上面的代码中,我们使用到了 slot 插槽,代码本身比较容易理解,但是注意我们这边还引入了一个新的东西,就是 shadow这也是 webcomponents 的三大特性之一shadow DOM中的结构是与外界隔离的外界是无法获取到内部dom的它可以理解为一颗单独的dom树隐藏的dom树。因此组件内部的样式也和外界完全隔离即使下面的 p 也使用了 custom-style 的类名。

有关shadow DOM将会在后面具体进行介绍。

事件

有了参数之后不能离开事件Event对吧我们想添加一个文本的点击事件。继续来改造升级。

// 定义自定义组件
class CustomComponent extends HTMLElement {
  constructor() {
    super();
    const shadow = this.attachShadow({ mode: "closed" });

    const template = document.getElementById("constomCompinentTemplate");
    const content = template.content.cloneNode(true);

    // 从 this 上获取 text 属性,如果有值就赋值给 content
    const textValue = this.getAttribute("text");
    const textDOM = content.querySelector(".component-text");
    if (textValue) {
      textDOM.innerHTML = textValue;
    }

    // 绑定事件
    textDOM.addEventListener("click", (e) => {
      e.stopPropagation();
      alert("Hello Web Components");
    });

    shadow.appendChild(content);
  }
}
window.customElements.define("custom-component", CustomComponent);

在上面的demo中我们为里面的 p 元素绑定了一个点击事件,并且使用了e.stopPropagation()方法阻止了事件冒泡。

这里有个知识点,自定义事件 new Event()options有几个参数可以设置冒泡行为方式其中就有关于Shadow DOM的。我们来看一下:

var options = {
  detail : {
    ...
  },
  composed: false, // Boolean 类型,默认值为 false指示事件是否会在 ShadowDOM 根节点之外触发侦听器
  bubbles: true, // Boolean 类型,默认值为 false表示该事件是否冒泡
  canceable: false // Boolean 类型,默认值为 false表示该事件是否能被取消
}
var myEvent = new CustomEvent(eventname, options);

Shadow DOM

Shadow DOM 允许将隐藏的 DOM 树附加到常规的 DOM 树中——它以 shadow root 节点为起始根节点,在这个根节点的下方,可以是任意元素,和普通的 DOM 元素一样。

image-20230215104608599

把本来DOM树中的一部分封装起来并且隐藏起来隐藏起来的树概念为Shadow Tree。把它理解成DOM上一棵特殊的子树称之为shadow tree或影子树。也是树但是很特殊树里面也是DOM就像我们上面用document.createElement创建的DOM一样。

影子树的根节点,我们称之为shadow root影子根

影子根的父节点,我称之为宿主shadow host

image-20230215105036312

在自定义元素中里面的结构已经变成了Shadow DOM。顺带说下attachShadow中的mode参数有两种值“open”、“closed”

  • open 表示可以通过页面内的 JavaScript 方法来获取 Shadow DOM例如使用 Element.shadowRoot 属性:
let myShadowDom = myCustomElem.shadowRoot;
  • closed 那么就不可以从外部获取Shadow DOM了myCustomElem.shadowRoot 将会返回 null

ShadowDOM的概念在HTML中非常常见举一个例子在 HTML 中有 Video 标签

<video 
 src="http://maoyan.meituan.net/movie/videos/854x4804c109134879943f4b24387adc040504b.mp4"
 controls
 width="500"
></video>

当我们使用该标签渲染一个视频的时候,会发现页面中会呈现出来一个完整的播放器,里面有播放视频的进度条、播放按钮、音量调节等。明明只有一个标签,为什么内部有如此丰富的内容呢?

image-20230215110726053

打开控制台查看结构时,看到的也仅仅是一个 video 标签而已我们可以打开控制台的【设置】勾选上【显示用户代理Shadow DOM】

image-20230215111137642

之后就可以看到在 video 中的 shadowDOM了

image-20230215111304144

因此像img、button、input、textarea、select、radio、checkboxvideo等等这些标签是不可以作为宿主的因为它们本身内部就已经有shadowDOM了。

Exparser框架原理

Exparser是微信小程序的组件组织框架内置在小程序基础库中为小程序提供各种各样的组件支撑。内置组件和自定义组件都有Exparser组织管理。

有关Exparser可参阅官网:https://developers.weixin.qq.com/ebook?action=get_post_info&docid=0000aac998c9b09b00863377251c0a


-EOF-