carbon-components-angular
Version:
Next generation components
328 lines • 37.7 kB
JavaScript
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"]}