@doku-dev/doku-fragment
Version:
A new Angular UI library that moving away from Bootstrap and built from scratch.
124 lines • 18.1 kB
JavaScript
import { Directive, HostBinding, HostListener, Inject, Input, createComponent, } from '@angular/core';
import { BehaviorSubject, ReplaySubject, filter, takeUntil } from 'rxjs';
import { DokuSorterIcon } from './sorter-icon.component';
import { DokuTable } from './table.directive';
import * as i0 from "@angular/core";
import * as i1 from "./table.directive";
export class DokuTableSortable {
constructor(envInjector, appRef, elementRef, table) {
this.envInjector = envInjector;
this.appRef = appRef;
this.elementRef = elementRef;
this.table = table;
this.direction$ = new BehaviorSubject('');
this.destroy$ = new ReplaySubject();
}
get classes() {
return ['d-table-sortable'];
}
ngOnInit() {
this.table?.['sortChange$']
.pipe(filter(() => !!this.sortable), filter(() => !!this.sorterIconComponent), takeUntil(this.destroy$))
.subscribe(({ activeSorter }) => {
if (this.direction$.value !== '' && activeSorter?.column !== this.sortable) {
// Reset direction because the active sorter is on another column.
this.direction$.next('');
}
if (activeSorter?.column === this.sortable &&
this.direction$.value !== activeSorter?.direction) {
// Update direction from table active sorter
this.direction$.next(activeSorter?.direction || '');
}
});
this.direction$
.pipe(filter(() => !!this.sortable), filter(() => !!this.sorterIconComponent), takeUntil(this.destroy$))
.subscribe((direction) => {
// Update sorter icon based on active direction.
if (direction === 'ascending') {
this.sorterIconComponent?.setInput('colorAsc', 'currentColor');
this.sorterIconComponent?.setInput('colorDesc', 'none');
}
else if (direction === 'descending') {
this.sorterIconComponent?.setInput('colorAsc', 'none');
this.sorterIconComponent?.setInput('colorDesc', 'currentColor');
}
else {
this.sorterIconComponent?.setInput('colorAsc', 'currentColor');
this.sorterIconComponent?.setInput('colorDesc', 'currentColor');
}
});
}
ngOnChanges(changes) {
if (changes['sortable']) {
const sortable = changes['sortable'].currentValue;
sortable ? this.appendElement() : this.removeElement();
}
}
ngOnDestroy() {
this.removeElement();
this.destroy$.next(true);
this.destroy$.complete();
}
onClick() {
if (!this.sortable)
return;
this.direction$.next(this.nextDirection(this.direction$.value));
this.table?.['_sortChange'].emit({
activeSorter: {
column: this.sortable,
direction: this.direction$.value,
},
});
}
appendElement() {
if (this.sorterIconComponent)
return;
this.sorterIconComponent = this.createSorterIconComponent();
const cell = this.elementRef.nativeElement;
cell.appendChild(this.sorterIconComponent.location.nativeElement);
cell.classList.add('d-table-th-with-sorter');
}
removeElement() {
this.sorterIconComponent?.destroy();
this.sorterIconComponent = undefined;
const cell = this.elementRef.nativeElement;
cell.classList.remove('d-table-th-with-sorter');
}
createSorterIconComponent() {
const component = createComponent(DokuSorterIcon, { environmentInjector: this.envInjector });
component.location.nativeElement.className = 'd-table-sorter-icon';
this.appRef.attachView(component.hostView);
return component;
}
nextDirection(currentDirection) {
const flows = ['', 'ascending', 'descending'];
const flowIndex = flows.findIndex((flow) => flow === currentDirection);
let nextIndex = flowIndex + 1;
if (flowIndex < 0 || flowIndex >= flows.length - 1)
nextIndex = 0;
return flows[nextIndex];
}
}
DokuTableSortable.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTableSortable, deps: [{ token: i0.EnvironmentInjector }, { token: i0.ApplicationRef }, { token: i0.ElementRef }, { token: DokuTable }], target: i0.ɵɵFactoryTarget.Directive });
DokuTableSortable.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "15.2.9", type: DokuTableSortable, isStandalone: true, selector: "th[doku-sortable]", inputs: { sortable: ["doku-sortable", "sortable"] }, host: { listeners: { "click": "onClick()" }, properties: { "class": "this.classes" } }, exportAs: ["dokuSortable"], usesOnChanges: true, ngImport: i0 });
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DokuTableSortable, decorators: [{
type: Directive,
args: [{
selector: 'th[doku-sortable]',
exportAs: 'dokuSortable',
standalone: true,
}]
}], ctorParameters: function () { return [{ type: i0.EnvironmentInjector }, { type: i0.ApplicationRef }, { type: i0.ElementRef }, { type: i1.DokuTable, decorators: [{
type: Inject,
args: [DokuTable]
}] }]; }, propDecorators: { sortable: [{
type: Input,
args: ['doku-sortable']
}], classes: [{
type: HostBinding,
args: ['class']
}], onClick: [{
type: HostListener,
args: ['click']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-sortable.directive.js","sourceRoot":"","sources":["../../../../../../projects/doku-fragment/src/lib/table/table-sortable.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgClass } from '@angular/common';\nimport {\n  ApplicationRef,\n  ComponentRef,\n  Directive,\n  ElementRef,\n  EnvironmentInjector,\n  HostBinding,\n  HostListener,\n  Inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  SimpleChanges,\n  createComponent,\n} from '@angular/core';\nimport { BehaviorSubject, ReplaySubject, filter, takeUntil } from 'rxjs';\nimport { DokuSorterIcon } from './sorter-icon.component';\nimport { DokuTable } from './table.directive';\nimport { DokuActiveSorter } from './table.interface';\n\n@Directive({\n  selector: 'th[doku-sortable]',\n  exportAs: 'dokuSortable',\n  standalone: true,\n})\nexport class DokuTableSortable implements OnInit, OnChanges, OnDestroy {\n  /**\n   * Column name that will be sorted.\n   * @default undefined\n   */\n  @Input('doku-sortable') sortable?: string;\n\n  private direction$ = new BehaviorSubject<DokuActiveSorter['direction']>('');\n  private sorterIconComponent?: ComponentRef<DokuSorterIcon>;\n\n  private destroy$ = new ReplaySubject();\n\n  constructor(\n    private envInjector: EnvironmentInjector,\n    private appRef: ApplicationRef,\n    private elementRef: ElementRef,\n    @Inject(DokuTable) private table?: DokuTable\n  ) {}\n\n  @HostBinding('class')\n  protected get classes(): NgClass['ngClass'] {\n    return ['d-table-sortable'];\n  }\n\n  ngOnInit(): void {\n    this.table?.['sortChange$']\n      .pipe(\n        filter(() => !!this.sortable),\n        filter(() => !!this.sorterIconComponent),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(({ activeSorter }) => {\n        if (this.direction$.value !== '' && activeSorter?.column !== this.sortable) {\n          // Reset direction because the active sorter is on another column.\n          this.direction$.next('');\n        }\n\n        if (\n          activeSorter?.column === this.sortable &&\n          this.direction$.value !== activeSorter?.direction\n        ) {\n          // Update direction from table active sorter\n          this.direction$.next(activeSorter?.direction || '');\n        }\n      });\n\n    this.direction$\n      .pipe(\n        filter(() => !!this.sortable),\n        filter(() => !!this.sorterIconComponent),\n        takeUntil(this.destroy$)\n      )\n      .subscribe((direction) => {\n        // Update sorter icon based on active direction.\n        if (direction === 'ascending') {\n          this.sorterIconComponent?.setInput('colorAsc', 'currentColor');\n          this.sorterIconComponent?.setInput('colorDesc', 'none');\n        } else if (direction === 'descending') {\n          this.sorterIconComponent?.setInput('colorAsc', 'none');\n          this.sorterIconComponent?.setInput('colorDesc', 'currentColor');\n        } else {\n          this.sorterIconComponent?.setInput('colorAsc', 'currentColor');\n          this.sorterIconComponent?.setInput('colorDesc', 'currentColor');\n        }\n      });\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['sortable']) {\n      const sortable = changes['sortable'].currentValue;\n      sortable ? this.appendElement() : this.removeElement();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.removeElement();\n    this.destroy$.next(true);\n    this.destroy$.complete();\n  }\n\n  @HostListener('click')\n  protected onClick() {\n    if (!this.sortable) return;\n    this.direction$.next(this.nextDirection(this.direction$.value));\n    this.table?.['_sortChange'].emit({\n      activeSorter: {\n        column: this.sortable,\n        direction: this.direction$.value,\n      },\n    });\n  }\n\n  private appendElement() {\n    if (this.sorterIconComponent) return;\n    this.sorterIconComponent = this.createSorterIconComponent();\n    const cell = this.elementRef.nativeElement as HTMLTableCellElement;\n    cell.appendChild(this.sorterIconComponent.location.nativeElement);\n    cell.classList.add('d-table-th-with-sorter');\n  }\n\n  private removeElement() {\n    this.sorterIconComponent?.destroy();\n    this.sorterIconComponent = undefined;\n    const cell = this.elementRef.nativeElement as HTMLTableCellElement;\n    cell.classList.remove('d-table-th-with-sorter');\n  }\n\n  private createSorterIconComponent() {\n    const component = createComponent(DokuSorterIcon, { environmentInjector: this.envInjector });\n    (component.location.nativeElement as HTMLElement).className = 'd-table-sorter-icon';\n    this.appRef.attachView(component.hostView);\n    return component;\n  }\n\n  private nextDirection(\n    currentDirection: DokuActiveSorter['direction']\n  ): DokuActiveSorter['direction'] {\n    const flows: DokuActiveSorter['direction'][] = ['', 'ascending', 'descending'];\n    const flowIndex = flows.findIndex((flow) => flow === currentDirection);\n    let nextIndex: number = flowIndex + 1;\n    if (flowIndex < 0 || flowIndex >= flows.length - 1) nextIndex = 0;\n    return flows[nextIndex];\n  }\n}\n"]}