@clr/angular
Version:
Angular components for Clarity
204 lines (202 loc) • 22.1 kB
JavaScript
/*
* Copyright (c) 2016-2025 Broadcom. All Rights Reserved.
* The term "Broadcom" refers to Broadcom Inc. and/or its subsidiaries.
* This software is released under MIT license.
* The full license information can be found in LICENSE in the root directory of this project.
*/
import { Component, Input } from '@angular/core';
import { startWith, tap } from 'rxjs';
import * as i0 from "@angular/core";
import * as i1 from "./providers/wizard-navigation.service";
import * as i2 from "./providers/page-collection.service";
import * as i3 from "../utils";
import * as i4 from "@angular/common";
import * as i5 from "../icon/icon";
export class ClrWizardStepnavItem {
constructor(navService, pageCollection, commonStrings, elementRef) {
this.navService = navService;
this.pageCollection = pageCollection;
this.commonStrings = commonStrings;
this.elementRef = elementRef;
/**
* This is used to prevent the steps from scrolling as the user clicks on the steps.
*/
this.skipNextScroll = false;
}
get id() {
this.pageGuard();
return this.pageCollection.getStepItemIdForPage(this.page);
}
get stepAriaCurrent() {
return this.isCurrent && 'step';
}
get isDisabled() {
this.pageGuard();
return this.page.disabled || this.navService.wizardStopNavigation || this.navService.wizardDisableStepnav;
}
get isCurrent() {
this.pageGuard();
return this.page.current;
}
get isComplete() {
this.pageGuard();
return this.page.completed;
}
get hasError() {
this.pageGuard();
return this.page.hasError && this.isComplete;
}
get canNavigate() {
this.pageGuard();
return this.pageCollection.previousPageIsCompleted(this.page);
}
get stepIconId() {
return `${this.id}-step-icon`;
}
get stepTextId() {
return `${this.id}-step-text`;
}
get stepNumberId() {
return `${this.id}-step-number`;
}
get stepTitleId() {
return `${this.id}-step-title`;
}
get labelledby() {
const textIds = [this.stepTextId, this.stepNumberId, this.stepTitleId];
const allIds = this.isComplete ? [this.stepIconId, ...textIds] : textIds;
return allIds.join(' ');
}
get icon() {
if (this.isCurrent) {
return {
shape: 'dot-circle',
label: this.commonStrings.keys.wizardStepCurrent || this.commonStrings.keys.timelineStepCurrent,
};
}
else if (this.hasError) {
return {
shape: 'error-standard',
label: this.commonStrings.keys.wizardStepError || this.commonStrings.keys.timelineStepError,
};
}
else if (this.isComplete) {
return {
shape: 'success-standard',
label: this.commonStrings.keys.wizardStepSuccess || this.commonStrings.keys.timelineStepSuccess,
};
}
else {
return null;
}
}
ngOnInit() {
this.subscription = this.ensureCurrentStepIsScrolledIntoView().subscribe();
}
ngOnDestroy() {
this.subscription?.unsubscribe();
}
click() {
this.pageGuard();
// if we click on our own stepnav or a disabled stepnav, we don't want to do anything
if (this.isDisabled || this.isCurrent) {
return;
}
this.skipNextScroll = true;
this.navService.goTo(this.page);
}
pageGuard() {
if (!this.page) {
throw new Error('Wizard stepnav item is not associated with a wizard page.');
}
}
ensureCurrentStepIsScrolledIntoView() {
// Don't use "smooth" scrolling when the wizard is first opened to avoid a delay in scrolling the current step into view.
// The current step when the wizard is opened might not be the first step. For example, the wizard can be closed and re-opened.
let scrollBehavior = 'auto';
return this.navService.currentPageChanged.pipe(startWith(this.navService.currentPage), tap(currentPage => {
if (!this.skipNextScroll && currentPage === this.page) {
this.elementRef.nativeElement.scrollIntoView({ behavior: scrollBehavior, block: 'center', inline: 'center' });
}
scrollBehavior = 'smooth';
this.skipNextScroll = false;
}));
}
}
ClrWizardStepnavItem.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrWizardStepnavItem, deps: [{ token: i1.WizardNavigationService }, { token: i2.PageCollectionService }, { token: i3.ClrCommonStringsService }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
ClrWizardStepnavItem.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.2", type: ClrWizardStepnavItem, selector: "[clr-wizard-stepnav-item]", inputs: { page: "page" }, host: { properties: { "id": "id", "attr.aria-current": "stepAriaCurrent", "attr.aria-controls": "page.id", "class.clr-nav-link": "true", "class.nav-item": "true", "class.active": "isCurrent", "class.disabled": "isDisabled", "class.no-click": "!canNavigate", "class.complete": "isComplete", "class.error": "hasError" } }, ngImport: i0, template: `
<button
type="button"
class="btn btn-link clr-wizard-stepnav-link"
(click)="click()"
[attr.disabled]="isDisabled ? '' : null"
[attr.aria-labelledby]="labelledby"
>
<div class="clr-wizard-stepnav-link-icon">
<cds-icon
*ngIf="icon; let icon"
[id]="stepIconId"
role="img"
class="clr-wizard-stepnav-link-icon"
[attr.shape]="icon.shape"
[attr.aria-label]="icon.label"
></cds-icon>
</div>
<span [id]="stepTextId" class="clr-sr-only">{{ commonStrings.keys.wizardStep }}</span>
<div [id]="stepNumberId" class="clr-wizard-stepnav-link-page-number">
<ng-content></ng-content>
</div>
<span [id]="stepTitleId" class="clr-wizard-stepnav-link-title">
<ng-template [ngTemplateOutlet]="page.navTitle"></ng-template>
</span>
</button>
`, isInline: true, dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i5.CdsIconCustomTag, selector: "cds-icon" }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.2", ngImport: i0, type: ClrWizardStepnavItem, decorators: [{
type: Component,
args: [{
selector: '[clr-wizard-stepnav-item]',
template: `
<button
type="button"
class="btn btn-link clr-wizard-stepnav-link"
(click)="click()"
[attr.disabled]="isDisabled ? '' : null"
[attr.aria-labelledby]="labelledby"
>
<div class="clr-wizard-stepnav-link-icon">
<cds-icon
*ngIf="icon; let icon"
[id]="stepIconId"
role="img"
class="clr-wizard-stepnav-link-icon"
[attr.shape]="icon.shape"
[attr.aria-label]="icon.label"
></cds-icon>
</div>
<span [id]="stepTextId" class="clr-sr-only">{{ commonStrings.keys.wizardStep }}</span>
<div [id]="stepNumberId" class="clr-wizard-stepnav-link-page-number">
<ng-content></ng-content>
</div>
<span [id]="stepTitleId" class="clr-wizard-stepnav-link-title">
<ng-template [ngTemplateOutlet]="page.navTitle"></ng-template>
</span>
</button>
`,
host: {
'[id]': 'id',
'[attr.aria-current]': 'stepAriaCurrent',
'[attr.aria-controls]': 'page.id',
'[class.clr-nav-link]': 'true',
'[class.nav-item]': 'true',
'[class.active]': 'isCurrent',
'[class.disabled]': 'isDisabled',
'[class.no-click]': '!canNavigate',
'[class.complete]': 'isComplete',
'[class.error]': 'hasError',
},
}]
}], ctorParameters: function () { return [{ type: i1.WizardNavigationService }, { type: i2.PageCollectionService }, { type: i3.ClrCommonStringsService }, { type: i0.ElementRef }]; }, propDecorators: { page: [{
type: Input,
args: ['page']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"wizard-stepnav-item.js","sourceRoot":"","sources":["../../../../projects/angular/src/wizard/wizard-stepnav-item.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2016-2025 Broadcom. All Rights Reserved.\n * The term \"Broadcom\" refers to Broadcom Inc. and/or its subsidiaries.\n * This software is released under MIT license.\n * The full license information can be found in LICENSE in the root directory of this project.\n */\n\nimport { Component, ElementRef, Input, OnDestroy, OnInit } from '@angular/core';\nimport { startWith, Subscription, tap } from 'rxjs';\n\nimport { ClrCommonStringsService } from '../utils';\nimport { PageCollectionService } from './providers/page-collection.service';\nimport { WizardNavigationService } from './providers/wizard-navigation.service';\nimport { ClrWizardPage } from './wizard-page';\n\n@Component({\n  selector: '[clr-wizard-stepnav-item]',\n  template: `\n    <button\n      type=\"button\"\n      class=\"btn btn-link clr-wizard-stepnav-link\"\n      (click)=\"click()\"\n      [attr.disabled]=\"isDisabled ? '' : null\"\n      [attr.aria-labelledby]=\"labelledby\"\n    >\n      <div class=\"clr-wizard-stepnav-link-icon\">\n        <cds-icon\n          *ngIf=\"icon; let icon\"\n          [id]=\"stepIconId\"\n          role=\"img\"\n          class=\"clr-wizard-stepnav-link-icon\"\n          [attr.shape]=\"icon.shape\"\n          [attr.aria-label]=\"icon.label\"\n        ></cds-icon>\n      </div>\n\n      <span [id]=\"stepTextId\" class=\"clr-sr-only\">{{ commonStrings.keys.wizardStep }}</span>\n      <div [id]=\"stepNumberId\" class=\"clr-wizard-stepnav-link-page-number\">\n        <ng-content></ng-content>\n      </div>\n      <span [id]=\"stepTitleId\" class=\"clr-wizard-stepnav-link-title\">\n        <ng-template [ngTemplateOutlet]=\"page.navTitle\"></ng-template>\n      </span>\n    </button>\n  `,\n  host: {\n    '[id]': 'id',\n    '[attr.aria-current]': 'stepAriaCurrent',\n    '[attr.aria-controls]': 'page.id',\n    '[class.clr-nav-link]': 'true',\n    '[class.nav-item]': 'true',\n    '[class.active]': 'isCurrent',\n    '[class.disabled]': 'isDisabled',\n    '[class.no-click]': '!canNavigate',\n    '[class.complete]': 'isComplete',\n    '[class.error]': 'hasError',\n  },\n})\nexport class ClrWizardStepnavItem implements OnInit, OnDestroy {\n  @Input('page') page: ClrWizardPage;\n\n  private subscription: Subscription;\n\n  /**\n   * This is used to prevent the steps from scrolling as the user clicks on the steps.\n   */\n  private skipNextScroll = false;\n\n  constructor(\n    public navService: WizardNavigationService,\n    public pageCollection: PageCollectionService,\n    public commonStrings: ClrCommonStringsService,\n    private readonly elementRef: ElementRef<HTMLElement>\n  ) {}\n\n  get id(): string {\n    this.pageGuard();\n    return this.pageCollection.getStepItemIdForPage(this.page);\n  }\n\n  get stepAriaCurrent(): string {\n    return this.isCurrent && 'step';\n  }\n\n  get isDisabled(): boolean {\n    this.pageGuard();\n    return this.page.disabled || this.navService.wizardStopNavigation || this.navService.wizardDisableStepnav;\n  }\n\n  get isCurrent(): boolean {\n    this.pageGuard();\n    return this.page.current;\n  }\n\n  get isComplete(): boolean {\n    this.pageGuard();\n    return this.page.completed;\n  }\n\n  get hasError(): boolean {\n    this.pageGuard();\n    return this.page.hasError && this.isComplete;\n  }\n\n  get canNavigate(): boolean {\n    this.pageGuard();\n    return this.pageCollection.previousPageIsCompleted(this.page);\n  }\n\n  protected get stepIconId() {\n    return `${this.id}-step-icon`;\n  }\n\n  protected get stepTextId() {\n    return `${this.id}-step-text`;\n  }\n\n  protected get stepNumberId() {\n    return `${this.id}-step-number`;\n  }\n\n  protected get stepTitleId() {\n    return `${this.id}-step-title`;\n  }\n\n  protected get labelledby() {\n    const textIds = [this.stepTextId, this.stepNumberId, this.stepTitleId];\n    const allIds = this.isComplete ? [this.stepIconId, ...textIds] : textIds;\n\n    return allIds.join(' ');\n  }\n\n  protected get icon(): { shape: string; label: string } | null {\n    if (this.isCurrent) {\n      return {\n        shape: 'dot-circle',\n        label: this.commonStrings.keys.wizardStepCurrent || this.commonStrings.keys.timelineStepCurrent,\n      };\n    } else if (this.hasError) {\n      return {\n        shape: 'error-standard',\n        label: this.commonStrings.keys.wizardStepError || this.commonStrings.keys.timelineStepError,\n      };\n    } else if (this.isComplete) {\n      return {\n        shape: 'success-standard',\n        label: this.commonStrings.keys.wizardStepSuccess || this.commonStrings.keys.timelineStepSuccess,\n      };\n    } else {\n      return null;\n    }\n  }\n\n  ngOnInit() {\n    this.subscription = this.ensureCurrentStepIsScrolledIntoView().subscribe();\n  }\n\n  ngOnDestroy() {\n    this.subscription?.unsubscribe();\n  }\n\n  click(): void {\n    this.pageGuard();\n\n    // if we click on our own stepnav or a disabled stepnav, we don't want to do anything\n    if (this.isDisabled || this.isCurrent) {\n      return;\n    }\n\n    this.skipNextScroll = true;\n    this.navService.goTo(this.page);\n  }\n\n  private pageGuard(): void {\n    if (!this.page) {\n      throw new Error('Wizard stepnav item is not associated with a wizard page.');\n    }\n  }\n\n  private ensureCurrentStepIsScrolledIntoView() {\n    // Don't use \"smooth\" scrolling when the wizard is first opened to avoid a delay in scrolling the current step into view.\n    // The current step when the wizard is opened might not be the first step. For example, the wizard can be closed and re-opened.\n    let scrollBehavior: ScrollBehavior = 'auto';\n\n    return this.navService.currentPageChanged.pipe(\n      startWith(this.navService.currentPage),\n      tap(currentPage => {\n        if (!this.skipNextScroll && currentPage === this.page) {\n          this.elementRef.nativeElement.scrollIntoView({ behavior: scrollBehavior, block: 'center', inline: 'center' });\n        }\n\n        scrollBehavior = 'smooth';\n        this.skipNextScroll = false;\n      })\n    );\n  }\n}\n"]}