From a0de6796d167abdb57c0a646a836976ba53cf4ff Mon Sep 17 00:00:00 2001 From: Chintan Kavathia Date: Tue, 12 Nov 2024 15:43:40 +0530 Subject: [PATCH 1/2] refactor: header component removed unnecessary transform calculations for scrolling header along with body scroll. --- .../lib/components/datatable.component.html | 1 - .../src/lib/components/datatable.component.ts | 10 ++- .../lib/components/header/header.component.ts | 85 ++++--------------- 3 files changed, 23 insertions(+), 73 deletions(-) diff --git a/projects/ngx-datatable/src/lib/components/datatable.component.html b/projects/ngx-datatable/src/lib/components/datatable.component.html index d51309697..ebdaaa0ef 100644 --- a/projects/ngx-datatable/src/lib/components/datatable.component.html +++ b/projects/ngx-datatable/src/lib/components/datatable.component.html @@ -7,7 +7,6 @@ [sortType]="sortType" [scrollbarH]="scrollbarH" [innerWidth]="_innerWidth" - [offsetX]="_offsetX | async" [dealsWithGroup]="groupedRows !== undefined" [columns]="_internalColumns" [headerHeight]="headerHeight" diff --git a/projects/ngx-datatable/src/lib/components/datatable.component.ts b/projects/ngx-datatable/src/lib/components/datatable.component.ts index 361bcf498..68d04599c 100644 --- a/projects/ngx-datatable/src/lib/components/datatable.component.ts +++ b/projects/ngx-datatable/src/lib/components/datatable.component.ts @@ -663,6 +663,9 @@ export class DatatableComponent @ViewChild(DataTableHeaderComponent) headerComponent: DataTableHeaderComponent; + @ViewChild(DataTableHeaderComponent, { read: ElementRef }) + headerElement: ElementRef; + @ViewChild(DataTableBodyComponent, { read: ElementRef }) private bodyElement: ElementRef; @ContentChild(DatatableRowDefDirective, { @@ -939,7 +942,7 @@ export class DatatableComponent this.bodyComponent.cd.markForCheck(); } - if (this.headerComponent && this.headerComponent._columnGroupWidths.total !== width) { + if (this.headerComponent && this.headerComponent._columnGroupWidths().total !== width) { this.headerComponent.columns = [...this._internalColumns]; } @@ -1004,7 +1007,10 @@ export class DatatableComponent * The body triggered a scroll event. */ onBodyScroll(event: ScrollEvent): void { - this._offsetX.next(event.offsetX); + // if horizontal scroll is enabled we update the header scroll position + if (this.headerElement && this.scrollbarH && !isNaN(event.offsetX)) { + this.headerElement.nativeElement.scrollLeft = event.offsetX; + } this.scroll.emit(event); this.cd.detectChanges(); } diff --git a/projects/ngx-datatable/src/lib/components/header/header.component.ts b/projects/ngx-datatable/src/lib/components/header/header.component.ts index 651b631f1..f7e3be5e9 100644 --- a/projects/ngx-datatable/src/lib/components/header/header.component.ts +++ b/projects/ngx-datatable/src/lib/components/header/header.component.ts @@ -1,15 +1,12 @@ import { ChangeDetectionStrategy, - ChangeDetectorRef, Component, EventEmitter, HostBinding, inject, Input, - OnChanges, - OnDestroy, Output, - SimpleChanges, + signal, TemplateRef } from '@angular/core'; import { columnGroupWidths, columnsByPin, columnsByPinArr } from '../../utils/column'; @@ -23,10 +20,10 @@ import { SortPropDir, SortType } from '../../types/public.types'; -import { NgStyle } from '@angular/common'; import { ScrollbarHelper } from '../../services/scrollbar-helper.service'; import { TableColumn } from '../../types/table-column.type'; import { + ColumnGroupWidth, OrderableReorderEvent, PinnedColumns, TargetChangedEvent @@ -40,11 +37,14 @@ import { orderable (reorder)="onColumnReordered($event)" (targetChanged)="onTargetChanged($event)" - [style.width.px]="_columnGroupWidths.total" + [style.width.px]="_columnGroupWidths().total" class="datatable-header-inner" > @for (colGroup of _columnsByPin; track colGroup.type) { -
+
@for (column of colGroup.columns; track column.$$id) { { if (this._columns) { const colByPin = columnsByPin(this._columns); - this._columnGroupWidths = columnGroupWidths(colByPin, this._columns); - this.setStylesByGroup(); + this._columnGroupWidths.set(columnGroupWidths(colByPin, this._columns)); } }); } @@ -146,8 +144,7 @@ export class DataTableHeaderComponent implements OnDestroy, OnChanges { const colsByPin = columnsByPin(val); this._columnsByPin = columnsByPinArr(val); setTimeout(() => { - this._columnGroupWidths = columnGroupWidths(colsByPin, val); - this.setStylesByGroup(); + this._columnGroupWidths.set(columnGroupWidths(colsByPin, val)); }); } @@ -155,15 +152,6 @@ export class DataTableHeaderComponent implements OnDestroy, OnChanges { return this._columns; } - @Input() - set offsetX(val: number) { - this._offsetX = val; - this.setStylesByGroup(); - } - get offsetX() { - return this._offsetX; - } - @Output() sort: EventEmitter = new EventEmitter(); @Output() reorder: EventEmitter = new EventEmitter(); @Output() resize: EventEmitter = new EventEmitter(); @@ -172,33 +160,15 @@ export class DataTableHeaderComponent implements OnDestroy, OnChanges { @Output() columnContextmenu = new EventEmitter<{ event: MouseEvent; column: TableColumn }>(false); _columnsByPin: PinnedColumns[]; - _columnGroupWidths: any = { + _columnGroupWidths = signal({ + left: 0, + center: 0, + right: 0, total: 100 - }; + }); _innerWidth: number; - _offsetX: number; _columns: TableColumn[]; _headerHeight: string; - _styleByGroup: { - left: NgStyle['ngStyle']; - center: NgStyle['ngStyle']; - right: NgStyle['ngStyle']; - } = { left: {}, center: {}, right: {} }; - - private destroyed = false; - - ngOnChanges(changes: SimpleChanges): void { - if (changes.verticalScrollVisible) { - this._styleByGroup.right = this.calcStylesByGroup('right'); - if (!this.destroyed) { - this.cd.detectChanges(); - } - } - } - - ngOnDestroy(): void { - this.destroyed = true; - } onLongPressStart({ event, model }: { event: MouseEvent; model: TableColumn }) { model.dragging = true; @@ -344,29 +314,4 @@ export class DataTableHeaderComponent implements OnDestroy, OnChanges { return sorts; } - - setStylesByGroup() { - this._styleByGroup.left = this.calcStylesByGroup('left'); - this._styleByGroup.center = this.calcStylesByGroup('center'); - this._styleByGroup.right = this.calcStylesByGroup('right'); - if (!this.destroyed) { - this.cd.detectChanges(); - } - } - - calcStylesByGroup(group: 'center' | 'right' | 'left'): NgStyle['ngStyle'] { - const widths = this._columnGroupWidths; - - if (group === 'center') { - return { - transform: `translateX(${this.offsetX * -1}px)`, - width: `${widths[group]}px`, - willChange: 'transform' - }; - } - - return { - width: `${widths[group]}px` - }; - } } From e8184773acb247700f9b6233fedc6d39c4fd283e Mon Sep 17 00:00:00 2001 From: Maxi Date: Thu, 21 Nov 2024 09:43:48 +0100 Subject: [PATCH 2/2] fix: do not bind content as `innerHTML` by default (#126) BREAKING CHANGE: Previously, cell values were bound using `innerHTML`. With this change they are now bound using normal data binding. This means that any html markup will no longer be rendered. To restore the previous behavior set `bindAsUnsafeHtml` on columns where needed. We decided to change this behavior, as binding `innerHTML` can lead to HTML injection. Especially in table content which are often untrusted user generated content. BREAKING CHANGE: Header cell names are now bound using data binding instead of `innerHTML`. Use a `headerTemplate` to provide custom html markup. --- .../src/lib/components/body/body-cell.component.ts | 6 +++++- .../src/lib/components/columns/column.directive.ts | 1 + .../src/lib/components/header/header-cell.component.ts | 3 ++- projects/ngx-datatable/src/lib/types/table-column.type.ts | 8 ++++++++ 4 files changed, 16 insertions(+), 2 deletions(-) diff --git a/projects/ngx-datatable/src/lib/components/body/body-cell.component.ts b/projects/ngx-datatable/src/lib/components/body/body-cell.component.ts index ecc48a985..fe7cfb605 100644 --- a/projects/ngx-datatable/src/lib/components/body/body-cell.component.ts +++ b/projects/ngx-datatable/src/lib/components/body/body-cell.component.ts @@ -76,7 +76,11 @@ import { AsyncPipe, NgTemplateOutlet } from '@angular/common'; } @if (!column.cellTemplate) { - + @if (column.bindAsUnsafeHtml) { + + } @else { + {{ value }} + } } @else { implements TableColumn, OnChanges { private columnChangesService = inject(ColumnChangesService); @Input() name: string; @Input() prop: TableColumnProp; + @Input({ transform: booleanAttribute }) bindAsUnsafeHtml?: boolean; @Input({ transform: booleanAttribute }) frozenLeft: boolean; @Input({ transform: booleanAttribute }) frozenRight: boolean; @Input({ transform: numberAttribute }) flexGrow: number; diff --git a/projects/ngx-datatable/src/lib/components/header/header-cell.component.ts b/projects/ngx-datatable/src/lib/components/header/header-cell.component.ts index ad534bf39..9be783394 100644 --- a/projects/ngx-datatable/src/lib/components/header/header-cell.component.ts +++ b/projects/ngx-datatable/src/lib/components/header/header-cell.component.ts @@ -47,7 +47,8 @@ import { NgTemplateOutlet } from '@angular/common'; } @else { - + + {{ name }} } diff --git a/projects/ngx-datatable/src/lib/types/table-column.type.ts b/projects/ngx-datatable/src/lib/types/table-column.type.ts index a8d1505a9..c4d636a64 100644 --- a/projects/ngx-datatable/src/lib/types/table-column.type.ts +++ b/projects/ngx-datatable/src/lib/types/table-column.type.ts @@ -119,6 +119,14 @@ export interface TableColumn { */ prop?: TableColumnProp; + /** + * By default, the property is bound using normal data binding `{{content}}`. + * If this property is set to true, the property will be bound as ``. + * + * **DANGER** If enabling this feature, make sure the source of the data is trusted. This can be a vector for HTML injection attacks. + */ + bindAsUnsafeHtml?: boolean; + /** * Cell template ref */