当前位置: 首页 > article >正文

Web Component 教程(五):从 Lit-html 到 LitElement,简化组件开发

前言

在现代前端开发中,Web 组件是一种非常流行的技术,它允许我们创建可重用的、自包含的 UI 元素。而 Lit-html 是一个简洁高效库,用于在 Web 组件中进行渲染。在这篇教程中,我们一步步学习如何 Lit-html 来创建 Web Component。

我们将从基础概念开始,逐步到高级功能和策略,使你能够创建高效、可展的前端应用。无论是初学者还是经验丰富的开发,这篇教程都将帮助你掌 Web Component和 Lit-html 的使用方法

什么是 Web 组件?

Web 组件是一套不同的技术,它们允许开发者创建自定义的、可重用的 HTML 元素。这些技术包括:

  1. Custom Elements:自定义元素。
  2. Shadow DOM:用于封装元素的样式和结构。
  3. HTML Templates:用于定义模板内容。

什么是 Lit-html?

Lit-html 是一个快速且高效的 HTML 模板库,它允许我们使用 JavaScript 标签模板字符串来编写 HTML 模板。它的主要优点包括:

  • 高性能:通过最小化 DOM 更新来提高性能。
  • 简洁:使用模板字符串语法,代码简洁易读。
  • 灵活:支持各种动态内容和条件渲染。

开始使用 Lit-html 和 Web 组件

首先,我们需要安装 lit-html 库。在项目的根目录下运行以下命令:

npm install lit-html

安装完成后,我们就可以开始编写 Web 组件了。下面是一个使用 lit-html 创建简单 Web 组件的示例:

  1. 创建一个新的 JavaScript 文件 my-element.js
import { html, render } from 'lit-html';class MyElement extends HTMLElement {constructor() {super();this.attachShadow({ mode: 'open' });}connectedCallback() {this.render();}render() {const template = html`<style>.container {padding: 16px;background-color: #f0f0f0;border-radius: 8px;text-align: center;}h1 {color: #333;}</style><div class="container"><h1>Hello, Lit-html!</h1><p>This is a simple Web component.</p></div>`;render(template, this.shadowRoot);}
}customElements.define('my-element', MyElement);

在这个示例中,我们创建了一个名为 MyElement 的自定义元素,并使用 lit-htmlhtml 函数来定义模板内容。通过 render 函数,我们将模板渲染到组件的 Shadow DOM 中。

  1. 在 HTML 文件中包含并使用这个自定义元素:
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Web Component with Lit-html</title>
</head>
<body><my-element></my-element><script type="module" src="./my-element.js"></script>
</body>
</html>

当我们打开这个 HTML 文件时,我们会看到一个样式化的盒子,显示 “Hello, Lit-html!” 和一段简单的文本。

添加动态内容

我们可以进一步扩展这个示例,添加一些动态内容。例如,我们可以添加一个按钮,通过点击按钮改变显示的文本:

import { html, render } from 'lit-html';class MyElement extends HTMLElement {constructor() {super();this.attachShadow({ mode: 'open' });this.counter = 0;}connectedCallback() {this.render();}increment() {this.counter += 1;this.render();}render() {const template = html`<style>.container {padding: 16px;background-color: #f0f0f0;border-radius: 8px;text-align: center;}h1 {color: #333;}button {padding: 8px 16px;font-size: 16px;margin-top: 16px;}</style><div class="container"><h1>Counter: ${this.counter}</h1><button @click="${() => this.increment()}">Increment</button></div>`;render(template, this.shadowRoot);}
}customElements.define('my-element', MyElement);

现在,当我们点击按钮时,计数器值会增加,并通过重新渲染模板来更新显示的值。

处理属性和事件

在 Web 组件中,我们经常需要处理属性和事件。接下来,我们会展示如何在 Lit-html 中处理这些情况。

属性绑定

我们可以通过使用 ${} 语法来动态绑定属性。例如,我们可以扩展上面的示例,让组件接收一个 title 属性,并根据这个属性来显示内容:

