UNPKG

carbon-components-angular

Version:
328 lines 37.7 kB
import { Component, Input, HostListener, HostBinding, ViewChild, ContentChildren, ViewChildren } from "@angular/core"; import { BaseTabHeader } from "./base-tab-header.component"; import { Tab } from "./tab.component"; import * as i0 from "@angular/core"; import * as i1 from "carbon-components-angular/utils"; import * as i2 from "carbon-components-angular/i18n"; import * as i3 from "@angular/common"; const VERTICAL_TAB_HEIGHT = 64; /** * The `TabHeadersVertical` component renders tab headers in a vertical * orientation. It contains the `Tab` items and supports keyboard navigation * via ArrowUp/ArrowDown/Home/End. */ export class TabHeadersVertical extends BaseTabHeader { constructor(elementRef, changeDetectorRef, eventService, renderer, i18n) { super(elementRef, changeDetectorRef, eventService, renderer); this.elementRef = elementRef; this.changeDetectorRef = changeDetectorRef; this.eventService = eventService; this.renderer = renderer; this.i18n = i18n; /** * i18n strings for the tab list `aria-label` fallback. */ this.translations = this.i18n.get().TABS; this.verticalClass = true; /** * Focused tab index when `followFocus` is false (manual activation) */ this.activeIndex = null; /** * Whether the tab list is overflowing at the top (some tabs are clipped). */ this.isOverflowingTop = false; /** * Whether the tab list is overflowing at the bottom (some tabs are clipped). */ this.isOverflowingBottom = false; this.listScrollHandler = () => this.updateOverflowState(); this.type = "contained"; } keyboardInput(event) { if (!this.tabs) { return; } const tabsArray = this.tabs.toArray(); const enabledTabs = tabsArray.filter(tab => !tab.disabled); if (enabledTabs.length === 0) { return; } const referenceIndex = this.followFocus ? this.currentSelectedTab : (this.activeIndex !== null ? this.activeIndex : this.currentSelectedTab); const currentEnabledIndex = Math.max(0, enabledTabs.indexOf(tabsArray[referenceIndex])); let nextEnabledIndex = currentEnabledIndex; let handled = false; if (event.key === "ArrowDown") { nextEnabledIndex = (currentEnabledIndex + 1) % enabledTabs.length; handled = true; } else if (event.key === "ArrowUp") { nextEnabledIndex = (enabledTabs.length + currentEnabledIndex - 1) % enabledTabs.length; handled = true; } else if (event.key === "Home") { nextEnabledIndex = 0; handled = true; } else if (event.key === "End") { nextEnabledIndex = enabledTabs.length - 1; handled = true; } if (handled) { event.preventDefault(); const nextTab = enabledTabs[nextEnabledIndex]; const nextIndex = tabsArray.indexOf(nextTab); if (this.followFocus) { this.selectTab(nextTab, nextIndex); } else { this.activeIndex = nextIndex; } this.allTabHeaders.toArray()[nextIndex].nativeElement.focus(); return; } if ((event.key === " " || event.key === "Spacebar") && !this.followFocus) { const focusIndex = this.activeIndex !== null ? this.activeIndex : this.currentSelectedTab; this.selectTab(tabsArray[focusIndex], focusIndex); } } handleBlur(event) { const relatedTarget = event.relatedTarget; const container = this.headerContainer?.nativeElement; if (container && relatedTarget && container.contains(relatedTarget)) { return; } if (!this.followFocus) { this.activeIndex = this.currentSelectedTab; } } ngOnInit() { this.resizeObserver = new ResizeObserver(() => { this.updateOverflowState(); this.changeDetectorRef.detectChanges(); }); this.resizeObserver.observe(this.headerContainer.nativeElement); this.headerContainer.nativeElement.addEventListener("scroll", this.listScrollHandler); } ngOnDestroy() { this.resizeObserver?.unobserve(this.headerContainer.nativeElement); this.headerContainer.nativeElement.removeEventListener("scroll", this.listScrollHandler); } ngAfterContentInit() { if (!this.tabInput) { this.tabs = this.tabQuery; } else { this.tabs = this.tabInput; } this.tabs.forEach(tab => tab.cacheActive = this.cacheActive); this.tabs.changes.subscribe(() => { this.setFirstTab(); this.changeDetectorRef.markForCheck(); }); this.setFirstTab(); } ngOnChanges(changes) { if (this.tabs && changes.cacheActive) { this.tabs.forEach(tab => tab.cacheActive = this.cacheActive); } } onTabFocus(index) { if (this.followFocus) { this.currentSelectedTab = index; } else { this.activeIndex = index; } this.scrollSelectedTabIntoView(); } selectTab(tab, tabIndex) { if (tab.disabled) { return; } this.currentSelectedTab = tabIndex; this.activeIndex = tabIndex; this.tabs.forEach(_tab => _tab.active = false); tab.active = true; tab.doSelect(); this.scrollSelectedTabIntoView(); } getSelectedTab() { const selected = this.tabs.find(tab => tab.active); if (selected) { return selected; } return { headingIsTemplate: false, heading: "" }; } updateOverflowState() { const element = this.headerContainer?.nativeElement; if (!element) { return; } const halfTabHeight = VERTICAL_TAB_HEIGHT / 2; this.isOverflowingBottom = element.scrollTop + element.clientHeight + halfTabHeight <= element.scrollHeight; this.isOverflowingTop = element.scrollTop > halfTabHeight; this.changeDetectorRef.markForCheck(); } scrollSelectedTabIntoView() { const container = this.headerContainer?.nativeElement; if (!container) { return; } const selectedHeader = this.allTabHeaders?.toArray()[this.currentSelectedTab]?.nativeElement; if (!selectedHeader) { return; } const containerRect = container.getBoundingClientRect(); const selectedRect = selectedHeader.getBoundingClientRect(); const halfTabHeight = VERTICAL_TAB_HEIGHT / 2; if (selectedRect.top - halfTabHeight < containerRect.top || selectedRect.top - containerRect.top + VERTICAL_TAB_HEIGHT + halfTabHeight > containerRect.height) { container.scrollTo({ top: Math.max(0, (this.currentSelectedTab - 1) * VERTICAL_TAB_HEIGHT), behavior: "smooth" }); } } setFirstTab() { setTimeout(() => { let firstTab = this.tabs.find(tab => tab.active); if (!firstTab && this.tabs.first) { firstTab = this.tabs.first; firstTab.active = true; } if (firstTab) { this.currentSelectedTab = this.tabs.toArray().indexOf(firstTab); this.activeIndex = this.currentSelectedTab; firstTab.doSelect(); this.updateOverflowState(); } }); } } TabHeadersVertical.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: TabHeadersVertical, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: i1.EventService }, { token: i0.Renderer2 }, { token: i2.I18n }], target: i0.ɵɵFactoryTarget.Component }); TabHeadersVertical.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: TabHeadersVertical, selector: "cds-tab-headers-vertical, ibm-tab-headers-vertical", inputs: { tabInput: ["tabs", "tabInput"], translations: "translations" }, host: { listeners: { "keydown": "keyboardInput($event)", "blur": "handleBlur($event)" }, properties: { "class.cds--tabs--vertical": "this.verticalClass" } }, queries: [{ propertyName: "tabQuery", predicate: Tab }], viewQueries: [{ propertyName: "headerContainer", first: true, predicate: ["tabList"], descendants: true, static: true }, { propertyName: "allTabHeaders", predicate: ["tabItem"], descendants: true }], usesInheritance: true, usesOnChanges: true, ngImport: i0, template: ` <div *ngIf="isOverflowingTop" class="cds--tab--list-gradient_top"></div> <div #tabList class="cds--tab--list" role="tablist" [attr.aria-label]="ariaLabel || translations.HEADER_ARIA_LABEL" [attr.aria-labelledby]="ariaLabelledby || null"> <ng-container [ngTemplateOutlet]="contentBefore"></ng-container> <button *ngFor="let tab of tabs; let i = index;" #tabItem role="tab" [attr.aria-selected]="tab.active" [attr.tabindex]="(tab.active?0:-1)" [attr.aria-controls]="tab.id" [attr.aria-disabled]="tab.disabled" [disabled]="tab.disabled" [ngClass]="{ 'cds--tabs__nav-item--selected': tab.active, 'cds--tabs__nav-item--disabled': tab.disabled }" class="cds--tabs__nav-item cds--tabs__nav-link" type="button" draggable="false" id="{{tab.id}}-header" [attr.title]="tab.title || (!tab.headingIsTemplate ? tab.heading : null)" (focus)="onTabFocus(i)" (click)="selectTab(tab, i)"> <div class="cds--tabs__nav-item-label-wrapper"> <span class="cds--tabs__nav-item-label"> <ng-container *ngIf="!tab.headingIsTemplate"> {{ tab.heading }} </ng-container> <ng-template *ngIf="tab.headingIsTemplate" [ngTemplateOutlet]="tab.heading" [ngTemplateOutletContext]="{$implicit: tab.context}"> </ng-template> </span> </div> </button> <ng-container [ngTemplateOutlet]="contentAfter"></ng-container> </div> <div *ngIf="isOverflowingBottom" class="cds--tab--list-gradient_bottom"></div> `, isInline: true, dependencies: [{ kind: "directive", type: i3.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: TabHeadersVertical, decorators: [{ type: Component, args: [{ selector: "cds-tab-headers-vertical, ibm-tab-headers-vertical", template: ` <div *ngIf="isOverflowingTop" class="cds--tab--list-gradient_top"></div> <div #tabList class="cds--tab--list" role="tablist" [attr.aria-label]="ariaLabel || translations.HEADER_ARIA_LABEL" [attr.aria-labelledby]="ariaLabelledby || null"> <ng-container [ngTemplateOutlet]="contentBefore"></ng-container> <button *ngFor="let tab of tabs; let i = index;" #tabItem role="tab" [attr.aria-selected]="tab.active" [attr.tabindex]="(tab.active?0:-1)" [attr.aria-controls]="tab.id" [attr.aria-disabled]="tab.disabled" [disabled]="tab.disabled" [ngClass]="{ 'cds--tabs__nav-item--selected': tab.active, 'cds--tabs__nav-item--disabled': tab.disabled }" class="cds--tabs__nav-item cds--tabs__nav-link" type="button" draggable="false" id="{{tab.id}}-header" [attr.title]="tab.title || (!tab.headingIsTemplate ? tab.heading : null)" (focus)="onTabFocus(i)" (click)="selectTab(tab, i)"> <div class="cds--tabs__nav-item-label-wrapper"> <span class="cds--tabs__nav-item-label"> <ng-container *ngIf="!tab.headingIsTemplate"> {{ tab.heading }} </ng-container> <ng-template *ngIf="tab.headingIsTemplate" [ngTemplateOutlet]="tab.heading" [ngTemplateOutletContext]="{$implicit: tab.context}"> </ng-template> </span> </div> </button> <ng-container [ngTemplateOutlet]="contentAfter"></ng-container> </div> <div *ngIf="isOverflowingBottom" class="cds--tab--list-gradient_bottom"></div> ` }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }, { type: i1.EventService }, { type: i0.Renderer2 }, { type: i2.I18n }]; }, propDecorators: { tabInput: [{ type: Input, args: ["tabs"] }], translations: [{ type: Input }], verticalClass: [{ type: HostBinding, args: ["class.cds--tabs--vertical"] }], headerContainer: [{ type: ViewChild, args: ["tabList", { static: true }] }], tabQuery: [{ type: ContentChildren, args: [Tab] }], allTabHeaders: [{ type: ViewChildren, args: ["tabItem"] }], keyboardInput: [{ type: HostListener, args: ["keydown", ["$event"]] }], handleBlur: [{ type: HostListener, args: ["blur", ["$event"]] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab-headers-vertical.component.js","sourceRoot":"","sources":["../../../src/tabs/tab-headers-vertical.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n\tComponent,\r\n\tQueryList,\r\n\tInput,\r\n\tHostListener,\r\n\tHostBinding,\r\n\tViewChild,\r\n\tContentChildren,\r\n\tAfterContentInit,\r\n\tViewChildren,\r\n\tElementRef,\r\n\tOnChanges,\r\n\tSimpleChanges,\r\n\tOnDestroy,\r\n\tOnInit,\r\n\tChangeDetectorRef,\r\n\tRenderer2\r\n} from \"@angular/core\";\r\nimport { EventService } from \"carbon-components-angular/utils\";\r\nimport { I18n } from \"carbon-components-angular/i18n\";\r\n\r\nimport { BaseTabHeader } from \"./base-tab-header.component\";\r\nimport { Tab } from \"./tab.component\";\r\n\r\nconst VERTICAL_TAB_HEIGHT = 64;\r\n\r\n/**\r\n * The `TabHeadersVertical` component renders tab headers in a vertical\r\n * orientation. It contains the `Tab` items and supports keyboard navigation\r\n * via ArrowUp/ArrowDown/Home/End.\r\n */\r\n@Component({\r\n\tselector: \"cds-tab-headers-vertical, ibm-tab-headers-vertical\",\r\n\ttemplate: `\r\n\t\t<div *ngIf=\"isOverflowingTop\" class=\"cds--tab--list-gradient_top\"></div>\r\n\t\t<div\r\n\t\t\t#tabList\r\n\t\t\tclass=\"cds--tab--list\"\r\n\t\t\trole=\"tablist\"\r\n\t\t\t[attr.aria-label]=\"ariaLabel || translations.HEADER_ARIA_LABEL\"\r\n\t\t\t[attr.aria-labelledby]=\"ariaLabelledby || null\">\r\n\t\t\t<ng-container [ngTemplateOutlet]=\"contentBefore\"></ng-container>\r\n\t\t\t<button\r\n\t\t\t\t*ngFor=\"let tab of tabs; let i = index;\"\r\n\t\t\t\t#tabItem\r\n\t\t\t\trole=\"tab\"\r\n\t\t\t\t[attr.aria-selected]=\"tab.active\"\r\n\t\t\t\t[attr.tabindex]=\"(tab.active?0:-1)\"\r\n\t\t\t\t[attr.aria-controls]=\"tab.id\"\r\n\t\t\t\t[attr.aria-disabled]=\"tab.disabled\"\r\n\t\t\t\t[disabled]=\"tab.disabled\"\r\n\t\t\t\t[ngClass]=\"{\r\n\t\t\t\t\t'cds--tabs__nav-item--selected': tab.active,\r\n\t\t\t\t\t'cds--tabs__nav-item--disabled': tab.disabled\r\n\t\t\t\t}\"\r\n\t\t\t\tclass=\"cds--tabs__nav-item cds--tabs__nav-link\"\r\n\t\t\t\ttype=\"button\"\r\n\t\t\t\tdraggable=\"false\"\r\n\t\t\t\tid=\"{{tab.id}}-header\"\r\n\t\t\t\t[attr.title]=\"tab.title || (!tab.headingIsTemplate ? tab.heading : null)\"\r\n\t\t\t\t(focus)=\"onTabFocus(i)\"\r\n\t\t\t\t(click)=\"selectTab(tab, i)\">\r\n\t\t\t\t<div class=\"cds--tabs__nav-item-label-wrapper\">\r\n\t\t\t\t\t<span class=\"cds--tabs__nav-item-label\">\r\n\t\t\t\t\t\t<ng-container *ngIf=\"!tab.headingIsTemplate\">\r\n\t\t\t\t\t\t\t{{ tab.heading }}\r\n\t\t\t\t\t\t</ng-container>\r\n\t\t\t\t\t\t<ng-template\r\n\t\t\t\t\t\t\t*ngIf=\"tab.headingIsTemplate\"\r\n\t\t\t\t\t\t\t[ngTemplateOutlet]=\"tab.heading\"\r\n\t\t\t\t\t\t\t[ngTemplateOutletContext]=\"{$implicit: tab.context}\">\r\n\t\t\t\t\t\t</ng-template>\r\n\t\t\t\t\t</span>\r\n\t\t\t\t</div>\r\n\t\t\t</button>\r\n\t\t\t<ng-container [ngTemplateOutlet]=\"contentAfter\"></ng-container>\r\n\t\t</div>\r\n\t\t<div *ngIf=\"isOverflowingBottom\" class=\"cds--tab--list-gradient_bottom\"></div>\r\n\t`\r\n})\r\nexport class TabHeadersVertical extends BaseTabHeader implements AfterContentInit, OnChanges, OnDestroy, OnInit {\r\n\t/**\r\n\t * List of `Tab` components.\r\n\t */\r\n\t// disable the next line because we need to rename the input\r\n\t// tslint:disable-next-line\r\n\t@Input(\"tabs\") tabInput: QueryList<Tab>;\r\n\r\n\t/**\r\n\t * i18n strings for the tab list `aria-label` fallback.\r\n\t */\r\n\t@Input() translations = this.i18n.get().TABS;\r\n\r\n\t@HostBinding(\"class.cds--tabs--vertical\") verticalClass = true;\r\n\r\n\t@ViewChild(\"tabList\", { static: true }) headerContainer: ElementRef<HTMLElement>;\r\n\r\n\t/**\r\n\t * ContentChild of all the tabs\r\n\t */\r\n\t@ContentChildren(Tab) tabQuery: QueryList<Tab>;\r\n\ttabs: QueryList<Tab>;\r\n\r\n\t@ViewChildren(\"tabItem\") allTabHeaders: QueryList<ElementRef>;\r\n\r\n\t/**\r\n\t * Focused tab index when `followFocus` is false (manual activation)\r\n\t */\r\n\tactiveIndex: number | null = null;\r\n\r\n\t/**\r\n\t * Whether the tab list is overflowing at the top (some tabs are clipped).\r\n\t */\r\n\tisOverflowingTop = false;\r\n\t/**\r\n\t * Whether the tab list is overflowing at the bottom (some tabs are clipped).\r\n\t */\r\n\tisOverflowingBottom = false;\r\n\r\n\tprivate resizeObserver: ResizeObserver;\r\n\r\n\tconstructor(\r\n\t\tprotected elementRef: ElementRef,\r\n\t\tprotected changeDetectorRef: ChangeDetectorRef,\r\n\t\tprotected eventService: EventService,\r\n\t\tprotected renderer: Renderer2,\r\n\t\tprotected i18n: I18n\r\n\t) {\r\n\t\tsuper(elementRef, changeDetectorRef, eventService, renderer);\r\n\t\tthis.type = \"contained\";\r\n\t}\r\n\r\n\t@HostListener(\"keydown\", [\"$event\"])\r\n\tkeyboardInput(event: KeyboardEvent) {\r\n\t\tif (!this.tabs) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tconst tabsArray = this.tabs.toArray();\r\n\t\tconst enabledTabs = tabsArray.filter(tab => !tab.disabled);\r\n\t\tif (enabledTabs.length === 0) {\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tconst referenceIndex = this.followFocus ?\r\n\t\t\tthis.currentSelectedTab :\r\n\t\t\t(this.activeIndex !== null ? this.activeIndex : this.currentSelectedTab);\r\n\t\tconst currentEnabledIndex = Math.max(0, enabledTabs.indexOf(tabsArray[referenceIndex]));\r\n\r\n\t\tlet nextEnabledIndex = currentEnabledIndex;\r\n\t\tlet handled = false;\r\n\r\n\t\tif (event.key === \"ArrowDown\") {\r\n\t\t\tnextEnabledIndex = (currentEnabledIndex + 1) % enabledTabs.length;\r\n\t\t\thandled = true;\r\n\t\t} else if (event.key === \"ArrowUp\") {\r\n\t\t\tnextEnabledIndex = (enabledTabs.length + currentEnabledIndex - 1) % enabledTabs.length;\r\n\t\t\thandled = true;\r\n\t\t} else if (event.key === \"Home\") {\r\n\t\t\tnextEnabledIndex = 0;\r\n\t\t\thandled = true;\r\n\t\t} else if (event.key === \"End\") {\r\n\t\t\tnextEnabledIndex = enabledTabs.length - 1;\r\n\t\t\thandled = true;\r\n\t\t}\r\n\r\n\t\tif (handled) {\r\n\t\t\tevent.preventDefault();\r\n\t\t\tconst nextTab = enabledTabs[nextEnabledIndex];\r\n\t\t\tconst nextIndex = tabsArray.indexOf(nextTab);\r\n\r\n\t\t\tif (this.followFocus) {\r\n\t\t\t\tthis.selectTab(nextTab, nextIndex);\r\n\t\t\t} else {\r\n\t\t\t\tthis.activeIndex = nextIndex;\r\n\t\t\t}\r\n\t\t\tthis.allTabHeaders.toArray()[nextIndex].nativeElement.focus();\r\n\t\t\treturn;\r\n\t\t}\r\n\r\n\t\tif ((event.key === \" \" || event.key === \"Spacebar\") && !this.followFocus) {\r\n\t\t\tconst focusIndex = this.activeIndex !== null ? this.activeIndex : this.currentSelectedTab;\r\n\t\t\tthis.selectTab(tabsArray[focusIndex], focusIndex);\r\n\t\t}\r\n\t}\r\n\r\n\t@HostListener(\"blur\", [\"$event\"])\r\n\thandleBlur(event: FocusEvent) {\r\n\t\tconst relatedTarget = event.relatedTarget as Node | null;\r\n\t\tconst container = this.headerContainer?.nativeElement;\r\n\t\tif (container && relatedTarget && container.contains(relatedTarget)) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tif (!this.followFocus) {\r\n\t\t\tthis.activeIndex = this.currentSelectedTab;\r\n\t\t}\r\n\t}\r\n\r\n\tngOnInit(): void {\r\n\t\tthis.resizeObserver = new ResizeObserver(() => {\r\n\t\t\tthis.updateOverflowState();\r\n\t\t\tthis.changeDetectorRef.detectChanges();\r\n\t\t});\r\n\t\tthis.resizeObserver.observe(this.headerContainer.nativeElement);\r\n\t\tthis.headerContainer.nativeElement.addEventListener(\"scroll\", this.listScrollHandler);\r\n\t}\r\n\r\n\tngOnDestroy(): void {\r\n\t\tthis.resizeObserver?.unobserve(this.headerContainer.nativeElement);\r\n\t\tthis.headerContainer.nativeElement.removeEventListener(\"scroll\", this.listScrollHandler);\r\n\t}\r\n\r\n\tngAfterContentInit() {\r\n\t\tif (!this.tabInput) {\r\n\t\t\tthis.tabs = this.tabQuery;\r\n\t\t} else {\r\n\t\t\tthis.tabs = this.tabInput;\r\n\t\t}\r\n\r\n\t\tthis.tabs.forEach(tab => tab.cacheActive = this.cacheActive);\r\n\t\tthis.tabs.changes.subscribe(() => {\r\n\t\t\tthis.setFirstTab();\r\n\t\t\tthis.changeDetectorRef.markForCheck();\r\n\t\t});\r\n\t\tthis.setFirstTab();\r\n\t}\r\n\r\n\tngOnChanges(changes: SimpleChanges) {\r\n\t\tif (this.tabs && changes.cacheActive) {\r\n\t\t\tthis.tabs.forEach(tab => tab.cacheActive = this.cacheActive);\r\n\t\t}\r\n\t}\r\n\r\n\tonTabFocus(index: number) {\r\n\t\tif (this.followFocus) {\r\n\t\t\tthis.currentSelectedTab = index;\r\n\t\t} else {\r\n\t\t\tthis.activeIndex = index;\r\n\t\t}\r\n\t\tthis.scrollSelectedTabIntoView();\r\n\t}\r\n\r\n\tselectTab(tab: Tab, tabIndex: number) {\r\n\t\tif (tab.disabled) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tthis.currentSelectedTab = tabIndex;\r\n\t\tthis.activeIndex = tabIndex;\r\n\t\tthis.tabs.forEach(_tab => _tab.active = false);\r\n\t\ttab.active = true;\r\n\t\ttab.doSelect();\r\n\t\tthis.scrollSelectedTabIntoView();\r\n\t}\r\n\r\n\tgetSelectedTab(): any {\r\n\t\tconst selected = this.tabs.find(tab => tab.active);\r\n\t\tif (selected) {\r\n\t\t\treturn selected;\r\n\t\t}\r\n\t\treturn { headingIsTemplate: false, heading: \"\" };\r\n\t}\r\n\r\n\tprotected updateOverflowState() {\r\n\t\tconst element = this.headerContainer?.nativeElement;\r\n\t\tif (!element) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tconst halfTabHeight = VERTICAL_TAB_HEIGHT / 2;\r\n\t\tthis.isOverflowingBottom =\r\n\t\t\telement.scrollTop + element.clientHeight + halfTabHeight <= element.scrollHeight;\r\n\t\tthis.isOverflowingTop = element.scrollTop > halfTabHeight;\r\n\t\tthis.changeDetectorRef.markForCheck();\r\n\t}\r\n\r\n\tprotected scrollSelectedTabIntoView() {\r\n\t\tconst container = this.headerContainer?.nativeElement;\r\n\t\tif (!container) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tconst selectedHeader = this.allTabHeaders?.toArray()[this.currentSelectedTab]?.nativeElement;\r\n\t\tif (!selectedHeader) {\r\n\t\t\treturn;\r\n\t\t}\r\n\t\tconst containerRect = container.getBoundingClientRect();\r\n\t\tconst selectedRect = selectedHeader.getBoundingClientRect();\r\n\t\tconst halfTabHeight = VERTICAL_TAB_HEIGHT / 2;\r\n\r\n\t\tif (\r\n\t\t\tselectedRect.top - halfTabHeight < containerRect.top ||\r\n\t\t\tselectedRect.top - containerRect.top + VERTICAL_TAB_HEIGHT + halfTabHeight > containerRect.height\r\n\t\t) {\r\n\t\t\tcontainer.scrollTo({\r\n\t\t\t\ttop: Math.max(0, (this.currentSelectedTab - 1) * VERTICAL_TAB_HEIGHT),\r\n\t\t\t\tbehavior: \"smooth\"\r\n\t\t\t});\r\n\t\t}\r\n\t}\r\n\r\n\tprotected setFirstTab() {\r\n\t\tsetTimeout(() => {\r\n\t\t\tlet firstTab = this.tabs.find(tab => tab.active);\r\n\t\t\tif (!firstTab && this.tabs.first) {\r\n\t\t\t\tfirstTab = this.tabs.first;\r\n\t\t\t\tfirstTab.active = true;\r\n\t\t\t}\r\n\t\t\tif (firstTab) {\r\n\t\t\t\tthis.currentSelectedTab = this.tabs.toArray().indexOf(firstTab);\r\n\t\t\t\tthis.activeIndex = this.currentSelectedTab;\r\n\t\t\t\tfirstTab.doSelect();\r\n\t\t\t\tthis.updateOverflowState();\r\n\t\t\t}\r\n\t\t});\r\n\t}\r\n\tprivate listScrollHandler = () => this.updateOverflowState();\r\n}\r\n"]}