Loading [MathJax]/jax/output/CommonHTML/config.js
前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
圈层
工具
发布
首页
学习
活动
专区
圈层
工具
MCP广场
社区首页 >专栏 >Web Components 初探

Web Components 初探

作者头像
QQ音乐前端团队
发布于 2018-10-27 04:54:04
发布于 2018-10-27 04:54:04
2.7K00
代码可运行
举报
运行总次数:0
代码可运行

原文

任何UI框架或库最期望目标之一是帮助我们建立通用的模式或约定。这些约定使UI代码易于共享并为其提供理论基础。很长一段时间,每个框架或库都有自己的实现或UI组件版本。仅在该组件的生态中,这些组件可以实现代码复用。如果给定的UI组件/插件需要在不同的技术依赖中使用,往往由于特定的生态系统限制而成为局限。 例如,如果我编写一个Angular库并想在我的Vue应用程序中使用我的Angular下拉列表,目前还无法直接做到。如果有一个基于任意JavaScript库或框架编写的通用标准组件可以在任何浏览器中用,那就很好了!这是Web Components的愿景。

Web Components是标准化的底层浏览器API的集合,方便我们创建共享的可重用UI组件。在这次介绍中,我们通过构建一个Web组件的例子来介绍其中的一些API。我们的组件是一个简单的计数器组件,当有单击操作时实现数字加减。

为了制作我们的计数器组件,将介绍三种不同的API,Custom Elements,Templates 和Shadow DOM API。首先介绍Custom Elements API。

Custom Elements

Custom Elements API允许我们自定义HTML节点并添加行为给这些节点。使用Custom Elements API,我们还可以扩展原生HTML节点。首先,让我们创建一个最小的自定义节点。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
class XCounter extends HTMLElement {    constructor() {        super();    }    connectedCallback() {        this.innerHTML = `            <p>Hello From Web Component</p>        `;    }}customElements.define('x-counter', XCounter);   

上面我们用最少的代码量来创建我们的第一个自定义节点。我们继承HTMLElement类来创建自定义节点。在我们的自定义类中,可以定义模板和我们想要的任何行为。在特定生命周期的钩子函数connectedCallback()中,我们将模板赋值给节点的innerHTML属性。构造函数执行完且节点插入DOM之后才会调用connectedCallback()方法。

在我们定义Custom Element类之后,我们需要注册该节点。 customElements.define('x-counter',XCounter);注册节点时,我们传递两个参数。类引用和我们将在HTML中引用的节点的名称。在命名我们的节点时,名称中必须至少有一个破折号。Custom Elements 命名规定至少需要一个破折号,以防止命名与现有HTML节点发生冲突。

此时我们已经创建了Web Component 的基本结构,下面通过添加一个完整的模板来创建我们的计数器组件。

Template and Shadow DOM APIs

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const template = document.createElement('template');template.innerHTML = `        <style>            button, p {            display: inline-block;            }        </style>        <button aria-label="decrement">-</button>            <p>0</p>        <button aria-label="increment">+</button>    `;class XCounter extends HTMLElement {    constructor() {        super();        this.root = this.attachShadow({ mode: 'open' });        this.root.appendChild(template.content.cloneNode(true));    }}customElements.define('x-counter', XCounter);

Template和Shadow DOM API将允许我们创建高度封装的、高性能的组件。首先我们声明一个新的template节点。使用document.createElement('template');我们可以定义一个独立的原生HTML template。创建template 让我们不必立即将其插入DOM就可以构建HTML节点树。通过使用template,我们可以做到只创建一次template,然后在每次创建组件实例时重复使用它。

我们通过Shadow DOM API而不是之前的innerHTML示例添加我们新创建的template。通过Shadow DOM API将模板添加到我们的组件,我们在构造函数中添加如下代码。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
this.root = this.attachShadow({ mode: 'open' });this.root.appendChild(template.content.cloneNode(true));

