Web Components入门不完全指北

目前流行的各类前端框架,不管是react, angular还是vue,都有一个共同点,那就是支持组件化开发,但事实上随着浏览器的发展,现在浏览器也原生支持组件式开发,本文将通过介绍Web Components 的三个主要概念,带读者无障碍入门Web Components

1.Custom Elements(自定义组件)

顾名思义,Web Component 提供了一套模式让我们自定义组件,采用自定义类的方式,当我们需要自定义一个组件,我们需要声明一个类,并继承自原生的DOM 类,下面先给出一个示例

class MyCustomElement extends HTMLElement {constructor() {super();}connectedCallback() {console.log("自定义元素添加至页面。");}disconnectedCallback() {console.log("自定义元素从页面中移除。");}adoptedCallback() {console.log("自定义元素移动至新页面。");}attributeChangedCallback(name, oldValue, newValue) {console.log(`属性 ${name} 已变更。`);}
}customElements.define("my-custom-element", MyCustomElement);

如上面的代码所示,我们定义了一个最简单的类,这个类继承自HTMLElement,它有四个发方法,分别代表组件的四个生命周期,接着我们需要调用customElements.define这个方法来注册一个自定义组件,这个方法包含三个参数,第一个参数为自定义组件的名字,第二个参数我们需要传入刚刚定义的类,第三个参数是一个可选参数,后文会提及,这里可以先忽略,这样我们就定义个一个最简单的组件,这个组件没有任何功能没有任何形态,但是会有生命周期,我们可以像下面这样调用它

<my-custom-element></my-custom-element>

页面渲染之后我们可以看到控制台打印出下面的文本

接下来简单讲讲四个生命周期,第1,2,4这三个生命周期比较好理解,分别代表组件被添加到页面,从页面移除,以及组件参数的变化,对于组件参数,下文会提及,这里先有个概念就可以了,第三个生命周期adoptedCallback 相对来说比较难理解,移动到新页面的意思是从一个html文档移动到另一个html文档,一般情况下如果同一个组件,从普通页面移动到iframe里面,会触发这个事件,有兴趣的小伙伴可以自己尝试一下。自此我们简单的了解了如何创建一个最简单的自定义组件,接下来我们来了解一下Web Components 的核心,Shadow DOM。

2.Shadow DOM(影子dom)

Shadow DOM 这个名字直译过来就是影子DOM,大部分人看到这个名字可能会联想到react或者vue的虚拟dom,但这两者基本上不是一个东西,虚拟dom是用js对dom结构做一个映射,用js对象来表示dom结构,用来提升性能的一个东西,而影子dom反而可以理解为一个实实在在的dom,但提供了样式,dom结构隔离等功能的东西,为了方便理解,我们来看看下面的html代码

<!DOCTYPE html><html><head><meta charset="utf-8"/><title>web components</title></head><body><input type="range"/><input/></body>
</html>

我们可以看到下面的页面

有没有考虑过一个问题,同样是input控件,为什么可以展现出不同的功能和样式?当然肯定有人会说是浏览器根据标签传入的参数渲染的结果,这么说也没毛病,但其实我们可以仔细窥探一下这里面的实现原理,首先我们打开一下浏览器的一下开关,路径为 设置/performance/Elements, 把下面这个开关打开

打开之后我们会发现,dom结构发生了很大的变化,我们看到了一些原本被隐藏起来的东西

shadow-root 底下的dom结构明明白白的告诉你这个组件是怎么被构建的,包括样式结构等等,可以这么理解,这部分就是原生组件的shadow dom,而Web Components把这部分功能开放出来,我们也可以利用shadow dom来构建自己的自定义组件,当然构建自定义组件不一定需要用到shadow dom, 只是用上它,能让组件更好的内聚,避免被其它组件影响,shadow dom 提供了一个隔离机制,可以使得组件内部的dom结构和style都仅作用于组件内部,并且不受外部的影响,外部也无法访问到组件内部的东西,从而避免全局样式之间的互相影响,不同组件之间你甚至可以使用相同类名而不用添加hash之类的东西用于区分

接下来我们开看看如何用shadow dom 创建一个简单的组件,我们在第一个组件里面添加一些内容

