@@ -312,6 +312,12 @@ interface ZoneType {
312
312
/** @internal */
313
313
type _PatchFn = ( global : Window , Zone : ZoneType , api : _ZonePrivate ) => void ;
314
314
315
+ /** @internal */
316
+ interface BeforeRunTaskStatus {
317
+ reEntryGuard : boolean ;
318
+ previousTask : Task ;
319
+ }
320
+
315
321
/** @internal */
316
322
interface _ZonePrivate {
317
323
currentZoneFrame : ( ) => _ZoneFrame ;
@@ -323,6 +329,8 @@ interface _ZonePrivate {
323
329
patchEventTargetMethods :
324
330
( obj : any , addFnName ?: string , removeFnName ?: string , metaCreator ?: any ) => boolean ;
325
331
patchOnProperties : ( obj : any , properties : string [ ] ) => void ;
332
+ beforeRunTask : ( zone : Zone , task : Task ) => BeforeRunTaskStatus ;
333
+ afterRunTask : ( zone : Zone , beforeRunTaskStatus : BeforeRunTaskStatus , task : Task ) => void ;
326
334
}
327
335
328
336
/** @internal */
@@ -757,8 +765,7 @@ const Zone: ZoneType = (function(global: any) {
757
765
}
758
766
}
759
767
760
-
761
- runTask ( task : Task , applyThis ?: any , applyArgs ?: any ) : any {
768
+ beforeRunTask ( task : Task ) {
762
769
if ( task . zone != this ) {
763
770
throw new Error (
764
771
'A task can only be run in the zone of creation! (Creation: ' +
@@ -772,7 +779,7 @@ const Zone: ZoneType = (function(global: any) {
772
779
// typescript compiler will complain below
773
780
const isNotScheduled = task . state === notScheduled ;
774
781
if ( isNotScheduled && task . type === eventTask ) {
775
- return ;
782
+ return null ;
776
783
}
777
784
778
785
const reEntryGuard = task . state != running ;
@@ -781,6 +788,33 @@ const Zone: ZoneType = (function(global: any) {
781
788
const previousTask = _currentTask ;
782
789
_currentTask = task ;
783
790
_currentZoneFrame = { parent : _currentZoneFrame , zone : this } ;
791
+ //(process as any)._rawDebug('currentFrame increase ', _currentZoneFrame && _currentZoneFrame.zone.name, task.source);
792
+ return {
793
+ reEntryGuard : reEntryGuard ,
794
+ previousTask : previousTask
795
+ }
796
+ }
797
+
798
+ afterRunTask ( beforeRunTaskStatus : BeforeRunTaskStatus , task : Task ) {
799
+ // if the task's state is notScheduled or unknown, then it has already been cancelled
800
+ // we should not reset the state to scheduled
801
+ if ( task . state !== notScheduled && task . state !== unknown ) {
802
+ if ( task . type == eventTask || ( task . data && task . data . isPeriodic ) ) {
803
+ beforeRunTaskStatus . reEntryGuard && ( task as ZoneTask < any > ) . _transitionTo ( scheduled , running ) ;
804
+ } else {
805
+ task . runCount = 0 ;
806
+ this . _updateTaskCount ( task as ZoneTask < any > , - 1 ) ;
807
+ beforeRunTaskStatus . reEntryGuard &&
808
+ ( task as ZoneTask < any > ) . _transitionTo ( notScheduled , running , notScheduled ) ;
809
+ }
810
+ }
811
+ _currentZoneFrame = _currentZoneFrame . parent ;
812
+ //(process as any)._rawDebug('currentFrame decrease ', _currentZoneFrame && _currentZoneFrame.zone.name, task.source);
813
+ _currentTask = beforeRunTaskStatus . previousTask ;
814
+ }
815
+
816
+ runTask ( task : Task , applyThis ?: any , applyArgs ?: any ) : any {
817
+ const beforeRunTaskStatus = this . beforeRunTask ( task ) ;
784
818
try {
785
819
if ( task . type == macroTask && task . data && ! task . data . isPeriodic ) {
786
820
task . cancelFn = null ;
@@ -793,20 +827,7 @@ const Zone: ZoneType = (function(global: any) {
793
827
}
794
828
}
795
829
} finally {
796
- // if the task's state is notScheduled or unknown, then it has already been cancelled
797
- // we should not reset the state to scheduled
798
- if ( task . state !== notScheduled && task . state !== unknown ) {
799
- if ( task . type == eventTask || ( task . data && task . data . isPeriodic ) ) {
800
- reEntryGuard && ( task as ZoneTask < any > ) . _transitionTo ( scheduled , running ) ;
801
- } else {
802
- task . runCount = 0 ;
803
- this . _updateTaskCount ( task as ZoneTask < any > , - 1 ) ;
804
- reEntryGuard &&
805
- ( task as ZoneTask < any > ) . _transitionTo ( notScheduled , running , notScheduled ) ;
806
- }
807
- }
808
- _currentZoneFrame = _currentZoneFrame . parent ;
809
- _currentTask = previousTask ;
830
+ this . afterRunTask ( beforeRunTaskStatus , task ) ;
810
831
}
811
832
}
812
833
@@ -1241,10 +1262,13 @@ const Zone: ZoneType = (function(global: any) {
1241
1262
// we must bootstrap the initial task creation by manually scheduling the drain
1242
1263
if ( _numberOfNestedTaskFrames === 0 && _microTaskQueue . length === 0 ) {
1243
1264
// We are not running in Task, so we need to kickstart the microtask queue.
1265
+ // @JiaLiPassion , use native promise if async_hooks is available
1244
1266
if ( global [ symbolPromise ] ) {
1245
1267
global [ symbolPromise ] . resolve ( 0 ) [ symbolThen ] ( drainMicroTaskQueue ) ;
1246
- } else {
1268
+ } else if ( global [ symbolSetTimeout ] ) {
1247
1269
global [ symbolSetTimeout ] ( drainMicroTaskQueue , 0 ) ;
1270
+ } else {
1271
+ Promise . resolve ( 0 ) . then ( drainMicroTaskQueue ) ;
1248
1272
}
1249
1273
}
1250
1274
task && _microTaskQueue . push ( task ) ;
@@ -1294,7 +1318,13 @@ const Zone: ZoneType = (function(global: any) {
1294
1318
scheduleMicroTask : scheduleMicroTask ,
1295
1319
showUncaughtError : ( ) => ! ( Zone as any ) [ __symbol__ ( 'ignoreConsoleErrorUncaughtError' ) ] ,
1296
1320
patchEventTargetMethods : ( ) => false ,
1297
- patchOnProperties : noop
1321
+ patchOnProperties : noop ,
1322
+ beforeRunTask : ( zone : Zone , task : Task ) => {
1323
+ return zone . beforeRunTask ( task ) ;
1324
+ } ,
1325
+ afterRunTask : ( zone : Zone , beforeRunTaskStatus : BeforeRunTaskStatus , task : Task ) => {
1326
+ return zone . afterRunTask ( beforeRunTaskStatus , task ) ;
1327
+ }
1298
1328
} ;
1299
1329
let _currentZoneFrame : _ZoneFrame = { parent : null , zone : new Zone ( null , null ) } ;
1300
1330
let _currentTask : Task = null ;
@@ -1306,7 +1336,6 @@ const Zone: ZoneType = (function(global: any) {
1306
1336
return '__zone_symbol__' + name ;
1307
1337
}
1308
1338
1309
-
1310
1339
performanceMeasure ( 'Zone' , 'Zone' ) ;
1311
1340
return global [ 'Zone' ] = Zone ;
1312
1341
} ) ( typeof window !== 'undefined' && window || typeof self !== 'undefined' && self || global ) ;
0 commit comments