使用JavaScript实现带权重和总计功能的双按钮点击计数器

使用JavaScript实现带权重和总计功能的双按钮点击计数器

本教程将指导您如何使用javascripthtml数据属性,构建一个包含两个独立点击计数器的页面,并实现一个全局总计功能。其中一个计数器每次点击使总计增加1,另一个计数器则按照设定的权重(例如每9次点击使总计增加1)来更新总计。我们将通过扩展现有的`clickcount`类和html配置来达到这一目标,确保代码模块化、易于维护。

教程背景与目标

在网页开发中,我们经常需要实现用户交互的计数功能。本教程将解决一个具体问题:在一个HTML页面上,存在两个独立的点击计数器,每个计数器都有自己的显示区域和重置按钮。在此基础上,我们需要添加一个全局的总计显示区域,该总计的更新规则如下:

当第一个按钮(例如“Verified Videos”)被点击时,全局总计直接增加1。当第二个按钮(例如“Document Count”)的点击次数达到特定倍数(例如9、18、27等)时,全局总计增加1(即每9次点击增加1)。

为了实现这一目标,我们将对现有的JavaScript ClickCount 类进行扩展,并优化HTML结构以传递必要的配置信息。

HTML结构优化:引入配置参数

为了使每个计数器能够灵活地配置其对全局总计的影响,我们将利用HTML的data-*属性来传递额外信息。具体来说,我们将扩展 data-config-click 属性,增加 weight 和 resultElement 两个字段。

weight: 用于指定该计数器对总计的权重。如果每次点击总计增加1,则 weight 为1。如果每N次点击总计增加1,则 weight 为N。resultElement: 用于指定全局总计显示元素的CSS选择器,以便JavaScript能够找到并更新它。

此外,我们需要一个专门的HTML元素来显示全局总计。

立即学习“Java免费学习笔记(深入)”;

更新后的 HTML (index.html):

    Click counter    

Total Units = 0

关键变化说明:

两个 c-block div 中的 data-config-click 属性都增加了 “weight” 和 “resultElement” 字段。”Document Count” 计数器设置 weight: 9,表示每9次点击总计增加1。”Verified Videos” 计数器设置 weight: 1,表示每次点击总计增加1。resultElement: “#total-units” 指向了页面底部的总计显示元素。页面底部添加了一个 span 元素,id=”total-units”,用于显示全局总计,并初始化为0。

CSS样式(保持不变)

CSS样式主要负责页面的布局和视觉呈现,对于本教程的功能实现没有直接影响,因此可以沿用原有的样式。

style.css (与原代码基本一致):

