OSDN Git Service

Version 0.6.137, fix X.EventDispatcher.unlisten & remove X.Node.destroy.
[pettanr/clientJs.git] / 0.6.x / js / 01_core / 13_XEventDispatcher.js
index 29ec939..946cbfc 100644 (file)
@@ -1,26 +1,35 @@
-/*\r
- * <p>EventDispatcher インスタンスのメンバ(_listeners)でイベントリスナをイベント名(string)やイベントID(5~以上の number, フレームワーク内で定義、5 以上になる理由は後述)をキーとする Array で記憶します。\r
- * Arrayには、__CallbackHash__ というハッシュ、または関数が蓄えられています。\r
+/**\r
+ * <p>EventDispatcher インスタンスのメンバ(_listeners)でイベントリスナをイベント名(string)や\r
+ * イベントID(5~以上の number, フレームワーク内で定義、5 以上になる理由は後述)をキーとする Array で記憶します。\r
+ * \r
+ * <p>Arrayには、__CallbackHash__ というハッシュ、または関数が蓄えられています。\r
  * \r
- * また、dispatch 中の状態と操作を記録し不整合が起きないようにするためのプロパティを持ち、0 から 4 の番号が与えられています。\r
+ * <p>また、イベントターゲット(EventDispatcher[ '_rawObject' ])に渡された再利用可能クロージャの控えを _listeners[0] に記憶します。(ACTUAL_HANDLER)\r
+ * \r
+ * <p>dispatch 中の状態と操作を記録し不整合が起きないようにするためのプロパティ(_listeners[1]~_listeners[4])を持ちます。イベントID が 5 から始まるのはこのためです。\r
  * \r
  * <dl>\r
  * <dt>0:ACTUAL_HANDLER\r
  * <dd>イベントターゲットの addEventListener 等に渡される実際の関数(再利用可能クロージャ)を控えています。\r
  * <dt>1:DISPATCHING number\r
- * <dd>イベントディスパッチ中か?またディスパッチがネストした場合、その深さを記憶します。\r
+ * <dd>dispatch 中か?さらにインスタンス自身の dispatch がネストした場合、その深さを記憶します。\r
  * <dt>2:RESERVES Array\r
- * <dd>ã\82¤ã\83\99ã\83³ã\83\88ç\99ºç\81«ä¸­ã\81« listen() ã\81\8cå\91¼ã\81°ã\82\8cã\81\9få ´å\90\88ã\81«å¼\95æ\95°ã\82\92è\93\84ã\81\88ã\80\81å\85¨ã\81¦ã\81®ã\83\87ã\82£ã\82¹ã\83\91ã\83\83ã\83\81ã\81®å®\8cäº\86æ\99\82(_dispatching===0)に再度 listen() するための一時ストアです。\r
+ * <dd>ã\82¤ã\83\99ã\83³ã\83\88ç\99ºç\81«ä¸­ã\81« listen() ã\81\8cå\91¼ã\81°ã\82\8cã\81\9få ´å\90\88ã\81«å¼\95æ\95°ã\82\92è\93\84ã\81\88ã\80\81å®\8cäº\86æ\99\82(DISPATCHING===0)に再度 listen() するための一時ストアです。\r
  * <dt>3:UNLISTENS Array\r
- * <dd>ã\82¤ã\83\99ã\83³ã\83\88ç\99ºç\81«ä¸­ã\81« unlisten() ã\81\8cå\91¼ã\81°ã\82\8cã\81\9få ´å\90\88ã\81«å¯¾è±¡ã\83ªã\82¹ã\83\8aã\82\92è¨\98æ\86¶ã\81\97ã\80\81ã\83ªã\82¹ã\83\8aã\81\8cå\91¼ã\81°ã\82\8cã\81ªã\81\84ã\82\88ã\81\86ã\81«ã\81\97ã\81¾ã\81\99ã\80\82å\85¨ã\81¦ã\81®ã\83\87ã\82£ã\82¹ã\83\91ã\83\83ã\83\81ã\81®å®\8cäº\86æ\99\82(_dispatching===0)に再度 unlisten() します。\r
+ * <dd>ã\82¤ã\83\99ã\83³ã\83\88ç\99ºç\81«ä¸­ã\81« unlisten() ã\81\8cå\91¼ã\81°ã\82\8cã\81\9få ´å\90\88ã\81«å¯¾è±¡ã\83ªã\82¹ã\83\8aã\82\92è¨\98æ\86¶ã\81\97ã\80\81ã\83ªã\82¹ã\83\8aã\81\8cå\91¼ã\81°ã\82\8cã\81ªã\81\84ã\82\88ã\81\86ã\81«ã\81\97ã\81¾ã\81\99ã\80\82å®\8cäº\86æ\99\82(DISPATCHING===0)に再度 unlisten() します。\r
  * <dt>4:KILL_RESERVED boolean\r
- * <dd>イベント発火中に kill() が呼ばれた場合に、全てのディスパッチの完了時(_dispatching===0)に再度 kill() するためのフラグです。\r
+ * <dd>dispatch 中に kill() が呼ばれた場合に一旦 kill をキャンセルし、完了時(DISPATCHING===0)に再度 kill() するためのフラグです。\r
  * </dl>\r
+ * \r
+ * @class __X_EventDispatcher_Listeners__\r
+ * @private\r
+ * @abstract\r
  */\r
