javascript的事件触发器介绍的实现


Posted in Javascript onJune 05, 2014

事件触发器从字面意思上可以很好的理解,就是用来触发事件的,但是有些没有用过的朋友可能就会迷惑了,事件不是通常都由用户在页面上的实际操作来触发的吗?这个观点不完全正确,因为有些事件必须由程序来实现,如自定义事件,jQuery的ajax框架的一些自定义事件就必须由事件触发器来实现。当然,在一些特殊情况下,用事件触发器来触发事件比用户的实际操作来触发事件更方便。

对于实现事件触发器,浏览器都有原生的方法来支持,但是在兼容性上又有很大的出入,这种兼容性的问题完全在意料之中,IE有自己的方法,其他标准浏览器也有一套方法,不说谁的方法好与不好,对于WEB开发者来说搞出几套方法就是对开发人员的一种折磨。IE支持fireEvent方法来实现事件触发,标准浏览器支持dispatchEvent来实现事件触发,两面派的IE9是两者都支持。下面是出自prototype.js的源码(其实我是从司徒正美的博客复制过来的):

var fireEvent = function(element,event){
 if (document.createEventObject){
  // IE浏览器支持fireEvent方法
  var evt = document.createEventObject();
  return element.fireEvent('on'+event,evt)
 }
 else{
  // 其他标准浏览器使用dispatchEvent方法
  var evt = document.createEvent( 'HTMLEvents' );
  // initEvent接受3个参数:
  // 事件类型,是否冒泡,是否阻止浏览器的默认行为
  evt.initEvent(event, true, true);  
  return !element.dispatchEvent(evt);
 }
};

上面的方法可以兼容主流的浏览器以实现事件触发器的功能。但是对于一些封装好的事件处理系统,如jQuery的event模块,就没这么简单了,只能通过模拟来实现了。我在之前写过一个很简单的事件处理系统,最近又碰到自定义事件的需求,于是在之前的事件系统的基础上模拟了一个事件触发器,代码如下:

/**
 * 事件触发器
 * @param { Object } DOM元素
 * @param { String / Object } 事件类型 / event对象
 * @param { Array }  传递给事件处理函数的附加参数
 * @param { Boolean } 是否冒泡
 */
trigger : function( elem, event, data, isStopPropagation ){
 var type = event.type || event,
  // 冒泡的父元素,一直到document、window
  parent = elem.parentNode || 
   elem.ownerDocument || 
   elem === elem.ownerDocument && win,
  eventHandler = $.data( elem, type + 'Handler' ); isStopPropagation = typeof data === 'boolean' ? 
  data : (isStopPropagation || false);
 data = data && isArray( data ) ? data : [];
 // 创建自定义的event对象  
 event = typeof event === 'object' ? 
  event : {
   type : type,
   preventDefault : noop,
   stopPropagation : function(){
    isStopPropagation = true;
   }
  };
 event.target = elem;  
 data.unshift( event );
 if( eventHandler ){
  eventHandler.call( elem, data );
 }
 // 递归调用自身来模拟冒泡
 if( parent && !isStopPropagation ){
  data.shift();
  this.trigger( parent, event, data );
 }
}

模拟的原理并不难,给某元素绑定一个事件处理函数,如果有触发事件的实际操作就会执行相应的事件处理函数,所以要达到事件触发器的功能只要获取到相应的事件处理函数然后执行就差不多了,这是最基本的。

在实际的事件发生时浏览器会生成一个event对象,里面包含了一些事件发生时的属性和信息。如果没有实际的事件发生是没有这个event对象的,所以上面的代码也创建了一个event对象满足最基本的功能。

还有事件冒泡,如果没有实际的事件发生,自然也不会有冒泡的行为,那么如果要模拟冒泡的功能,就必须不断的查找父元素并检查是否绑定了相同类型的事件,直至到document和window为止,如果结构复杂,这种递归调用的方法性能估计会不怎么样。

最后是浏览器的默认行为,我觉得这个要去模拟相当麻烦,麻烦到不知如何去实现,比如a标签默认的跳转,我测试了jQuery的trigger,也没有实现,但是一些其他的行为貌似又在API手册中有介绍。毕竟这个功能不是很重要,暂时也没做过多的深究。

 

 

