OSDN Git Service

Version 0.6.131, cleanup X.EventDispatcher.
[pettanr/clientJs.git] / 0.6.x / js / 01_core / 13_XEventDispatcher.js
index 25c2d11..36a11fb 100644 (file)
@@ -29,7 +29,33 @@ var X_EventDispatcher_once       = false,
        X_EventDispatcher_unlock     = false,\r
        X_EventDispatcher_needsIndex = false,\r
        \r
-       X_EventDispatcher_safariPreventDefault = false; // Safari3-\r
+       X_EventDispatcher_safariPreventDefault = false, // Safari3-\r
+       \r
+       X_EventDispatcher_EVENT_TARGET_TYPE = {\r
+               OTHER        : 0,\r
+               XHR          : 1,\r
+               SILVER_LIGHT : 2\r
+       },\r
+       \r
+       X_EventDispatcher_LAZY_TIMERS = {}; // Object.<number, X.EventDispatcher> number は timerID\r
+       \r
+/*\r
+ * イベントリスナをイベント名(string)や数値(1~,フレームワーク内で定義)をキーとするArrayで記憶します。\r
+ * Arrayには、{k:種類,x:コンテキスト(thisObject),f:関数,s:サプリメントする引数の配列} というハッシュ、または関数が蓄えられています。\r
+ * また、dispatch 中の状態と操作を記録し不整合が起きないようにするためのプロパティを持ちます。\r
+ * @typedef {(\r
+ *     {\r
+ *      _handleEvent  : function,\r
+ *      _dispatching  : number,\r
+ *      _reserves     : (Array|undefined),\r
+ *      _unlistens    : {Object.<(number|string), Array.<(X.Callback|function)>>},\r
+ *      _killReserved : (Boolean|undefiend)\r
+ *  }\r
+ *   |\r
+ *  Object.<(number|string), Array.<(callbackHash|function)>>\r
+ * )}\r
+ */\r
+var X_EventDispatcher_listeners;\r
 \r
 // ------------------------------------------------------------------------- //\r
 // --- interface ----------------------------------------------------------- //\r