//components/MyCustomElement.jsclass MyCustomElement extends HTMLElement {constructor() {super();}connectedCallback() {console.log("自定义元素添加至页面。");const shadow = this.attachShadow({ mode: "open" });const sheet = new CSSStyleSheet();sheet.replaceSync(".wrapper {color:red;}");const wrapper = document.createElement("span");wrapper.innerHTML = this.getAttribute("content");wrapper.setAttribute("class", "wrapper");shadow.adoptedStyleSheets = [sheet];shadow.appendChild(wrapper);const style = document.createElement("style");style.textContent = `.wrapper{color:red;}`;}customElements.define("my-custom-element", MyCustomElement);
<!-- index.html --><my-custom-element content="wrapper1"></my-custom-element>

我们选择在connectedCallback 添加了一些代码,目的是在自定义组件添加至页面之后,我们通过shadow dom 给它内部添加一个span并指定样式,下面我们简单解释一下代码

shadow dom不能脱离宿主单独存在,所以我们在一开始通过调用attachShadow 方法给当前组件附加影子dom,这个方法支持传入一个mode参数,这个参数可以设置为“open”或者“closed”, open代表开放shadow dom, 虽然我们无法直接访问到shadow dom里面的元素,但还是可以通过特定途径访问得到,closed则代表对外部彻底关闭(一个前提,这些都是基于君子协议,并非可以真正封闭组件,通过浏览器插件等方式依旧可以访问到内部的内容,这并不是一个严格的安全措施)

添加样式的代码比较简单,就不进行解释,接下来我们通过getAttribute方法来获取组件的content属性,并将它作为span的内容,我们可以看到渲染出了文字,并添加了样式

现在我们在html上添加一些对比内容来展示一下shadow dom对于样式跟dom的隔离效果