import { html, render } from 'lit-html';class MyElement extends HTMLElement {constructor() {super();this.attachShadow({ mode: 'open' });this.counter = 0;}static get observedAttributes() {return ['title'];}attributeChangedCallback(name, oldValue, newValue) {if (name === 'title') {this.title = newValue;this.render();}}connectedCallback() {this.render();}increment() {this.counter += 1;this.render();}render() {const template = html`<style>.container {padding: 16px;background-color: #f0f0f0;border-radius: 8px;text-align: center;}h1 {color: #333;}button {padding: 8px 16px;font-size: 16px;margin-top: 16px;}</style><div class="container"><h1>${this.title}</h1><p>Counter: ${this.counter}</p><button @click="${() => this.increment()}">Increment</button></div>`;render(template, this.shadowRoot);}
}customElements.define('my-element', MyElement);

在这个示例中,我们定义了 observedAttributes 静态方法来指定要观察的属性。当 title 属性发生变化时,会触发 attributeChangedCallback 方法,并更新组件的状态。

我们可以在 HTML 中传递 title 属性如下:

<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><meta name="viewport" content="width=device-width, initial-scale=1.0"><title>Web Component with Lit-html</title>
</head>
<body><my-element title="Dynamic Title"></my-element><script type="module" src="./my-element.js"></script>
</body>
</html>

事件处理

在 Lit-html 中处理事件非常简单。我们已经看到如何使用 @click 事件处理器来处理按钮点击事件。实际上,我们可以处理各种 DOM 事件,例如 input 事件、change 事件等。

下面是一个示例,展示如何通过输入框来更新组件的内容:

import { html, render } from 'lit-html';class MyElement extends HTMLElement {constructor() {super();this.attachShadow({ mode: 'open' });this.counter = 0;this.title = 'Default Title';}connectedCallback() {this.render();}handleInput(event) {this.title = event.target.value;this.render();}increment() {this.counter += 1;this.render();}render() {const template = html`<style>.container {padding: 16px;background-color: #f0f0f0;border-radius: 8px;text-align: center;}h1 {color: #333;}button, input {padding: 8px 16px;font-size: 16px;margin-top: 16px;}input {display: block;margin-bottom: 16px;}</style><div class="container"><input type="text" @input="${this.handleInput.bind(this)}" placeholder="Enter title" /><h1>${this.title}</h1><p>Counter: ${this.counter}</p><button @click="${this.increment.bind(this)}">Increment</button></div>`;render(template, this.shadowRoot);}
}customElements.define('my-element', MyElement);

在这个示例中,我们添加了一个输入框,并通过 @input 事件处理器来绑定 handleInput 方法。当用户在输入框中输入文本时,组件的标题会实时更新。

高级功能与优化策略

在前面的教程中,我们介绍了如何使用 Lit-html 创建基本的 Web 组件,并处理属性和事件。在这一部分,我们将深入探讨一些高级功能和优化策略,以便你能够创建更高效、可扩展的组件。

使用 LitElement 简化组件开发

虽然我们可以直接使用 Lit-html 创建 Web 组件,但 Lit-html 的兄弟项目 LitElement 提供了更高层的抽象,使得开发更加简便。LitElement 封装了一些常见的操作,如属性观察和更新、Shadow DOM 的管理等。

我们可以使用 LitElement 来创建一个自定义组件:

import { LitElement, html, css } from 'lit-element';class MyElement extends LitElement {static get properties() {return {title: { type: String },counter: { type: Number }};}constructor() {super();this.title = 'Default Title';this.counter = 0;}static get styles() {return css`.container {padding: 16px;background-color: #f0f0f0;border-radius: 8px;text-align: center;}h1 {color: #333;}button, input {padding: 8px 16px;font-size: 16px;margin-top: 16px;}input {display: block;margin-bottom: 16px;}`;}handleInput(event) {this.title = event.target.value;}increment() {this.counter += 1;}render() {return html`<div class="container"><input type="text" @input="${this.handleInput}" placeholder="Enter title" /><h1>${this.title}</h1><p>Counter: ${this.counter}</p><button @click="${this.increment}">Increment</button></div>`;}
}customElements.define('my-element', MyElement);

