@doku-dev/doku-fragment
Version:
A new Angular UI library that moving away from Bootstrap and built from scratch.
139 lines • 16.5 kB
JavaScript
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"]}