1. DOM事件模型

DOM是微软和网景发生“浏览器大战”时期留下的产物,后来被“W3C”进行标准化,标准化一代代升级与改进,目前已经推行至第四代,即 level1(DOM1)、level2(DOM2)、level3(DOM3)、level4(DOM4)。事件模型是DOM的一部分,在不同的发展时期有不同的定义。

1.1 DOM0 / DOM1时期的事件模型

DOM0指的是未被“W3C”标准化前的DOM,DOM被“W3C”正式标准化后才开始的DOM1,因此DOM1是DOM0的整理和归纳,也就是说DOM事件模型最开始的定义是在DOM0 / DOM1 时期,这时的DOM事件模型如下:

① HTML中的onevent事件属性

  1. //在元素上使用 HTML attribute on{eventtype}
  2. <div onclick="alert('old')">点击div</div>
  3. //赋值为事件处理函数
  4. <div onclick="fn()"></div> //必须是加括号的调用形式
  5. <div onclick="fn.call()"></div> //相当于fn()
  6. <script>
  7. function fn(){
  8. console.log('ok');
  9. }
  10. </script>
  11. //此种方式的事件信息怎么传递
  12. <div onclick="pe(event)">点击</div> //这里必须传递 event 关键字对象
  13. <script>
  14. function pe(e){
  15. console.log(e); //事件信息
  16. }
  17. </script>

② js 操作DOM节点的事件属性

  1. //通过 JavaScript 设置页面元素相应的事件属性
  2. let div = document.getElementById("a");
  3. div.onclick = function() { alert('new') };
  4. //当函数单独定义时
  5. function fn(){
  6. alert("new");
  7. }
  8. div.onclick = fn; //必须是不加括号,传递地址的形式

1.2 DOM2时期的事件模型

DOM2中的事件模型已经非常完善了,在DOM3中并未对事件模型进行修改,因此至今的事件模型是以DOM2事件标准为基准的。DOM2标准事件模型采用了事件监听队列,如下:

  1. //addEventListener是添加事件监听,removeEventListener是移除事件监听
  2. let div = document.getElementById("a");
  3. div.addEventListener('click', function(e){
  4. console.log('点击div')
  5. })
  6. //基本语法
  7. target.addEventListener(type, listener, useCapture);
  8. target.addEventListener(type, listener, options);
  9. type:表示监听事件类型的字符串
  10. listener:事件处理函数,形参为事件信息
  11. useCaptureBoolean类型的值,表示是否在捕获阶段触发事件,默认为false,默认在冒泡阶段处理事件,若设置为true,则会在捕获阶段处理事件。
  12. options:一个指定有关 listener 属性的可选参数对象,相比于useCapture,可以有更多的设置,options对象的属性值都是Boolean值,默认全部为false
  13. {capture: 是否捕获阶段监听, once: 是否只监听一次, passive: 是否忽略preventDefault }

2. DOM事件机制(事件流)

DOM事件流的出现是在DOM节点中事件发生时常见的一种现象中产生的,如下问题:

  1. <div class="爷爷" onclick="console.log('我是爷爷')">
  2. <div class="爸爸" onclick="console.log('我是爸爸')">
  3. <div class="儿子" onclick="console.log('我是儿子')">
  4. 文字
  5. </div>
  6. </div>
  7. </div>
  8. //1. 点击了“文字”后,算不算点击了儿子?算不算点击了“爸爸”,算不算点击了“爷爷”?
  9. // 答案是都算,点击元素内部的任一元素节点,都算点击了该元素。这就涉及到一个事件流的问题

由上述可知,事件是会传递的,但是事件会以怎样的顺序进行传递执行?

在上述的例子中,我们第一眼就想到的就是,点击了“文字”,会依次向上传递,先执行离“文字”最近的儿子的click事件处理程序,再执行父亲的,最后执行爷爷的。没错,在“浏览器大战”时微软的IE浏览器就是按照由内向外的事件流顺序定义DOM事件流的。但是和其对立的网景公司却是反着定义的,网景公司的DOM事件流传递顺序是由外层向内层执行,先执行爷爷的,再执行爸爸的,最后执行儿子的。

在W3C规定的DOM2中统一了DOM事件机制的标准,即规定事件的传递顺序是先从外层向内层依次传递,称之为“捕获阶段”,再从内层向外层依次传递,称之为“冒泡阶段”。但并不意味着一个事件处理程序要在“捕获”和“冒泡”同时执行两次,而是用户选择其事件处理程序的执行时期是在“捕获阶段”还是“冒泡阶段”。在DOM2的 addEventListener函数的第三个参数(useCapture)就是让用户选择该处理程序是放在“捕获阶段”执行还是“冒泡阶段”执行。默认是 false,即事件监听机制默认是在“冒泡阶段”,也就是用 addEventListener 定义的事件处理程序是默认在冒泡阶段执行。

但是无论事件在哪个阶段执行,一个完整的事件流都是先“捕获阶段”,再“冒泡阶段”,捕获和冒泡都检查一遍。除此之外,还添加了一个“目标阶段”,就是用户真正点击的元素的事件处理阶段。

