OSDN Git Service

Version 0.6.163, fix __ClassBase__.kill().
[pettanr/clientJs.git] / 0.6.x / js / 01_core / 13_XClass.js
index ebe050b..2923520 100644 (file)
@@ -24,8 +24,9 @@ var
        X_Class_CALL_SUPER_STACK   = [],\r
        X_Class_traits             = null,\r
        X_Class_useObjectCreate    = false, // !!Object.create, http://jsperf.com/prototype-vs-object-create-perf\r
+       // Opera Mobile 12.10 Android11 IS01 でクラスのメンバが欠落する問題に遭遇。__proto__ を辞めると動作,,,\r
        X_Class_use_proto_         = !X_UA[ 'OperaMobile' ] && !X_UA[ 'OperaTablet' ] && !!X_emptyFunction.prototype.__proto__,\r
-               // Opera Mobile 12.10 Android11 IS01 でクラスのメンバが欠落する問題に遭遇。__proto__ を辞めると動作,,,\r
+       X_Class_SEAL_KILLING       = [],\r
 \r
 X_Class_CommonMethods =\r
 /** @lends __ClassBase__.prototype */\r
@@ -34,41 +35,71 @@ X_Class_CommonMethods =
         * 全ての動的メンバを削除して、インスタンスを破棄する。<br>\r
         * インスタンスが X.EventDispatcher とそのサブクラスの場合、次の動作をする。\r
         * <ol>\r
-        * <li>X.Event.BEFORE_KILL_INSTANCE を発火する。戻り値のビットフラグに X_Callback.PREVENT_DEFAULT が立つ場合、破棄をキャンセルし X.Event.KILL_INSTANCE_CANCELED を発火する。\r
+        * <li>X.Event.BEFORE_KILL_INSTANCE を発火する。戻り値のビットフラグに X_Callback.PREVENT_DEFAULT が立つ場合、破棄をキャンセルし X.Event.KILL_INSTANCE_CANCELED を発火する。この間に kill() が呼ばれても無視される。\r
         * <li>破棄に進む場合は、X.Event.KILL_INSTANCE を発火する。\r
         * <li>dispatch 中は、インスタンスの全ての dispatch が終了するまで実際の破棄を待つ。\r
         * <li>実際の破棄では、インスタンスのメンバの削除に加えて全てのイベントリスナを解除する。\r
         */\r
        // TODO kill したインスタンスのイベントが残っていないか?これは開発用のみ\r
        'kill' : function(){\r
-               var def, listeners, p;\r
+               var listeners, flag, p, timers, def;\r
                \r
-               // TODO kill 中の kill の呼び出しを防ぐ, 破棄済のインスタンスへの kill\r
+               // TODO 破棄済のインスタンスへの kill\r
                \r
                if( this[ 'instanceOf' ]( X_EventDispatcher ) ){\r
 \r
-                       if( this[ 'dispatch' ]( X_EVENT_BEFORE_KILL_INSTANCE ) & X_Callback_PREVENT_DEFAULT ){\r
-                               this[ 'dispatch' ]( X_EVENT_KILL_INSTANCE_CANCELED );\r
-                               return;\r
-                       };\r
-                       \r
                        listeners = this[ '_listeners' ];\r
-                       \r
-                       if( listeners && listeners[ X_LISTENERS_DISPATCHING ] ){\r
-                               listeners[ X_LISTENERS_KILL_RESERVED ] = true;\r
-                               return;\r
+\r
+                       // SEAL のタイミングは、イベント中なので listeners が存在する\r
+                       if( listeners && X_Class_SEAL_KILLING.length && X_Class_SEAL_KILLING.indexOf( this ) !== -1 ) return;\r
+\r
+                       // listeners がない場合、イベントの登録がないため、BEFORE_KILL_INSTANCE は呼ばれない。\r
+                       // KILL_RESERVED == true の場合、BEFORE_KILL_INSTANCE は呼ばれない。\r
+                       if( listeners && !listeners[ X_LISTENERS_KILL_RESERVED ] && listeners[ X_EVENT_BEFORE_KILL_INSTANCE ] ){\r
+                               X_Class_SEAL_KILLING[ X_Class_SEAL_KILLING.length ] = this;\r
+                               \r
+                               if( this[ 'dispatch' ]( X_EVENT_BEFORE_KILL_INSTANCE ) & X_Callback_PREVENT_DEFAULT ){\r
+                                       this[ 'dispatch' ]( X_EVENT_KILL_INSTANCE_CANCELED );\r
+                                       // BEFORE_KILL_INSTANCE, KILL_INSTANCE_CANCELED 内で kill() しても PREVENT_DEFAULT の場合はこれを無視する。\r
+                                       flag = true;\r
+                               };\r
+                               \r
+                               X_Class_SEAL_KILLING.length === 1 ?\r
+                                       ( X_Class_SEAL_KILLING.length = 0 ) :\r
+                                       X_Class_SEAL_KILLING.splice( X_Class_SEAL_KILLING.indexOf( this ), 1 );\r
+\r
+                               if( flag ) return;\r
                        };\r
 \r
+                       if( listeners = this[ '_listeners' ] ){// unlisten 等で listeners が破棄されている場合があるので取り直し。\r
+                               if( listeners[ X_LISTENERS_DISPATCHING ] ){\r
+                                       listeners[ X_LISTENERS_KILL_RESERVED ] = true;\r
+                                       return;\r
+                               };\r
+                               \r
+                               if( listeners[ X_EVENT_KILL_INSTANCE ] ){\r
+                                       X_Class_SEAL_KILLING[ X_Class_SEAL_KILLING.length ] = this;\r
+\r
+                                       listeners[ X_LISTENERS_KILL_RESERVED ] = false;                                 \r
+                                       this[ 'dispatch' ]( X_EVENT_KILL_INSTANCE );\r
+                                       \r
+                                       X_Class_SEAL_KILLING.length === 1 ?\r
+                                               ( X_Class_SEAL_KILLING.length = 0 ) :\r
+                                               X_Class_SEAL_KILLING.splice( X_Class_SEAL_KILLING.indexOf( this ), 1 );\r
+                               };\r
+                               \r
+                               X_EventDispatcher_unlistenAll( this );\r
+                       };\r
+\r
+                       timers = X_EventDispatcher_LAZY_TIMERS;\r
+\r
                        // asyncDispatch の削除\r
-                       for( p in X_EventDispatcher_LAZY_TIMERS ){\r
-                               if( X_EventDispatcher_LAZY_TIMERS[ p ] === this ){\r
+                       for( p in timers ){\r
+                               if( timers[ p ] === this ){\r
                                         // delete X_EventDispatcher_LAZY_TIMERS[ p ]; コレ不要\r
                                        X_Timer_remove( p );\r
                                };\r
                        };\r
-                       \r
-                       this[ 'dispatch' ]( X_EVENT_KILL_INSTANCE );\r
-                       listeners && X_EventDispatcher_unlistenAll( this );\r
                };\r
                \r
                X_Object_clear( this );\r