使用 LitElement,我们的代码变得更加简洁和结构化:

  • 属性管理:通过 static get properties() 方法自动处理属性变化。
  • 样式定义:通过 static get styles() 方法轻松定义和应用组件样式。
  • 渲染方法:使用 render() 方法进行模板渲染,简化了组件的更新和渲染逻辑。

使用指令优化模板渲染

Lit-html 提供了一些指令,可以帮助我们优化模板渲染。常见的指令包括:

repeat

repeat 指令用于高效地渲染列表:

import { repeat } from 'lit-html/directives/repeat';class MyElement extends LitElement {static get properties() {return {items: { type: Array }};}constructor() {super();this.items = ['Item 1', 'Item 2', 'Item 3'];}render() {return html`<ul>${repeat(this.items, (item) => item, (item, index) => html`<li>${index}: ${item}</li>`)}</ul>`;}
}
customElements.define('my-element', MyElement);
cache

cache 指令用于缓存渲染结果,提高性能:

import { cache } from 'lit-html/directives/cache';class MyElement extends LitElement {static get properties() {return {showContent: { type: Boolean }};}constructor() {super();this.showContent = true;}toggleContent() {this.showContent = !this.showContent;}render() {return html`<button @click="${this.toggleContent}">Toggle Content</button>${cache(this.showContent ? html`<p>Content is shown.</p>` : html``)}`;}
}
customElements.define('my-element', MyElement);

状态管理

对于复杂的应用,状态管理变得至关重要。我们可以结合外部状态管理库(如 Redux 或 MobX)来管理应用状态。

使用 Redux

首先,安装 Redux:

npm install redux

然后,我们可以创建一个简单的 Redux store,并将其集成到 LitElement 组件中:

import { createStore } from 'redux';
import { LitElement, html } from 'lit-element';// Redux reducer
const initialState = { counter: 0 };function counterReducer(state = initialState, action) {switch (action.type) {case 'INCREMENT':return { counter: state.counter + 1 };default:return state;}
}// Redux store
const store = createStore(counterReducer);class MyElement extends LitElement {constructor() {super();this.unsubscribe = store.subscribe(() => this.requestUpdate());}disconnectedCallback() {super.disconnectedCallback();this.unsubscribe();}increment() {store.dispatch({ type: 'INCREMENT' });}render() {const state = store.getState();return html`<div><p>Counter: ${state.counter}</p><button @click="${this.increment}">Increment</button></div>`;}
}customElements.define('my-element', MyElement);

在这个示例中,我们创建了一个 Redux store,并通过 subscribe 方法监听状态变化。当状态变化时,我们调用 requestUpdate 方法触发重新渲染。

总结

这篇教程,我们从基础到高级详细介绍了如何使用 Lit-html 和 LitElement 创建 Web 组件,并展示了处理属性和事件、使用指令优化渲染、以及结合状态管理的方式。除了基本用法,我们还探索了与其他库和工具的集成、最佳实践以及性能优化策略。

相关文章:

Web Component 教程(五):从 Lit-html 到 LitElement,简化组件开发

前言 在现代前端开发中&#xff0c;Web 组件是一种非常流行的技术&#xff0c;它允许我们创建可重用的、自包含的 UI 元素。而 Lit-html 是一个简洁高效库&#xff0c;用于在 Web 组件中进行渲染。在这篇教程中&#xff0c;我们一步步学习如何 Lit-html 来创建 Web Component。…...

Vue3:构建高效用户界面的利器

一、Vue.js 简介​ Vue.js&#xff08;读音 /vjuː/, 类似于 view&#xff09;是一套构建用户界面的渐进式框架。它只关注视图层&#xff0c;采用自底向上增量开发的设计。Vue 的目标是通过尽可能简单的 API 实现响应的数据绑定和组合的视图组件 &#xff0c;学习起来非常简单…...

