vue.js - 奇怪的 event 对象

作者: ebfc7d0362e4 | 来源:发表于2017-09-05 14:41 被阅读526次

好久都没有写点东西了, 前段时间工作搞得头大,真的就是一起加班到死了。废话不多说,写这篇文章是因为这次因为 event 对象闹了一个乌龙,以此总结一下。

一、event 对象

(一)事件的 event 对象

你说你是搞前端的,那么你肯定就知道事件,知道事件,你就肯定知道 event 对象吧?各种的库、框架多少都有针对 event 对象的处理。比如 jquery,通过它内部进行一定的封装,我们开发的时候,就无需关注 event 对象的部分兼容性问题。最典型的,如果我们要阻止默认事件,在 chrome 等浏览器中,我们可能要写一个:

event.preventDefault();

而在 IE 中,我们则需要写:

event.returnValue = false;

多亏了 jquery ,跨浏览器的实现,我们统一只需要写:

event.preventDefault();

兼容?jquery 内部帮我们搞定了。类似的还有比如阻止事件冒泡以以及事件绑定(addEventListener / attachEvent)等,简单到很多的后端都会使用 $('xxx').bind(...),这不是我们今天的重点,我们往下看。

(二)vue 中的 event 对象

我们知道,相比于 jquery,vue 的事件绑定可以显得更加直观和便捷,我们只需要在模板上添加一个 v-on 指令(还可以简写为 @),即可完成类似于 $('xxx').bind 的效果,少了一个利用选择器查询元素的操作。我们知道,jquery 中,event 对象会被默认当做实参传入到处理函数中,如下:

$('body').bind('click', function (event) {
  console.log(typeof event);        // object 
});

这里直接就获取到了 event 对象,那么问题来了,vue 中呢?

<div id="app">
    <button v-on:click="click">click me</button>
</div>
...
var app = new Vue({
    el: '#app',
    methods: {
        click(event) {
            console.log(typeof event);    // object
        }
    }
});

这里的实现方式看起来和 jquery 是一致的啊,但是实际上,vue 比 jquery 要要复杂得多,jquery 官方也明确的说,v-on 不简单是 addEventListener 的语法糖。在 jquery 中,我们传入到 bind 方法中的回调,只能是一个函数表类型的变量或者一个匿名函数,传递的时候,还不能执行它(在后面加上一堆圆括号),否则就变成了取这一个函数的返回值作为事件回调。而我们知道,vue 的 v-on 指令接受的值可以是函数执行的形式,比如 v-on:click="click(233)" 。这里我们可以传递任何需要传递的参数,甚至可以不传递参数:

<div id="app">
    <button v-on:click="click()">click me</button>
</div>
...
var app = new Vue({
    el: '#app',
    methods: {
        click(event) {
            console.log(typeof event);    // undefined
        }
    }
});

咦?我的 event 对象呢?怎么不见了?打印看看 arguments.length 也是 0,说明这时候确实没有实参被传入进来。T_T,那我们如果既需要传递参数,又需要用到 event 对象,这个该怎么办呢?

(三)$event

翻看 vue 文档,不难发现,其实我们可以通过将一个特殊变量 $event 传入到回调中解决这个问题:

<div id="app">
    <button v-on:click="click($event, 233)">click me</button>
</div>
...
var app = new Vue({
    el: '#app',
    methods: {
        click(event, val) {
            console.log(typeof event);    // object
        }
    }
});

好吧,这样看起来就正常了。
简单总结来说:

  1. 使用不带圆括号的形式,event 对象将被自动当做实参传入;
  2. 使用带圆括号的形式,我们需要使用 $event 变量显式传入 event 对象。

二、乌龙
前面都算是铺垫吧,现在真正的乌龙来了。
翻看小伙伴儿的代码,偶然看到了类似下面的代码:

<div id="app">
    <button v-on:click="click(233)">click me</button>
</div>
...
var app = new Vue({
    el: '#app',
    methods: {
        click(val) {
            console.log(typeof event);    // object
        }
    }
});

看到这一段代码,我的内心是崩溃的,丢进 chrome 里面一跑,尼玛还真可以,打印 arguments.length,也是正常的 1。尼玛!这是什么鬼?毁三观啊?
既没有传入实参,也没有接收的形参,这个 event 对象的来源,要么是上级作用链,要么。。。是全局作用域。。。全局的,不禁想到了 window.event
。再次上 MDN 确认了一下,果然,window.event,ie 和 chrome 都在 window 对象上有这样一个属性:

window.event

代码丢进 Firefox 中运行,event 果然就变成了 undefined 了。额,这个我也不知道说什么了。。。

相关文章

  • vue.js - 奇怪的 event 对象

    好久都没有写点东西了, 前段时间工作搞得头大,真的就是一起加班到死了。废话不多说,写这篇文章是因为这次因为 eve...

  • event 事件对象

    event 对象及使用 event 对象是什么? event 对象表示事件的状态,当 dom 树中某个事件被触发时...

  • 创建和分派事件

    event = new Event(typeArg[, eventInit]) 创建一个新的 Event 对象 ...

  • JS事件 Event

    获得event对象兼容性写法 :event || (event = window.event); 获得target...

  • 2.基础:Vue.js 组件的三个 API:prop、event

    基础:Vue.js 组件的三个 API:prop、event、slot 如果您已经对 Vue.js 组件的基础用法...

  • python停止线程?暂停?

    Python 通过threading.Event()产生一个event对象。event对象维护一个内部标志(标志初...

  • event对象

    Js的event对象用来描述Js的事件,它主要作用于IE和NN4(Netscape Navigator4)以后的各...

  • event 对象

    在触发DOM上的某个时间时,会产生一个事件对象event,这个对象中包含着所有与事件相关的信息,所有的浏览器都支持...

  • Event对象

    Event 对象代表事件的状态,比如事件在其中发生的元素、键盘按键的状态、鼠标的位置、鼠标按钮的状态。事件通常与函...

  • event对象

网友评论

  • 施主画个猿:作者挺有意思的😂
    ebfc7d0362e4:@施主画个猿 哈哈,只是发现问题比较爱瞎琢磨
  • abd84017b1e6:Event这个是有特殊意义的,你误会了
    ebfc7d0362e4:你是指哪方面?
  • d5d9940fcd48:还是没有理解。。
    ebfc7d0362e4:@IsmeJay
    第一种是没有带圆括号,注意这时在处理函数的形参中是定义了 event 的,vue 在执行的时候,自动将 event 对象作为参数传入到其中了,所以我们取到的 event 对象其实是真正传入到处理函数中的参数,一定程度上可以理解成是该函数的局部变量;
    第二种是带了圆括号的,这时的形参表里并没有定义 event 这一个参数,当前作用域并没有定义 evnet 这一个对象,所以 js 会尝试在定义这个处理函数的上级作用域中搜索 event 变量,仍未找到,最终到达全局作用域上,找到了 window.event 这一个对象,但是这一个对象并不是我们期望的 event 对象。
    准确的说,是因为这一个对象存在兼容性问题,js 代码中常见的 var event = event || window.event; 就是这个道理,查 MDN 上说只有 IE/Edge/Chrome 支持。(实际上 Safari 也支持)
    d5d9940fcd48:@江湖z 最后是如果没有传入$event,也可以获取到,而上一个就是undefined,为什么,不是都有window.event
    ebfc7d0362e4:是哪一块嘛?

本文标题:vue.js - 奇怪的 event 对象

本文链接:https://www.haomeiwen.com/subject/bepujxtx.html