Javascript 相关文章推荐
验证码在IE中不刷新而谷歌等浏览器正常的解决方案
Mar 18 Javascript
JavaScript实现的使用键盘控制人物走动实例
Aug 27 Javascript
JQuery入门基础小实例(1)
Sep 17 Javascript
jQuery fadeOut 异步实例代码详解
Aug 18 Javascript
Angular+Bootstrap+Spring Boot实现分页功能实例代码
Jul 21 Javascript
几个你不知道的技巧助你写出更优雅的vue.js代码
Jun 11 Javascript
前端axios下载excel文件(二进制)的处理方法
Jul 31 Javascript
解决vue props 拿不到值的问题
Sep 11 Javascript
javascript面向对象创建对象的方式小结
Jul 29 Javascript
Vue+Element ui 根据后台返回数据设置动态表头操作
Sep 21 Javascript
JS继承最简单的理解方式
Mar 31 Javascript
Ajax实现局部刷新的方法实例
Mar 31 Javascript
javascript中attribute和property的区别详解
Jun 05 #Javascript
使用jquery实现的一个图片延迟加载插件(含图片延迟加载原理)
Jun 05 #Javascript
用js的document.write输出的广告无阻塞加载的方法
Jun 05 #Javascript
javascript数组去重方法终极总结
Jun 05 #Javascript
javascript设计模式之解释器模式详解
Jun 05 #Javascript
javascript监听鼠标滚轮事件浅析
Jun 05 #Javascript
详解JavaScript语法对{}处理的坑爹之处
Jun 05 #Javascript
You might like
php win下Socket方式发邮件类
2009/08/21 PHP
php输入流php://input使用示例(php发送图片流到服务器)
2013/12/25 PHP
PHP使用DOMDocument类生成HTML实例(包含常见标签元素)
2014/06/25 PHP
destoon首页调用求购供应信息的地区名称的方法
2014/08/21 PHP
Laravel如何友好的修改.env配置文件详解
2017/06/07 PHP
php生成毫秒时间戳的实例讲解
2017/09/22 PHP
javascript中的prototype属性使用说明(函数功能扩展)
2010/08/16 Javascript
JavaScript 注册事件代码
2011/01/27 Javascript
javascript中的对象创建 实例附注释
2011/02/08 Javascript
由JavaScript中call()方法引发的对面向对象继承机制call的思考
2011/09/12 Javascript
用JavaScript实现动画效果的方法
2013/07/20 Javascript
js中return false(阻止)的用法
2013/08/14 Javascript
js与运算符和或运算符的妙用
2014/02/14 Javascript
node.js中的fs.fstatSync方法使用说明
2014/12/15 Javascript
jQuery插件实现控制网页元素动态居中显示
2015/03/24 Javascript
JQuery插件ajaxfileupload.js异步上传文件实例
2015/05/19 Javascript
Bootstrap modal使用及点击外部不消失的解决方法
2016/12/13 Javascript
js下拉菜单生成器dropMenu使用方法详解
2017/08/01 Javascript
详解react服务端渲染(同构)的方法
2017/09/21 Javascript
解决Vue+Electron下Vuex的Dispatch没有效果问题
2019/05/20 Javascript
使用Phantomjs和Node完成网页的截屏快照的方法
2019/07/16 Javascript
小程序实现左滑删除的效果的实例代码
2020/10/19 Javascript
python格式化字符串实例总结
2014/09/28 Python
Tensorflow简单验证码识别应用
2017/05/25 Python
python实现任意位置文件分割的实例
2018/12/14 Python
33个Python爬虫项目实战(推荐)
2019/07/08 Python
Python基于BeautifulSoup和requests实现的爬虫功能示例
2019/08/02 Python
解决django后台管理界面添加中文内容乱码问题
2019/11/15 Python
Python使用正则表达式实现爬虫数据抽取
2020/08/17 Python
民生工程实施方案
2014/03/22 职场文书
幼儿园春季开学寄语
2014/04/03 职场文书
健康教育评估方案
2014/05/25 职场文书
公司2015年终工作总结
2015/05/26 职场文书
运动会通讯稿50字
2015/07/20 职场文书
纯 CSS 自定义多行省略的问题(从原理到实现)
2021/11/11 HTML / CSS
mysql字段为NULL索引是否会失效实例详解
2022/05/30 MySQL