UNPKG

@doku-dev/doku-fragment

Version:

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

249 lines 35.7 kB
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"]}