0

0

Bubble.js:针对常见问题的高效 1.6K 解决方案

PHPz

PHPz

发布时间:2023-08-28 23:21:02

|

871人浏览过

|

来源于php中文网

原创

bubble.js:针对常见问题的高效 1.6k 解决方案

Web 开发中最常见的任务之一是事件管理。我们的 JavaScript 代码通常监听 DOM 元素调度的事件。

这就是我们从用户那里获取信息的方式:也就是说,他或她点击、打字、与我们的页面交互,我们需要知道这种情况是否发生。添加事件监听器看起来很简单,但可能是一个艰难的过程。

在本文中,我们将看到一个真实的案例问题及其 1.6K 解决方案。

问题

我的一个朋友是一名初级开发人员。因此,他对普通 JavaScript 没有太多经验;然而,他不得不开始使用 AngularJS 和 Ember 等框架,而对 DOM 与 JavaScript 的关系没有基本的了解。在担任初级开发人员期间,他负责一个小项目:一个几乎不涉及 JavaScript 的单页活动网站。他遇到了一个小但非常有趣的问题,最终促使我编写了 Bubble.js。

假设我们有一个弹出窗口。一个样式漂亮的 元素:


这是我们用来显示消息的代码:

var popup = document.querySelector('.popup');
var showMessage = function(msg) {
    popup.style.display = 'block';
    popup.innerHTML = msg;
}
...
showMessage('Loading. Please wait.');

我们还有另一个函数 hideMessage ,它将 display 属性更改为 none 并隐藏弹出窗口。该方法可能适用于最一般的情况,但仍然存在一些问题。

例如,如果问题一一出现,我们需要实现额外的逻辑。假设我们必须向弹出窗口的内容添加两个按钮 -

var content = 'Are you sure?
'; content += 'Yes'; content += 'No'; showMessage(content);

那么,我们如何知道用户点击了它们呢?我们必须为每个链接添加事件侦听器。

例如:

var addListeners = function(yesCB, noCB) {
    popup.querySelector('.popup--yes').addEventListener('click', yesCB);
    popup.querySelector('.popup--no').addEventListener('click', noCB);
}
showMessage(content);
addListeners(function() {
    console.log('Yes button clicked');
}, function() {
    console.log('No button clicked');
});

上面的代码在第一次运行时有效。如果我们需要一个新按钮怎么办,或者更糟糕的是,如果我们需要不同类型的按钮怎么办?也就是说,如果我们继续使用 元素但使用不同的类名会怎么样?我们不能使用相同的 addListeners 函数,并且为弹出窗口的每个变体创建一个新方法很烦人。

以下是问题显而易见的地方:

  • 我们必须一次又一次地添加监听器。事实上,每次弹出窗口的 中的 HTML 发生更改时,我们都必须执行此操作。
  • 仅当弹出窗口的内容更新时,我们才能附加事件侦听器。仅在 showMessage 调用之后。我们必须始终考虑这一点并同步两个流程。
  • 添加侦听器的代码有一个硬依赖项 - popup 变量。我们需要调用它的 querySelector 函数,而不是 document.querySelector。否则,我们可能会选择错误的元素。
  • 一旦我们更改消息中的逻辑,我们就必须更改选择器,并且可能还更改 addEventListener 调用。它一点也不干。

必须有更好的方法来做到这一点。

是的,有更好的方法。不,解决方案不是使用框架。

在揭晓答案之前,我们先来谈谈 DOM 树中的事件。

了解事件处理

事件是网络开发的重要组成部分。它们为我们的应用程序添加了交互性,并充当业务逻辑和用户之间的桥梁。每个 DOM 元素都可以调度事件。我们所要做的就是订阅这些事件并处理接收到的Event对象。

有一个术语“事件传播”,它代表“事件冒泡”和“事件捕获”,这两者都是 DOM 中事件处理的两种方式。让我们使用以下标记来看看它们之间的区别。


我们将把 click 事件处理程序附加到这两个元素。不过,因为互相嵌套,所以他们都会收到 click 事件。

document.querySelector('.wrapper').addEventListener('click', function(e) {
    console.log('.wrapper clicked');
});
document.querySelector('a').addEventListener('click', function(e) {
    console.log('a clicked');
});

按下链接后,我们会在控制台中看到以下输出:

a clicked
.wrapper clicked

因此,这两个元素确实都接收了 click 事件。首先是链接,然后是 。这就是冒泡效果。从最深的元素到它的父母。有一种方法可以阻止冒泡。每个处理程序都会接收一个具有 stopPropagation 方法的事件对象:

document.querySelector('a').addEventListener('click', function(e) {
    e.stopPropagation();
    console.log('a clicked');
});

