利用自定义元素实现页面多处独立库存计数器

利用自定义元素实现页面多处独立库存计数器

本文详细介绍了如何使用Web Components中的自定义元素(Custom Elements)来解决在同一页面上显示多个独立库存计数器的问题。通过封装计数逻辑和状态管理到可重用的标签中,每个计数器都能拥有独立的初始数量、随机递减逻辑以及通过localStorage实现的状态持久化,有效避免了传统脚本因ID冲突和全局状态共享导致的问题,提升了组件的模块化和可维护性。

1. 问题背景与传统方法局限性

在网页开发中,我们经常需要展示动态变化的数值,例如商品库存、活动剩余名额等。当页面上只有一个这样的计数器时,使用简单的JavaScript配合DOM元素的ID属性即可实现。然而,当需要在同一页面上展示多个独立的计数器时,传统方法往往会遇到挑战。

最初的实现方式可能如下所示,它通过一个全局的元素和一个脚本来管理库存:

    const setQty = (qty) => {        qtySpan.innerHTML = qty;        if (qty == 0) return;        let parts = Math.floor((Math.random() * 3) + 1);        if (parts > qty) parts = qty;        const msec =  Math.floor(((Math.random() * 15) + 15) * 1000);        qty -= parts;        // Save the updated quantity in localStorage        localStorage.setItem('saved_countdown', qty);        setTimeout(() => setQty(qty), msec);    }    // Get the saved countdown value from localStorage, or use default value of 57 if not found    const defaultQty = localStorage.getItem('saved_countdown') ?? 57;    const qtySpan = document.getElementById('qty');    // Set the initial value of the quantity    setQty(defaultQty);

这种方法在只有一个计数器时工作良好。但当尝试通过复制脚本并更改ID(如qty1, qty2等)来创建多个计数器时,会发现只有第一个或某一个计数器能够正常工作。其根本原因在于:

ID的唯一性问题: document.getElementById()方法只会返回页面上第一个匹配指定ID的元素。即使你为每个计数器设置了不同的ID,但每个脚本实例都试图操作一个名为qtySpan的变量,并且可能由于作用域或执行顺序问题,导致它们最终都指向同一个DOM元素,或者只有第一个脚本成功绑定。全局状态共享: 原始脚本中的localStorage.setItem(‘saved_countdown’, qty);使用了固定的键名’saved_countdown’。这意味着所有计数器实例都会尝试读写同一个localStorage条目,导致它们的状态相互覆盖,无法独立保存和恢复。

为了解决这些问题,我们需要一种更模块化、更具封装性的方法来创建可复用的UI组件,而Web Components中的自定义元素(Custom Elements)正是为此而生。

2. 解决方案:利用自定义元素

自定义元素允许我们定义自己的HTML标签,并为其关联特定的JavaScript行为和样式。通过这种方式,每个计数器都可以成为一个独立的组件,拥有自己的内部状态和逻辑,互不干扰。

我们将创建一个名为的自定义元素,它将具备以下特性:

独立的计数逻辑: 每个实例都将运行自己的倒计时逻辑。可配置的初始数量: 通过quantity属性设置计数器的起始值。独立的持久化存储: 通过storage-key属性为每个计数器指定一个唯一的localStorage键名,实现状态的独立保存和恢复。

2.1 自定义元素的核心代码

以下是自定义元素的完整实现:

customElements.define('stock-counter', class extends HTMLElement {  // quantity 属性的 getter 方法  get quantity() {    // 1. 检查是否设置了 storage-key 并且 localStorage 中有存储的值    if (this.storageKey !== null) {      const value = Number(localStorage.getItem(this.storageKey));      // 如果存储的值是有效的数字且不为 0,则优先使用它      if (!Number.isNaN(value) && value !== 0) {        return value;      }    }    // 2. 如果没有有效的存储值,则从 quantity 属性获取初始值    const value = Number(this.getAttribute('quantity'));    // 如果 quantity 属性值无效,则返回 0    if (Number.isNaN(value)) {      return 0;    }    return value;  }  // quantity 属性的 setter 方法  set quantity(value) {    if (!isNaN(value)) {      // 如果设置了 storage-key,则将新值存储到 localStorage      if (this.storageKey !== null) {        localStorage.setItem(this.storageKey, value);      }      // 更新元素的 quantity 属性      this.setAttribute('quantity', value);    }  }  // storage-key 属性的 getter 方法  get storageKey() {    return this.getAttribute('storage-key');  }  // 当元素被添加到文档流时调用  connectedCallback() {    this.count(); // 启动计数器  }  // 核心计数逻辑方法  count = () => {    const qty = this.quantity; // 获取当前数量    this.textContent = qty;     // 更新元素显示文本    if (qty === 0) {      return; // 数量为 0 时停止计数    }    let parts = Math.floor((Math.random() * 3) + 1); // 随机生成递减数量 (1-3)    if (parts > qty) {      parts = qty; // 确保递减数量不超过当前数量    }    this.quantity -= parts; // 递减数量,同时触发 setter 更新 localStorage 和属性    const msec = Math.floor(((Math.random() * 15) + 15) * 1000); // 随机生成下一次递减的间隔时间 (15-30秒)    setTimeout(this.count, msec); // 在指定时间后再次调用 count 方法  };});

2.2 代码解析

customElements.define(‘stock-counter’, class extends HTMLElement { … });

这是定义自定义元素的标准语法。’stock-counter’是自定义元素的标签名,必须包含连字符。class extends HTMLElement 表示我们的自定义元素继承自HTMLElement,从而拥有所有标准HTML元素的特性。

quantity getter/setter

get quantity(): 这个getter负责获取当前计数器的数量。它遵循一个优先级逻辑:优先从localStorage获取: 如果设置了storage-key属性,并且localStorage中存在一个有效且不为0的值,则使用该值。这确保了页面刷新后计数器能从上次的状态恢复。其次从quantity属性获取: 如果localStorage中没有有效值,则从HTML标签的quantity属性中获取初始值。最后默认为0: 如果quantity属性值无效(例如非数字),则默认为0。set quantity(value): 这个setter负责设置计数器的数量。它会将新值存储到localStorage(如果storage-key已设置)。同时,它会更新元素的quantity属性,确保DOM属性与内部状态同步。

storageKey getter

简单地返回storage-key属性的值。这个属性用于指定localStorage中存储该计数器状态的键名。

connectedCallback()

这是一个Web Components的生命周期回调函数,当自定义元素首次被添加到文档DOM时,浏览器会自动调用它。在这里,我们调用this.count()来启动计数器。

count = () => { … };

这是实现随机递减的核心逻辑。this.textContent = qty;:将当前数量显示在自定义元素内部。随机递减逻辑:parts变量用于生成每次递减的数量(1到3之间)。this.quantity -= parts;:更新计数器的数量。注意这里调用的是我们自定义的quantity setter,它会自动处理localStorage的更新。setTimeout(this.count, msec);:使用setTimeout实现递归调用,每隔一段随机时间(15到30秒)执行一次递减,模拟库存缓慢减少的场景。

3. 如何使用自定义元素

一旦自定义元素被定义,你就可以像使用任何标准HTML标签一样在页面中使用它。

            多库存计数器示例            stock-counter {            display: inline-block; /* 使其表现得像行内块元素 */            padding: 5px 10px;            margin: 10px;            border: 1px solid #ccc;            background-color: #f9f9f9;            font-size: 1.2em;            font-weight: bold;            min-width: 50px;            text-align: center;        }        

产品库存状态

产品A库存:
产品B库存:
产品C库存:
产品D库存 (无持久化):
customElements.define('stock-counter', class extends HTMLElement { get quantity() { if (this.storageKey !== null) { const value = Number(localStorage.getItem(this.storageKey)); if (!Number.isNaN(value) && value !== 0) { return value; } } const value = Number(this.getAttribute('quantity')); if (Number.isNaN(value)) { return 0; } return value; } set quantity(value) { if (!isNaN(value)) { if (this.storageKey !== null) { localStorage.setItem(this.storageKey, value); } this.setAttribute('quantity', value); } } get storageKey() { return this.getAttribute('storage-key'); } connectedCallback() { this.count(); } count = () => { const qty = this.quantity; this.textContent = qty; if (qty === 0) { return; } let parts = Math.floor((Math.random() * 3) + 1); if (parts > qty) { parts = qty; } this.quantity -= parts; const msec = Math.floor(((Math.random() * 15) + 15) * 1000); setTimeout(this.count, msec); }; });

在上面的示例中:

每个标签都代表一个独立的库存计数器。quantity属性设置了每个计数器的初始库存量。storage-key属性为每个计数器提供了唯一的localStorage键名,确保它们的状态可以独立地保存和恢复。第四个计数器没有设置storage-key,这意味着它的状态不会被持久化,每次页面加载都会从quantity=”25″开始。

4. 注意事项与总结

浏览器兼容性: 自定义元素是Web Components规范的一部分,现代浏览器(Chrome, Firefox, Edge, Safari)对其支持良好。对于旧版浏览器,可能需要引入Polyfill。storage-key的重要性: 如果你需要计数器在页面刷新后保持其状态,务必为每个独立的实例提供一个唯一的storage-key。否则,它们仍可能共享localStorage中的同一个存储位置。默认行为: 如果省略storage-key属性,计数器将不会将状态保存到localStorage,每次页面加载都会从quantity属性指定的初始值开始。CSS样式: 自定义元素默认是行内元素,你可以通过CSS display属性(如display: block;或display: inline-block;)来控制其布局和样式。

通过采用自定义元素,我们成功地将复杂的、相互独立的计数器逻辑封装成可复用的HTML标签。这种方法不仅解决了传统脚本在多实例场景下的问题,还极大地提升了代码的模块化、可读性和可维护性,是构建现代Web应用程序的强大工具

以上就是利用自定义元素实现页面多处独立库存计数器的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1527365.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月20日 19:21:34
下一篇 2025年12月20日 19:21:40

相关推荐

  • 在代码压缩和混淆过程中,如何保证 JavaScript 源码的安全性?

    JavaScript 无法绝对安全,但可通过混淆和压缩提升逆向难度。使用专业工具如 JavaScript Obfuscator 进行变量函数重命名、控制流扁平化、字符串加密及添加调试保护,结合 Webpack 或 Vite 在构建时集成混淆与压缩,禁用或偏移 source map,并将敏感逻辑(如认…

    2025年12月20日
    000
  • JavaScript Prettier代码格式化

    Prettier是提升JavaScript代码可读性和团队协作效率的主流格式化工具,支持多种语言,能自动处理空格、换行、引号等格式问题。通过npm或yarn在项目中本地安装可避免版本不一致问题。支持配置文件如.prettierrc或prettier.config.js来自定义规则,常用配置包括sem…

    2025年12月20日
    000
  • Node.js异步编程:正确处理HTTP请求与数据同步

    本文深入探讨了Node.js中因`https.get`等异步操作未等待完成就返回结果,导致外部变量未更新的问题。通过分析Node.js的事件循环和非阻塞I/O机制,教程将详细介绍如何利用Promise和`async/await`语法,确保所有异步请求完成后再处理数据并发送响应,从而解决数据同步难题,…

    2025年12月20日
    000
  • React TypeScript中嵌套数组状态的不可变更新策略

    本文深入探讨了在React和TypeScript环境中,如何高效且安全地管理和更新嵌套在对象中的数组状态。通过分析常见的TypeError问题,我们强调了React状态不可变性原则的重要性,并提供了具体的代码示例,演示了如何使用useState的函数式更新和数组的map方法来精确地修改、添加嵌套数组…

    2025年12月20日
    000
  • React中Textarea滚动条不显示:诊断与解决方案

    本文旨在解决react应用中多行文本输入框(textarea)滚动条不显示的问题。核心症结在于开发者常误用`input type=”textarea”`,而正确的做法应是直接使用html “ 标签。本教程将详细阐述如何正确在react中集成和使用“元素,并通过css样式确…

    2025年12月20日
    000
  • 解决Socket.io与Express.js CORS策略冲突的全面指南

    本文旨在解决node.js应用中,当express.js与socket.io共存时,即使已配置express的cors头,socket.io连接仍可能遭遇cors策略阻塞的问题。我们将深入探讨cors机制,解释express与socket.io处理cors的差异,并提供针对socket.io的独立c…

    2025年12月20日
    000
  • 使用jQuery高效实现卡片内信息面板的显示与隐藏切换

    本文将指导您如何使用jquery,以简洁高效的方式实现卡片(card)组件内部信息面板的显示与隐藏切换功能。我们将探讨两种常见模式:独立的面板切换和手风琴式(accordion)面板切换,并通过最佳实践和示例代码,帮助您优化事件处理、元素查找及css类管理,避免常见的开发陷阱。 在现代Web应用中,…

    2025年12月20日 好文分享
    000
  • 使用 jQuery 实现卡片内元素显示/隐藏切换的专业指南

    本教程详细讲解如何使用 jquery 在卡片(`coin-card`)内部点击按钮时,切换隐藏内容面板(`more-info-panel`)的显示状态。文章将重点介绍事件绑定、dom 遍历技巧,并提供两种切换模式:独立切换和手风琴式切换,确保代码简洁、高效且易于维护。 在现代网页设计中,卡片式布局(…

    2025年12月20日 好文分享
    000
  • JavaScript 中 `window.onload` 与异步操作的正确姿势

    window.onload 在处理异步操作时可能出现时序问题,导致脚本行为不稳定。本文将详细解释 window.onload 的执行机制,并提供一种健壮的解决方案:通过将 window.onload 定义为 async 函数,并使用 await 确保所有异步数据加载完成后再执行相关初始化逻辑,从而保…

    2025年12月20日
    000
  • 解决React Redux刷新时localStorage数据丢失问题

    本文旨在解决react redux应用中,刷新页面时`localstorage`数据丢失的问题。我们将深入探讨如何利用`useeffect`钩子和redux状态管理,实现数据的加载与持久化,避免常见错误如无限循环,并提供清晰的示例代码和最佳实践,确保应用状态在页面刷新后依然保持一致。 在单页应用(S…

    2025年12月20日
    000
  • Node.js异步编程实践:解决https.get回调中数据更新不同步问题

    在node.js的开发实践中,处理异步操作是核心技能之一。然而,由于javascript的单线程非阻塞特性,不正确地管理异步流程常常会导致意想不到的结果,例如本文将探讨的,在`https.get`等网络请求的回调函数中更新的数据,在外部作用域却无法正确获取的问题。这种现象的根源在于对异步执行顺序的误…

    2025年12月20日
    000
  • JavaScript DOM diff算法实现

    答案是DOM diff算法通过比较新旧虚拟DOM树差异来最小化真实DOM操作。首先定义虚拟节点结构h函数,再实现render函数将虚拟节点转为真实DOM;diff函数处理五种情况:新节点不存在则删除、文本节点直接替换、标签不同替换元素、标签相同更新属性、递归对比子节点;最后通过实例展示1秒后更新视图…

    2025年12月20日
    000
  • 如何构建一个不依赖框架的、声明式的 JavaScript 渲染引擎?

    答案:通过虚拟DOM、Diff算法与Proxy响应式系统实现声明式渲染。用纯函数组件描述UI,状态变化时自动最小化更新视图,核心为vnode生成、比对与副作用追踪。 构建一个不依赖框架的声明式 JavaScript 渲染引擎,核心在于将 UI 视为状态的函数,并通过观察状态变化自动更新视图。不需要 …

    2025年12月20日
    000
  • Mongoose 数据复制 VersionError 深度解析与解决方案

    本文深入探讨了在 mongoose 中将文档从一个集合复制到另一个集合时可能遇到的 `versionerror`。我们将解析该错误产生的根本原因,即 mongoose 对文档实例和版本控制的内部处理机制。文章将提供安全且推荐的解决方案,通过将 mongoose 文档转换为普通 javascript …

    2025年12月20日
    000
  • JavaScript中高效生成指定范围唯一随机数:避免栈溢出的策略

    本文深入探讨了在javascript中生成指定范围唯一随机数时可能遇到的rangeerror: maximum call stack size exceeded问题。通过分析导致栈溢出的低效递归方法,文章介绍了一种基于数组操作和洗牌算法的高效解决方案,该方法简洁、性能优越,能够有效避免递归陷阱,确保…

    2025年12月20日
    000
  • 如何实现一个支持语法高亮的代码编辑器核心?

    实现语法高亮编辑器核心需结合文本解析与实时渲染,采用行数组模型存储文本及token,通过虚拟DOM和CSS定位高效渲染;基于词法分析设计轻量tokenizer,用正则逐行匹配关键词、字符串、注释等,并处理跨行状态;构建可扩展的语言配置系统,支持多语言规则动态加载;优化性能,仅渲染可视区、缓存结果、防…

    2025年12月20日
    000
  • JavaScript移动端适配方案

    移动端适配中,JavaScript通过动态设置rem根字体、处理高清屏1px边框、响应式事件兼容等方式辅助实现自适应布局。首先根据设计稿宽度与设备实际宽度计算根字体大小,使页面等比缩放;结合viewport meta确保视口正确;利用devicePixelRatio判断dpr,通过伪元素或类名实现高…

    2025年12月20日
    000
  • 动态加载Bootstrap Carousel的初始化与重置

    在使用%ignore_a_1%动态向dom添加bootstrap carousel组件后,需要显式地对其进行初始化以确保功能正常。本文将介绍两种基于javascript的无jquery初始化方法:`new bootstrap.carousel()`构造函数和`bootstrap.carousel.g…

    2025年12月20日
    000
  • 构建FormData以向C#控制器发送包含文件字段的数组数据

    本教程详细阐述了如何使用javascript `formdata`对象,将包含文件类型属性的复杂对象数组高效地上传至c# asp.net mvc后端控制器。我们将探讨前端数据构建的正确方法,包括文件和文本字段的索引命名约定,以及后端控制器如何自动绑定这些数据,确保文件上传和数据传输的完整性与准确性。…

    2025年12月20日
    000
  • jQuery实现卡片内隐藏面板的切换显示:两种模式详解

    本文详细阐述了如何使用jquery在卡片组件内部实现隐藏面板的切换显示功能。通过优化事件绑定机制和dom遍历方法,我们展示了两种核心模式:简单的独立切换和手风琴式的排他性切换。教程强调了避免混合使用`onclick`与jquery事件绑定、高效利用`closest()`和`find()`进行元素定位…

    2025年12月20日
    000

发表回复

登录后才能评论
关注微信