LeetCode 2614.对角线上的质数:遍历(质数判断)

【LetMeFly】2614.对角线上的质数&#xff1a;遍历(质数判断) 力扣题目链接&#xff1a;https://leetcode.cn/problems/prime-in-diagonal/ 给你一个下标从 0 开始的二维整数数组 nums 。 返回位于 nums 至少一条 对角线 上的最大 质数 。如果任一对角线上均不存在质数&…...

红日靶场(二)——个人笔记

靶场搭建 新增VMnet2网卡 **web&#xff1a;**需要配置两张网卡&#xff0c;分别是外网出访NAT模式和内网域环境仅主机模式下的VMnet2网卡。 **PC&#xff1a;**跟web一样&#xff0c;也是需要配置两张网卡&#xff0c;分别是外网出访NAT模式和内网域环境仅主机模式下的VMn…...

实时视频分析的破局之道:蓝耘 MaaS 如何与海螺 AI 视频实现高效协同

一、蓝耘 MaaS 平台&#xff1a;AI 模型全生命周期管理的智能引擎 蓝耘 MaaS&#xff08;Model-as-a-Service&#xff09;平台是由蓝耘科技推出的 AI 模型全生命周期管理平台&#xff0c;专注于为企业和开发者提供从模型训练、推理到部署的一站式解决方案。依托云原生架构、高…...

清晰易懂的 Swift 安装与配置教程

初学者也能看懂的 Swift 安装与配置教程 本教程将手把手教你如何在 macOS 系统上安装 Swift&#xff0c;配置依赖包缓存位置&#xff0c;并指出新手容易踩坑的细节。即使你是零基础小白&#xff0c;也能快速上手&#xff01; 一、安装 Swift&#xff08;macOS 环境&#xff09…...

大数据 ETL 异常值缺失值处理完整方案

在大数据时代,数据已成为推动业务创新与决策优化的重要资产。然而,数据的海量、异构及实时性往往伴随着噪声、错误记录以及缺失现象,严重影响下游分析模型的准确性和可靠性。尤其在 ETL(抽取、转换、加载)环节中,如何在海量数据流中迅速甄别并处理异常数据,便成为决定整…...

macOS homebrew - 切换源

https://mirrors.tuna.tsinghua.edu.cn/help/homebrew/ 环境变量中 添加&#xff1a; export HOMEBREW_BREW_GIT_REMOTE"https://mirrors.tuna.tsinghua.edu.cn/git/homebrew/brew.git" export HOMEBREW_CORE_GIT_REMOTE"https://mirrors.tuna.tsinghua.edu.cn…...

如何基于Gone编写一个Goner对接Apollo配置中心(下)—— 对组件进行单元测试

项目地址&#xff1a;https://github.com/gone-io/gone 原文地址&#xff1a;https://github.com/gone-io/goner/blob/main/docs/test_goner.md 本文介绍的例子&#xff0c;代码在&#xff1a;https://github.com/gone-io/goner/blob/main/apollo 文章目录 引言编写“可测试”的…...

走进Java:String字符串的基本使用

❀❀❀ 大佬求个关注吧~祝您开心每一天 ❀❀❀ 目录 一、什么是String 二、如何定义一个String 1. 用双引号定义 2. 通过构造函数定义 三、String中的一些常用方法 1 字符串比较 1.1 字符串使用 1.2 字符串使用equals() 1.3 使用 equalsIgnoreCase() 1.4 cpmpareTo…...

python系列之元组(Tuple)

不为失败找理由&#xff0c;只为成功找方法。所有的不甘&#xff0c;因为还心存梦想&#xff0c;所以在你放弃之前&#xff0c;好好拼一把&#xff0c;只怕心老&#xff0c;不怕路长。 python系列之元组&#xff08;Turple&#xff09; 一、元组是什么&#xff1f;——给新手的…...

破解验证码新利器:基于百度OCR与captcha-killer-modified插件的免费调用教程