通过使用 stopPropagation 函数,我们表明该事件不应发送给父级。

有时我们可能需要颠倒顺序并让外部元素捕获事件。为此,我们必须在 addEventListener 中使用第三个参数。如果我们传递 true 作为值,我们将进行事件捕获。例如:

document.querySelector('.wrapper').addEventListener('click', function(e) {
    console.log('.wrapper clicked');
}, true);
document.querySelector('a').addEventListener('click', function(e) {
    console.log('a clicked');
}, true);

这就是我们与页面交互时浏览器处理事件的方式。

我秀秀淘宝客api源码
我秀秀淘宝客api源码

程序介绍:程序采用.net 2.0进行开发,全自动应用淘客api,自动采集信息,无需,手工更新,源码完全开放。(程序改进 无需填入阿里妈妈淘客API 您只要修改app_code文件下的config.cs文件中的id为你的淘客id即可)针对淘客3/300毫秒的查询限制,系统采用相应的解决方案,可以解决大部分因此限制带来的问题;程序采用全局异常,避免偶尔没考虑到的异常带来的问题;程序源码全部开放,请使

下载

解决方案

好吧,那么我们为什么要在文章中花一部分时间讨论冒泡和捕获呢?我们提到它们是因为冒泡是弹出窗口问题的答案。我们不应该将事件监听器设置为链接,而是直接设置为

var content = 'Are you sure?
'; content += 'Yes'; content += 'No'; var addListeners = function() { popup.addEventListener('click', function(e) { var link = e.target; }); } showMessage(content); addListeners();

通过遵循这种方法,我们消除了开头列出的问题。

  • 只有一个事件监听器,我们只添加一次。无论我们在弹出窗口中放入什么,事件的捕获都会在其父级中发生。
  • 我们不受附加内容的约束。换句话说,我们不关心 showMessage 何时被调用。只要 popup 变量处于活动状态,我们就会捕获事件。
  • 因为我们调用了 addListeners 一次,所以我们也使用了 popup 变量一次。我们不必保留它或在方法之间传递它。
  • 我们的代码变得灵活,因为我们选择不关心传递给 showMessage 的 HTML。我们可以访问被点击的锚点,因为 e.target 指向被按下的元素。

上面的代码比我们一开始的代码要好。然而,仍然无法以同样的方式发挥作用。正如我们所说, e.target 指向点击的 标签。因此,我们将使用它来区分按钮。

var addListeners = function(callbacks) {
    popup.addEventListener('click', function(e) {
        var link = e.target;
        var buttonType = link.getAttribute('class');
        if(callbacks[buttonType]) {
            callbacks[buttonType](e);
        }
    });
}
...
addListeners({
    'popup--yes': function() {
        console.log('Yes');
    },
    'popup--no': function() {
        console.log('No');
    }
});

我们获取了 class 属性的值并将其用作键。不同的类指向不同的回调。

但是,使用 class 属性并不是一个好主意。它保留用于将视觉样式应用于元素,并且其值可能随时更改。作为 JavaScript 开发人员,我们应该使用 data 属性。

var content = 'Are you sure?
'; content += 'Yes'; content += 'No';

我们的代码也变得更好了。我们可以删除 addListeners 函数中使用的引号:

addListeners({
    yes: function() {
        console.log('Yes');
    },
    no: function() {
        console.log('No');
    }
});

结果可以在这个 JSBin 中看到。

泡泡.js

我在几个项目中应用了上面的解决方案,因此将其包装为一个库是有意义的。它称为 Bubble.js,可在 GitHub 中找到。这是一个 1.6K 的文件,它的作用与我们上面所做的完全一样。

让我们将弹出示例转换为使用 Bubble.js。我们必须更改的第一件事是使用的标记:

var content = 'Are you sure?
'; content += 'Yes'; content += 'No';

我们应该使用 data-bubble-action 而不是 data-action

一旦我们在页面中包含 bubble.min.js ,我们就有了一个全局 bubble 函数。它接受 DOM 元素选择器并返回库的 API。 on 方法是添加监听器的方法:

bubble('.popup')
.on('yes', function() {
    console.log('Yes');
})
.on('no', function() {
    console.log('No');
});

还有一种替代语法:

bubble('.popup').on({
    yes: function() {
        console.log('Yes');
    },
    no: function() {
        console.log('No');
    }
});

默认情况下,Bubble.js 侦听 click 事件,但有一个选项可以更改该设置。让我们添加一个输入字段并监听其 keyup 事件:


JavaScript 处理程序仍然接收 Event 对象。因此,在这种情况下,我们可以显示字段的文本:

