@hxui/angular
Version:
An Angular library based on the [HXUI design system](https://hxui.io).
272 lines (269 loc) • 24.7 kB
JavaScript
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"]}