Custom Elements 实现输入框组件(Input)

面试官:小伙子,你的代码为什么这么丝滑?

前言

在实际项目中,很多时候都需要用到一些自定义的组件。此时,我们要么自己手写一遍,要么使用第三方的 UI 库,然而前者会浪费很多时间,而后者的样式定制又会非常麻烦,特别是在多人开发的产品中,更多的时候我们需要自己实现一个轻量级的组件库。

而自定义组件一般涉及到 DOM 操作,这里就推荐一种前端技术——Custom Elements,是的,新时代的 Web 组件标准来了。

什么是 Custom Elements

Custom Elements 是由浏览器原生支持的一门技术,它使得开发者可以定义自己的 HTML 元素,扩展现有的 HTML 元素,并能够像使用标准 HTML 元素一样使用它们。换句话说,Custom Elements 是一种让我们可以自定义 HTML element 的技术。

Custom Elements 叧可以创建一些内置 Web API 和浏览器 UI 部件,并能够实现一些自定义的行为。

实现 Input 组件

定义 Input 元素

在定义 Input 元素之前,我们需要先导入Web Components 的主要 API:CustomElementRegistryHTMLElementShadowRoot

----- - --------------- ------------ ---------- - - -------

CustomElementRegistry 中有两个关键方法:

  • define(name, constructor):用来定义一个新的自定义元素,并返回一个构造函数;
  • get(name):用来获取当前文档和上下文中配置的自定义元素命名的元素构造函数。

下面,我们就来定义一个名为 InputElement 的自定义元素。

----- ------------ ------- ----------- -
  ------------- -
    -------- ----------
  -
-
--------------------------------- --------------

上面的代码中,我们继承了 HTMLElement 构造函数,并且用 customElements.define 方法注册了 wc-input。在这一步,定义 Input 元素已经完成了。

添加 Shadow DOM

接下来,让我们为 InputElement 添加 Shadow DOMShadow DOM 是一种独立的 DOM 树结构,它从主 DOM 树中抽象出来,可以用于封装自定义组件的内部实现。

----- ------------ ------- ----------- -
  ------------- -
    -------- ----------

    ----- ---------- - ------------------- ----- ------ ---
    -- -- ------ --- ---
    ----- ------- - ------------------------------
    ----------------------------- -----------
    --------------------------------
    -- -- ----- --
    ----- ----- - --------------------------------
    --------------------------- ----------------
    ---------------------------
  -
-
--------------------------------- --------------

在上面的代码中,我们通过 attachShadow 方法将 Shadow DOM 树附加到 InputElement 上,然后创建一个名为 wrapper 的 div 节点,并设置属性 class 为 wrapper。接着,我们通过 appendChild 方法将它添加到 Shadow DOM 树上。

然后,我们在 wrapper 节点内部再次创建一个名为 input 的元素,并设置属性 class 为 custom-input,然后再将其添加到 wrapper 内部,并在屏幕上显示出来。

添加 CSS 样式

既然我们已经定义了一个 InputElement 元素,我们就需要为它添加一些样式。

-------- -
  -------- -----
  ------------ -------
  ----------------- ------- -- -- ------
  -------- ----
  -------------- ----
-
------------- -
  ------- -----
  ------ -----
  -------- -----
  ---------- -----
-

我们为 Wrapper 节点设置了一些 CSS 样式,如 display: flex 使得它成为了一个 flex box,align-items: center让它的子元素垂直居中,background-colorpaddingborder-radius 为它的样式提供了一些装饰性质,其中 border-radius 是为了更好的 UI 设计。

同时,我们也在 Input 元素添加了一些自定义的样式。

添加事件

现在我们已经实现了一个基本的 Input 组件,但是,我们在实际项目开发中必须考虑到组件的可用性。这时,我们就可以为 Input 元素添加事件来提高其互动性。

在本案例中,我为 Input 元素添加了两个事件:focusblur。当用户进入输入框时,将会触发 focus 事件;当用户离开输入框时,则会触发 blur 事件。

----- ------------ ------- ----------- -
  ------------- -
    -------- ----------
 
    ----- ---------- - ------------------- ----- ------ ---
    -- -- ------ --- ---
    ----- ------- - ------------------------------
    ----------------------------- -----------
    --------------------------------
    -- -- ----- --
    ----- ----- - --------------------------------
    --------------------------- ----------------
    ------------------------------- -- -- -
      ------------------ --------
    ---
    ------------------------------ -- -- -
      ------------------ -------
    ---
    ---------------------------
  -
-
--------------------------------- --------------