<!-- index.html -->
<!DOCTYPE html><html><head><meta charset="utf-8"/><title>web components</title><link href="index.css" rel="stylesheet"/></head><body><h1>web components</h1><my-custom-element content="wrapper1" id="myCustomElement"></my-custom-element><span class="wrapper">wrapper2</span><button id="test">get dom from shadow dom</button><script src="./components/MyCustomElement.js" defer></script><script src="index.js"></script></body>
</html>
/* index.css */
.wrapper{color:blue;
}
//index.js
window.onload = () => {document.getElementById("test").addEventListener("click",() => {var wrapper = document.querySelectorAll(".wrapper");if (wrapper) {console.log("1");console.log(wrapper);console.log(wrapper[0].innerHTML)}//通过宿主的shadowRoot来获取内部domwrapper = document.getElementById("myCustomElement").shadowRoot.querySelectorAll(".wrapper");if (wrapper) {console.log("2");console.log(wrapper);console.log(wrapper[0].innerHTML)}},false);
};

运行代码之后我们看到是下图的效果

红色的是组件内部定义的颜色,蓝色是组件外定义的全局样式,两者之间没有相互影响,样式之间做到了互相隔离,事实上不管我们以上么样的方式引入样式,组件的样式都不会影响到其他的组件。我们再尝试点击按钮,我们可以看到打印了下面的结果

我们发现,如果用普通的查询手段我们是无法查询到自定义组件内部的dom节点的,但我们可以通过宿主的shadowRoot来获取内部的dom,但这是有一个前提的,就是前文提到的,我们把mode参数设置成了open, 如果我们把mode修改成closed, 输出则会变成这样,我们无法访问到shadowRoot

shadowRoot 返回了null,因此我们的代码报错了 。

接下来,我们把第一段里面的坑填回去,我们现在已经知道我们可以通过Web Components来创建自定义组件了,但这对于日常的开发是远远不够的,我们有时候会有这样的需求,我们需要基于原有的组件进行二次开发,也就是说我们想继承原生组件的功能,但我们想新增一些功能上去,这样改怎么做呢,答案是回到customElements.define,我们添加第三个参数,把需要继承的原生组件名字赋值给extends,类似下面,就是自定义一个继承自textarea的组件

customElements.define("my-text-area", MyTextArea, { extends: "textarea" });

下面我们举个简单的例子,实现一个很常见的需求,我们将实现一个自定义的textarea, 下方显示能接受的最大字数和当前字数

//components/MyTextArea.js
class MyTextArea extends HTMLTextAreaElement {constructor() {super();}connectedCallback() {console.log("自定义元素添加至页面2。");var div = document.createElement("div");div.innerText = `0/${this.getAttribute("maxlength")}`;this.parentNode.insertBefore(div, this.nextSibling);this.addEventListener("input",(e) => {div.innerText = `${e.currentTarget.value.length}/${this.getAttribute("maxlength")}`;},false);}
}customElements.define("my-text-area", MyTextArea, { extends: "textarea" });
<!-- index.html -->
<!DOCTYPE html><html><head><meta charset="utf-8"/><title>web components</title><link href="index.css" rel="stylesheet"/></head><body><h1>web components</h1><textarea is="my-text-area" maxlength="20"></textarea><textarea is="my-text-area" maxlength="50"></textarea><script src="./components/MyTextArea.js" defer></script><script src="index.js"></script></body>
</html>

通过上述代码,我们可以看到运行的效果如下

 我们通过继承textarea组件,可以获取原生组件的所有功能,并能在这个功能的基础上扩展出自己的功能,上面我们做的事情很简单,在自定义组件被加载之后,我们通getAttribute函数获取当前传入的maxlength来获取允许输入的最大值,然后监听input事件,实时获取当前的字数,实现过程比较简单,这里就不再赘述了

3.Template and Slot(模板与插槽)

其实到了这里,我们已经基本上能够利用上述的功能来实现我们业务需求了,但还不够灵活,例如我们日常开发过程中经常会有这样的需求,在自定义组件下面给他添加子组件,就像普通的div一样,可以层层嵌套,类似这样:

<my-element><div><div>let's have some differences</div></div>
</my-element>

基于但不仅仅基于这种需求,我们引入另外两个概念

3.1 Template(模板)

模板,顾名思义,我们可以在html文件里面定义一系列可重用的模板,这些模板除非最终被使用,否则不会渲染到页面中来,我们看看下面的代码

<!-- index.html -->
<!DOCTYPE html><template id="my-paragraph"><p>my paragrapy</p>
</template><html><head><meta charset="utf-8"/><title>web components</title></head><body><h1>web components</h1><my-slot-element></my-slot-element><script src="./components/MySlotElement.js" defer></script></body>
</html>
//components/MySlotElement.jsclass MySlotElement extends HTMLElement {constructor() {super();}connectedCallback() {console.log("自定义元素添加至页面3。");let template = document.getElementById("my-paragraph");let templateContent = template.content;const shadowRoot = this.attachShadow({ mode: "open" });shadowRoot.appendChild(templateContent.cloneNode(true));}
}customElements.define("my-slot-element", MySlotElement);

 上述代码实现的效果如下

上述的代码通过id获取到当前的template,通过template.content我们可以获取到当前模板的内容,然后我们我么们通过cloneNode方法把template的内容复制一份添加到影子DOM里面,注意我们传入了一个参数true,表示递归复制此节点下的所有子孙节点

当然,如果只是仅仅实现这样一个html代码重用的效果,template可能显得不是很有必要,毕竟我们有很多的方法来实现代码的重用,之所以把它拎出来讲,是因为template实现了插槽(slot)的功能

3.2 Slot(插槽)

Slot给template模板提供了自定义扩展的功能,我们可以在特定的位置先埋下插槽,在后续使用模板的过程中,我们可以把特定的代码插入到对应的插槽当中,我们来对上面的代码做一些扩展,我们修改一下template的定义,实现插槽的功能

<!DOCTYPE html><template id="my-paragraph"><p>my paragrapy</p><slot name="my-text">my defalut text</slot>
</template><html><head><meta charset="utf-8"/><title>web components</title><link href="index.css" rel="stylesheet"/></head><body><h1>web components</h1><my-slot-element><div slot="my-text"><div>let's have some differences</div></div></my-slot-element><script src="./components/MySlotElement.js" defer></script></body>
</html>

 实现的效果如下

我们在template里面预先埋入了一个叫my-text的插槽,然后我们在自定义组件的子节点编写了自定义的内容,并且指定了slot的名称,我们发现渲染之后,子节点被渲染到于P标签同一个层级的dom结构中,注意,如果我们在template中删除slot标签,自定义组件的子节点是不会被渲染出来的,而如果我们没有传入自定义内容到指定的slot中,slot中的默认内容将会被展示,这里由于篇幅原因,就不展示了,有兴趣的话可以自行测试

看到这里写过vue的小伙伴应该会觉得很熟悉,这个写法几乎如出一辙,事实上vue设计slot的时候是借鉴了Web Components的slot的(没错,Web Component出来的时间比vue早)

4.兼容性

接下来就是喜闻乐见的兼容性程度了,这关乎这个技术能否直接应用到生产当中,截至到本篇文章发布,目前浏览器对主要的特性支持如下

除了个别api没被支持之外,大部分的api都已经被主流的浏览器所支持,当然了,哪里有不兼容,哪里就会有polyfills, 例如polyfills可以对大部分不兼容的api提供支持,各位可以自行探索

以上,希望各位封装组件愉快,有什么疑问见解或者想交流的都可以评论区见

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/224562.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

vue场景 无分页列表条件过滤,子组件多选来自父组件的列表

日常开发中&#xff0c;经常会遇到下面场景&#xff1a; 页面加载一个无分页列表&#xff0c;同时工具栏设置多个条件可对列表过滤的场景(典型的就是关键字模糊查询)父组件传给子组件列表&#xff0c;子组件中需要多选列表多选&#xff0c;选择结果返回父组件 1 无分页列表过…

电子电器架构刷写方案——General Flash Bootloader

电子电器架构刷写方案——General Flash Bootloader 我是穿拖鞋的汉子&#xff0c;魔都中坚持长期主义的汽车电子工程师。 注&#xff1a;文章1万字左右&#xff0c;深度思考者入&#xff01;&#xff01;&#xff01; 老规矩&#xff0c;分享一段喜欢的文字&#xff0c;避免…

CentOS 7 Tomcat服务的安装

前提 安装ava https://blog.csdn.net/qq_36940806/article/details/134945175?spm1001.2014.3001.5501 1. 下载 wget https://mirrors.tuna.tsinghua.edu.cn/apache/tomcat/tomcat-9/v9.0.84/bin/apache-tomcat-9.0.84.tar.gzps: 可选择自己需要的版本下载安装https://mirr…

mysql原理--基于成本的优化

1.什么是成本 我们之前老说 MySQL 执行一个查询可以有不同的执行方案&#xff0c;它会选择其中成本最低&#xff0c;或者说代价最低的那种方案去真正的执行查询。不过我们之前对 成本 的描述是非常模糊的&#xff0c;其实在 MySQL 中一条查询语句的执行成本是由下边这两个方面组…

Kali Linux—借助 SET+MSF 进行网络钓鱼、生成木马、获主机shell、权限提升、远程监控、钓鱼邮件等完整渗透测试(一)

社会工程学—世界头号黑客凯文米特尼克在《欺骗的艺术》中曾提到&#xff0c;这是一种通过对受害者心理弱点、本能反应、好奇心、信任、贪婪等心理陷阱进行诸如欺骗、伤害等危害手段。 SET最常用的攻击方法有&#xff1a;用恶意附件对目标进行 E-mail 钓鱼攻击、Java Applet攻…

Unity之DOTweenPath轨迹移动

Unity之DOTweenPath轨迹移动 一、介绍 DOTweenPath二、操作说明1、Scene View Commands2、INfo3、Tween Options4、Path Tween Options5、Path Editor Options&#xff1a;轨迹编辑参数&#xff0c;就不介绍了6、ResetPath&#xff1a;重置轨迹7、Events&#xff1a;8、WayPoin…

Liteos移植_STM32_HAL库

0 开发环境 STM32CubeMX(HAL库)keil 5正点原子探索者STM32F4ZET6LiteOS-develop分支 1 STM32CubeMX创建工程 如果有自己的工程&#xff0c;直接从LiteOS源码获取开始 关于STM32CubeMX的安装&#xff0c;看我另一篇博客STM32CubeMX安装 工程配置 创建新工程 选择芯片【STM32F…

ElasticSearch入门介绍和实战

目录 1.ElasticSearch简介 1.1 ElasticSearch&#xff08;简称ES&#xff09; 1.2 ElasticSearch与Lucene的关系 1.3 哪些公司在使用Elasticsearch 1.4 ES vs Solr比较 1.4.1 ES vs Solr 检索速度 2. Lucene全文检索框架 2.1 什么是全文检索 2.2 分词原理之倒排索引…

JavaScript进阶(事件+获取元素+操作元素)

目录 事件基础 事件组成 执行事件的步骤 获取元素 根据ID获取元素 根据标签名获取元素 获取ol中的小li 类选择器&#xff08;html5新增的I9以上支持&#xff09; 获取body和html 操作元素 innerText和innerHtml 表单标签 样式属性操作 操作元素总结 事件基础 事…

设计模式--桥接模式

实验9&#xff1a;桥接模式 本次实验属于模仿型实验&#xff0c;通过本次实验学生将掌握以下内容&#xff1a; 1、理解桥接模式的动机&#xff0c;掌握该模式的结构&#xff1b; 2、能够利用桥接模式解决实际问题。 [实验任务]&#xff1a;两个维度的桥接模式 用桥接模式…

MySql的mvcc原理

目录 一、什么是mvcc? 二、什么是当前读,快照读? 当前读 快照读 三、mvcc实现原理 版本链 undo日志 Undo log 的用途 Read View(读视图) Read View几个属性 五、RR、RC级别下生成时机 一、什么是mvcc? mvcc全称Multi-Version Concurrency Control&#xff0c;即…

华为HCIA认证H12-811题库新增

801、[单选题]178/832、在系统视图下键入什么命令可以切换到用户视图? A quit B souter C system-view D user-view 试题答案&#xff1a;A 试题解析&#xff1a;在系统视图下键入quit命令退出到用户视图。因此答案选A。 802、[单选题]“网络管理员在三层交换机上创建了V…

17个常用经典数据可视化图表与冷门图表

数据可视化是创建信息图形表示的过程。随着可视化技术的飞速发展&#xff0c;可以利用强大的可视化工具选择合适的数据可视化图表来展示数据。以下专业人士都应该知道的一些最重要的数据可视化图表。 常见数据可视化图表 饼图 饼图是最常见和最基本的数据可视化图表之一。饼图…

虚拟机的下载、安装(模拟出服务器)

下载 vmware workstation&#xff08;收费的虚拟机&#xff09; 下载vbox 网址&#xff1a;Oracle VM VirtualBox&#xff08;免费的虚拟机&#xff09; 以下选择一个下载即可&#xff0c;建议下载vbox&#xff0c;因为是免费的。安装的时候默认下一步即可&#xff08;路径最好…

elasticsearch-py 8.x的一些优势

​ 早在 2022 年 2 月,当 Elasticsearch 8.0 发布时,Python 客户端也发布了 8.0 版本。它是对 7.x 客户端的部分重写,并带有许多不错的功能(如下所述),但也带有弃用警告和重大更改。今天,客户端的 7.17 版本仍然相对流行,每月下载量超过 100 万次,占 8.x 下载量的 ~50…

了解OAuth 2.0以及社交登录认证授权流程

1.前言 目前在写一个电商项目&#xff0c;可以通过手机号进行注册登录&#xff0c;为了方便用户使用本平台的系统&#xff0c;引入社交登录功能&#xff0c;这里使用的是gittee。 2.OAuth 2.0介绍 当谈到网络安全和身份验证时&#xff0c;OAuth 2.0&#xff08;开放授权 2.0&a…

迪文屏开发保姆级教程5—表盘时钟和文本RTC显示

这篇文章要讲啥事呢&#xff1f; 本篇文章主要介绍了在DGBUS平台上使用表盘时钟和文本时钟RTC显示功能的方法。 文哥悄悄话&#xff1a; 官方开发指南PDF&#xff1a;&#xff08;不方便下载的私聊我发给你&#xff09; https://download.csdn.net/download/qq_21370051/8864…

微服务实战系列之Dubbo(上)

前言 随着一年一度冬至的到来&#xff0c;2023的步伐也将远去。而博主的系列文章&#xff0c;也将从今天起&#xff0c;越来越聚焦如何构建微服务“内核”上。前序系列文章几乎囊括了微服务的方方面面&#xff0c;无论使用什么框架、组件或工具&#xff0c;皆可拿来用之。 那么…

Docker介绍、常用命令与操作

Docker介绍、常用命令与操作 学习前言为什么要学习DockerDocker里的必要基础概念常用命令与操作1、基础操作a、查看docker相关信息b、启动或者关闭docker 2、容器操作a、启动一个镜像i、后台运行ii、前台运行 b、容器运行情况查看c、日志查看d、容器删除 3、镜像操作a、镜像拉取…

“一篇长文教你进行全方位的使用appium“

随着移动应用的日益普及&#xff0c;移动应用的测试成为了软件开发的重要组成部分。Python&#xff0c;作为一种易于学习&#xff0c;功能强大的编程语言&#xff0c;特别适合进行这种测试。本文将详细介绍如何使用Python进行APP测试&#xff0c;并附带一个实例。 Python 和 A…