本文介绍一个jquery的小技巧,能让任意组件对象都能支持类似DOM的事件管理,也就是说除了派发事件,添加或删除事件监听器,还能支持事件冒泡,阻止事件默认行为等等。在jquery的帮助下,使用这个方法来管理普通对象的事件就跟管理DOM对象的事件一模一样,虽然在最后当你看到这个小技巧的具体内容时,你可能会觉得原来如此或者不过如此,但是我觉得如果能把普通的发布-订阅模式的实现改成DOM类似的事件机制,那开发出来的组件一定会有更大的灵活性和扩展性,而且我也是第一次使用这种方法(见识太浅的原因),觉得它的使用价值还蛮大的,所以就把它分享出来了。

在正式介绍这个技巧之前,得先说一下我之前考虑的一种方法,也就是发布-订阅模式,看看它能解决什么问题以及它存在的问题。

1. 发布-订阅模式

很多博客包括书本上都说javascript要实现组件的自定义事件的话,可以采用发布-订阅模式,起初我也是坚定不移地这么认为的,于是用jquery的$.Callbacks写了一个:

  1. define(function(require, exports, module) {
  2.  
  3. var $ = require('jquery');
  4. var Class = require('./class');
  5.  
  6. function isFunc(f) {
  7. return Object.prototype.toString.apply(f) === '[object Function]';
  8. }
  9.  
  10. /**
  11. * 这个基类可以让普通的类具备事件驱动的能力
  12. * 提供类似jq的on off trigger方法,不考虑one方法,也不考虑命名空间
  13. * 举例:
  14. * var e = new EventBase();
  15. * e.on('load', function(){
  16. * console.log('loaded');
  17. * });
  18. * e.trigger('load');//loaded
  19. * e.off('load');
  20. */
  21. var EventBase = Class({
  22. instanceMembers: {
  23. init: function () {
  24. this.events = {};
  25. //把$.Callbacks的flag设置成一个实例属性,以便子类可以覆盖
  26. this.CALLBACKS_FLAG = 'unique';
  27. },
  28. on: function (type, callback) {
  29. type = $.trim(type);
  30. //如果type或者callback参数无效则不处理
  31. if (!(type && isFunc(callback))) return;
  32.  
  33. var event = this.events[type];
  34. if (!event) {
  35. //定义一个新的jq队列,且该队列不能添加重复的回调
  36. event = this.events[type] = $.Callbacks(this.CALLBACKS_FLAG);
  37. }
  38. //把callback添加到这个队列中,这个队列可以通过type来访问
  39. event.add(callback);
  40. },
  41. off: function (type, callback) {
  42. type = $.trim(type);
  43. if (!type) return;
  44.  
  45. var event = this.events[type];
  46. if (!event) return;
  47.  
  48. if (isFunc(callback)) {
  49. //如果同时传递type跟callback,则将callback从type对应的队列中移除
  50. event.remove(callback);
  51. } else {
  52. //否则就移除整个type对应的队列
  53. delete this.events[type];
  54. }
  55. },
  56. trigger: function () {
  57. var args = [].slice.apply(arguments),
  58. type = args[0];//第一个参数转为type
  59.  
  60. type = $.trim(type);
  61. if (!type) return;
  62.  
  63. var event = this.events[type];
  64. if (!event) return;
  65.  
  66. //用剩下的参数来触发type对应的回调
  67. //同时把回调的上下文设置成当前实例
  68. event.fireWith(this, args.slice(1));
  69. }
  70. }
  71. });
  72.  
  73. return EventBase;
  74. });

(基于seajs以及《详解Javascript的继承实现》介绍的继承库class.js)