在代码中,我们先添加了两个事件的监听器,然后我们在输入框中使用 console.log 输出事件的触发。

属性数据绑定

Custom Elements 还支持自定义属性,我们可以使用自定义属性来绑定数据,通过属性的更改去改变组件状态,最后从当前输入框获取相应的数据。

----- ------------ ------- ----------- -
  ------------- -
    --------

    ----- ---------- - ------------------- ----- ------ ---
    ----- ------- - ------------------------------
    ----------------------------- -----------
    --------------------------------

    ----- ----- - --------------------------------
    --------------------------- ----------------

    ------------------------------- -- -- -
      ---------------------------- ----
    ---
    ------------------------------ -- -- -
      --------------------------------
    ---

    ---------------------------
  -
  ------ --- -------------------- -
    ------ ----------
  -
  
  ------------------------------ --------- --------- -
    -- ----- --- -------- -
      ----- ----- - ---------------------------------------
      ----------- - ---------
    -
  -
  --- ------- -
    ----- ----- - ---------------------------------------
    ------ ------------
  -
  --- ------------ -
    -------------------------- -------
  -
-
--------------------------------- --------------

代码中可以看到,为了绑定数据,我们:

  • 使用this.setAttribute方法设置名为value的自定义属性;

  • 通过事件attributeChangedCallback监听到value属性的变更,并在其中将输入框的值更新为属性的值;

  • 使用this.getAttribute获取属性值。

通过这种方式,我们可以将数据从当前产生 Input 的组件地址传递,并使其与原始组件同步属性数据。

使用 Input 组件

到这里,我们已经完成了一个简单的 Custom Elements Input 组件,现在我们现在就来简单使用一下定义好的自定义组件:

--------- -----
------
------
  ----- ----------------
  ------------- -------- -------------
  -------
    ---- -
      -------- -----
      ---------------- -------
      ------------ -------
      ------- ------
    -
  --------
-------
------
  --------- -------------------------
-------
-------

以上是通过自定义标签 wc-input 来使用 Custom Elements 实现的浏览器兼容性,目前除 IE 外,其它现代浏览器都支持 Custom Elements。

总结

最终,自定义元素依然是标准的 DOM 对象,遵守 web 标准,他可以处理复杂逻辑、样式和 DOM 集成的详细细节。并且 Custom Elements 在浏览器的响应式设计、跨设备布局等应用场景,均具备不同的价值。 Custom Elements 底层 API 的机制使其更加自由。自定义元素可以注册为现有 element 的子元素,其他元素也可以充当 shadow DOM 中的自定义元素的宿主。

当然,这篇文章中的 Input 组件只是做为一个示例来说明 Custom Elements 的基础用法,底层原理还很复杂,如果让你开发一个真正的组件库以及项目,还需要更多丰富的实践和经验积累。

来源:JavaScript中文网 ,转载请联系管理员! 本文地址:https://www.javascriptcn.com/post/64a7a8c448841e98944253bb


