UNPKG

@engie-group/fluid-design-system-angular

Version:

Fluid Design System Angular

149 lines 21.1 kB
import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, Input, Output, ViewEncapsulation } from '@angular/core'; import { Subject, takeUntil } from 'rxjs'; import { Utils } from '../../utils/utils.util'; import { TabComponent } from '../tab/tab.component'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; export class TabsComponent { constructor(cdr) { this.cdr = cdr; this.tabClass = 'nj-tab'; this.unsubscribe = new Subject(); /** * Tabs density */ this.density = 'normal'; /** * Output that emits selected tab index */ this.selectedTab = new EventEmitter(); } ngAfterViewInit() { // A render is being done after view init so setTimeout allows us to delay any attribute modification // in the call stack so it can be taken account on next render only setTimeout(() => { this.tabs = this.tabsList?.toArray(); this.initializeClickListener(); this.initializeKeydownListener(); const activeIndex = this.tabs?.findIndex(tab => tab?.isActive); this.goToTab(activeIndex === -1 ? 0 : activeIndex, false); }); } ngOnDestroy() { this.unsubscribe.next(); this.unsubscribe.complete(); } /** * Check if tab is active * @param index index of tab to check */ isActiveTab(index) { return this.activeTabIndex === index; } /** * Allows you to navigate to tab * @param index index of tab to select * @param emit emits if set to true */ goToTab(index, emit = true) { const validIndex = (Utils.isUndefinedOrNull(index) || index < 0) ? 0 : index; this.activeTabIndex = validIndex; this.renderTemplate(validIndex); if (emit) { this.selectedTab.emit(validIndex); } this.cdr.markForCheck(); } /** * @ignore */ focusNextFocusableTab() { const focusableTabs = Array.from(this.tabs).filter(tab => !tab.isDisabled); const focusedTabIndex = focusableTabs.findIndex(tab => document.activeElement === tab.tab.nativeElement); const nextFocusableTab = focusableTabs[(focusedTabIndex + 1) % focusableTabs.length]; nextFocusableTab.tab.nativeElement.focus(); } /** * @ignore */ focusPreviousFocusableTab() { const focusableTabs = Array.from(this.tabs).filter(tab => !tab.isDisabled); const focusedTabIndex = focusableTabs.findIndex(tab => document.activeElement === tab.tab.nativeElement); const previousFocusableTabIndex = focusedTabIndex === 0 ? focusableTabs.length - 1 : focusedTabIndex - 1; const previousFocusableTab = focusableTabs[previousFocusableTabIndex]; previousFocusableTab.tab.nativeElement.focus(); } /** * @ignore */ handleTabKeydown(event) { switch (event.key) { case 'ArrowRight': event.preventDefault(); this.focusNextFocusableTab(); break; case 'ArrowLeft': event.preventDefault(); this.focusPreviousFocusableTab(); break; default: } } getDensityClass() { if (!this.density || this.density === 'normal') { return; } return `${this.tabClass}--${this.density}`; } initializeClickListener() { if (Utils.isUndefinedOrNull(this.tabs)) { return; } this.tabs.forEach((tab, index) => { tab.tabSelect .pipe(takeUntil(this.unsubscribe)) .subscribe(_ => { this.goToTab(index); }); }); } initializeKeydownListener() { if (Utils.isUndefinedOrNull(this.tabs)) { return; } this.tabs.forEach(tab => { tab.tabMove.pipe(takeUntil(this.unsubscribe)) .subscribe(event => { this.handleTabKeydown(event); }); }); } renderTemplate(index) { for (const tab of this.tabs) { if (tab?.isActive) { tab.setIsActive(false); } } this.tabToRender = this.tabs?.[index]; this.tabToRender.setIsActive(true); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TabsComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: TabsComponent, isStandalone: true, selector: "nj-tabs", inputs: { label: "label", density: "density" }, outputs: { selectedTab: "selectedTab" }, queries: [{ propertyName: "tabsList", predicate: TabComponent }], ngImport: i0, template: "<div class=\"nj-tab\" [ngClass]=\"getDensityClass()\">\n <div class=\"nj-tab__items\" role=\"tablist\" [attr.aria-label]=\"label\">\n <ng-content></ng-content>\n </div>\n <div class=\"nj-tab__content-wrapper\">\n <div class=\"nj-tab__content nj-tab__content--active\" *ngIf=\"tabToRender\"\n [attr.id]=\"tabToRender.tabContentId\"\n [attr.aria-labelledby]=\"tabToRender.tabContentAriaLabelledBy\"\n role=\"tabpanel\"\n tabIndex=\"0\">\n <ng-container *ngTemplateOutlet=\"tabToRender.contentTemplateRef\"></ng-container>\n </div>\n </div>\n</div>\n", styles: [".nj-tab__content-wrapper{margin:var(--nj-semantic-size-spacing-24) 0}.nj-tab--stretched nj-tab{display:inline-flex;flex-grow:1}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: TabsComponent, decorators: [{ type: Component, args: [{ selector: 'nj-tabs', changeDetection: ChangeDetectionStrategy.OnPush, encapsulation: ViewEncapsulation.None, standalone: true, imports: [TabComponent, CommonModule], template: "<div class=\"nj-tab\" [ngClass]=\"getDensityClass()\">\n <div class=\"nj-tab__items\" role=\"tablist\" [attr.aria-label]=\"label\">\n <ng-content></ng-content>\n </div>\n <div class=\"nj-tab__content-wrapper\">\n <div class=\"nj-tab__content nj-tab__content--active\" *ngIf=\"tabToRender\"\n [attr.id]=\"tabToRender.tabContentId\"\n [attr.aria-labelledby]=\"tabToRender.tabContentAriaLabelledBy\"\n role=\"tabpanel\"\n tabIndex=\"0\">\n <ng-container *ngTemplateOutlet=\"tabToRender.contentTemplateRef\"></ng-container>\n </div>\n </div>\n</div>\n", styles: [".nj-tab__content-wrapper{margin:var(--nj-semantic-size-spacing-24) 0}.nj-tab--stretched nj-tab{display:inline-flex;flex-grow:1}\n"] }] }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { label: [{ type: Input }], density: [{ type: Input }], tabsList: [{ type: ContentChildren, args: [TabComponent] }], selectedTab: [{ type: Output }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs.component.js","sourceRoot":"","sources":["../../../../src/components/tabs/tabs.component.ts","../../../../src/components/tabs/tabs.component.html"],"names":[],"mappings":"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","sourcesContent":["import {CommonModule} from '@angular/common';\nimport {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ContentChildren,\n  EventEmitter,\n  Input,\n  OnDestroy,\n  Output,\n  QueryList,\n  ViewEncapsulation\n} from '@angular/core';\nimport {Subject, takeUntil} from 'rxjs';\nimport {Utils} from '../../utils/utils.util';\nimport {TabComponent} from '../tab/tab.component';\nimport {TabsDensity} from './tabs.model';\n\n@Component({\n  selector: 'nj-tabs',\n  templateUrl: './tabs.component.html',\n  styleUrls: ['./tabs.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  encapsulation: ViewEncapsulation.None,\n  standalone: true,\n  imports: [TabComponent, CommonModule]\n})\nexport class TabsComponent implements AfterViewInit, OnDestroy {\n\n  private activeTabIndex;\n\n  private readonly tabClass = 'nj-tab';\n\n  private unsubscribe: Subject<void> = new Subject<void>();\n\n  /**\n   * @ignore\n   */\n  public tabToRender: TabComponent;\n\n  /**\n   * @ignore\n   */\n  public tabs: TabComponent[];\n\n  /**\n   * Label for the tablist element\n   */\n  @Input() label: string;\n\n  /**\n   * Tabs density\n   */\n  @Input() density: TabsDensity = 'normal';\n\n  /**\n   * @ignore\n   */\n  @ContentChildren(TabComponent) tabsList: QueryList<TabComponent>;\n\n  /**\n   * Output that emits selected tab index\n   */\n  @Output() selectedTab: EventEmitter<number> = new EventEmitter<number>();\n\n  constructor(private cdr: ChangeDetectorRef) {\n  }\n\n  ngAfterViewInit() {\n    // A render is being done after view init so setTimeout allows us to delay any attribute modification\n    // in the call stack so it can be taken account on next render only\n    setTimeout(() => {\n      this.tabs = this.tabsList?.toArray();\n\n      this.initializeClickListener();\n      this.initializeKeydownListener();\n\n      const activeIndex = this.tabs?.findIndex(tab => tab?.isActive);\n      this.goToTab(activeIndex === -1 ? 0 : activeIndex, false);\n    });\n  }\n\n  ngOnDestroy() {\n    this.unsubscribe.next();\n    this.unsubscribe.complete();\n  }\n\n  /**\n   * Check if tab is active\n   * @param index index of tab to check\n   */\n  isActiveTab(index: number): boolean {\n    return this.activeTabIndex === index;\n  }\n\n  /**\n   * Allows you to navigate to tab\n   * @param index index of tab to select\n   * @param emit emits if set to true\n   */\n  goToTab(index: number, emit = true) {\n    const validIndex = (Utils.isUndefinedOrNull(index) || index < 0) ? 0 : index;\n\n    this.activeTabIndex = validIndex;\n    this.renderTemplate(validIndex);\n    if (emit) {\n      this.selectedTab.emit(validIndex);\n    }\n    this.cdr.markForCheck();\n  }\n\n  /**\n   * @ignore\n   */\n  focusNextFocusableTab() {\n    const focusableTabs = Array.from(this.tabs).filter(tab => !tab.isDisabled);\n    const focusedTabIndex = focusableTabs.findIndex(tab => document.activeElement === tab.tab.nativeElement);\n    const nextFocusableTab = focusableTabs[(focusedTabIndex + 1) % focusableTabs.length];\n\n    nextFocusableTab.tab.nativeElement.focus();\n  }\n\n  /**\n   * @ignore\n   */\n  focusPreviousFocusableTab() {\n    const focusableTabs = Array.from(this.tabs).filter(tab => !tab.isDisabled);\n    const focusedTabIndex = focusableTabs.findIndex(tab => document.activeElement === tab.tab.nativeElement);\n\n    const previousFocusableTabIndex = focusedTabIndex === 0\n      ? focusableTabs.length - 1\n      : focusedTabIndex - 1;\n\n    const previousFocusableTab = focusableTabs[previousFocusableTabIndex];\n\n    previousFocusableTab.tab.nativeElement.focus();\n  }\n\n  /**\n   * @ignore\n   */\n  handleTabKeydown(event: KeyboardEvent) {\n    switch (event.key) {\n      case 'ArrowRight':\n        event.preventDefault();\n        this.focusNextFocusableTab();\n        break;\n      case 'ArrowLeft':\n        event.preventDefault();\n        this.focusPreviousFocusableTab();\n        break;\n      default:\n    }\n  }\n\n  getDensityClass() {\n    if(!this.density || this.density === 'normal') {\n      return;\n    }\n    return `${this.tabClass}--${this.density}`;\n  }\n\n  private initializeClickListener() {\n    if (Utils.isUndefinedOrNull(this.tabs)) {\n      return;\n    }\n    this.tabs.forEach((tab, index) => {\n      tab.tabSelect\n        .pipe(takeUntil(this.unsubscribe))\n        .subscribe(_ => {\n          this.goToTab(index);\n        });\n    });\n  }\n\n  private initializeKeydownListener() {\n    if (Utils.isUndefinedOrNull(this.tabs)) {\n      return;\n    }\n    this.tabs.forEach(tab => {\n      tab.tabMove.pipe(takeUntil(this.unsubscribe))\n      .subscribe(event => {\n        this.handleTabKeydown(event);\n      });\n    });\n  }\n\n  private renderTemplate(index: number) {\n    for (const tab of this.tabs) {\n      if (tab?.isActive) {\n        tab.setIsActive(false);\n      }\n    }\n    this.tabToRender = this.tabs?.[index];\n    this.tabToRender.setIsActive(true);\n  }\n}\n","<div class=\"nj-tab\" [ngClass]=\"getDensityClass()\">\n  <div class=\"nj-tab__items\" role=\"tablist\" [attr.aria-label]=\"label\">\n    <ng-content></ng-content>\n  </div>\n  <div class=\"nj-tab__content-wrapper\">\n    <div class=\"nj-tab__content nj-tab__content--active\" *ngIf=\"tabToRender\"\n         [attr.id]=\"tabToRender.tabContentId\"\n         [attr.aria-labelledby]=\"tabToRender.tabContentAriaLabelledBy\"\n         role=\"tabpanel\"\n         tabIndex=\"0\">\n      <ng-container *ngTemplateOutlet=\"tabToRender.contentTemplateRef\"></ng-container>\n    </div>\n  </div>\n</div>\n"]}