只要任何组件继承这个EventBase,就能继承它提供的on off trigger方法来完成消息的订阅,发布和取消订阅功能,比如我下面想要实现的这个FileUploadBaseView:

  1. define(function(require, exports, module) {
  2.  
  3. var $ = require('jquery');
  4. var Class = require('./class');
  5. var EventBase = require('./eventBase');
  6.  
  7. var DEFAULTS = {
  8. data: [], //要展示的数据列表,列表元素必须是object类型的,如[{url: 'xxx.png'},{url: 'yyyy.png'}]
  9. sizeLimit: 0, //用来限制BaseView中的展示的元素个数,为0表示不限制
  10. readonly: false, //用来控制BaseView中的元素是否允许增加和删除
  11. onBeforeRender: $.noop, //对应beforeRender事件,在render方法调用前触发
  12. onRender: $.noop, //对应render事件,在render方法调用后触发
  13. onBeforeAppend: $.noop, //对应beforeAppend事件,在append方法调用前触发
  14. onAppend: $.noop, //对应append事件,在append方法调用后触发
  15. onBeforeRemove: $.noop, //对应beforeRemove事件,在remove方法调用前触发
  16. onRemove: $.noop //对应remove事件,在remove方法调用后触发
  17. };
  18.  
  19. /**
  20. * 数据解析,给每个元素的添加一个唯一标识_uuid,方便查找
  21. */
  22. function resolveData(ctx, data){
  23. var time = new Date().getTime();
  24. return $.map(data, function(d){
  25. d._uuid = '_uuid' + time + Math.floor(Math.random() * 100000);
  26. });
  27. }
  28.  
  29. var FileUploadBaseView = Class({
  30. instanceMembers: {
  31. init: function (options) {
  32. this.base();
  33. this.options = this.getOptions(options);
  34. },
  35. getOptions: function(options) {
  36. return $.extend({}, DEFAULTS, options);
  37. },
  38. render: function(){
  39.  
  40. },
  41. append: function(data){
  42.  
  43. },
  44. remove: function(prop){
  45.  
  46. }
  47. },
  48. extend: EventBase
  49. });
  50.  
  51. return FileUploadBaseView;
  52. });

实际调用测试如下:

测试中,实例化了一个FileUploadBaseView对象f,并设置了它的name属性,通过on方法添加一个跟hello相关的监听器,最后通过trigger方法触发了hello的监听器,并传递了额外的两个参数,在监听器内部除了可以通过监听器的函数参数访问到trigger传递过来的数据,还能通过this访问f对象。

从目前的结果来说,这个方式看起来还不错,但是在我想要继续实现FileUploadBaseView的时候碰到了问题。你看我在设计这个组件的时候那几个订阅相关的option:

 
我原本的设计是:这些订阅都是成对定义,一对订阅跟某个实例方法对应,比如带before的那个订阅会在相应的实例方法(render)调用前触发,不带before的那个订阅会在相应的实例方法(render)调用后触发,而且还要求带before的那个订阅如果返回false,就不执行相应的实例方法以及后面的订阅。最后这个设计要求是考虑到在调用组件的实例方法之前,有可能因为一些特殊的原因,必须得取消当前实例方法的调用,比如调用remove方法时有的数据不能remove,那么就可以在before订阅里面做一些校验,能删除的返回true,不能删除的返回false,然后在实例方法中触发before的订阅后加一个判断就可以了,类似下面的这种做法:

但是这个做法只能在单纯的回调函数模式里实现,在发布-订阅模式下是行不通的,因为回调函数只会跟一个函数引用相关,而发布-订阅模式里,同一个消息可能有多个订阅,如果把这种做法应用到发布-订阅里面,当调用this.trigger('beforeRender')的时候,会把跟beforeRender关联的所有订阅全部调用一次,那么以哪个订阅的返回值为准呢?也许你会说可以用队列中的最后一个订阅的返回值为准,在大多数情况下也许这么干没问题,但是当我们把“以队列最后的一个订阅返回值作为判断标准”这个逻辑加入到EventBase中的时候,会出现一个很大的风险,就是外部在使用的时候,一定得清楚地管理好订阅的顺序,一定要把那个跟校验等一些特殊逻辑相关的订阅放在最后面才行,而这种跟语法、编译没有关系,对编码顺序有要求的开发方式会给软件带来比较大的安全隐患,谁能保证任何时候任何场景都能控制好订阅的顺序呢,更何况公司里面可能还有些后来的新人,压根不知道你写的东西还有这样的限制。