在节点的构造函数中创建Shadow DOM,从而无需使用connectedCallback钩子函数来创建。 Shadow DOM为我们的组件创造一个高度封装的且隔离的DOM环境。 Shadow DOM会保护我们的HTML不被全局CSS或外部JavaScript污染。例如,在我们上面的模板中,我们有以下CSS:

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
button, p {    display: inline-block;}

在我们的Shadow DOM template中定义样式时,我们的Web组件中的按钮和段落标记将使用内联样式进行设置。样式不会全局泄露,全局CSS样式也不会覆盖我们的template。现在我们已经设置并创建了template,我们需要在按钮上添加一些click事件处理。

Properties

为了与Web Components通信,我们主要通过组件上定义的公共属性来进行数据传递。对于我们的组件,我们将创建一个公共属性value 。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const template = document.createElement('template');template.innerHTML = `<style>    button, p {    display: inline-block;    }</style><button aria-label="decrement">-</button>    <p>0</p><button aria-label="increment">+</button>`;class XCounter extends HTMLElement {    set value(value) {        this._value = value;        this.valueElement.innerText = this._value;    }    get value() {        return this._value;    }    constructor() {        super();        this._value = 0;        this.root = this.attachShadow({ mode: 'open' });        this.root.appendChild(template.content.cloneNode(true));        this.valueElement = this.root.querySelector('p');        this.incrementButton = this.root.querySelectorAll('button')[1];        this.decrementButton = this.root.querySelectorAll('button')[0];        this.incrementButton            .addEventListener('click', (e) => this.value++);        this.decrementButton            .addEventListener('click', (e) => this.value--);    }}customElements.define('x-counter', XCounter);

在我们的组件上,我们为value属性创建了一个get和set方法。使用getter和setter,我们可以触发对template的更新。我们有一个私有的变量value来保留计数器值。在我们的setter中,我们使用下面的代码来更新数值:this.valueElement.innerText = this.value;。

使用我们的公共的 value getter方法,我们可以动态设置计数器的值。例如,我们获得对节点的引用,像其他HTML节点一样与它进行交互。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<x-counter></x-counter>
代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import 'counter.js';const counter = document.querySelector('x-counter');counter.value = 10;

如您所见,我们可以查询自定义节点并像设置任何其他原生HTML节点一样设置自定义属性。使用我们的组件,我们可以通过输入属性将数据传递给它,但是如果我们希望组件在用户更改计数器值时通知我们怎么办?接下来,我们将介绍自定义事件。

Events

