@doku-dev/doku-fragment
Version:
A new Angular UI library that moving away from Bootstrap and built from scratch.
176 lines • 22.7 kB
JavaScript
import { CommonModule } from '@angular/common';
import { ChangeDetectionStrategy, Component, ContentChildren, EventEmitter, HostBinding, Input, Output, ViewEncapsulation, } from '@angular/core';
import { ReplaySubject, map, pairwise, shareReplay, startWith, takeUntil } from 'rxjs';
import { DOKU_TAB, DOKU_TABS } from './token';
import { ViewElement } from './view-element';
import * as i0 from "@angular/core";
export class DokuTabs {
constructor(elementRef, renderer, appRef) {
this.elementRef = elementRef;
this.renderer = renderer;
this.appRef = appRef;
/**
* Initial active id of the tab.
* Default to first tab.
*/
this.activeId = '';
/**
* Whether to use content card styles for the tab body.
*
* @default false
*/
this.useContentCard = false;
/**
* Listen to active tab change.
*/
this.activeTabChange = new EventEmitter();
this.classes = 'd-tabs';
this.destroy$ = new ReplaySubject();
/**
* It used in tab component.
*/
this.tabChangeForChild$ = this.activeTabChange
.asObservable()
.pipe(shareReplay(), takeUntil(this.destroy$));
}
ngAfterContentInit() {
// Create tabs header and body, then append them.
this.tabsHeaderElement = ViewElement.createTabsHeader({ renderer: this.renderer });
this.tabsBodyElement = ViewElement.createTabsBody({ renderer: this.renderer });
this.elementRef.nativeElement.append(this.tabsHeaderElement, this.tabsBodyElement);
// Set initial active tab
const activeId = this.tabItems?.first ? this.tabItems.first.id : null;
if (!this.isValidActiveId(this.activeId) && activeId) {
this.updateActiveId(activeId);
}
this.tabItems?.changes
.pipe(startWith(this.tabItems), map((items) => items.toArray()), startWith([]), pairwise(), takeUntil(this.destroy$))
.subscribe(([prevItems, currentItems]) => {
this.updateTabs(prevItems, currentItems);
this.notifyTabChange(this.activeId);
});
}
ngOnDestroy() {
this.destroy$.next(true);
this.destroy$.complete();
}
/**
* Change the active tab.
* @param id Tab id.
*/
changeActiveTab(id) {
this.updateActiveId(id, { emitTabChange: true });
}
updateActiveId(id, options) {
if (this.activeId === id)
return;
this.activeId = id;
if (options?.emitTabChange)
this.notifyTabChange(id);
}
notifyTabChange(itemId) {
const tab = this.getTabById(itemId);
if (!tab)
return;
this.activeTabChange.emit(tab);
}
getTabById(id) {
return this.tabItems?.find((tab) => tab.id === id) || null;
}
isValidActiveId(id) {
return !!this.tabItems?.some((tab) => tab.id === id);
}
/**
* Update tab elements when adding more or removing tab items.
*/
updateTabs(prevItems, currentItems) {
const removedItems = prevItems.filter((prevItem) => !currentItems.some((currentItem) => prevItem.id === currentItem.id));
const newItems = currentItems.filter((currentItem) => !prevItems.some((prevItem) => currentItem.id === prevItem.id));
removedItems.forEach((item) => {
this.removeTabLabel(item);
this.removeTabContent(item);
});
newItems.forEach((item) => {
this.addTabLabel(item);
this.addTabContent(item);
});
}
addTabLabel(item) {
if (!this.tabsHeaderElement || !item['labelTemplate'])
return;
if (item['labelElement'] || item['labelViewRef'])
return;
item['labelElement'] = ViewElement.createTabLabel({ renderer: this.renderer });
item['labelViewRef'] = ViewElement.appendLabelToTabsHeader({
applicationRef: this.appRef,
renderer: this.renderer,
headerElement: this.tabsHeaderElement,
labelElement: item['labelElement'],
labelTemplate: item['labelTemplate'],
});
}
removeTabLabel(item) {
item['labelElement']?.remove();
item['labelViewRef']?.destroy();
}
addTabContent(item) {
if (!this.tabsBodyElement || !item['contentTemplate'])
return;
if (item['contentElement'] || item['contentViewRef'])
return;
item['contentElement'] = ViewElement.createTabContent({ renderer: this.renderer });
item['contentViewRef'] = ViewElement.appendContentToTabsBody({
applicationRef: this.appRef,
renderer: this.renderer,
bodyElement: this.tabsBodyElement,
contentElement: item['contentElement'],
contentTemplate: item['contentTemplate'],
lazyLoad: item.lazyLoad,
});
}
removeTabContent(item) {
item['contentElement']?.remove();
item['contentViewRef']?.destroy();
}
}
DokuTabs.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTabs, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.ApplicationRef }], target: i0.ɵɵFactoryTarget.Component });
DokuTabs.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DokuTabs, isStandalone: true, selector: "doku-tabs", inputs: { activeId: "activeId", useContentCard: "useContentCard" }, outputs: { activeTabChange: "activeTabChange" }, host: { properties: { "class.d-tabs-content-card": "this.useContentCard", "class": "this.classes" } }, providers: [
{
provide: DOKU_TABS,
useExisting: DokuTabs,
},
], queries: [{ propertyName: "tabItems", predicate: DOKU_TAB }], exportAs: ["dokuTabs"], ngImport: i0, template: '', isInline: true, dependencies: [{ kind: "ngmodule", type: CommonModule }], changeDetection: i0.ChangeDetectionStrategy.OnPush, encapsulation: i0.ViewEncapsulation.None });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTabs, decorators: [{
type: Component,
args: [{
selector: 'doku-tabs',
exportAs: 'dokuTabs',
standalone: true,
imports: [CommonModule],
template: '',
encapsulation: ViewEncapsulation.None,
changeDetection: ChangeDetectionStrategy.OnPush,
providers: [
{
provide: DOKU_TABS,
useExisting: DokuTabs,
},
],
}]
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.ApplicationRef }]; }, propDecorators: { activeId: [{
type: Input
}], useContentCard: [{
type: HostBinding,
args: ['class.d-tabs-content-card']
}, {
type: Input
}], activeTabChange: [{
type: Output
}], classes: [{
type: HostBinding,
args: ['class']
}], tabItems: [{
type: ContentChildren,
args: [DOKU_TAB, { descendants: false }]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tabs.component.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/tabs/tabs.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule, NgClass } from '@angular/common';\nimport {\n  AfterContentInit,\n  ApplicationRef,\n  ChangeDetectionStrategy,\n  Component,\n  ContentChildren,\n  ElementRef,\n  EventEmitter,\n  HostBinding,\n  Input,\n  OnDestroy,\n  Output,\n  QueryList,\n  Renderer2,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { ReplaySubject, map, pairwise, shareReplay, startWith, takeUntil } from 'rxjs';\nimport { DokuTab } from './tab.component';\nimport { DOKU_TAB, DOKU_TABS } from './token';\nimport { ViewElement } from './view-element';\n\n@Component({\n  selector: 'doku-tabs',\n  exportAs: 'dokuTabs',\n  standalone: true,\n  imports: [CommonModule],\n  template: '',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: DOKU_TABS,\n      useExisting: DokuTabs,\n    },\n  ],\n})\nexport class DokuTabs implements AfterContentInit, OnDestroy {\n  /**\n   * Initial active id of the tab.\n   * Default to first tab.\n   */\n  @Input() activeId = '';\n\n  /**\n   * Whether to use content card styles for the tab body.\n   *\n   * @default false\n   */\n  @HostBinding('class.d-tabs-content-card')\n  @Input()\n  useContentCard = false;\n\n  /**\n   * Listen to active tab change.\n   */\n  @Output() activeTabChange = new EventEmitter<DokuTab>();\n\n  @HostBinding('class')\n  protected readonly classes: NgClass['ngClass'] = 'd-tabs';\n\n  @ContentChildren(DOKU_TAB, { descendants: false }) private tabItems?: QueryList<DokuTab>;\n\n  private tabsHeaderElement?: HTMLDivElement;\n  private tabsBodyElement?: HTMLDivElement;\n\n  private destroy$ = new ReplaySubject();\n\n  /**\n   * It used in tab component.\n   */\n  protected tabChangeForChild$ = this.activeTabChange\n    .asObservable()\n    .pipe(shareReplay(), takeUntil(this.destroy$));\n\n  constructor(\n    private elementRef: ElementRef,\n    private renderer: Renderer2,\n    private appRef: ApplicationRef\n  ) {}\n\n  ngAfterContentInit(): void {\n    // Create tabs header and body, then append them.\n    this.tabsHeaderElement = ViewElement.createTabsHeader({ renderer: this.renderer });\n    this.tabsBodyElement = ViewElement.createTabsBody({ renderer: this.renderer });\n    (this.elementRef.nativeElement as HTMLElement).append(\n      this.tabsHeaderElement,\n      this.tabsBodyElement\n    );\n\n    // Set initial active tab\n    const activeId = this.tabItems?.first ? this.tabItems.first.id : null;\n    if (!this.isValidActiveId(this.activeId) && activeId) {\n      this.updateActiveId(activeId);\n    }\n\n    this.tabItems?.changes\n      .pipe(\n        startWith(this.tabItems),\n        map((items: QueryList<DokuTab>) => items.toArray()),\n        startWith([]),\n        pairwise(),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(([prevItems, currentItems]) => {\n        this.updateTabs(prevItems, currentItems);\n        this.notifyTabChange(this.activeId);\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  /**\n   * Change the active tab.\n   * @param id Tab id.\n   */\n  changeActiveTab(id: string) {\n    this.updateActiveId(id, { emitTabChange: true });\n  }\n\n  private updateActiveId(id: string, options?: { emitTabChange?: boolean }) {\n    if (this.activeId === id) return;\n    this.activeId = id;\n    if (options?.emitTabChange) this.notifyTabChange(id);\n  }\n\n  private notifyTabChange(itemId: string) {\n    const tab = this.getTabById(itemId);\n    if (!tab) return;\n    this.activeTabChange.emit(tab);\n  }\n\n  private getTabById(id: string): DokuTab | null {\n    return this.tabItems?.find((tab) => tab.id === id) || null;\n  }\n\n  private isValidActiveId(id: string): boolean {\n    return !!this.tabItems?.some((tab) => tab.id === id);\n  }\n\n  /**\n   * Update tab elements when adding more or removing tab items.\n   */\n  private updateTabs(prevItems: DokuTab[], currentItems: DokuTab[]) {\n    const removedItems = prevItems.filter(\n      (prevItem) => !currentItems.some((currentItem) => prevItem.id === currentItem.id)\n    );\n    const newItems = currentItems.filter(\n      (currentItem) => !prevItems.some((prevItem) => currentItem.id === prevItem.id)\n    );\n\n    removedItems.forEach((item) => {\n      this.removeTabLabel(item);\n      this.removeTabContent(item);\n    });\n\n    newItems.forEach((item) => {\n      this.addTabLabel(item);\n      this.addTabContent(item);\n    });\n  }\n\n  private addTabLabel(item: DokuTab) {\n    if (!this.tabsHeaderElement || !item['labelTemplate']) return;\n    if (item['labelElement'] || item['labelViewRef']) return;\n\n    item['labelElement'] = ViewElement.createTabLabel({ renderer: this.renderer });\n    item['labelViewRef'] = ViewElement.appendLabelToTabsHeader({\n      applicationRef: this.appRef,\n      renderer: this.renderer,\n      headerElement: this.tabsHeaderElement,\n      labelElement: item['labelElement'],\n      labelTemplate: item['labelTemplate'],\n    });\n  }\n\n  private removeTabLabel(item: DokuTab) {\n    item['labelElement']?.remove();\n    item['labelViewRef']?.destroy();\n  }\n\n  private addTabContent(item: DokuTab) {\n    if (!this.tabsBodyElement || !item['contentTemplate']) return;\n    if (item['contentElement'] || item['contentViewRef']) return;\n\n    item['contentElement'] = ViewElement.createTabContent({ renderer: this.renderer });\n    item['contentViewRef'] = ViewElement.appendContentToTabsBody({\n      applicationRef: this.appRef,\n      renderer: this.renderer,\n      bodyElement: this.tabsBodyElement,\n      contentElement: item['contentElement'],\n      contentTemplate: item['contentTemplate'],\n      lazyLoad: item.lazyLoad,\n    });\n  }\n\n  private removeTabContent(item: DokuTab) {\n    item['contentElement']?.remove();\n    item['contentViewRef']?.destroy();\n  }\n}\n"]}