UNPKG

@angular/cdk

Version:

Angular Material Component Development Kit

129 lines 15.8 kB
/** * @license * Copyright Google LLC 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 */ import { __awaiter, __extends, __generator } from "tslib"; import { HarnessEnvironment } from '@angular/cdk/testing'; import { flush } from '@angular/core/testing'; import { takeWhile } from 'rxjs/operators'; import { TaskStateZoneInterceptor } from './task-state-zone-interceptor'; import { UnitTestElement } from './unit-test-element'; /** A `HarnessEnvironment` implementation for Angular's Testbed. */ var TestbedHarnessEnvironment = /** @class */ (function (_super) { __extends(TestbedHarnessEnvironment, _super); function TestbedHarnessEnvironment(rawRootElement, _fixture) { var _this = _super.call(this, rawRootElement) || this; _this._fixture = _fixture; _this._destroyed = false; _this._taskState = TaskStateZoneInterceptor.setup(); _fixture.componentRef.onDestroy(function () { return _this._destroyed = true; }); return _this; } /** Creates a `HarnessLoader` rooted at the given fixture's root element. */ TestbedHarnessEnvironment.loader = function (fixture) { return new TestbedHarnessEnvironment(fixture.nativeElement, fixture); }; /** * Creates a `HarnessLoader` at the document root. This can be used if harnesses are * located outside of a fixture (e.g. overlays appended to the document body). */ TestbedHarnessEnvironment.documentRootLoader = function (fixture) { return new TestbedHarnessEnvironment(document.body, fixture); }; /** * Creates an instance of the given harness type, using the fixture's root element as the * harness's host element. This method should be used when creating a harness for the root element * of a fixture, as components do not have the correct selector when they are created as the root * of the fixture. */ TestbedHarnessEnvironment.harnessForFixture = function (fixture, harnessType) { return __awaiter(this, void 0, void 0, function () { var environment; return __generator(this, function (_a) { switch (_a.label) { case 0: environment = new TestbedHarnessEnvironment(fixture.nativeElement, fixture); return [4 /*yield*/, environment.forceStabilize()]; case 1: _a.sent(); return [2 /*return*/, environment.createComponentHarness(harnessType, fixture.nativeElement)]; } }); }); }; TestbedHarnessEnvironment.prototype.forceStabilize = function () { return __awaiter(this, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: if (this._destroyed) { throw Error('Harness is attempting to use a fixture that has already been destroyed.'); } this._fixture.detectChanges(); return [4 /*yield*/, this._fixture.whenStable()]; case 1: _a.sent(); return [2 /*return*/]; } }); }); }; TestbedHarnessEnvironment.prototype.waitForTasksOutsideAngular = function () { return __awaiter(this, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: // If we run in the fake async zone, we run "flush" to run any scheduled tasks. This // ensures that the harnesses behave inside of the FakeAsyncTestZone similar to the // "AsyncTestZone" and the root zone (i.e. neither fakeAsync or async). Note that we // cannot just rely on the task state observable to become stable because the state will // never change. This is because the task queue will be only drained if the fake async // zone is being flushed. if (Zone.current.get('FakeAsyncTestZoneSpec')) { flush(); } // Wait until the task queue has been drained and the zone is stable. Note that // we cannot rely on "fixture.whenStable" since it does not catch tasks scheduled // outside of the Angular zone. For test harnesses, we want to ensure that the // app is fully stabilized and therefore need to use our own zone interceptor. return [4 /*yield*/, this._taskState.pipe(takeWhile(function (state) { return !state.stable; })).toPromise()]; case 1: // Wait until the task queue has been drained and the zone is stable. Note that // we cannot rely on "fixture.whenStable" since it does not catch tasks scheduled // outside of the Angular zone. For test harnesses, we want to ensure that the // app is fully stabilized and therefore need to use our own zone interceptor. _a.sent(); return [2 /*return*/]; } }); }); }; TestbedHarnessEnvironment.prototype.getDocumentRoot = function () { return document.body; }; TestbedHarnessEnvironment.prototype.createTestElement = function (element) { var _this = this; return new UnitTestElement(element, function () { return _this.forceStabilize(); }); }; TestbedHarnessEnvironment.prototype.createEnvironment = function (element) { return new TestbedHarnessEnvironment(element, this._fixture); }; TestbedHarnessEnvironment.prototype.getAllRawElements = function (selector) { return __awaiter(this, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: return [4 /*yield*/, this.forceStabilize()]; case 1: _a.sent(); return [2 /*return*/, Array.from(this.rawRootElement.querySelectorAll(selector))]; } }); }); }; return TestbedHarnessEnvironment; }(HarnessEnvironment)); export { TestbedHarnessEnvironment }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"testbed-harness-environment.js","sourceRoot":"","sources":["../../../../../../../../../../../../src/cdk/testing/testbed/testbed-harness-environment.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport {\n  ComponentHarness,\n  ComponentHarnessConstructor,\n  HarnessEnvironment,\n  HarnessLoader,\n  TestElement\n} from '@angular/cdk/testing';\nimport {ComponentFixture, flush} from '@angular/core/testing';\nimport {Observable} from 'rxjs';\nimport {takeWhile} from 'rxjs/operators';\nimport {TaskState, TaskStateZoneInterceptor} from './task-state-zone-interceptor';\nimport {UnitTestElement} from './unit-test-element';\n\n\n/** A `HarnessEnvironment` implementation for Angular's Testbed. */\nexport class TestbedHarnessEnvironment extends HarnessEnvironment<Element> {\n  private _destroyed = false;\n\n  /** Observable that emits whenever the test task state changes. */\n  private _taskState: Observable<TaskState>;\n\n  protected constructor(rawRootElement: Element, private _fixture: ComponentFixture<unknown>) {\n    super(rawRootElement);\n    this._taskState = TaskStateZoneInterceptor.setup();\n    _fixture.componentRef.onDestroy(() => this._destroyed = true);\n  }\n\n  /** Creates a `HarnessLoader` rooted at the given fixture's root element. */\n  static loader(fixture: ComponentFixture<unknown>): HarnessLoader {\n    return new TestbedHarnessEnvironment(fixture.nativeElement, fixture);\n  }\n\n  /**\n   * Creates a `HarnessLoader` at the document root. This can be used if harnesses are\n   * located outside of a fixture (e.g. overlays appended to the document body).\n   */\n  static documentRootLoader(fixture: ComponentFixture<unknown>): HarnessLoader {\n    return new TestbedHarnessEnvironment(document.body, fixture);\n  }\n\n  /**\n   * Creates an instance of the given harness type, using the fixture's root element as the\n   * harness's host element. This method should be used when creating a harness for the root element\n   * of a fixture, as components do not have the correct selector when they are created as the root\n   * of the fixture.\n   */\n  static async harnessForFixture<T extends ComponentHarness>(\n      fixture: ComponentFixture<unknown>, harnessType: ComponentHarnessConstructor<T>): Promise<T> {\n    const environment = new TestbedHarnessEnvironment(fixture.nativeElement, fixture);\n    await environment.forceStabilize();\n    return environment.createComponentHarness(harnessType, fixture.nativeElement);\n  }\n\n  async forceStabilize(): Promise<void> {\n    if (this._destroyed) {\n      throw Error('Harness is attempting to use a fixture that has already been destroyed.');\n    }\n\n    this._fixture.detectChanges();\n    await this._fixture.whenStable();\n  }\n\n  async waitForTasksOutsideAngular(): Promise<void> {\n    // If we run in the fake async zone, we run \"flush\" to run any scheduled tasks. This\n    // ensures that the harnesses behave inside of the FakeAsyncTestZone similar to the\n    // \"AsyncTestZone\" and the root zone (i.e. neither fakeAsync or async). Note that we\n    // cannot just rely on the task state observable to become stable because the state will\n    // never change. This is because the task queue will be only drained if the fake async\n    // zone is being flushed.\n    if (Zone!.current.get('FakeAsyncTestZoneSpec')) {\n      flush();\n    }\n\n    // Wait until the task queue has been drained and the zone is stable. Note that\n    // we cannot rely on \"fixture.whenStable\" since it does not catch tasks scheduled\n    // outside of the Angular zone. For test harnesses, we want to ensure that the\n    // app is fully stabilized and therefore need to use our own zone interceptor.\n    await this._taskState.pipe(takeWhile(state => !state.stable)).toPromise();\n  }\n\n  protected getDocumentRoot(): Element {\n    return document.body;\n  }\n\n  protected createTestElement(element: Element): TestElement {\n    return new UnitTestElement(element, () => this.forceStabilize());\n  }\n\n  protected createEnvironment(element: Element): HarnessEnvironment<Element> {\n    return new TestbedHarnessEnvironment(element, this._fixture);\n  }\n\n  protected async getAllRawElements(selector: string): Promise<Element[]> {\n    await this.forceStabilize();\n    return Array.from(this.rawRootElement.querySelectorAll(selector));\n  }\n}\n"]}