解决这个问题的完美方式,就是像DOM对象的事件那样,在消息发布的时候,不是简简单单的发布一个消息字符串,而是把这个消息封装成一个对象,这个对象会传递给它所有的订阅,哪个订阅里觉得应该阻止这个消息发布之后的逻辑,只要调用这个消息的preventDefault()方法,然后在外部发布完消息后,调用消息的isDefaultPrevented()方法判断一下即可:

而这个做法跟使用jquery管理DOM对象的事件是一样的思路,比如bootstrap的大部分组件以及我在前面一些博客中写的组件都是用的这个方法来增加额外的判断逻辑,比如bootstrap的alert组件在close方法执行的时候有一段这样的判断:

按照这个思路去改造EventBase是一个解决问题的方法,但是jquery的一个小技巧,能够让我们把整个普通对象的事件管理变得更加简单,下面就让我们来瞧一瞧它的庐山真面目。

2. jquery小技巧模式

1)技巧一

如果在定义组件的时候,这个组件是跟DOM对象有关联的,比如下面这种形式:

那么我们可以完全给这个组件添加on off trigger one这几个常用事件管理的方法,然后将这些方法代理到$element的相应方法上: 

通过代理,当调用组件的on方法时,其实调用的是$element的on方法,这样的话这种类型的组件就能支持完美的事件管理了。

2)技巧二

第一个技巧只能适用于跟DOM有关联的组件,对于那些跟DOM完全没有关联的组件该怎么添加像前面这样完美的事件管理机制呢?其实方法也很简单,只是我自己以前真的是没这么用过,所以这一次用起来才会觉得特别新鲜: 

看截图中框起来的部分,只要给jquery的构造函数传递一个空对象,它就会返回一个完美支持事件管理的jquery对象。而且除了事件管理的功能外,由于它是一个jquery对象。所以jquery原型上的所有方法它都能调用,将来要是需要借用jquery其它的跟DOM无关的方法,说不定也能参考这个小技巧来实现。

3. 完美的事件管理实现

考虑到第2部分介绍的2种方式里面有重复的逻辑代码,如果把它们结合起来的话,就可以适用所有的开发组件的场景,也就能达到本文标题和开篇提到的让任意对象支持事件管理功能的目标了,所以最后结合前面两个技巧,把EventBase改造如下(是不是够简单):

  1. define(function(require, exports, module) {
  2.  
  3. var $ = require('jquery');
  4. var Class = require('./class');
  5.  
  6. /**
  7. * 这个基类可以让普通的类具备jquery对象的事件管理能力
  8. */
  9. var EventBase = Class({
  10. instanceMembers: {
  11. init: function (_jqObject) {
  12. this._jqObject = _jqObject && _jqObject instanceof $ && _jqObject || $({});
  13. },
  14. on: function(){
  15. return $.fn.on.apply(this._jqObject, arguments);
  16. },
  17. one: function(){
  18. return $.fn.one.apply(this._jqObject, arguments);
  19. },
  20. off: function(){
  21. return $.fn.off.apply(this._jqObject, arguments);
  22. },
  23. trigger: function(){
  24. return $.fn.trigger.apply(this._jqObject, arguments);
  25. }
  26. }
  27. });
  28.  
  29. return EventBase;
  30. });

实际调用测试如下

1)模拟跟DOM关联的组件

测试代码一:

  1. define(function(require, exports, module) {
  2. var $ = require('jquery');
  3. var Class = require('mod/class');
  4. var EventBase = require('mod/eventBase');
  5.  
  6. var Demo = window.demo = Class({
  7. instanceMembers: {
  8. init: function (element,options) {
  9. this.$element = $(element);
  10. this.base(this.$element);
  11.  
  12. //添加监听
  13. this.on('beforeRender', $.proxy(options.onBeforeRender, this));
  14. this.on('render', $.proxy(options.onRender, this));
  15. },
  16. render: function () {
  17. //触发beforeRender事件
  18. var e = $.Event('beforeRender');
  19. this.trigger(e);
  20. if(e.isDefaultPrevented())return;
  21. //主要逻辑代码
  22. console.log('render complete!');
  23. //触发render事件
  24. this.trigger('render');
  25. }
  26. },
  27. extend: EventBase
  28. });
  29.  
  30. var demo = new Demo('#demo', {
  31. onBeforeRender: function(e) {
  32. console.log('beforeRender event triggered!');
  33. },
  34. onRender: function(e) {
  35. console.log('render event triggered!');
  36. }
  37. });
  38.  
  39. demo.render();
  40. });

