UNPKG

@coreui/angular

Version:

CoreUI Components Library for Angular

96 lines 13.2 kB
import { booleanAttribute, computed, DestroyRef, Directive, effect, ElementRef, inject, Input, input, signal, untracked } from '@angular/core'; import { takeUntilDestroyed } from '@angular/core/rxjs-interop'; import { fromEvent, merge, takeWhile } from 'rxjs'; import { filter, tap } from 'rxjs/operators'; import { TabsService } from '../tabs.service'; import * as i0 from "@angular/core"; export class TabDirective { constructor() { this.#destroyRef = inject(DestroyRef); this.#elementRef = inject(ElementRef); this.#tabsService = inject(TabsService); this.#disabled = signal(false); this.attrDisabled = computed(() => this.#disabled() || null); /** * Item key. * @type string | number * @required */ this.itemKey = input.required(); /** * Element id attribute * @type string * @default undefined */ this.id = input(); /** * aria-controls attribute * @type string * @default undefined */ this.ariaControls = input(undefined, { alias: 'aria-controls' }); this.isActive = computed(() => !this.#disabled() && this.#tabsService.activeItemKey() === this.itemKey()); this.hostClasses = computed(() => ({ 'nav-link': true, active: this.isActive(), disabled: this.#disabled() })); this.propId = computed(() => this.id() ?? `${this.#tabsService.id()}-tab-${this.itemKey()}`); this.attrAriaControls = computed(() => this.ariaControls() ?? `${this.#tabsService.id()}-panel-${this.itemKey()}`); this.disabledEffect = effect(() => { if (!this.#disabled()) { const click$ = fromEvent(this.#elementRef.nativeElement, 'click'); const focusIn$ = fromEvent(this.#elementRef.nativeElement, 'focusin'); merge(focusIn$, click$) .pipe(filter(($event) => !this.#disabled()), tap(($event) => { this.#tabsService.activeItemKey.set(untracked(this.itemKey)); }), takeWhile(() => !this.#disabled()), takeUntilDestroyed(this.#destroyRef)) .subscribe(); } }, { allowSignalWrites: true }); } #destroyRef; #elementRef; #tabsService; /** * Disabled attribute * @type boolean * @default false */ set disabled(value) { this.#disabled.set(value); } get disabled() { return this.#disabled(); } #disabled; focus(origin) { this.#elementRef.nativeElement.focus(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: TabDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive }); } static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.12", type: TabDirective, isStandalone: true, selector: "button[cTab]", inputs: { disabled: { classPropertyName: "disabled", publicName: "disabled", isSignal: false, isRequired: false, transformFunction: booleanAttribute }, itemKey: { classPropertyName: "itemKey", publicName: "itemKey", isSignal: true, isRequired: true, transformFunction: null }, id: { classPropertyName: "id", publicName: "id", isSignal: true, isRequired: false, transformFunction: null }, ariaControls: { classPropertyName: "ariaControls", publicName: "aria-controls", isSignal: true, isRequired: false, transformFunction: null } }, host: { attributes: { "type": "button", "role": "tab" }, properties: { "class": "hostClasses()", "attr.aria-selected": "isActive()", "attr.aria-controls": "attrAriaControls()", "attr.disabled": "attrDisabled() || null", "id": "propId()", "tabindex": "isActive() ? 0 : -1" } }, exportAs: ["cTab"], ngImport: i0 }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: TabDirective, decorators: [{ type: Directive, args: [{ exportAs: 'cTab', selector: 'button[cTab]', standalone: true, host: { '[class]': 'hostClasses()', type: 'button', role: 'tab', '[attr.aria-selected]': 'isActive()', '[attr.aria-controls]': 'attrAriaControls()', '[attr.disabled]': 'attrDisabled() || null', '[id]': 'propId()', '[tabindex]': 'isActive() ? 0 : -1' } }] }], propDecorators: { disabled: [{ type: Input, args: [{ transform: booleanAttribute }] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab.directive.js","sourceRoot":"","sources":["../../../../../../projects/coreui-angular/src/lib/tabs-2/tab/tab.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { FocusableOption, FocusOrigin } from '@angular/cdk/a11y';\nimport {\n  booleanAttribute,\n  computed,\n  DestroyRef,\n  Directive,\n  effect,\n  ElementRef,\n  inject,\n  Input,\n  input,\n  InputSignal,\n  signal,\n  untracked\n} from '@angular/core';\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\nimport { fromEvent, merge, takeWhile } from 'rxjs';\nimport { filter, tap } from 'rxjs/operators';\nimport { TabsService } from '../tabs.service';\n\n@Directive({\n  exportAs: 'cTab',\n  selector: 'button[cTab]',\n  standalone: true,\n  host: {\n    '[class]': 'hostClasses()',\n    type: 'button',\n    role: 'tab',\n    '[attr.aria-selected]': 'isActive()',\n    '[attr.aria-controls]': 'attrAriaControls()',\n    '[attr.disabled]': 'attrDisabled() || null',\n    '[id]': 'propId()',\n    '[tabindex]': 'isActive() ? 0 : -1'\n  }\n})\nexport class TabDirective implements FocusableOption {\n  readonly #destroyRef = inject(DestroyRef);\n  readonly #elementRef = inject(ElementRef);\n  readonly #tabsService = inject(TabsService);\n\n  /**\n   * Disabled attribute\n   * @type boolean\n   * @default false\n   */\n  @Input({ transform: booleanAttribute })\n  set disabled(value: boolean) {\n    this.#disabled.set(value);\n  }\n\n  get disabled() {\n    return this.#disabled();\n  }\n\n  readonly #disabled = signal(false);\n  readonly attrDisabled = computed(() => this.#disabled() || null);\n\n  /**\n   * Item key.\n   * @type string | number\n   * @required\n   */\n  readonly itemKey: InputSignal<string | number> = input.required<string | number>();\n\n  /**\n   * Element id attribute\n   * @type string\n   * @default undefined\n   */\n  readonly id: InputSignal<string | undefined> = input<string>();\n\n  /**\n   * aria-controls attribute\n   * @type string\n   * @default undefined\n   */\n  readonly ariaControls: InputSignal<string | undefined> = input<string | undefined>(undefined, {\n    alias: 'aria-controls'\n  });\n\n  readonly isActive = computed<boolean>(\n    () => !this.#disabled() && this.#tabsService.activeItemKey() === this.itemKey()\n  );\n\n  readonly hostClasses = computed(() => ({\n    'nav-link': true,\n    active: this.isActive(),\n    disabled: this.#disabled()\n  }));\n\n  readonly propId = computed(() => this.id() ?? `${this.#tabsService.id()}-tab-${this.itemKey()}`);\n\n  readonly attrAriaControls = computed(\n    () => this.ariaControls() ?? `${this.#tabsService.id()}-panel-${this.itemKey()}`\n  );\n\n  disabledEffect = effect(\n    () => {\n      if (!this.#disabled()) {\n        const click$ = fromEvent<MouseEvent>(this.#elementRef.nativeElement, 'click');\n        const focusIn$ = fromEvent<FocusEvent>(this.#elementRef.nativeElement, 'focusin');\n\n        merge(focusIn$, click$)\n          .pipe(\n            filter(($event) => !this.#disabled()),\n            tap(($event) => {\n              this.#tabsService.activeItemKey.set(untracked(this.itemKey));\n            }),\n            takeWhile(() => !this.#disabled()),\n            takeUntilDestroyed(this.#destroyRef)\n          )\n          .subscribe();\n      }\n    },\n    { allowSignalWrites: true }\n  );\n\n  focus(origin?: FocusOrigin): void {\n    this.#elementRef.nativeElement.focus();\n  }\n}\n"]}