UNPKG

@doku-dev/doku-fragment

Version:

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

189 lines 26 kB
import { Directive, HostBinding, Inject, Input, Optional, } from '@angular/core'; import { ReplaySubject, filter, fromEvent, startWith, takeUntil } from 'rxjs'; import { DokuTable } from './table.directive'; import * as i0 from "@angular/core"; import * as i1 from "./table.directive"; export var DokuStickyColumnPosition; (function (DokuStickyColumnPosition) { DokuStickyColumnPosition["Left"] = "left"; DokuStickyColumnPosition["Right"] = "right"; })(DokuStickyColumnPosition || (DokuStickyColumnPosition = {})); export class DokuTableStickyColumn { constructor(elementRef, renderer, ngZone, dokuTable) { this.elementRef = elementRef; this.renderer = renderer; this.ngZone = ngZone; this.dokuTable = dokuTable; this.destroy$ = new ReplaySubject(); } get element() { return this.elementRef.nativeElement; } get isStickyLeft() { return this.position === DokuStickyColumnPosition.Left; } get isStickyRight() { return this.position === DokuStickyColumnPosition.Right; } ngOnChanges(changes) { if (changes['position']?.previousValue !== changes['position']?.currentValue) { this.ngZone.runOutsideAngular(() => { this.observer?.disconnect(); setTimeout(() => { if (this.isStickyRight) this.handleDynamicStyleRight(); if (this.isStickyLeft) this.handleDynamicStyleLeft(); }, 0); }); } } ngAfterViewInit() { // Giving delay to detect table container being initialized. setTimeout(() => { this.watchTableContainerScrollEvent(); }, 0); } ngOnDestroy() { this.observer?.disconnect(); this.destroy$.next(true); this.destroy$.complete(); } handleDynamicStyleRight() { // Get next sticky column const nextStickyColumn = this.element.nextElementSibling; if (!nextStickyColumn) return; if (!nextStickyColumn.classList.contains('d-table-sticky-column-right')) return; this.observer = new ResizeObserver(() => { // If has next sticky column, get the width of the column. const nextStickyColumnWidth = this.getSiblingElementClientWidth(nextStickyColumn, 'next'); // Set the start `right` point of the current sticky column // from the next sticky column width. const rightValue = nextStickyColumnWidth + 'px'; this.renderer.setStyle(this.element, 'right', rightValue); }); if (this.element.parentElement) { this.observer.observe(this.element.parentElement); } } handleDynamicStyleLeft() { // Get previous sticky column const prevStickyColumn = this.element.previousElementSibling; if (!prevStickyColumn) return; // if (prevStickyColumn.classList.contains('d-table-sticky-column-left')) return; this.observer = new ResizeObserver(() => { // If has previous sticky column, get the width of the column. const prevStickyColumnWidth = this.getSiblingElementClientWidth(prevStickyColumn, 'previous'); // Set the start `left` point of the current sticky column // from the prev sticky column width. const leftValue = prevStickyColumnWidth + 'px'; this.renderer.setStyle(this.element, 'left', leftValue); }); if (this.element.parentElement) { this.observer.observe(this.element.parentElement); } } getSiblingElementClientWidth(nextElement, siblingPosition) { const sibling = siblingPosition === 'previous' ? nextElement.previousElementSibling : nextElement.nextElementSibling; if (sibling) { const siblingClientWidth = this.getSiblingElementClientWidth(sibling, siblingPosition); return nextElement.getBoundingClientRect().width + siblingClientWidth; } return nextElement.getBoundingClientRect().width; } watchTableContainerScrollEvent() { this.ngZone.runOutsideAngular(() => { const tableElement = this.dokuTable?.['elementRef'].nativeElement; const tableContainerElement = this.dokuTable?.['containerElement']; if (!tableContainerElement || !tableElement) return; fromEvent(tableContainerElement, 'scroll') .pipe(startWith(new Event('scroll')), filter(() => !!this.position), takeUntil(this.destroy$)) .subscribe(() => { this.handleStickyLeftShadow(tableElement, tableContainerElement); this.handleStickyRightShadow(tableElement, tableContainerElement); }); }); } handleStickyLeftShadow(_, tableContainerElement) { if (!this.isLastStickyColumn()) return; if (this.position !== DokuStickyColumnPosition.Left) return; tableContainerElement.scrollLeft > 0 ? this.toggleShadowClass('add') : this.toggleShadowClass('remove'); } handleStickyRightShadow(tableElement, tableContainerElement) { if (!this.isLastStickyColumn()) return; if (this.position !== DokuStickyColumnPosition.Right) return; // Don't need to provide the shadow because the table element has no more columns to scroll. if (tableElement.clientWidth < tableContainerElement.clientWidth) return; const offsetWidth = tableElement.clientWidth - tableContainerElement.clientWidth; const offsetScroll = offsetWidth - tableContainerElement.scrollLeft; // Giving check offset to 0.5 because the client width value is rounded to nearest integer. offsetScroll > 0.5 ? this.toggleShadowClass('add') : this.toggleShadowClass('remove'); } /** * Check whether this column is the last sticky column. * * For the `left` position, will check if no sticky column after. * For the `right` position, will check if no sticky column before. */ isLastStickyColumn() { if (this.isStickyLeft) { const nextElement = this.element.nextElementSibling; if (nextElement?.classList.contains('d-table-sticky-column-left')) return false; return true; } if (this.isStickyRight) { const prevElement = this.element.previousElementSibling; if (prevElement?.classList.contains('d-table-sticky-column-right')) return false; return true; } return false; } toggleShadowClass(type) { if (type === 'add') { this.renderer.addClass(this.element, 'd-table-sticky-column-shadow'); } if (type === 'remove') { this.renderer.removeClass(this.element, 'd-table-sticky-column-shadow'); } } } DokuTableStickyColumn.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTableStickyColumn, deps: [{ token: i0.ElementRef }, { token: i0.Renderer2 }, { token: i0.NgZone }, { token: DokuTable, optional: true }], target: i0.ɵɵFactoryTarget.Directive }); DokuTableStickyColumn.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.9", type: DokuTableStickyColumn, isStandalone: true, selector: "th[doku-sticky-column], td[doku-sticky-column]", inputs: { position: ["doku-sticky-column", "position"] }, host: { properties: { "class.d-table-sticky-column-left": "this.isStickyLeft", "class.d-table-sticky-column-right": "this.isStickyRight" } }, exportAs: ["dokuStickyColumn"], usesOnChanges: true, ngImport: i0 }); i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTableStickyColumn, decorators: [{ type: Directive, args: [{ selector: 'th[doku-sticky-column], td[doku-sticky-column]', exportAs: 'dokuStickyColumn', standalone: true, }] }], ctorParameters: function () { return [{ type: i0.ElementRef }, { type: i0.Renderer2 }, { type: i0.NgZone }, { type: i1.DokuTable, decorators: [{ type: Optional }, { type: Inject, args: [DokuTable] }] }]; }, propDecorators: { position: [{ type: Input, args: ['doku-sticky-column'] }], isStickyLeft: [{ type: HostBinding, args: ['class.d-table-sticky-column-left'] }], isStickyRight: [{ type: HostBinding, args: ['class.d-table-sticky-column-right'] }] } }); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-sticky-column.directive.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/table/table-sticky-column.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Directive,\n  ElementRef,\n  HostBinding,\n  Inject,\n  Input,\n  NgZone,\n  OnChanges,\n  OnDestroy,\n  Optional,\n  Renderer2,\n  SimpleChanges,\n} from '@angular/core';\nimport { ReplaySubject, filter, fromEvent, startWith, takeUntil } from 'rxjs';\nimport { DokuTable } from './table.directive';\n\nexport enum DokuStickyColumnPosition {\n  Left = 'left',\n  Right = 'right',\n}\n\n@Directive({\n  selector: 'th[doku-sticky-column], td[doku-sticky-column]',\n  exportAs: 'dokuStickyColumn',\n  standalone: true,\n})\nexport class DokuTableStickyColumn implements OnChanges, OnDestroy, AfterViewInit {\n  @Input('doku-sticky-column') position?: DokuStickyColumnPosition;\n\n  private observer?: ResizeObserver;\n  private destroy$ = new ReplaySubject();\n\n  constructor(\n    private elementRef: ElementRef,\n    private renderer: Renderer2,\n    private ngZone: NgZone,\n    @Optional() @Inject(DokuTable) private dokuTable?: DokuTable\n  ) {}\n\n  protected get element(): HTMLElement {\n    return this.elementRef.nativeElement;\n  }\n\n  @HostBinding('class.d-table-sticky-column-left')\n  protected get isStickyLeft(): boolean {\n    return this.position === DokuStickyColumnPosition.Left;\n  }\n\n  @HostBinding('class.d-table-sticky-column-right')\n  protected get isStickyRight(): boolean {\n    return this.position === DokuStickyColumnPosition.Right;\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['position']?.previousValue !== changes['position']?.currentValue) {\n      this.ngZone.runOutsideAngular(() => {\n        this.observer?.disconnect();\n\n        setTimeout(() => {\n          if (this.isStickyRight) this.handleDynamicStyleRight();\n          if (this.isStickyLeft) this.handleDynamicStyleLeft();\n        }, 0);\n      });\n    }\n  }\n\n  ngAfterViewInit(): void {\n    // Giving delay to detect table container being initialized.\n    setTimeout(() => {\n      this.watchTableContainerScrollEvent();\n    }, 0);\n  }\n\n  ngOnDestroy(): void {\n    this.observer?.disconnect();\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  private handleDynamicStyleRight() {\n    // Get next sticky column\n    const nextStickyColumn = this.element.nextElementSibling;\n    if (!nextStickyColumn) return;\n    if (!nextStickyColumn.classList.contains('d-table-sticky-column-right')) return;\n\n    this.observer = new ResizeObserver(() => {\n      // If has next sticky column, get the width of the column.\n      const nextStickyColumnWidth = this.getSiblingElementClientWidth(nextStickyColumn, 'next');\n\n      // Set the start `right` point of the current sticky column\n      // from the next sticky column width.\n      const rightValue = nextStickyColumnWidth + 'px';\n      this.renderer.setStyle(this.element, 'right', rightValue);\n    });\n\n    if (this.element.parentElement) {\n      this.observer.observe(this.element.parentElement);\n    }\n  }\n\n  private handleDynamicStyleLeft() {\n    // Get previous sticky column\n    const prevStickyColumn = this.element.previousElementSibling;\n    if (!prevStickyColumn) return;\n    // if (prevStickyColumn.classList.contains('d-table-sticky-column-left')) return;\n\n    this.observer = new ResizeObserver(() => {\n      // If has previous sticky column, get the width of the column.\n      const prevStickyColumnWidth = this.getSiblingElementClientWidth(prevStickyColumn, 'previous');\n\n      // Set the start `left` point of the current sticky column\n      // from the prev sticky column width.\n      const leftValue = prevStickyColumnWidth + 'px';\n      this.renderer.setStyle(this.element, 'left', leftValue);\n    });\n\n    if (this.element.parentElement) {\n      this.observer.observe(this.element.parentElement);\n    }\n  }\n\n  private getSiblingElementClientWidth(\n    nextElement: Element,\n    siblingPosition: 'previous' | 'next'\n  ): number {\n    const sibling =\n      siblingPosition === 'previous'\n        ? (nextElement.previousElementSibling as Element)\n        : (nextElement.nextElementSibling as Element);\n\n    if (sibling) {\n      const siblingClientWidth = this.getSiblingElementClientWidth(sibling, siblingPosition);\n      return nextElement.getBoundingClientRect().width + siblingClientWidth;\n    }\n\n    return nextElement.getBoundingClientRect().width;\n  }\n\n  private watchTableContainerScrollEvent() {\n    this.ngZone.runOutsideAngular(() => {\n      const tableElement: HTMLTableElement = this.dokuTable?.['elementRef'].nativeElement;\n      const tableContainerElement = this.dokuTable?.['containerElement'];\n\n      if (!tableContainerElement || !tableElement) return;\n\n      fromEvent(tableContainerElement, 'scroll')\n        .pipe(\n          startWith(new Event('scroll')),\n          filter(() => !!this.position),\n          takeUntil(this.destroy$)\n        )\n        .subscribe(() => {\n          this.handleStickyLeftShadow(tableElement, tableContainerElement);\n          this.handleStickyRightShadow(tableElement, tableContainerElement);\n        });\n    });\n  }\n\n  private handleStickyLeftShadow(_: HTMLTableElement, tableContainerElement: HTMLElement) {\n    if (!this.isLastStickyColumn()) return;\n    if (this.position !== DokuStickyColumnPosition.Left) return;\n\n    tableContainerElement.scrollLeft > 0\n      ? this.toggleShadowClass('add')\n      : this.toggleShadowClass('remove');\n  }\n\n  private handleStickyRightShadow(\n    tableElement: HTMLTableElement,\n    tableContainerElement: HTMLElement\n  ) {\n    if (!this.isLastStickyColumn()) return;\n    if (this.position !== DokuStickyColumnPosition.Right) return;\n\n    // Don't need to provide the shadow because the table element has no more columns to scroll.\n    if (tableElement.clientWidth < tableContainerElement.clientWidth) return;\n\n    const offsetWidth = tableElement.clientWidth - tableContainerElement.clientWidth;\n    const offsetScroll = offsetWidth - tableContainerElement.scrollLeft;\n\n    // Giving check offset to 0.5 because the client width value is rounded to nearest integer.\n    offsetScroll > 0.5 ? this.toggleShadowClass('add') : this.toggleShadowClass('remove');\n  }\n\n  /**\n   * Check whether this column is the last sticky column.\n   *\n   * For the `left` position, will check if no sticky column after.\n   * For the `right` position, will check if no sticky column before.\n   */\n  private isLastStickyColumn(): boolean {\n    if (this.isStickyLeft) {\n      const nextElement = this.element.nextElementSibling;\n      if (nextElement?.classList.contains('d-table-sticky-column-left')) return false;\n      return true;\n    }\n\n    if (this.isStickyRight) {\n      const prevElement = this.element.previousElementSibling;\n      if (prevElement?.classList.contains('d-table-sticky-column-right')) return false;\n      return true;\n    }\n\n    return false;\n  }\n\n  private toggleShadowClass(type: 'add' | 'remove'): void {\n    if (type === 'add') {\n      this.renderer.addClass(this.element, 'd-table-sticky-column-shadow');\n    }\n\n    if (type === 'remove') {\n      this.renderer.removeClass(this.element, 'd-table-sticky-column-shadow');\n    }\n  }\n}\n"]}