在这个测试里, 我定义了一个跟DOM关联的Demo组件并继承了EventBase这个事件管理的类,给beforeRender事件和render事件都添加了一个监听,render方法中也有打印信息来模拟真实的逻辑,实例化Demo的时候用到了#demo这个DOM元素,最后的测试结果是:

完全与预期一致。

测试代码二:

  1. define(function(require, exports, module) {
  2. var $ = require('jquery');
  3. var Class = require('mod/class');
  4. var EventBase = require('mod/eventBase');
  5.  
  6. var Demo = window.demo = Class({
  7. instanceMembers: {
  8. init: function (element,options) {
  9. this.$element = $(element);
  10. this.base(this.$element);
  11.  
  12. //添加监听
  13. this.on('beforeRender', $.proxy(options.onBeforeRender, this));
  14. this.on('render', $.proxy(options.onRender, this));
  15. },
  16. render: function () {
  17. //触发beforeRender事件
  18. var e = $.Event('beforeRender');
  19. this.trigger(e);
  20. if(e.isDefaultPrevented())return;
  21. //主要逻辑代码
  22. console.log('render complete!');
  23. //触发render事件
  24. this.trigger('render');
  25. }
  26. },
  27. extend: EventBase
  28. });
  29.  
  30. var demo = new Demo('#demo', {
  31. onBeforeRender: function(e) {
  32. console.log('beforeRender event triggered!');
  33. },
  34. onRender: function(e) {
  35. console.log('render event triggered!');
  36. }
  37. });
  38.  
  39. demo.on('beforeRender', function(e) {
  40. e.preventDefault();
  41. console.log('beforeRender event triggered 2!');
  42. });
  43.  
  44. demo.on('beforeRender', function(e) {
  45. console.log('beforeRender event triggered 3!');
  46. });
  47.  
  48. demo.render();
  49. });

在这个测试了, 我定义了一个跟DOM相关的Demo组件并继承了EventBase这个事件管理的类,给beforeRender事件添加了3个监听,其中一个有加prevetDefault()的调用,而且该回调还不是最后一个,最后的测试结果是:

从结果可以看到,render方法的主要逻辑代码跟后面的render事件都没有执行,所有beforeRender的监听器都执行了,说明e.preventDefault()生效了,而且它没有对beforeRender的事件队列产生影响。

2)模拟跟DOM无关联的普通对象

测试代码一:

  1. define(function(require, exports, module) {
  2. var $ = require('jquery');
  3. var Class = require('mod/class');
  4. var EventBase = require('mod/eventBase');
  5.  
  6. var Demo = window.demo = Class({
  7. instanceMembers: {
  8. init: function (options) {
  9. this.base();
  10.  
  11. //添加监听
  12. this.on('beforeRender', $.proxy(options.onBeforeRender, this));
  13. this.on('render', $.proxy(options.onRender, this));
  14. },
  15. render: function () {
  16. //触发beforeRender事件
  17. var e = $.Event('beforeRender');
  18. this.trigger(e);
  19. if(e.isDefaultPrevented())return;
  20. //主要逻辑代码
  21. console.log('render complete!');
  22. //触发render事件
  23. this.trigger('render');
  24. }
  25. },
  26. extend: EventBase
  27. });
  28.  
  29. var demo = new Demo({
  30. onBeforeRender: function(e) {
  31. console.log('beforeRender event triggered!');
  32. },
  33. onRender: function(e) {
  34. console.log('render event triggered!');
  35. }
  36. });
  37.  
  38. demo.render();
  39. });

在这个测试里, 我定义了一个跟DOM无关的Demo组件并继承了EventBase这个事件管理的类,给beforeRender事件和render事件都添加了一个监听,render方法中也有打印信息来模拟真实的逻辑,最后的测试结果是:

完全与预期的一致。

