UNPKG

@nova-ui/bits

Version:

SolarWinds Nova Framework

182 lines 38.5 kB
// © 2022 SolarWinds Worldwide, LLC. All rights reserved. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. import { ChangeDetectionStrategy, ChangeDetectorRef, Component, ContentChildren, ElementRef, EventEmitter, HostBinding, Input, NgZone, Output, QueryList, ViewChild, } from "@angular/core"; import { TabHeadingComponent } from "../tab-heading/tab-heading.component"; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; import * as i2 from "../../icon/icon.component"; // <example-url>./../examples/index.html#/tabgroup</example-url> export class TabHeadingGroupComponent { get isVertical() { return this.vertical; } constructor(el, changeDetectorRef, ngZone) { this.el = el; this.changeDetectorRef = changeDetectorRef; this.ngZone = ngZone; /** * Emits id of selected tab */ this.selected = new EventEmitter(); this.leftTraverseEnabled = true; this.rightTraverseEnabled = false; this.hasTraverse = false; this._traverseButtonsWidth = 60; this._traverseStepSize = 50; this._tabSelectedSubscriptions = []; } ngAfterViewInit() { // Observing the size of the component to check traverse this._ro = new ResizeObserver((entries) => entries.forEach(() => this.checkTraverse())); this.ngZone.runOutsideAngular(() => { this._ro.observe(this.resizableArea.nativeElement); this._ro.observe(this.el.nativeElement); }); // Making the first tab in group active by default this.setActiveTab(); this.subscribeToSelection(); this._changesSubscription = this._tabs.changes.subscribe((changedTabs) => { this.setActiveTab(); this._tabSelectedSubscriptions.forEach((sub) => sub.unsubscribe()); this._tabSelectedSubscriptions = []; this.subscribeToSelection(); }); } setActiveTab() { if (this._tabs.length && !this.getActiveTab()) { this._tabs.first.active = true; this.selected.emit(this._tabs.first.tabId); } } getActiveTab() { return this._tabs.filter((tab) => tab.active)[0]; } checkTraverse() { this.hasTraverse = this.allowTraverse(); this.changeDetectorRef.detectChanges(); if (!this.hasTraverse) { this.setNewShift("0px"); } } allowTraverse() { const holderSize = this.getElementSize("nui-tab-headings__holder"); const contentSize = this.getElementSize("nui-tab-headings__container"); if (this.vertical) { return false; } return holderSize + this._traverseButtonsWidth <= contentSize; } traverseRight() { const margin = this.getCurrentShift(); if (this.isTraverseRightAllowed(margin)) { const traverseStep = Math.abs(this.getNumberFromPixels(margin)) < this._traverseStepSize ? Math.abs(this.getNumberFromPixels(margin)) : this._traverseStepSize; this.setNewShift(this.addPixels(margin, traverseStep)); this.rightTraverseEnabled = this.isTraverseRightAllowed(this.addPixels(margin, traverseStep)); this.leftTraverseEnabled = this.isTraverseLeftAllowed(this.addPixels(margin, traverseStep)); } } traverseLeft() { const margin = this.getCurrentShift(); if (this.isTraverseLeftAllowed(margin)) { const tabsSize = this.getElementSize("nui-tab-headings__container"); const tabHolderSize = this.getElementSize("nui-tab-headings__holder"); const maxAllowedMargin = Math.abs(tabsSize - tabHolderSize + this._traverseButtonsWidth); const leftMarginValue = Math.abs(this.getNumberFromPixels(margin)); const traverseStep = Math.min(maxAllowedMargin - leftMarginValue, this._traverseStepSize); this.setNewShift(this.addPixels(margin, -traverseStep)); this.rightTraverseEnabled = this.isTraverseRightAllowed(this.addPixels(margin, -traverseStep)); this.leftTraverseEnabled = this.isTraverseLeftAllowed(this.addPixels(margin, -traverseStep)); } } // Subscribing to all tabs and their 'selected' event. Once the event has been fired the origin tab becomes active. subscribeToSelection() { this._tabs.forEach((tab) => { this._tabSelectedSubscriptions.push(tab.selected.subscribe((currentTab) => { if (!currentTab.active && !currentTab.disabled) { // Making all elements in array inactive to make than current one active this._tabs.forEach((tabHeading) => { tabHeading.active = false; }); currentTab.active = true; this.changeDetectorRef.markForCheck(); this.changeDetectorRef.detectChanges(); this.selected.emit(currentTab.tabId); } })); }); } isTraverseLeftAllowed(leftMargin) { const tabsSize = this.getElementSize("nui-tab-headings__container"); const tabHolderSize = this.getElementSize("nui-tab-headings__holder"); const maxAllowedMargin = Math.abs(tabsSize - tabHolderSize + this._traverseButtonsWidth); const margin = Math.abs(this.getNumberFromPixels(leftMargin)); return margin < maxAllowedMargin; } isTraverseRightAllowed(margin) { return this.getNumberFromPixels(margin) < 0; } getElementSize(selector) { return this.el.nativeElement.querySelector("." + selector).offsetWidth; } getNumberFromPixels(pixels) { return pixels.indexOf("px") ? Number(pixels.substring(0, pixels.indexOf("px"))) : 0; } getCurrentShift() { return this.el.nativeElement.querySelector(".nui-tab-headings__container").style.marginLeft; } setNewShift(newShift) { this.el.nativeElement.querySelector(".nui-tab-headings__container").style.marginTop = "0px"; this.el.nativeElement.querySelector(".nui-tab-headings__container").style.marginLeft = newShift; } addPixels(currentValue, increment) { const value = this.getNumberFromPixels(currentValue); return value + increment + "px"; } ngOnDestroy() { this._changesSubscription.unsubscribe(); this._tabSelectedSubscriptions.forEach((sub) => sub.unsubscribe()); this._ro.disconnect(); } static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TabHeadingGroupComponent, deps: [{ token: i0.ElementRef }, { token: i0.ChangeDetectorRef }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); } static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: TabHeadingGroupComponent, selector: "nui-tab-heading-group", inputs: { vertical: "vertical" }, outputs: { selected: "selected" }, host: { attributes: { "role": "tablist" }, properties: { "class.vertical": "this.isVertical" } }, queries: [{ propertyName: "_tabs", predicate: TabHeadingComponent }], viewQueries: [{ propertyName: "resizableArea", first: true, predicate: ["resizableArea"], descendants: true }], ngImport: i0, template: "<div class=\"nui-tab-headings__holder\" (click)=\"$event.preventDefault()\">\n <div\n class=\"btn-caret-left\"\n [class.disabled]=\"!rightTraverseEnabled\"\n *ngIf=\"hasTraverse\"\n (click)=\"traverseRight()\"\n >\n <nui-icon\n iconSize=\"default\"\n icon=\"caret-left\"\n [iconColor]=\"!rightTraverseEnabled ? 'disabled-gray' : 'gray'\"\n ></nui-icon>\n </div>\n <div class=\"nui-tab-headings\">\n <div\n class=\"nui-tab-headings__container\"\n #resizableArea\n [class.flex-column]=\"vertical\"\n >\n <ng-content></ng-content>\n </div>\n </div>\n <div\n class=\"btn-caret-right\"\n [class.disabled]=\"!leftTraverseEnabled\"\n *ngIf=\"hasTraverse\"\n (click)=\"traverseLeft()\"\n >\n <nui-icon\n iconSize=\"default\"\n icon=\"caret-right\"\n [iconColor]=\"!leftTraverseEnabled ? 'disabled-gray' : 'gray'\"\n ></nui-icon>\n </div>\n</div>\n", styles: [":host{display:flex;height:100%}:host .nui-tab-headings{display:flex;justify-content:flex-start;margin-bottom:0;overflow:hidden}:host .nui-tab-headings__holder{display:flex;border-bottom:1px solid #d9d9d9;border-bottom:1px solid var(--nui-color-line-default, #d9d9d9);width:100%}:host .nui-tab-headings__holder .btn-caret-right{align-items:center;justify-content:center;display:flex;flex:1;background:var(--nui-color-bg-content,#fff);min-width:40px;max-width:40px;min-height:40px;z-index:0;border-bottom:none;cursor:pointer;border-left:1px solid;border-left-color:var(--nui-color-line-default,#d9d9d9);box-shadow:-6px 0 5px -5px var(--nui-shadow-color, rgba(17, 17, 17, .3))}:host .nui-tab-headings__holder .btn-caret-right.disabled{cursor:auto}:host .nui-tab-headings__holder .btn-caret-left{border-right:1px solid;border-right-color:var(--nui-color-line-default,#d9d9d9);box-shadow:6px 0 5px -5px var(--nui-shadow-color, rgba(17, 17, 17, .3));align-items:center;justify-content:center;display:flex;flex:1;background:var(--nui-color-bg-content,#fff);min-width:40px;max-width:40px;min-height:40px;z-index:0;border-bottom:none;cursor:pointer}:host .nui-tab-headings__holder .btn-caret-left.disabled{cursor:auto}:host .nui-tab-headings__container{display:flex}.caret-button{align-items:center;justify-content:center;display:flex;flex:1;background:var(--nui-color-bg-content,#fff);min-width:40px;max-width:40px;min-height:40px;z-index:0;border-bottom:none;cursor:pointer}.caret-button.disabled{cursor:auto}:host-context(nui-tab-heading-group.vertical) .nui-tab-headings__holder{flex-direction:column;border-right:1px solid;border-right-color:var(--nui-color-line-default,#d9d9d9);border-bottom:0;background:var(--nui-color-bg-secondary,#fafafa)}:host-context(nui-tab-heading-group.vertical) .nui-tab-headings__container{width:100%}\n"], dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.IconComponent, selector: "nui-icon", inputs: ["iconColor", "brushType", "iconHoverColor", "iconSize", "cssClass", "fillContainer", "status", "childStatus", "icon", "counter"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); } } i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: TabHeadingGroupComponent, decorators: [{ type: Component, args: [{ selector: "nui-tab-heading-group", changeDetection: ChangeDetectionStrategy.OnPush, host: { role: "tablist" }, template: "<div class=\"nui-tab-headings__holder\" (click)=\"$event.preventDefault()\">\n <div\n class=\"btn-caret-left\"\n [class.disabled]=\"!rightTraverseEnabled\"\n *ngIf=\"hasTraverse\"\n (click)=\"traverseRight()\"\n >\n <nui-icon\n iconSize=\"default\"\n icon=\"caret-left\"\n [iconColor]=\"!rightTraverseEnabled ? 'disabled-gray' : 'gray'\"\n ></nui-icon>\n </div>\n <div class=\"nui-tab-headings\">\n <div\n class=\"nui-tab-headings__container\"\n #resizableArea\n [class.flex-column]=\"vertical\"\n >\n <ng-content></ng-content>\n </div>\n </div>\n <div\n class=\"btn-caret-right\"\n [class.disabled]=\"!leftTraverseEnabled\"\n *ngIf=\"hasTraverse\"\n (click)=\"traverseLeft()\"\n >\n <nui-icon\n iconSize=\"default\"\n icon=\"caret-right\"\n [iconColor]=\"!leftTraverseEnabled ? 'disabled-gray' : 'gray'\"\n ></nui-icon>\n </div>\n</div>\n", styles: [":host{display:flex;height:100%}:host .nui-tab-headings{display:flex;justify-content:flex-start;margin-bottom:0;overflow:hidden}:host .nui-tab-headings__holder{display:flex;border-bottom:1px solid #d9d9d9;border-bottom:1px solid var(--nui-color-line-default, #d9d9d9);width:100%}:host .nui-tab-headings__holder .btn-caret-right{align-items:center;justify-content:center;display:flex;flex:1;background:var(--nui-color-bg-content,#fff);min-width:40px;max-width:40px;min-height:40px;z-index:0;border-bottom:none;cursor:pointer;border-left:1px solid;border-left-color:var(--nui-color-line-default,#d9d9d9);box-shadow:-6px 0 5px -5px var(--nui-shadow-color, rgba(17, 17, 17, .3))}:host .nui-tab-headings__holder .btn-caret-right.disabled{cursor:auto}:host .nui-tab-headings__holder .btn-caret-left{border-right:1px solid;border-right-color:var(--nui-color-line-default,#d9d9d9);box-shadow:6px 0 5px -5px var(--nui-shadow-color, rgba(17, 17, 17, .3));align-items:center;justify-content:center;display:flex;flex:1;background:var(--nui-color-bg-content,#fff);min-width:40px;max-width:40px;min-height:40px;z-index:0;border-bottom:none;cursor:pointer}:host .nui-tab-headings__holder .btn-caret-left.disabled{cursor:auto}:host .nui-tab-headings__container{display:flex}.caret-button{align-items:center;justify-content:center;display:flex;flex:1;background:var(--nui-color-bg-content,#fff);min-width:40px;max-width:40px;min-height:40px;z-index:0;border-bottom:none;cursor:pointer}.caret-button.disabled{cursor:auto}:host-context(nui-tab-heading-group.vertical) .nui-tab-headings__holder{flex-direction:column;border-right:1px solid;border-right-color:var(--nui-color-line-default,#d9d9d9);border-bottom:0;background:var(--nui-color-bg-secondary,#fafafa)}:host-context(nui-tab-heading-group.vertical) .nui-tab-headings__container{width:100%}\n"] }] }], ctorParameters: () => [{ type: i0.ElementRef }, { type: i0.ChangeDetectorRef }, { type: i0.NgZone }], propDecorators: { _tabs: [{ type: ContentChildren, args: [TabHeadingComponent] }], resizableArea: [{ type: ViewChild, args: ["resizableArea"] }], vertical: [{ type: Input }], selected: [{ type: Output }], isVertical: [{ type: HostBinding, args: ["class.vertical"] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab-heading-group.component.js","sourceRoot":"","sources":["../../../../../src/lib/tabgroup/tab-heading-group/tab-heading-group.component.ts","../../../../../src/lib/tabgroup/tab-heading-group/tab-heading-group.component.html"],"names":[],"mappings":"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","sourcesContent":["// © 2022 SolarWinds Worldwide, LLC. All rights reserved.\n//\n// Permission is hereby granted, free of charge, to any person obtaining a copy\n//  of this software and associated documentation files (the \"Software\"), to\n//  deal in the Software without restriction, including without limitation the\n//  rights to use, copy, modify, merge, publish, distribute, sublicense, and/or\n//  sell copies of the Software, and to permit persons to whom the Software is\n//  furnished to do so, subject to the following conditions:\n//\n// The above copyright notice and this permission notice shall be included in\n//  all copies or substantial portions of the Software.\n//\n// THE SOFTWARE IS PROVIDED \"AS IS\", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR\n//  IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,\n//  FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE\n//  AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER\n//  LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,\n//  OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN\n//  THE SOFTWARE.\n\nimport {\n    AfterViewInit,\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ContentChildren,\n    ElementRef,\n    EventEmitter,\n    HostBinding,\n    Input,\n    NgZone,\n    OnDestroy,\n    Output,\n    QueryList,\n    ViewChild,\n} from \"@angular/core\";\nimport { Subscription } from \"rxjs\";\n\nimport { TabHeadingComponent } from \"../tab-heading/tab-heading.component\";\n\n// <example-url>./../examples/index.html#/tabgroup</example-url>\n@Component({\n    selector: \"nui-tab-heading-group\",\n    templateUrl: \"./tab-heading-group.component.html\",\n    styleUrls: [\"./tab-heading-group.component.less\"],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n    host: { role: \"tablist\" },\n})\nexport class TabHeadingGroupComponent implements OnDestroy, AfterViewInit {\n    @ContentChildren(TabHeadingComponent) _tabs: QueryList<TabHeadingComponent>;\n\n    @ViewChild(\"resizableArea\") resizableArea: ElementRef;\n\n    /** If true tabs will be placed vertically */\n    @Input() public vertical: boolean;\n\n    /**\n     * Emits id of selected tab\n     */\n    @Output() selected: EventEmitter<string> = new EventEmitter();\n\n    @HostBinding(\"class.vertical\") get isVertical(): boolean {\n        return this.vertical;\n    }\n\n    public leftTraverseEnabled = true;\n    public rightTraverseEnabled = false;\n    public hasTraverse = false;\n\n    private _traverseButtonsWidth = 60;\n    private _traverseStepSize = 50;\n    private _ro: ResizeObserver;\n    private _tabSelectedSubscriptions: Subscription[] = [];\n    private _changesSubscription: Subscription;\n\n    constructor(\n        private el: ElementRef,\n        private changeDetectorRef: ChangeDetectorRef,\n        private ngZone: NgZone\n    ) {}\n\n    public ngAfterViewInit(): void {\n        // Observing the size of the component to check traverse\n        this._ro = new ResizeObserver((entries) =>\n            entries.forEach(() => this.checkTraverse())\n        );\n        this.ngZone.runOutsideAngular(() => {\n            this._ro.observe(this.resizableArea.nativeElement);\n            this._ro.observe(this.el.nativeElement);\n        });\n\n        // Making the first tab in group active by default\n        this.setActiveTab();\n        this.subscribeToSelection();\n\n        this._changesSubscription = this._tabs.changes.subscribe(\n            (changedTabs: any) => {\n                this.setActiveTab();\n                this._tabSelectedSubscriptions.forEach((sub) =>\n                    sub.unsubscribe()\n                );\n                this._tabSelectedSubscriptions = [];\n                this.subscribeToSelection();\n            }\n        );\n    }\n\n    public setActiveTab(): void {\n        if (this._tabs.length && !this.getActiveTab()) {\n            this._tabs.first.active = true;\n            this.selected.emit(this._tabs.first.tabId);\n        }\n    }\n\n    public getActiveTab(): TabHeadingComponent {\n        return this._tabs.filter((tab: TabHeadingComponent) => tab.active)[0];\n    }\n\n    public checkTraverse(): void {\n        this.hasTraverse = this.allowTraverse();\n        this.changeDetectorRef.detectChanges();\n        if (!this.hasTraverse) {\n            this.setNewShift(\"0px\");\n        }\n    }\n\n    public allowTraverse(): boolean {\n        const holderSize = this.getElementSize(\"nui-tab-headings__holder\");\n        const contentSize = this.getElementSize(\"nui-tab-headings__container\");\n        if (this.vertical) {\n            return false;\n        }\n        return holderSize + this._traverseButtonsWidth <= contentSize;\n    }\n\n    public traverseRight(): void {\n        const margin = this.getCurrentShift();\n        if (this.isTraverseRightAllowed(margin)) {\n            const traverseStep =\n                Math.abs(this.getNumberFromPixels(margin)) <\n                this._traverseStepSize\n                    ? Math.abs(this.getNumberFromPixels(margin))\n                    : this._traverseStepSize;\n            this.setNewShift(this.addPixels(margin, traverseStep));\n            this.rightTraverseEnabled = this.isTraverseRightAllowed(\n                this.addPixels(margin, traverseStep)\n            );\n            this.leftTraverseEnabled = this.isTraverseLeftAllowed(\n                this.addPixels(margin, traverseStep)\n            );\n        }\n    }\n\n    public traverseLeft(): void {\n        const margin = this.getCurrentShift();\n        if (this.isTraverseLeftAllowed(margin)) {\n            const tabsSize = this.getElementSize(\"nui-tab-headings__container\");\n            const tabHolderSize = this.getElementSize(\n                \"nui-tab-headings__holder\"\n            );\n            const maxAllowedMargin = Math.abs(\n                tabsSize - tabHolderSize + this._traverseButtonsWidth\n            );\n            const leftMarginValue = Math.abs(this.getNumberFromPixels(margin));\n            const traverseStep = Math.min(\n                maxAllowedMargin - leftMarginValue,\n                this._traverseStepSize\n            );\n            this.setNewShift(this.addPixels(margin, -traverseStep));\n            this.rightTraverseEnabled = this.isTraverseRightAllowed(\n                this.addPixels(margin, -traverseStep)\n            );\n            this.leftTraverseEnabled = this.isTraverseLeftAllowed(\n                this.addPixels(margin, -traverseStep)\n            );\n        }\n    }\n\n    // Subscribing to all tabs and their 'selected' event. Once the event has been fired the origin tab becomes active.\n    private subscribeToSelection() {\n        this._tabs.forEach((tab: TabHeadingComponent) => {\n            this._tabSelectedSubscriptions.push(\n                tab.selected.subscribe((currentTab: TabHeadingComponent) => {\n                    if (!currentTab.active && !currentTab.disabled) {\n                        // Making all elements in array inactive to make than current one active\n                        this._tabs.forEach(\n                            (tabHeading: TabHeadingComponent) => {\n                                tabHeading.active = false;\n                            }\n                        );\n                        currentTab.active = true;\n                        this.changeDetectorRef.markForCheck();\n                        this.changeDetectorRef.detectChanges();\n                        this.selected.emit(currentTab.tabId);\n                    }\n                })\n            );\n        });\n    }\n\n    private isTraverseLeftAllowed(leftMargin: string): boolean {\n        const tabsSize = this.getElementSize(\"nui-tab-headings__container\");\n        const tabHolderSize = this.getElementSize(\"nui-tab-headings__holder\");\n        const maxAllowedMargin = Math.abs(\n            tabsSize - tabHolderSize + this._traverseButtonsWidth\n        );\n\n        const margin = Math.abs(this.getNumberFromPixels(leftMargin));\n        return margin < maxAllowedMargin;\n    }\n\n    private isTraverseRightAllowed(margin: string): boolean {\n        return this.getNumberFromPixels(margin) < 0;\n    }\n\n    private getElementSize(selector: string): number {\n        return this.el.nativeElement.querySelector(\".\" + selector).offsetWidth;\n    }\n\n    private getNumberFromPixels(pixels: string): number {\n        return pixels.indexOf(\"px\")\n            ? Number(pixels.substring(0, pixels.indexOf(\"px\")))\n            : 0;\n    }\n\n    private getCurrentShift(): string {\n        return this.el.nativeElement.querySelector(\n            \".nui-tab-headings__container\"\n        ).style.marginLeft;\n    }\n\n    private setNewShift(newShift: string): void {\n        this.el.nativeElement.querySelector(\n            \".nui-tab-headings__container\"\n        ).style.marginTop = \"0px\";\n        this.el.nativeElement.querySelector(\n            \".nui-tab-headings__container\"\n        ).style.marginLeft = newShift;\n    }\n\n    private addPixels(currentValue: string, increment: number): string {\n        const value = this.getNumberFromPixels(currentValue);\n        return value + increment + \"px\";\n    }\n\n    public ngOnDestroy(): void {\n        this._changesSubscription.unsubscribe();\n        this._tabSelectedSubscriptions.forEach((sub) => sub.unsubscribe());\n        this._ro.disconnect();\n    }\n}\n","<div class=\"nui-tab-headings__holder\" (click)=\"$event.preventDefault()\">\n    <div\n        class=\"btn-caret-left\"\n        [class.disabled]=\"!rightTraverseEnabled\"\n        *ngIf=\"hasTraverse\"\n        (click)=\"traverseRight()\"\n    >\n        <nui-icon\n            iconSize=\"default\"\n            icon=\"caret-left\"\n            [iconColor]=\"!rightTraverseEnabled ? 'disabled-gray' : 'gray'\"\n        ></nui-icon>\n    </div>\n    <div class=\"nui-tab-headings\">\n        <div\n            class=\"nui-tab-headings__container\"\n            #resizableArea\n            [class.flex-column]=\"vertical\"\n        >\n            <ng-content></ng-content>\n        </div>\n    </div>\n    <div\n        class=\"btn-caret-right\"\n        [class.disabled]=\"!leftTraverseEnabled\"\n        *ngIf=\"hasTraverse\"\n        (click)=\"traverseLeft()\"\n    >\n        <nui-icon\n            iconSize=\"default\"\n            icon=\"caret-right\"\n            [iconColor]=\"!leftTraverseEnabled ? 'disabled-gray' : 'gray'\"\n        ></nui-icon>\n    </div>\n</div>\n"]}