8 * 1. as3 の EventDispatcher ライクなクラス。そのまま使ったり、継承したり。コールバック中にイベントを追加したら?削除したら?にも対処している。
\r
9 * 2. _rawObject メンバがいる場合、addEventListener, attachEvent, on 等で生のブラウザオブジェクトにリスナを登録する。
\r
10 * window, document, HTMLElement, Image, XHR などが _rawObject
\r
14 * https://developer.mozilla.org/ja/docs/Web/API/EventTarget.addEventListener
\r
16 * EventListener がイベント処理中に EventTarget に追加された場合、それが現在のアクションによって実行されることはありませんが、浮上フェーズのように、後の段階のイベントフローで実行されるかもしれません。
\r
18 * https://developer.mozilla.org/ja/docs/Web/API/EventTarget.removeEventListener
\r
19 * イベントリスナーが イベントを処理中であるイベントターゲットから削除された場合、現在のアクションによってそのイベントリスナーが実行されることはありません。
\r
20 * イベントリスナーは、決して削除された後に実行されることはありません。
\r
21 * イベントターゲット上にある現在のどのイベントリスナーも指定していない引数付きの removeEventListener は、何の効果もありません。
\r
24 // ------------------------------------------------------------------------- //
\r
25 // ------------ local variables -------------------------------------------- //
\r
26 // ------------------------------------------------------------------------- //
\r
27 var X_EventDispatcher_once = false,
\r
28 X_EventDispatcher_lock = false,
\r
29 X_EventDispatcher_unlock = false,
\r
30 X_EventDispatcher_needsIndex = false,
\r
32 X_EventDispatcher_safariPreventDefault = false, // Safari3-
\r
34 X_EventDispatcher_EVENT_TARGET_TYPE = {
\r
40 X_EventDispatcher_LAZY_TIMERS = {}; // Object.<number, X.EventDispatcher> number は timerID
\r
43 * イベントリスナをイベント名(string)や数値(1~,フレームワーク内で定義)をキーとするArrayで記憶します。
\r
44 * Arrayには、{k:種類,x:コンテキスト(thisObject),f:関数,s:サプリメントする引数の配列} というハッシュ、または関数が蓄えられています。
\r
45 * また、dispatch 中の状態と操作を記録し不整合が起きないようにするためのプロパティを持ちます。
\r
48 * _handleEvent : function,
\r
49 * _dispatching : number,
\r
50 * _reserves : (Array|undefined),
\r
51 * _unlistens : {Object.<(number|string), Array.<(X.Callback|function)>>},
\r
52 * _killReserved : (Boolean|undefiend)
\r
55 * Object.<(number|string), Array.<(callbackHash|function)>>
\r
58 var X_EventDispatcher_listeners;
\r
60 // ------------------------------------------------------------------------- //
\r
61 // --- interface ----------------------------------------------------------- //
\r
62 // ------------------------------------------------------------------------- //
\r
65 * <p>イベントターゲット(widnow, document, Image, XHR, Silverlight 等)をラップする場合、通常は new 時に渡します。
\r
66 * <p>参照:コンストラクタ実体 {@link X.EventDispatcher.Constructor}
\r
67 * <p>アプリケーション独自のイベントをやり取りしたい、という場合、イベントターゲットは不要です。
\r
70 * @classdesc EventTarget オブジェクトをラップしたり、アプリケーションで独自に定義したイベントを発信するためのクラスです。
\r
71 * <p>listen, unlisten, dispatch という addEventListener, removeEventListener, dispatchEvent に対応する関数を持ちます。
\r
72 * <p>また listening という ActionScript3 の hasEventListener に相当する関数を持ちます。
\r
73 * <p>イベントターゲットオブジェクト(widnow, document, HTMLElement, XHR 等)が this._rawObject に設定されていた場合に、それらへ実際のイベント登録・解除も行います。
\r
74 * <p>このイベントの登録・解除はクロスブラウザで、IE5~8 の独自イベントの差異を吸収し、DOM0 に対しても複数のコールバックを登録することができます。
\r
75 * <p>またコールバックに対して、this コンテキストや、追加の引数を指定もできます。 this コンテキストを指定しなかった場合、EventDispatcher インスタンスがコールバックの this になります。
\r
76 * <p>unlisten() は、引数を指定しなかった場合、全てのイベントを解除します。ただし、systemListen 経由で登録されたハンドラは解除されません。
\r
77 * systemListen, systemUnlisten は、ライブラリ内のコードからしかアクセスできません。
\r
79 * @augments X_Class_CommonProps
\r
81 * @param {object=} opt_rawObject イベントターゲット(EventTarget)
\r
87 /** @lends X.EventDispatcher.prototype */
\r
91 * OTHER(Node,window,document,Image,Audio), XHR, Silverlight
\r
95 '_rawType' : X_EventDispatcher_EVENT_TARGET_TYPE.OTHER,
\r
98 * イベントリスナをイベント名(string)や数値(1~,フレームワーク内で定義)をキーとするArrayで記憶します。
\r
99 * Arrayには、{k:種類,x:コンテキスト(thisObject),f:関数,s:サプリメントする引数の配列} というハッシュ、または関数が蓄えられています。
\r
102 * @type {X_EventDispatcher_listeners}
\r
104 '_listeners' : null,
\r
107 * _rawObject には HTMLElement, window, document, XHR といったイベントターゲットオブジェクトを設定します。
\r
108 * _rawObject が設定されていると listen(), unlisten() 時に addEventListener(DOM Level2) や detachEvent(ie5~8), on~(DOM0) 等を操作します。
\r
109 * _rawObject は最初の listen() 前に設定しておかないと addEventListener 等が意図したように行われません。
\r
110 * X.Node では非同期に HTMLElement を生成していますが、要素生成以前に listen, unlisten を呼び出すことができます。これは適宜に X_EventDispatcher_toggleAllEvents を呼んで解決しているためです。
\r
114 '_rawObject' : null,
\r
117 * X.EventDispatcher のコンストラクタの実体。
\r
119 * @this {X.EventDispatcher}
\r
120 * @param {object=} opt_rawObject
\r
122 Constructor : function( opt_rawObject ){
\r
123 if( opt_rawObject ){
\r
124 this._rawObject = opt_rawObject;
\r
129 * 登録されたイベントリスナを呼び出す。イベントリスナの返り値(数値)を OR したものを返す。
\r
130 * @this {X.EventDispatcher}
\r
132 * @param {(eventHash|string|number)} e
\r
134 dispatch : X_EventDispatcher_dispatch,
\r
138 * @this {X.EventDispatcher}
\r
139 * @param {(string|number|Array.<(string,number)>)} type
\r
140 * @param {(listener|function|Array)=} opt_arg1
\r
141 * @param {(function|Array=} opt_arg2
\r
142 * @param {Array=} opt_arg3
\r
143 * @return {X.EventDispatcher}
\r
145 listen : X_EventDispatcher_listen,
\r
148 * 一度 dispatch された時に自動で unlisten されるフラグを立てて listen する。
\r
149 * @this {X.EventDispatcher}
\r
150 * @return {X.EventDispatcher}
\r
151 * @param {(string|number|Array.<(string,number)>)} type
\r
152 * @param {(listener|function|Array)=} opt_arg1
\r
153 * @param {(function|Array=} opt_arg2
\r
154 * @param {Array=} opt_arg3
\r
156 listenOnce : function( type, opt_arg1, opt_arg2, opt_arg3 ){
\r
157 X_EventDispatcher_once = true;
\r
158 this.listen( type, opt_arg1, opt_arg2, opt_arg3 );
\r
159 X_EventDispatcher_once = false;
\r
163 unlisten : X_EventDispatcher_unlisten,
\r
166 * <p>イベントリスナの登録状況を真偽値で返す。戻り値が数値(index)の場合もあるが、これは内部のみで使用。
\r
167 * <p>this.listening(); のように type を省略した場合、一つでも登録があれば true を返す。
\r
168 * <p>this.listening( 'myevent' ); と type だけを与えた場合、その type に登録があれば true を返す。
\r
169 * <p>type と イベントリスナの組み合わせが登録されているかを調べる場合は、listen 時の thisObject や args(Array) も一致させて渡す必要がある。
\r
172 * this.listen( [ 'myevent', 'yourevent' ], this, onMyEvent, args = [ 1, 'a' ] );
\r
173 * this.listening( 'myevent', this, onMyEvent, args ) === true;
\r
175 * @this {X.EventDispatcher}
\r
176 * @return {(number|boolean)}
\r
177 * @param {(string|number)=} opt_type
\r
178 * @param {(listener|function|Array|callbackHash)=} opt_arg1
\r
179 * @param {(function|Array=} opt_arg2
\r
180 * @param {Array=} opt_arg3
\r
182 listening : function( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
\r
183 var listeners = this[ '_listeners' ],
\r
184 lock = X_EventDispatcher_lock || X_EventDispatcher_unlock,
\r
185 list, cbHash, unlistens, i, f;
\r
187 if( opt_type === undefined ) return !!listeners;
\r
188 if( !listeners || !( list = listeners[ opt_type ] ) ) return false;
\r
189 if( opt_arg1 === undefined ) return true;
\r
194 cbHash = X_Callback_classifyCallbackArgs( opt_arg1, opt_arg2, opt_arg3, this );
\r
197 if( ( unlistens = listeners._unlistens ) && ( unlistens = unlistens[ opt_type ] ) ){
\r
198 for( i = unlistens.length; i; ){
\r
199 f = unlistens[ --i ];
\r
200 if( f === cbHash || ( f.x === cbHash.x && f.f === cbHash.f && f.s === cbHash.s && f.lock === lock ) ) return false;
\r
203 for( i = list.length; i; ){
\r
205 if( f === cbHash || ( f.x === cbHash.x && f.f === cbHash.f && f.s === cbHash.s && f.lock === lock ) ){
\r
206 // index を要求された場合、lock されていない、または unlock なら index を返す
\r
207 return X_EventDispatcher_needsIndex ? i : true;
\r
214 * delay(ミリ秒)後にイベントを dispatch する。戻り値は uid = X.Timer.add() のタイマーID(数値)。X.Timer.remove(uid) でタイマーを解除して dispatch を中止できる。
\r
215 * @this {X.EventDispatcher}
\r
216 * @param {(number|eventHash|string)=} delay ms 省略した場合は 0 として扱う asyncDispatch( 'myevent' ) -> asyncDispatch( 0, 'myevent' )
\r
217 * @param {(eventHash|string|number)=} e イベントを表す数値、文字列、{ type : XXX, ... } なオブジェクト
\r
218 * @return {number} X.Timer.add() の戻り値
\r
220 asyncDispatch : function( delay, e ){
\r
222 if( delay && e === undefined ){
\r
226 timerID = X.Timer.add( delay, 1, this, X_EventDispatcher_dispatch, [ e ] );
\r
227 X_EventDispatcher_LAZY_TIMERS[ timerID ] = this;
\r
233 // ------------------------------------------------------------------------- //
\r
234 // --- implements ---------------------------------------------------------- //
\r
235 // ------------------------------------------------------------------------- //
\r
238 * 登録されたイベントリスナを呼び出す。イベントリスナの返り値(数値)を OR したものを返す。イベントハッシュでなく、string|number を渡すと内部でイベントハッシュを作る
\r
239 * @this {X.EventDispatcher}
\r
240 * @return {number} X.Callback で定義された数値
\r
241 * @param {(eventHash|string|number)} e
\r
243 function X_EventDispatcher_dispatch( e ){
\r
244 var listeners = this[ '_listeners' ],
\r
245 ret = X_Callback_NONE,
\r
246 type = e[ 'type' ],
\r
247 list, unlistens, i, l, args, f, r, sysOnly, timerID;
\r
249 if( !listeners || !( list = listeners[ type || e ] ) ) return ret;
\r
253 e = { type : type = e };
\r
255 e.target = e.target || this;
\r
256 e.currentTarget = e.currentTarget || this;
\r
258 if( listeners._dispatching ){
\r
259 ++listeners._dispatching;
\r
261 listeners._dispatching = 1;
\r
266 listeners._unlistens = listeners._unlistens || {};
\r
267 unlistens = listeners._unlistens[ type ];
\r
269 for( i = 0; i < list.length; ++i ){
\r
272 unlistens = listeners._unlistens[ type ];
\r
274 if( unlistens && unlistens.indexOf( f ) !== -1 ) continue;
\r
276 //if( f !== X.emptyFunction ){
\r
278 r = X_Callback_proxyCallback( f, args || ( args = [ e ] ) );
\r
280 // r = f.call( this, e );
\r
284 if( f.once || r & X_Callback_UN_LISTEN ){
\r
285 // dispatch 中に unlisten が作られることがある
\r
287 unlistens = listeners._unlistens || ( listeners._unlistens = {} );
\r
288 unlistens = unlistens[ type ] || ( unlistens[ type ] = [] );
\r
290 unlistens.indexOf( f ) === -1 && ( unlistens[ unlistens.length ] = f );
\r
293 if( r & X.Callback.STOP_NOW ){
\r
296 ret |= X.Type.isFinite( r ) ? r : 0;
\r
299 if( ( --listeners._dispatching ) === 0 ){
\r
301 delete listeners._dispatching;
\r
303 // dispatch 中に listen されたイベントの追加
\r
304 if( list = listeners._reserves ){
\r
305 for( i = 0, l = list.length; i < l; ++i ){
\r
307 X_EventDispatcher_once = f[ 4 ];
\r
308 X_EventDispatcher_lock = f[ 5 ];
\r
309 this.listen( f[ 0 ], f[ 1 ], f[ 2 ], f[ 3 ] );
\r
310 X_EventDispatcher_once = false;
\r
311 X_EventDispatcher_lock = false;
\r
315 delete listeners._reserves;
\r
318 // dispatch 中に unlisten されたイベントの削除
\r
319 if( unlistens = listeners._unlistens ){
\r
320 delete listeners._unlistens;
\r
322 // _unlistens に入っている callbackHash は、lock をクリアしている
\r
323 X_EventDispatcher_unlock = true;
\r
324 for( type in unlistens ){
\r
325 //if( X_EMPTY_OBJECT[ type ] ) continue;
\r
326 list = unlistens[ type ];
\r
327 for( i = list.length; i; ){
\r
328 this.unlisten( type, list[ --i ] );
\r
331 delete unlistens[ type ];
\r
333 X_EventDispatcher_unlock = false;
\r
336 if( X_EventDispatcher_LAZY_TIMERS[ X_Timer_currentUID ] === this ){
\r
337 delete X_EventDispatcher_LAZY_TIMERS[ X_Timer_currentUID ];
\r
340 if( listeners._killReserved ){
\r
341 for( timerID in X_EventDispatcher_LAZY_TIMERS ){
\r
342 if( X_EventDispatcher_LAZY_TIMERS[ timerID ] === this ) return ret;
\r
353 * @this {X.EventDispatcher}
\r
354 * @memberOf X.EventDispatcher.prototype
\r
355 * @param {(string|number|Array.<(string,number)>)} type
\r
356 * @param {(listener|function|Array)=} opt_arg1
\r
357 * @param {(function|Array=} opt_arg2
\r
358 * @param {Array=} opt_arg3
\r
359 * @return {X.EventDispatcher}
\r
361 function X_EventDispatcher_listen( type, opt_arg1, opt_arg2, opt_arg3 ){
\r
362 var listeners = this[ '_listeners' ],
\r
363 i, raw, add, list, f;
\r
365 if( !type ) return this;
\r
367 if( listeners && listeners._dispatching ){
\r
368 if( !listeners._reserves ) listeners._reserves = [];
\r
369 listeners._reserves[ listeners._reserves.length ] = [ type, opt_arg1, opt_arg2, opt_arg3, X_EventDispatcher_once, X_EventDispatcher_lock ];
\r
373 if( X.Type.isArray( type ) ){
\r
374 for( i = type.length; i; ){
\r
375 this.listen( type[ --i ], opt_arg1, opt_arg2, opt_arg3 );
\r
380 raw = this._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( this );
\r
381 add = raw && ( !listeners || !listeners[ type ] ) && X.Type.isString( type );
\r
383 if( this.listening( type, opt_arg1 || this, opt_arg2, opt_arg3 ) ) return this;
\r
385 if( !listeners ) listeners = this[ '_listeners' ] = {};
\r
386 list = listeners[ type ] || ( listeners[ type ] = [] );
\r
388 add && X_EventDispatcher_addEvent( this, type, raw, list );
\r
390 f = X_Callback_classifyCallbackArgs( opt_arg1, opt_arg2, opt_arg3, this );
\r
391 list[ list.length ] = f;
\r
392 f.once = X_EventDispatcher_once;
\r
393 f.lock = X_EventDispatcher_lock;
\r
399 * X_EventDispatcher_systemUnlisten 経由でないと解除できないリスナの登録
\r
401 function X_EventDispatcher_systemListen( that, type, opt_arg1, opt_arg2, opt_arg3 ){
\r
402 X_EventDispatcher_lock = true;
\r
403 that.listen( type, opt_arg1, opt_arg2, opt_arg3 );
\r
404 X_EventDispatcher_lock = false;
\r
407 // TODO this.listen(type) は this リスナの登録なのに、this.unlisten(type)は全てのtypeの削除、と不一致
\r
411 * @this {X.EventDispatcher}
\r
412 * @return {X.EventDispatcher}
\r
413 * @param {(string|number|Array.<(string,number)>)=} opt_type
\r
414 * @param {(listener|function|Array)=} opt_arg1
\r
415 * @param {(function|Array=} opt_arg2
\r
416 * @param {Array=} opt_arg3
\r
418 function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
\r
419 var listeners = this[ '_listeners' ],
\r
420 list, reserves, unlistens, i, f, raw, k, empty;
\r
421 if( !listeners ) return this;
\r
423 if( X.Type.isArray( opt_type ) ){
\r
424 for( i = opt_type.length; i; ){
\r
425 this.unlisten( opt_type[ --i ], opt_arg1, opt_arg2, opt_arg3 );
\r
426 if( !opt_type[ i ] ){
\r
427 alert( '不正な unlisten Array' );
\r
433 if( opt_type === undefined ){
\r
435 for( opt_type in listeners ){
\r
436 //if( X_EMPTY_OBJECT[ opt_type ] ) continue;
\r
437 if( opt_type.charAt( 0 ) === '_' ) continue;
\r
438 list = listeners[ opt_type ];
\r
439 for( i = list.length; i; ){
\r
440 this.unlisten( opt_type, list[ --i ] ); // override されていることがあるので、必ず unlisten を使用
\r
442 // this.unlisten( opt_type ); これは無茶!
\r
446 if( opt_arg1 === undefined ){
\r
448 if( list = listeners[ opt_type ] ){
\r
449 for( i = list.length; i; ){
\r
450 this.unlisten( opt_type, list[ --i ] ); // override されていることがあるので、必ず unlisten を使用
\r
455 if( reserves = listeners._reserves ){
\r
456 for( i = reserves.length; i; ){
\r
457 f = reserves[ --i ];
\r
458 if( f[ 0 ] === opt_type && f[ 1 ] === opt_arg1 && f[ 2 ] === opt_arg2 && f[ 3 ] === opt_arg3 && ( !f[ 5 ] || X_EventDispatcher_unlock ) ){
\r
459 reserves.splice( i, 1 );
\r
460 if( !reserves.legth ) delete listeners._reserves;
\r
466 X_EventDispatcher_needsIndex = true;
\r
467 i = this.listening( opt_type, opt_arg1, opt_arg2, opt_arg3 );
\r
468 X_EventDispatcher_needsIndex = false;
\r
469 if( i === false ) return this;
\r
471 f = ( list = listeners[ opt_type ] )[ i ];
\r
473 if( unlistens = listeners._unlistens ){
\r
474 // _unlistens に入っている callbackHash は、lock のチェックは済んでいる
\r
475 ( unlistens = unlistens[ opt_type ] ) ?
\r
476 ( unlistens[ unlistens.length ] = f ) :
\r
477 ( listeners._unlistens[ opt_type ] = [ f ] );
\r
480 list.splice( i, 1 );
\r
481 if( !list.length ){
\r
482 raw = this._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( this );
\r
483 delete listeners[ opt_type ];
\r
484 //empty = X_Object_isEmpty( listeners );
\r
487 for( k in listeners ){
\r
488 if( k.charAt( 0 ) === '_' ) continue;
\r
492 if( raw && '' + parseFloat( opt_type ) !== '' + opt_type ){ // 数字イベントの除外
\r
493 X_EventDispatcher_removeEvent( this, opt_type, raw, list, !empty );
\r
495 if( empty ) delete this[ '_listeners' ];
\r
502 * X_EventDispatcher_systemListen から登録したイベントの解除
\r
504 function X_EventDispatcher_systemUnlisten( that, type, opt_arg1, opt_arg2, opt_arg3 ){
\r
505 X_EventDispatcher_unlock = true;
\r
506 that.unlisten( type, opt_arg1, opt_arg2, opt_arg3 );
\r
507 X_EventDispatcher_unlock = false;
\r
510 function X_EventDispatcher_addEvent( that, type, raw, list ){
\r
512 X_EventDispatcher_lock || ( type = X_Event_Rename[ type ] || type );
\r
514 if( X.Type.isArray( type ) ){
\r
515 for( i = type.length; i; ){
\r
516 X_EventDispatcher_systemListen( that, type[ --i ], X.emptyFunction );
\r
517 console.log( 'events fix > ' + type[ i ] );
\r
520 X_EventDispatcher_actualAddEvent( that, type, raw, list );
\r
524 var X_EventDispatcher_actualAddEvent =
\r
525 // Days on the Moon DOM Events とブラウザの実装
\r
526 // http://nanto.asablo.jp/blog/2007/03/23/1339502
\r
527 // Safari 2 では関数オブジェクトしか EventListener として使えませんが、Safari のナイトリービルドでは handleEvent メソッドを持つオブジェクトも EventListener として使えるようです。
\r
529 (function( that, type, raw, list ){
\r
531 switch( that[ '_rawType' ] ){
\r
532 case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :
\r
533 list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );
\r
534 list.sltoken = raw.AddEventListener( type, list.slcallback );
\r
537 case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :
\r
538 if( X_UA.Opera < 12 ){
\r
539 // Opera11- の XHR は event オブジェクトが返らないため, eventType 毎に callback を指定する addEventListener もない
\r
540 raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );
\r
545 // iOS と MacOSX Iron36 で発生。連続してアニメーションが起こると、クロージャの束縛された obj へのアクセスに失敗する。Win では起きない?
\r
546 // むしろ、MacOSX のブラウザ全般で起こる??
\r
547 if( ( X_UA.WebKit || X_UA.Blink ) &&
\r
548 ( type === 'webkitTransitionEnd' || type === 'transitionend' ||
\r
549 type === 'animationend' || type === 'webkitAnimationEnd' ||
\r
550 type === 'animationstart' || type === 'webkitAnimationStart' ||
\r
551 type === 'animationiteration' || type === 'webkitAnimationIteration' ) ){
\r
552 raw.addEventListener( type, X_EventDispatcher_iOSTransitionEndDispatch, false );
\r
554 f = that[ '_listeners' ]._handleEvent || ( that[ '_listeners' ]._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );
\r
556 if( raw.addEventListener ){
\r
557 raw.addEventListener( type, f, false );
\r
559 // Safari は Image, Opera7 は window
\r
560 raw[ 'on' + type ] = f;
\r
566 (function( that, type, raw, list ){
\r
568 switch( that[ '_rawType' ] ){
\r
569 case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :
\r
570 list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );
\r
571 list.sltoken = raw.AddEventListener( type, list.slcallback );
\r
574 case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :
\r
575 // ie8- の XHR は window.event が更新されないため, eventType 毎に callback を指定する
\r
576 raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );
\r
580 f = that[ '_listeners' ]._handleEvent || ( that[ '_listeners' ]._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );
\r
582 if( raw.attachEvent ){
\r
583 raw.attachEvent( 'on' + type, f );
\r
585 raw[ 'on' + type ] = f;
\r
590 (function( that, type, raw, list ){
\r
591 switch( that[ '_rawType' ] ){
\r
592 case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :
\r
593 // DOM0 で Silverlight ってあるの -> ie4 mobile?
\r
594 list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );
\r
595 list.sltoken = raw.AddEventListener( type, list.slcallback );
\r
598 case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :
\r
599 // ie4 mobile は XHR をサポート!
\r
600 raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );
\r
604 raw[ 'on' + type ] = that[ '_listeners' ]._handleEvent || ( that[ '_listeners' ]._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );
\r
610 * iOS の webkitTransitionEnd が連続して起こる場合、
\r
611 * コールバックの(that._handleEvent)クロージャ内の実際のコールバック(X_Callback_actualClosure:obj._)が
\r
612 * 参照できていない問題に遭遇、、、iOS3.1.3 & iOS6.1.5 で確認
\r
613 * animation も怪しい、、、
\r
615 function X_EventDispatcher_iOSTransitionEndDispatch( e ){
\r
616 return X_Node_getXNode( this ).dispatch( X_Event_RenameTo[ e.type ] || e.type );
\r
620 * Silverlight のイベントの概要
\r
621 * http://msdn.microsoft.com/ja-jp/library/cc189018%28v=vs.95%29.aspx#the_sender_parameter_and_event_data
\r
623 function X_EventDispatcher_sliverLightDispatch( sender, e, type ){
\r
624 return this.dispatch( type );
\r
627 function X_EventDispatcher_removeEvent( that, type, raw, list, skip ){
\r
629 X_EventDispatcher_unlock || ( type = X_Event_Rename[ type ] || type );
\r
631 if( X.Type.isArray( type ) ){
\r
632 for( i = type.length; i; ){
\r
633 X_EventDispatcher_systemUnlisten( that, type[ --i ], X.emptyFunction );
\r
636 X_EventDispatcher_actualRemoveEvent( that, type, raw, list, skip );
\r
640 var X_EventDispatcher_actualRemoveEvent =
\r
642 (function( that, type, raw, list, skip ){
\r
643 switch( that[ '_rawType' ] ){
\r
644 case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :
\r
645 raw.RemoveEventListener( type, list.sltoken ); // token
\r
646 X_Callback_correct( list.slcallback );
\r
647 delete list.sltoken;
\r
648 delete list.slcallback;
\r
651 case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :
\r
652 if( X_UA.Opera < 12 ){
\r
653 // Opera11- の XHR は event オブジェクトが返らないため, eventType 毎に callback を指定する addEventListener もない
\r
654 X_Callback_correct( raw[ 'on' + type ] );
\r
655 raw[ 'on' + type ] = '';
\r
660 if( ( X_UA.WebKit || X_UA.Blink ) &&
\r
661 ( type === 'webkitTransitionEnd' || type === 'transitionend' ||
\r
662 type === 'animationend' || type === 'webkitAnimationEnd' ||
\r
663 type === 'animationstart' || type === 'webkitAnimationStart' ||
\r
664 type === 'animationiteration' || type === 'webkitAnimationIteration' ) ){
\r
665 raw.removeEventListener( type, X_EventDispatcher_iOSTransitionEndDispatch, false );
\r
667 if( raw.addEventListener ){
\r
668 raw.removeEventListener( type, that[ '_listeners' ]._handleEvent, false );
\r
670 raw[ 'on' + type ] = null;
\r
673 if( !skip && that[ '_listeners' ]._handleEvent ){
\r
674 X_Callback_correct( that[ '_listeners' ]._handleEvent );
\r
675 delete that[ '_listeners' ]._handleEvent;
\r
680 (function( that, type, raw, list, skip ){
\r
681 switch( that[ '_rawType' ] ){
\r
682 case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :
\r
683 raw.RemoveEventListener( type, list.sltoken ); // token
\r
684 X_Callback_correct( list.slcallback );
\r
685 delete list.sltoken;
\r
686 delete list.slcallback;
\r
689 case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :
\r
690 X_Callback_correct( raw[ 'on' + type ] );
\r
691 raw[ 'on' + type ] = X.emptyFunction;
\r
692 raw[ 'on' + type ] = '';
\r
696 if( raw.attachEvent ){
\r
697 raw.detachEvent( 'on' + type, that[ '_listeners' ]._handleEvent );
\r
699 raw[ 'on' + type ] = X.emptyFunction;
\r
700 raw[ 'on' + type ] = '';
\r
704 X_Callback_correct( that[ '_listeners' ]._handleEvent );
\r
705 delete that[ '_listeners' ]._handleEvent;
\r
709 (function( that, type, raw, list, skip ){
\r
710 switch( that[ '_rawType' ] ){
\r
711 case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :
\r
712 raw.RemoveEventListener( type, list.sltoken ); // token
\r
713 X_Callback_correct( list.slcallback );
\r
714 delete list.sltoken;
\r
715 delete list.slcallback;
\r
718 case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :
\r
719 X_Callback_correct( raw[ 'on' + type ] );
\r
720 raw[ 'on' + type ] = X.emptyFunction;
\r
721 raw[ 'on' + type ] = '';
\r
725 raw[ 'on' + type ] = X.emptyFunction;
\r
726 raw[ 'on' + type ] = '';
\r
729 X_Callback_correct( that[ '_listeners' ]._handleEvent );
\r
730 delete that[ '_listeners' ]._handleEvent;
\r
736 // handleEvent を拡張可能にするために、クロージャに移動した
\r
737 // Is this in regard to the Safari 1.x preventDefault bug on click/dblclick?
\r
738 // https://groups.google.com/forum/#!msg/comp.lang.javascript/uYEuCHjHxnw/yKoHtZJPa1QJ
\r
739 var X_EventDispatcher_actualHandleEvent =
\r
740 X_UA_EVENT.IE4 || X_UA_EVENT.IE ? // ie45678 EVENT_IE & EVENT_DOM0 for ie4
\r
744 //if( event.type === 'readystatechange' && this._tag && X.Dom.Event._LOAD_FIX_TAGS[ this._tag ] ){
\r
745 //type = 'readystatechange';
\r
748 ret = this.dispatch( new X.Dom.Event( event, this, this._rawObject ) );
\r
750 if( ret & X.Callback.STOP_PROPAGATION ){
\r
751 event.cancelBubble = true;
\r
753 if( ret & X.Callback.PREVENT_DEFAULT ){
\r
754 this._tag === 'A' && this._rawObject.blur();
\r
755 return event.returnValue = false;
\r
758 //X_UA_EVENT.W3C & X_UA_EVENT.DOM0
\r
760 var ev = new X.Dom.Event( e, this ),
\r
761 ret = X_Callback_NONE,
\r
763 //console.log( '>>>>>>>>>> ' + e.type );
\r
764 // touch event -> pointer
\r
765 if( X.Type.isArray( ev ) ){
\r
766 if( ev.length === 0 ){
\r
767 // TouchEvent の後に発生した MouseEvent のキャンセル
\r
768 ret = X.Callback.STOP_PROPAGATION | X.Callback.PREVENT_DEFAULT;
\r
770 for( i = 0, l = ev.length; i < l; ++i ){
\r
771 console.log( 'handleEvent ' + ev[ i ].type );
\r
772 ret |= this.dispatch( ev[ i ] ) || 0;
\r
776 ret = this.dispatch( ev );
\r
779 if( ret & X.Callback.STOP_PROPAGATION ){
\r
780 e.stopPropagation();
\r
782 if( ret & X.Callback.PREVENT_DEFAULT ){
\r
783 this._tag === 'A' && this._rawObject.blur();
\r
784 e.preventDefault();
\r
785 if( X_UA.WebKit < 525.13 ){ // Safari3-
\r
786 if( e.type === 'click' || e.type === 'dbclick' ){
\r
787 X_EventDispatcher_safariPreventDefault = true;
\r
794 if( X_UA.WebKit < 525.13 ){ // Safari3-
\r
795 document.documentElement.onclick =
\r
796 document.documentElement.ondbclick = function( e ){
\r
797 if( X_EventDispatcher_safariPreventDefault ){
\r
798 X_EventDispatcher_safariPreventDefault = false;
\r
799 e.preventDefault();
\r
805 // イベントの退避、dom が画面から抜かれる場合に実施しておく
\r
807 function X_EventDispatcher_toggleAllEvents( that, add ){
\r
808 var list = that[ '_listeners' ],
\r
809 raw = that._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( that ),
\r
810 f = add ? X_EventDispatcher_addEvent : X_EventDispatcher_removeEvent,
\r
812 if( !list || !raw ) return;
\r
813 for( type in list ){
\r
814 //if( X_EMPTY_OBJECT[ type ] ) continue;
\r
815 if( type.charAt( 0 ) === '_' ) continue;
\r
817 if( '' + parseFloat( type ) !== type ){
\r
818 // TODO type rename はここ
\r
819 f( that, type, raw, list[ type ], true );
\r
825 console.log( 'X.Core.EventDispatcher' );
\r