gentics-ui-core
Version:
This is the common core framework for the Gentics CMS and Mesh UI, and other Angular applications.
173 lines • 21.7 kB
JavaScript
import { Component, ContentChildren, QueryList, Input, Output, EventEmitter } from '@angular/core';
import { Tab } from './tab.component';
import { coerceToBoolean } from '../../common/coerce-to-boolean';
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "@angular/router";
import * as i3 from "../icon/icon.directive";
/**
* Tabs can be either pure or stateful. Stateful tabs will keep track of which one is active by keeping an internal
* state.
*
* Pure tabs will only change the active tab when the `activeId` property is updated.
*
* ## Stateful Tabs
* ```html
* <gtx-tabs (tabChange)="goToTab($event)">
* <gtx-tab title="Details">Optional content here.</gtx-tab>
* <gtx-tab title="Orders"></gtx-tab>
* <gtx-tab title="Notes"></gtx-tab>
* </gtx-tabs>
* ```
*
* ## Pure Tabs
* ```html
* <gtx-tabs pure [activeId]="activeTab">
* <gtx-tab title="Details" id="1" (select)="activeTab = $event"></gtx-tab>
* <gtx-tab title="Orders" id="2" (select)="activeTab = $event"></gtx-tab>
* <gtx-tab title="Notes" id="3" (select)="activeTab = $event"></gtx-tab>
* </gtx-tabs>
* ```
*
* ## With `routerLink`
* A gtx-tab can take an optional `[routerLink]` binding which will set router links for the tabs.
* ```html
* <gtx-tabs pure [activeId]="activeTab">
* <gtx-tab title="Details" id="1" [routerLink]="['customer', 'details']"></gtx-tab>
* <gtx-tab title="Orders" id="2" [routerLink]="['customer', 'orders']"></gtx-tab>
* <gtx-tab title="Notes" id="3" [routerLink]="['customer', 'notes']"></gtx-tab>
* </gtx-tabs>
* ```
* * ##### Vertical Tabs
* A gtx-tabs can take an optional `vertical` property which allows to display tabs vertically.
* ```html
* <gtx-tabs vertical>
* <gtx-tab title="Details"></gtx-tab>
* <gtx-tab title="Orders"></gtx-tab>
* <gtx-tab title="Notes"></gtx-tab>
* </gtx-tabs>
* ```
* * * ##### Active Tabs with Icons
* A gtx-tabs can take an optional `hideTitle` property which allows to hide the title for non-active tabs with icons.
* ```html
* <gtx-tabs hideTitle>
* <gtx-tab icon="folder" title="Tab 1">Tab 1 Content</gtx-tab>
* <gtx-tab icon="cloud" title="Tab 2">Tab 2 Content</gtx-tab>
* </gtx-tabs>
* ```
*
*/
export class Tabs {
constructor() {
/**
* Fires an event whenever the active tab changes. Argument is the id of the selected tab.
*/
this.tabChange = new EventEmitter();
this.shouldHideTitle = false;
this.verticalTabs = false;
this.tabsShouldWrap = false;
this.isPure = false;
}
/**
* When present (or set to true), tabs are displayed vertically.
*/
set vertical(val) {
this.verticalTabs = coerceToBoolean(val);
}
/**
* When present (or set to true), only active tabs with icons will show the title.
* Non-active tabs with icons will hide the title, show only icon.
*/
set hideTitle(val) {
this.shouldHideTitle = coerceToBoolean(val);
}
/**
* When present, sets the tabs to pure (stateless) mode.
*/
set pure(val) {
this.isPure = val != null;
}
/**
* When present (or set to true), tabs which do not fit into the width of their container will wrap onto
* a new line. Otherwise, tabs will remain on one line and the contents will be elided if all the available
* space is filled.
*/
set wrap(val) {
this.tabsShouldWrap = coerceToBoolean(val);
}
ngAfterContentInit() {
if (this.isPure) {
this.tabsChangeSubscription = this.tabs.changes.subscribe(() => {
setTimeout(() => this.setActiveTab());
});
}
else {
let activeTabs = this.tabs.filter(tab => tab.active);
// if there is no active tab set, activate the first
if (activeTabs.length === 0) {
this.tabs.first.active = true;
}
}
this.tabs.notifyOnChanges();
}
ngOnChanges(changes) {
this.setActiveTab();
}
ngOnDestroy() {
if (this.tabsChangeSubscription) {
this.tabsChangeSubscription.unsubscribe();
}
}
/**
* Sets the tab with id === this.activeId to active.
*/
setActiveTab() {
if (this.tabs) {
let tabToActivate = this.tabs.filter(t => t.id === this.activeId)[0];
if (tabToActivate) {
this.setAsActive(tabToActivate);
}
}
}
/**
* Invoked when a tab link is clicked.
*/
selectTab(tab) {
if (tab.disabled) {
return;
}
if (!this.isPure) {
this.setAsActive(tab);
this.tabChange.emit(tab.id);
}
else {
tab.select.emit(tab.id);
}
}
setAsActive(tab) {
this.tabs.toArray().forEach(tab => tab.active = false);
tab.active = true;
}
}
/** @nocollapse */ Tabs.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: Tabs, deps: [], target: i0.ɵɵFactoryTarget.Component });
/** @nocollapse */ Tabs.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.8", type: Tabs, selector: "gtx-tabs", inputs: { vertical: "vertical", hideTitle: "hideTitle", activeId: "activeId", pure: "pure", wrap: "wrap" }, outputs: { tabChange: "tabChange" }, queries: [{ propertyName: "tabs", predicate: Tab }], usesOnChanges: true, ngImport: i0, template: "<ul class=\"tab-links\" [class.vertical]=\"verticalTabs\" [class.wrap]=\"tabsShouldWrap\">\n <li class=\"tab-link\" role=\"presentation\"\n *ngFor=\"let tab of tabs\"\n [title]=\"tab.title\"\n (click)=\"selectTab(tab)\"\n [ngClass]=\"{\n disabled: tab.disabled,\n 'is-active': tab.active,\n 'has-icon': !!tab.icon,\n 'icon-only': shouldHideTitle && !tab.active || (tab.icon && !tab.title)\n }\">\n <a role=\"tab\" *ngIf=\"tab.routerLink && !tab.disabled\" [routerLink]=\"tab.routerLink\">\n <icon *ngIf=\"tab.icon\">{{tab.icon}}</icon><span [class.animatedTitle]=\"shouldHideTitle\" *ngIf=\"!shouldHideTitle || tab.active || (!tab.active && !tab.icon)\">{{ tab.title }}</span></a>\n <a role=\"tab\" *ngIf=\"!tab.routerLink || tab.disabled\">\n <icon *ngIf=\"tab.icon\">{{tab.icon}}</icon><span [class.animatedTitle]=\"shouldHideTitle\" *ngIf=\"!shouldHideTitle || tab.active || (!tab.active && !tab.icon)\">{{ tab.title }}</span></a>\n </li>\n</ul>\n<ng-content></ng-content>\n", directives: [{ type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i2.RouterLinkWithHref, selector: "a[routerLink],area[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "preserveFragment", "skipLocationChange", "replaceUrl", "state", "relativeTo", "routerLink"] }, { type: i3.Icon, selector: "icon" }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.8", ngImport: i0, type: Tabs, decorators: [{
type: Component,
args: [{ selector: 'gtx-tabs', template: "<ul class=\"tab-links\" [class.vertical]=\"verticalTabs\" [class.wrap]=\"tabsShouldWrap\">\n <li class=\"tab-link\" role=\"presentation\"\n *ngFor=\"let tab of tabs\"\n [title]=\"tab.title\"\n (click)=\"selectTab(tab)\"\n [ngClass]=\"{\n disabled: tab.disabled,\n 'is-active': tab.active,\n 'has-icon': !!tab.icon,\n 'icon-only': shouldHideTitle && !tab.active || (tab.icon && !tab.title)\n }\">\n <a role=\"tab\" *ngIf=\"tab.routerLink && !tab.disabled\" [routerLink]=\"tab.routerLink\">\n <icon *ngIf=\"tab.icon\">{{tab.icon}}</icon><span [class.animatedTitle]=\"shouldHideTitle\" *ngIf=\"!shouldHideTitle || tab.active || (!tab.active && !tab.icon)\">{{ tab.title }}</span></a>\n <a role=\"tab\" *ngIf=\"!tab.routerLink || tab.disabled\">\n <icon *ngIf=\"tab.icon\">{{tab.icon}}</icon><span [class.animatedTitle]=\"shouldHideTitle\" *ngIf=\"!shouldHideTitle || tab.active || (!tab.active && !tab.icon)\">{{ tab.title }}</span></a>\n </li>\n</ul>\n<ng-content></ng-content>\n" }]
}], propDecorators: { tabs: [{
type: ContentChildren,
args: [Tab]
}], tabChange: [{
type: Output
}], vertical: [{
type: Input
}], hideTitle: [{
type: Input
}], activeId: [{
type: Input
}], pure: [{
type: Input
}], wrap: [{
type: Input
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs.component.js","sourceRoot":"","sources":["../../../../../src/components/tabs/tabs.component.ts","../../../../../src/components/tabs/tabs.tpl.html"],"names":[],"mappings":"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","sourcesContent":["import {Component, ContentChildren, QueryList, AfterContentInit, Input, Output, EventEmitter, SimpleChanges, OnChanges, OnDestroy} from '@angular/core';\nimport {Tab} from './tab.component';\nimport {coerceToBoolean} from '../../common/coerce-to-boolean';\nimport { Subscription } from 'rxjs';\n\n/**\n * Tabs can be either pure or stateful. Stateful tabs will keep track of which one is active by keeping an internal\n * state.\n *\n * Pure tabs will only change the active tab when the `activeId` property is updated.\n *\n * ## Stateful Tabs\n * ```html\n * <gtx-tabs (tabChange)=\"goToTab($event)\">\n *     <gtx-tab title=\"Details\">Optional content here.</gtx-tab>\n *     <gtx-tab title=\"Orders\"></gtx-tab>\n *     <gtx-tab title=\"Notes\"></gtx-tab>\n * </gtx-tabs>\n * ```\n *\n * ## Pure Tabs\n * ```html\n * <gtx-tabs pure [activeId]=\"activeTab\">\n *     <gtx-tab title=\"Details\" id=\"1\" (select)=\"activeTab = $event\"></gtx-tab>\n *     <gtx-tab title=\"Orders\" id=\"2\" (select)=\"activeTab = $event\"></gtx-tab>\n *     <gtx-tab title=\"Notes\" id=\"3\" (select)=\"activeTab = $event\"></gtx-tab>\n * </gtx-tabs>\n * ```\n *\n * ## With `routerLink`\n * A gtx-tab can take an optional `[routerLink]` binding which will set router links for the tabs.\n * ```html\n * <gtx-tabs pure [activeId]=\"activeTab\">\n *     <gtx-tab title=\"Details\" id=\"1\" [routerLink]=\"['customer', 'details']\"></gtx-tab>\n *     <gtx-tab title=\"Orders\" id=\"2\" [routerLink]=\"['customer', 'orders']\"></gtx-tab>\n *     <gtx-tab title=\"Notes\" id=\"3\" [routerLink]=\"['customer', 'notes']\"></gtx-tab>\n * </gtx-tabs>\n * ```\n *  * ##### Vertical Tabs\n * A gtx-tabs can take an optional `vertical` property which allows to display tabs vertically.\n * ```html\n * <gtx-tabs vertical>\n *        <gtx-tab title=\"Details\"></gtx-tab>\n *        <gtx-tab title=\"Orders\"></gtx-tab>\n *        <gtx-tab title=\"Notes\"></gtx-tab>\n * </gtx-tabs>\n * ```\n *  *  * ##### Active Tabs with Icons\n * A gtx-tabs can take an optional `hideTitle` property which allows to hide the title for non-active tabs with icons.\n * ```html\n * <gtx-tabs hideTitle>\n *           <gtx-tab icon=\"folder\" title=\"Tab 1\">Tab 1 Content</gtx-tab>\n *           <gtx-tab icon=\"cloud\" title=\"Tab 2\">Tab 2 Content</gtx-tab>\n * </gtx-tabs>\n * ```\n *\n */\n@Component({\n    selector: 'gtx-tabs',\n    templateUrl: './tabs.tpl.html'\n})\nexport class Tabs implements AfterContentInit, OnChanges, OnDestroy {\n\n    @ContentChildren(Tab) tabs: QueryList<Tab>;\n\n    /**\n     * Fires an event whenever the active tab changes. Argument is the id of the selected tab.\n     */\n    @Output() tabChange = new EventEmitter<string>();\n\n    /**\n     * When present (or set to true), tabs are displayed vertically.\n     */\n    @Input() set vertical(val: any) {\n        this.verticalTabs = coerceToBoolean(val);\n    }\n\n    /**\n     * When present (or set to true), only active tabs with icons will show the title.\n     * Non-active tabs with icons will hide the title, show only icon.\n     */\n    @Input() set hideTitle(val: any) {\n        this.shouldHideTitle = coerceToBoolean(val);\n    }\n\n    /**\n     * The id of the active tab. Should only be used in pure (stateless) mode.\n     */\n    @Input() activeId: string;\n\n    /**\n     * When present, sets the tabs to pure (stateless) mode.\n     */\n    @Input() set pure(val: any) {\n        this.isPure = val != null;\n    }\n\n    /**\n     * When present (or set to true), tabs which do not fit into the width of their container will wrap onto\n     * a new line. Otherwise, tabs will remain on one line and the contents will be elided if all the available\n     * space is filled.\n     */\n    @Input() set wrap(val: any) {\n        this.tabsShouldWrap = coerceToBoolean(val);\n    }\n\n    shouldHideTitle: boolean = false;\n    verticalTabs: boolean = false;\n    tabsShouldWrap: boolean = false;\n    private isPure: boolean = false;\n\n    tabsChangeSubscription: Subscription;\n\n    ngAfterContentInit(): void {\n        if (this.isPure) {\n            this.tabsChangeSubscription = this.tabs.changes.subscribe(() => {\n                setTimeout(() => this.setActiveTab());\n            });\n        } else {\n            let activeTabs = this.tabs.filter(tab => tab.active);\n\n            // if there is no active tab set, activate the first\n            if (activeTabs.length === 0) {\n                this.tabs.first.active = true;\n            }\n        }\n        this.tabs.notifyOnChanges();\n    }\n\n    ngOnChanges(changes: SimpleChanges): void {\n        this.setActiveTab();\n    }\n\n    ngOnDestroy(): void {\n        if (this.tabsChangeSubscription) {\n            this.tabsChangeSubscription.unsubscribe();\n        }\n    }\n\n    /**\n     * Sets the tab with id === this.activeId to active.\n     */\n    setActiveTab(): void {\n        if (this.tabs) {\n            let tabToActivate = this.tabs.filter(t => t.id === this.activeId)[0];\n            if (tabToActivate) {\n                this.setAsActive(tabToActivate);\n            }\n        }\n    }\n\n    /**\n     * Invoked when a tab link is clicked.\n     */\n    selectTab(tab: Tab): void {\n        if (tab.disabled) {\n            return;\n        }\n        if (!this.isPure) {\n            this.setAsActive(tab);\n            this.tabChange.emit(tab.id);\n        } else {\n            tab.select.emit(tab.id);\n        }\n    }\n\n    private setAsActive(tab: Tab): void {\n        this.tabs.toArray().forEach(tab => tab.active = false);\n        tab.active = true;\n    }\n\n}\n","<ul class=\"tab-links\" [class.vertical]=\"verticalTabs\" [class.wrap]=\"tabsShouldWrap\">\n    <li class=\"tab-link\" role=\"presentation\"\n            *ngFor=\"let tab of tabs\"\n            [title]=\"tab.title\"\n            (click)=\"selectTab(tab)\"\n            [ngClass]=\"{\n                disabled: tab.disabled,\n                'is-active': tab.active,\n                'has-icon': !!tab.icon,\n                'icon-only': shouldHideTitle && !tab.active || (tab.icon && !tab.title)\n            }\">\n        <a role=\"tab\" *ngIf=\"tab.routerLink && !tab.disabled\" [routerLink]=\"tab.routerLink\">\n            <icon *ngIf=\"tab.icon\">{{tab.icon}}</icon><span [class.animatedTitle]=\"shouldHideTitle\" *ngIf=\"!shouldHideTitle || tab.active || (!tab.active && !tab.icon)\">{{ tab.title }}</span></a>\n        <a role=\"tab\" *ngIf=\"!tab.routerLink || tab.disabled\">\n            <icon *ngIf=\"tab.icon\">{{tab.icon}}</icon><span [class.animatedTitle]=\"shouldHideTitle\" *ngIf=\"!shouldHideTitle || tab.active || (!tab.active && !tab.icon)\">{{ tab.title }}</span></a>\n    </li>\n</ul>\n<ng-content></ng-content>\n"]}