javascript事件模型其实就是观察者模式的体现,就是当事件触发时,监听该事件的全部触发事件都会被调用。
一、事件流
事件流主要分为两种,一种是事件捕获,另一种是事件冒泡。
事件捕获:事件从上向下进行触发,事件首先从最外侧父元素开始触发,然后一直向里延伸,知道触发到最内层的元素。
事件冒泡:事件从下向上进行触发,事件首先会触发最底层的子节点,然后一直向上触发父元素事件。
二、事件
1、DOM0事件,在不同的浏览器上面不存在兼容性问题,不过在触发事件中,默认使用的是事件冒泡。
//直接在html元素中绑定
<div id = "box1" onclick="func1()"></div>
//使用js获取元素标签绑定函数
let box1 = document.getElementbyId("box1")
box1.onclick = function() {
XXXXX
}
//移除监听函数
box1.onclick = null
IE事件模型
IE事件模型主要分为两个阶段:一个是事件处理阶段,就是当事件到达该元素时,执行相关函数阶段。
第二个是事件冒泡阶段:执行完事件阶段一,然后就会执行进行事件冒泡。
//ie事件监听
//监听事件
attachEvent(eventType, handler)
//移除事件
detachEvent(eventType, handler)
//参数解析
eventType:事件类型,需要加上on
handler:事件处理函数。
//代码解析
<body>
<div id="box1">
<div id="box2"></div>
</div>
<button id="btn">点击移除box1事件</button>
<script>
var box1 = document.getElementById('box1');
var box2 = document.getElementById("box2");
var btn = document.getElementById("btn");
function func1() {
console.log("执行box1")
}
function func2() {
console.log("执行box2")
}
box1.attachEvent("onclick", func1)
box2.attachEvent("onclick", func2)
btn.attachEvent("onclick", function () {
box1.detachEvent("onclick", func1)
})
</script>
</body>
DOM2级模型
DOM2级模型的事件,除了IE6-IE8不支持外,其他的都支持。
该事件中主要存在三个过程:
事件捕获:从最外层父元素从向最内层子元素中进行事件传递,如果存在事件处理函数,则执行。
事件处理:执行最内层的时间。
事件冒泡:从最内层子元素向最外层元素进行事件传递,如果存在事件处理函数,则执行。
//事件处理函数
//监听事件
addEventListener(eventType, handler, useCapture)
//移除事件
removeEventListener(eventType, handler, useCapture)
//代码展示
<script>
var box1 = document.getElementById('box1');
var box2 = document.getElementById("box2");
box1.addEventListener("click", function() {
console.log("box1")
},false)
box2.addEventListener("click",function() {
console.log("box2")
},false)
</script>
第三个参数来设置时事件捕获还是事件冒泡。默认是事件冒泡(false).
三、事件对象
当事件触发时,默认会创建一个新的对象,并且该对象携带一些属性和方法。并且该对象会作为第一个参数传递给监听函数。
DOM事件中常见的属性:
type:用于获取事件类型
target:获取事件目标
stopPropagation()阻止事件冒泡
preventDefault()阻止事件默认行为
IE事件模型常见的属性:
type用于获取事件类型
srcElement获取事件目标
cancelBubble阻止事件冒泡
returnValue阻止事件默认行为
四、event wrapper
因为存在DOM0和DOM2和IE事件模型,所以我们可以对其进行统一封装来统一调用。这样来解决兼容性问题。
const EventUtils = {
// 主要从DOM0|DOM2|IE事件模型来封装
// 监听事件
addEvent: function(element, type, handler) {
if(element.addEventListener) {
element.addEventListener(type, handler, false)
}else if(element.attachEvent) {
element.attachEvent("on" + type, handler)
}else {
element["on" + type] = handler
}
},
// 移除事件
removeEvent: function(element, type, handler) {
if(element.removeEventListener) {
element.removeEventListener(type, handler, false)
}else if(element.detachEvent) {
element.detachEvent("on" + type, handler)
}else {
element["on" + type] = handler
}
},
// 获取事件目标
getTarget: function(event) {
return event.target || event.srcElement;
},
// 获取event对象的引用,取到事件的所有信息,确保随时使用event
getEvent: function(event) {
return event || window.event
},
// 组织默认事件
stopPropagation: function(event) {
if(event.stopPropagation) {
event.stopPropagation()
}else{
event.cancelBubble = true
}
},
// 取消事件的默认行为
preventDefault: function(event) {
if(event.preventDefault) {
event.preventDefault()
}else {
event.returnValue = false
}
}
}
五、事件代理/事件委托
事件代理,也称为事件委托,就是将子元素中的事件全部放到父元素中进行触发。
<body>
<div id="box">
<button id="btn1">按钮1</button>
<button id="btn2">按钮2</button>
<button id="btn3">按钮3</button>
</div>
<script>
let box = document.getElementById('box')
let btn1 = document.getElementById("btn1")
let btn2 = document.getElementById("btn2")
let btn3 = document.getElementById("btn3")
box.addEventListener("click", function(event) {
if(event.target.getAttribute("id") === "btn1") {
console.log("您点击的是按钮一")
}else if(event.target.getAttribute("id") === "btn2") {
console.log("您点击的是按钮二")
}else {
console.log("您点击的是按钮三")
}
})
</script>
</body>
这样让所有事件全部由父元素进行处理,利用的是事件冒泡。
在react中的事件使用的也是事件委托。