gpt4 book ai didi

Javascript 事件委托(delegate)/冒泡

转载 作者:行者123 更新时间:2023-12-01 00:17:13 27 4
gpt4 key购买 nike

我想禁用 SPA 路由器正文部分中所有具有“nolink”类的链接。为了实现这一点,我使用了事件委托(delegate),它不能很好地处理嵌套元素。 (下面是简化的代码)。

HTML:

<header id="header">
<nobr>
<a href="home" class="nolink">
<img src="image.png">
<span>Title</span>
</a>
</nobr>
<ul>
<li><a href="link1" class="nolink">Link 1</a></li>
<li><a href="link2" class="nolink">Link 2</a></li>
<li><a href="link3" class="nolink">Link 3</a></li>
<li><a href="link4" class="nolink">Link 4</a></li>
</ul>
</header>

JavaScript:

class Delegator {
constructor(wrapper) {
this.wrapper = wrapper || document.body;
}

add({ selector, event, callback }) {
this.wrapper.addEventListener(event, e => {
const target = e.target;
if (target.matches(selector) /* bad code: */ || target.parentElement.matches(selector) || target.parentElement.parentElement.matches(selector)) {
e.preventDefault();
callback(e);
}
})
}
}

function readyLink(event) {
let href = event.target.href /* bad code: */ || event.target.parentElement.href || event.target.parentElement.parentElement.href;
history.pushState(null, null, href);
event.preventDefault();
router.fetch(href.split("/").pop());
router.route(href);
}

const router = new Router(...);
const bodyDelegator = new Delegator();
bodyDelegator.add({
selector: `a.${nolink}`,
event: "click",
callback: readyLinks
});

由于单击图像或 span 元素时的事件路由,必须引用 target.parentElements 令我烦恼。我希望它们受到添加功能的影响,而不必为将来可能的使用指定一千个parentElements。

最佳答案

使用 Element.closest() 查找元素本身或其父元素之一是否与选择器匹配:

class Delegator {
constructor(wrapper) {
this.wrapper = wrapper || document.body;
}

add({ selector, event, callback }) {
this.wrapper.addEventListener(event, e => {
const target = e.target;

if (target.closest(selector)) {
e.preventDefault();
callback(e);
}
})
}
}

传递最接近的回调:

class Delegator {
constructor(wrapper) {
this.wrapper = wrapper || document.body;
}

add({ selector, event, callback }) {
this.wrapper.addEventListener(event, e => {
const target = e.target;

const actual = target.closest(selector);

if (actual) {
e.preventDefault();
callback(e, actual);
}
})
}
}

然后readyLink就可以从实际元素中获取href。我仍然会将事件传递给其他用途(例如调用 preventDefault)。

function readyLink(event, actualTarget) {
let href = actualTarget.href;
history.pushState(null, null, href);
event.preventDefault();
router.fetch(href.split("/").pop());
router.route(href);
}

关于Javascript 事件委托(delegate)/冒泡,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/59687745/

27 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com