UNPKG

@nebular/theme

Version:
282 lines 9.41 kB
/** * @license * Copyright Akveo. All Rights Reserved. * Licensed under the MIT License. See License.txt in the project root for license information. */ import { map, delay, filter } from 'rxjs/operators'; import { Component, Input, Output, EventEmitter, ContentChildren, QueryList, HostBinding, ChangeDetectorRef, } from '@angular/core'; import { ActivatedRoute } from '@angular/router'; import { convertToBoolProperty } from '../helpers'; /** * Specific tab container. * * ```ts * <nb-tab tabTitle="Users" * badgeText="99+" * badgeStatus="danger"> * <p>List of <strong>users</strong>.</p> * </nb-tab> ``` */ export class NbTabComponent { constructor() { this.activeValue = false; this.responsiveValue = false; this.disabledValue = false; /** * Badge status (adds specific styles): * 'primary', 'info', 'success', 'warning', 'danger' * @param {string} val */ this.badgeStatus = 'basic'; this.init = false; } /** * Use badge dot mode * @type {boolean} */ get badgeDot() { return this._badgeDot; } set badgeDot(val) { this._badgeDot = convertToBoolProperty(val); } /** * Item is disabled and cannot be opened. * @type {boolean} */ get disabled() { return this.disabledValue; } set disabled(val) { this.disabledValue = convertToBoolProperty(val); } /** * Show only icons when width is smaller than `tabs-icon-only-max-width` * @type {boolean} */ set responsive(val) { this.responsiveValue = convertToBoolProperty(val); } get responsive() { return this.responsiveValue; } /** * Specifies active tab * @returns {boolean} */ get active() { return this.activeValue; } set active(val) { this.activeValue = convertToBoolProperty(val); if (this.activeValue) { this.init = true; } } /** * Lazy load content before tab selection * TODO: rename, as lazy is by default, and this is more `instant load` * @param {boolean} val */ set lazyLoad(val) { this.init = convertToBoolProperty(val); } } NbTabComponent.decorators = [ { type: Component, args: [{ selector: 'nb-tab', template: ` <ng-container *ngIf="init"> <ng-content></ng-content> </ng-container> ` },] } ]; NbTabComponent.propDecorators = { tabTitle: [{ type: Input }], tabId: [{ type: Input }], badgeDot: [{ type: Input }], tabIcon: [{ type: Input }], disabled: [{ type: Input, args: ['disabled',] }, { type: HostBinding, args: ['class.disabled',] }], responsive: [{ type: Input }], route: [{ type: Input }], activeValue: [{ type: HostBinding, args: ['class.content-active',] }], active: [{ type: Input }], lazyLoad: [{ type: Input }], badgeText: [{ type: Input }], badgeStatus: [{ type: Input }], badgePosition: [{ type: Input }] }; // TODO: Combine tabset with route-tabset, so that we can: // - have similar interface // - easy to migrate from one to another // - can mix them both (route/content tab) /** * * Dynamic tabset component. * @stacked-example(Showcase, tabset/tabset-showcase.component) * * Basic tabset example * * ```html * <nb-tabset> * <nb-tab tabTitle="Simple Tab #1"> * Tab content 1 * </nb-tab> * <nb-tab tabTitle="Simple Tab #2"> * Tab content 2 * </nb-tab> * </nb-tabset> * ``` * * ### Installation * * Import `NbTabsetModule` to your feature module. * ```ts * @NgModule({ * imports: [ * // ... * NbTabsetModule, * ], * }) * export class PageModule { } * ``` * ### Usage * * It is also possible to set a badge to a particular tab: * @stacked-example(Tab With Badge, tabset/tabset-badge.component) * * and we can set it to full a width of a parent component * @stacked-example(Full Width, tabset/tabset-width.component) * * `tabIcon` should be used to add an icon to the tab. Icon can also be combined with title. * `responsive` tab property if set allows you to hide the title on smaller screens * (`tabs-icon-only-max-width` property) for better responsive behaviour. You can open the following example and make * your screen smaller - titles will be hidden in the last tabset in the list: * * @stacked-example(Icon, tabset/tabset-icon.component) * * It is also possible to disable a tab using `disabled` property: * @stacked-example(Disabled Tab, tabset/tabset-disabled.component) * * @styles * * tabset-background-color: * tabset-border-radius: * tabset-shadow: * tabset-tab-background-color: * tabset-tab-padding: * tabset-tab-text-color: * tabset-tab-text-font-family: * tabset-tab-text-font-size: * tabset-tab-text-font-weight: * tabset-tab-text-line-height: * tabset-tab-text-transform: * tabset-tab-underline-width: * tabset-tab-underline-color: * tabset-tab-active-background-color: * tabset-tab-active-text-color: * tabset-tab-active-underline-color: * tabset-tab-focus-background-color: * tabset-tab-focus-text-color: * tabset-tab-focus-underline-color: * tabset-tab-hover-background-color: * tabset-tab-hover-text-color: * tabset-tab-hover-underline-color: * tabset-tab-disabled-background-color: * tabset-tab-disabled-text-color: * tabset-tab-disabled-underline-color: * tabset-divider-color: * tabset-divider-style: * tabset-divider-width: * tabset-content-background-color: * tabset-content-padding: * tabset-content-text-color: * tabset-content-text-font-family: * tabset-content-text-font-size: * tabset-content-text-font-weight: * tabset-content-text-line-height: * tabset-scrollbar-color: * tabset-scrollbar-background-color: * tabset-scrollbar-width: * tabset-tab-text-hide-breakpoint: */ export class NbTabsetComponent { constructor(route, changeDetectorRef) { this.route = route; this.changeDetectorRef = changeDetectorRef; this.fullWidthValue = false; /** * Emits when tab is selected * @type EventEmitter<any> */ this.changeTab = new EventEmitter(); } /** * Take full width of a parent * @param {boolean} val */ set fullWidth(val) { this.fullWidthValue = convertToBoolProperty(val); } // TODO: refactoring this component, avoid change detection loop ngAfterContentInit() { this.route.params .pipe(map((params) => this.tabs.find((tab) => this.routeParam ? tab.route === params[this.routeParam] : tab.active)), delay(0), map((tab) => tab || this.tabs.first), filter((tab) => !!tab)) .subscribe((tabToSelect) => { this.selectTab(tabToSelect); this.changeDetectorRef.markForCheck(); }); } // TODO: navigate to routeParam selectTab(selectedTab) { if (!selectedTab.disabled) { this.tabs.forEach(tab => tab.active = tab === selectedTab); this.changeTab.emit(selectedTab); } } } NbTabsetComponent.decorators = [ { type: Component, args: [{ selector: 'nb-tabset', template: ` <ul class="tabset"> <li *ngFor="let tab of tabs" (click)="selectTab(tab)" (keyup.space)="selectTab(tab)" (keyup.enter)="selectTab(tab)" [class.responsive]="tab.responsive" [class.active]="tab.active" [class.disabled]="tab.disabled" [attr.tabindex]="tab.disabled ? -1 : 0" class="tab"> <a href (click)="$event.preventDefault()" tabindex="-1" class="tab-link"> <nb-icon *ngIf="tab.tabIcon" [config]="tab.tabIcon"></nb-icon> <span *ngIf="tab.tabTitle" class="tab-text">{{ tab.tabTitle }}</span> </a> <nb-badge *ngIf="tab.badgeText || tab.badgeDot" [text]="tab.badgeText" [dotMode]="tab.badgeDot" [status]="tab.badgeStatus" [position]="tab.badgePosition"> </nb-badge> </li> </ul> <ng-content select="nb-tab"></ng-content> `, styles: [":host{display:block}:host.full-width .tabset{justify-content:space-around}:host ::ng-deep nb-tab{flex:1;-ms-flex:1 1 auto;overflow:auto;display:none}:host ::ng-deep nb-tab.content-active{display:block}:host .tabset{display:flex;flex-direction:row;list-style-type:none;margin:0;padding:0}:host .tabset .tab{margin-bottom:-1px;text-align:center;position:relative}:host .tabset .tab.active a::before{display:block}:host .tabset .tab a{display:flex;position:relative;text-decoration:none}:host .tabset .tab a::before{position:absolute;content:'';width:100%;border-radius:3px;bottom:-2px;left:0}:host .tabset .tab a nb-icon{vertical-align:middle}[dir=ltr] :host .tabset .tab a nb-icon+span{margin-left:.5rem}[dir=rtl] :host .tabset .tab a nb-icon+span{margin-right:.5rem}\n"] },] } ]; NbTabsetComponent.ctorParameters = () => [ { type: ActivatedRoute }, { type: ChangeDetectorRef } ]; NbTabsetComponent.propDecorators = { tabs: [{ type: ContentChildren, args: [NbTabComponent,] }], fullWidthValue: [{ type: HostBinding, args: ['class.full-width',] }], fullWidth: [{ type: Input }], routeParam: [{ type: Input }], changeTab: [{ type: Output }] }; //# sourceMappingURL=tabset.component.js.map