UNPKG

@hxui/angular

Version:

An Angular library based on the [HXUI design system](https://hxui.io).

272 lines (269 loc) 24.7 kB
import { Component, ContentChildren, forwardRef, Input, QueryList, } from '@angular/core'; import { TabDirective } from './tab.directive'; import { TabsetConfig } from './tabset.config'; import * as i0 from "@angular/core"; import * as i1 from "./tabset.config"; import * as i2 from "@angular/common"; import * as i3 from "./ng-transclude.directive"; export class TabsetComponent { constructor(config) { /** if true tabs will be placed vertically */ this.vertical = false; /** if true tabs fill the container and have a consistent width */ this.justified = false; this.hasInfo = false; this.stickyHeader = false; this.stickyHeaderOffset = 0; this.tag = 'link'; this.changeFn = async () => true; this.tabs = []; Object.assign(this, config); } /** navigation context class: 'tabs' or 'pills' */ get type() { return this._type; } set type(value) { this._type = value; } ngAfterContentInit() { // get all active tabs const activeTabs = this._tabList.filter(tab => tab.active); // if there is no active tab set, activate the first if (activeTabs.length === 0) { this._selectTab(this._tabList.last); } } _selectTab(tab) { // deactivate all tabs this._tabList.toArray().forEach(tab => (tab.active = false)); // activate the tab the user has clicked on. tab.active = true; } selectTab(tab) { this.changeFn().then(res => !!res && this._selectTab(tab)); } ngOnDestroy() { this.isDestroyed = true; } addTab(tab) { this.tabs.push(tab); tab.active = this.tabs.length === 1 && tab.active !== false; } removeTab(tab, options = { reselect: true, emit: true }) { const index = this.tabs.indexOf(tab); if (index === -1 || this.isDestroyed) { return; } // Select a new tab if the tab to be removed is selected and not destroyed if (options.reselect && tab.active && this.hasAvailableTabs(index)) { const newActiveIndex = this.getClosestTabIndex(index); this.tabs[newActiveIndex].active = true; } if (options.emit) { tab.removed.emit(tab); } this.tabs.splice(index, 1); if (tab.elementRef.nativeElement && tab.elementRef.nativeElement.remove) { tab.elementRef.nativeElement.remove(); } } getStickyHeaderPosition() { return this.stickyHeader ? 'sticky' : 'relative'; } getClosestTabIndex(index) { const tabsLength = this.tabs.length; if (!tabsLength) { return -1; } for (let step = 1; step <= tabsLength; step += 1) { const prevIndex = index - step; const nextIndex = index + step; if (this.tabs[prevIndex] && !this.tabs[prevIndex].disabled) { return prevIndex; } if (this.tabs[nextIndex] && !this.tabs[nextIndex].disabled) { return nextIndex; } } return -1; } hasAvailableTabs(index) { const tabsLength = this.tabs.length; if (!tabsLength) { return false; } for (let i = 0; i < tabsLength; i += 1) { if (!this.tabs[i].disabled && i !== index) { return true; } } return false; } } TabsetComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: TabsetComponent, deps: [{ token: i1.TabsetConfig }], target: i0.ɵɵFactoryTarget.Component }); TabsetComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.11", type: TabsetComponent, selector: "hx-tabset", inputs: { vertical: "vertical", justified: "justified", hasInfo: "hasInfo", type: "type", contentCustomClass: "contentCustomClass", stickyHeader: "stickyHeader", stickyHeaderOffset: "stickyHeaderOffset", tag: "tag", changeFn: "changeFn" }, host: { classAttribute: "hx-tab-container" }, queries: [{ propertyName: "_tabList", predicate: i0.forwardRef(function () { return TabDirective; }) }], ngImport: i0, template: ` <ul class="hx-nav hx-nav-{{ type }}" [ngStyle]="{ position: getStickyHeaderPosition(), 'top.rem': stickyHeaderOffset }" [class.is-vertical]="vertical" [class.is-justified]="justified" [class.has-info]="hasInfo" > <li *ngFor="let tab of tabs" [ngClass]="['hx-nav-item', tab.customClass || '']" [class.is-active]="!!tab?.active" [class.is-disabled]="!!tab?.disabled" [class.is-button]="tag === 'button'" > <button *ngIf="tag === 'button'" class="hx-nav-link" [class.is-active]="!!tab?.active" [class.is-disabled]="!!tab?.disabled" [attr.disabled]="!!tab?.disabled ? '' : null" (click)="selectTab(tab)" > <span [ngTransclude]="tab.headingRef">{{ tab.heading }}</span> <span *ngIf="tab.removable"> <span (click)="removeTab(tab)" class="icon close-outline is-small" ></span> </span> </button> <a *ngIf="tag === 'link'" class="hx-nav-link" [class.is-active]="!!tab?.active" [class.is-disabled]="!!tab?.disabled" [attr.disabled]="!!tab?.disabled ? '' : null" (click)="selectTab(tab)" > <span [ngTransclude]="tab.headingRef">{{ tab.heading }}</span> <span *ngIf="tab.removable"> <span (click)="removeTab(tab)" class="icon close-outline is-small" ></span> </span> </a> </li> </ul> <div class="hx-tab-content {{ contentCustomClass }}"> <ng-content></ng-content> </div> `, isInline: true, styles: [":host,ul.hx-nav{background-color:inherit}button.hx-nav-link{border-width:0;border-bottom-width:1px;border-color:transparent;background-color:transparent;line-height:1.5;cursor:pointer}:where(.is-justified) .is-button.hx-nav-item{display:flex}:where(.is-justified .is-button) button.hx-nav-link{display:flex}\n"], directives: [{ type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i3.NgTranscludeDirective, selector: "[ngTransclude]", inputs: ["ngTransclude"] }] }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.11", ngImport: i0, type: TabsetComponent, decorators: [{ type: Component, args: [{ selector: 'hx-tabset', host: { class: 'hx-tab-container', }, template: ` <ul class="hx-nav hx-nav-{{ type }}" [ngStyle]="{ position: getStickyHeaderPosition(), 'top.rem': stickyHeaderOffset }" [class.is-vertical]="vertical" [class.is-justified]="justified" [class.has-info]="hasInfo" > <li *ngFor="let tab of tabs" [ngClass]="['hx-nav-item', tab.customClass || '']" [class.is-active]="!!tab?.active" [class.is-disabled]="!!tab?.disabled" [class.is-button]="tag === 'button'" > <button *ngIf="tag === 'button'" class="hx-nav-link" [class.is-active]="!!tab?.active" [class.is-disabled]="!!tab?.disabled" [attr.disabled]="!!tab?.disabled ? '' : null" (click)="selectTab(tab)" > <span [ngTransclude]="tab.headingRef">{{ tab.heading }}</span> <span *ngIf="tab.removable"> <span (click)="removeTab(tab)" class="icon close-outline is-small" ></span> </span> </button> <a *ngIf="tag === 'link'" class="hx-nav-link" [class.is-active]="!!tab?.active" [class.is-disabled]="!!tab?.disabled" [attr.disabled]="!!tab?.disabled ? '' : null" (click)="selectTab(tab)" > <span [ngTransclude]="tab.headingRef">{{ tab.heading }}</span> <span *ngIf="tab.removable"> <span (click)="removeTab(tab)" class="icon close-outline is-small" ></span> </span> </a> </li> </ul> <div class="hx-tab-content {{ contentCustomClass }}"> <ng-content></ng-content> </div> `, styles: [ ` :host, ul.hx-nav { background-color: inherit; } button.hx-nav-link { border-width: 0; border-bottom-width: 1px; border-color: transparent; background-color: transparent; line-height: 1.5; cursor: pointer; } :where(.is-justified) .is-button.hx-nav-item { display: flex; } :where(.is-justified .is-button) button.hx-nav-link { display: flex; } `, ], }] }], ctorParameters: function () { return [{ type: i1.TabsetConfig }]; }, propDecorators: { vertical: [{ type: Input }], justified: [{ type: Input }], hasInfo: [{ type: Input }], type: [{ type: Input }], contentCustomClass: [{ type: Input }], stickyHeader: [{ type: Input }], stickyHeaderOffset: [{ type: Input }], tag: [{ type: Input }], changeFn: [{ type: Input }], _tabList: [{ type: ContentChildren, args: [forwardRef(() => TabDirective)] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabset.component.js","sourceRoot":"","sources":["../../../../../projects/hx-ui/src/lib/tabs/tabset.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\r\n  AfterContentInit,\r\n  Component,\r\n  ContentChildren,\r\n  forwardRef,\r\n  Input,\r\n  OnDestroy,\r\n  QueryList,\r\n} from '@angular/core';\r\nimport { TabDirective } from './tab.directive';\r\nimport { TabsetConfig } from './tabset.config';\r\n\r\n@Component({\r\n  selector: 'hx-tabset',\r\n  host: {\r\n    class: 'hx-tab-container',\r\n  },\r\n  template: `\r\n    <ul\r\n      class=\"hx-nav hx-nav-{{ type }}\"\r\n      [ngStyle]=\"{\r\n        position: getStickyHeaderPosition(),\r\n        'top.rem': stickyHeaderOffset\r\n      }\"\r\n      [class.is-vertical]=\"vertical\"\r\n      [class.is-justified]=\"justified\"\r\n      [class.has-info]=\"hasInfo\"\r\n    >\r\n      <li\r\n        *ngFor=\"let tab of tabs\"\r\n        [ngClass]=\"['hx-nav-item', tab.customClass || '']\"\r\n        [class.is-active]=\"!!tab?.active\"\r\n        [class.is-disabled]=\"!!tab?.disabled\"\r\n        [class.is-button]=\"tag === 'button'\"\r\n      >\r\n        <button\r\n          *ngIf=\"tag === 'button'\"\r\n          class=\"hx-nav-link\"\r\n          [class.is-active]=\"!!tab?.active\"\r\n          [class.is-disabled]=\"!!tab?.disabled\"\r\n          [attr.disabled]=\"!!tab?.disabled ? '' : null\"\r\n          (click)=\"selectTab(tab)\"\r\n        >\r\n          <span [ngTransclude]=\"tab.headingRef\">{{ tab.heading }}</span>\r\n          <span *ngIf=\"tab.removable\">\r\n            <span\r\n              (click)=\"removeTab(tab)\"\r\n              class=\"icon close-outline is-small\"\r\n            ></span>\r\n          </span>\r\n        </button>\r\n        <a\r\n          *ngIf=\"tag === 'link'\"\r\n          class=\"hx-nav-link\"\r\n          [class.is-active]=\"!!tab?.active\"\r\n          [class.is-disabled]=\"!!tab?.disabled\"\r\n          [attr.disabled]=\"!!tab?.disabled ? '' : null\"\r\n          (click)=\"selectTab(tab)\"\r\n        >\r\n          <span [ngTransclude]=\"tab.headingRef\">{{ tab.heading }}</span>\r\n          <span *ngIf=\"tab.removable\">\r\n            <span\r\n              (click)=\"removeTab(tab)\"\r\n              class=\"icon close-outline is-small\"\r\n            ></span>\r\n          </span>\r\n        </a>\r\n      </li>\r\n    </ul>\r\n    <div class=\"hx-tab-content {{ contentCustomClass }}\">\r\n      <ng-content></ng-content>\r\n    </div>\r\n  `,\r\n  styles: [\r\n    `\r\n      :host,\r\n      ul.hx-nav {\r\n        background-color: inherit;\r\n      }\r\n\r\n      button.hx-nav-link {\r\n        border-width: 0;\r\n        border-bottom-width: 1px;\r\n        border-color: transparent;\r\n        background-color: transparent;\r\n        line-height: 1.5;\r\n        cursor: pointer;\r\n      }\r\n\r\n      :where(.is-justified) .is-button.hx-nav-item {\r\n        display: flex;\r\n      }\r\n\r\n      :where(.is-justified .is-button) button.hx-nav-link {\r\n        display: flex;\r\n      }\r\n    `,\r\n  ],\r\n})\r\nexport class TabsetComponent implements OnDestroy, AfterContentInit {\r\n  /** if true tabs will be placed vertically */\r\n  @Input() vertical = false;\r\n\r\n  /** if true tabs fill the container and have a consistent width */\r\n  @Input() justified = false;\r\n\r\n  @Input() hasInfo = false;\r\n\r\n  /** navigation context class: 'tabs' or 'pills' */\r\n  @Input()\r\n  get type(): string {\r\n    return this._type;\r\n  }\r\n  set type(value: string) {\r\n    this._type = value;\r\n  }\r\n\r\n  @Input() contentCustomClass: string;\r\n  @Input() stickyHeader = false;\r\n  @Input() stickyHeaderOffset = 0;\r\n\r\n  @Input() tag: 'link' | 'button' = 'link';\r\n\r\n  @Input()\r\n  changeFn = async () => true;\r\n\r\n  @ContentChildren(forwardRef(() => TabDirective))\r\n  private _tabList: QueryList<TabDirective>;\r\n\r\n  tabs: TabDirective[] = [];\r\n\r\n  protected isDestroyed: boolean;\r\n  protected _type: string;\r\n\r\n  constructor(config: TabsetConfig) {\r\n    Object.assign(this, config);\r\n  }\r\n\r\n  ngAfterContentInit() {\r\n    // get all active tabs\r\n    const activeTabs = this._tabList.filter(tab => tab.active);\r\n\r\n    // if there is no active tab set, activate the first\r\n    if (activeTabs.length === 0) {\r\n      this._selectTab(this._tabList.last);\r\n    }\r\n  }\r\n\r\n  private _selectTab(tab: TabDirective) {\r\n    // deactivate all tabs\r\n    this._tabList.toArray().forEach(tab => (tab.active = false));\r\n\r\n    // activate the tab the user has clicked on.\r\n    tab.active = true;\r\n  }\r\n\r\n  selectTab(tab: TabDirective) {\r\n    this.changeFn().then(res => !!res && this._selectTab(tab));\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.isDestroyed = true;\r\n  }\r\n\r\n  addTab(tab: TabDirective): void {\r\n    this.tabs.push(tab);\r\n    tab.active = this.tabs.length === 1 && tab.active !== false;\r\n  }\r\n\r\n  removeTab(tab: TabDirective, options = { reselect: true, emit: true }): void {\r\n    const index = this.tabs.indexOf(tab);\r\n    if (index === -1 || this.isDestroyed) {\r\n      return;\r\n    }\r\n    // Select a new tab if the tab to be removed is selected and not destroyed\r\n    if (options.reselect && tab.active && this.hasAvailableTabs(index)) {\r\n      const newActiveIndex = this.getClosestTabIndex(index);\r\n      this.tabs[newActiveIndex].active = true;\r\n    }\r\n    if (options.emit) {\r\n      tab.removed.emit(tab);\r\n    }\r\n    this.tabs.splice(index, 1);\r\n    if (tab.elementRef.nativeElement && tab.elementRef.nativeElement.remove) {\r\n      tab.elementRef.nativeElement.remove();\r\n    }\r\n  }\r\n\r\n  getStickyHeaderPosition(): string {\r\n    return this.stickyHeader ? 'sticky' : 'relative';\r\n  }\r\n\r\n  protected getClosestTabIndex(index: number): number {\r\n    const tabsLength = this.tabs.length;\r\n    if (!tabsLength) {\r\n      return -1;\r\n    }\r\n\r\n    for (let step = 1; step <= tabsLength; step += 1) {\r\n      const prevIndex = index - step;\r\n      const nextIndex = index + step;\r\n      if (this.tabs[prevIndex] && !this.tabs[prevIndex].disabled) {\r\n        return prevIndex;\r\n      }\r\n      if (this.tabs[nextIndex] && !this.tabs[nextIndex].disabled) {\r\n        return nextIndex;\r\n      }\r\n    }\r\n    return -1;\r\n  }\r\n\r\n  protected hasAvailableTabs(index: number): boolean {\r\n    const tabsLength = this.tabs.length;\r\n    if (!tabsLength) {\r\n      return false;\r\n    }\r\n\r\n    for (let i = 0; i < tabsLength; i += 1) {\r\n      if (!this.tabs[i].disabled && i !== index) {\r\n        return true;\r\n      }\r\n    }\r\n    return false;\r\n  }\r\n}\r\n"]}