捕获阶段 ====> 目标阶段 ====> 冒泡阶段

捕获和冒泡都可以被阻止继续传播,阻止传播的方法是:e.stopPropagation()

最全的DOM事件笔记 - 图1

补充: 一般情况下,如果在捕获阶段和冒泡阶段都有对应的事件处理函数,一般是先执行捕获,再执行冒泡。但是如果只有一个div被监听时,对其来说,捕获和冒泡是同级的,因此会按照其代码顺序,谁先注册先执行。

3. DOM事件委托(代理)

事件委托是指本该自己监听的事件交给父元素或祖先元素监听,然后在祖先元素的监听函数中判断是否是触发的当前元素的事件,并进行相应的处理操作。事件委托就是由祖先元素监听事件,并根据事件来源统一处理。

有时候,页面元素是动态生成的,提前写好的事件绑定可能在元素还未出现时就已经执行,则事件处理则无法执行,除非在生成时再次绑定。这样比较麻烦,因此可以用事件代理,由祖先元素监听事件后进行相应的处理。被委托的祖先元素一般是页面中不变的一直存在的元素。那么被委托的祖先元素如何分发事件呢?

3.1 通过 e.target 分发事件

target 和 currentTarget 是DOM事件对象e上的属性,这两个属性是在事件委托中才各有各的作用,在不是事件委托的自己负责事件监听的DOM事件对象中,二者是相同的,都是元素本身。

二者的区别在于: e.target 是用户操作的实际对象,而 e.currentTarget 是程序员监听的对象。非箭头事件处理函数中的 this 是 e.currentTarget。

因此可以通过 e.target 来判断当前操作的元素,从而进行事件分发。

  1. <div class="grandpa">
  2. <div class="father">
  3. <div class="child">
  4. <span class="text">文字</span>
  5. </div>
  6. </div>
  7. </div>
  1. let grandpa = document.querySelector(".grandpa");
  2. grandpa.addEventListener('click',(e)=>{
  3. const t = e.target;
  4. if(t.className === "text"){
  5. console.log("span元素被点击了");
  6. }
  7. console.log(e.target); // 是用户实际点击的元素
  8. console.log(e.currentTarget); //一直是 div.grandpa 元素
  9. })

3.2 通过 path 分发事件

在上述的 e.target 分发事件存在不准确的问题,比如div.parent元素进行事件监听时,想监听的是div.child元素,但是点击child内部的”span”元素时也算点击了child元素,应该触发child的点击事件,但是这里单单凭借e.target获得到的是”span”元素,无法用于判断到点击了child元素。这里可以使用DOM事件对象的”path”属性。

这是点击 span 时的 “path” 属性

最全的DOM事件笔记 - 图2

这是点击 child 时的 “path” 属性

最全的DOM事件笔记 - 图3

因此 “path” 属性是一个存储了”捕获和冒泡”的所有事件传递的路径元素信息,从 e.target 被操作的元素本身到其祖宗元素。可以通过该”path”属性进行精确的判断,上述例子修改如下:

  1. let grandpa = document.querySelector(".grandpa");
  2. grandpa.addEventListener('click',(e)=>{
  3. let child = e.path.find(el=>el.className === 'child');
  4. if(child){
  5. console.log("child 被点击了"); //点击 span 元素时也会输出"child被点击了"
  6. }
  7. })

3.3 事件委托优点

① 省监听数,省内存
② 可以监听动态元素

4. DOM事件对象

4.1 e.target VS e.currentTarget

e.target 是操作对象本身,e.currentTarget是程序员监听的对象

4.2 e.path

e.path是存放了从 e.target 元素到最顶端祖先元素的事件传递路径信息的数组。

4.3 e.type

e.type是事件类型,如’click’事件。

4.4 e.bubbles VS e.cancelable

e.bubbles 和 e.cancelable 都是布尔值,e.bubbles 表示该事件是否支持冒泡,e.cancelable 表示该事件是否可以取消冒泡,如scroll不支持取消冒泡。

5. 自定义事件

浏览器提供了100多种事件,详细可见 DOM事件。但是浏览器也允许用户自定义事件,自定义事件需要考虑三个问题:
①自定义事件的事件定义
通过 “CustomEvent” 构造函数构造出一个自定义事件对象,第一个参数是自定义事件名,第二个参数是一个对象,是事件对象的定义。默认的 bubbles 等都是false。
②自定义事件的触发
通过dispatchEvent(event)函数触发自定义事件。
③自定义事件的监听
和普通事件一样的监听方法。

  1. let grandpa = document.querySelector(".grandpa");
  2. grandpa.addEventListener('click',(e)=>{
  3. const event = new CustomEvent('sayHi',{
  4. detail: {name: 'myEvent', content: 'hello'},
  5. bubbles: true,
  6. cancelable: true
  7. }); //① 自定义事件本身
  8. grandpa.dispatchEvent(event); //② 触发自定义事件
  9. })
  10. grandpa.addEventListener('sayHi',(e)=>{
  11. console.log(e);
  12. console.log(e.detail.content); //输出 hello
  13. }) //③ 监听自定义事件,和监听普通事件一样。