carbon-components-angular
Version:
Next generation components
243 lines • 19.2 kB
JavaScript
import { Component, Input, Output, EventEmitter, ContentChildren, ContentChild } from "@angular/core";
import { Tab } from "./tab.component";
import { TabHeaders } from "./tab-headers.component";
import * as i0 from "@angular/core";
import * as i1 from "@angular/common";
import * as i2 from "./tab-headers.component";
import * as i3 from "./tab-skeleton.component";
/**
* Build out your application's tabs using this component.
* This is the parent of the `Tab` and `TabHeader` components.
*
* [See demo](../../?path=/story/components-tabs--basic)
*
* `Tabs` expects a set of `n-tab` elements
*
* ```html
* <cds-tabs>
* <cds-tab heading='tab1'>
* tab 1 content
* </cds-tab>
* <cds-tab heading='tab1'>
* tab 2 content
* </cds-tab>
* <!-- ... -->
* <cds-tab heading='tab1'>
* tab n content
* </cds-tab>
* </cds-tabs>
* ```
*/
export class Tabs {
constructor() {
/**
* Takes either `top` or `bottom` to place `TabHeader` relative to the tab panels.
*/
this.position = "top";
/**
* Set to `true` to have `Tab` items cached and not reloaded on tab switching.
*/
this.cacheActive = false;
/**
* Set to 'true' to have tabs automatically activated and have their content displayed when they receive focus.
*/
this.followFocus = true;
/**
* When `true`, sets each tab panel `tabindex` to `-1` for navigation-style usage.
*/
this.isNavigation = false;
/**
* Visual style of the tab headers: `line` or `contained`.
*/
this.type = "line";
/**
* Theme for tab headers: `light` or `dark`.
*/
this.theme = "dark";
/**
* When `true`, renders the tabs skeleton loading state.
*/
this.skeleton = false;
/**
* Evenly sized tabs across the row (contained, fewer than 9 tabs).
*/
this.fullWidth = false;
/**
* Show a close control on each tab.
*/
this.dismissable = false;
/**
* Scroll the active tab into view on focus/select.
*/
this.scrollIntoView = false;
/**
* Debounce (ms) for tab list scroll events; affects overflow chevron updates.
*/
this.scrollDebounceWait = 200;
/**
* Emits when a tab close control is used (with `dismissable`).
* The emitted value is the tab index.
*/
this.tabClose = new EventEmitter();
}
/**
* After content is initialized update `Tab`s to cache (if turned on) and set the initial
* selected Tab item.
*/
ngAfterContentInit() {
if (this.tabHeaders) {
this.tabHeaders.cacheActive = this.cacheActive;
}
this.tabs.forEach(tab => {
tab.tabIndex = this.isNavigation ? null : 0;
});
}
ngOnChanges(changes) {
if (this.tabHeaders && changes.cacheActive) {
this.tabHeaders.cacheActive = this.cacheActive;
}
if (this.tabs && changes.isNavigation) {
this.tabs.forEach(tab => {
tab.tabIndex = this.isNavigation ? null : 0;
});
}
}
/**
* true if the cds-tab's are passed directly to the component as children
*/
hasTabHeaders() {
return this.tabs.length > 0;
}
}
Tabs.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: Tabs, deps: [], target: i0.ɵɵFactoryTarget.Component });
Tabs.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: Tabs, selector: "cds-tabs, ibm-tabs", inputs: { position: "position", cacheActive: "cacheActive", followFocus: "followFocus", isNavigation: "isNavigation", ariaLabel: "ariaLabel", ariaLabelledby: "ariaLabelledby", type: "type", theme: "theme", skeleton: "skeleton", iconSize: "iconSize", fullWidth: "fullWidth", dismissable: "dismissable", scrollIntoView: "scrollIntoView", scrollDebounceWait: "scrollDebounceWait" }, outputs: { tabClose: "tabClose" }, queries: [{ propertyName: "tabHeaders", first: true, predicate: TabHeaders, descendants: true }, { propertyName: "tabs", predicate: Tab }], usesOnChanges: true, ngImport: i0, template: `
<ng-container *ngIf="skeleton">
<cds-tabs-skeleton [contained]="type === 'contained'"></cds-tabs-skeleton>
</ng-container>
<ng-container *ngIf="!skeleton">
<cds-tab-headers
*ngIf="hasTabHeaders() && position === 'top'"
[]="theme"
[]="tabs"
[]="followFocus"
[]="cacheActive"
[]="before"
[]="after"
[]="ariaLabel"
[]="ariaLabelledby"
[]="type"
[]="iconSize"
[]="fullWidth"
[]="dismissable"
[]="scrollIntoView"
[]="scrollDebounceWait"
(tabClose)="tabClose.emit($event)">
</cds-tab-headers>
<ng-content></ng-content>
<ng-template
<ng-content select="[before]"></ng-content>
</ng-template>
<ng-template
<ng-content select="[after]"></ng-content>
</ng-template>
<cds-tab-headers
*ngIf="hasTabHeaders() && position === 'bottom'"
[]="tabs"
[]="cacheActive"
[]="type"
[]="iconSize"
[]="fullWidth"
[]="dismissable"
[]="scrollIntoView"
[]="scrollDebounceWait"
(tabClose)="tabClose.emit($event)">
</cds-tab-headers>
</ng-container>
`, isInline: true, dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.TabHeaders, selector: "cds-tab-headers, ibm-tab-headers", inputs: ["tabs", "translations"], outputs: ["tabClose"] }, { kind: "component", type: i3.TabSkeleton, selector: "cds-tabs-skeleton, ibm-tabs-skeleton", inputs: ["numOftabs", "contained"] }] });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: Tabs, decorators: [{
type: Component,
args: [{
selector: "cds-tabs, ibm-tabs",
template: `
<ng-container *ngIf="skeleton">
<cds-tabs-skeleton [contained]="type === 'contained'"></cds-tabs-skeleton>
</ng-container>
<ng-container *ngIf="!skeleton">
<cds-tab-headers
*ngIf="hasTabHeaders() && position === 'top'"
[]="theme"
[]="tabs"
[]="followFocus"
[]="cacheActive"
[]="before"
[]="after"
[]="ariaLabel"
[]="ariaLabelledby"
[]="type"
[]="iconSize"
[]="fullWidth"
[]="dismissable"
[]="scrollIntoView"
[]="scrollDebounceWait"
(tabClose)="tabClose.emit($event)">
</cds-tab-headers>
<ng-content></ng-content>
<ng-template
<ng-content select="[before]"></ng-content>
</ng-template>
<ng-template
<ng-content select="[after]"></ng-content>
</ng-template>
<cds-tab-headers
*ngIf="hasTabHeaders() && position === 'bottom'"
[]="tabs"
[]="cacheActive"
[]="type"
[]="iconSize"
[]="fullWidth"
[]="dismissable"
[]="scrollIntoView"
[]="scrollDebounceWait"
(tabClose)="tabClose.emit($event)">
</cds-tab-headers>
</ng-container>
`
}]
}], propDecorators: { position: [{
type: Input
}], cacheActive: [{
type: Input
}], followFocus: [{
type: Input
}], isNavigation: [{
type: Input
}], ariaLabel: [{
type: Input
}], ariaLabelledby: [{
type: Input
}], type: [{
type: Input
}], theme: [{
type: Input
}], skeleton: [{
type: Input
}], iconSize: [{
type: Input
}], fullWidth: [{
type: Input
}], dismissable: [{
type: Input
}], scrollIntoView: [{
type: Input
}], scrollDebounceWait: [{
type: Input
}], tabClose: [{
type: Output
}], tabs: [{
type: ContentChildren,
args: [Tab, { descendants: false }]
}], tabHeaders: [{
type: ContentChild,
args: [TabHeaders]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs.component.js","sourceRoot":"","sources":["../../../src/tabs/tabs.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n\tComponent,\n\tInput,\n\tOutput,\n\tEventEmitter,\n\tContentChildren,\n\tQueryList,\n\tAfterContentInit,\n\tContentChild,\n\tOnChanges,\n\tSimpleChanges\n} from \"@angular/core\";\nimport { Tab } from \"./tab.component\";\nimport { TabHeaders } from \"./tab-headers.component\";\n\n/**\n * Build out your application's tabs using this component.\n * This is the parent of the `Tab` and `TabHeader` components.\n *\n * [See demo](../../?path=/story/components-tabs--basic)\n *\n * `Tabs` expects a set of `n-tab` elements\n *\n * ```html\n * <cds-tabs>\n * \t<cds-tab heading='tab1'>\n * \t\ttab 1 content\n * \t</cds-tab>\n * \t<cds-tab heading='tab1'>\n * \t\ttab 2 content\n * \t</cds-tab>\n * \t<!-- ... -->\n * \t<cds-tab heading='tab1'>\n * \t\ttab n content\n * \t</cds-tab>\n * </cds-tabs>\n * ```\n */\n@Component({\n\tselector: \"cds-tabs, ibm-tabs\",\n\ttemplate: `\n\t\t<ng-container *ngIf=\"skeleton\">\n\t\t\t<cds-tabs-skeleton [contained]=\"type === 'contained'\"></cds-tabs-skeleton>\n\t\t</ng-container>\n\t\t<ng-container *ngIf=\"!skeleton\">\n\t\t\t<cds-tab-headers\n\t\t\t\t*ngIf=\"hasTabHeaders() && position === 'top'\"\n\t\t\t\t[theme]=\"theme\"\n\t\t\t\t[tabs]=\"tabs\"\n\t\t\t\t[followFocus]=\"followFocus\"\n\t\t\t\t[cacheActive]=\"cacheActive\"\n\t\t\t\t[contentBefore]=\"before\"\n\t\t\t\t[contentAfter]=\"after\"\n\t\t\t\t[ariaLabel]=\"ariaLabel\"\n\t\t\t\t[ariaLabelledby]=\"ariaLabelledby\"\n\t\t\t\t[type]=\"type\"\n\t\t\t\t[iconSize]=\"iconSize\"\n\t\t\t\t[fullWidth]=\"fullWidth\"\n\t\t\t\t[dismissable]=\"dismissable\"\n\t\t\t\t[scrollIntoView]=\"scrollIntoView\"\n\t\t\t\t[scrollDebounceWait]=\"scrollDebounceWait\"\n\t\t\t\t(tabClose)=\"tabClose.emit($event)\">\n\t\t\t</cds-tab-headers>\n\t\t\t<ng-content></ng-content>\n\t\t\t<ng-template #before>\n\t\t\t\t<ng-content select=\"[before]\"></ng-content>\n\t\t\t</ng-template>\n\t\t\t<ng-template #after>\n\t\t\t\t<ng-content select=\"[after]\"></ng-content>\n\t\t\t</ng-template>\n\t\t\t<cds-tab-headers\n\t\t\t\t*ngIf=\"hasTabHeaders() && position === 'bottom'\"\n\t\t\t\t[tabs]=\"tabs\"\n\t\t\t\t[cacheActive]=\"cacheActive\"\n\t\t\t\t[type]=\"type\"\n\t\t\t\t[iconSize]=\"iconSize\"\n\t\t\t\t[fullWidth]=\"fullWidth\"\n\t\t\t\t[dismissable]=\"dismissable\"\n\t\t\t\t[scrollIntoView]=\"scrollIntoView\"\n\t\t\t\t[scrollDebounceWait]=\"scrollDebounceWait\"\n\t\t\t\t(tabClose)=\"tabClose.emit($event)\">\n\t\t\t</cds-tab-headers>\n\t\t</ng-container>\n\t`\n})\nexport class Tabs implements AfterContentInit, OnChanges {\n\t/**\n\t * Takes either `top` or `bottom` to place `TabHeader` relative to the tab panels.\n\t */\n\t@Input() position: \"top\" | \"bottom\" = \"top\";\n\t/**\n\t * Set to `true` to have `Tab` items cached and not reloaded on tab switching.\n\t */\n\t@Input() cacheActive = false;\n\t/**\n\t * Set to 'true' to have tabs automatically activated and have their content displayed when they receive focus.\n\t */\n\t@Input() followFocus = true;\n\t/**\n\t * When `true`, sets each tab panel `tabindex` to `-1` for navigation-style usage.\n\t */\n\t@Input() isNavigation = false;\n\t/**\n\t * Sets the aria label on the `TabHeader`s nav element.\n\t */\n\t@Input() ariaLabel: string;\n\t/**\n\t * Sets the aria labelledby on the `TabHeader`s nav element.\n\t */\n\t@Input() ariaLabelledby: string;\n\t/**\n\t * Visual style of the tab headers: `line` or `contained`.\n\t */\n\t@Input() type: \"line\" | \"contained\" = \"line\";\n\t/**\n\t * Theme for tab headers: `light` or `dark`.\n\t */\n\t@Input() theme: \"light\" | \"dark\" = \"dark\";\n\t/**\n\t * When `true`, renders the tabs skeleton loading state.\n\t */\n\t@Input() skeleton = false;\n\t/**\n\t * When using icon-only tabs, icon size: `default` (16px) or `lg` (20px).\n\t */\n\t@Input() iconSize: \"default\" | \"lg\";\n\t/**\n\t * Evenly sized tabs across the row (contained, fewer than 9 tabs).\n\t */\n\t@Input() fullWidth = false;\n\t/**\n\t * Show a close control on each tab.\n\t */\n\t@Input() dismissable = false;\n\t/**\n\t * Scroll the active tab into view on focus/select.\n\t */\n\t@Input() scrollIntoView = false;\n\t/**\n\t * Debounce (ms) for tab list scroll events; affects overflow chevron updates.\n\t */\n\t@Input() scrollDebounceWait = 200;\n\n\t/**\n\t * Emits when a tab close control is used (with `dismissable`).\n\t * The emitted value is the tab index.\n\t */\n\t@Output() tabClose: EventEmitter<number> = new EventEmitter<number>();\n\n\t/**\n\t * Maintains a `QueryList` of the `Tab` elements and updates if `Tab`s are added or removed.\n\t */\n\t@ContentChildren(Tab, { descendants: false }) tabs: QueryList<Tab>;\n\t/**\n\t * Content child of the projected header component\n\t */\n\t@ContentChild(TabHeaders) tabHeaders;\n\n\t/**\n\t * After content is initialized update `Tab`s to cache (if turned on) and set the initial\n\t * selected Tab item.\n\t */\n\tngAfterContentInit() {\n\t\tif (this.tabHeaders) {\n\t\t\tthis.tabHeaders.cacheActive = this.cacheActive;\n\t\t}\n\n\t\tthis.tabs.forEach(tab => {\n\t\t\ttab.tabIndex = this.isNavigation ? null : 0;\n\t\t});\n\t}\n\n\tngOnChanges(changes: SimpleChanges) {\n\t\tif (this.tabHeaders && changes.cacheActive) {\n\t\t\tthis.tabHeaders.cacheActive = this.cacheActive;\n\t\t}\n\n\t\tif (this.tabs && changes.isNavigation) {\n\t\t\tthis.tabs.forEach(tab => {\n\t\t\t\ttab.tabIndex = this.isNavigation ? null : 0;\n\t\t\t});\n\t\t}\n\t}\n\n\t/**\n\t * true if the cds-tab's are passed directly to the component as children\n\t */\n\thasTabHeaders() {\n\t\treturn this.tabs.length > 0;\n\t}\n}\n"]}