jiechenjiechen

The world is quiet here.

© jiechen

Rebuild in 2023   |   Start in 2021
Total View 0 Site Visitors 0

初识WebComponent

2023年2月16日front-end1988 字约 13 分钟

背景

组件化 已经成为目前主流的前端开发模式,其可复用性这一大特点是一众复制粘贴工程师的福音。目前我们实现组件化主要是依托于各大框架如 VueReactAngular 。这些框架基本都是在遵从浏览器的规则下制定出自己的一套开发规则和书写语法使开发者的项目获得组件化的能力

随着近年来组件化框架的盛行,官方也推行了一套组件化的解决方案和原生API上的支持 —— Web Component

Web Component 是什么

Web Components 是一系列加入 w3cHTMLDOM 的特性,使得开发者可以创建可复用的组件

由于 web components 是由 w3c 组织去推动的,因此它很有可能在不久的将来成为浏览器的一个标配。

关键字:原生、定制化标签

使用 Web Component 编写的组件是脱离框架的,换言之,也就是说使用 Web Component 开发的组件库,是适配所有框架的,不会像 Antd 这样需要对 VueReact 等框架出不同的版本

使用 Web Component

Web Component 核心技术

  • Custom elements(自定义元素):一组 JavaScript API,允许您定义 custom elements 及其行为,然后可以在您的用户界面中按照需要使用它们
  • Shadow DOM(影子DOM):一组 JavaScript API,用于将封装的 “影子” DOM 树 附加到元素(与主文档DOM分开呈现)并控制其关联的功能。通过这种方式,您可以保持元素的功能私有,这样它们就可以被脚本化和样式化,而不用担心与文档的其他部分发生冲突
  • HTML templates(HTML模板)< template >< slot > 元素使您可以编写不在呈现页面中显示的标记模板。然后它们可以作为自定义元素结构的基础被多次重用
  • HTML Imports(HTML导入):一旦定义了 自定义组件,最简单的重用它的方法就是使其定义细节保存在一个单独的文件中,然后使用导入机制将其导入到想要实际使用它的页面中。 HTML 导入就是这样一种机制,尽管存在争议 — Mozilla 根本不同意这种方法,并打算在将来实现更合适的

实现一个简单的组件

  1. 定义自定组件:
    1
    2
    3
    4
    5
    6
    7
    8
    class MyButton extends HTMLElement {
    constructor () {
    super();
    const template = document.getElementById('mybutton');
    const content = template.content.cloneNode(true);
    this.appendChild(content);
    }
    }

(神似 react)

  1. 定义组件模板:

    1
    2
    3
    <template id="mybutton">
    <button>Add</button>
    </template>
  2. 注册组件:

    1
    window.customElements.define('my-button', MyButton);
  3. 使用组件:

    1
    2
    3
    <body>
    <my-button></my-button>
    </body>

这样, 一个简单的 Web Component 就完成了。

生命周期

和一般框架中的组件一样,Web Component 的组件为了支持更多场景的应用也是有生命周期的。

常用的生命周期方法如下:

  • connectedCallback
    web component 被添加到 DOM 时,会调用这个回调函数,这个函数只会被执行一次。可以在这个回调函数中完成一些初始化操作,比如更加参数设置组件的样式。
  • disconnectedCallback
    web component 从文档 DOM 中删除时执行。
  • adoptedCallback
    web component 被移动到新文档时执行。
  • attributeChangedCallback
    被监听的属性发生变化时执行

与React的结合

就像刚刚所使用的,看起来 WebComponent 和 React 很想,但实际上二者是互补的关系

React中使用的API都是声明式的,react封装了对DOM的操作并做了一定的优化; 而WebComponent中则是命令式的,它的方法都是基于原生DOM进行操作的(要不然咋说它是原生组件技术,doge)

react官方也有说明,详情参考:Web Components – React

在 Web Component 中使用 React

1
2
3
4
5
6
7
8
9
10
11
class XSearch extends HTMLElement {
connectedCallback() {
const mountPoint = document.createElement('span');
this.attachShadow({ mode: 'open' }).appendChild(mountPoint);

const name = this.getAttribute('name');
const url = 'https://www.google.com/search?q=' + encodeURIComponent(name);
ReactDOM.render(<a href={url}>{name}</a>, mountPoint);
}
}
customElements.define('x-search', XSearch);

