如何定义一个事件来更改具有不同事件的元素?

发布于 2025-01-06 13:02:04 字数 724 浏览 0 评论 0原文

对于模糊的标题,抱歉,很难用一行字来解释...

无论如何,假设我有一个按钮,其操作由 data-action 属性定义。当用户单击该按钮时,会发生一些事情,并且按钮的操作将更改为第二状态。

然而,在实现这一点时,我发现为第二个状态定义的事件在触发第一个状态后会自动运行,就好像“单击”在函数调用之后持续存在并触发第二个事件一样。

这是一个 jsFiddle

单击一次应显示日志“转换为“foo””,第二次单击应显示日志“转换为“bar””,但如果您尝试一下,您会看到两个事件都被触发。

我怎样才能阻止这个?

编辑:感谢 Dimitar 的帮助 -

var action = ((this.retrieve('action', 'foo') != this.getProperty('data-action')) ? 'bar' : 'foo' );
if (action == 'foo') {
    console.log('executing "foo"');
    this.store('action', 'bar');
}
else {
    console.log('executing "bar"');
    this.store('action', 'foo');
}

Apologies for the vague title, it's kind of hard to explain in a line...

In any case, let's say I have a button with an action as defined by the data-action property. When the user clicks said button, something happens, and the button's action will change to its second state.

However, implementing this, I am finding that the event that is defined for that second state gets run automatically after the first state is triggered, as though the "click" persists past the function call and triggers the second event.

Here is a jsFiddle

One click should show the log 'transitioning to "foo"', and a second click should show the log 'transitioning to "bar"', but if you try it out, you'll see that both events are fired.

How can I stop this?

Edit: Thanks Dimitar for the help -

var action = ((this.retrieve('action', 'foo') != this.getProperty('data-action')) ? 'bar' : 'foo' );
if (action == 'foo') {
    console.log('executing "foo"');
    this.store('action', 'bar');
}
else {
    console.log('executing "bar"');
    this.store('action', 'foo');
}

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(2

怪我闹别瞎闹 2025-01-13 13:02:04

就像洛伦佐所说,当你委托它并更改用作委托标识符的属性时,这对我来说听起来像是反模式。我认为添加延迟也不是一个好的模式。

要么使用在元素上更改的第二个属性,或者更好的是,更改一个 var,该 var 在初始事件从属性读取状态后将状态保留在元素存储中以提高性能。

$(document.body).addEvents({
    'click:relay([data-action=foo])': function() {
        this.store("action", ((this.retrieve("action") || this.get("data-action")) === 'foo') ? 'bar' : 'foo');
        this.set("html", 'State + ' + this.retrieve("action"));
    }
});

like lorenzo said, when you delegate it and change the property, which is used as the delegation identifier, it sounds like an anti-pattern to me. I don't think that adding a delay is a good pattern either.

Either use a second property that you change on the element instead, or better yet, change a var that keeps state in element storage after initial event reads it from a property for performance.

$(document.body).addEvents({
    'click:relay([data-action=foo])': function() {
        this.store("action", ((this.retrieve("action") || this.get("data-action")) === 'foo') ? 'bar' : 'foo');
        this.set("html", 'State + ' + this.retrieve("action"));
    }
});
蓝天 2025-01-13 13:02:04

很好的情况:)

我看到更改事件处理程序声明的顺序解决了第一次单击的问题(并推迟了下一次单击的问题)。

而且,我认为 Mootools 上没有问题:您单击按钮,事件将 DOM 向上冒泡到主体,主体检查所有其附加事件(在您的情况下是委托的)并首先停止,因为它找到了匹配项[data-action="bar"]。并执行相关函数。主体现在继续检查其他委托匹配(它没有理由停止!),并找到第二个再次匹配的 [data-action="foo"]

恐怕如果不使用一些标志/状态变量或延迟函数执行,你就无法避免这种情况。我认为最后一个是最好的,或者至少是最简洁的解决方案(如果适用,我不知道延迟是否会给您带来问题)。如果您稍微延迟函数执行(以毫秒为单位),您只需跳过委托事件检查。演示此处

Nice situation :)

I saw that changing the order of event handlers declaration solve the problem for the first click (and postpone it for the next ones).

And, I think there is no issue on Mootools: you click the button, the event bubbles up the DOM to the body, the body check all its attach events (delegated, in your case) and stop at first because it finds a match for [data-action="bar"]. And the related function is executed. The body now continues to check for other delegation matches (it have no reason to stop!), and finds the second one [data-action="foo"] matching again.

I'm afraid you cannot avoid that without using some flag/state variable or delaying the function execution. I think this last one is the best, or at least the most concise, solution (if applicable, I don't know if the delay can cause you problems). If you delay the function execution a bit (in ms), you just jump over the delegation event checking. Demo here.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文