@@ -61,14 +87,19 @@ X.EventDispatcher =
            /** @lends X.EventDispatcher.prototype */\r
                {\r
 \r
-\r
-\r
-               // TODO _rawObjectType EventTarget, XHR, Silverlight, ...\r
+               /**\r
+                * OTHER(Node,window,document,Image,Audio), XHR, Silverlight\r
+                * @private\r
+                * @type {number}\r
+                */\r
+                       '_rawType'      : X_EventDispatcher_EVENT_TARGET_TYPE.OTHER,\r
+               \r
                /**\r
                 * イベントリスナをイベント名(string)や数値(1~,フレームワーク内で定義)をキーとするArrayで記憶します。\r
                 * Arrayには、{k:種類,x:コンテキスト(thisObject),f:関数,s:サプリメントする引数の配列} というハッシュ、または関数が蓄えられています。\r
+                * \r
                 * @private\r
-                * @type {Object.<(number|string), Array.<(callbackHash|function)>>}\r
+                * @type {X_EventDispatcher_listeners}\r
                 */\r
                        '_listeners'    : null,\r
 \r
@@ -76,62 +107,15 @@ X.EventDispatcher =
                 * _rawObject には HTMLElement, window, document, XHR といったイベントターゲットオブジェクトを設定します。\r
                 * _rawObject が設定されていると listen(), unlisten() 時に addEventListener(DOM Level2) や detachEvent(ie5~8), on~(DOM0) 等を操作します。\r
                 * _rawObject は最初の listen() 前に設定しておかないと addEventListener 等が意図したように行われません。\r
-                * X.Node ã\81§ã\81¯é\9d\9eå\90\8cæ\9c\9fã\81« HTMLElement ã\82\92ç\94\9fæ\88\90ã\81\97ã\81¦ã\81\84ã\81¦、要素生成以前に listen, unlisten を呼び出すことができます。これは適宜に X_EventDispatcher_toggleAllEvents を呼んで解決しているためです。\r
+                * X.Node ã\81§ã\81¯é\9d\9eå\90\8cæ\9c\9fã\81« HTMLElement ã\82\92ç\94\9fæ\88\90ã\81\97ã\81¦ã\81\84ã\81¾ã\81\99ã\81\8c、要素生成以前に listen, unlisten を呼び出すことができます。これは適宜に X_EventDispatcher_toggleAllEvents を呼んで解決しているためです。\r
                 * @private\r
                 * @type {Object}\r
                 */\r
                        '_rawObject'    : null,\r
-               \r
-               /**\r
-                * _rawObject がある場合、イベントターゲットオブジェクトに対して addEventListener, attachEvent, onXX 時に実際に渡される関数。\r
-                * イベントタイプが変わっても、単一の オブジェクトを使いまわす。(但し例外は、IE8以下の XHR|MSXML に対して。)最初の listen で生成されて、最後の unlisten で削除される。\r
-                * @private\r
-                * @type {X.Callback}\r
-                */\r
-                       '_handleEvent'  : null,\r
-                       \r
-               /**\r
-                * dispatch 中に dispatch が呼ばれた際に、そのネストの深さを保存する。\r
-                * dispatch() 終了時に _dispatching が 0 の場合に、現在のインスタンスの dispatch がすべて終わったことになる。\r
-                * @private\r
-                * @type {number}\r
-                */\r
-                       '_dispatching'  : 0,     // dispatch 中の unlisten で使用\r
-\r
-               /**\r
-                * dispatch 中に listen が呼ばれた場合に、配列のindexがずれることを避けるため、一旦保持する。\r
-                * _dispatching が 0 のときに _reserves を使って listen() を呼び出す。\r
-                * @private\r
-                * @type {Object.<(number|string), Array.<(X.Callback|function)>>}\r
-                */\r
-                       '_reserves'     : null,\r
-               \r
-               /**\r
-                * dispatch 中に unlisten が呼ばれた場合に、配列のindexがずれることを避けるため、一旦保持する。\r
-                * _dispatching が 0 のときに _unlistens を使って unlisten() を呼び出す。\r
-                * @private\r
-                * @type {Object.<(number|string), Array.<(X.Callback|function)>>}\r
-                */\r
-                       '_unlistens'    : null,\r
-                       \r
-               /**\r
-                * dispatch 中に kill が呼ばれた場合に、X.EventDispatcher インスタンスの削除を dispatch 後にずらすために立てるフラグ。\r
-                * @private\r
-                * @type {boolean}\r
-                */\r
-                       '_killReserved' : false,\r
-\r
-               /**\r
-                * asyncDispatch 中に kill が呼ばれた場合に、X.EventDispatcher インスタンスの削除を最後のタイマーの発火後にずらすために立てるフラグ。\r
-                * TODO asyncDispatch の解除はどうする?\r
-                * @private\r
-                * @type {boolean}\r
-                */\r
-                       '_lastLazyID' : false,\r
                        \r
            /**\r
             * X.EventDispatcher のコンストラクタの実体。\r
-            * @@@constructs\r
+            * @constructs\r
             * @this {X.EventDispatcher}\r
             * @param {object=} opt_rawObject\r
             */\r
@@ -196,30 +180,31 @@ X.EventDispatcher =
                 * @param {Array=} opt_arg3\r
                 */                     \r
                        listening : function( opt_type, opt_arg1, opt_arg2, opt_arg3 ){\r
-                               var list = this._listeners,\r
-                                       unlistens, i, f, hash;\r
+                               var listeners = this[ '_listeners' ],\r
+                                       lock      = X_EventDispatcher_lock || X_EventDispatcher_unlock,\r
+                                       list, cbHash, unlistens, i, f;\r
                                \r
-                               if( opt_type === undefined ) return !!list;\r
-                               if( !list || !( list = list[ opt_type ] ) ) return false;\r
+                               if( opt_type === undefined ) return !!listeners;\r
+                               if( !listeners || !( list = listeners[ opt_type ] ) ) return false;\r
                                if( opt_arg1 === undefined ) return true;\r
                                \r
                                if( opt_arg1.k ){\r
-                                       hash = opt_arg1;\r
+                                       cbHash = opt_arg1;\r
                                } else {\r
-                                       hash = X_Callback_classifyCallbackArgs( opt_arg1, opt_arg2, opt_arg3, this );\r
+                                       cbHash = X_Callback_classifyCallbackArgs( opt_arg1, opt_arg2, opt_arg3, this );\r
                                };\r
                                \r
-                               if( ( unlistens = this._unlistens ) && ( unlistens = unlistens[ opt_type ] ) ){\r
+                               if( ( unlistens = listeners._unlistens ) && ( unlistens = unlistens[ opt_type ] ) ){\r
                                        for( i = unlistens.length; i; ){\r
                                                f = unlistens[ --i ];\r
-                                               if( f === hash || ( f.x === hash.x && f.f === hash.f && f.s === hash.s ) ) return false;\r
+                                               if( f === cbHash || ( f.x === cbHash.x && f.f === cbHash.f && f.s === cbHash.s && f.lock === lock ) ) return false;\r
                                        };\r
                                };\r
                                for( i = list.length; i; ){\r
                                        f = list[ --i ];\r
-                                       if( f === hash || ( f.x === hash.x && f.f === hash.f && f.s === hash.s ) ){\r
+                                       if( f === cbHash || ( f.x === cbHash.x && f.f === cbHash.f && f.s === cbHash.s && f.lock === lock ) ){\r
                                                // index を要求された場合、lock されていない、または unlock なら index を返す\r
-                                               return X_EventDispatcher_needsIndex ? ( X_EventDispatcher_unlock || !f.lock ? i : false ) : true;\r
+                                               return X_EventDispatcher_needsIndex ? i : true;\r
                                        };\r
                                };\r
                                return false;\r
@@ -233,11 +218,14 @@ X.EventDispatcher =
                 * @return {number} X.Timer.add() の戻り値\r
                 */                     \r
                        asyncDispatch : function( delay, e ){\r
+                               var timerID;\r
                                if( delay && e === undefined ){\r
                                        e = delay;\r
                                        delay = 0;\r
                                };\r
-                               return this[ '_lastLazyID' ] = X.Timer.add( delay, 1, this, X_EventDispatcher_dispatch, [ e ] );\r
+                               timerID = X.Timer.add( delay, 1, this, X_EventDispatcher_dispatch, [ e ] );\r
+                               X_EventDispatcher_LAZY_TIMERS[ timerID ] = this;\r
+                               return timerID;\r
                        }\r
                }\r
        );\r
@@ -253,12 +241,12 @@ X.EventDispatcher =
  * @param {(eventHash|string|number)} e\r
  */\r
 function X_EventDispatcher_dispatch( e ){\r
-       var list  = this[ '_listeners' ],\r
-               ret   = X_Callback_NONE,\r
-               type  = e[ 'type' ],\r
-               unlistens, i, l, args, f, r, sysOnly;\r
+       var listeners = this[ '_listeners' ],\r
+               ret       = X_Callback_NONE,\r
+               type      = e[ 'type' ],\r
+               list, unlistens, i, l, args, f, r, sysOnly, timerID;\r
        \r
-       if( !list || !( list = list[ type || e ] ) ) return ret;\r
+       if( !listeners || !( list = listeners[ type || e ] ) ) return ret;\r
        \r
        // 数値, 文字が渡された場合\r
        if( !type ){\r
@@ -267,17 +255,21 @@ function X_EventDispatcher_dispatch( e ){
        e.target        = e.target || this;\r
        e.currentTarget = e.currentTarget || this;\r
        \r
-       ++this._dispatching;\r
+       if( listeners._dispatching ){\r
+               ++listeners._dispatching;\r
+       } else {\r
+               listeners._dispatching = 1;\r
+       };\r
        \r
        // todo:\r
        // type も保存\r
-       this._unlistens = this._unlistens || {};\r
-       unlistens = this._unlistens[ type ];\r
+       listeners._unlistens = listeners._unlistens || {};\r
+       unlistens = listeners._unlistens[ type ];\r
        \r
        for( i = 0; i < list.length; ++i ){\r
                f = list[ i ];\r
                if( !unlistens ){\r
-                       unlistens = this._unlistens[ type ];\r
+                       unlistens = listeners._unlistens[ type ];\r
                };\r
                if( unlistens && unlistens.indexOf( f ) !== -1 ) continue;\r
                \r
@@ -292,7 +284,7 @@ function X_EventDispatcher_dispatch( e ){
                if( f.once || r & X_Callback_UN_LISTEN ){\r
                        // dispatch 中に unlisten が作られることがある\r
                        if( !unlistens ){\r
-                               unlistens = this._unlistens || ( this._unlistens = {} );\r
+                               unlistens = listeners._unlistens || ( listeners._unlistens = {} );\r
                                unlistens = unlistens[ type ] || ( unlistens[ type ] = [] );\r
                        };\r
                        unlistens.indexOf( f ) === -1 && ( unlistens[ unlistens.length ] = f );\r
@@ -304,33 +296,12 @@ function X_EventDispatcher_dispatch( e ){
                ret |= X.Type.isFinite( r ) ? r : 0;\r
        };\r
        \r
-       if( ( --this._dispatching ) === 0 ){\r
-               \r
-               if( this[ '_lastLazyID' ] && X_Timer_currentUID === this[ '_lastLazyID' ] ){\r
-                       delete this[ '_lastLazyID' ];\r
-               };\r
-               \r
-               // dispatch 中に unlisten された要素の削除\r
-               unlistens = this._unlistens;\r
-               delete this._dispatching;\r
-               delete this._unlistens;                                 \r
-               // _unlistens に入っている callbackHash は、lock をクリアしている\r
-               X_EventDispatcher_unlock = true;\r
-               for( type in unlistens ){\r
-                       //if( X_EMPTY_OBJECT[ type ] ) continue;\r
-                       list = unlistens[ type ];\r
-                       for( i = list.length; i; ){\r
-                               this.unlisten( type, list[ --i ] );\r
-                       };\r
-                       list.length = 0;\r
-                       delete unlistens[ type ];\r
-               };\r
-               X_EventDispatcher_unlock = false;\r
+       if( ( --listeners._dispatching ) === 0 ){\r
+\r
+               delete listeners._dispatching;\r
                \r
-               if( this._killReserved ){\r
-                       this.kill();\r
-               } else\r
-               if( list = this._reserves ){\r
+               // dispatch 中に listen されたイベントの追加\r
+               if( list = listeners._reserves ){\r
                        for( i = 0, l = list.length; i < l; ++i ){\r
                                f = list[ i ];\r
                                X_EventDispatcher_once = f[ 4 ];\r
@@ -341,7 +312,36 @@ function X_EventDispatcher_dispatch( e ){
                                f.length = 0;\r
                        };\r
                        list.length = 0;\r
-                       delete this._reserves;\r
+                       delete listeners._reserves;\r
+               };              \r
+               \r
+               // dispatch 中に unlisten されたイベントの削除\r
+               if( unlistens = listeners._unlistens ){\r
+                       delete listeners._unlistens;\r
+                       \r
+                       // _unlistens に入っている callbackHash は、lock をクリアしている\r
+                       X_EventDispatcher_unlock = true;\r
+                       for( type in unlistens ){\r
+                               //if( X_EMPTY_OBJECT[ type ] ) continue;\r
+                               list = unlistens[ type ];\r
+                               for( i = list.length; i; ){\r
+                                       this.unlisten( type, list[ --i ] );\r
+                               };\r
+                               list.length = 0;\r
+                               delete unlistens[ type ];\r
+                       };\r
+                       X_EventDispatcher_unlock = false;                       \r
+               };\r
+               \r
+               if( X_EventDispatcher_LAZY_TIMERS[ X_Timer_currentUID ] === this ){\r
+                       delete X_EventDispatcher_LAZY_TIMERS[ X_Timer_currentUID ];\r
+               };\r
+\r
+               if( listeners._killReserved ){\r
+                       for( timerID in X_EventDispatcher_LAZY_TIMERS ){\r
+                               if( X_EventDispatcher_LAZY_TIMERS[ timerID ] === this ) return ret;\r
+                       };                      \r
+                       this.kill();\r
                };\r
        };\r
        \r
@@ -359,14 +359,14 @@ function X_EventDispatcher_dispatch( e ){
  * @return {X.EventDispatcher}\r
  */\r
 function X_EventDispatcher_listen( type, opt_arg1, opt_arg2, opt_arg3 ){\r
-       var list = this._listeners,\r
-               i, raw, add, f;\r
+       var listeners = this[ '_listeners' ],\r
+               i, raw, add, list, f;\r
 \r
        if( !type ) return this;\r
        \r
-       if( this._dispatching ){\r
-               if( !this._reserves ) this._reserves = [];\r
-               this._reserves[ this._reserves.length ] = [ type, opt_arg1, opt_arg2, opt_arg3, X_EventDispatcher_once, X_EventDispatcher_lock ];\r
+       if( listeners && listeners._dispatching ){\r
+               if( !listeners._reserves ) listeners._reserves = [];\r
+               listeners._reserves[ listeners._reserves.length ] = [ type, opt_arg1, opt_arg2, opt_arg3, X_EventDispatcher_once, X_EventDispatcher_lock ];\r
                return this;\r
        };\r
        \r
@@ -378,12 +378,12 @@ function X_EventDispatcher_listen( type, opt_arg1, opt_arg2, opt_arg3 ){
        };\r
        \r
        raw = this._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( this );\r
-       add = raw && ( !list || !list[ type ] ) && X.Type.isString( type );\r
-       \r
+       add = raw && ( !listeners || !listeners[ type ] ) && X.Type.isString( type );\r
+\r
        if( this.listening( type, opt_arg1 || this, opt_arg2, opt_arg3 ) ) return this;\r
 \r
-       if( !list ) list = this._listeners = {};\r
-       if( !( list = list[ type ] ) ) list = this._listeners[ type ] = [];\r
+       if( !listeners ) listeners = this[ '_listeners' ] = {};\r
+       list = listeners[ type ] || ( listeners[ type ] = [] );\r
        \r
        add && X_EventDispatcher_addEvent( this, type, raw, list );\r
        \r
@@ -416,9 +416,9 @@ function X_EventDispatcher_systemListen( that, type, opt_arg1, opt_arg2, opt_arg
  * @param {Array=} opt_arg3\r
  */\r
 function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){\r
-       var list = this._listeners,\r
-               _list, reserves, unlistens, i, f, raw, empty;\r
-       if( !list ) return this;\r
+       var listeners = this[ '_listeners' ],\r
+               list, reserves, unlistens, i, f, raw, k, empty;\r
+       if( !listeners ) return this;\r
        \r
        if( X.Type.isArray( opt_type ) ){\r
                for( i = opt_type.length; i; ){\r
@@ -432,11 +432,12 @@ function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
        \r
        if( opt_type === undefined ){\r
                // 全て削除\r
-               for( opt_type in list ){\r
+               for( opt_type in listeners ){\r
                        //if( X_EMPTY_OBJECT[ opt_type ] ) continue;\r
-                       _list = list[ opt_type ];\r
-                       for( i = _list.length; i; ){\r
-                               this.unlisten( opt_type, _list[ --i ] ); // override されていることがあるので、必ず unlisten を使用\r
+                       if( opt_type.charAt( 0 ) === '_' ) continue;\r
+                       list = listeners[ opt_type ];\r
+                       for( i = list.length; i; ){\r
+                               this.unlisten( opt_type, list[ --i ] ); // override されていることがあるので、必ず unlisten を使用\r
                        };\r
                        // this.unlisten( opt_type ); これは無茶!\r
                };\r
@@ -444,19 +445,19 @@ function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
        } else\r
        if( opt_arg1 === undefined ){\r
                // 同一タイプを全て削除\r
-               if( _list = list[ opt_type ] ){\r
-                       for( i = _list.length; i; ){\r
-                               this.unlisten( opt_type, _list[ --i ] ); // override されていることがあるので、必ず unlisten を使用\r
+               if( list = listeners[ opt_type ] ){\r
+                       for( i = list.length; i; ){\r
+                               this.unlisten( opt_type, list[ --i ] ); // override されていることがあるので、必ず unlisten を使用\r
                        };\r
                };\r
                return this;\r
        } else\r
-       if( reserves = this._reserves ){\r
+       if( reserves = listeners._reserves ){\r
                for( i = reserves.length; i; ){\r
                        f = reserves[ --i ];\r
-                       if( f[ 0 ] === opt_type && f[ 1 ] === opt_arg1 && f[ 2 ] === opt_arg2 && f[ 3 ] === opt_arg3 ){\r
+                       if( f[ 0 ] === opt_type && f[ 1 ] === opt_arg1 && f[ 2 ] === opt_arg2 && f[ 3 ] === opt_arg3 && ( !f[ 5 ] || X_EventDispatcher_unlock ) ){\r
                                reserves.splice( i, 1 );\r
-                               if( !reserves.legth ) delete this._reserves;\r
+                               if( !reserves.legth ) delete listeners._reserves;\r
                                return this;\r
                        };\r
                };\r
@@ -467,22 +468,31 @@ function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
        X_EventDispatcher_needsIndex = false;\r
        if( i === false ) return this;\r
 \r
-       f = ( _list = list[ opt_type ] )[ i ];\r
-       // _unlistens に入っている callbackHash は、lock をクリアしている\r
-       if( unlistens = this._unlistens ){\r
+       f = ( list = listeners[ opt_type ] )[ i ];\r
+       \r
+       if( unlistens = listeners._unlistens ){\r
+               // _unlistens に入っている callbackHash は、lock のチェックは済んでいる\r
                ( unlistens = unlistens[ opt_type ] ) ?\r
                        ( unlistens[ unlistens.length ] = f ) :\r
-                       ( this._unlistens[ opt_type ] = [ f ] );\r
+                       ( listeners._unlistens[ opt_type ] = [ f ] );\r
        } else {\r
                delete f.once;\r
-               _list.splice( i, 1 );\r
-               if( !_list.length ){\r
+               list.splice( i, 1 );\r
+               if( !list.length ){\r
                        raw  = this._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( this );\r
-                       delete list[ opt_type ];\r
-                       if( empty = X_Object_isEmpty( list ) ) delete this._listeners;\r
+                       delete listeners[ opt_type ];\r
+                       //empty = X_Object_isEmpty( listeners );\r
+                       // TODO カウンター\r
+                       empty = true;\r
+                       for( k in listeners ){\r
+                               if( k.charAt( 0 ) === '_' ) continue;\r
+                               empty = false;\r
+                               break;\r
+                       };\r
                        if( raw && '' + parseFloat( opt_type ) !== '' + opt_type ){ // 数字イベントの除外\r
-                               X_EventDispatcher_removeEvent( this, opt_type, raw, _list, !empty );\r
+                               X_EventDispatcher_removeEvent( this, opt_type, raw, list, !empty );\r
                        };\r
+                       if( empty ) delete this[ '_listeners' ];\r
                };\r
        };\r
        return this;\r
@@ -515,66 +525,84 @@ var X_EventDispatcher_actualAddEvent =
        // Days on the Moon DOM Events とブラウザの実装 \r
        // http://nanto.asablo.jp/blog/2007/03/23/1339502\r
        // Safari 2 では関数オブジェクトしか EventListener として使えませんが、Safari のナイトリービルドでは handleEvent メソッドを持つオブジェクトも EventListener として使えるようです。\r
-       X_UA_EVENT.W3C /* && ( X_UA.WebKit < 525.13 || X_UA.Opera7 || X_UA.NetFront < 4 ) */ ? // Safari3-\r
+       X_UA_EVENT.W3C ?\r
                (function( that, type, raw, list ){\r
-                       if( that._isXHR && X_UA.Opera < 12 ){\r
-                               // Opera11- の XHR は event オブジェクトが返らないため, eventType 毎に callback を指定する addEventListener もない\r
-                               raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );\r
-                       } else\r
-                       if( that._isSilverlight ){\r
-                               list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );\r
-                               list.sltoken    = raw.AddEventListener( type, list.slcallback );\r
-                       } else\r
-                       // iOS と MacOSX Iron36 で発生。連続してアニメーションが起こると、クロージャの束縛された obj へのアクセスに失敗する。Win では起きない?\r
-                       // むしろ、MacOSX のブラウザ全般で起こる??\r
-                       if( ( X_UA.WebKit || X_UA.Blink ) &&\r
-                               ( type === 'webkitTransitionEnd' || type === 'transitionend' ||\r
-                                 type === 'animationend'        || type === 'webkitAnimationEnd' ||\r
-                                 type === 'animationstart'      || type === 'webkitAnimationStart' ||\r
-                                 type === 'animationiteration'  || type === 'webkitAnimationIteration' ) ){\r
-                               raw.addEventListener( type, X_EventDispatcher_iOSTransitionEndDispatch, false );\r
-                       } else {\r
-                               console.log( 'event > ' + type );\r
-                               that._handleEvent || ( that._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );\r
+                       var f;\r
+                       switch( that[ '_rawType' ] ){\r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :\r
+                                       list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );\r
+                                       list.sltoken    = raw.AddEventListener( type, list.slcallback );\r
+                                       break;\r
+                               \r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :\r
+                                       if( X_UA.Opera < 12 ){\r
+                                               // Opera11- の XHR は event オブジェクトが返らないため, eventType 毎に callback を指定する addEventListener もない\r
+                                               raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );\r
+                                               break;\r
+                                       };\r
 \r
-                               if( raw.addEventListener ){\r
-                                       raw.addEventListener( type, that._handleEvent, false );\r
-                               } else {\r
-                                       // Safari は Image, Opera7 は window\r
-                                       raw[ 'on' + type ] = that._handleEvent;\r
-                               };\r
+                               default :\r
+                                       // iOS と MacOSX Iron36 で発生。連続してアニメーションが起こると、クロージャの束縛された obj へのアクセスに失敗する。Win では起きない?\r
+                                       // むしろ、MacOSX のブラウザ全般で起こる??\r
+                                       if( ( X_UA.WebKit || X_UA.Blink ) &&\r
+                                               ( type === 'webkitTransitionEnd' || type === 'transitionend' ||\r
+                                                 type === 'animationend'        || type === 'webkitAnimationEnd' ||\r
+                                                 type === 'animationstart'      || type === 'webkitAnimationStart' ||\r
+                                                 type === 'animationiteration'  || type === 'webkitAnimationIteration' ) ){\r
+                                               raw.addEventListener( type, X_EventDispatcher_iOSTransitionEndDispatch, false );\r
+                                       } else {\r
+                                               f = that[ '_listeners' ]._handleEvent || ( that[ '_listeners' ]._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );\r
+               \r
+                                               if( raw.addEventListener ){\r
+                                                       raw.addEventListener( type, f, false );\r
+                                               } else {\r
+                                                       // Safari は Image, Opera7 は window\r
+                                                       raw[ 'on' + type ] = f;\r
+                                               };\r
+                                       };\r
                        };\r
                }) :\r
        X_UA_EVENT.IE ?\r
                (function( that, type, raw, list ){\r
-                       if( that._isXHR ){\r
-                               // ie8- の XHR は window.event が更新されないため, eventType 毎に callback を指定する\r
-                               raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );\r
-                       } else                                  \r
-                       if( that._isSilverlight ){\r
-                               list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );\r
-                               list.sltoken    = raw.AddEventListener( type, list.slcallback );\r
-                       } else {                                \r
-                               that._handleEvent || ( that._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );\r
+                       var f;\r
+                       switch( that[ '_rawType' ] ){   \r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :\r
+                                       list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );\r
+                                       list.sltoken    = raw.AddEventListener( type, list.slcallback );\r
+                                       break;                          \r
                                \r
-                               if( raw.attachEvent ){\r
-                                       raw.attachEvent( 'on' + type, that._handleEvent );\r
-                               } else {\r
-                                       raw[ 'on' + type ] = that._handleEvent;\r
-                               };\r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :\r
+                                       // ie8- の XHR は window.event が更新されないため, eventType 毎に callback を指定する\r
+                                       raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );\r
+                                       break;\r
+                               \r
+                               default :\r
+                                       f = that[ '_listeners' ]._handleEvent || ( that[ '_listeners' ]._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );\r
+                                       \r
+                                       if( raw.attachEvent ){\r
+                                               raw.attachEvent( 'on' + type, f );\r
+                                       } else {\r
+                                               raw[ 'on' + type ] = f;\r
+                                       };\r
+                                       break;\r
                        };\r
                }) :\r
                (function( that, type, raw, list ){\r
-                       if( that._isXHR ){\r
-                               // ie4 mobile は XHR をサポート!\r
-                               raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );\r
-                       } else\r
-                       if( that._isSilverlight ){\r
-                               // DOM0 で Silverlight ってあるの?\r
-                               list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );\r
-                               list.sltoken    = raw.AddEventListener( type, list.slcallback );\r
-                       } else {\r
-                               raw[ 'on' + type ] = that._handleEvent || ( that._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );\r
+                       switch( that[ '_rawType' ] ){\r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :\r
+                                       // DOM0 で Silverlight ってあるの -> ie4 mobile?\r
+                                       list.slcallback = X_Callback_create( that, X_EventDispatcher_sliverLightDispatch, [ type ] );\r
+                                       list.sltoken    = raw.AddEventListener( type, list.slcallback );\r
+                                       break;                          \r
+                               \r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :\r
+                                       // ie4 mobile は XHR をサポート!\r
+                                       raw[ 'on' + type ] = X_Callback_create( that, X_EventDispatcher_dispatch, [ type ] );\r
+                                       break;\r
+\r
+                               default :\r
+                                       raw[ 'on' + type ] = that[ '_listeners' ]._handleEvent || ( that[ '_listeners' ]._handleEvent = X_Callback_create( that, X_EventDispatcher_actualHandleEvent ) );\r
+                                       break;\r
                        };\r
                });\r
 \r
@@ -610,82 +638,97 @@ function X_EventDispatcher_removeEvent( that, type, raw, list, skip ){
 };\r
 \r
 var X_EventDispatcher_actualRemoveEvent =\r
-       X_UA_EVENT.W3C /*&& ( X_UA.WebKit < 525.13 || X_UA.Opera7 || X_UA.NetFront < 4 )*/ ? // Safari3-\r
+       X_UA_EVENT.W3C ?\r
                (function( that, type, raw, list, skip ){\r
-                       if( that._isXHR && X_UA.Opera < 12 ){\r
-                               X_Callback_correct( raw[ 'on' + type ] );\r
-                               raw[ 'on' + type ] = '';\r
-                       } else  \r
-                       if( that._isSilverlight ){\r
-                               raw.RemoveEventListener( type, list.sltoken ); // token\r
-                               X_Callback_correct( list.slcallback );\r
-                               delete list.sltoken;\r
-                               delete list.slcallback;\r
-                       } else\r
-                       if( ( X_UA.WebKit || X_UA.Blink ) &&\r
-                               ( type === 'webkitTransitionEnd' || type === 'transitionend' ||\r
-                                 type === 'animationend'        || type === 'webkitAnimationEnd' ||\r
-                                 type === 'animationstart'      || type === 'webkitAnimationStart' ||\r
-                                 type === 'animationiteration'  || type === 'webkitAnimationIteration' ) ){\r
-                               raw.removeEventListener( type, X_EventDispatcher_iOSTransitionEndDispatch, false );\r
-                       } else {\r
-                                                               \r
-                               if( raw.addEventListener ){\r
-                                       raw.removeEventListener( type, that._handleEvent, false );\r
-                               } else {\r
-                                       raw[ 'on' + type ] = null;\r
-                               };\r
-                               if( !skip ){\r
-                                       X_Callback_correct( that._handleEvent );\r
-                                       delete that._handleEvent;\r
-                               };\r
+                       switch( that[ '_rawType' ] ){\r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :\r
+                                       raw.RemoveEventListener( type, list.sltoken ); // token\r
+                                       X_Callback_correct( list.slcallback );\r
+                                       delete list.sltoken;\r
+                                       delete list.slcallback;\r
+                                       break;\r
+                               \r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :\r
+                                       if( X_UA.Opera < 12 ){\r
+                                               // Opera11- の XHR は event オブジェクトが返らないため, eventType 毎に callback を指定する addEventListener もない\r
+                                               X_Callback_correct( raw[ 'on' + type ] );\r
+                                               raw[ 'on' + type ] = '';\r
+                                               break;\r
+                                       };\r
+\r
+                               default :\r
+                                       if( ( X_UA.WebKit || X_UA.Blink ) &&\r
+                                               ( type === 'webkitTransitionEnd' || type === 'transitionend' ||\r
+                                                 type === 'animationend'        || type === 'webkitAnimationEnd' ||\r
+                                                 type === 'animationstart'      || type === 'webkitAnimationStart' ||\r
+                                                 type === 'animationiteration'  || type === 'webkitAnimationIteration' ) ){\r
+                                               raw.removeEventListener( type, X_EventDispatcher_iOSTransitionEndDispatch, false );\r
+                                       } else                  \r
+                                       if( raw.addEventListener ){\r
+                                               raw.removeEventListener( type, that[ '_listeners' ]._handleEvent, false );\r
+                                       } else {\r
+                                               raw[ 'on' + type ] = null;\r
+                                       };\r
+                                       \r
+                                       if( !skip && that[ '_listeners' ]._handleEvent ){\r
+                                               X_Callback_correct( that[ '_listeners' ]._handleEvent );\r
+                                               delete that[ '_listeners' ]._handleEvent;\r
+                                       };\r
                        };\r
                }) :\r
        X_UA_EVENT.IE ?\r
                (function( that, type, raw, list, skip ){\r
-                       if( that._isXHR ){\r
-                               X_Callback_correct( raw[ 'on' + type ] );\r
-                               raw[ 'on' + type ] = X.emptyFunction;\r
-                               raw[ 'on' + type ] = '';\r
-                       } else  \r
-                       if( that._isSilverlight ){\r
-                               raw.RemoveEventListener( type, list.sltoken ); // token\r
-                               X_Callback_correct( list.slcallback );\r
-                               delete list.sltoken;\r
-                               delete list.slcallback;\r
-                       } else {\r
-                               if( raw.attachEvent ){\r
-                                       raw.detachEvent( 'on' + type, that._handleEvent );\r
-                               } else {\r
+                       switch( that[ '_rawType' ] ){\r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :\r
+                                       raw.RemoveEventListener( type, list.sltoken ); // token\r
+                                       X_Callback_correct( list.slcallback );\r
+                                       delete list.sltoken;\r
+                                       delete list.slcallback;\r
+                                       break;\r
+                               \r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :\r
+                                       X_Callback_correct( raw[ 'on' + type ] );\r
                                        raw[ 'on' + type ] = X.emptyFunction;\r
                                        raw[ 'on' + type ] = '';\r
-                               };\r
-                               \r
-                               if( !skip ){\r
-                                       X_Callback_correct( that._handleEvent );\r
-                                       delete that._handleEvent;\r
-                               };\r
+                                       break;\r
+\r
+                               default :\r
+                                       if( raw.attachEvent ){\r
+                                               raw.detachEvent( 'on' + type, that[ '_listeners' ]._handleEvent );\r
+                                       } else {\r
+                                               raw[ 'on' + type ] = X.emptyFunction;\r
+                                               raw[ 'on' + type ] = '';\r
+                                       };\r
+                                       \r
+                                       if( !skip ){\r
+                                               X_Callback_correct( that[ '_listeners' ]._handleEvent );\r
+                                               delete that[ '_listeners' ]._handleEvent;\r
+                                       };\r
                        };\r
                }) :\r
                (function( that, type, raw, list, skip ){\r
-                       if( that._isXHR ){\r
-                               X_Callback_correct( raw[ 'on' + type ] );\r
-                               raw[ 'on' + type ] = X.emptyFunction;\r
-                               raw[ 'on' + type ] = '';\r
-                       } else\r
-                       if( that._isSilverlight ){\r
-                               raw.RemoveEventListener( type, list.sltoken ); // token\r
-                               X_Callback_correct( list.slcallback );\r
-                               delete list.sltoken;\r
-                               delete list.slcallback;\r
-                       } else {\r
-                               raw[ 'on' + type ] = X.emptyFunction;\r
-                               raw[ 'on' + type ] = '';\r
+                       switch( that[ '_rawType' ] ){\r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.SILVER_LIGHT :\r
+                                       raw.RemoveEventListener( type, list.sltoken ); // token\r
+                                       X_Callback_correct( list.slcallback );\r
+                                       delete list.sltoken;\r
+                                       delete list.slcallback;\r
+                                       break;\r
                                \r
-                               if( !skip ){\r
-                                       X_Callback_correct( that._handleEvent );\r
-                                       delete that._handleEvent;\r
-                               };\r
+                               case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :\r
+                                       X_Callback_correct( raw[ 'on' + type ] );\r
+                                       raw[ 'on' + type ] = X.emptyFunction;\r
+                                       raw[ 'on' + type ] = '';\r
+                                       break;\r
+\r
+                               default :\r
+                                       raw[ 'on' + type ] = X.emptyFunction;\r
+                                       raw[ 'on' + type ] = '';\r
+                                       \r
+                                       if( !skip ){\r
+                                               X_Callback_correct( that[ '_listeners' ]._handleEvent );\r
+                                               delete that[ '_listeners' ]._handleEvent;\r
+                                       };\r
                        };\r
                });\r
 \r
@@ -762,13 +805,14 @@ if( X_UA.WebKit < 525.13 ){ // Safari3-
 // イベントの退避、dom が画面から抜かれる場合に実施しておく\r
 // 退避したイベントの復帰\r
 function X_EventDispatcher_toggleAllEvents( that, add ){\r
-       var list = that._listeners,\r
+       var list = that[ '_listeners' ],\r
                raw  = that._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( that ),\r
                f    = add ? X_EventDispatcher_addEvent : X_EventDispatcher_removeEvent,\r
                type;\r
        if( !list || !raw ) return;\r
        for( type in list ){\r
                //if( X_EMPTY_OBJECT[ type ] ) continue;\r
+               if( type.charAt( 0 ) === '_' ) continue;\r
                // 数字イベントの除外\r
                if( '' + parseFloat( type ) !== type ){\r
                        // TODO type rename はここ\r