在 React 中使用 Web Component

1
2
3
4
5
class HelloMessage extends React.Component {
render() {
return <div>Hello <x-search>{this.props.name}</x-search>!</div>;
}
}

Web Components 的组件 video 可能会公开 play()pause() 方法。要访问 Web Components 的命令式 API,你需要使用 ref 直接与 DOM 节点进行交互
如果你使用的是第三方 Web Components,那么最好的解决方案是编写 React 组件包装该 Web Components

Web Components 触发的事件可能无法通过 React 渲染树正确的传递。 你需要在 React 组件中手动添加事件处理器来处理这些事件。

demo演示

https://p3-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/63928cab8ae54cb5be882ec5eb1dc757~tplv-k3u1fbpfcp-zoom-in-crop-mark:4536:0:0:0.awebp

如果我们想要实现原生组件复用,就需要把代码写在一个js文件里面,引入该js文件,就等于引入了组件。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
//index.html
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta http-equiv="X-UA-Compatible" content="IE=edge" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
</head>
//引入编写好的组件,在这里引入文件,注意要添加defer关键字
<script src="./MyList/index.js" defer></script>
<body>
<div>
//使用组件
<my-list id="node">
<!--原生支持插槽 -->
<slot>web component</slot>
</my-list>
</div>
<script>
//因为是原生,所以我们需要获取dom节点行后续操作
const node = document.getElementById("node");
//我们将变量转换一下格式,就能传递给子组件
node.dataset.arr = JSON.stringify(["吃饭", "睡觉"]);
</script>
</body>
</html>
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
//index.js
const template = document.createElement("template");
//在js文件中,我们想要书写html和css就必须要借助innerHTML,在其内部书写我们的样式和结构
template.innerHTML = `
<style>
#contain {
display: flex;
flex-direction: column
}
input {
width: 200px
}
</style>
<div id="contain">
<span><slot></slot></span>
<div>
<input type="text" id=input>
<button id="mybutton" data-text1="111111">添加</button>
</div>
</div>
`;
class MyList extends HTMLElement {
constructor() {
//因为我们的组件继承于HTMLElement,所以需要调用super关键字
super();
// 获取标签
const content = template.content.cloneNode(true);
const mybutton = content.getElementById("mybutton");
const input = content.getElementById("input");
const contain = content.getElementById("contain");

// 获取props
const arr = JSON.parse(this.dataset.arr);
//进行事件的监听
mybutton.addEventListener("click", () => {
arr.push(input.value)
const li = document.createElement("li");
li.innerText = input.value;
contain.appendChild(li);
});
// 将数据渲染到页面
arr.forEach((item) => {
const li = document.createElement("li");
li.innerText = item;
contain.appendChild(li);
});
//初始化一个影子dom
this.attachShadow({ mode: "closed" }).appendChild(content);
}
}
// 注册组件
window.customElements.define("my-list", MyList);

相应框架

从上面的案例看的出来这种原生dom操作的开发效率还是太低,这里再推荐一个WebComponent的封装框架:Stencil

有人就疑惑了,WebComponent不是强调不依赖vue、react等框架吗?

是的,它是不依赖vue、react等框架,但并不表示他不能像js拥有jQuery一样,拥有自己的封装库。
封装出来的语法题和强依赖的运行环境,二者的关系需要弄清楚

框架使用示例

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
import { Component, Prop, h } from '@stencil/core';

@Component({
tag: 'my-component', // the name of the component's custom HTML tag
styleUrl: 'my-component.css', // css styles to apply to the component
shadow: true, // this component uses the ShadowDOM
})
export class MyComponent {
// The component accepts two arguments:
@Prop() first: string;
@Prop() last: string;

//The following HTML is rendered when our component is used
render() {
return (
<div>
Hello, my name is {this.first} {this.last}
</div>
);
}
}

使用

1
<my-component first="Stencil" last="JS"></my-component>

参考

Web Component | MDN

Web Component入门

Stencil