测试代码二:

  1. define(function(require, exports, module) {
  2. var $ = require('jquery');
  3. var Class = require('mod/class');
  4. var EventBase = require('mod/eventBase');
  5.  
  6. var Demo = window.demo = Class({
  7. instanceMembers: {
  8. init: function (options) {
  9. this.base();
  10.  
  11. //添加监听
  12. this.on('beforeRender', $.proxy(options.onBeforeRender, this));
  13. this.on('render', $.proxy(options.onRender, this));
  14. },
  15. render: function () {
  16. //触发beforeRender事件
  17. var e = $.Event('beforeRender');
  18. this.trigger(e);
  19. if(e.isDefaultPrevented())return;
  20. //主要逻辑代码
  21. console.log('render complete!');
  22. //触发render事件
  23. this.trigger('render');
  24. }
  25. },
  26. extend: EventBase
  27. });
  28.  
  29. var demo = new Demo({
  30. onBeforeRender: function(e) {
  31. console.log('beforeRender event triggered!');
  32. },
  33. onRender: function(e) {
  34. console.log('render event triggered!');
  35. }
  36. });
  37.  
  38. demo.on('beforeRender', function(e) {
  39. e.preventDefault();
  40. console.log('beforeRender event triggered 2!');
  41. });
  42.  
  43. demo.on('beforeRender', function(e) {
  44. console.log('beforeRender event triggered 3!');
  45. });
  46.  
  47. demo.render();
  48. });

在这个测试了, 我定义了一个跟DOM无关的Demo组件并继承了EventBase这个事件管理的类,给beforeRender事件添加了3个监听,其中一个有加prevetDefault()的调用,而且该回调还不是最后一个,最后的测试结果是:

从结果可以看到,render方法的主要逻辑代码跟后面的render事件都没有执行,所有beforeRender的监听器都执行了,说明e.preventDefault()生效了,而且它没有对beforeRender的事件队列产生影响。

所以从2个测试来看,通过改造后的EventBase,我们得到了一个可以让任意对象支持jquery事件管理机制的方法,将来在考虑用事件机制来解耦的时候,就不用再去考虑前面第一个介绍的发布-订阅模式了,而且相对而言这个方法功能更强更稳定,也更符合你平常使用jquery操作DOM的习惯。

4. 本文小结

有2点需要再说明一下的是:

1)即使不用jquery按照第1部分最后提出的思路,把第一部分常规的发布-订阅模式改造一下也可以的,只不过用jquery更加简洁些;

2)最终用jquery 的事件机制来实现任意对象的事件管理,一方面是用到了代理模式,更重要的还是要用发布-订阅模式,只不过最后的这个实现是由jquery帮我们把第一部分的发布-订阅实现改造好了而已。

最后真切地希望这篇分享能够给你的工作带来一些帮助,谢谢阅读:)


补充于2016-04-08:

自定义事件的名称,有的时候会跟jquery内部的一些事件名称冲突,我遇到一种情况:定义一个跟DOM关联的组件时,我用到了一个自定义事件remove,当我调用this.trigger('remove')的时候,竟然把这个组件关联DOM元素从DOM中删除了,估计这个事件名称已经在jquery内部使用,当在DOM元素上触发这个事件后,会导致元素remove。所以当你碰到类似的意外情况时,回头想想自定义事件的名称是不是跟jquery某些API有关联,然后试试换一个名称是不是就能解决问题。

