当前位置:首页 > 文章列表 > 文章 > 前端 > 自定义元素打造独立库存计数器

自定义元素打造独立库存计数器

2025-10-17 15:18:38 0浏览 收藏

还在为网页上多个独立库存计数器而烦恼吗?本文为你提供了一个高效的解决方案:利用Web Components中的自定义元素``,轻松打造多处独立库存计数器。告别传统JavaScript脚本ID冲突和全局状态共享的困扰,``将计数逻辑和状态管理封装在可重用的标签中,每个计数器都拥有独立的初始数量、随机递减逻辑,并通过localStorage实现状态持久化。本文将深入解析``的实现原理和使用方法,助你提升组件的模块化和可维护性,构建更强大的Web应用。无论你是前端工程师还是Web开发爱好者,都能从中受益,让库存管理更简单高效。

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

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

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

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

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

<span class="qty" id="qty"></span>

<script>
    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);
</script>

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

  1. ID的唯一性问题: document.getElementById()方法只会返回页面上第一个匹配指定ID的元素。即使你为每个计数器设置了不同的ID,但每个脚本实例都试图操作一个名为qtySpan的变量,并且可能由于作用域或执行顺序问题,导致它们最终都指向同一个DOM元素,或者只有第一个脚本成功绑定。
  2. 全局状态共享: 原始脚本中的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 代码解析

  1. customElements.define('stock-counter', class extends HTMLElement { ... });

    • 这是定义自定义元素的标准语法。'stock-counter'是自定义元素的标签名,必须包含连字符。
    • class extends HTMLElement 表示我们的自定义元素继承自HTMLElement,从而拥有所有标准HTML元素的特性。
  2. 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属性与内部状态同步。
  3. storageKey getter

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

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

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

3. 如何使用自定义元素

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

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>多库存计数器示例</title>
    <style>
        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;
        }
    </style>
</head>
<body>
    <h1>产品库存状态</h1>

    <div>
        产品A库存: <stock-counter quantity="40" storage-key="countdown-product-a"></stock-counter>
    </div>
    <div>
        产品B库存: <stock-counter quantity="50" storage-key="countdown-product-b"></stock-counter>
    </div>
    <div>
        产品C库存: <stock-counter quantity="80" storage-key="countdown-product-c"></stock-counter>
    </div>
    <div>
        产品D库存 (无持久化): <stock-counter quantity="25"></stock-counter>
    </div>

    <!-- 将自定义元素的定义脚本放在 body 结束标签前,确保 DOM 元素已加载 -->
    <script>
        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);
          };
        });
    </script>
</body>
</html>

在上面的示例中:

  • 每个标签都代表一个独立的库存计数器。
  • 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应用程序的强大工具。

今天关于《自定义元素打造独立库存计数器》的内容介绍就到此结束,如果有什么疑问或者建议,可以在golang学习网公众号下多多回复交流;文中若有不正之处,也希望回复留言以告知!

高德地图后台运行原因揭秘高德地图后台运行原因揭秘
上一篇
高德地图后台运行原因揭秘
Pythonmmap内存映射优化方法
下一篇
Pythonmmap内存映射优化方法
查看更多
最新文章
查看更多
课程推荐
  • 前端进阶之JavaScript设计模式
    前端进阶之JavaScript设计模式
    设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
    543次学习
  • GO语言核心编程课程
    GO语言核心编程课程
    本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
    516次学习
  • 简单聊聊mysql8与网络通信
    简单聊聊mysql8与网络通信
    如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
    500次学习
  • JavaScript正则表达式基础与实战
    JavaScript正则表达式基础与实战
    在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
    487次学习
  • 从零制作响应式网站—Grid布局
    从零制作响应式网站—Grid布局
    本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
    485次学习
查看更多
AI推荐
  • ChatExcel酷表:告别Excel难题,北大团队AI助手助您轻松处理数据
    ChatExcel酷表
    ChatExcel酷表是由北京大学团队打造的Excel聊天机器人,用自然语言操控表格,简化数据处理,告别繁琐操作,提升工作效率!适用于学生、上班族及政府人员。
    3180次使用
  • Any绘本:开源免费AI绘本创作工具深度解析
    Any绘本
    探索Any绘本(anypicturebook.com/zh),一款开源免费的AI绘本创作工具,基于Google Gemini与Flux AI模型,让您轻松创作个性化绘本。适用于家庭、教育、创作等多种场景,零门槛,高自由度,技术透明,本地可控。
    3391次使用
  • 可赞AI:AI驱动办公可视化智能工具,一键高效生成文档图表脑图
    可赞AI
    可赞AI,AI驱动的办公可视化智能工具,助您轻松实现文本与可视化元素高效转化。无论是智能文档生成、多格式文本解析,还是一键生成专业图表、脑图、知识卡片,可赞AI都能让信息处理更清晰高效。覆盖数据汇报、会议纪要、内容营销等全场景,大幅提升办公效率,降低专业门槛,是您提升工作效率的得力助手。
    3420次使用
  • 星月写作:AI网文创作神器,助力爆款小说速成
    星月写作
    星月写作是国内首款聚焦中文网络小说创作的AI辅助工具,解决网文作者从构思到变现的全流程痛点。AI扫榜、专属模板、全链路适配,助力新人快速上手,资深作者效率倍增。
    4526次使用
  • MagicLight.ai:叙事驱动AI动画视频创作平台 | 高效生成专业级故事动画
    MagicLight
    MagicLight.ai是全球首款叙事驱动型AI动画视频创作平台,专注于解决从故事想法到完整动画的全流程痛点。它通过自研AI模型,保障角色、风格、场景高度一致性,让零动画经验者也能高效产出专业级叙事内容。广泛适用于独立创作者、动画工作室、教育机构及企业营销,助您轻松实现创意落地与商业化。
    3800次使用
微信登录更方便
  • 密码登录
  • 注册账号
登录即同意 用户协议隐私政策
返回登录
  • 重置密码