UNPKG

@doku-dev/doku-fragment

Version:

A new Angular UI library that moving away from Bootstrap and built from scratch.

176 lines 22.7 kB
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"]}