jquery技巧之让任何组件都支持类似DOM的事件管理的更多相关文章

  1. Jquery图片上传组件,支持多文件上传

    Jquery图片上传组件,支持多文件上传http://www.jq22.com/jquery-info230jQuery File Upload 是一个Jquery图片上传组件,支持多文件上传.取消. ...

  2. jQuery UI Autocomplete是jQuery UI的自动完成组件(share)

    官网:http://jqueryui.com/autocomplete/ 以下分享自:http://www.cnblogs.com/yuzhongwusan/archive/2012/06/04/25 ...

  3. jquery技巧总结

    jquery技巧总结一.简介 1.1.概述随着WEB2.0及ajax思想在互联网上的快速发展传播,陆续出现了一些优秀的Js框架,其中比较著名的有Prototype.YUI.jQuery.mootool ...

  4. 周末大礼:jQuery技巧总结

    一.简介 1.1.概述 随着WEB2.0及ajax思想在互联网上的快速发展传播,陆续出现了一些优秀的Js框架,其中比较著名的有Prototype.YUI.jQuery.mootools.Bindows ...

  5. 第二百二十六节,jQuery EasyUI,Tree(树)组件

    jQuery EasyUI,Tree(树)组件 本节课重点了解 EasyUI 中 Tree(树)组件的使用方法,这个组件依赖于 Draggable(拖 动)和 Droppable(放置)组件. 一.加 ...

  6. JQuery 技巧总结

    一.简介 1.1.概述 随着WEB2.0及ajax思想在互联网上的快速发展传播,陆续出现了一些优秀的Js框架,其中比较著名的有Prototype.YUI.jQuery.mootools.Bindows ...

  7. 风云流水 jQuery技巧总结 (转)

    jQuery技巧总结 (转) 一.简介 1.1.概述 随着WEB2.0及ajax思想在互联网上的快速发展传播,陆续出现了一些优秀的Js框架,其中比较著名的有Prototype.YUI.jQuery.m ...

  8. jQuery技巧大放送

    1.关于页面元素的引用 通过jquery的$()引用元素包括通过id.class.元素名以及元素的层级关系及dom或者xpath条件等方法,且返回的对象为jquery对象(集合对象),不能直接调用do ...

  9. 基于Bootstrap的JQuery TreeView树形控件,数据支持json字符串、list集合(MVC5)<二>

    上篇博客给大家介绍了基于Bootstrap的JQuery TreeView树形控件,数据支持json字符串.list集合(MVC5)<一>, 其中的两种方式都显得有些冗余.接着上篇博客继续 ...

随机推荐

  1. 多文档上传(upload multiple documents)功能不能使用怎么办?

    问题描述: 在SharePoint 2010的文档库里选择documents标签,然后选择upload document下拉菜单,你会发现upload multiple documents那个按钮是灰 ...

  2. 你的应用是如何被替换的,App劫持病毒剖析

    一.App劫持病毒介绍 App劫持是指执行流程被重定向,又可分为Activity劫持.安装劫持.流量劫持.函数执行劫持等.本文将对近期利用Acticity劫持和安装劫持的病毒进行分析. 二.Activ ...

  3. Code First系列之视图,存储过程和异步API

    返回<8天掌握EF的Code First开发>总目录 本篇目录 视图View 存储过程 使用存储过程CRUD 异步API 本章小结 自我测试 本系列的源码本人已托管于coding上:点击查 ...

  4. ABP理论学习之Abp Session

    返回总目录 本篇目录 介绍 注入Session 使用Session属性 介绍 当应用程序要求用户登录时,那么应用程序也需要知道当前用户正在执行的操作.虽然ASP.NET本身在展现层提供了Session ...

  5. ASP.NET MVC 5 - 查询Details和Delete方法

    在这部分教程中,接下来我们将讨论自动生成的Details和Delete方法. 查询Details和Delete方法 打开Movie控制器并查看Details方法. public ActionResul ...

  6. WCF basicHttpBinding之Message Security Mode

    原创地址:http://www.cnblogs.com/jfzhu/p/4067873.html 转载请注明出处 前面的文章<WCF Security基本概念>介绍了WCF的securit ...

  7. java stopwatch 功能

    C#中有一个stopwatch的功能,主要是用来监测程序执行时间的.java之前一直都在用如下方式完成: public static void main(String[] args) { long s ...

  8. JBoss AS

    即 JBoss Application Server - JBoss Community The JBoss AS community project has been renamed to the ...

  9. web应用中使用JavaMail发送邮件

    现在很多的网站都提供有用户注册功能, 通常我们注册成功之后就会收到一封来自注册网站的邮件.邮件里面的内容可能包含了我们的注册的用户名和密码以及一个激活账户的超链接等信息.今天我们也来实现一个这样的功能 ...

  10. LINQ系列:Linq to Object联接操作符

    联接是指将一个数据源对象与另一个数据源对象进行关联或联合的操作.这两个数据源对象通过一个共同的值或属性进行关联. LINQ的联接操作符将包含可匹配(或相同)关键字的两个或多个数据源中的值进行匹配. L ...