破解验证码新利器&#xff1a;基于百度OCR与captcha-killer-modified插件的免费调用教程 引言 免责声明&#xff1a; 本文提供的信息仅供参考&#xff0c;不承担因操作产生的任何损失。读者需自行判断内容适用性&#xff0c;并遵守法律法规。作者不鼓励非法行为&#xff0c;保…...

批量删除 PPT 中的所有图片、某张指定图片或者所有二维码图片

PPT 文档中的图片如何删除呢&#xff1f;相信很多小伙伴或碰到类似的需求。比如我们需要删除 PPT 文档中的某一张图片或者某张二维码图片&#xff0c;如果每一页都有这张图片&#xff0c;或者有很多 ppt 都有同一张要删除的图片&#xff0c;我们应该怎么快速的完成删除呢&#…...

大模型开发(六):LoRA项目——新媒体评论智能分类与信息抽取系统

LoRA项目——新媒体评论智能分类与信息抽取系统 0 前言1 项目介绍1.1 项目功能1.2 技术原理1.3 软硬件环境1.4 项目结构 2 数据介绍与处理2.1 数据集介绍2.2 数据处理2.3 数据导入器 3 模型训练3.1 配置文件3.2 工具函数3.3 模型训练3.4 模型评估 4 模型推理 0 前言 微调里面&…...

mysql-innodb存储引擎主键索引叶子结点数据结构(非单纯的双向链表)

我们应该清楚行记录是放在页中的。 compact行记录格式&#xff1a; 主要介绍几个比较重要的参数 heap_no&#xff1a; 页号 record_type&#xff1a; 0 表示普通类型&#xff08;叶子结点&#xff09;&#xff0c;1表示B树的非叶子节点 &#xff0c;2 表示最小记录&#xff…...

MySQL 进阶学习文档

一、存储引擎 1.1 核心架构 四层架构&#xff1a;连接层 → 服务层 → 引擎层 → 存储层插件式存储引擎&#xff1a;不同引擎独立管理数据存储&#xff0c;可动态选择 1.2 主流引擎对比 特性InnoDB&#xff08;默认&#xff09;MyISAMMemory事务支持✅ 支持❌ 不支持❌ 不支…...

物联网为什么用MQTT不用 HTTP 或 UDP?

先来两个代码对比&#xff0c;上传温度数据给服务器。 MQTT代码示例 // MQTT 客户端连接到 MQTT 服务器 mqttClient.connect("mqtt://broker.server.com:8883", clientId) // 订阅特定主题 mqttClient.subscribe("sensor/data", qos1) // …...

Vmware中的centos7连接上网

有很多刚刚开始配置了centos7&#xff0c;然后发现不能上网现在来解决这个问题。 测试能不能上网 先还原这个设置&#xff0c;如果没有动过的话就不用&#xff0c;连接模式是NAT模式 然后进去设置网络环境&#xff0c;记得是用超级用户设置 vi /etc/sysconfig/network-script…...

【AI知识】常见的优化器及其原理:梯度下降、动量梯度下降、AdaGrad、RMSProp、Adam、AdamW

常见的优化器 梯度下降&#xff08;Gradient Descent, GD&#xff09;局部最小值、全局最小值和鞍点凸函数和非凸函数动量梯度下降&#xff08;Momentum&#xff09;自适应学习率优化器AdaGrad&#xff08;Adaptive Gradient Algorithm&#xff09;​RMSProp&#xff08;Root M…...

线性规划的标准形式

标准形式的定义 目标函数&#xff1a;最大化线性目标函数 其中&#xff0c;x 是决策变量向量&#xff0c;c 是目标系数向量。 约束条件&#xff1a;等式形式约束 A x b, 其中&#xff0c;A 是约束系数矩阵&#xff0c;b 是常数项向量。 变量非负约束&#xff1a; 。 因此…...

网络安全应急入门到实战

奇安信&#xff1a;95015网络安全应急响应分析报告&#xff08;2022-2024年&#xff09;官网可以下载 https://github.com/Bypass007/Emergency-Response-Notes 应急响应实战笔记 网络安全应急响应技术实战指南 .pdf 常见场景 第4章 勒索病毒网络安全应急响应 第5章 挖矿木…...

