相关文章推荐
坚强的大蒜  ·  js ...·  5 月前    · 

谷歌公司由于掌握了 Chrome 浏览器,一直在推动浏览器的原生组件,即 Web Components API 。相比第三方框架,原生组件简单直接,符合直觉,不用加载任何外部模块,代码量小。目前,它还在不断发展,但已经可用于生产环境。

Web Components API 内容很多,本文不是全面的教程,只是一个简单演示,让大家看一下怎么用它开发组件。

一、自定义元素

下图是一个用户卡片。

本文演示如何把这个卡片,写成 Web Components 组件,这里是最后的 完整代码

网页只要插入下面的代码,就会显示用户卡片。

<user-card></user-card>

这种自定义的 HTML 标签,称为自定义元素(custom element)。根据规范,自定义元素的名称必须包含连词线,用与区别原生的 HTML 元素。所以, <user-card> 不能写成 <usercard>

二、 customElements.define()

自定义元素需要使用 JavaScript 定义一个类,所有 <user-card> 都会是这个类的实例。

class UserCard extends HTMLElement { constructor() { super();

上面代码中, UserCard 就是自定义元素的类。注意,这个类的父类是 HTMLElement ,因此继承了 HTML 元素的特性。

接着,使用浏览器原生的 customElements.define() 方法,告诉浏览器 <user-card> 元素与这个类关联。

window.customElements.define('user-card', UserCard);

三、自定义元素的内容

自定义元素 <user-card> 目前还是空的,下面在类里面给出这个元素的内容。

class UserCard extends HTMLElement { constructor() { super(); var image = document.createElement('img'); image.src = 'https://semantic-ui.com/images/avatar2/large/kristy.png'; image.classList.add('image'); var container = document.createElement('div'); container.classList.add('container'); var name = document.createElement('p'); name.classList.add('name'); name.innerText = 'User Name'; var email = document.createElement('p'); email.classList.add('email'); email.innerText = ' [email protected] '; var button = document.createElement('button'); button.classList.add('button'); button.innerText = 'Follow'; container.append(name, email, button); this.append(image, container);

上面代码最后一行, this.append() this 表示自定义元素实例。

完成这一步以后,自定义元素内部的 DOM 结构就已经生成了。

四、 <template> 标签

使用 JavaScript 写上一节的 DOM 结构很麻烦,Web Components API 提供了 <template> 标签,可以在它里面使用 HTML 定义 DOM。

<template id="userCardTemplate"> <img src="https://semantic-ui.com/images/avatar2/large/kristy.png" class="image"> <div class="container"> <p class="name">User Name</p> <p class="email"> [email protected] </p> <button class="button">Follow</button> </template>

然后,改写一下自定义元素的类,为自定义元素加载 <template>

class UserCard extends HTMLElement { constructor() { super(); var templateElem = document.getElementById('userCardTemplate'); var content = templateElem.content.cloneNode(true); this.appendChild(content);

上面代码中,获取 <template> 节点以后,克隆了它的所有子元素,这是因为可能有多个自定义元素的实例,这个模板还要留给其他实例使用,所以不能直接移动它的子元素。

到这一步为止,完整的代码如下。

<user-card></user-card> <template>...</template> <script> class UserCard extends HTMLElement { constructor() { super(); var templateElem = document.getElementById('userCardTemplate'); var content = templateElem.content.cloneNode(true); this.appendChild(content); window.customElements.define('user-card', UserCard); </script> </body>

五、添加样式

自定义元素还没有样式,可以给它指定全局样式,比如下面这样。

user-card { /* ... */

但是,组件的样式应该与代码封装在一起,只对自定义元素生效,不影响外部的全局样式。所以,可以把样式写在 <template> 里面。

<template id="userCardTemplate"> <style> :host { display: flex; align-items: center; width: 450px; height: 180px; background-color: #d4d4d4; border: 1px solid #d5d5d5; box-shadow: 1px 1px 5px rgba(0, 0, 0, 0.1); border-radius: 3px; overflow: hidden; padding: 10px; box-sizing: border-box; font-family: 'Poppins', sans-serif; .image { flex: 0 0 auto; width: 160px; height: 160px; vertical-align: middle; border-radius: 5px; .container { box-sizing: border-box; padding: 20px; height: 160px; .container > .name { font-size: 20px; font-weight: 600; line-height: 1; margin: 0; margin-bottom: 5px; .container > .email { font-size: 12px; opacity: 0.75; line-height: 1; margin: 0; margin-bottom: 15px; .container > .button { padding: 10px 25px; font-size: 12px; border-radius: 5px; text-transform: uppercase; </style> <img src="https://semantic-ui.com/images/avatar2/large/kristy.png" class="image"> <div class="container"> <p class="name">User Name</p> <p class="email"> [email protected] </p> <button class="button">Follow</button> </template>

上面代码中, <template> 样式里面的 :host 伪类,指代自定义元素本身。

六、自定义元素的参数

<user-card> 内容现在是在 <template> 里面设定的,为了方便使用,把它改成参数。

<user-card image="https://semantic-ui.com/images/avatar2/large/kristy.png" name="User Name" email=" [email protected] " ></user-card>

<template> 代码也相应改造。

<template id="userCardTemplate"> <style>...</style> <img class="image"> <div class="container"> <p class="name"></p> <p class="email"></p> <button class="button">Follow John</button> </template>

最后,改一下类的代码,把参数加到自定义元素里面。

class UserCard extends HTMLElement { constructor() { super(); var templateElem = document.getElementById('userCardTemplate'); var content = templateElem.content.cloneNode(true); content.querySelector('img').setAttribute('src', this.getAttribute('image')); content.querySelector('.container>.name').innerText = this.getAttribute('name'); content.querySelector('.container>.email').innerText = this.getAttribute('email'); this.appendChild(content); window.customElements.define('user-card', UserCard);

七、Shadow DOM

我们不希望用户能够看到 <user-card> 的内部代码,Web Component 允许内部代码隐藏起来,这叫做 Shadow DOM,即这部分 DOM 默认与外部 DOM 隔离,内部任何代码都无法影响外部。

自定义元素的 this.attachShadow() 方法开启 Shadow DOM,详见下面的代码。

class UserCard extends HTMLElement { constructor() { super(); var shadow = this.attachShadow( { mode: 'closed' } ); var templateElem = document.getElementById('userCardTemplate'); var content = templateElem.content.cloneNode(true); content.querySelector('img').setAttribute('src', this.getAttribute('image')); content.querySelector('.container>.name').innerText = this.getAttribute('name'); content.querySelector('.container>.email').innerText = this.getAttribute('email'); shadow.appendChild(content); window.customElements.define('user-card', UserCard);

上面代码中, this.attachShadow() 方法的参数 { mode: 'closed' } ,表示 Shadow DOM 是封闭的,不允许外部访问。

至此,这个 Web Component 组件就完成了,完整代码可以访问 这里 。可以看到,整个过程还是很简单的,不像第三方框架那样有复杂的 API。

八、组件的扩展

在前面的基础上,可以对组件进行扩展。

(1)与用户互动

用户卡片是一个静态组件,如果要与用户互动,也很简单,就是在类里面监听各种事件。

this.$button = shadow.querySelector('button'); this.$button.addEventListener('click', () => { // do something

(2)组件的封装

上面的例子中, <template> 与网页代码放在一起,其实可以用脚本把 <template> 注入网页。这样的话,JavaScript 脚本跟 <template> 就能封装成一个 JS 文件,成为独立的组件文件。网页只要加载这个脚本,就能使用 <user-card> 组件。

这里就不展开了,更多 Web Components 的高级用法,可以接着学习下面两篇文章。

  • Web Components Tutorial for Beginners
  • Custom Elements v1: Reusable Web Components

    用过google的polymer来写web-component,是一个基于openlayers的地图组件,要求是高度自治,在vue和react都可以用。
    写小组件还不错,但是项目的依赖管理和打包挺麻烦,官方的脚手架不支持webpack。组件功能一复杂就很头疼。
    遗憾的是polymer已经很久没有更新过了。

    slot 也挺有用(常用)的啊,这个你应该提一下。
    我觉得 HTML 可以直接写在一个变量里(可以用模板引用可变的参数),然后直接用 innerHTML 而不是每次都要用 template。
    现在 Edge 好像还不支持(更别说IE),所以要排除或提示用户用Chrome(内核)浏览器了。

    这个让我有理由不去花时间精力学习 Vue/React 了(很不喜欢 Webpack 的麻烦ness)
    (我的个人网址就实践了Web Component,应该会离不开了)

    五、添加样式 这一节用了:host,:host只在template作为影子元素时生效
    参考https://developer.mozilla.org/zh-CN/docs/Web/CSS/:host

    在成为影子元素前没有局部的概念,想要定义容器的样式需要按照往常的方式定义。

    在js文件前面加上如下代码:
    let target_div = document.createElement("div");
    target_div.innerHTML = ''; //引号内填写的HTML代码
    document.body.append(target_div);

    let target_template = document.createElement("template");
    target_template.setAttribute("id", "userCardTemplate");
    target_template.innerHTML = `
    <style>
    //内容太长了,自己去复制样式
    </style>

    <img class="image">
    <div class="container">
    <p class="name"></p>
    <p class="email"></p>
    <button class="button">Follow John</button>
    `; //引号内填写的HTML代码
    document.body.append(target_template);

    class UserCard extends HTMLElement {
    //太长了,自己复制内容
    window.customElements.define("user-card", UserCard);

    然后index.html里使用:
    <user-card
    image="https://semantic-ui.com/images/avatar2/large/kristy.png"
    name="User Name"
    email="
    [email protected] "
    ></user-card>
    <script src="components/userCard.js"></script>
    </body>

    就可以了,正常显示~

    template 是模板 —— 模板是提前准备好的,可以用来无限复用。
    你也可以用js createElement 动态生成template, 很多时候没有直接引用template 方便,但是更灵活。

    template 模板不用保存在html,工程化时应该放到单独的组件中

  •