1 /**
  2  * Hilo
  3  * Copyright 2015 alibaba.com
  4  * Licensed under the MIT License
  5  */
  6 
  7 /**
  8  * @class EventMixin是一个包含事件相关功能的mixin。可以通过 Class.mix(target, EventMixin) 来为target增加事件功能。
  9  * @mixin
 10  * @static
 11  * @module hilo/event/EventMixin
 12  * @requires hilo/core/Class
 13  */
 14 var EventMixin = {
 15     _listeners: null,
 16 
 17     /**
 18      * 增加一个事件监听。
 19      * @param {String} type 要监听的事件类型。
 20      * @param {Function} listener 事件监听回调函数。
 21      * @param {Boolean} once 是否是一次性监听,即回调函数响应一次后即删除,不再响应。
 22      * @returns {Object} 对象本身。链式调用支持。
 23      */
 24     on: function(type, listener, once){
 25         var listeners = (this._listeners = this._listeners || {});
 26         var eventListeners = (listeners[type] = listeners[type] || []);
 27         for(var i = 0, len = eventListeners.length; i < len; i++){
 28             var el = eventListeners[i];
 29             if(el.listener === listener) return;
 30         }
 31         eventListeners.push({listener:listener, once:once});
 32         return this;
 33     },
 34 
 35     /**
 36      * 删除一个事件监听。如果不传入任何参数,则删除所有的事件监听;如果不传入第二个参数,则删除指定类型的所有事件监听。
 37      * @param {String} type 要删除监听的事件类型。
 38      * @param {Function} listener 要删除监听的回调函数。
 39      * @returns {Object} 对象本身。链式调用支持。
 40      */
 41     off: function(type, listener){
 42         //remove all event listeners
 43         if(arguments.length == 0){
 44             this._listeners = null;
 45             return this;
 46         }
 47 
 48         var eventListeners = this._listeners && this._listeners[type];
 49         if(eventListeners){
 50             //remove event listeners by specified type
 51             if(arguments.length == 1){
 52                 delete this._listeners[type];
 53                 return this;
 54             }
 55 
 56             for(var i = 0, len = eventListeners.length; i < len; i++){
 57                 var el = eventListeners[i];
 58                 if(el.listener === listener){
 59                     eventListeners.splice(i, 1);
 60                     if(eventListeners.length === 0) delete this._listeners[type];
 61                     break;
 62                 }
 63             }
 64         }
 65         return this;
 66     },
 67 
 68     /**
 69      * 发送事件。当第一个参数类型为Object时,则把它作为一个整体事件对象。
 70      * @param {String} type 要发送的事件类型。
 71      * @param {Object} detail 要发送的事件的具体信息,即事件随带参数。
 72      * @returns {Boolean} 是否成功调度事件。
 73      */
 74     fire: function(type, detail){
 75         var event, eventType;
 76         if(typeof type === 'string'){
 77             eventType = type;
 78         }else{
 79             event = type;
 80             eventType = type.type;
 81         }
 82 
 83         var listeners = this._listeners;
 84         if(!listeners) return false;
 85 
 86         var eventListeners = listeners[eventType];
 87         if(eventListeners){
 88             eventListeners = eventListeners.slice(0);
 89             event = event || new EventObject(eventType, this, detail);
 90             if(event._stopped) return false;
 91 
 92             for(var i = 0; i < eventListeners.length; i++){
 93                 var el = eventListeners[i];
 94                 el.listener.call(this, event);
 95                 if(el.once) eventListeners.splice(i--, 1);
 96             }
 97 
 98             if(eventListeners.length == 0) delete listeners[eventType];
 99             return true;
100         }
101         return false;
102     }
103 };
104 
105 /**
106  * 事件对象类。当前仅为内部类,以后有需求的话可能会考虑独立为公开类。
107  */
108 var EventObject = Class.create({
109     constructor: function EventObject(type, target, detail){
110         this.type = type;
111         this.target = target;
112         this.detail = detail;
113         this.timeStamp = +new Date();
114     },
115 
116     type: null,
117     target: null,
118     detail: null,
119     timeStamp: 0,
120 
121     stopImmediatePropagation: function(){
122         this._stopped = true;
123     }
124 });
125 
126 //Trick: `stopImmediatePropagation` compatibility
127 var RawEvent = window.Event;
128 if(RawEvent){
129     var proto = RawEvent.prototype,
130         stop = proto.stopImmediatePropagation;
131     proto.stopImmediatePropagation = function(){
132         stop && stop.call(this);
133         this._stopped = true;
134     }
135 }
136