@doku-dev/doku-fragment
Version:
A new Angular UI library that moving away from Bootstrap and built from scratch.
249 lines • 35.7 kB
JavaScript
import { Directive, EventEmitter, HostBinding, Input, Output, } from '@angular/core';
import { ReplaySubject, distinctUntilChanged, fromEvent, shareReplay, takeUntil } from 'rxjs';
import { TemplateUtil } from './template.util';
import * as i0 from "@angular/core";
export class DokuTable {
constructor(elementRef, renderer, ngZone, envInjector, appRef, injector) {
this.elementRef = elementRef;
this.renderer = renderer;
this.ngZone = ngZone;
this.envInjector = envInjector;
this.appRef = appRef;
this.injector = injector;
/**
* Listen for column sort changes.
*/
this.sortChange = new EventEmitter();
this._sortChange = new EventEmitter();
this.sortChange$ = this._sortChange.asObservable().pipe(shareReplay());
this.destroy$ = new ReplaySubject();
}
get classes() {
return ['d-table'];
}
get totalColumn() {
return this.elementRef.nativeElement.querySelectorAll('thead tr th').length;
}
ngOnInit() {
this._sortChange
.asObservable()
.pipe(distinctUntilChanged((previous, current) => previous.activeSorter === current.activeSorter), takeUntil(this.destroy$))
.subscribe((props) => {
const emitEvent = props?.options?.emitEvent ?? true;
if (emitEvent)
this.sortChange.emit(props?.activeSorter);
});
}
ngOnDestroy() {
this.containerElement = undefined;
this.filledElement = undefined;
this.noDataElement = undefined;
this.noDataContentRef?.destroy();
this.loadingElement = undefined;
this.loadingContentRef?.destroy();
this.errorElement = undefined;
this.errorContentRef?.destroy();
this.destroy$.next(true);
this.destroy$.complete();
}
ngOnChanges(changes) {
const activeSorter = changes['activeSorter'];
if (activeSorter?.previousValue !== activeSorter?.currentValue) {
setTimeout(() => this._sortChange.emit({ activeSorter: activeSorter?.currentValue }), 0);
}
const status = changes['status'];
if (status?.previousValue !== status?.currentValue) {
this.handleBodyVisibilityByStatus();
}
const customNoData = changes['customNoData'];
if (customNoData?.previousValue !== customNoData?.currentValue) {
this.handleCustomNoData();
}
const customLoading = changes['customLoading'];
if (customLoading?.previousValue !== customLoading?.currentValue) {
this.handleCustomLoading();
}
const customError = changes['customError'];
if (customError?.previousValue !== customError?.currentValue) {
this.handleCustomError();
}
}
ngAfterViewInit() {
this.containerElement = this.createContainer();
const parentNode = this.elementRef.nativeElement.parentNode;
// Insert container element before the table element.
this.renderer.insertBefore(parentNode, this.containerElement, this.elementRef.nativeElement);
// Remove the table element.
this.renderer.removeChild(parentNode, this.elementRef.nativeElement);
// Append the table element inside the container element.
this.renderer.appendChild(this.containerElement, this.elementRef.nativeElement);
// Assign filled tbody to variable
this.filledElement = this.elementRef.nativeElement.querySelector('tbody');
// Append others tbody (no data, loading)
this.initNoDataElement();
this.initLoadingElement();
this.initErrorElement();
// Handle table body visibility
this.handleBodyVisibilityByStatus();
// Handle resize when window resized
this.onResize();
}
/**
* Change active sorter programmatically.
* @param activeSorter Value of the active sorter. Provide `undefined` to clear sorter.
* @param options.emitEvent Whether to emit an event for `sortChange`. Default is `true`.
*/
changeActiveSorter(activeSorter, options) {
this.activeSorter = activeSorter;
this._sortChange.emit({ activeSorter, options });
}
onResize() {
this.ngZone.runOutsideAngular(() => {
fromEvent(window, 'resize')
.pipe(takeUntil(this.destroy$))
.subscribe(() => {
this.resizeNoDataElementWidth();
this.resizeLoadingElementWidth();
this.resizeErrorElementWidth();
});
});
}
createContainer() {
const element = this.renderer.createElement('div');
this.renderer.addClass(element, 'd-table-container');
return element;
}
initNoDataElement() {
this.noDataElement = TemplateUtil.createNoDataElement({
renderer: this.renderer,
totalColumn: this.totalColumn,
});
this.renderer.insertBefore(this.elementRef.nativeElement, this.noDataElement.bodyElement, this.filledElement);
this.handleCustomNoData();
this.resizeNoDataElementWidth();
}
handleCustomNoData() {
if (!this.noDataElement)
return;
this.noDataContentRef?.destroy();
TemplateUtil.updateNoDataContent({
appRef: this.appRef,
noDataElement: this.noDataElement,
renderer: this.renderer,
content: this.customNoData,
injector: this.injector,
});
}
resizeNoDataElementWidth() {
if (!this.noDataElement)
return;
const containerWidth = this.containerElement?.clientWidth
? this.containerElement.clientWidth + 'px'
: '100%';
this.noDataElement.contentElement.style.width = containerWidth;
}
initLoadingElement() {
this.loadingElement = TemplateUtil.createLoadingElement({
renderer: this.renderer,
totalColumn: this.totalColumn,
applicationRef: this.appRef,
environmentInjector: this.envInjector,
});
this.renderer.insertBefore(this.elementRef.nativeElement, this.loadingElement.bodyElement, this.filledElement);
this.handleCustomLoading();
this.resizeLoadingElementWidth();
}
resizeLoadingElementWidth() {
if (!this.loadingElement)
return;
const containerWidth = this.containerElement?.clientWidth
? this.containerElement.clientWidth + 'px'
: '100%';
this.loadingElement.contentElement.style.width = containerWidth;
}
handleCustomLoading() {
if (!this.loadingElement)
return;
this.loadingContentRef?.destroy();
TemplateUtil.updateLoadingContent({
appRef: this.appRef,
envInjector: this.envInjector,
loadingElement: this.loadingElement,
renderer: this.renderer,
content: this.customLoading,
injector: this.injector,
});
}
initErrorElement() {
this.errorElement = TemplateUtil.createErrorElement({
renderer: this.renderer,
totalColumn: this.totalColumn,
});
this.renderer.insertBefore(this.elementRef.nativeElement, this.errorElement.bodyElement, this.filledElement);
this.handleCustomError();
this.resizeErrorElementWidth();
}
handleCustomError() {
if (!this.errorElement)
return;
this.errorContentRef?.destroy();
TemplateUtil.updateErrorContent({
appRef: this.appRef,
errorElement: this.errorElement,
renderer: this.renderer,
content: this.customError,
injector: this.injector,
});
}
resizeErrorElementWidth() {
if (!this.errorElement)
return;
const containerWidth = this.containerElement?.clientWidth
? this.containerElement.clientWidth + 'px'
: '100%';
this.errorElement.contentElement.style.width = containerWidth;
}
handleBodyVisibilityByStatus() {
if (!this.filledElement || !this.noDataElement || !this.loadingElement || !this.errorElement) {
return;
}
this.filledElement.style.display = 'none';
this.noDataElement.bodyElement.style.display = 'none';
this.loadingElement.bodyElement.style.display = 'none';
this.errorElement.bodyElement.style.display = 'none';
if (this.status === 'loading')
this.loadingElement.bodyElement.style.display = '';
if (this.status === 'empty')
this.noDataElement.bodyElement.style.display = '';
if (this.status === 'error')
this.errorElement.bodyElement.style.display = '';
if (!this.status)
this.filledElement.style.display = '';
}
}
DokuTable.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTable, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.NgZone }, { token: i0.EnvironmentInjector }, { token: i0.ApplicationRef }, { token: i0.Injector }], target: i0.ɵɵFactoryTarget.Directive });
DokuTable.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.9", type: DokuTable, isStandalone: true, selector: "[doku-table]", inputs: { activeSorter: "activeSorter", status: "status", customNoData: "customNoData", customLoading: "customLoading", customError: "customError" }, outputs: { sortChange: "sortChange" }, host: { properties: { "class": "this.classes" } }, exportAs: ["dokuTable"], usesOnChanges: true, ngImport: i0 });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTable, decorators: [{
type: Directive,
args: [{
selector: '[doku-table]',
exportAs: 'dokuTable',
standalone: true,
}]
}], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.NgZone }, { type: i0.EnvironmentInjector }, { type: i0.ApplicationRef }, { type: i0.Injector }]; }, propDecorators: { activeSorter: [{
type: Input
}], status: [{
type: Input
}], customNoData: [{
type: Input
}], customLoading: [{
type: Input
}], customError: [{
type: Input
}], sortChange: [{
type: Output
}], classes: [{
type: HostBinding,
args: ['class']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.directive.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/table/table.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgClass } from '@angular/common';\nimport {\n  AfterViewInit,\n  ApplicationRef,\n  Directive,\n  ElementRef,\n  EnvironmentInjector,\n  EventEmitter,\n  HostBinding,\n  Injector,\n  Input,\n  NgZone,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  Renderer2,\n  SimpleChanges,\n  TemplateRef,\n} from '@angular/core';\nimport { ReplaySubject, distinctUntilChanged, fromEvent, shareReplay, takeUntil } from 'rxjs';\nimport { DokuActiveSorter } from './table.interface';\nimport { TemplateUtil } from './template.util';\n\n@Directive({\n  selector: '[doku-table]',\n  exportAs: 'dokuTable',\n  standalone: true,\n})\nexport class DokuTable implements AfterViewInit, OnChanges, OnDestroy, OnInit {\n  /**\n   * Initial active sorter for column.\n   * @default undefined\n   */\n  @Input() activeSorter?: DokuActiveSorter;\n\n  /**\n   * The status of the data.\n   *\n   * - `loading`, will show loading content on the table body\n   * - `empty`, will show no data content on the table body\n   * - `error`, will show error content on the table body\n   *\n   * If status is `undefined`, it means data is provided.\n   *\n   * @default undefined\n   */\n  @Input() status?: 'loading' | 'empty' | 'error';\n\n  /**\n   * Custom text or template on the table body when status is `empty`.\n   *\n   * @default undefined\n   */\n  @Input() customNoData?: string | TemplateRef<unknown>;\n\n  /**\n   * Custom text or template on the table body when status is `loading`.\n   *\n   * @default undefined\n   */\n  @Input() customLoading?: string | TemplateRef<unknown>;\n\n  /**\n   * Custom text or template on the table body when status is `error`.\n   */\n  @Input() customError?: string | TemplateRef<unknown>;\n\n  /**\n   * Listen for column sort changes.\n   */\n  @Output()\n  sortChange = new EventEmitter<DokuActiveSorter | undefined>();\n\n  protected _sortChange = new EventEmitter<{\n    activeSorter?: DokuActiveSorter | undefined;\n    options?: { emitEvent?: boolean };\n  }>();\n  protected sortChange$ = this._sortChange.asObservable().pipe(shareReplay());\n\n  private containerElement?: HTMLDivElement;\n\n  private filledElement?: HTMLTableSectionElement;\n  private noDataElement?: ReturnType<typeof TemplateUtil.createNoDataElement>;\n  private noDataContentRef?: ReturnType<typeof TemplateUtil.updateNoDataContent>;\n  private loadingElement?: ReturnType<typeof TemplateUtil.createLoadingElement>;\n  private loadingContentRef?: ReturnType<typeof TemplateUtil.updateLoadingContent>;\n  private errorElement?: ReturnType<typeof TemplateUtil.createErrorElement>;\n  private errorContentRef?: ReturnType<typeof TemplateUtil.updateErrorContent>;\n\n  private destroy$ = new ReplaySubject();\n\n  constructor(\n    private elementRef: ElementRef,\n    private renderer: Renderer2,\n    private ngZone: NgZone,\n    private envInjector: EnvironmentInjector,\n    private appRef: ApplicationRef,\n    private injector: Injector\n  ) {}\n\n  @HostBinding('class')\n  protected get classes(): NgClass['ngClass'] {\n    return ['d-table'];\n  }\n\n  protected get totalColumn(): number {\n    return (this.elementRef.nativeElement as HTMLElement).querySelectorAll('thead tr th').length;\n  }\n\n  ngOnInit(): void {\n    this._sortChange\n      .asObservable()\n      .pipe(\n        distinctUntilChanged((previous, current) => previous.activeSorter === current.activeSorter),\n        takeUntil(this.destroy$)\n      )\n      .subscribe((props) => {\n        const emitEvent = props?.options?.emitEvent ?? true;\n        if (emitEvent) this.sortChange.emit(props?.activeSorter);\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.containerElement = undefined;\n    this.filledElement = undefined;\n    this.noDataElement = undefined;\n    this.noDataContentRef?.destroy();\n    this.loadingElement = undefined;\n    this.loadingContentRef?.destroy();\n    this.errorElement = undefined;\n    this.errorContentRef?.destroy();\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    const activeSorter = changes['activeSorter'];\n    if (activeSorter?.previousValue !== activeSorter?.currentValue) {\n      setTimeout(() => this._sortChange.emit({ activeSorter: activeSorter?.currentValue }), 0);\n    }\n\n    const status = changes['status'];\n    if (status?.previousValue !== status?.currentValue) {\n      this.handleBodyVisibilityByStatus();\n    }\n\n    const customNoData = changes['customNoData'];\n    if (customNoData?.previousValue !== customNoData?.currentValue) {\n      this.handleCustomNoData();\n    }\n\n    const customLoading = changes['customLoading'];\n    if (customLoading?.previousValue !== customLoading?.currentValue) {\n      this.handleCustomLoading();\n    }\n\n    const customError = changes['customError'];\n    if (customError?.previousValue !== customError?.currentValue) {\n      this.handleCustomError();\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.containerElement = this.createContainer();\n    const parentNode = (this.elementRef.nativeElement as HTMLElement).parentNode;\n\n    // Insert container element before the table element.\n    this.renderer.insertBefore(parentNode, this.containerElement, this.elementRef.nativeElement);\n\n    // Remove the table element.\n    this.renderer.removeChild(parentNode, this.elementRef.nativeElement);\n\n    // Append the table element inside the container element.\n    this.renderer.appendChild(this.containerElement, this.elementRef.nativeElement);\n\n    // Assign filled tbody to variable\n    this.filledElement = this.elementRef.nativeElement.querySelector('tbody');\n\n    // Append others tbody (no data, loading)\n    this.initNoDataElement();\n    this.initLoadingElement();\n    this.initErrorElement();\n\n    // Handle table body visibility\n    this.handleBodyVisibilityByStatus();\n\n    // Handle resize when window resized\n    this.onResize();\n  }\n\n  /**\n   * Change active sorter programmatically.\n   * @param activeSorter Value of the active sorter. Provide `undefined` to clear sorter.\n   * @param options.emitEvent Whether to emit an event for `sortChange`. Default is `true`.\n   */\n  changeActiveSorter(activeSorter?: DokuActiveSorter, options?: { emitEvent?: boolean }) {\n    this.activeSorter = activeSorter;\n    this._sortChange.emit({ activeSorter, options });\n  }\n\n  private onResize() {\n    this.ngZone.runOutsideAngular(() => {\n      fromEvent(window, 'resize')\n        .pipe(takeUntil(this.destroy$))\n        .subscribe(() => {\n          this.resizeNoDataElementWidth();\n          this.resizeLoadingElementWidth();\n          this.resizeErrorElementWidth();\n        });\n    });\n  }\n\n  private createContainer() {\n    const element: HTMLDivElement = this.renderer.createElement('div');\n    this.renderer.addClass(element, 'd-table-container');\n    return element;\n  }\n\n  private initNoDataElement() {\n    this.noDataElement = TemplateUtil.createNoDataElement({\n      renderer: this.renderer,\n      totalColumn: this.totalColumn,\n    });\n    this.renderer.insertBefore(\n      this.elementRef.nativeElement,\n      this.noDataElement.bodyElement,\n      this.filledElement\n    );\n    this.handleCustomNoData();\n    this.resizeNoDataElementWidth();\n  }\n\n  private handleCustomNoData() {\n    if (!this.noDataElement) return;\n    this.noDataContentRef?.destroy();\n    TemplateUtil.updateNoDataContent({\n      appRef: this.appRef,\n      noDataElement: this.noDataElement,\n      renderer: this.renderer,\n      content: this.customNoData,\n      injector: this.injector,\n    });\n  }\n\n  private resizeNoDataElementWidth() {\n    if (!this.noDataElement) return;\n    const containerWidth = this.containerElement?.clientWidth\n      ? this.containerElement.clientWidth + 'px'\n      : '100%';\n    this.noDataElement.contentElement.style.width = containerWidth;\n  }\n\n  private initLoadingElement() {\n    this.loadingElement = TemplateUtil.createLoadingElement({\n      renderer: this.renderer,\n      totalColumn: this.totalColumn,\n      applicationRef: this.appRef,\n      environmentInjector: this.envInjector,\n    });\n    this.renderer.insertBefore(\n      this.elementRef.nativeElement,\n      this.loadingElement.bodyElement,\n      this.filledElement\n    );\n    this.handleCustomLoading();\n    this.resizeLoadingElementWidth();\n  }\n\n  private resizeLoadingElementWidth() {\n    if (!this.loadingElement) return;\n    const containerWidth = this.containerElement?.clientWidth\n      ? this.containerElement.clientWidth + 'px'\n      : '100%';\n    this.loadingElement.contentElement.style.width = containerWidth;\n  }\n\n  private handleCustomLoading() {\n    if (!this.loadingElement) return;\n    this.loadingContentRef?.destroy();\n    TemplateUtil.updateLoadingContent({\n      appRef: this.appRef,\n      envInjector: this.envInjector,\n      loadingElement: this.loadingElement,\n      renderer: this.renderer,\n      content: this.customLoading,\n      injector: this.injector,\n    });\n  }\n\n  private initErrorElement() {\n    this.errorElement = TemplateUtil.createErrorElement({\n      renderer: this.renderer,\n      totalColumn: this.totalColumn,\n    });\n    this.renderer.insertBefore(\n      this.elementRef.nativeElement,\n      this.errorElement.bodyElement,\n      this.filledElement\n    );\n    this.handleCustomError();\n    this.resizeErrorElementWidth();\n  }\n\n  private handleCustomError() {\n    if (!this.errorElement) return;\n    this.errorContentRef?.destroy();\n    TemplateUtil.updateErrorContent({\n      appRef: this.appRef,\n      errorElement: this.errorElement,\n      renderer: this.renderer,\n      content: this.customError,\n      injector: this.injector,\n    });\n  }\n\n  private resizeErrorElementWidth() {\n    if (!this.errorElement) return;\n    const containerWidth = this.containerElement?.clientWidth\n      ? this.containerElement.clientWidth + 'px'\n      : '100%';\n    this.errorElement.contentElement.style.width = containerWidth;\n  }\n\n  private handleBodyVisibilityByStatus() {\n    if (!this.filledElement || !this.noDataElement || !this.loadingElement || !this.errorElement) {\n      return;\n    }\n\n    this.filledElement.style.display = 'none';\n    this.noDataElement.bodyElement.style.display = 'none';\n    this.loadingElement.bodyElement.style.display = 'none';\n    this.errorElement.bodyElement.style.display = 'none';\n\n    if (this.status === 'loading') this.loadingElement.bodyElement.style.display = '';\n    if (this.status === 'empty') this.noDataElement.bodyElement.style.display = '';\n    if (this.status === 'error') this.errorElement.bodyElement.style.display = '';\n    if (!this.status) this.filledElement.style.display = '';\n  }\n}\n"]}