UNPKG

@coreui/angular

Version:

CoreUI Components Library for Angular

93 lines 14.3 kB
import { FocusKeyManager } from '@angular/cdk/a11y'; import { Component, computed, contentChildren, DestroyRef, effect, HostListener, inject, input, untracked } from '@angular/core'; import { takeUntilDestroyed } from '@angular/core/rxjs-interop'; import { tap } from 'rxjs/operators'; import { TabDirective } from '../tab/tab.directive'; import { TabsService } from '../tabs.service'; import * as i0 from "@angular/core"; export class TabsListComponent { constructor() { this.#destroyRef = inject(DestroyRef); this.tabsService = inject(TabsService); /** * Specify a layout type for component. * @type 'fill' | 'justified' | undefined * @default undefined */ this.layout = input(); /** * Set the variant to tabs, pills or underline. * @type 'pills' | 'tabs' | 'underline' | 'underline-border' | undefined * @default undefined */ this.variant = input(); /** * Set the role to tab list. * @default 'tablist' */ this.role = input('tablist'); this.hostClasses = computed(() => ({ nav: true, [`nav-${this.layout()}`]: this.layout(), [`nav-${this.variant()}`]: this.variant() })); this.tabs = contentChildren(TabDirective); this.tabsEffect = effect(() => { if (this.tabs().length === 0) { return; } this.#focusKeyManager = new FocusKeyManager(this.tabs()) .skipPredicate((tab) => tab.disabled === true) .withHorizontalOrientation('ltr') .withHomeAndEnd() .withWrap(); this.#focusKeyManager.change .pipe(tap((value) => { this.tabsService.activeItemKey.set(this.#focusKeyManager.activeItem?.itemKey()); this.tabsService.activeItem.set(this.#focusKeyManager.activeItem); }), takeUntilDestroyed(this.#destroyRef)) .subscribe(); const activeItem = this.tabs().find((tab) => untracked(tab.isActive)) ?? this.tabs().find((tab) => !tab.disabled); const activeItemIndex = this.tabs().findIndex((tab) => tab === activeItem); this.#focusKeyManager?.updateActiveItem(activeItemIndex < 0 ? 0 : activeItemIndex); this.tabsService.activeItemKey.set(this.#focusKeyManager.activeItem?.itemKey()); this.tabsService.activeItem.set(this.#focusKeyManager.activeItem); }, { allowSignalWrites: true }); this.tabsServiceEffect = effect(() => { const activeItemIndex = this.tabs().findIndex((tab) => untracked(tab.isActive) && untracked(tab.itemKey) === this.tabsService.activeItemKey()); this.#focusKeyManager?.updateActiveItem(activeItemIndex < 0 ? 0 : activeItemIndex); }, { allowSignalWrites: true }); } #destroyRef; #focusKeyManager; onKeydown($event) { if (['ArrowLeft', 'ArrowRight'].includes($event.key)) { this.#focusKeyManager.onKeydown($event); return; } if (['Tab'].includes($event.key)) { this.#focusKeyManager?.tabOut.next(); } return; } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: TabsListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.2.0", version: "18.2.12", type: TabsListComponent, isStandalone: true, selector: "c-tabs-list", inputs: { layout: { classPropertyName: "layout", publicName: "layout", isSignal: true, isRequired: false, transformFunction: null }, variant: { classPropertyName: "variant", publicName: "variant", isSignal: true, isRequired: false, transformFunction: null }, role: { classPropertyName: "role", publicName: "role", isSignal: true, isRequired: false, transformFunction: null } }, host: { listeners: { "keydown": "onKeydown($event)" }, properties: { "attr.role": "role()", "class": "hostClasses()" } }, queries: [{ propertyName: "tabs", predicate: TabDirective, isSignal: true }], exportAs: ["cTabsList"], ngImport: i0, template: '<ng-content />', isInline: true }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: TabsListComponent, decorators: [{ type: Component, args: [{ exportAs: 'cTabsList', selector: 'c-tabs-list', standalone: true, imports: [], template: '<ng-content />', host: { '[attr.role]': 'role()', '[class]': 'hostClasses()' } }] }], propDecorators: { onKeydown: [{ type: HostListener, args: ['keydown', ['$event']] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs-list.component.js","sourceRoot":"","sources":["../../../../../../projects/coreui-angular/src/lib/tabs-2/tabs-list/tabs-list.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { FocusKeyManager } from '@angular/cdk/a11y';\nimport {\n  Component,\n  computed,\n  contentChildren,\n  DestroyRef,\n  effect,\n  HostListener,\n  inject,\n  input,\n  InputSignal,\n  untracked\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { tap } from 'rxjs/operators';\nimport { TabDirective } from '../tab/tab.directive';\nimport { TabsService } from '../tabs.service';\n\n@Component({\n  exportAs: 'cTabsList',\n  selector: 'c-tabs-list',\n  standalone: true,\n  imports: [],\n  template: '<ng-content />',\n  host: {\n    '[attr.role]': 'role()',\n    '[class]': 'hostClasses()'\n  }\n})\nexport class TabsListComponent {\n  readonly #destroyRef = inject(DestroyRef);\n  readonly tabsService = inject(TabsService);\n\n  /**\n   * Specify a layout type for component.\n   * @type 'fill' | 'justified' | undefined\n   * @default undefined\n   */\n  readonly layout: InputSignal<'fill' | 'justified' | undefined> = input();\n\n  /**\n   * Set the variant to tabs, pills or underline.\n   * @type 'pills' | 'tabs' | 'underline' | 'underline-border' | undefined\n   * @default undefined\n   */\n  readonly variant: InputSignal<'pills' | 'tabs' | 'underline' | 'underline-border' | undefined> = input();\n\n  /**\n   * Set the role to tab list.\n   * @default 'tablist'\n   */\n  readonly role = input('tablist');\n\n  readonly hostClasses = computed(() => ({\n    nav: true,\n    [`nav-${this.layout()}`]: this.layout(),\n    [`nav-${this.variant()}`]: this.variant()\n  }));\n\n  readonly tabs = contentChildren(TabDirective);\n  #focusKeyManager!: FocusKeyManager<TabDirective>;\n\n  readonly tabsEffect = effect(\n    () => {\n      if (this.tabs().length === 0) {\n        return;\n      }\n      this.#focusKeyManager = new FocusKeyManager(this.tabs())\n        .skipPredicate((tab) => tab.disabled === true)\n        .withHorizontalOrientation('ltr')\n        .withHomeAndEnd()\n        .withWrap();\n\n      this.#focusKeyManager.change\n        .pipe(\n          tap((value) => {\n            this.tabsService.activeItemKey.set(this.#focusKeyManager.activeItem?.itemKey());\n            this.tabsService.activeItem.set(this.#focusKeyManager.activeItem);\n          }),\n          takeUntilDestroyed(this.#destroyRef)\n        )\n        .subscribe();\n\n      const activeItem = this.tabs().find((tab) => untracked(tab.isActive)) ?? this.tabs().find((tab) => !tab.disabled);\n      const activeItemIndex = this.tabs().findIndex((tab) => tab === activeItem);\n      this.#focusKeyManager?.updateActiveItem(activeItemIndex < 0 ? 0 : activeItemIndex);\n      this.tabsService.activeItemKey.set(this.#focusKeyManager.activeItem?.itemKey());\n      this.tabsService.activeItem.set(this.#focusKeyManager.activeItem);\n    },\n    { allowSignalWrites: true }\n  );\n\n  tabsServiceEffect = effect(\n    () => {\n      const activeItemIndex = this.tabs().findIndex(\n        (tab) => untracked(tab.isActive) && untracked(tab.itemKey) === this.tabsService.activeItemKey()\n      );\n      this.#focusKeyManager?.updateActiveItem(activeItemIndex < 0 ? 0 : activeItemIndex);\n    },\n    { allowSignalWrites: true }\n  );\n\n  @HostListener('keydown', ['$event'])\n  onKeydown($event: any) {\n    if (['ArrowLeft', 'ArrowRight'].includes($event.key)) {\n      this.#focusKeyManager.onKeydown($event);\n      return;\n    }\n    if (['Tab'].includes($event.key)) {\n      this.#focusKeyManager?.tabOut.next();\n    }\n    return;\n  }\n}\n"]}