应用程序安全趋势:左移安全、人工智能和开源恶意软件

软件是大多数行业业务运营的核心&#xff0c;这意味着应用程序安全从未如此重要。 随着组织采用云原生架构、微服务和开源组件&#xff0c;攻击面不断扩大。结果是&#xff1a;攻击者渴望利用的易受攻击和恶意依赖项数量不断增加。 2025 年&#xff0c;安全团队将面临日益复杂…...

ospf动态路由

一、为什么使用动态路由 OSPF&#xff08;open shortest path first开放最短路径优先&#xff09;是内部网关协议(IGP)的一种&#xff0c;基于链路状态算法(LS)。 OSPF企业级路由协议(RFC2328 OSPFv2)&#xff0c;核心重点协议 OSPF共三个版本&#xff0c;OSPFV1主要是实验室…...

【视频】OrinNX+Ubuntu20.04:移植OpenCV-4.11.0 with CUDA(含opencv_contrib )

1、源码下载 github下载地址如下,选择最新版本4.11 https://github.com/opencv/opencv/releases/tag/4.11.0 https://github.com/opencv/opencv_contrib/releases/tag/4.11.02、安装依赖库 1)对图片编码格式的支持 sudo apt install zlib1g-dev libjpeg8-dev libwebp-dev…...

基于单片机控制的电动汽车双闭环调速系统(论文+源码)

2.1系统方案 在本次设计中&#xff0c;其系统整个框图如下图3.1所示&#xff0c;其主要的核心控制模块由电源供电模块&#xff0c;晶振电路&#xff0c;驱动电路模块&#xff0c;霍尔传感器&#xff0c;按键模块&#xff0c;复位电路&#xff0c;LCD液晶显示及直流电机等组成。…...

【病毒分析】伪造微软官网+勒索加密+支付威胁,CTF中勒索病毒解密题目真实还原!

1.背景 该CTF挑战题目完整复现了黑客的攻击链路&#xff0c;攻击者通过伪造钓鱼页面引导受害者下载恶意软件。用户访问伪造的 Microsoft 365 官网后&#xff0c;在点击“Windows Installer (64-bit)”下载选项时&#xff0c;页面会自动跳转至伪造的 GitHub 项目链接&#xff0…...

PDF Reader Pro for Mac v4.9.0 PDF编辑/批注/OCR/转换工具 支持M、Intel芯片

PDF Reader Pro 是一款用户必备的集管理、编辑、转换、阅读功能于一体的专业的全能PDF阅读专家。快速、易用、强大&#xff0c;让您出色完成 PDF 工作。 应用介绍 PDF Reader Pro&#xff0c;一款功能齐全且强大的PDF阅读和编辑软件。支持PDF阅读、批注、PDF编辑、PDF格式转换…...

神经网络中层与层之间的关联

目录 1. 层与层之间的核心关联&#xff1a;数据流动与参数传递 1.1 数据流动&#xff08;Forward Propagation&#xff09; 1.2 参数传递&#xff08;Backward Propagation&#xff09; 2. 常见层与层之间的关联模式 2.1 典型全连接网络&#xff08;如手写数字分类&#xf…...

PowerShell 美化 增强教程

PowerShell Windows Terminal 美化 & 增强教程 Windows Terminal PowerShell 默认外观和功能较为基础&#xff0c;但通过 Oh My Posh 及其他增强工具&#xff0c;你可以打造一个更美观、更高效的终端环境。本教程提供完整的安装、美化和优化步骤&#xff0c;包括常见问题…...

机械革命蛟龙16pro玩游戏闪屏

我查过原因&#xff0c;好像是AMD显卡对游戏用了可变刷新率就出bug了&#xff0c;可能是那个游戏不适合用可变刷新率技术。 解决办法&#xff1a; 1.桌面右键鼠标&#xff0c;出现如下标签&#xff0c;点击AMD Software&#xff1a;Adrenalin Edition 2.选择闪屏的游戏&#x…...