[6a3a178] | 1 | /**
|
---|
| 2 | * @license
|
---|
| 3 | * Copyright Google LLC All Rights Reserved.
|
---|
| 4 | *
|
---|
| 5 | * Use of this source code is governed by an MIT-style license that can be
|
---|
| 6 | * found in the LICENSE file at https://angular.io/license
|
---|
| 7 | */
|
---|
| 8 | import { __awaiter } from "tslib";
|
---|
| 9 | import { handleAutoChangeDetectionStatus, HarnessEnvironment, stopHandlingAutoChangeDetectionStatus } from '@angular/cdk/testing';
|
---|
| 10 | import { flush } from '@angular/core/testing';
|
---|
| 11 | import { takeWhile } from 'rxjs/operators';
|
---|
| 12 | import { TaskStateZoneInterceptor } from './task-state-zone-interceptor';
|
---|
| 13 | import { UnitTestElement } from './unit-test-element';
|
---|
| 14 | /** The default environment options. */
|
---|
| 15 | const defaultEnvironmentOptions = {
|
---|
| 16 | queryFn: (selector, root) => root.querySelectorAll(selector)
|
---|
| 17 | };
|
---|
| 18 | /** Whether auto change detection is currently disabled. */
|
---|
| 19 | let disableAutoChangeDetection = false;
|
---|
| 20 | /**
|
---|
| 21 | * The set of non-destroyed fixtures currently being used by `TestbedHarnessEnvironment` instances.
|
---|
| 22 | */
|
---|
| 23 | const activeFixtures = new Set();
|
---|
| 24 | /**
|
---|
| 25 | * Installs a handler for change detection batching status changes for a specific fixture.
|
---|
| 26 | * @param fixture The fixture to handle change detection batching for.
|
---|
| 27 | */
|
---|
| 28 | function installAutoChangeDetectionStatusHandler(fixture) {
|
---|
| 29 | if (!activeFixtures.size) {
|
---|
| 30 | handleAutoChangeDetectionStatus(({ isDisabled, onDetectChangesNow }) => {
|
---|
| 31 | disableAutoChangeDetection = isDisabled;
|
---|
| 32 | if (onDetectChangesNow) {
|
---|
| 33 | Promise.all(Array.from(activeFixtures).map(detectChanges)).then(onDetectChangesNow);
|
---|
| 34 | }
|
---|
| 35 | });
|
---|
| 36 | }
|
---|
| 37 | activeFixtures.add(fixture);
|
---|
| 38 | }
|
---|
| 39 | /**
|
---|
| 40 | * Uninstalls a handler for change detection batching status changes for a specific fixture.
|
---|
| 41 | * @param fixture The fixture to stop handling change detection batching for.
|
---|
| 42 | */
|
---|
| 43 | function uninstallAutoChangeDetectionStatusHandler(fixture) {
|
---|
| 44 | activeFixtures.delete(fixture);
|
---|
| 45 | if (!activeFixtures.size) {
|
---|
| 46 | stopHandlingAutoChangeDetectionStatus();
|
---|
| 47 | }
|
---|
| 48 | }
|
---|
| 49 | /** Whether we are currently in the fake async zone. */
|
---|
| 50 | function isInFakeAsyncZone() {
|
---|
| 51 | return Zone.current.get('FakeAsyncTestZoneSpec') != null;
|
---|
| 52 | }
|
---|
| 53 | /**
|
---|
| 54 | * Triggers change detection for a specific fixture.
|
---|
| 55 | * @param fixture The fixture to trigger change detection for.
|
---|
| 56 | */
|
---|
| 57 | function detectChanges(fixture) {
|
---|
| 58 | return __awaiter(this, void 0, void 0, function* () {
|
---|
| 59 | fixture.detectChanges();
|
---|
| 60 | if (isInFakeAsyncZone()) {
|
---|
| 61 | flush();
|
---|
| 62 | }
|
---|
| 63 | else {
|
---|
| 64 | yield fixture.whenStable();
|
---|
| 65 | }
|
---|
| 66 | });
|
---|
| 67 | }
|
---|
| 68 | /** A `HarnessEnvironment` implementation for Angular's Testbed. */
|
---|
| 69 | export class TestbedHarnessEnvironment extends HarnessEnvironment {
|
---|
| 70 | constructor(rawRootElement, _fixture, options) {
|
---|
| 71 | super(rawRootElement);
|
---|
| 72 | this._fixture = _fixture;
|
---|
| 73 | /** Whether the environment has been destroyed. */
|
---|
| 74 | this._destroyed = false;
|
---|
| 75 | this._options = Object.assign(Object.assign({}, defaultEnvironmentOptions), options);
|
---|
| 76 | this._taskState = TaskStateZoneInterceptor.setup();
|
---|
| 77 | installAutoChangeDetectionStatusHandler(_fixture);
|
---|
| 78 | _fixture.componentRef.onDestroy(() => {
|
---|
| 79 | uninstallAutoChangeDetectionStatusHandler(_fixture);
|
---|
| 80 | this._destroyed = true;
|
---|
| 81 | });
|
---|
| 82 | }
|
---|
| 83 | /** Creates a `HarnessLoader` rooted at the given fixture's root element. */
|
---|
| 84 | static loader(fixture, options) {
|
---|
| 85 | return new TestbedHarnessEnvironment(fixture.nativeElement, fixture, options);
|
---|
| 86 | }
|
---|
| 87 | /**
|
---|
| 88 | * Creates a `HarnessLoader` at the document root. This can be used if harnesses are
|
---|
| 89 | * located outside of a fixture (e.g. overlays appended to the document body).
|
---|
| 90 | */
|
---|
| 91 | static documentRootLoader(fixture, options) {
|
---|
| 92 | return new TestbedHarnessEnvironment(document.body, fixture, options);
|
---|
| 93 | }
|
---|
| 94 | /** Gets the native DOM element corresponding to the given TestElement. */
|
---|
| 95 | static getNativeElement(el) {
|
---|
| 96 | if (el instanceof UnitTestElement) {
|
---|
| 97 | return el.element;
|
---|
| 98 | }
|
---|
| 99 | throw Error('This TestElement was not created by the TestbedHarnessEnvironment');
|
---|
| 100 | }
|
---|
| 101 | /**
|
---|
| 102 | * Creates an instance of the given harness type, using the fixture's root element as the
|
---|
| 103 | * harness's host element. This method should be used when creating a harness for the root element
|
---|
| 104 | * of a fixture, as components do not have the correct selector when they are created as the root
|
---|
| 105 | * of the fixture.
|
---|
| 106 | */
|
---|
| 107 | static harnessForFixture(fixture, harnessType, options) {
|
---|
| 108 | return __awaiter(this, void 0, void 0, function* () {
|
---|
| 109 | const environment = new TestbedHarnessEnvironment(fixture.nativeElement, fixture, options);
|
---|
| 110 | yield environment.forceStabilize();
|
---|
| 111 | return environment.createComponentHarness(harnessType, fixture.nativeElement);
|
---|
| 112 | });
|
---|
| 113 | }
|
---|
| 114 | /**
|
---|
| 115 | * Flushes change detection and async tasks captured in the Angular zone.
|
---|
| 116 | * In most cases it should not be necessary to call this manually. However, there may be some edge
|
---|
| 117 | * cases where it is needed to fully flush animation events.
|
---|
| 118 | */
|
---|
| 119 | forceStabilize() {
|
---|
| 120 | return __awaiter(this, void 0, void 0, function* () {
|
---|
| 121 | if (!disableAutoChangeDetection) {
|
---|
| 122 | if (this._destroyed) {
|
---|
| 123 | throw Error('Harness is attempting to use a fixture that has already been destroyed.');
|
---|
| 124 | }
|
---|
| 125 | yield detectChanges(this._fixture);
|
---|
| 126 | }
|
---|
| 127 | });
|
---|
| 128 | }
|
---|
| 129 | /**
|
---|
| 130 | * Waits for all scheduled or running async tasks to complete. This allows harness
|
---|
| 131 | * authors to wait for async tasks outside of the Angular zone.
|
---|
| 132 | */
|
---|
| 133 | waitForTasksOutsideAngular() {
|
---|
| 134 | return __awaiter(this, void 0, void 0, function* () {
|
---|
| 135 | // If we run in the fake async zone, we run "flush" to run any scheduled tasks. This
|
---|
| 136 | // ensures that the harnesses behave inside of the FakeAsyncTestZone similar to the
|
---|
| 137 | // "AsyncTestZone" and the root zone (i.e. neither fakeAsync or async). Note that we
|
---|
| 138 | // cannot just rely on the task state observable to become stable because the state will
|
---|
| 139 | // never change. This is because the task queue will be only drained if the fake async
|
---|
| 140 | // zone is being flushed.
|
---|
| 141 | if (isInFakeAsyncZone()) {
|
---|
| 142 | flush();
|
---|
| 143 | }
|
---|
| 144 | // Wait until the task queue has been drained and the zone is stable. Note that
|
---|
| 145 | // we cannot rely on "fixture.whenStable" since it does not catch tasks scheduled
|
---|
| 146 | // outside of the Angular zone. For test harnesses, we want to ensure that the
|
---|
| 147 | // app is fully stabilized and therefore need to use our own zone interceptor.
|
---|
| 148 | yield this._taskState.pipe(takeWhile(state => !state.stable)).toPromise();
|
---|
| 149 | });
|
---|
| 150 | }
|
---|
| 151 | /** Gets the root element for the document. */
|
---|
| 152 | getDocumentRoot() {
|
---|
| 153 | return document.body;
|
---|
| 154 | }
|
---|
| 155 | /** Creates a `TestElement` from a raw element. */
|
---|
| 156 | createTestElement(element) {
|
---|
| 157 | return new UnitTestElement(element, () => this.forceStabilize());
|
---|
| 158 | }
|
---|
| 159 | /** Creates a `HarnessLoader` rooted at the given raw element. */
|
---|
| 160 | createEnvironment(element) {
|
---|
| 161 | return new TestbedHarnessEnvironment(element, this._fixture, this._options);
|
---|
| 162 | }
|
---|
| 163 | /**
|
---|
| 164 | * Gets a list of all elements matching the given selector under this environment's root element.
|
---|
| 165 | */
|
---|
| 166 | getAllRawElements(selector) {
|
---|
| 167 | return __awaiter(this, void 0, void 0, function* () {
|
---|
| 168 | yield this.forceStabilize();
|
---|
| 169 | return Array.from(this._options.queryFn(selector, this.rawRootElement));
|
---|
| 170 | });
|
---|
| 171 | }
|
---|
| 172 | }
|
---|
| 173 | //# 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  handleAutoChangeDetectionStatus,\n  HarnessEnvironment,\n  HarnessLoader,\n  stopHandlingAutoChangeDetectionStatus,\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/** Options to configure the environment. */\nexport interface TestbedHarnessEnvironmentOptions {\n  /** The query function used to find DOM elements. */\n  queryFn: (selector: string, root: Element) => Iterable<Element> | ArrayLike<Element>;\n}\n\n/** The default environment options. */\nconst defaultEnvironmentOptions: TestbedHarnessEnvironmentOptions = {\n  queryFn: (selector: string, root: Element) => root.querySelectorAll(selector)\n};\n\n/** Whether auto change detection is currently disabled. */\nlet disableAutoChangeDetection = false;\n\n/**\n * The set of non-destroyed fixtures currently being used by `TestbedHarnessEnvironment` instances.\n */\nconst activeFixtures = new Set<ComponentFixture<unknown>>();\n\n/**\n * Installs a handler for change detection batching status changes for a specific fixture.\n * @param fixture The fixture to handle change detection batching for.\n */\nfunction installAutoChangeDetectionStatusHandler(fixture: ComponentFixture<unknown>) {\n  if (!activeFixtures.size) {\n    handleAutoChangeDetectionStatus(({isDisabled, onDetectChangesNow}) => {\n      disableAutoChangeDetection = isDisabled;\n      if (onDetectChangesNow) {\n        Promise.all(Array.from(activeFixtures).map(detectChanges)).then(onDetectChangesNow);\n      }\n    });\n  }\n  activeFixtures.add(fixture);\n}\n\n/**\n * Uninstalls a handler for change detection batching status changes for a specific fixture.\n * @param fixture The fixture to stop handling change detection batching for.\n */\nfunction uninstallAutoChangeDetectionStatusHandler(fixture: ComponentFixture<unknown>) {\n  activeFixtures.delete(fixture);\n  if (!activeFixtures.size) {\n    stopHandlingAutoChangeDetectionStatus();\n  }\n}\n\n/** Whether we are currently in the fake async zone. */\nfunction isInFakeAsyncZone() {\n  return Zone!.current.get('FakeAsyncTestZoneSpec') != null;\n}\n\n/**\n * Triggers change detection for a specific fixture.\n * @param fixture The fixture to trigger change detection for.\n */\nasync function detectChanges(fixture: ComponentFixture<unknown>) {\n  fixture.detectChanges();\n  if (isInFakeAsyncZone()) {\n    flush();\n  } else {\n    await fixture.whenStable();\n  }\n}\n\n/** A `HarnessEnvironment` implementation for Angular's Testbed. */\nexport class TestbedHarnessEnvironment extends HarnessEnvironment<Element> {\n  /** Whether the environment has been destroyed. */\n  private _destroyed = false;\n\n  /** Observable that emits whenever the test task state changes. */\n  private _taskState: Observable<TaskState>;\n\n  /** The options for this environment. */\n  private _options: TestbedHarnessEnvironmentOptions;\n\n  protected constructor(rawRootElement: Element, private _fixture: ComponentFixture<unknown>,\n      options?: TestbedHarnessEnvironmentOptions) {\n    super(rawRootElement);\n    this._options = {...defaultEnvironmentOptions, ...options};\n    this._taskState = TaskStateZoneInterceptor.setup();\n    installAutoChangeDetectionStatusHandler(_fixture);\n    _fixture.componentRef.onDestroy(() => {\n      uninstallAutoChangeDetectionStatusHandler(_fixture);\n      this._destroyed = true;\n    });\n  }\n\n  /** Creates a `HarnessLoader` rooted at the given fixture's root element. */\n  static loader(fixture: ComponentFixture<unknown>, options?: TestbedHarnessEnvironmentOptions):\n      HarnessLoader {\n    return new TestbedHarnessEnvironment(fixture.nativeElement, fixture, options);\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>,\n      options?: TestbedHarnessEnvironmentOptions): HarnessLoader {\n    return new TestbedHarnessEnvironment(document.body, fixture, options);\n  }\n\n  /** Gets the native DOM element corresponding to the given TestElement. */\n  static getNativeElement(el: TestElement): Element {\n    if (el instanceof UnitTestElement) {\n      return el.element;\n    }\n    throw Error('This TestElement was not created by the TestbedHarnessEnvironment');\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>,\n      options?: TestbedHarnessEnvironmentOptions): Promise<T> {\n    const environment = new TestbedHarnessEnvironment(fixture.nativeElement, fixture, options);\n    await environment.forceStabilize();\n    return environment.createComponentHarness(harnessType, fixture.nativeElement);\n  }\n\n  /**\n   * Flushes change detection and async tasks captured in the Angular zone.\n   * In most cases it should not be necessary to call this manually. However, there may be some edge\n   * cases where it is needed to fully flush animation events.\n   */\n  async forceStabilize(): Promise<void> {\n    if (!disableAutoChangeDetection) {\n      if (this._destroyed) {\n        throw Error('Harness is attempting to use a fixture that has already been destroyed.');\n      }\n\n      await detectChanges(this._fixture);\n    }\n  }\n\n  /**\n   * Waits for all scheduled or running async tasks to complete. This allows harness\n   * authors to wait for async tasks outside of the Angular zone.\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 (isInFakeAsyncZone()) {\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  /** Gets the root element for the document. */\n  protected getDocumentRoot(): Element {\n    return document.body;\n  }\n\n  /** Creates a `TestElement` from a raw element. */\n  protected createTestElement(element: Element): TestElement {\n    return new UnitTestElement(element, () => this.forceStabilize());\n  }\n\n  /** Creates a `HarnessLoader` rooted at the given raw element. */\n  protected createEnvironment(element: Element): HarnessEnvironment<Element> {\n    return new TestbedHarnessEnvironment(element, this._fixture, this._options);\n  }\n\n  /**\n   * Gets a list of all elements matching the given selector under this environment's root element.\n   */\n  protected async getAllRawElements(selector: string): Promise<Element[]> {\n    await this.forceStabilize();\n    return Array.from(this._options.queryFn(selector, this.rawRootElement));\n  }\n}\n"]} |
---|