@asadi/angular-date-components
Version:
`Angular Date Components` is a comprehensive angular library of date-related components designed to meet the needs of applications that require localization based on various calendar systems. While the package currently includes two powerful components (S
113 lines • 17.9 kB
JavaScript
import { Directive, EventEmitter, HostListener, Output } from '@angular/core';
import { BehaviorSubject } from 'rxjs';
import { ADCTableCell } from '../utils/table-cell.tools';
import { ADCTableRowTools } from '../utils/table-view-row.tools';
import * as i0 from "@angular/core";
export class TableViewControllerDirective {
onZoomChange(event) {
if (event.key == 'Shift' && !this.isZoomed) {
this.isZoomed = true;
this.focusIn();
}
else if (event.key == 'Shift' && this.isZoomed) {
this.isZoomed = false;
this.focusOut();
}
}
ngOnInit() {
this.childrenChangesObserver = new MutationObserver((changes) => {
if (this.scrollListener)
this.scrollListener();
const tableCells = [];
changes.filter(c => c.addedNodes[0] instanceof HTMLElement && c.addedNodes[0].classList.contains('row')).forEach(row => {
const children = Array().slice.call(row.addedNodes[0].children);
const cells = Array().slice.call(children[0].children);
cells.filter(item => item.getAttribute('selectable') == 'true').forEach(item => tableCells.push(new ADCTableCell(item)));
});
this.applyScrollingBehavior();
this.childrenChangesSubject.next(tableCells);
if (this.isZoomed) {
this.isZoomed = false;
this.focusOut();
}
this.viewReadyEvent.emit();
});
this.childrenChangesObserver.observe(this.container, { childList: true });
}
get container() {
return this.elRef.nativeElement;
}
get scrollableContainer() {
return this.container.parentElement;
}
constructor(renderer, elRef) {
this.renderer = renderer;
this.elRef = elRef;
this.isZoomed = false;
this.childrenChangesSubject = new BehaviorSubject([]);
this.viewReadyEvent = new EventEmitter();
this.scrollListener = null;
}
applyScrollingBehavior() {
this.scrollListener = this.renderer.listen(this.scrollableContainer, 'scroll', () => {
if (this.isZoomed) {
this.scrollableContainer.scrollLeft = 0;
this.scrollableContainer.scrollTop = 0;
return;
}
const stickyItems = this.container.querySelectorAll('.row');
Array().slice.call(stickyItems).forEach(item => {
Array().slice.call(item.children).filter(child => child.getAttribute('sticky') == 'true').forEach(stickyChild => {
stickyChild.style.transform = `translateX(-${this.scrollableContainer.scrollLeft}px)`;
});
});
});
}
focusIn() {
const childWidth = this.container.scrollWidth;
const parentWidth = this.scrollableContainer.clientWidth;
const childHeight = this.container.scrollHeight;
const parentHeight = this.scrollableContainer.clientHeight;
const scaleX = parentWidth / childWidth * 100;
const scaleY = parentHeight / childHeight * 100;
this.renderer.setStyle(this.container, 'transform', `scaleX(${scaleX}%) scaleY(${scaleY}%)`);
requestAnimationFrame(() => {
this.scrollableContainer.scrollLeft = 0;
this.scrollableContainer.scrollTop = 0;
});
}
focusOut() {
this.renderer.setStyle(this.container, 'transform', `scaleX(${100}%) scaleY(${100}%)`);
}
row(rowIndex) {
const allRows = [].slice.call(this.container.getElementsByClassName("row"));
const row = allRows.filter((r) => r.getAttribute('row-index') == rowIndex.toString())[0];
return new ADCTableRowTools(this.renderer, row);
}
cellChanges() {
return this.childrenChangesSubject.asObservable();
}
rowsCount() {
return [].slice.call(this.elRef.nativeElement.getElementsByClassName("row")).length;
}
ngOnDestroy() {
if (this.scrollListener)
this.scrollListener();
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.7", ngImport: i0, type: TableViewControllerDirective, deps: [{ token: i0.Renderer2 }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive }); }
static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.0.7", type: TableViewControllerDirective, isStandalone: true, selector: "[tableViewController]", outputs: { viewReadyEvent: "viewReady" }, host: { listeners: { "window:keydown": "onZoomChange($event)" } }, ngImport: i0 }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.7", ngImport: i0, type: TableViewControllerDirective, decorators: [{
type: Directive,
args: [{
selector: '[tableViewController]',
standalone: true,
}]
}], ctorParameters: () => [{ type: i0.Renderer2 }, { type: i0.ElementRef }], propDecorators: { viewReadyEvent: [{
type: Output,
args: ['viewReady']
}], onZoomChange: [{
type: HostListener,
args: ['window:keydown', ['$event']]
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-view-controller.directive.js","sourceRoot":"","sources":["../../../../../../../projects/asadi/angular-date-components/core/src/directives/table-view-controller.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import { Directive, ElementRef, EventEmitter, HostListener, OnDestroy, OnInit, Output, Renderer2 } from '@angular/core';\r\nimport { BehaviorSubject, Observable } from 'rxjs';\r\nimport { ADCITableViewCTRL } from '../interface';\r\nimport { ADCTableCell } from '../utils/table-cell.tools';\r\nimport { ADCTableRowTools } from '../utils/table-view-row.tools';\r\n\r\n@Directive({\r\n  selector: '[tableViewController]',\r\n  standalone: true,\r\n})\r\nexport class TableViewControllerDirective implements OnInit, ADCITableViewCTRL, OnDestroy{\r\n\r\n  private isZoomed = false;\r\n\r\n  private childrenChangesObserver!: MutationObserver;\r\n  private childrenChangesSubject = new BehaviorSubject<ADCTableCell[]>([]);\r\n\r\n  @Output('viewReady')\r\n  viewReadyEvent = new EventEmitter<void>();\r\n\r\n\r\n  @HostListener('window:keydown', ['$event'])\r\n  onZoomChange(event: KeyboardEvent): void\r\n  {\r\n    if(event.key == 'Shift' && !this.isZoomed)\r\n    {\r\n      this.isZoomed = true;\r\n      this.focusIn();\r\n    }\r\n    else if(event.key == 'Shift' && this.isZoomed)\r\n    {\r\n      this.isZoomed = false;\r\n      this.focusOut();\r\n    }\r\n  }\r\n\r\n  ngOnInit(): void \r\n  {\r\n    this.childrenChangesObserver = new MutationObserver((changes) => {\r\n\r\n      if(this.scrollListener)\r\n        this.scrollListener();\r\n\r\n      const tableCells: ADCTableCell[] = [];\r\n\r\n      changes.filter(c => c.addedNodes[0] instanceof HTMLElement && c.addedNodes[0].classList.contains('row')).forEach(row => {\r\n        const children = Array().slice.call((row.addedNodes[0] as HTMLElement).children) as HTMLElement[];\r\n\r\n        const cells = Array().slice.call(children[0].children) as HTMLElement[];\r\n\r\n        cells.filter(item => item.getAttribute('selectable') == 'true').forEach(item => tableCells.push(new ADCTableCell(item)));\r\n      });\r\n\r\n      this.applyScrollingBehavior();\r\n\r\n      this.childrenChangesSubject.next(tableCells);\r\n\r\n      if(this.isZoomed)\r\n      {\r\n        this.isZoomed = false;\r\n        this.focusOut();\r\n      }\r\n\r\n      this.viewReadyEvent.emit();\r\n    });\r\n\r\n    this.childrenChangesObserver.observe(this.container, {childList: true});\r\n  }\r\n\r\n  private get container(): HTMLElement\r\n  {\r\n    return this.elRef.nativeElement;\r\n  }\r\n\r\n  get scrollableContainer(): HTMLElement\r\n  {\r\n    return this.container.parentElement!;\r\n  }\r\n\r\n  private scrollListener: (() => void) | null = null;\r\n\r\n  constructor(\r\n    private renderer: Renderer2,\r\n    private elRef: ElementRef\r\n  ) {\r\n  }\r\n\r\n  private applyScrollingBehavior(): void\r\n  {\r\n    this.scrollListener = this.renderer.listen(this.scrollableContainer, 'scroll', () => {\r\n\r\n      if(this.isZoomed)\r\n      {\r\n        this.scrollableContainer.scrollLeft = 0;\r\n        this.scrollableContainer.scrollTop = 0;\r\n\r\n        return;\r\n      }\r\n\r\n      const stickyItems = this.container.querySelectorAll('.row');\r\n\r\n      (Array().slice.call(stickyItems) as HTMLElement[]).forEach(item => {\r\n\r\n        (Array().slice.call(item.children) as HTMLElement[]).filter(child => child.getAttribute('sticky') == 'true').forEach(stickyChild => {\r\n          stickyChild.style.transform = `translateX(-${this.scrollableContainer.scrollLeft}px)`\r\n        })\r\n      })\r\n\r\n\r\n    });\r\n  }\r\n\r\n  private focusIn(): void\r\n  {\r\n    const childWidth = this.container.scrollWidth;\r\n    const parentWidth = this.scrollableContainer.clientWidth;\r\n\r\n    const childHeight = this.container.scrollHeight;\r\n    const parentHeight = this.scrollableContainer.clientHeight;\r\n\r\n    const scaleX = parentWidth / childWidth * 100;\r\n    const scaleY = parentHeight / childHeight * 100;\r\n\r\n    this.renderer.setStyle(this.container, 'transform', `scaleX(${scaleX}%) scaleY(${scaleY}%)`);\r\n\r\n    requestAnimationFrame(() => {\r\n      this.scrollableContainer.scrollLeft = 0;\r\n      this.scrollableContainer.scrollTop = 0;\r\n    })\r\n  }\r\n\r\n  private focusOut(): void\r\n  {\r\n    this.renderer.setStyle(this.container, 'transform', `scaleX(${100}%) scaleY(${100}%)`);\r\n  }\r\n\r\n\r\n  row(rowIndex: number): ADCTableRowTools \r\n  {\r\n    const allRows = [].slice.call(this.container.getElementsByClassName(\"row\")) as HTMLElement[];\r\n    const row = allRows.filter((r) => r.getAttribute('row-index') == rowIndex.toString())[0];\r\n\r\n    return new ADCTableRowTools(this.renderer, row);\r\n  }\r\n\r\n  cellChanges(): Observable<ADCTableCell[]>\r\n  {\r\n    return this.childrenChangesSubject.asObservable();\r\n  }\r\n\r\n  rowsCount(): number \r\n  {\r\n    return [].slice.call(this.elRef.nativeElement.getElementsByClassName(\"row\")).length;\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    if(this.scrollListener) this.scrollListener();\r\n  }\r\n\r\n}\r\n"]}