UNPKG

@doku-dev/doku-fragment

Version:

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

139 lines 16.5 kB
import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, ContentChild, Inject, Input, ViewEncapsulation, } from '@angular/core'; import { ReplaySubject, delay, filter, fromEvent, of, switchMap, takeUntil } from 'rxjs'; import { DokuTabContent } from './tab-content.directive'; import { DokuTabLabel } from './tab-label.directive'; import { DOKU_TAB, DOKU_TABS } from './token'; import { ViewElement } from './view-element'; import * as i0 from "@angular/core"; import * as i1 from "./tabs.component"; let nextId = 1; export class DokuTab { constructor(appRef, renderer, tabs) { this.appRef = appRef; this.renderer = renderer; this.tabs = tabs; /** * Unique id of the tab. * Default value is auto-generated. * * @default 'd-tab-[nextId]'' */ this.id = `d-tab-${nextId++}`; /** * Whether the tab is disabled. * * The tab content is not going to be loaded on disabled tab. * * @default false */ this.disabled = false; /** * Whether the tab content is lazy loaded which means * the content will be loaded when the tab is active and * destroyed when away from the tab. * * @default false */ this.lazyLoad = false; this.destroy$ = new ReplaySubject(); } ngAfterContentInit() { this.tabs?.['tabChangeForChild$'].pipe(takeUntil(this.destroy$)).subscribe((activeItem) => { this.handleActiveState(activeItem); this.handleDisabledState(); this.handleLazyLoaded(activeItem); }); of(true) .pipe(delay(100), // eslint-disable-next-line @typescript-eslint/no-non-null-assertion switchMap(() => fromEvent(this.labelElement, 'click')), filter(() => !this.disabled), takeUntil(this.destroy$)) .subscribe(() => { this.tabs?.changeActiveTab(this.id); }); } ngOnDestroy() { this.destroy$.next(true); this.destroy$.complete(); } handleActiveState(activeItem) { if (!this.labelElement || !this.contentElement) return; if (activeItem?.id === this.id) { this.renderer.addClass(this.labelElement, 'active'); this.renderer.addClass(this.contentElement, 'active'); } else { this.renderer.removeClass(this.labelElement, 'active'); this.renderer.removeClass(this.contentElement, 'active'); } } handleDisabledState() { if (!this.labelElement || !this.contentElement) return; if (this.disabled) { this.renderer.addClass(this.labelElement, 'disabled'); this.renderer.addClass(this.contentElement, 'disabled'); } else { this.renderer.removeClass(this.labelElement, 'disabled'); this.renderer.removeClass(this.contentElement, 'disabled'); } } handleLazyLoaded(activeItem) { if (!this.lazyLoad) return; if (activeItem?.id === this.id && this.contentTemplate && this.contentElement) { this.contentViewRef = ViewElement.createContentViewRefAndAppend({ applicationRef: this.appRef, renderer: this.renderer, contentElement: this.contentElement, contentTemplate: this.contentTemplate, }); } else { this.contentViewRef?.destroy(); } } } DokuTab.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTab, deps: [{ token: i0.ApplicationRef }, { token: i0.Renderer2 }, { token: DOKU_TABS }], target: i0.ɵɵFactoryTarget.Component }); DokuTab.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DokuTab, isStandalone: true, selector: "doku-tab", inputs: { id: "id", disabled: "disabled", lazyLoad: "lazyLoad" }, providers: [ { provide: DOKU_TAB, useExisting: DokuTab, }, ], queries: [{ propertyName: "labelTemplate", first: true, predicate: DokuTabLabel }, { propertyName: "contentTemplate", first: true, predicate: DokuTabContent }], exportAs: ["dokuTab"], 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: DokuTab, decorators: [{ type: Component, args: [{ selector: 'doku-tab', exportAs: 'dokuTab', standalone: true, imports: [CommonModule], template: '', encapsulation: ViewEncapsulation.None, changeDetection: ChangeDetectionStrategy.OnPush, providers: [ { provide: DOKU_TAB, useExisting: DokuTab, }, ], }] }], ctorParameters: function () { return [{ type: i0.ApplicationRef }, { type: i0.Renderer2 }, { type: i1.DokuTabs, decorators: [{ type: Inject, args: [DOKU_TABS] }] }]; }, propDecorators: { id: [{ type: Input }], disabled: [{ type: Input }], lazyLoad: [{ type: Input }], labelTemplate: [{ type: ContentChild, args: [DokuTabLabel, { descendants: false }] }], contentTemplate: [{ type: ContentChild, args: [DokuTabContent, { descendants: false }] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"tab.component.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/tabs/tab.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  AfterContentInit,\n  ApplicationRef,\n  ChangeDetectionStrategy,\n  Component,\n  ContentChild,\n  EmbeddedViewRef,\n  Inject,\n  Input,\n  OnDestroy,\n  Renderer2,\n  ViewEncapsulation,\n} from '@angular/core';\nimport { ReplaySubject, delay, filter, fromEvent, of, switchMap, takeUntil } from 'rxjs';\nimport { DokuTabContent } from './tab-content.directive';\nimport { DokuTabLabel } from './tab-label.directive';\nimport { DokuTabs } from './tabs.component';\nimport { DOKU_TAB, DOKU_TABS } from './token';\nimport { ViewElement } from './view-element';\n\nlet nextId = 1;\n\n@Component({\n  selector: 'doku-tab',\n  exportAs: 'dokuTab',\n  standalone: true,\n  imports: [CommonModule],\n  template: '',\n  encapsulation: ViewEncapsulation.None,\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  providers: [\n    {\n      provide: DOKU_TAB,\n      useExisting: DokuTab,\n    },\n  ],\n})\nexport class DokuTab implements OnDestroy, AfterContentInit {\n  /**\n   * Unique id of the tab.\n   * Default value is auto-generated.\n   *\n   * @default 'd-tab-[nextId]''\n   */\n  @Input() id = `d-tab-${nextId++}`;\n\n  /**\n   * Whether the tab is disabled.\n   *\n   * The tab content is not going to be loaded on disabled tab.\n   *\n   * @default false\n   */\n  @Input() disabled = false;\n\n  /**\n   * Whether the tab content is lazy loaded which means\n   * the content will be loaded when the tab is active and\n   * destroyed when away from the tab.\n   *\n   * @default false\n   */\n  @Input() lazyLoad = false;\n\n  protected labelElement?: HTMLDivElement;\n  protected labelViewRef?: EmbeddedViewRef<any>;\n  protected contentElement?: HTMLDivElement;\n  protected contentViewRef?: EmbeddedViewRef<any>;\n\n  @ContentChild(DokuTabLabel, { descendants: false }) protected labelTemplate?: DokuTabLabel;\n  @ContentChild(DokuTabContent, { descendants: false }) protected contentTemplate?: DokuTabContent;\n\n  private destroy$ = new ReplaySubject();\n\n  constructor(\n    private appRef: ApplicationRef,\n    private renderer: Renderer2,\n    @Inject(DOKU_TABS) private tabs?: DokuTabs\n  ) {}\n\n  ngAfterContentInit(): void {\n    this.tabs?.['tabChangeForChild$'].pipe(takeUntil(this.destroy$)).subscribe((activeItem) => {\n      this.handleActiveState(activeItem);\n      this.handleDisabledState();\n      this.handleLazyLoaded(activeItem);\n    });\n\n    of(true)\n      .pipe(\n        delay(100),\n        // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n        switchMap(() => fromEvent(this.labelElement!, 'click')),\n        filter(() => !this.disabled),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(() => {\n        this.tabs?.changeActiveTab(this.id);\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  private handleActiveState(activeItem: DokuTab) {\n    if (!this.labelElement || !this.contentElement) return;\n\n    if (activeItem?.id === this.id) {\n      this.renderer.addClass(this.labelElement, 'active');\n      this.renderer.addClass(this.contentElement, 'active');\n    } else {\n      this.renderer.removeClass(this.labelElement, 'active');\n      this.renderer.removeClass(this.contentElement, 'active');\n    }\n  }\n\n  private handleDisabledState() {\n    if (!this.labelElement || !this.contentElement) return;\n\n    if (this.disabled) {\n      this.renderer.addClass(this.labelElement, 'disabled');\n      this.renderer.addClass(this.contentElement, 'disabled');\n    } else {\n      this.renderer.removeClass(this.labelElement, 'disabled');\n      this.renderer.removeClass(this.contentElement, 'disabled');\n    }\n  }\n\n  private handleLazyLoaded(activeItem: DokuTab) {\n    if (!this.lazyLoad) return;\n    if (activeItem?.id === this.id && this.contentTemplate && this.contentElement) {\n      this.contentViewRef = ViewElement.createContentViewRefAndAppend({\n        applicationRef: this.appRef,\n        renderer: this.renderer,\n        contentElement: this.contentElement,\n        contentTemplate: this.contentTemplate,\n      });\n    } else {\n      this.contentViewRef?.destroy();\n    }\n  }\n}\n"]}