猜你喜欢

  • Fastify 入门:安装和使用指南

    Fastify 是 Node.js 上一个快速且低延迟的 Web 框架,主要用于构建高效的 Restful API。与其他 Node.js 框架相比,Fastify 的性能更加强大,支持异步编程,具有...

    21 小时前
  • 怎样在 Ubuntu 系统中安装 PM2

    PM2是一款开源的Node.js进程管理器,可以帮助我们更方便的管理我们的Node.js应用程序,减少Node.js应用程序宕机的可能性。本文将详细介绍如何在Ubuntu系统上安装PM2,并提供相关示...

    21 小时前
  • React 测试工具 Enzyme:一个入门指南

    在前端开发中,我们经常需要对 React 组件进行测试来确保其行为与预期一致。而 Enzyme 就是一款广受欢迎的 React 测试工具,它提供了强大的 API 来帮助我们进行组件测试。

    21 小时前
  • 解析 ES12 引入的数字分隔符

    随着 JavaScript 语言的不断发展,ES12 (即 ECMAScript 2021)引入了新的语法:数字分隔符。这个新的语法允许使用下划线(_)来分隔数字,使得数字更加易读和易于理解。

    21 小时前
  • 如何在 PWA 中添加自定义的缓存策略?

    随着移动设备的普及和网络技术的不断进步,越来越多的应用程序开始采用 Progressive Web App(PWA)技术。PWA可以将网站应用程序转化为离线可访问、快速响应和具有本地化功能的应用程序,...

    21 小时前
  • 如何使用 Babel 将代码转换成 ES2015

    在前端开发中,越来越多的开发者开始将目光投向了新一代的 JavaScript 语言规范 ES2015(也叫作 ECMAScript 6)。ES2015 引入了许多新特性和语言功能,可以让我们写出更加简...

    21 小时前
  • Express.js 实现登陆验证教程

    随着互联网的发展,越来越多的网站和应用需要实现用户注册和登录功能。这些功能对于网站和应用的安全性和用户体验至关重要。本文将介绍如何使用 Express.js 实现登陆验证功能,让你的网站和应用更加安全...

    21 小时前
  • Redux 中如何使用中间件?

    什么是 Redux 中间件? Redux 中间件是 Redux 应用程序中的可插入功能,它允许在派发和处理 Redux Action 之间添加其他操作和逻辑。中间件可以处理异步操作、调用 API,添加...

    21 小时前
  • 使用 Jest 测试插件——vue-test-utils

    Vue.js 是一个流行的前端 JavaScript 框架,可以帮助我们构建交互式的 web 应用程序。在开发过程中,测试是一个重要的步骤,有助于确保我们的代码在不同情况下的正确性。

    21 小时前
  • Sequelize 够用吗?ORM 到底是不是好东西?

    随着前端技术的不断发展,越来越多的开发者开始使用 ORM(对象关系映射)框架来管理数据库。其中,Sequelize 是一个广泛使用的 ORM 框架之一,它可以让你使用 JavaScript 来操作数据...

    21 小时前
  • PostgreSQL 10 的新功能和性能优化

    PostgreSQL 是一款功能强大、可扩展性好的关系型数据库,被广泛地用于各种 web 应用和企业应用中。在最新的 PostgreSQL 10 版本中,新增了一些重要的新功能和性能优化,本文将详细介...

    21 小时前
  • 如何使用 Socket.io 和 MongoDB 实现聊天室?

    前端技术的发展让实时聊天变得越来越普遍,我们可以通过使用 Socket.io 和 MongoDB 实现一个简单的实时聊天室。在本文中,我们将介绍如何使用这两个技术来实现聊天室。

    21 小时前
  • 详解:Dockerfile 中 ADD 与 COPY 的区别

    详解:Dockerfile 中 ADD 与 COPY 的区别 在 Dockerfile 文件中,ADD 和 COPY 都是用于将文件从本地复制到 Docker 镜像中。然而,它们有着不同的用法和作用。

    21 小时前
  • Kubernetes 中 RBAC 实现权限控制的方法及注意事项

    在 Kubernetes 中,RBAC 是一种用于授权用户访问 API 资源的方法。通过 RBAC,用户可以设置不同的访问权限,以便于控制 Kubernetes 集群中各种资源的访问情况。

    21 小时前
  • 解决 CSS Flexbox 实现横向滚动条的问题

    在开发 Web 应用时,经常需要在页面中实现横向滚动条,使得页面内容能够轮廓展示。实现过程中,CSS Flexbox 布局经常被使用。然而,在使用 Flexbox 实现横向滚动条时,往往会遇到一些问题...

    21 小时前
  • 常见错误解决方案 - Express.js 使用

    Express.js 是 Node.js 最流行的 Web 应用程序框架,它提供了简单而强大的 API 来构建 Web 应用。然而,即使是最流行的框架也只是一种工具,使用不当或者疏忽都容易出现问题。

    1 天前
  • Cypress 自动化测试实战:端到端测试篇

    Cypress 是一个现代化的端到端测试工具,它是专门为现代 Web 应用程序打造的。Cypress 拥有丰富的 API,易于使用和学习,同时提供了一个交互式的测试运行器和强大的调试工具。

    1 天前
  • 如何使用 ES11 中的 Promise.allSettled 方法实现批量异步请求

    如何使用 ES11 中的 Promise.allSettled 方法实现批量异步请求 在前端开发中,经常需要发送多个异步请求,这时候我们可以使用 Promise.all 方法来处理,但是如果其中一个请...

    1 天前
  • Next.js HMR 原理解析

    在前端开发中,HMR(热模块替换)已经成为了一个常见的开发技术,可以大大提高开发效率和代码质量。Next.js 是一个流行的 React 服务端渲染框架,它也提供了 HMR 的支持。

    1 天前
  • Redux 中如何处理持久化数据?

    Redux 是一个流行的 JavaScript 应用程序状态管理库,但是我们如何在 Redux 中处理持久化数据?在本文中,我们将探讨一些在 Redux 中处理持久化数据的方法,并提供一些示例代码和最...

    1 天前