2016-06-23 09:47:54 -07:00
|
|
|
/**
|
|
|
|
* @license
|
|
|
|
* Copyright Google Inc. All Rights Reserved.
|
|
|
|
*
|
|
|
|
* Use of this source code is governed by an MIT-style license that can be
|
|
|
|
* found in the LICENSE file at https://angular.io/license
|
|
|
|
*/
|
|
|
|
|
2016-06-08 16:38:52 -07:00
|
|
|
import {Injectable} from '../di/decorators';
|
|
|
|
import {ObservableWrapper} from '../facade/async';
|
2016-05-31 15:22:59 -07:00
|
|
|
import {Map, MapWrapper} from '../facade/collection';
|
|
|
|
import {BaseException} from '../facade/exceptions';
|
2016-06-08 16:38:52 -07:00
|
|
|
import {scheduleMicroTask} from '../facade/lang';
|
2015-10-08 13:33:22 -07:00
|
|
|
import {NgZone} from '../zone/ng_zone';
|
2016-06-08 16:38:52 -07:00
|
|
|
|
2015-03-23 16:46:18 -07:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The Testability service provides testing hooks that can be accessed from
|
|
|
|
* the browser and by services such as Protractor. Each bootstrapped Angular
|
|
|
|
* application on the page will have an instance of Testability.
|
2016-05-25 15:00:05 -07:00
|
|
|
* @experimental
|
2015-03-23 16:46:18 -07:00
|
|
|
*/
|
2015-03-31 12:36:43 -07:00
|
|
|
@Injectable()
|
2015-03-23 16:46:18 -07:00
|
|
|
export class Testability {
|
2015-10-09 17:21:25 -07:00
|
|
|
/** @internal */
|
2015-07-24 12:46:12 -07:00
|
|
|
_pendingCount: number = 0;
|
2016-03-28 14:25:22 -07:00
|
|
|
/** @internal */
|
2016-02-25 14:24:17 -08:00
|
|
|
_isZoneStable: boolean = true;
|
2016-01-05 12:56:24 -08:00
|
|
|
/**
|
|
|
|
* Whether any work was done since the last 'whenStable' callback. This is
|
|
|
|
* useful to detect if this could have potentially destabilized another
|
|
|
|
* component while it is stabilizing.
|
|
|
|
* @internal
|
|
|
|
*/
|
|
|
|
_didWork: boolean = false;
|
2015-10-09 17:21:25 -07:00
|
|
|
/** @internal */
|
2015-08-28 11:29:19 -07:00
|
|
|
_callbacks: Function[] = [];
|
2016-02-25 14:24:17 -08:00
|
|
|
constructor(private _ngZone: NgZone) { this._watchAngularEvents(); }
|
2015-03-23 16:46:18 -07:00
|
|
|
|
2015-10-09 17:21:25 -07:00
|
|
|
/** @internal */
|
2016-02-25 14:24:17 -08:00
|
|
|
_watchAngularEvents(): void {
|
|
|
|
ObservableWrapper.subscribe(this._ngZone.onUnstable, (_) => {
|
2016-01-05 12:56:24 -08:00
|
|
|
this._didWork = true;
|
2016-02-25 14:24:17 -08:00
|
|
|
this._isZoneStable = false;
|
2016-01-05 12:56:24 -08:00
|
|
|
});
|
2015-11-03 18:45:55 -08:00
|
|
|
|
2016-02-25 14:24:17 -08:00
|
|
|
this._ngZone.runOutsideAngular(() => {
|
|
|
|
ObservableWrapper.subscribe(this._ngZone.onStable, (_) => {
|
|
|
|
NgZone.assertNotInAngularZone();
|
|
|
|
scheduleMicroTask(() => {
|
|
|
|
this._isZoneStable = true;
|
2015-11-03 18:45:55 -08:00
|
|
|
this._runCallbacksIfReady();
|
2016-02-25 14:24:17 -08:00
|
|
|
});
|
2015-11-03 18:45:55 -08:00
|
|
|
});
|
|
|
|
});
|
2015-03-23 16:46:18 -07:00
|
|
|
}
|
|
|
|
|
2015-07-24 12:46:12 -07:00
|
|
|
increasePendingRequestCount(): number {
|
|
|
|
this._pendingCount += 1;
|
2016-01-05 12:56:24 -08:00
|
|
|
this._didWork = true;
|
2015-07-24 12:46:12 -07:00
|
|
|
return this._pendingCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
decreasePendingRequestCount(): number {
|
|
|
|
this._pendingCount -= 1;
|
2015-03-23 16:46:18 -07:00
|
|
|
if (this._pendingCount < 0) {
|
|
|
|
throw new BaseException('pending async requests below zero');
|
|
|
|
}
|
2015-07-24 12:46:12 -07:00
|
|
|
this._runCallbacksIfReady();
|
2015-03-23 16:46:18 -07:00
|
|
|
return this._pendingCount;
|
|
|
|
}
|
|
|
|
|
2016-02-25 14:24:17 -08:00
|
|
|
isStable(): boolean {
|
|
|
|
return this._isZoneStable && this._pendingCount == 0 && !this._ngZone.hasPendingMacrotasks;
|
|
|
|
}
|
2015-08-27 17:44:59 +02:00
|
|
|
|
2015-10-09 17:21:25 -07:00
|
|
|
/** @internal */
|
2015-07-24 12:46:12 -07:00
|
|
|
_runCallbacksIfReady(): void {
|
2016-02-25 14:24:17 -08:00
|
|
|
if (this.isStable()) {
|
|
|
|
// Schedules the call backs in a new frame so that it is always async.
|
|
|
|
scheduleMicroTask(() => {
|
|
|
|
while (this._callbacks.length !== 0) {
|
|
|
|
(this._callbacks.pop())(this._didWork);
|
|
|
|
}
|
|
|
|
this._didWork = false;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
// Not Ready
|
2016-01-05 12:56:24 -08:00
|
|
|
this._didWork = true;
|
2015-03-23 16:46:18 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-07-24 12:46:12 -07:00
|
|
|
whenStable(callback: Function): void {
|
2015-06-17 11:17:21 -07:00
|
|
|
this._callbacks.push(callback);
|
2015-07-24 12:46:12 -07:00
|
|
|
this._runCallbacksIfReady();
|
2015-03-23 16:46:18 -07:00
|
|
|
}
|
|
|
|
|
2015-07-24 12:46:12 -07:00
|
|
|
getPendingRequestCount(): number { return this._pendingCount; }
|
|
|
|
|
2015-10-10 22:11:13 -07:00
|
|
|
findBindings(using: any, provider: string, exactMatch: boolean): any[] {
|
|
|
|
// TODO(juliemr): implement.
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
findProviders(using: any, provider: string, exactMatch: boolean): any[] {
|
2015-03-23 16:46:18 -07:00
|
|
|
// TODO(juliemr): implement.
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-12-03 15:49:09 -08:00
|
|
|
/**
|
|
|
|
* A global registry of {@link Testability} instances for specific elements.
|
2016-05-25 15:00:05 -07:00
|
|
|
* @experimental
|
2015-12-03 15:49:09 -08:00
|
|
|
*/
|
2015-03-31 12:36:43 -07:00
|
|
|
@Injectable()
|
2015-03-23 16:46:18 -07:00
|
|
|
export class TestabilityRegistry {
|
2015-10-09 17:21:25 -07:00
|
|
|
/** @internal */
|
2015-09-29 11:11:06 -07:00
|
|
|
_applications = new Map<any, Testability>();
|
2015-03-23 16:46:18 -07:00
|
|
|
|
2015-11-17 15:24:36 -08:00
|
|
|
constructor() { _testabilityGetter.addToWindow(this); }
|
2015-03-23 16:46:18 -07:00
|
|
|
|
2015-07-07 20:03:00 -07:00
|
|
|
registerApplication(token: any, testability: Testability) {
|
2015-06-17 16:21:40 -07:00
|
|
|
this._applications.set(token, testability);
|
2015-03-23 16:46:18 -07:00
|
|
|
}
|
|
|
|
|
2015-11-17 15:24:36 -08:00
|
|
|
getTestability(elem: any): Testability { return this._applications.get(elem); }
|
|
|
|
|
2015-08-28 11:29:19 -07:00
|
|
|
getAllTestabilities(): Testability[] { return MapWrapper.values(this._applications); }
|
2015-07-30 15:51:06 -07:00
|
|
|
|
2016-02-18 13:51:20 -08:00
|
|
|
getAllRootElements(): any[] { return MapWrapper.keys(this._applications); }
|
|
|
|
|
2015-08-11 22:34:59 -07:00
|
|
|
findTestabilityInTree(elem: Node, findInAncestors: boolean = true): Testability {
|
2015-11-17 15:24:36 -08:00
|
|
|
return _testabilityGetter.findTestabilityInTree(this, elem, findInAncestors);
|
2015-03-23 16:46:18 -07:00
|
|
|
}
|
|
|
|
}
|
2015-09-02 15:19:26 -07:00
|
|
|
|
2015-12-03 15:49:09 -08:00
|
|
|
/**
|
|
|
|
* Adapter interface for retrieving the `Testability` service associated for a
|
|
|
|
* particular context.
|
|
|
|
*/
|
2015-11-17 15:24:36 -08:00
|
|
|
export interface GetTestability {
|
|
|
|
addToWindow(registry: TestabilityRegistry): void;
|
2016-06-08 16:38:52 -07:00
|
|
|
findTestabilityInTree(registry: TestabilityRegistry, elem: any, findInAncestors: boolean):
|
|
|
|
Testability;
|
2015-11-17 15:24:36 -08:00
|
|
|
}
|
2015-09-02 15:19:26 -07:00
|
|
|
|
2016-04-25 21:58:48 -07:00
|
|
|
/* @ts2dart_const */
|
2015-11-17 15:24:36 -08:00
|
|
|
class _NoopGetTestability implements GetTestability {
|
2015-09-02 15:19:26 -07:00
|
|
|
addToWindow(registry: TestabilityRegistry): void {}
|
2016-06-08 16:38:52 -07:00
|
|
|
findTestabilityInTree(registry: TestabilityRegistry, elem: any, findInAncestors: boolean):
|
|
|
|
Testability {
|
2015-11-17 15:24:36 -08:00
|
|
|
return null;
|
|
|
|
}
|
2015-09-02 15:19:26 -07:00
|
|
|
}
|
|
|
|
|
2015-12-03 15:49:09 -08:00
|
|
|
/**
|
|
|
|
* Set the {@link GetTestability} implementation used by the Angular testing framework.
|
2016-05-25 15:00:05 -07:00
|
|
|
* @experimental
|
2015-12-03 15:49:09 -08:00
|
|
|
*/
|
2015-09-02 15:19:26 -07:00
|
|
|
export function setTestabilityGetter(getter: GetTestability): void {
|
2015-11-17 15:24:36 -08:00
|
|
|
_testabilityGetter = getter;
|
2015-09-02 15:19:26 -07:00
|
|
|
}
|
|
|
|
|
2016-04-25 21:47:33 -07:00
|
|
|
var _testabilityGetter: GetTestability = /*@ts2dart_const*/ new _NoopGetTestability();
|