UNPKG

@angular/material

Version:
91 lines 14.2 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 { ContentContainerComponentHarness, HarnessPredicate, } from '@angular/cdk/testing'; /** Harness for interacting with a standard Angular Material step in tests. */ class MatStepHarness extends ContentContainerComponentHarness { /** The selector for the host element of a `MatStep` instance. */ static { this.hostSelector = '.mat-step-header'; } /** * Gets a `HarnessPredicate` that can be used to search for a `MatStepHarness` that meets * certain criteria. * @param options Options for filtering which steps are considered a match. * @return a `HarnessPredicate` configured with the given options. */ static with(options = {}) { return new HarnessPredicate(MatStepHarness, options) .addOption('label', options.label, (harness, label) => HarnessPredicate.stringMatches(harness.getLabel(), label)) .addOption('selected', options.selected, async (harness, selected) => (await harness.isSelected()) === selected) .addOption('completed', options.completed, async (harness, completed) => (await harness.isCompleted()) === completed) .addOption('invalid', options.invalid, async (harness, invalid) => (await harness.hasErrors()) === invalid); } /** Gets the label of the step. */ async getLabel() { return (await this.locatorFor('.mat-step-text-label')()).text(); } /** Gets the `aria-label` of the step. */ async getAriaLabel() { return (await this.host()).getAttribute('aria-label'); } /** Gets the value of the `aria-labelledby` attribute. */ async getAriaLabelledby() { return (await this.host()).getAttribute('aria-labelledby'); } /** Whether the step is selected. */ async isSelected() { const host = await this.host(); return (await host.getAttribute('aria-selected')) === 'true'; } /** Whether the step has been filled out. */ async isCompleted() { const state = await this._getIconState(); return state === 'done' || (state === 'edit' && !(await this.isSelected())); } /** * Whether the step is currently showing its error state. Note that this doesn't mean that there * are or aren't any invalid form controls inside the step, but that the step is showing its * error-specific styling which depends on there being invalid controls, as well as the * `ErrorStateMatcher` determining that an error should be shown and that the `showErrors` * option was enabled through the `STEPPER_GLOBAL_OPTIONS` injection token. */ async hasErrors() { return (await this._getIconState()) === 'error'; } /** Whether the step is optional. */ async isOptional() { // If the node with the optional text is present, it means that the step is optional. const optionalNode = await this.locatorForOptional('.mat-step-optional')(); return !!optionalNode; } /** * Selects the given step by clicking on the label. The step may not be selected * if the stepper doesn't allow it (e.g. if there are validation errors). */ async select() { await (await this.host()).click(); } async getRootHarnessLoader() { const contentId = await (await this.host()).getAttribute('aria-controls'); return this.documentRootLocatorFactory().harnessLoaderFor(`#${contentId}`); } /** * Gets the state of the step. Note that we have a `StepState` which we could use to type the * return value, but it's basically the same as `string`, because the type has `| string`. */ async _getIconState() { // The state is exposed on the icon with a class that looks like `mat-step-icon-state-{{state}}` const icon = await this.locatorFor('.mat-step-icon')(); const classes = (await icon.getAttribute('class')); const match = classes.match(/mat-step-icon-state-([a-z]+)/); if (!match) { throw Error(`Could not determine step state from "${classes}".`); } return match[1]; } } export { MatStepHarness }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"step-harness.js","sourceRoot":"","sources":["../../../../../../../src/material/stepper/testing/step-harness.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH,OAAO,EACL,gCAAgC,EAChC,gBAAgB,GAEjB,MAAM,sBAAsB,CAAC;AAG9B,8EAA8E;AAC9E,MAAa,cAAe,SAAQ,gCAAwC;IAC1E,iEAAiE;aAC1D,iBAAY,GAAG,kBAAkB,CAAC;IAEzC;;;;;OAKG;IACH,MAAM,CAAC,IAAI,CAAC,UAA8B,EAAE;QAC1C,OAAO,IAAI,gBAAgB,CAAC,cAAc,EAAE,OAAO,CAAC;aACjD,SAAS,CAAC,OAAO,EAAE,OAAO,CAAC,KAAK,EAAE,CAAC,OAAO,EAAE,KAAK,EAAE,EAAE,CACpD,gBAAgB,CAAC,aAAa,CAAC,OAAO,CAAC,QAAQ,EAAE,EAAE,KAAK,CAAC,CAC1D;aACA,SAAS,CACR,UAAU,EACV,OAAO,CAAC,QAAQ,EAChB,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,EAAE,CAAC,CAAC,MAAM,OAAO,CAAC,UAAU,EAAE,CAAC,KAAK,QAAQ,CACvE;aACA,SAAS,CACR,WAAW,EACX,OAAO,CAAC,SAAS,EACjB,KAAK,EAAE,OAAO,EAAE,SAAS,EAAE,EAAE,CAAC,CAAC,MAAM,OAAO,CAAC,WAAW,EAAE,CAAC,KAAK,SAAS,CAC1E;aACA,SAAS,CACR,SAAS,EACT,OAAO,CAAC,OAAO,EACf,KAAK,EAAE,OAAO,EAAE,OAAO,EAAE,EAAE,CAAC,CAAC,MAAM,OAAO,CAAC,SAAS,EAAE,CAAC,KAAK,OAAO,CACpE,CAAC;IACN,CAAC;IAED,kCAAkC;IAClC,KAAK,CAAC,QAAQ;QACZ,OAAO,CAAC,MAAM,IAAI,CAAC,UAAU,CAAC,sBAAsB,CAAC,EAAE,CAAC,CAAC,IAAI,EAAE,CAAC;IAClE,CAAC;IAED,yCAAyC;IACzC,KAAK,CAAC,YAAY;QAChB,OAAO,CAAC,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,YAAY,CAAC,YAAY,CAAC,CAAC;IACxD,CAAC;IAED,yDAAyD;IACzD,KAAK,CAAC,iBAAiB;QACrB,OAAO,CAAC,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,YAAY,CAAC,iBAAiB,CAAC,CAAC;IAC7D,CAAC;IAED,oCAAoC;IACpC,KAAK,CAAC,UAAU;QACd,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;QAC/B,OAAO,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC,CAAC,KAAK,MAAM,CAAC;IAC/D,CAAC;IAED,4CAA4C;IAC5C,KAAK,CAAC,WAAW;QACf,MAAM,KAAK,GAAG,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC;QACzC,OAAO,KAAK,KAAK,MAAM,IAAI,CAAC,KAAK,KAAK,MAAM,IAAI,CAAC,CAAC,MAAM,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC;IAC9E,CAAC;IAED;;;;;;OAMG;IACH,KAAK,CAAC,SAAS;QACb,OAAO,CAAC,MAAM,IAAI,CAAC,aAAa,EAAE,CAAC,KAAK,OAAO,CAAC;IAClD,CAAC;IAED,oCAAoC;IACpC,KAAK,CAAC,UAAU;QACd,qFAAqF;QACrF,MAAM,YAAY,GAAG,MAAM,IAAI,CAAC,kBAAkB,CAAC,oBAAoB,CAAC,EAAE,CAAC;QAC3E,OAAO,CAAC,CAAC,YAAY,CAAC;IACxB,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,MAAM;QACV,MAAM,CAAC,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC;IACpC,CAAC;IAEkB,KAAK,CAAC,oBAAoB;QAC3C,MAAM,SAAS,GAAG,MAAM,CAAC,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,YAAY,CAAC,eAAe,CAAC,CAAC;QAC1E,OAAO,IAAI,CAAC,0BAA0B,EAAE,CAAC,gBAAgB,CAAC,IAAI,SAAS,EAAE,CAAC,CAAC;IAC7E,CAAC;IAED;;;OAGG;IACK,KAAK,CAAC,aAAa;QACzB,gGAAgG;QAChG,MAAM,IAAI,GAAG,MAAM,IAAI,CAAC,UAAU,CAAC,gBAAgB,CAAC,EAAE,CAAC;QACvD,MAAM,OAAO,GAAG,CAAC,MAAM,IAAI,CAAC,YAAY,CAAC,OAAO,CAAC,CAAE,CAAC;QACpD,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,8BAA8B,CAAC,CAAC;QAE5D,IAAI,CAAC,KAAK,EAAE;YACV,MAAM,KAAK,CAAC,wCAAwC,OAAO,IAAI,CAAC,CAAC;SAClE;QAED,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC;IAClB,CAAC;;SAzGU,cAAc","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  ContentContainerComponentHarness,\n  HarnessPredicate,\n  HarnessLoader,\n} from '@angular/cdk/testing';\nimport {StepHarnessFilters} from './step-harness-filters';\n\n/** Harness for interacting with a standard Angular Material step in tests. */\nexport class MatStepHarness extends ContentContainerComponentHarness<string> {\n  /** The selector for the host element of a `MatStep` instance. */\n  static hostSelector = '.mat-step-header';\n\n  /**\n   * Gets a `HarnessPredicate` that can be used to search for a `MatStepHarness` that meets\n   * certain criteria.\n   * @param options Options for filtering which steps are considered a match.\n   * @return a `HarnessPredicate` configured with the given options.\n   */\n  static with(options: StepHarnessFilters = {}): HarnessPredicate<MatStepHarness> {\n    return new HarnessPredicate(MatStepHarness, options)\n      .addOption('label', options.label, (harness, label) =>\n        HarnessPredicate.stringMatches(harness.getLabel(), label),\n      )\n      .addOption(\n        'selected',\n        options.selected,\n        async (harness, selected) => (await harness.isSelected()) === selected,\n      )\n      .addOption(\n        'completed',\n        options.completed,\n        async (harness, completed) => (await harness.isCompleted()) === completed,\n      )\n      .addOption(\n        'invalid',\n        options.invalid,\n        async (harness, invalid) => (await harness.hasErrors()) === invalid,\n      );\n  }\n\n  /** Gets the label of the step. */\n  async getLabel(): Promise<string> {\n    return (await this.locatorFor('.mat-step-text-label')()).text();\n  }\n\n  /** Gets the `aria-label` of the step. */\n  async getAriaLabel(): Promise<string | null> {\n    return (await this.host()).getAttribute('aria-label');\n  }\n\n  /** Gets the value of the `aria-labelledby` attribute. */\n  async getAriaLabelledby(): Promise<string | null> {\n    return (await this.host()).getAttribute('aria-labelledby');\n  }\n\n  /** Whether the step is selected. */\n  async isSelected(): Promise<boolean> {\n    const host = await this.host();\n    return (await host.getAttribute('aria-selected')) === 'true';\n  }\n\n  /** Whether the step has been filled out. */\n  async isCompleted(): Promise<boolean> {\n    const state = await this._getIconState();\n    return state === 'done' || (state === 'edit' && !(await this.isSelected()));\n  }\n\n  /**\n   * Whether the step is currently showing its error state. Note that this doesn't mean that there\n   * are or aren't any invalid form controls inside the step, but that the step is showing its\n   * error-specific styling which depends on there being invalid controls, as well as the\n   * `ErrorStateMatcher` determining that an error should be shown and that the `showErrors`\n   * option was enabled through the `STEPPER_GLOBAL_OPTIONS` injection token.\n   */\n  async hasErrors(): Promise<boolean> {\n    return (await this._getIconState()) === 'error';\n  }\n\n  /** Whether the step is optional. */\n  async isOptional(): Promise<boolean> {\n    // If the node with the optional text is present, it means that the step is optional.\n    const optionalNode = await this.locatorForOptional('.mat-step-optional')();\n    return !!optionalNode;\n  }\n\n  /**\n   * Selects the given step by clicking on the label. The step may not be selected\n   * if the stepper doesn't allow it (e.g. if there are validation errors).\n   */\n  async select(): Promise<void> {\n    await (await this.host()).click();\n  }\n\n  protected override async getRootHarnessLoader(): Promise<HarnessLoader> {\n    const contentId = await (await this.host()).getAttribute('aria-controls');\n    return this.documentRootLocatorFactory().harnessLoaderFor(`#${contentId}`);\n  }\n\n  /**\n   * Gets the state of the step. Note that we have a `StepState` which we could use to type the\n   * return value, but it's basically the same as `string`, because the type has `| string`.\n   */\n  private async _getIconState(): Promise<string> {\n    // The state is exposed on the icon with a class that looks like `mat-step-icon-state-{{state}}`\n    const icon = await this.locatorFor('.mat-step-icon')();\n    const classes = (await icon.getAttribute('class'))!;\n    const match = classes.match(/mat-step-icon-state-([a-z]+)/);\n\n    if (!match) {\n      throw Error(`Could not determine step state from \"${classes}\".`);\n    }\n\n    return match[1];\n  }\n}\n"]}