body {  background: #000000;  height: 100vh;  font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto, Oxygen, Ubuntu, Cantarell, "Open Sans", "Helvetica Neue", sans-serif;}.block-wrapper {  width: 100vw;  display: flex;  justify-content: center;  margin: auto;}.c-block {  margin: 50px;}button,b {  display: block;  text-align: center;}button {  border: none;  cursor: pointer;}button:focus {  outline: none;}.doctile {  background-color: #000000;  margin-top: 40px;  width: 375px;  height: 425px;  border-radius: 12px;  border: solid;  border-color: rgb(170, 170, 170, 0.5);  display: flex;  justify-content: center;  margin: 20px;}.vidtile {  background-color: #000000;  margin-top: 40px;  width: 375px;  height: 425px;  border-radius: 12px;  border: solid;  border-color: rgb(170, 170, 170, 0.5);  display: flex;  justify-content: center;  margin: 20px;}.button {  margin-top: 40px;  color: rgb(0, 190, 255);  padding: 4px 10px;  border-radius: 6px;  border: solid;  border-color: rgb(170, 170, 170, 0.5);  font-size: 32px;}.button:hover {  opacity: 0.9;}.button[disabled] {  opacity: 0.7;  cursor: not-allowed;}.button-primary {  background-color: rgb(85, 85, 85, .4);}.button-secondary {  background-color: rgb(85, 85, 85, .4);}.button-reset {  background-color: rgb(85, 85, 85, .4);  display: block;  color: rgb(0, 190, 255);  margin: 12px auto 0;  font-size: 16px;  padding: 4px 56px;  border-radius: 6px;  border: solid;  border-width: 2px;  border-color: rgb(170, 170, 170, 0.5);}.result {  color: rgb(0, 190, 255);  font-size: 100px;  margin-bottom: 8px;}.totaltile {  width: 500px;  height: 50px;  padding: 4px 10px;  border-radius: 6px;  border: solid;  border-color: rgb(170, 170, 170, 0.5);  font-size: 32px;  display: flex;  justify-content: center;  margin: auto;}p {  color: rgb(0, 190, 255);  font-size: 25px;  margin-top: 10px;}

JavaScript逻辑实现:扩展ClickCount类

核心的逻辑修改将在 script.js 文件中完成。我们将扩展 ClickCount 类,使其能够处理全局总计的更新,并根据配置的 weight 值计算贡献。

更新后的 JavaScript (script.js):

window.onload = () => {  class ClickCount {    constructor(obj) {      this.triggerNode = obj.triggerNode;      this.targetNode = obj.targetNode;      this.resetNode = obj.resetNode;      this.weight = obj.weight; // 新增:权重      this.resultNode = obj.resultNode; // 新增:全局总计显示节点      this.count = 0;      this.handleClick();      this.handleCount();    }    /**     * 更新全局总计:增加当前计数器的贡献     * 逻辑:计算当前点击前后的加权贡献差值,并更新总计。     * 例如,如果weight=9,count从8到9,则加权贡献从0增加到1。     * 如果count从17到18,则加权贡献从1增加到2。     */    increaseTotal() {      // 获取当前全局总计,如果非数字则默认为0      let currentTotal = isNaN(this.resultNode.innerHTML)        ? 0        : parseInt(this.resultNode.innerHTML);      // 计算当前点击前,该计数器的加权贡献      const previousWeightedCount = parseInt(this.count / this.weight);      // 从总计中减去旧的加权贡献      currentTotal -= previousWeightedCount;      // 计算当前点击后(count + 1),该计数器的加权贡献      const newWeightedCount = parseInt((this.count + 1) / this.weight);      // 将新的加权贡献加回到总计中      currentTotal += newWeightedCount;      // 更新全局总计显示      this.resultNode.innerHTML = currentTotal;    }    /**     * 更新全局总计:减少当前计数器的贡献(用于重置)     * 逻辑:计算当前计数器的加权贡献,并从总计中减去。     */    decreaseTotal() {      // 获取当前全局总计,如果非数字则默认为0      let currentTotal = isNaN(this.resultNode.innerHTML)        ? 0        : parseInt(this.resultNode.innerHTML);      // 计算当前点击数对应的加权贡献      const weightedCount = parseInt(this.count / this.weight);      // 从总计中减去该贡献      currentTotal -= weightedCount;      // 更新全局总计显示      this.resultNode.innerHTML = currentTotal;    }    /**     * 处理点击事件     */    handleClick() {      this.triggerNode.addEventListener("click", () => {        this.increaseTotal(); // 在计数器自身增加前,先更新全局总计        this.handleCount({          count: this.count++, // 增加计数器自身点击数        });      });      this.resetNode.addEventListener("click", () => this.handleReset());    }    /**     * 更新计数器自身的显示     */    handleCount() {      this.targetNode.innerHTML = this.count;    }    /**     * 处理重置事件     */    handleReset() {      this.decreaseTotal(); // 在计数器自身重置前,先从全局总计中移除贡献      this.count = 0; // 重置计数器自身点击数      this.handleCount();    }  }  // 遍历所有配置了点击计数器的HTML块  const clickBlock = document.querySelectorAll("[data-config-click]");  clickBlock.forEach((block) => {    const config = JSON.parse(block.dataset.configClick);    // 解构获取所有配置参数,包括新增的 weight 和 resultElement    const { trigger, target, reset, weight, resultElement } = config.selectors;    // 获取DOM节点    const triggerNode = block.querySelector(trigger);    const targetNode = block.querySelector(target);    const resetNode = block.querySelector(reset);    // 获取全局总计显示节点    const resultNode = document.querySelector(resultElement);    // 确保关键节点存在,然后创建 ClickCount 实例    if (triggerNode && targetNode) {      new ClickCount({        triggerNode,        targetNode,        resetNode,        resultNode, // 传递全局总计节点        weight, // 传递权重      });    }  });};

关键 JavaScript 变化说明:

window.onload 包装: 将所有初始化代码放入 window.onload 事件监听器中,确保DOM完全加载后再执行脚本,避免因元素未加载而导致的错误。ClickCount 构造函数扩展:新增 this.weight 和 this.resultNode 属性,用于存储当前计数器的权重和全局总计的DOM节点。increaseTotal() 方法:这是实现加权总计的核心。它首先获取当前的全局总计。然后,它计算当前点击前(this.count)该计数器对总计的贡献 (parseInt(this.count / this.weight)),并从 currentTotal 中减去。接着,计算点击后(this.count + 1)该计数器对总计的贡献 (parseInt((this.count + 1) / this.weight)),并加回到 currentTotal 中。这种“先减后加”的逻辑确保了即使连续多次调用 increaseTotal,只要 this.count 未改变,总计也不会重复增加。它准确地反映了在 this.count 增加1时,加权总计是否应该增加1。最后,更新 resultNode 的 innerHTML。decreaseTotal() 方法:用于处理计数器重置时,从全局总计中移除其贡献。它计算当前 this.count 对应的加权贡献,并直接从 currentTotal 中减去。handleClick() 方法修改:在 this.count++ 之前调用 this.increaseTotal(),确保在更新自身计数的同时,也更新全局总计。handleReset() 方法修改:在 this.count = 0 之前调用 this.decreaseTotal(),确保在重置自身计数的同时,也从全局总计中移除其贡献。初始化逻辑修改:在 clickBlock.forEach 循环中,从 config.selectors 中解构出 weight 和 resultElement。使用 document.querySelector(resultElement) 获取全局总计的DOM节点 resultNode。在创建 ClickCount 实例时,将 resultNode 和 weight 作为参数传递进去。

运行效果与注意事项

完成上述HTML、CSS和JavaScript的修改后,保存文件并在浏览器中打开 index.html。

点击“Verified Videos”按钮,其自身计数会增加,同时页面底部的“Total Units”也会增加1。点击“Document Count”按钮,其自身计数会增加。每当其计数达到9、18、27等时,“Total Units”会增加1。每个计数器的“Reset”按钮将重置该计数器,并同时从“Total Units”中减去该计数器之前贡献的加权值。

注意事项:

DOM加载顺序: 使用 window.onload 是一个良好的实践,确保在JavaScript尝试访问DOM元素时,这些元素已经存在于文档中。数据属性的解析: JSON.parse() 用于将 data-config-click 属性中的JSON字符串转换为JavaScript对象。请确保JSON格式正确,否则会导致解析错误。全局总计的唯一性: resultElement 应该指向一个唯一的元素(例如通过 id 选择器),以便所有计数器都能更新同一个全局总计。整数除法: parseInt(count / weight) 实现了向下取整的整数除法,这正是计算加权贡献所需的行为。健壮性: 在 increaseTotal 和 decreaseTotal 方法中,通过 isNaN(this.resultNode.innerHTML) ? 0 : parseInt(this.resultNode.innerHTML) 来获取 currentTotal,增加了代码的健壮性,防止因 resultNode 内容非数字而导致的问题。

总结

通过本教程,我们成功地将两个独立的点击计数器与一个具有复杂加权规则的全局总计功能结合起来。我们利用HTML数据属性实现了灵活的配置,并通过扩展JavaScript类使其具备了管理全局总计的能力。这种模块化和可配置的设计模式在构建复杂的交互式网页应用时非常有用。

以上就是使用JavaScript实现带权重和总计功能的双按钮点击计数器的详细内容,更多请关注创想鸟其它相关文章!

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

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月23日 14:05:58
下一篇 2025年12月23日 14:06:14

相关推荐

  • 如何使用LocalStorage持久化动态HTML表格数据

    本教程详细介绍了如何利用web storage api中的`localstorage`来持久化动态生成的html表格数据。通过将表格的每一行数据存储为一个javascript对象,并将其序列化为json字符串后存入`localstorage`,即使页面刷新,用户也能保留并继续操作之前添加的数据。文章…

    2025年12月23日
    000
  • 解决CSS样式表已加载但未生效的问题:路径配置详解

    当css样式表在浏览器中显示已加载但未对html元素生效时,常见原因在于其引用路径不正确。本文将深入探讨这一问题,并提供一个简单而有效的解决方案:通过明确使用相对路径前缀`./`来确保浏览器能正确解析css文件的位置,从而使样式得以正确应用。理解和掌握文件路径的正确配置是前端开发中避免此类常见问题的…

    2025年12月23日
    000
  • 解决Bootstrap导航栏响应式布局问题:data-bs-*属性的正确使用

    本教程旨在解决Bootstrap导航栏在不同视口下响应式行为异常的问题。核心症结在于错误使用了`data-mdb-*`而非标准的`data-bs-*`属性来控制导航栏的折叠功能。文章将详细阐述Bootstrap导航栏的响应式原理,并通过代码示例演示如何正确配置`data-bs-toggle`和`da…

    2025年12月23日
    000
  • PHP多语言网站切换机制实现教程

    本教程旨在详细阐述如何在php应用中实现一个健壮的多语言切换机制。我们将探讨如何利用会话(session)和url参数来管理用户选择的语言,并通过模块化的函数封装翻译逻辑,有效避免常见的变量作用域问题,并提供清晰的代码示例,帮助开发者构建易于维护和扩展的多语言网站。 在当今全球化的网络环境中,为网站…

    2025年12月23日
    000
  • 解决HTML元素尺寸不一致问题:深入理解CSS box-sizing 属性

    本教程深入探讨html元素,尤其是“,在设置`width`和`height`时出现尺寸不一致的常见问题。核心原因在于css `box-sizing` 属性的默认值 `content-box`,它导致 `padding` 和 `border` 会在声明尺寸之外额外增加。通过将 `box-s…

    2025年12月23日
    000
  • 响应式布局中内容居中对齐的Flexbox解决方案

    本文详细阐述了在响应式网页设计中,如何有效解决内容居中对齐问题,特别是在屏幕尺寸变化时保持元素居中。通过对比传统方法(如 `margin: auto` 结合 `position: absolute`)的局限性,重点介绍了使用 css flexbox(弹性盒子)模型,结合 `display: flex…

    2025年12月23日
    000
  • 掌握CSS Grid响应式布局:解决项目堆叠与全宽显示问题

    本教程旨在解决css grid布局在响应式设计中,网格项目无法在小屏幕下自动堆叠并占据全宽的问题。我们将详细讲解如何利用css媒体查询(`@media`)动态调整网格列结构(`grid-template-columns`)以及重置特定项目的定位属性,确保内容在不同设备上都能优雅展示,并提供代码示例和…

    2025年12月23日
    000
  • 使用在线工具快速识别浏览器视口与Bootstrap断点

    了解浏览器视口尺寸及其对应的bootstrap断点对于前端响应式设计至关重要。本文将介绍如何利用一个简单实用的在线工具,快速准确地识别当前浏览器的视口宽度所匹配的bootstrap响应式尺寸命名(如x-small、small等),从而辅助开发者进行布局调整和测试,优化多设备用户体验。 在现代Web开…

    2025年12月23日
    000
  • 深入解析CSS与Adobe光学字距调整:是否存在等效方案?

    adobe illustrator的光学字距调整通过专有算法动态分析字符形状以优化视觉间距。而css的font-kerning属性则依赖于字体文件中预设的度量字距调整数据。本文将深入探讨两者机制的根本差异,并明确指出css目前尚无直接等效于adobe算法化光学字距调整的功能。 理解Adobe光学字距…

    2025年12月23日
    000
  • 解决React中组件嵌套导致的输入框失焦问题

    本教程旨在解决react应用中常见的输入框失焦问题,该问题通常由组件在父组件内部定义所引起。通过将内部组件提升为独立组件并以props形式传递必要数据和函数,可以有效避免不必要的重渲染,从而保持输入框的焦点,提升用户体验。 引言:React输入框失焦的常见困境 在React开发中,开发者有时会遇到一…

    好文分享 2025年12月23日
    000
  • 构建健壮的XPath:应对动态DOM结构的策略

    本文深入探讨在web自动化中,如何解决因网页dom结构动态变化导致xpath失效的问题。通过分析html元素的稳定属性和文本内容,文章介绍了使用相对路径、`contains()`函数结合类名和文本内容来构建更健壮、更具弹性的xpath表达式,确保即使在元素位置动态变化时也能准确地定位目标元素,从而显…

    2025年12月23日
    000
  • 深入理解 document.querySelector 与表单提交事件处理

    本文旨在澄清 `document.queryselector` 的工作原理,特别是其如何选择指定元素,并解释表单 `submit` 事件的监听机制。我们将通过示例代码,详细阐述如何精确地选择html元素,以及如何在不直接选择提交按钮的情况下,有效地监听并处理表单的提交行为,从而避免常见的理解误区。 …

    2025年12月23日
    000
  • 掌握CSS Flexbox order 属性:实现联动DOM元素高效视觉重排序

    本教程探讨了在网页中同步移动或重排多个关联DOM元素的挑战,特别是当它们位于不同区域时。针对传统jQuery `insertAfter` 方法可能遇到的复杂性和性能问题,文章重点介绍了如何利用CSS Flexbox的 `order` 属性实现高效、声明式的视觉重排序。通过详细的代码示例和解释,读者将…

    好文分享 2025年12月23日
    000
  • CSS Grid布局中高度继承与fr单位的深度解析与实践

    本文深入探讨了css grid布局中子容器高度未按预期继承父容器高度的问题,尤其是在使用`fr`单位定义行高时。通过一个具体的卡片布局案例,文章详细解释了为何内部grid容器需要明确设置`height: 100%`才能正确响应父容器的高度,并使得`1fr`单位能够有效计算。教程提供了详细的代码示例和…

    2025年12月23日
    000
  • 解决JavaScript To-Do应用中动态列表项删除的逻辑问题

    本教程旨在解决JavaScript To-Do应用中动态列表项删除功能失效或错位的问题。我们将深入探讨如何通过优化数据管理、实现事件委托机制,以及确保删除操作与特定列表项精确关联,从而构建一个健壮且用户体验良好的删除功能,避免删除行为与预期不符的常见错误。 核心问题分析:为何删除按钮行为异常? 在开…

    2025年12月23日
    000
  • 在jQuery Selectivity插件中动态添加新选项的教程

    本教程详细介绍了如何在jQuery Selectivity插件中动态添加新选项,特别是当数据来源于服务器端时。文章强调了使用Selectivity提供的`add`方法而非直接DOM操作,并指导如何将服务器端数据正确序列化为JSON格式,以便JavaScript能够顺利处理并更新下拉列表,同时提醒了在…

    2025年12月23日
    000
  • HTML5日期选择器的高级控制:利用jQuery UI实现联动与编程开启

    本文探讨了html5原生日期选择器在编程控制上的局限性,特别是在事件触发后无法直接开启另一个日期选择器的问题。针对这一挑战,文章推荐使用功能更强大的jquery ui datepicker。通过详细的步骤和示例代码,演示了如何集成jquery ui datepicker,并利用其提供的api实现日期…

    2025年12月23日
    000
  • 响应式图片缩放:确保图片在不同屏幕尺寸下自适应的CSS指南

    本教程旨在解决图片在不同屏幕尺寸下无法自适应缩放的问题。我们将深入探讨如何利用CSS的相对单位(如百分比)和媒体查询(Media Queries)来创建响应式图片,确保图片在桌面和移动设备上都能优雅地调整大小并保持正确的宽高比,从而提升用户体验。 引言:理解响应式图片的重要性 在当今多设备并存的互联…

    2025年12月23日 好文分享
    000
  • JavaScript动态列表删除功能的最佳实践:数据驱动与事件委托

    引言:动态列表删除功能的常见挑战 在开发基于JavaScript的Web应用,特别是涉及动态内容(如待办事项列表、评论区等)时,实现准确的删除功能是一个常见需求。开发者经常会遇到一些困惑,例如点击某个删除按钮时,却意外地删除了其他列表项,或者删除操作未能正确反映在用户界面上。这种问题的根源通常在于D…

    2025年12月23日
    000
  • 优化Leaflet弹出层:动态内容中缺失图片的优雅处理策略

    本教程旨在解决leaflet地图弹出层中因缺失图片链接而显示“图片损坏”图标的问题。通过引入条件渲染机制,我们演示如何智能地检查图片链接的有效性,并仅在链接存在时才渲染对应的“标签,从而消除视觉上的瑕疵,提升用户体验,并使代码更加健壮和可维护。 在Leaflet地图应用中,动态生成弹出层内容并包含…

    2025年12月23日 好文分享
    000

发表回复

登录后才能评论
关注微信