bubble('.popup').on({
    ...
    input: function(e) {
        console.log('New value: ' + e.target.value);
    }
});

有时我们需要捕获同一元素发出的多个事件而不是一个。 data-bubble-action 接受以逗号分隔的多个值:


在此处查找 JSBin 中的最终变体。

后备

本文提供的解决方案完全依赖于事件冒泡。在某些情况下 e.target 可能不会指向我们需要的元素。

例如:


如果我们将鼠标放在“choose”上并执行单击,则调度事件的元素不是 标记,而是 span 元素。

摘要

诚然,与 DOM 的通信是我们应用程序开发的重要组成部分,但我们使用框架只是为了绕过这种通信是一种常见的做法。

我们不喜欢一次又一次地添加监听器。我们不喜欢调试奇怪的双事件触发错误。事实上,如果我们了解浏览器的工作原理,我们就能够消除这些问题。

Bubble.js 只是几个小时阅读和一小时编码的结果 - 它是我们针对最常见问题之一的 1.6K 解决方案。

本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

拼多多赚钱的5种方法 拼多多赚钱的5种方法
拼多多赚钱的5种方法 拼多多赚钱的5种方法

在拼多多上赚钱主要可以通过无货源模式一件代发、精细化运营特色店铺、参与官方高流量活动、利用拼团机制社交裂变,以及成为多多进宝推广员这5种方法实现。核心策略在于通过低成本、高效率的供应链管理与营销,利用平台社交电商红利实现盈利。

109

2026.01.26

edge浏览器怎样设置主页 edge浏览器自定义设置教程
edge浏览器怎样设置主页 edge浏览器自定义设置教程

在Edge浏览器中设置主页,请依次点击右上角“...”图标 > 设置 > 开始、主页和新建标签页。在“Microsoft Edge 启动时”选择“打开以下页面”,点击“添加新页面”并输入网址。若要使用主页按钮,需在“外观”设置中开启“显示主页按钮”并设定网址。

16

2026.01.26

苹果官方查询网站 苹果手机正品激活查询入口
苹果官方查询网站 苹果手机正品激活查询入口

苹果官方查询网站主要通过 checkcoverage.apple.com/cn/zh/ 进行,可用于查询序列号(SN)对应的保修状态、激活日期及技术支持服务。此外,查找丢失设备请使用 iCloud.com/find,购买信息与物流可访问 Apple (中国大陆) 订单状态页面。

136

2026.01.26

npd人格什么意思 npd人格有什么特征
npd人格什么意思 npd人格有什么特征

NPD(Narcissistic Personality Disorder)即自恋型人格障碍,是一种心理健康问题,特点是极度夸大自我重要性、需要过度赞美与关注,同时极度缺乏共情能力,背后常掩藏着低自尊和不安全感,影响人际关系、工作和生活,通常在青少年时期开始显现,需由专业人士诊断。

7

2026.01.26

windows安全中心怎么关闭 windows安全中心怎么执行操作
windows安全中心怎么关闭 windows安全中心怎么执行操作

关闭Windows安全中心(Windows Defender)可通过系统设置暂时关闭,或使用组策略/注册表永久关闭。最简单的方法是:进入设置 > 隐私和安全性 > Windows安全中心 > 病毒和威胁防护 > 管理设置,将实时保护等选项关闭。

6

2026.01.26

2026年春运抢票攻略大全 春运抢票攻略教你三招手【技巧】
2026年春运抢票攻略大全 春运抢票攻略教你三招手【技巧】

铁路12306提供起售时间查询、起售提醒、购票预填、候补购票及误购限时免费退票五项服务,并强调官方渠道唯一性与信息安全。

122

2026.01.26

个人所得税税率表2026 个人所得税率最新税率表
个人所得税税率表2026 个人所得税率最新税率表

以工资薪金所得为例,应纳税额 = 应纳税所得额 × 税率 - 速算扣除数。应纳税所得额 = 月度收入 - 5000 元 - 专项扣除 - 专项附加扣除 - 依法确定的其他扣除。假设某员工月工资 10000 元,专项扣除 1000 元,专项附加扣除 2000 元,当月应纳税所得额为 10000 - 5000 - 1000 - 2000 = 2000 元,对应税率为 3%,速算扣除数为 0,则当月应纳税额为 2000×3% = 60 元。

35

2026.01.26

oppo云服务官网登录入口 oppo云服务登录手机版
oppo云服务官网登录入口 oppo云服务登录手机版

oppo云服务https://cloud.oppo.com/可以在云端安全存储您的照片、视频、联系人、便签等重要数据。当您的手机数据意外丢失或者需要更换手机时,可以随时将这些存储在云端的数据快速恢复到手机中。

121

2026.01.26

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号