@catull/igniteui-angular
Version:
Ignite UI for Angular is a dependency-free Angular toolkit for building modern web apps
452 lines • 52.6 kB
JavaScript
var IgxTabsComponent_1;
import { __decorate, __metadata } from "tslib";
import { CommonModule } from '@angular/common';
import { AfterViewInit, Component, ContentChildren, ElementRef, EventEmitter, forwardRef, HostBinding, Input, NgModule, Output, QueryList, ViewChild, ViewChildren, OnDestroy, NgZone } from '@angular/core';
import { IgxBadgeModule } from '../badge/badge.component';
import { IgxRippleModule } from '../directives/ripple/ripple.directive';
import { IgxIconModule } from '../icon/index';
import { IgxTabItemComponent } from './tab-item.component';
import { IgxTabsGroupComponent } from './tabs-group.component';
import { IgxLeftButtonStyleDirective, IgxRightButtonStyleDirective, IgxTabItemTemplateDirective } from './tabs.directives';
import { IgxTabsBase } from './tabs.common';
import ResizeObserver from 'resize-observer-polyfill';
export var TabsType;
(function (TabsType) {
TabsType["FIXED"] = "fixed";
TabsType["CONTENTFIT"] = "contentfit";
})(TabsType || (TabsType = {}));
let IgxTabsComponent = IgxTabsComponent_1 = class IgxTabsComponent {
constructor(_element, _ngZone) {
this._element = _element;
this._ngZone = _ngZone;
/**
*@hidden
*/
this.selectedIndexChange = new EventEmitter();
/**
* Defines the tab header sizing mode. You can choose between `contentfit` or `fixed`.
* By default the header sizing mode is `contentfit`.
* ```html
* <igx-tabs tabsType="fixed">
* <igx-tabs-group label="HOME">Home</igx-tabs-group>
* </igx-tabs>
* ```
*/
this.tabsType = 'contentfit';
/**
* @hidden
*/
this.class = '';
/**
* Emitted when a tab item is deselected.
* ```html
* <igx-tabs (onTabItemDeselected)="itemDeselected($event)">
* <igx-tabs-group label="Tab 1">This is Tab 1 content.</igx-tabs-group>
* <igx-tabs-group label="Tab 2">This is Tab 2 content.</igx-tabs-group>
* </igx-tabs>
* ```
* ```typescript
* itemDeselected(e){
* const tabGroup = e.group;
* const tabItem = e.tab;
* }
* ```
*/
this.onTabItemDeselected = new EventEmitter();
/**
* Emitted when a tab item is selected.
* ```html
* <igx-tabs (onTabItemSelected)="itemSelected($event)">
* <igx-tabs-group label="Tab 1">This is Tab 1 content.</igx-tabs-group>
* <igx-tabs-group label="Tab 2">This is Tab 2 content.</igx-tabs-group>
* </igx-tabs>
* ```
* ```typescript
* itemSelected(e){
* const tabGroup = e.group;
* const tabItem = e.tab;
* }
* ```
*/
this.onTabItemSelected = new EventEmitter();
/**
* @hidden
*/
this.offset = 0;
this._selectedIndex = -1;
}
/**
* An @Input property that sets the value of the `selectedIndex`.
* Default value is 0.
* ```html
* <igx-tabs selectedIndex="1">
* ```
*
* Two-way data binding.
* ```html
* <igx-tabs [(selectedIndex)]="model.selectedIndex">
* ```
*/
get selectedIndex() {
return this._selectedIndex;
}
set selectedIndex(index) {
const newIndex = typeof index !== 'number' ? parseInt(index, 10) : index;
if (this._selectedIndex !== newIndex) {
if (this.tabs && this.tabs.length > 0) {
const newTab = this.tabs.toArray()[newIndex];
if (newTab) {
this.performSelectionChange(newTab);
}
}
else {
this._selectedIndex = newIndex;
}
}
}
/**
* Provides an observable collection of all `IgxTabItemComponent`s.
* First try to get them as content children if not available get them as view children.
* ```typescript
* const tabItems = this.myTabComponent.tabs;
* ```
*/
get tabs() {
if (this.hasContentTabs) {
return this.contentTabs;
}
return this.viewTabs;
}
/**
*@hidden
*/
get hasContentTabs() {
return (this.contentTabs && this.contentTabs.length > 0);
}
/**
* @hidden
*/
get cssClass() {
const defaultStyle = `igx-tabs`;
const fixedStyle = `igx-tabs--fixed`;
const iconStyle = `igx-tabs--icons`;
const iconLabelFoundInGroups = this.groups.find((group) => group.icon != null && group.label != null);
const iconLabelFoundInTabs = this.contentTabs.find((tab) => tab.icon != null && tab.label != null);
let css;
switch (TabsType[this.tabsType.toUpperCase()]) {
case TabsType.FIXED: {
css = fixedStyle;
break;
}
default: {
css = defaultStyle;
break;
}
}
// Layout fix for items with icons
if (iconLabelFoundInGroups !== undefined || iconLabelFoundInTabs !== undefined) {
css = `${css} ${iconStyle}`;
}
return `${css} ${this.class}`;
}
/**
* @hidden
*/
scrollLeft(event) {
this.scroll(false);
}
/**
* @hidden
*/
scrollRight(event) {
this.scroll(true);
}
/**
* @hidden
*/
scrollElement(element, scrollRight) {
const viewPortWidth = this.viewPort.nativeElement.offsetWidth;
this.offset = (scrollRight) ? element.offsetWidth + element.offsetLeft - viewPortWidth : element.offsetLeft;
this.itemsContainer.nativeElement.style.transform = `translate(${-this.offset}px)`;
}
/**
* Gets the selected `IgxTabItemComponent`.
* ```
* const selectedItem = this.myTabComponent.selectedTabItem;
* ```
*/
get selectedTabItem() {
if (this.tabs && this.selectedIndex !== undefined) {
return this.tabs.toArray()[this.selectedIndex];
}
}
/**
* @hidden
*/
ngAfterViewInit() {
if (this._selectedIndex === -1) {
this.tabs.forEach((t) => {
if (t.isSelected) {
this._selectedIndex = t.index;
}
});
}
if (!this.hasContentTabs && (this.selectedIndex < 0 || this.selectedIndex >= this.groups.length)) {
this._selectedIndex = 0;
}
requestAnimationFrame(() => {
const newTab = this.tabs.toArray()[this._selectedIndex];
if (newTab) {
this.performSelection(newTab);
}
else {
this.hideIndicator();
}
});
this._groupChanges$ = this.groups.changes.subscribe(() => {
this.resetSelectionOnCollectionChanged();
});
this._ngZone.runOutsideAngular(() => {
this._resizeObserver = new ResizeObserver(() => {
if (!this.hasContentTabs && this._selectedIndex >= 0 && this._selectedIndex < this.tabs.length) {
const newTab = this.tabs.toArray()[this._selectedIndex];
this.transformContentAnimation(newTab, 0);
}
});
this._resizeObserver.observe(this.tabsContainer.nativeElement);
});
}
/**
* @hidden
*/
ngOnDestroy() {
if (this._groupChanges$) {
this._groupChanges$.unsubscribe();
}
this._ngZone.runOutsideAngular(() => {
this._resizeObserver.disconnect();
});
}
resetSelectionOnCollectionChanged() {
requestAnimationFrame(() => {
const currentTab = this.tabs.toArray()[this.selectedIndex];
if (currentTab) {
this.performSelectionChange(currentTab);
}
else if (this.selectedIndex >= this.tabs.length) {
this.performSelectionChange(this.tabs.last);
}
else {
this.hideIndicator();
}
});
}
scroll(scrollRight) {
const tabsArray = this.tabs.toArray();
for (const tab of tabsArray) {
const element = tab.nativeTabItem.nativeElement;
if (scrollRight) {
if (element.offsetWidth + element.offsetLeft > this.viewPort.nativeElement.offsetWidth + this.offset) {
this.scrollElement(element, scrollRight);
break;
}
}
else {
if (element.offsetWidth + element.offsetLeft >= this.offset) {
this.scrollElement(element, scrollRight);
break;
}
}
}
}
/**
* @hidden
*/
performSelectionChange(newTab) {
const oldTab = this.selectedTabItem;
if (oldTab) {
this.performDeselection(oldTab);
}
if (newTab) {
this.performSelection(newTab);
}
else {
// if there is no new selected tab hide the selection indicator
this.hideIndicator();
}
this.selectedIndexChange.emit(this._selectedIndex);
}
performDeselection(oldTab) {
oldTab.setSelectedInternal(false);
const oldTabRelatedGroup = this.groups.toArray()[oldTab.index];
if (oldTabRelatedGroup) {
oldTabRelatedGroup.setSelectedInternal(false);
}
this._selectedIndex = -1;
this.onTabItemDeselected.emit({ tab: oldTab, group: oldTabRelatedGroup });
}
performSelection(newTab) {
newTab.setSelectedInternal(true);
this._selectedIndex = newTab.index;
let newTabRelatedGroup = null;
if (!this.hasContentTabs && this.groups) {
newTabRelatedGroup = this.groups.toArray()[newTab.index];
if (newTabRelatedGroup) {
newTabRelatedGroup.setSelectedInternal(true);
}
}
this.onTabItemSelected.emit({ tab: newTab, group: newTabRelatedGroup });
requestAnimationFrame(() => {
// bring the new selected tab into view if it is not
this.bringNewTabIntoView(newTab);
// animate the new selection indicator
this.transformIndicatorAnimation(newTab.nativeTabItem.nativeElement);
// animate the new tab's group content
if (!this.hasContentTabs) {
this.transformContentAnimation(newTab, 0.2);
}
});
}
bringNewTabIntoView(newTab) {
const tabNativeElement = newTab.nativeTabItem.nativeElement;
// Scroll left if there is need
if (tabNativeElement.offsetLeft < this.offset) {
this.scrollElement(tabNativeElement, false);
}
// Scroll right if there is need
const viewPortOffsetWidth = this.viewPort.nativeElement.offsetWidth;
const delta = (tabNativeElement.offsetLeft + tabNativeElement.offsetWidth) - (viewPortOffsetWidth + this.offset);
// Fix for IE 11, a difference is accumulated from the widths calculations
if (delta > 1) {
this.scrollElement(tabNativeElement, true);
}
}
/**
* @hidden
*/
// animation for the new panel/group (not needed for tab only mode)
transformContentAnimation(tab, duration) {
const contentOffset = this.tabsContainer.nativeElement.offsetWidth * tab.index;
this.contentsContainer.nativeElement.style.transitionDuration = duration > 0 ? `${duration}s` : 'initial';
this.contentsContainer.nativeElement.style.transform = `translate(${-contentOffset}px)`;
}
/**
* @hidden
*/
transformIndicatorAnimation(element, duration = 0.3) {
if (this.selectedIndicator) {
this.selectedIndicator.nativeElement.style.visibility = 'visible';
this.selectedIndicator.nativeElement.style.transitionDuration = duration > 0 ? `${duration}s` : 'initial';
this.selectedIndicator.nativeElement.style.width = `${element.offsetWidth}px`;
this.selectedIndicator.nativeElement.style.transform = `translate(${element.offsetLeft}px)`;
}
}
hideIndicator() {
if (this.selectedIndicator) {
this.selectedIndicator.nativeElement.style.visibility = 'hidden';
}
}
};
IgxTabsComponent.ctorParameters = () => [
{ type: ElementRef },
{ type: NgZone }
];
__decorate([
ContentChildren(forwardRef(() => IgxTabsGroupComponent)),
__metadata("design:type", QueryList)
], IgxTabsComponent.prototype, "groups", void 0);
__decorate([
ContentChildren(forwardRef(() => IgxTabItemComponent)),
__metadata("design:type", QueryList)
], IgxTabsComponent.prototype, "contentTabs", void 0);
__decorate([
Input(),
__metadata("design:type", Number),
__metadata("design:paramtypes", [Number])
], IgxTabsComponent.prototype, "selectedIndex", null);
__decorate([
Output(),
__metadata("design:type", Object)
], IgxTabsComponent.prototype, "selectedIndexChange", void 0);
__decorate([
Input('tabsType'),
__metadata("design:type", String)
], IgxTabsComponent.prototype, "tabsType", void 0);
__decorate([
Input(),
__metadata("design:type", Object)
], IgxTabsComponent.prototype, "class", void 0);
__decorate([
Output(),
__metadata("design:type", Object)
], IgxTabsComponent.prototype, "onTabItemDeselected", void 0);
__decorate([
Output(),
__metadata("design:type", Object)
], IgxTabsComponent.prototype, "onTabItemSelected", void 0);
__decorate([
ViewChild('contentsContainer', { static: true }),
__metadata("design:type", ElementRef)
], IgxTabsComponent.prototype, "contentsContainer", void 0);
__decorate([
ViewChild('headerContainer', { static: true }),
__metadata("design:type", ElementRef)
], IgxTabsComponent.prototype, "headerContainer", void 0);
__decorate([
ViewChild('itemsContainer', { static: true }),
__metadata("design:type", ElementRef)
], IgxTabsComponent.prototype, "itemsContainer", void 0);
__decorate([
ViewChild('selectedIndicator'),
__metadata("design:type", ElementRef)
], IgxTabsComponent.prototype, "selectedIndicator", void 0);
__decorate([
ViewChild('tabsContainer', { static: true }),
__metadata("design:type", ElementRef)
], IgxTabsComponent.prototype, "tabsContainer", void 0);
__decorate([
ViewChild('viewPort', { static: true }),
__metadata("design:type", ElementRef)
], IgxTabsComponent.prototype, "viewPort", void 0);
__decorate([
ViewChildren(forwardRef(() => IgxTabItemComponent)),
__metadata("design:type", QueryList)
], IgxTabsComponent.prototype, "viewTabs", void 0);
__decorate([
HostBinding('attr.class'),
__metadata("design:type", Object),
__metadata("design:paramtypes", [])
], IgxTabsComponent.prototype, "cssClass", null);
IgxTabsComponent = IgxTabsComponent_1 = __decorate([
Component({
selector: 'igx-tabs',
template: "<!-- TODO Remove tab container from here -->\n<div #tabsContainer>\n <div class=\"igx-tabs__header\" #headerContainer>\n <button igxRipple class=\"igx-tabs__header-button\" igxButton=\"icon\" (click)=\"scrollLeft($event)\" igxLeftButtonStyle>\n <igx-icon fontSet=\"material\">navigate_before</igx-icon>\n </button>\n <div class=\"igx-tabs__header-wrapper-fixed\" #viewPort>\n <div #itemsContainer class=\"igx-tabs__header-wrapper-fluid\">\n <ng-container *ngIf=\"!hasContentTabs\">\n <igx-tab-item igxRipple *ngFor=\"let group of groups\" [relatedGroup]=\"group\">\n </igx-tab-item>\n </ng-container>\n <ng-content select=\"igx-tab-item\"></ng-content>\n <div #selectedIndicator *ngIf=\"groups.length > 0 || contentTabs.length > 0\" class=\"igx-tabs__header-menu-item-indicator\"></div>\n </div>\n </div>\n <button igxRipple class=\"igx-tabs__header-button\" igxButton=\"icon\" (click)=\"scrollRight($event)\" igxRightButtonStyle>\n <igx-icon fontSet=\"material\">navigate_next</igx-icon>\n </button>\n </div>\n <div class=\"igx-tabs__content-fixed\">\n <div #contentsContainer class=\"igx-tabs__content-fluid\">\n <ng-content></ng-content>\n </div>\n </div>\n</div>",
providers: [{ provide: IgxTabsBase, useExisting: IgxTabsComponent_1 }]
}),
__metadata("design:paramtypes", [ElementRef, NgZone])
], IgxTabsComponent);
export { IgxTabsComponent };
/**
* @hidden
*/
let IgxTabsModule = class IgxTabsModule {
};
IgxTabsModule = __decorate([
NgModule({
declarations: [IgxTabsComponent,
IgxTabsGroupComponent,
IgxTabItemComponent,
IgxTabItemTemplateDirective,
IgxRightButtonStyleDirective,
IgxLeftButtonStyleDirective],
exports: [IgxTabsComponent,
IgxTabsGroupComponent,
IgxTabItemComponent,
IgxTabItemTemplateDirective,
IgxRightButtonStyleDirective,
IgxLeftButtonStyleDirective],
imports: [CommonModule, IgxBadgeModule, IgxIconModule, IgxRippleModule]
})
], IgxTabsModule);
export { IgxTabsModule };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs.component.js","sourceRoot":"ng://igniteui-angular/","sources":["lib/tabs/tabs.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n    AfterViewInit,\n    Component,\n    ContentChildren,\n    ElementRef,\n    EventEmitter,\n    forwardRef,\n    HostBinding,\n    Input,\n    NgModule,\n    Output,\n    QueryList,\n    ViewChild,\n    ViewChildren,\n    OnDestroy,\n    NgZone\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { IgxBadgeModule } from '../badge/badge.component';\nimport { IgxRippleModule } from '../directives/ripple/ripple.directive';\nimport { IgxIconModule } from '../icon/index';\nimport { IgxTabItemComponent } from './tab-item.component';\nimport { IgxTabsGroupComponent } from './tabs-group.component';\nimport { IgxLeftButtonStyleDirective, IgxRightButtonStyleDirective, IgxTabItemTemplateDirective } from './tabs.directives';\nimport { IgxTabsBase, IgxTabItemBase } from './tabs.common';\nimport ResizeObserver from 'resize-observer-polyfill';\n\nexport enum TabsType {\n    FIXED = 'fixed',\n    CONTENTFIT = 'contentfit'\n}\n\n@Component({\n    selector: 'igx-tabs',\n    templateUrl: 'tabs.component.html',\n    providers: [{ provide: IgxTabsBase, useExisting: IgxTabsComponent }]\n})\n\nexport class IgxTabsComponent implements IgxTabsBase, AfterViewInit, OnDestroy {\n    /**\n    * Provides an observable collection of all `IgxTabsGroupComponent`s.\n    * ```typescript\n    * const groupItems = this.myTabComponent.groups;\n    * ```\n    */\n    @ContentChildren(forwardRef(() => IgxTabsGroupComponent))\n    public groups: QueryList<IgxTabsGroupComponent>;\n\n    /**\n    * Provides an observable collection of all `IgxTabItemComponent`s defined in the page.\n    * ```typescript\n    * const tabItems = this.myTabComponent.contentTabs;\n    * ```\n    */\n    @ContentChildren(forwardRef(() => IgxTabItemComponent))\n    public contentTabs: QueryList<IgxTabItemComponent>;\n\n    /**\n    * An @Input property that sets the value of the `selectedIndex`.\n    * Default value is 0.\n    * ```html\n    * <igx-tabs selectedIndex=\"1\">\n    * ```\n    *\n    * Two-way data binding.\n    * ```html\n    * <igx-tabs [(selectedIndex)]=\"model.selectedIndex\">\n    * ```\n    */\n    @Input()\n    public get selectedIndex(): number {\n        return this._selectedIndex;\n    }\n\n    public set selectedIndex(index: number) {\n        const newIndex = typeof index !== 'number' ? parseInt(index, 10) : index;\n        if (this._selectedIndex !== newIndex) {\n            if (this.tabs && this.tabs.length > 0) {\n                const newTab = this.tabs.toArray()[newIndex];\n                if (newTab) {\n                   this.performSelectionChange(newTab);\n                }\n            } else {\n                this._selectedIndex = newIndex;\n            }\n        }\n    }\n\n    /**\n     *@hidden\n     */\n    @Output()\n    public selectedIndexChange = new EventEmitter<number>();\n\n    /**\n     * Defines the tab header sizing mode. You can choose between `contentfit` or `fixed`.\n     * By default the header sizing mode is `contentfit`.\n     * ```html\n     * <igx-tabs tabsType=\"fixed\">\n     *     <igx-tabs-group label=\"HOME\">Home</igx-tabs-group>\n     * </igx-tabs>\n     * ```\n     */\n    @Input('tabsType')\n    public tabsType: string | TabsType = 'contentfit';\n\n    /**\n    * @hidden\n    */\n    @Input()\n    public class = '';\n\n    /**\n     * Emitted when a tab item is deselected.\n     * ```html\n     * <igx-tabs (onTabItemDeselected)=\"itemDeselected($event)\">\n     *      <igx-tabs-group label=\"Tab 1\">This is Tab 1 content.</igx-tabs-group>\n     *      <igx-tabs-group label=\"Tab 2\">This is Tab 2 content.</igx-tabs-group>\n     * </igx-tabs>\n     * ```\n     * ```typescript\n     * itemDeselected(e){\n     *      const tabGroup = e.group;\n     *      const tabItem = e.tab;\n     * }\n     * ```\n     */\n    @Output()\n    public onTabItemDeselected = new EventEmitter();\n\n    /**\n    * Emitted when a tab item is selected.\n    * ```html\n    * <igx-tabs (onTabItemSelected)=\"itemSelected($event)\">\n    *      <igx-tabs-group label=\"Tab 1\">This is Tab 1 content.</igx-tabs-group>\n    *      <igx-tabs-group label=\"Tab 2\">This is Tab 2 content.</igx-tabs-group>\n    * </igx-tabs>\n    * ```\n    * ```typescript\n    * itemSelected(e){\n    *      const tabGroup = e.group;\n    *      const tabItem = e.tab;\n    * }\n    * ```\n    */\n    @Output()\n    public onTabItemSelected = new EventEmitter();\n\n    /**\n     * @hidden\n     */\n    @ViewChild('contentsContainer', { static: true })\n    public contentsContainer: ElementRef;\n\n    /**\n     * @hidden\n     */\n    @ViewChild('headerContainer', { static: true })\n    public headerContainer: ElementRef;\n\n    /**\n     * @hidden\n     */\n    @ViewChild('itemsContainer', { static: true })\n    public itemsContainer: ElementRef;\n\n    /**\n     * @hidden\n     */\n    @ViewChild('selectedIndicator')\n    public selectedIndicator: ElementRef;\n\n    /**\n    * @hidden\n    */\n    @ViewChild('tabsContainer', { static: true })\n    public tabsContainer: ElementRef;\n\n    /**\n     * @hidden\n     */\n    @ViewChild('viewPort', { static: true })\n    public viewPort: ElementRef;\n\n    /**\n     * Provides an observable collection of all `IgxTabItemComponent`s.\n     * ```typescript\n     * const tabItems = this.myTabComponent.viewTabs;\n     * ```\n     */\n    @ViewChildren(forwardRef(() => IgxTabItemComponent))\n    public viewTabs: QueryList<IgxTabItemComponent>;\n\n    /**\n     * Provides an observable collection of all `IgxTabItemComponent`s.\n     * First try to get them as content children if not available get them as view children.\n     * ```typescript\n     * const tabItems = this.myTabComponent.tabs;\n     * ```\n     */\n    public get tabs(): QueryList<IgxTabItemComponent> {\n        if (this.hasContentTabs) {\n            return this.contentTabs;\n        }\n        return this.viewTabs;\n    }\n\n    /**\n     *@hidden\n     */\n    public get hasContentTabs(): boolean {\n        return (this.contentTabs && this.contentTabs.length > 0);\n    }\n\n    /**\n     * @hidden\n     */\n    public calculatedWidth: number;\n\n    /**\n     * @hidden\n     */\n    public visibleItemsWidth: number;\n\n    /**\n     * @hidden\n     */\n    public offset = 0;\n\n    private _groupChanges$: Subscription;\n    private _selectedIndex = -1;\n    private _resizeObserver: ResizeObserver;\n\n    /**\n     * @hidden\n     */\n    @HostBinding('attr.class')\n    public get cssClass() {\n        const defaultStyle = `igx-tabs`;\n        const fixedStyle = `igx-tabs--fixed`;\n        const iconStyle = `igx-tabs--icons`;\n        const iconLabelFoundInGroups = this.groups.find((group) => group.icon != null && group.label != null);\n        const iconLabelFoundInTabs = this.contentTabs.find((tab) => tab.icon != null && tab.label != null);\n        let css;\n        switch (TabsType[this.tabsType.toUpperCase()]) {\n            case TabsType.FIXED: {\n                css = fixedStyle;\n                break;\n            }\n            default: {\n                css = defaultStyle;\n                break;\n            }\n        }\n\n        // Layout fix for items with icons\n        if (iconLabelFoundInGroups !== undefined || iconLabelFoundInTabs !== undefined) {\n            css = `${css} ${iconStyle}`;\n        }\n\n        return `${css} ${this.class}`;\n    }\n\n    /**\n     * @hidden\n     */\n    public scrollLeft(event): void {\n        this.scroll(false);\n    }\n\n    /**\n     * @hidden\n     */\n    public scrollRight(event): void {\n        this.scroll(true);\n    }\n\n    /**\n     * @hidden\n     */\n    public scrollElement(element: any, scrollRight: boolean): void {\n        const viewPortWidth = this.viewPort.nativeElement.offsetWidth;\n\n        this.offset = (scrollRight) ? element.offsetWidth + element.offsetLeft - viewPortWidth : element.offsetLeft;\n        this.itemsContainer.nativeElement.style.transform = `translate(${-this.offset}px)`;\n    }\n\n    /**\n     * Gets the selected `IgxTabItemComponent`.\n     * ```\n     * const selectedItem = this.myTabComponent.selectedTabItem;\n     * ```\n     */\n    get selectedTabItem(): IgxTabItemComponent {\n        if (this.tabs && this.selectedIndex !== undefined) {\n            return this.tabs.toArray()[this.selectedIndex];\n        }\n    }\n\n    constructor(private _element: ElementRef, private _ngZone: NgZone) {\n    }\n\n    /**\n     * @hidden\n     */\n    public ngAfterViewInit() {\n        if (this._selectedIndex === -1) {\n            this.tabs.forEach((t) => {\n                if (t.isSelected) {\n                    this._selectedIndex = t.index;\n                }\n            });\n        }\n\n        if (!this.hasContentTabs && (this.selectedIndex < 0 || this.selectedIndex >= this.groups.length)) {\n            this._selectedIndex = 0;\n        }\n\n        requestAnimationFrame(() => {\n            const newTab = this.tabs.toArray()[this._selectedIndex];\n            if (newTab) {\n                this.performSelection(newTab);\n            } else {\n                this.hideIndicator();\n            }\n        });\n\n        this._groupChanges$ = this.groups.changes.subscribe(() => {\n            this.resetSelectionOnCollectionChanged();\n        });\n\n        this._ngZone.runOutsideAngular(() => {\n            this._resizeObserver = new ResizeObserver(() => {\n                if (!this.hasContentTabs && this._selectedIndex >= 0 && this._selectedIndex < this.tabs.length) {\n                    const newTab = this.tabs.toArray()[this._selectedIndex];\n                    this.transformContentAnimation(newTab, 0);\n                }\n            });\n\n            this._resizeObserver.observe(this.tabsContainer.nativeElement);\n        });\n    }\n\n    /**\n     * @hidden\n     */\n    public ngOnDestroy(): void {\n        if (this._groupChanges$) {\n            this._groupChanges$.unsubscribe();\n        }\n\n        this._ngZone.runOutsideAngular(() => {\n            this._resizeObserver.disconnect();\n        });\n    }\n\n    private resetSelectionOnCollectionChanged(): void {\n        requestAnimationFrame(() => {\n            const currentTab = this.tabs.toArray()[this.selectedIndex];\n            if (currentTab) {\n                this.performSelectionChange(currentTab);\n            } else if (this.selectedIndex >= this.tabs.length) {\n                this.performSelectionChange(this.tabs.last);\n            } else {\n                this.hideIndicator();\n            }\n        });\n    }\n\n    private scroll(scrollRight: boolean): void {\n        const tabsArray = this.tabs.toArray();\n        for (const tab of tabsArray) {\n            const element = tab.nativeTabItem.nativeElement;\n            if (scrollRight) {\n                if (element.offsetWidth + element.offsetLeft > this.viewPort.nativeElement.offsetWidth + this.offset) {\n                    this.scrollElement(element, scrollRight);\n                    break;\n                }\n            } else {\n                if (element.offsetWidth + element.offsetLeft >= this.offset) {\n                    this.scrollElement(element, scrollRight);\n                    break;\n                }\n            }\n        }\n    }\n\n    /**\n     * @hidden\n     */\n    public performSelectionChange(newTab: IgxTabItemBase): void {\n        const oldTab = this.selectedTabItem;\n        if (oldTab) {\n            this.performDeselection(oldTab);\n        }\n        if (newTab) {\n            this.performSelection(newTab);\n        } else {\n            // if there is no new selected tab hide the selection indicator\n            this.hideIndicator();\n        }\n        this.selectedIndexChange.emit(this._selectedIndex);\n    }\n\n    private performDeselection(oldTab: IgxTabItemBase): void {\n        oldTab.setSelectedInternal(false);\n        const oldTabRelatedGroup = this.groups.toArray()[oldTab.index];\n        if (oldTabRelatedGroup) {\n            oldTabRelatedGroup.setSelectedInternal(false);\n        }\n        this._selectedIndex = -1;\n        this.onTabItemDeselected.emit({ tab: oldTab, group: oldTabRelatedGroup });\n    }\n\n    private performSelection(newTab: IgxTabItemBase): void {\n        newTab.setSelectedInternal(true);\n        this._selectedIndex = newTab.index;\n\n        let newTabRelatedGroup = null;\n        if (!this.hasContentTabs && this.groups) {\n            newTabRelatedGroup = this.groups.toArray()[newTab.index];\n            if (newTabRelatedGroup) {\n                newTabRelatedGroup.setSelectedInternal(true);\n            }\n        }\n\n        this.onTabItemSelected.emit({ tab: newTab, group: newTabRelatedGroup });\n\n        requestAnimationFrame(() => {\n            // bring the new selected tab into view if it is not\n            this.bringNewTabIntoView(newTab);\n            // animate the new selection indicator\n            this.transformIndicatorAnimation(newTab.nativeTabItem.nativeElement);\n            // animate the new tab's group content\n            if (!this.hasContentTabs) {\n                this.transformContentAnimation(newTab, 0.2);\n            }\n        });\n    }\n\n    private bringNewTabIntoView(newTab: IgxTabItemBase): void {\n        const tabNativeElement = newTab.nativeTabItem.nativeElement;\n\n        // Scroll left if there is need\n        if (tabNativeElement.offsetLeft < this.offset) {\n            this.scrollElement(tabNativeElement, false);\n        }\n\n        // Scroll right if there is need\n        const viewPortOffsetWidth = this.viewPort.nativeElement.offsetWidth;\n        const delta = (tabNativeElement.offsetLeft + tabNativeElement.offsetWidth) - (viewPortOffsetWidth + this.offset);\n\n        // Fix for IE 11, a difference is accumulated from the widths calculations\n        if (delta > 1)