addEventListener实现事件绑定const btn = document.getElementById('btn1')
btn.addEventListener('click', event => {
console.log('clicked')
})
// 通用的绑定函数
function bindEvent(elem, type, fn){
elem.addEventListener(type, fn)
}
const a = document.getElementById('link1')
bindEvent(a, 'click', e=>{
e.preventDefault() // 阻止默认行为
alert('clicked')
})
DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>事件绑定title>
<style>
div {
border: 1px solid #ccc;
margin: 10px 0;
padding: 0 10px;
}
style>
head>
<body>
<button id="btn1">一个按钮button>
<script>
// 通用的绑定函数
function bindEvent(elem, type, fn) {
elem.addEventListener(type, fn)
}
const btn1 = document.getElementById('btn1')
bindEvent(btn1, 'click', event =>{
console.log(event.target) // 获取触发的元素
event.preventDefault() // 阻止默认行为
alert('clicked')
})
script>
body>
html>

<div id="div1">
<p id="p1">激活p>
<p id="p2">取消p>
<p id="p3">取消p>
<p id="p4">取消p>
div>
<div id="div2">
<p id="p5">取消p>
<p id="p6">取消p>
div>
<script>
// 通用的绑定函数
function bindEvent(elem, type, fn) {
elem.addEventListener(type, fn)
}
const p1 = document.getElementById('p1')
const body = document.body
bindEvent(p1, 'click', event=>{
console.log('激活')
})
bindEvent(body, 'click', event=>{
console.log('body clicked')
})
</script>
事件冒泡的情况:点击“激活”的p标签,会触发p标签绑定的事件,同时也会触发body绑定的事件。

此时,我们在“激活”的p标签的绑定事件中加入stopPropagation(),就可以防止事件冒泡。
<script>
// 通用的绑定函数
function bindEvent(elem, type, fn) {
elem.addEventListener(type, fn)
}
const p1 = document.getElementById('p1')
const body = document.body
bindEvent(p1, 'click', event=>{
event.stopPropagation() // 阻止事件冒泡
console.log('激活')
})
bindEvent(body, 'click', event=>{
console.log('body clicked')
})
</script>

事件代理是利用了事件冒泡的机制完成的。当需要对多个元素绑定相同的事件时,可以将事件绑定在这些元素的父元素上,利用冒泡机制实现子元素的事件绑定。
事件代理的优点:
DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>事件代理title>
head>
<body>
<div id="div1">
<a href="#">a1a><br>
<a href="#">a2a><br>
<a href="#">a3a><br>
<a href="#">a4a><br>
div>
<script>
function bindEvent(elem, type, fn){
elem.addEventListener(type, fn)
}
const div1 = document.getElementById('div1')
bindEvent(div1, 'click', event=>{
event.preventDefault() // 阻止跳转链接的默认行为
const target = event.target
if(target.nodeName === 'A'){
alert(target.innerHTML)
}
})
script>
body>
html>

function bindEvent(elem, type, selector, fn){
if(fn == null){
fn = selector
selector = null
}
elem.addEventListener(type, event=>{
const target = event.target
if(selector){
// 代理绑定
if(target.matches(selector)){
fn.call(target, event)
}
}else{
// 普通绑定
fn.call(target, event)
}
})
}