-var X_Listeners,\r
-\r
+var\r
        /** @enum {number} */\r
-       X_Listeners_ = {\r
+       X_Listeners_ =\r
+               /** @lends __X_EventDispatcher_Listeners__ */\r
+               {\r
                        ACTUAL_HANDLER : 0,\r
                        DISPATCHING    : 1,\r
                        RESERVES       : 2,\r
@@ -41,6 +50,9 @@ var X_EventDispatcher_once       = false,
        \r
        X_EventDispatcher_safariPreventDefault = false, // Safari3-\r
        \r
+       /**\r
+        * @enum {number} \r
+        */\r
        X_EventDispatcher_EVENT_TARGET_TYPE = {\r
                OTHER        : 0,\r
                XHR          : 1,\r
@@ -78,7 +90,7 @@ var X_EventDispatcher_once       = false,
  * <p>listen, unlisten, dispatch という addEventListener, removeEventListener, dispatchEvent に対応する関数を持ちます。\r
  * また listening という ActionScript3 の hasEventListener に相当する関数を持ちます。\r
  * \r
- * <p>イベントターゲットオブジェクト(widnow, document, HTMLElement, XHR, Silverlight 等)が this._rawObject に設定されていた場合に、それらへ実際のイベント登録・解除も行います。\r
+ * <p>イベントターゲットオブジェクト(widnow, document, HTMLElement, XHR, Silverlight 等)が this[ '_rawObject' ] に設定されていた場合に、それらへ実際のイベント登録・解除も行います。\r
  * このイベントの登録・解除はクロスブラウザで、IE5~8 の独自イベントの差異を吸収し、DOM0 に対しても複数のコールバックを登録することができます。\r
  * \r
  * <p>またコールバックに対して、this コンテキストや、追加の引数を指定もできます。 this コンテキストを指定しなかった場合、EventDispatcher インスタンスがコールバックの this になります。 \r
@@ -93,8 +105,8 @@ var X_EventDispatcher_once       = false,
  * @constructs EventDispatcher\r
  * @extends {__ClassBase__}\r
  */\r
-var EventDispatcher = X.EventDispatcher =\r
-       X.Class.create(\r
+var X_EventDispatcher = X[ 'EventDispatcher' ] =\r
+       X_Class_create(\r
                'EventDispatcher',\r
                \r
            /** @lends EventDispatcher.prototype */\r
@@ -112,7 +124,7 @@ var EventDispatcher = X.EventDispatcher =
                 * Arrayには、{k:種類,x:コンテキスト(thisObject),f:関数,s:サプリメントする引数の配列} というハッシュ、または関数が蓄えられています。\r
                 * \r
                 * @private\r
-                * @type {X_Listeners}\r
+                * @type {__X_EventDispatcher_Listeners__}\r
                 */\r
                        '_listeners'    : null,\r
 \r
@@ -132,15 +144,15 @@ var EventDispatcher = X.EventDispatcher =
                 * アプリケーション独自のイベントをやり取りしたいだけ、という場合イベントターゲットは指定しません。\r
             * @param {object=} opt_rawObject\r
             */\r
-                       Constructor : function( opt_rawObject ){\r
+                       'Constructor' : function( opt_rawObject ){\r
                                if( opt_rawObject ){\r
-                                       this._rawObject = opt_rawObject;\r
+                                       this[ '_rawObject' ] = opt_rawObject;\r
                                };\r
                        },\r
 \r
-                       dispatch : X_EventDispatcher_dispatch,\r
+                       'dispatch' : X_EventDispatcher_dispatch,\r
                        \r
-                       listen : X_EventDispatcher_listen,\r
+                       'listen' : X_EventDispatcher_listen,\r
                \r
                /**\r
                 * dispatch 時に自動で unlisten されるフラグを立てて listen する。\r
@@ -150,24 +162,24 @@ var EventDispatcher = X.EventDispatcher =
                 * @param {Array} [opt_arg3=] コールバック時の引数を配列に入れる。引数がひとつでも配列を使用する。省略した場合引数なし。unlisten() に使用するので、配列も適宜に保持しておくこと。\r
                 * @return {EventDispatcher} チェインメソッド\r
                 */\r
-                       listenOnce : function( type, opt_arg1, opt_arg2, opt_arg3 ){\r
+                       'listenOnce' : function( type, opt_arg1, opt_arg2, opt_arg3 ){\r
                                X_EventDispatcher_once = true;\r
-                               this.listen( type, opt_arg1, opt_arg2, opt_arg3 );\r
+                               this[ 'listen' ]( type, opt_arg1, opt_arg2, opt_arg3 );\r
                                X_EventDispatcher_once = false;\r
                                return this;\r
                        },\r
 \r
-                       unlisten : X_EventDispatcher_unlisten,\r
+                       'unlisten' : X_EventDispatcher_unlisten,\r
 \r
                /**\r
                 * <p>イベントリスナの登録状況を真偽値で返す。戻り値が数値(index)の場合もあるが、これは内部のみで使用。\r
-                * <p>this.listening(); のように type を省略した場合、一つでも登録があれば true を返す。\r
-                * <p>this.listening( 'myevent' ); と type だけを与えた場合、その type に登録があれば true を返す。\r
+                * <p>this[ 'listening' ](); のように type を省略した場合、一つでも登録があれば true を返す。\r
+                * <p>this[ 'listening' ]( 'myevent' ); と type だけを与えた場合、その type に登録があれば true を返す。\r
                 * <p>type と イベントリスナの組み合わせが登録されているかを調べる場合は、listen 時の thisObject や args(Array) も一致させて渡す必要がある。\r
                 * \r
                 * @example \r
-                *  this.listen( [ 'myevent', 'yourevent' ], this, onMyEvent, args = [ 1, 'a' ] );\r
-                *  this.listening( 'myevent', this, onMyEvent, args ) === true;\r
+                *  this[ 'listen' ]( [ 'myevent', 'yourevent' ], this, onMyEvent, args = [ 1, 'a' ] );\r
+                *  this[ 'listening' ]( 'myevent', this, onMyEvent, args ) === true;\r
                 * \r
                 * @return {number|boolean}\r
                 * @param {string|number} opt_type\r
@@ -175,7 +187,7 @@ var EventDispatcher = X.EventDispatcher =
                 * @param {function|Array} opt_arg2\r
                 * @param {Array} opt_arg3\r
                 */                     \r
-                       listening : function( opt_type, opt_arg1, opt_arg2, opt_arg3 ){\r
+                       'listening' : function( opt_type, opt_arg1, opt_arg2, opt_arg3 ){\r
                                var listeners = this[ '_listeners' ],\r
                                        lock      = X_EventDispatcher_lock || X_EventDispatcher_unlock,\r
                                        list, cbHash, unlistens, i, f;\r
@@ -184,7 +196,7 @@ var EventDispatcher = X.EventDispatcher =
                                if( !listeners || !( list = listeners[ opt_type ] ) ) return false;\r
                                if( opt_arg1 === undefined ) return true;\r
                                \r
-                               if( opt_arg1.k ){\r
+                               if( opt_arg1.kind ){\r
                                        cbHash = opt_arg1;\r
                                } else {\r
                                        cbHash = X_Callback_classifyCallbackArgs( opt_arg1, opt_arg2, opt_arg3, this );\r
@@ -193,12 +205,12 @@ var EventDispatcher = X.EventDispatcher =
                                if( ( unlistens = listeners[ X_Listeners_.UNLISTENS ] ) && ( unlistens = unlistens[ opt_type ] ) ){\r
                                        for( i = unlistens.length; i; ){\r
                                                f = unlistens[ --i ];\r
-                                               if( f === cbHash || ( f.x === cbHash.x && f.f === cbHash.f && f.s === cbHash.s && f.lock === lock ) ) return false;\r
+                                               if( f === cbHash || ( f.context === cbHash.context && f.func === cbHash.func && f.supplement === cbHash.supplement && f.lock === lock ) ) return false;\r
                                        };\r
                                };\r
                                for( i = list.length; i; ){\r
                                        f = list[ --i ];\r
-                                       if( f === cbHash || ( f.x === cbHash.x && f.f === cbHash.f && f.s === cbHash.s && f.lock === lock ) ){\r
+                                       if( f === cbHash || ( f.context === cbHash.context && f.func === cbHash.func && f.supplement === cbHash.supplement && f.lock === lock ) ){\r
                                                // index を要求された場合、lock されていない、または unlock なら index を返す\r
                                                return X_EventDispatcher_needsIndex ? i : true;\r
                                        };\r
@@ -207,22 +219,22 @@ var EventDispatcher = X.EventDispatcher =
                        },\r
 \r
                /**\r
-                * delay(ミリ秒)後にイベントを dispatch する。戻り値は uid = X.Timer.add() のタイマーID(数値)。X.Timer.remove(uid) でタイマーを解除して dispatch を中止できる。\r
-                * kill() 時には内部でまだ呼ばれていないタイマーの X.Timer.remove() が行われる。\r
-                * @example this.asyncDispatch( 'myevent' );\r
+                * delay(ミリ秒)後にイベントを dispatch する。戻り値は uid = X.Timer.once() のタイマーID(数値)。X.Timer.remove(uid) でタイマーを解除して dispatch を中止できる。\r
+                * kill() 時には内部でまだ呼ばれていないタイマーの X.Timer.remove() が行われる。インスタンスが破棄された後にタイマーが呼ばれることがないので神経質にならなくても安全に使える。\r
+                * @example this[ 'asyncDispatch' ]( 'myevent' );\r
                 * // どちらのコードも同じ動作をする。\r
-                * this.asyncDispatch( 0, 'myevent' );\r
+                * this[ 'asyncDispatch' ]( 0, 'myevent' );\r
                 * @param {number|eventHash|string} delay ms 省略した場合は 0 として扱う asyncDispatch( 'myevent' ) -> asyncDispatch( 0, 'myevent' )\r
                 * @param {eventHash|string|number} e イベントを表す数値、文字列、{ type : XXX, ... } なオブジェクト\r
                 * @return {number} X.Timer.add() の戻り値\r
                 */                     \r
-                       asyncDispatch : function( delay, e ){\r
+                       'asyncDispatch' : function( delay, e ){\r
                                var timerID;\r
                                if( delay && e === undefined ){\r
                                        e = delay;\r
                                        delay = 0;\r
                                };\r
-                               timerID = 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
@@ -252,10 +264,10 @@ function X_EventDispatcher_dispatch( e ){
        \r
        // 数値, 文字が渡された場合\r
        if( !type ){\r
-               e = { type : type = e };\r
+               e = { 'type' : type = e };\r
        };\r
-       e.target        = e.target || this;\r
-       e.currentTarget = e.currentTarget || this;\r
+       e[ 'target' ]        = e[ 'target' ] || this;\r
+       e[ 'currentTarget' ] = e[ 'currentTarget' ] || this;\r
        \r
        if( listeners[ X_Listeners_.DISPATCHING ] ){\r
                ++listeners[ X_Listeners_.DISPATCHING ];\r
@@ -286,10 +298,10 @@ function X_EventDispatcher_dispatch( e ){
                        unlistens.indexOf( f ) === -1 && ( unlistens[ unlistens.length ] = f );\r
                };\r
 \r
-               if( r & X.Callback.STOP_NOW ){\r
+               if( r & X_Callback_STOP_NOW ){\r
                        sysOnly = true;\r
                };\r
-               ret |= X.Type.isFinite( r ) ? r : 0;\r
+               ret |= X_Type_isFinite( r ) ? r : 0;\r
        };\r
        \r
        if( ( --listeners[ X_Listeners_.DISPATCHING ] ) === 0 ){\r
@@ -302,12 +314,11 @@ function X_EventDispatcher_dispatch( e ){
                                f = list[ i ];\r
                                X_EventDispatcher_once = f[ 4 ];\r
                                X_EventDispatcher_lock = f[ 5 ];\r
-                               this.listen( f[ 0 ], f[ 1 ], f[ 2 ], f[ 3 ] );\r
-                               X_EventDispatcher_once = false;\r
-                               X_EventDispatcher_lock = false;\r
+                               this[ 'listen' ]( f[ 0 ], f[ 1 ], f[ 2 ], f[ 3 ] );     \r
                                f.length = 0;\r
                        };\r
                        list.length = 0;\r
+                       X_EventDispatcher_once = X_EventDispatcher_lock = false;\r
                        delete listeners[ X_Listeners_.RESERVES ];\r
                };              \r
                \r
@@ -321,7 +332,7 @@ function X_EventDispatcher_dispatch( e ){
                                //if( X_EMPTY_OBJECT[ type ] ) continue;\r
                                list = unlistens[ type ];\r
                                for( i = list.length; i; ){\r
-                                       this.unlisten( type, list[ --i ] );\r
+                                       this[ 'unlisten' ]( type, list[ --i ] );\r
                                };\r
                                list.length = 0;\r
                                delete unlistens[ type ];\r
@@ -338,7 +349,7 @@ function X_EventDispatcher_dispatch( e ){
                        for( timerID in X_EventDispatcher_LAZY_TIMERS ){\r
                                if( X_EventDispatcher_LAZY_TIMERS[ timerID ] === this ) return ret;\r
                        }; */\r
-                       this.kill();\r
+                       this[ 'kill' ]();\r
                };\r
        };\r
        \r
@@ -350,25 +361,25 @@ function X_EventDispatcher_dispatch( e ){
  * ユーザーが触ることは無いが、システム内部でロックフラグを立てたリスナは、立てられていないフラグとは区別される。\r
  * この仕組みによってシステムの登録したリスナを、システム外から不用意に削除されることを回避する。\r
  * @example // 'myEvent' に対して、 this コンテキストを指定して 、コールバック関数を渡す。\r
- * this.listen( 'myEvent', context, func );\r
+ * this[ 'listen' ]( 'myEvent', context, func );\r
  * // 'myEvent' に対して、 this コンテキストを指定して 、コールバック関数と追加の引数を渡す。\r
  * args = [ 'arg1', 'arg2', 3 ]; // unlisten( 'myEvent', context, func, args ) で使用するので Array も控えておく。\r
- * this.listen( 'myEvent', context, func, args );\r
+ * this[ 'listen' ]( 'myEvent', context, func, args );\r
  * // 'myEvent' に対して、 listener オブジェクトを渡す。listener は handleEvent という関数を持つオブジェクトのこと。\r
  * listener.handleEvent = function( e ){};\r
- * this.listen( 'myEvent', listener );\r
+ * this[ 'listen' ]( 'myEvent', listener );\r
  * // 'myEvent' に対して、 listener オブジェクトと追加の引数を渡す。\r
  * listener.handleEvent = function( e, arg1, arg2, arg3 ){};\r
- * this.listen( 'myEvent', listener, [ arg1, arg2, arg3 ] );\r
+ * this[ 'listen' ]( 'myEvent', listener, [ arg1, arg2, arg3 ] );\r
  * // 'myEvent' に対して、 function を渡す。\r
- * this.listen( 'myEvent', onMyEvent );\r
+ * this[ 'listen' ]( 'myEvent', onMyEvent );\r
  * // 'myEvent' に対して、 function と追加の引数を渡す。\r
- * this.listen( 'myEvent', onMyEvent, args );\r
+ * this[ 'listen' ]( 'myEvent', onMyEvent, args );\r
  * // 次の二つは同じ動作です。 this コンテキストが与えられなかった場合、コールバックの this は発火元インスタンスになります。\r
- * this.listen( 'myEvent', this [, func [, args ] ] );\r
- * this.listen( 'myEvent' [, func [, args ] ] );\r
+ * this[ 'listen' ]( 'myEvent', this [, func [, args ] ] );\r
+ * this[ 'listen' ]( 'myEvent' [, func [, args ] ] );\r
  * // 複数のイベントタイプを同時に登録。コールバックは同じ指定が使われる。\r
- * this.listen( [ 'open', 'close', 'ready' ], onUpdate );\r
+ * this[ 'listen' ]( [ 'open', 'close', 'ready' ], onUpdate );\r
  * \r
  * @alias EventDispatcher.prototype.listen\r
  * @param {string|number|Array.<string,number>} type 配列を指定した場合、複数のイベントタイプに対して同じコールバックを登録する。\r
@@ -389,17 +400,17 @@ function X_EventDispatcher_listen( type, opt_arg1, opt_arg2, opt_arg3 ){
                return this;\r
        };\r
        \r
-       if( X.Type.isArray( type ) ){\r
+       if( X_Type_isArray( type ) ){\r
                for( i = type.length; i; ){\r
-                       this.listen( type[ --i ], opt_arg1, opt_arg2, opt_arg3 );\r
+                       this[ 'listen' ]( type[ --i ], opt_arg1, opt_arg2, opt_arg3 );\r
                };\r
                return this;\r
        };\r
        \r
-       raw = this._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( this );\r
-       add = raw && ( !listeners || !listeners[ type ] ) && X.Type.isString( type );\r
+       raw = this[ '_rawObject' ] || X_UA_DOM.IE4 && X_Node__ie4getRawNode( this );\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
+       if( this[ 'listening' ]( type, opt_arg1 || this, opt_arg2, opt_arg3 ) ) return this;\r
 \r
        if( !listeners ) listeners = this[ '_listeners' ] = {};\r
        list = listeners[ type ] || ( listeners[ type ] = [] );\r
@@ -419,17 +430,15 @@ function X_EventDispatcher_listen( type, opt_arg1, opt_arg2, opt_arg3 ){
  */\r
 function X_EventDispatcher_systemListen( that, type, opt_arg1, opt_arg2, opt_arg3 ){\r
        X_EventDispatcher_lock = true;\r
-       that.listen( type, opt_arg1, opt_arg2, opt_arg3 );\r
+       that[ 'listen' ]( type, opt_arg1, opt_arg2, opt_arg3 );\r
        X_EventDispatcher_lock = false;\r
 };\r
 \r
-// TODO this.listen(type) は this リスナの登録なのに、this.unlisten(type)は全てのtypeの削除、と不一致\r
-\r
 /**\r
  * イベントリスナの解除を行う。登録時と同じ引数を与える必要がある。kill() ですべてのイベントが解除されるので、途中で解除されるイベント以外は kill() に任せてしまってよい。\r
  * @alias EventDispatcher.prototype.unlisten\r
  * @return {EventDispatcher}\r
- * @param {string|number|Array.<string,number>} opt_type\r
+ * @param {string|number|Array.<string,number>} opt_type イベントID, イベント名、またはその配列\r
  * @param {listener|function|Array} opt_arg1\r
  * @param {function|Array} opt_arg2\r
  * @param {Array} opt_arg3\r
@@ -439,38 +448,16 @@ function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
                list, reserves, unlistens, i, f, raw, k, empty;\r
        if( !listeners ) return this;\r
        \r
-       if( X.Type.isArray( opt_type ) ){\r
+       if( X_Type_isArray( opt_type ) ){\r
                for( i = opt_type.length; i; ){\r
-                       this.unlisten( opt_type[ --i ], opt_arg1, opt_arg2, opt_arg3 );\r
+                       this[ 'unlisten' ]( opt_type[ --i ], opt_arg1, opt_arg2, opt_arg3 );\r
                        if( !opt_type[ i ] ){\r
                                alert( '不正な unlisten Array' );\r
                        };\r
                };\r
                return this;\r
        };\r
-       \r
-       if( opt_type === undefined ){\r
-               // 全て削除\r
-               for( opt_type in listeners ){\r
-                       //if( X_EMPTY_OBJECT[ opt_type ] ) continue;\r
-                       if( opt_type < X_Listeners_.KILL_RESERVED ) 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
-               return this;\r
-       } else\r
-       if( opt_arg1 === undefined ){\r
-               // 同一タイプを全て削除\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
+\r
        if( reserves = listeners[ X_Listeners_.RESERVES ] ){\r
                for( i = reserves.length; i; ){\r
                        f = reserves[ --i ];\r
@@ -483,7 +470,7 @@ function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
        };\r
        \r
        X_EventDispatcher_needsIndex = true;\r
-       i = this.listening( opt_type, opt_arg1, opt_arg2, opt_arg3 );\r
+       i = this[ 'listening' ]( opt_type, opt_arg1, opt_arg2, opt_arg3 );\r
        X_EventDispatcher_needsIndex = false;\r
        if( i === false ) return this;\r
 \r
@@ -498,13 +485,13 @@ function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
                delete f.once;\r
                list.splice( i, 1 );\r
                if( !list.length ){\r
-                       raw  = this._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( this );\r
+                       raw  = this[ '_rawObject' ] || X_UA_DOM.IE4 && X_Node__ie4getRawNode( this );\r
                        delete listeners[ opt_type ];\r
                        //empty = X_Object_isEmpty( listeners );\r
                        // TODO カウンター\r
                        empty = true;\r
                        for( k in listeners ){\r
-                               if( k < X_Listeners_.KILL_RESERVED ) continue;\r
+                               if( k <= X_Listeners_.KILL_RESERVED ) continue;\r
                                empty = false;\r
                                break;\r
                        };\r
@@ -522,17 +509,32 @@ function X_EventDispatcher_unlisten( opt_type, opt_arg1, opt_arg2, opt_arg3 ){
  */\r
 function X_EventDispatcher_systemUnlisten( that, type, opt_arg1, opt_arg2, opt_arg3 ){\r
        X_EventDispatcher_unlock = true;\r
-       that.unlisten( type, opt_arg1, opt_arg2, opt_arg3 );\r
+       that[ 'unlisten' ]( type, opt_arg1, opt_arg2, opt_arg3 );\r
        X_EventDispatcher_unlock = false;\r
 };\r
 \r
+function X_EventDispatcher_unlistenAll( that ){\r
+       var listeners = that[ '_listeners' ],\r
+               type, list, i;\r
+       if( !listeners ) return;\r
+       \r
+       for( type in listeners ){\r
+               //if( X_EMPTY_OBJECT[ opt_type ] ) continue;\r
+               if( type <= X_Listeners_.KILL_RESERVED ) continue;\r
+               list = listeners[ type ];\r
+               for( i = list.length; i; ){\r
+                       that[ 'unlisten' ]( type, list[ --i ] );\r
+               };\r
+       };\r
+};\r
+\r
 function X_EventDispatcher_addEvent( that, type, raw, list ){\r
        var i;\r
        X_EventDispatcher_lock || ( type = X_Event_Rename[ type ] || type );\r
        \r
-       if( X.Type.isArray( type ) ){\r
+       if( X_Type_isArray( type ) ){\r
                for( i = type.length; i; ){\r
-                       X_EventDispatcher_systemListen( that, type[ --i ], X.emptyFunction );\r
+                       X_EventDispatcher_systemListen( that, type[ --i ], X_emptyFunction );\r
                        console.log( 'events fix > ' + type[ i ] );\r
                };\r
        } else {\r
@@ -554,7 +556,7 @@ var X_EventDispatcher_actualAddEvent =
                                        break;\r
                                \r
                                case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :\r
-                                       if( X_UA.Opera < 12 ){\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
@@ -563,7 +565,7 @@ var X_EventDispatcher_actualAddEvent =
                                default :\r
                                        // iOS と MacOSX Iron36 で発生。連続してアニメーションが起こると、クロージャの束縛された obj へのアクセスに失敗する。Win では起きない?\r
                                        // むしろ、MacOSX のブラウザ全般で起こる??\r
-                                       if( ( X_UA.WebKit || X_UA.Blink ) &&\r
+                                       if( ( X_UA[ 'WebKit' ] || X_UA[ 'Blink' ] ) &&\r
                                                ( type === 'webkitTransitionEnd' || type === 'transitionend' ||\r
                                                  type === 'animationend'        || type === 'webkitAnimationEnd' ||\r
                                                  type === 'animationstart'      || type === 'webkitAnimationStart' ||\r
@@ -632,7 +634,7 @@ var X_EventDispatcher_actualAddEvent =
  * animation も怪しい、、、\r
  */\r
 function X_EventDispatcher_iOSTransitionEndDispatch( e ){\r
-       return X_Node_getXNode( this ).dispatch( X_Event_RenameTo[ e.type ] || e.type );\r
+       return X_Node_getXNode( this )[ 'dispatch' ]( X_Event_RenameTo[ e.type ] || e.type );\r
 };\r
 \r
 /*\r
@@ -640,16 +642,16 @@ function X_EventDispatcher_iOSTransitionEndDispatch( e ){
  * http://msdn.microsoft.com/ja-jp/library/cc189018%28v=vs.95%29.aspx#the_sender_parameter_and_event_data\r
  */\r
 function X_EventDispatcher_sliverLightDispatch( sender, e, type ){\r
-       return this.dispatch( type );\r
+       return this[ 'dispatch' ]( type );\r
 };\r
 \r
 function X_EventDispatcher_removeEvent( that, type, raw, list, skip ){\r
        var i;\r
        X_EventDispatcher_unlock || ( type = X_Event_Rename[ type ] || type );\r
        \r
-       if( X.Type.isArray( type ) ){\r
+       if( X_Type_isArray( type ) ){\r
                for( i = type.length; i; ){\r
-                       X_EventDispatcher_systemUnlisten( that, type[ --i ], X.emptyFunction );\r
+                       X_EventDispatcher_systemUnlisten( that, type[ --i ], X_emptyFunction );\r
                };\r
        } else {\r
                X_EventDispatcher_actualRemoveEvent( that, type, raw, list, skip );\r
@@ -668,7 +670,7 @@ var X_EventDispatcher_actualRemoveEvent =
                                        break;\r
                                \r
                                case X_EventDispatcher_EVENT_TARGET_TYPE.XHR :\r
-                                       if( X_UA.Opera < 12 ){\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
@@ -676,7 +678,7 @@ var X_EventDispatcher_actualRemoveEvent =
                                        };\r
 \r
                                default :\r
-                                       if( ( X_UA.WebKit || X_UA.Blink ) &&\r
+                                       if( ( X_UA[ 'WebKit' ] || X_UA[ 'Blink' ] ) &&\r
                                                ( type === 'webkitTransitionEnd' || type === 'transitionend' ||\r
                                                  type === 'animationend'        || type === 'webkitAnimationEnd' ||\r
                                                  type === 'animationstart'      || type === 'webkitAnimationStart' ||\r
@@ -707,7 +709,7 @@ var X_EventDispatcher_actualRemoveEvent =
                                \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 ] = X_emptyFunction;\r
                                        raw[ 'on' + type ] = '';\r
                                        break;\r
 \r
@@ -715,7 +717,7 @@ var X_EventDispatcher_actualRemoveEvent =
                                        if( raw.attachEvent ){\r
                                                raw.detachEvent( 'on' + type, that[ '_listeners' ][ X_Listeners_.ACTUAL_HANDLER ] );\r
                                        } else {\r
-                                               raw[ 'on' + type ] = X.emptyFunction;\r
+                                               raw[ 'on' + type ] = X_emptyFunction;\r
                                                raw[ 'on' + type ] = '';\r
                                        };\r
                                        \r
@@ -736,12 +738,12 @@ var X_EventDispatcher_actualRemoveEvent =
                                \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 ] = X_emptyFunction;\r
                                        raw[ 'on' + type ] = '';\r
                                        break;\r
 \r
                                default :\r
-                                       raw[ 'on' + type ] = X.emptyFunction;\r
+                                       raw[ 'on' + type ] = X_emptyFunction;\r
                                        raw[ 'on' + type ] = '';\r
                                        \r
                                        if( !skip ){\r
@@ -762,48 +764,44 @@ var X_EventDispatcher_actualHandleEvent =
                (function(){\r
                        var ret;\r
                        \r
-                       //if( event.type === 'readystatechange' && this._tag && X.Dom.Event._LOAD_FIX_TAGS[ this._tag ] ){\r
-                               //type = 'readystatechange';\r
-                       //};\r
-                       \r
-                       ret = this.dispatch( new X.Dom.Event( event, this, this._rawObject ) );\r
+                       ret = this[ 'dispatch' ]( new X_DomEvent( event, this, this[ '_rawObject' ] ) );\r
 \r
-                       if( ret & X.Callback.STOP_PROPAGATION ){\r
+                       if( ret & X_Callback_STOP_PROPAGATION ){\r
                                event.cancelBubble = true;\r
                        };\r
-                       if( ret & X.Callback.PREVENT_DEFAULT ){\r
-                               this._tag === 'A' && this._rawObject.blur();\r
+                       if( ret & X_Callback_PREVENT_DEFAULT ){\r
+                               this[ '_tag' ] === 'A' && this[ '_rawObject' ].blur();\r
                                return event.returnValue = false;\r
                        };\r
                }) :\r
        //X_UA_EVENT.W3C || X_UA_EVENT.DOM0\r
                (function( e ){\r
-                       var ev  = new X.Dom.Event( e, this ),\r
+                       var ev  = new X_DomEvent( e, this ),\r
                                ret = X_Callback_NONE,\r
                                i, l;\r
                        //console.log( '>>>>>>>>>> ' + e.type );\r
                        // touch event -> pointer\r
-                       if( X.Type.isArray( ev ) ){\r
+                       if( X_Type_isArray( ev ) ){\r
                                if( ev.length === 0 ){\r
                                        // TouchEvent の後に発生した MouseEvent のキャンセル\r
-                                       ret = X.Callback.STOP_PROPAGATION | X.Callback.PREVENT_DEFAULT;\r
+                                       ret = X_Callback_STOP_PROPAGATION | X_Callback_PREVENT_DEFAULT;\r
                                } else {\r
                                        for( i = 0, l = ev.length; i < l; ++i ){\r
                                                //console.log( 'handleEvent ' + ev[ i ].type );\r
-                                               ret |= this.dispatch( ev[ i ] ) || 0;\r
-                                       };                              \r
+                                               ret |= this[ 'dispatch' ]( ev[ i ] ) || 0;\r
+                                       };\r
                                };\r
                        } else {\r
-                               ret = this.dispatch( ev );\r
+                               ret = this[ 'dispatch' ]( ev );\r
                        };\r
                        \r
-                       if( ret & X.Callback.STOP_PROPAGATION ){\r
+                       if( ret & X_Callback_STOP_PROPAGATION ){\r
                                e.stopPropagation();\r
                        };\r
-                       if( ret & X.Callback.PREVENT_DEFAULT ){\r
-                               this._tag === 'A' && this._rawObject.blur();\r
+                       if( ret & X_Callback_PREVENT_DEFAULT ){\r
+                               this[ '_tag' ] === 'A' && this[ '_rawObject' ].blur();\r
                                e.preventDefault();\r
-                               if( X_UA.WebKit < 525.13 ){ // Safari3-\r
+                               if( X_UA[ 'WebKit' ] < 525.13 ){ // Safari3-\r
                                        if( e.type === 'click' || e.type === 'dbclick' ){\r
                                                X_EventDispatcher_safariPreventDefault = true;\r
                                        };\r
@@ -812,9 +810,9 @@ var X_EventDispatcher_actualHandleEvent =
                        };\r
                });\r
 \r
-if( X_UA.WebKit < 525.13 ){ // Safari3-\r
+if( X_UA[ 'WebKit' ] < 525.13 ){ // Safari3-\r
        document.documentElement.onclick =\r
-       document.documentElement.ondbclick = function( e ){\r
+       document.documentElement[ 'ondbclick' ] = function( e ){\r
                        if( X_EventDispatcher_safariPreventDefault ){\r
                                X_EventDispatcher_safariPreventDefault = false;\r
                                e.preventDefault();\r
@@ -827,13 +825,13 @@ if( X_UA.WebKit < 525.13 ){ // Safari3-
 // 退避したイベントの復帰\r
 function X_EventDispatcher_toggleAllEvents( that, add ){\r
        var list = that[ '_listeners' ],\r
-               raw  = that._rawObject || X_UA_DOM.IE4 && X_Node__ie4getRawNode( that ),\r
+               raw  = that[ '_rawObject' ] || X_UA_DOM.IE4 && X_Node__ie4getRawNode( that ),\r
                func = 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 < X_Listeners_.KILL_RESERVED ) continue;\r
+               //if( type <= X_Listeners_.KILL_RESERVED ) continue;\r
                // 数字イベントの除外\r
                if( !X_String_isNumberString( type ) ){\r
                        // TODO type rename はここ\r