就像任何HTML节点一样,我们的自定义节点可以发出自定义事件供我们监听。在我们例子中,我们想知道用户何时更新了计数器组件的值。我们来看看组件值的更新。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const template = document.createElement('template');template.innerHTML = `<style>    button, p {    display: inline-block;    }</style><button aria-label="decrement">-</button>    <p>0</p><button aria-label="increment">+</button>`;    class XCounter extends HTMLElement {    set value(value) {        this._value = value;        this.valueElement.innerText = this._value;        // trigger our custom event 'valueChange'        this.dispatchEvent(new CustomEvent('valueChange', { detail: this._value }));    }    get value() {        return this._value;    }    constructor() {        super();        this._value = 0;        this.root = this.attachShadow({ mode: 'open' });        this.root.appendChild(template.content.cloneNode(true));        this.valueElement = this.root.querySelector('p');        this.incrementButton = this.root.querySelectorAll('button')[1];        this.decrementButton = this.root.querySelectorAll('button')[0];        this.incrementButton            .addEventListener('click', (e) => this.value++);        this.decrementButton            .addEventListener('click', (e) => this.value--);    }}customElements.define('x-counter', XCounter);

使用我们的组件,我们在value属性的setter方法中添加一个自定义事件。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
this.dispatchEvent(new CustomEvent('valueChange', { detail: this._value }));

我们可以发送自定义事件。自定义事件类有两个参数。第一个参数是事件的名称;第二个参数是我们想要传回的数据。通常会传递包含已更改数据detail属性的对象。当我们的自定义事件发出时,我们能够监听事件,同时获取事件值以及节点触发事件的详细信息。为了监听事件,我们可以像标准HTML节点一样创建事件监听器。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
import 'counter.js';const counter = document.querySelector('x-counter');counter.value = 10;counter.addEventListener('valueChange', v => console.log(v));

在我们的代码中,我们可以监听自定义valueChange事件,在这里我们记录该值。

Attributes

有时,通过特性而不是属性将信息传递给组件也很方便。例如,我们可能想要传递一个初始值给我们的计数器。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
<x-counter value="5"></x-counter>

为此,我们需要为组件添加一些额外的代码。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
const template = document.createElement('template');template.innerHTML = `<style>    button, p {    display: inline-block;    }</style><button aria-label="decrement">-</button>    <p>0</p><button aria-label="increment">+</button>`;    class XCounter extends HTMLElement {    // Attributes we care about getting values from.    static get observedAttributes() {        return ['value'];    }    set value(value) {        this._value = value;        this.valueElement.innerText = this._value;        this.dispatchEvent(new CustomEvent('valueChange', { detail: this._value }));    }    get value() {        return this._value;    }    constructor() {        super();        this._value = 0;        this.root = this.attachShadow({ mode: 'open' });        this.root.appendChild(template.content.cloneNode(true));        this.valueElement = this.root.querySelector('p');        this.incrementButton = this.root.querySelectorAll('button')[1];        this.decrementButton = this.root.querySelectorAll('button')[0];        this.incrementButton            .addEventListener('click', (e) => this.value++);        this.decrementButton            .addEventListener('click', (e) => this.value--);    }    // Lifecycle hook called when a observed attribute changes    attributeChangedCallback(attrName, oldValue, newValue) {        if (attrName === 'value') {            this.value = parseInt(newValue, 10);        }    }}customElements.define('x-counter', XCounter);

在我们的组件上,我们必须添加两个部分。首先是我们希望在更改时收到通知的特性列表。这是Web Components所需的性能优化。

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
static get observedAttributes() {    return ['value'];}

在我们的组件上,我们还添加了一个生命周期中新的钩子函数。当HTML特性有更新时,就会调用attributeChangedCallback()

代码语言:javascript
代码运行次数:0
运行
AI代码解释
复制
attributeChangedCallback(attrName, oldValue, newValue) {    if (attrName === 'value') {        this.value = parseInt(newValue, 10);    }}

对于Web通信最佳实践,最好使用自定义属性而不是自定义特性。属性更灵活,可以处理复杂的数据类型,如对象或数组。使用属性时,因为HTML的限制所有值都被当做String类型。自定义特性虽然很有用,但始终从属性开始,并根据需要添加特性。如果使用Web Component创作工具(如StencilJS),该工具会自动连接属性中的特性并使其保持同步。

总结

使用Web Components,我们可以创建可重用的Web UI组件库。Web components在Chrome,Safari中已经支持,很快Firefox便会支持。通过polyfill,我们还可以支持Edge(现在正在实现Web Component API)和IE11。大多数现代JavaScript框架也支持Web Components,您可以在custom-elements-everywhere.com上看到完全支持。

想要深入了解有关Web Components的更多信息?查看我的早期发行书Web Component Essentials!

本文参与 腾讯云自媒体同步曝光计划,分享自微信公众号。
原始发表:2018-10-27,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 QQ音乐前端团队 微信公众号,前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体同步曝光计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
暂无评论
推荐阅读
编辑精选文章
换一批
Spring Boot 统一接口响应格式的正确姿势
熟悉 web 系统开发的同学可能比较熟悉,目前绝大多数的互联网软件平台基本都是前后端分离的开发模式,为了加快前后端接口对接速度,一套完善且规范的接口标准格式是非常有必要的,不仅能够提升开发效率,也会让代码看起来更加简洁、好维护。
星辰大海的精灵
2024/08/06
2390
Spring mvc中统一对ResponseBody进行封装
在一个前后端分离的项目中,需要对后端RestController里返回的ResponseBody进行统一的封装,让所有的API结果的都是json对象,带有是否成功的标志位,并且将实际的数据放到json的result字段中,例如:
大神带我来搬砖
2018/12/07
1.2K0
【SpringBoot】Spring 一站式解决方案:融合统一返回结果、异常处理与适配器模式
适配器模式, 也叫包装器模式. 将⼀个类的接⼝,转换成客⼾期望的另⼀个接⼝, 适配器让原本接⼝不兼容的类可以合作⽆间.
用户11288949
2025/01/17
1700
【SpringBoot】Spring 一站式解决方案:融合统一返回结果、异常处理与适配器模式
优美的统一返回结果处理
我们写项目一般都会自己写一个Result对象,然后去处理,但是有一个问题,逐渐的接口写多了之后
用户10136162
2022/11/15
7150
优美的统一返回结果处理
SpringBoot中统一API返回格式的两种方式
微服务中,由于各业务团队之间的对接,各个团队之间需要统一返回格式,这样解析时不容易出现错误。因此,有必要统一返回格式。下面我说下项目中常见的两种统一和变更返回值格式的方式
zhaozhen
2023/02/18
8580
springboot对返回值作统一处理方式
在使用springboot的使用,我们更加多的方式是返回json数据,直接返回,如下(比如返回一个对象):
用户7741497
2022/08/01
2.2K0
SpringBoot中如何参数校验、统一异常、统一响应以及自定义注解
@NotBlank:只能作用在String上 不能为null并且调用trim()后长度必须大于0
才疏学浅的木子
2022/11/13
4860
SpringBoot中如何参数校验、统一异常、统一响应以及自定义注解
SpringBoot 定义优雅全局统一 Restful API 响应和统一异常处理,太优雅了!
假如你作为项目组长,为 Spring Boot 项目设计一个规范的统一的RESTfulAPI 响应框架。
码哥字节
2024/11/23
3640
SpringBoot 定义优雅全局统一 Restful API 响应和统一异常处理,太优雅了!
Spring MVC ControllerAdvice深入解析
  Spring 在3.2版本后面增加了一个ControllerAdvice注解。网上的资料说的都是ControllerAdvice配合ExceptionHandler注解可以统一处理异常。而Spring MVC是如何做到的资料却比较少,下面会先给出使用的例子和踩过的一个坑。然后进行相应的源码分析,之后再介始ControllerAdvice另外的两种使用方式。
良辰美景TT
2018/09/11
1.5K0
拿去用,接口统一返回值,最简单的一种实现
上一篇中介绍了 RequestBodyAdvice 接口,可以对@RequestBody 进行增强,本文介绍另外一个相似的接口:ResponseBodyAdvice,这个可以对@ResponseBody 进行增强,可以拦截@ResponseBody 标注的方法的返回值,对返回值进行统一处理,比如进行加密、包装等操作;比如通过他可以实现统一的返回值。
路人甲Java
2021/10/20
1.3K0
统一定制API返回格式,我只做了这几件事
转自:blog.csdn.net/qq_34347620/article/details/102239179
猿天地
2020/09/22
2.2K0
Spring Boot 无侵入式 实现 API 接口统一 JSON 格式返回
无侵入式 统一返回JSON格式 其实本没有没打算写这篇博客的,但还是要写一下写这篇博客的起因是因为,现在呆着的这家公司居然没有统一的API返回格式?,询问主管他居然告诉我用HTTP状态码就够用了(fx
架构师修炼
2020/11/19
1.2K0
spring 使用ResponseBodyAdvice处理响应负载
spring中,使用@ResponseBody注解controller方法,或者返回ResponseEntity对象,返回的数据将通过HttpMessageConverter转换后写入响应的body区域
路过君
2021/10/15
1.1K0
[Spring cloud 一步步实现广告系统] 4. 通用代码模块设计
我们在annotation包下面添加一个注解com.sxzhongf.ad.common.annotation.IgnoreResponseAdvice,用它来标柱是否需要支持上面的统一返回拦截。
Isaac Zhang
2019/09/10
1.1K0
[Spring cloud 一步步实现广告系统] 4. 通用代码模块设计
每天用SpringBoot,还不懂RESTful API返回统一数据格式是怎么实现的?
有童鞋说,我们项目都做了这种处理,就是在每个 API 都单独工具类将返回值进行封装,但这种不够优雅;我想写最少的代码完成这件事,也许有童鞋说,加几个注解就解决问题了,说的没错,但这篇文章主要是为了说明为什么加了几个注解就解决问题了,目的是希望大家知其所以然。
用户1516716
2019/09/11
1.7K0
每天用SpringBoot,还不懂RESTful API返回统一数据格式是怎么实现的?
每天用SpringBoot,还不懂RESTful API返回统一数据格式是怎么实现的?
有童鞋说,我们项目都做了这种处理,就是在每个 API 都单独工具类将返回值进行封装,但这种不够优雅;我想写最少的代码完成这件事,也许有童鞋说,加几个注解就解决问题了,说的没错,但这篇文章主要是为了说明为什么加了几个注解就解决问题了,目的是希望大家知其所以然。
用户4172423
2019/09/04
1.1K0
每天用SpringBoot,还不懂RESTful API返回统一数据格式是怎么实现的?
基于 RequestBodyAdvice 与 ResponseBodyAdvice 实现统一加密和解密
在日常开发中,有时候经常需要和第三方接口打交道,有时候是我方调用别人的第三方接口,有时候是别人在调用我方的第三方接口,那么为了调用接口的安全性,一般都会对传输的数据进行加密操作,如果每个接口都由我们自己去手动加密和解密,那么工作量太大而且代码冗余。那么有没有简单的方法,借助 spring 提供的 RequestBodyAdvice 和 ResponseBodyAdvice 可以实现解密和加密操作。
BUG弄潮儿
2021/06/25
1.7K0
基于 RequestBodyAdvice 与 ResponseBodyAdvice 实现统一加密和解密
Controller层代码这么写,简洁又优雅!
点击上方“芋道源码”,选择“设为星标” 管她前浪,还是后浪? 能浪的浪,才是好浪! 每天 10:33 更新文章,每天掉亿点点头发... 源码精品专栏 原创 | Java 2021 超神之路,很肝~ 中文详细注释的开源项目 RPC 框架 Dubbo 源码解析 网络应用框架 Netty 源码解析 消息中间件 RocketMQ 源码解析 数据库中间件 Sharding-JDBC 和 MyCAT 源码解析 作业调度中间件 Elastic-Job 源码解析 分布式事务中间件 TCC-Transaction
芋道源码
2022/08/29
3780
Controller层代码这么写,简洁又优雅!
【JavaEE进阶】拦截器与统一功能处理
如果要在以上 Spring AOP的切面中实现用户登录权限效验的功能,有以下两个问题:
xxxflower
2023/10/16
2940
【JavaEE进阶】拦截器与统一功能处理
SpringBoot统一接口返回和全局异常处理
项目中最常见到的是封装一个工具类,类中定义需要返回的字段信息,把需要返回前端的接口信息,通过该类进行封装,这样就可以解决返回格式不统一的现象了。
用户7741497
2022/02/28
6870
推荐阅读
相关推荐
Spring Boot 统一接口响应格式的正确姿势
更多 >
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档
本文部分代码块支持一键运行,欢迎体验
本文部分代码块支持一键运行,欢迎体验