ngx-simple-datatables 1.17.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -0,0 +1,483 @@
1
+ import { CommonModule, isPlatformBrowser } from "@angular/common";
2
+ import { ChangeDetectionStrategy, Component, Input, ViewChild, Inject, PLATFORM_ID, ContentChild, } from "@angular/core";
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "@angular/common";
5
+ export class NgxSimpleDatatableComponent {
6
+ constructor(cdr, platformId, ngZone) {
7
+ this.cdr = cdr;
8
+ this.platformId = platformId;
9
+ this.ngZone = ngZone;
10
+ this.columns = [];
11
+ this.data = [];
12
+ this.rowHeight = 40;
13
+ this.headerHeight = 50;
14
+ this.bufferSize = 10;
15
+ // Virtualization properties
16
+ this.visibleStartIndex = 0;
17
+ this.visibleEndIndex = 0;
18
+ this.visibleRows = [];
19
+ this.totalHeight = 0;
20
+ this.offsetY = 0;
21
+ this.containerHeight = 0;
22
+ // Column properties
23
+ this.leftFrozenColumns = [];
24
+ this.centerColumns = [];
25
+ this.rightFrozenColumns = [];
26
+ this.columnWidths = {};
27
+ // Sorting
28
+ this.sortState = { field: "", direction: null };
29
+ // Resizing
30
+ this.isResizing = false;
31
+ this.resizingColumn = "";
32
+ this.startX = 0;
33
+ this.startWidth = 0;
34
+ // Scroll listeners
35
+ this.scrollListener = null;
36
+ this.resizeListener = null;
37
+ this.storageKey = "ngx-simple-datatable-column-widths";
38
+ this.scrollRequestId = null;
39
+ this.resizeTimer = null;
40
+ this.mouseMoveHandler = null;
41
+ this.mouseUpHandler = null;
42
+ // Track last scroll time to prevent feedback loops
43
+ this.lastScrollTime = 0;
44
+ this.SCROLL_DEBOUNCE = 10; // ms
45
+ // Track scroll position for header transform
46
+ this.headerTransform = "translateX(0)";
47
+ // TrackBy functions for performance
48
+ this.trackByRowIndex = (index) => this.visibleStartIndex + index;
49
+ this.trackByColumnField = (index, column) => column.field;
50
+ this.isBrowser = isPlatformBrowser(this.platformId);
51
+ }
52
+ ngOnInit() {
53
+ this.initializeColumns();
54
+ if (this.isBrowser) {
55
+ this.setupEventListeners();
56
+ }
57
+ this.calculateVirtualization();
58
+ }
59
+ ngOnDestroy() {
60
+ if (this.isBrowser) {
61
+ this.removeEventListeners();
62
+ // Clean up any pending animations or timeouts
63
+ if (this.scrollRequestId) {
64
+ cancelAnimationFrame(this.scrollRequestId);
65
+ }
66
+ if (this.resizeTimer) {
67
+ clearTimeout(this.resizeTimer);
68
+ }
69
+ }
70
+ }
71
+ ngAfterContentInit() {
72
+ // if (!this.headerTemplate) {
73
+ // throw new Error("ngx-simple-datatable requires a headerTemplate.");
74
+ // }
75
+ // if (!this.cellTemplate) {
76
+ // throw new Error("ngx-simple-datatable requires a cellTemplate.");
77
+ // }
78
+ }
79
+ initializeColumns() {
80
+ this.leftFrozenColumns = this.columns.filter((col) => col.freeze === "left");
81
+ this.rightFrozenColumns = this.columns.filter((col) => col.freeze === "right");
82
+ this.centerColumns = this.columns.filter((col) => !col.freeze);
83
+ // Load saved column widths if available
84
+ const savedWidths = this.loadColumnWidths();
85
+ // Initialize column widths
86
+ this.columns.forEach((col) => {
87
+ if (col.width) {
88
+ this.columnWidths[col.field] = Number.parseInt(col.width.replace("px", ""));
89
+ }
90
+ else if (savedWidths && savedWidths[col.field]) {
91
+ this.columnWidths[col.field] = savedWidths[col.field];
92
+ }
93
+ else {
94
+ this.columnWidths[col.field] = 150; // default width
95
+ }
96
+ });
97
+ }
98
+ setupEventListeners() {
99
+ // Store references for cleanup
100
+ this.scrollListener = (event) => this.onBodyScroll(event);
101
+ this.resizeListener = () => this.onResize();
102
+ this.mouseMoveHandler = (event) => this.onMouseMove(event);
103
+ this.mouseUpHandler = (event) => this.onMouseUp();
104
+ // Add event listeners
105
+ if (this.isBrowser) {
106
+ // For the header scroll (horizontal only)
107
+ if (this.headerRow?.nativeElement) {
108
+ this.headerRow.nativeElement.addEventListener("scroll", () => this.onHeaderScroll(null));
109
+ }
110
+ // For the body scroll (both vertical and horizontal)
111
+ if (this.tableBody?.nativeElement && this.scrollListener) {
112
+ this.tableBody.nativeElement.addEventListener("scroll", this.scrollListener);
113
+ }
114
+ // Window resize and mouse events
115
+ if (this.resizeListener) {
116
+ window.addEventListener("resize", this.resizeListener);
117
+ }
118
+ if (this.mouseMoveHandler) {
119
+ document.addEventListener("mousemove", this.mouseMoveHandler);
120
+ }
121
+ if (this.mouseUpHandler) {
122
+ document.addEventListener("mouseup", this.mouseUpHandler);
123
+ }
124
+ }
125
+ }
126
+ removeEventListeners() {
127
+ if (this.isBrowser) {
128
+ // Clear any pending animations or timeouts
129
+ if (this.scrollRequestId) {
130
+ cancelAnimationFrame(this.scrollRequestId);
131
+ this.scrollRequestId = null;
132
+ }
133
+ if (this.resizeTimer) {
134
+ clearTimeout(this.resizeTimer);
135
+ this.resizeTimer = null;
136
+ }
137
+ // Reset all event listeners
138
+ this.scrollListener = null;
139
+ this.resizeListener = null;
140
+ this.mouseMoveHandler = null;
141
+ this.mouseUpHandler = null;
142
+ }
143
+ }
144
+ calculateVirtualization() {
145
+ if (!this.tableContainer || !this.tableBody)
146
+ return;
147
+ // Update container height based on available space
148
+ const container = this.tableBody.nativeElement;
149
+ this.containerHeight = container.clientHeight - this.headerHeight;
150
+ this.totalHeight = this.data.length * this.rowHeight;
151
+ // Get scroll position
152
+ const scrollTop = container.scrollTop;
153
+ // Calculate visible rows
154
+ const visibleRowCount = Math.ceil(this.containerHeight / this.rowHeight) + 1;
155
+ // Calculate start and end indices with buffer
156
+ this.visibleStartIndex = Math.max(0, Math.floor(scrollTop / this.rowHeight) - this.bufferSize);
157
+ this.visibleEndIndex = Math.min(this.data.length - 1, this.visibleStartIndex + visibleRowCount + this.bufferSize * 2);
158
+ // Update visible rows
159
+ this.visibleRows = this.data.slice(this.visibleStartIndex, this.visibleEndIndex + 1);
160
+ console.log(scrollTop, this.visibleStartIndex, this.visibleEndIndex);
161
+ // Update vertical offset for virtual scrolling
162
+ this.offsetY = this.visibleStartIndex * this.rowHeight;
163
+ // Trigger change detection in the next tick to avoid ExpressionChangedAfterItHasBeenCheckedError
164
+ Promise.resolve().then(() => {
165
+ this.cdr.detectChanges();
166
+ });
167
+ }
168
+ // Handle body scroll events
169
+ onBodyScroll(event) {
170
+ if (!event)
171
+ return;
172
+ const now = Date.now();
173
+ if (now - this.lastScrollTime < this.SCROLL_DEBOUNCE)
174
+ return;
175
+ this.lastScrollTime = now;
176
+ const target = event.target;
177
+ // Sync header scroll with body (horizontally)
178
+ // if (this.headerRow?.nativeElement) {
179
+ // this.headerRow.nativeElement.scrollLeft = target.scrollLeft;
180
+ this.syncHorizontalScroll(target.scrollLeft);
181
+ // }
182
+ // Handle virtualization (vertical scrolling)
183
+ if (this.scrollRequestId) {
184
+ cancelAnimationFrame(this.scrollRequestId);
185
+ }
186
+ this.scrollRequestId = requestAnimationFrame(() => {
187
+ this.calculateVirtualization();
188
+ this.scrollRequestId = null;
189
+ });
190
+ }
191
+ // Handle header scroll events
192
+ onHeaderScroll(event) {
193
+ if (!event)
194
+ return;
195
+ const now = Date.now();
196
+ if (now - this.lastScrollTime < this.SCROLL_DEBOUNCE)
197
+ return;
198
+ this.lastScrollTime = now;
199
+ // Sync body scroll with header (horizontally)
200
+ const scrollLeft = event.target.scrollLeft;
201
+ if (this.tableBody?.nativeElement) {
202
+ this.tableBody.nativeElement.scrollLeft = scrollLeft;
203
+ }
204
+ }
205
+ onResize() {
206
+ // Debounce resize events for better performance
207
+ if (this.resizeTimer) {
208
+ clearTimeout(this.resizeTimer);
209
+ }
210
+ this.resizeTimer = setTimeout(() => {
211
+ this.calculateVirtualization();
212
+ this.resizeTimer = null;
213
+ }, 50);
214
+ }
215
+ syncHorizontalScroll(scrollLeft) {
216
+ if (!this.tableContainer)
217
+ return;
218
+ // Update the transform value
219
+ this.headerTransform = `translateX(-${scrollLeft}px)`;
220
+ // Trigger change detection
221
+ this.cdr.detectChanges();
222
+ }
223
+ // Sorting functionality
224
+ onSort(column) {
225
+ if (!column.sortable)
226
+ return;
227
+ if (this.sortState.field === column.field) {
228
+ // Toggle sort direction
229
+ if (this.sortState.direction === "asc") {
230
+ this.sortState.direction = "desc";
231
+ }
232
+ else if (this.sortState.direction === "desc") {
233
+ this.sortState.direction = null;
234
+ this.sortState.field = "";
235
+ }
236
+ else {
237
+ this.sortState.direction = "asc";
238
+ }
239
+ }
240
+ else {
241
+ this.sortState.field = column.field;
242
+ this.sortState.direction = "asc";
243
+ }
244
+ this.applySorting();
245
+ }
246
+ applySorting() {
247
+ if (!this.sortState.direction || !this.sortState.field) {
248
+ // Reset to original order if no sorting
249
+ return;
250
+ }
251
+ this.data.sort((a, b) => {
252
+ const aValue = a[this.sortState.field];
253
+ const bValue = b[this.sortState.field];
254
+ let comparison = 0;
255
+ if (aValue > bValue)
256
+ comparison = 1;
257
+ if (aValue < bValue)
258
+ comparison = -1;
259
+ return this.sortState.direction === "desc" ? -comparison : comparison;
260
+ });
261
+ this.calculateVirtualization();
262
+ }
263
+ getSortIcon(column) {
264
+ if (!column.sortable || this.sortState.field !== column.field)
265
+ return "";
266
+ if (this.sortState.direction === "asc")
267
+ return "▲";
268
+ if (this.sortState.direction === "desc")
269
+ return "▼";
270
+ return "";
271
+ }
272
+ // Column resizing functionality
273
+ onResizeStart(event, column) {
274
+ event.preventDefault();
275
+ this.isResizing = true;
276
+ this.resizingColumn = column.field;
277
+ this.startX = event.clientX;
278
+ this.startWidth = this.columnWidths[column.field];
279
+ if (this.isBrowser) {
280
+ document.body.style.cursor = "col-resize";
281
+ document.body.style.userSelect = "none";
282
+ }
283
+ }
284
+ onMouseMove(event) {
285
+ if (!this.isResizing || !this.resizingColumn)
286
+ return;
287
+ this.ngZone.runOutsideAngular(() => {
288
+ const deltaX = event.clientX - this.startX;
289
+ const newWidth = Math.max(50, this.startWidth + deltaX); // Minimum width of 50px
290
+ // Only update if width actually changed
291
+ if (this.columnWidths[this.resizingColumn] !== newWidth) {
292
+ this.columnWidths[this.resizingColumn] = newWidth;
293
+ this.cdr.detectChanges();
294
+ }
295
+ });
296
+ }
297
+ onMouseUp() {
298
+ if (this.isResizing) {
299
+ this.isResizing = false;
300
+ if (this.isBrowser) {
301
+ document.body.style.cursor = "";
302
+ document.body.style.userSelect = "";
303
+ // Save column widths when resizing is complete
304
+ this.saveColumnWidths();
305
+ }
306
+ this.resizingColumn = "";
307
+ }
308
+ }
309
+ // Utility methods
310
+ getColumnWidth(column) {
311
+ return `${this.columnWidths[column.field]}px`;
312
+ }
313
+ getCellValue(row, column) {
314
+ const value = row[column.field];
315
+ return column.formatter
316
+ ? column.formatter(value, row)
317
+ : value?.toString() || "";
318
+ }
319
+ getLeftFrozenWidth() {
320
+ return this.leftFrozenColumns.reduce((sum, col) => sum + this.columnWidths[col.field], 0);
321
+ }
322
+ getRightFrozenWidth() {
323
+ return this.rightFrozenColumns.reduce((sum, col) => sum + this.columnWidths[col.field], 0);
324
+ }
325
+ // Accessibility methods
326
+ getAriaSort(column) {
327
+ if (this.sortState.field !== column.field || !column.sortable) {
328
+ return null;
329
+ }
330
+ return this.sortState.direction === "asc" ? "ascending" : "descending";
331
+ }
332
+ // Keyboard navigation
333
+ onKeydown(event) {
334
+ if (!this.tableContainer)
335
+ return;
336
+ const { key, ctrlKey, shiftKey } = event;
337
+ const scrollAmount = shiftKey ? 100 : 30; // Larger scroll with shift key
338
+ console.log(key);
339
+ switch (key) {
340
+ case "ArrowLeft":
341
+ this.tableContainer.nativeElement.scrollLeft -= scrollAmount;
342
+ break;
343
+ case "ArrowRight":
344
+ this.tableContainer.nativeElement.scrollLeft += scrollAmount;
345
+ break;
346
+ case "ArrowUp":
347
+ if (ctrlKey) {
348
+ this.tableContainer.nativeElement.scrollTo({
349
+ top: 0,
350
+ behavior: "smooth",
351
+ });
352
+ }
353
+ else {
354
+ this.tableContainer.nativeElement.scrollTop -= scrollAmount;
355
+ }
356
+ break;
357
+ case "ArrowDown":
358
+ if (ctrlKey) {
359
+ this.tableContainer.nativeElement.scrollTo({
360
+ top: this.tableContainer.nativeElement.scrollHeight,
361
+ behavior: "smooth",
362
+ });
363
+ }
364
+ else {
365
+ this.tableContainer.nativeElement.scrollTop += scrollAmount;
366
+ }
367
+ break;
368
+ case "1":
369
+ if (ctrlKey) {
370
+ this.tableContainer.nativeElement.scrollTo({
371
+ top: 0,
372
+ left: 0,
373
+ behavior: "smooth",
374
+ });
375
+ }
376
+ else {
377
+ this.tableContainer.nativeElement.scrollTo({
378
+ left: 0,
379
+ behavior: "smooth",
380
+ });
381
+ }
382
+ break;
383
+ case "0":
384
+ if (ctrlKey) {
385
+ this.tableContainer.nativeElement.scrollTo({
386
+ top: this.tableContainer.nativeElement.scrollHeight,
387
+ left: this.tableContainer.nativeElement.scrollWidth,
388
+ behavior: "smooth",
389
+ });
390
+ }
391
+ else {
392
+ this.tableContainer.nativeElement.scrollTo({
393
+ left: this.tableContainer.nativeElement.scrollWidth,
394
+ behavior: "smooth",
395
+ });
396
+ }
397
+ break;
398
+ default:
399
+ return; // Exit for other keys
400
+ }
401
+ event.preventDefault();
402
+ }
403
+ // Save column widths to localStorage
404
+ saveColumnWidths() {
405
+ if (this.isBrowser) {
406
+ try {
407
+ localStorage.setItem(this.storageKey, JSON.stringify(this.columnWidths));
408
+ }
409
+ catch (e) {
410
+ console.warn("Failed to save column widths to localStorage", e);
411
+ }
412
+ }
413
+ }
414
+ // Load column widths from localStorage
415
+ loadColumnWidths() {
416
+ if (!this.isBrowser)
417
+ return null;
418
+ try {
419
+ const saved = localStorage.getItem(this.storageKey);
420
+ return saved ? JSON.parse(saved) : null;
421
+ }
422
+ catch (e) {
423
+ console.warn("Failed to load column widths from localStorage", e);
424
+ return null;
425
+ }
426
+ }
427
+ // Clear saved column widths
428
+ clearSavedWidths() {
429
+ if (this.isBrowser) {
430
+ try {
431
+ localStorage.removeItem(this.storageKey);
432
+ // Reset to default widths
433
+ this.columns.forEach((col) => {
434
+ this.columnWidths[col.field] = col.width
435
+ ? Number.parseInt(col.width.replace("px", ""))
436
+ : 150;
437
+ });
438
+ this.cdr.detectChanges();
439
+ }
440
+ catch (e) {
441
+ console.warn("Failed to clear saved column widths", e);
442
+ }
443
+ }
444
+ }
445
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NgxSimpleDatatableComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: PLATFORM_ID }, { token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
446
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: NgxSimpleDatatableComponent, isStandalone: true, selector: "ngx-simple-datatable", inputs: { columns: "columns", data: "data", rowHeight: "rowHeight", headerHeight: "headerHeight", bufferSize: "bufferSize" }, queries: [{ propertyName: "headerTemplate", first: true, predicate: ["headerTemplate"], descendants: true, static: true }, { propertyName: "cellTemplate", first: true, predicate: ["cellTemplate"], descendants: true, static: true }], viewQueries: [{ propertyName: "tableContainer", first: true, predicate: ["tableContainer"], descendants: true, static: true }, { propertyName: "tableBody", first: true, predicate: ["tableBody"], descendants: true, static: true }, { propertyName: "headerRow", first: true, predicate: ["headerRow"], descendants: true, static: true }, { propertyName: "headeCenterRow", first: true, predicate: ["headeCenterRow"], descendants: true, static: true }], ngImport: i0, template: "<div class=\"dynamic-table-container\" #tableContainer>\n <!-- Sticky Header -->\n <div class=\"table-header\" [style.height.px]=\"headerHeight\">\n <div class=\"header-row\" #headerRow (scroll)=\"onHeaderScroll($event)\">\n <!-- Left Frozen Columns -->\n <div class=\"frozen-left\" [style.width.px]=\"getLeftFrozenWidth()\" *ngIf=\"leftFrozenColumns.length > 0\"\n role=\"presentation\">\n <div class=\"header-cell\"\n *ngFor=\"let column of leftFrozenColumns; trackBy: trackByColumnField; let colIndex = index\"\n [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\"\n [class.sort-asc]=\"sortState.field === column.field && sortState.direction === 'asc'\"\n [class.sort-desc]=\"sortState.field === column.field && sortState.direction === 'desc'\"\n [attr.role]=\"'columnheader'\" [attr.aria-sort]=\"getAriaSort(column)\" [attr.aria-colindex]=\"colIndex + 1\"\n [attr.aria-label]=\"column.header + (column.sortable ? ' (click to sort)' : '')\" (click)=\"onSort(column)\"\n (keydown.enter)=\"onSort(column)\" (keydown.space)=\"onSort(column); $event.preventDefault()\"\n [tabindex]=\"column.sortable ? '0' : '-1'\" [attr.data-column-id]=\"column.field\">\n <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n <span class=\"header-text\">{{ column.header }}</span>\n </ng-container>\n <ng-template #customHeader>\n <ng-container [ngTemplateOutlet]=\"headerTemplate\" [ngTemplateOutletContext]=\"{ column: column }\">\n </ng-container>\n\n </ng-template>\n <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n </div>\n </div>\n\n <!-- Center Scrollable Columns -->\n <div class=\"center-columns\" #headeCenterRow role=\"presentation\" [ngStyle]=\"{'transform': headerTransform}\">\n <div class=\"header-cell\" *ngFor=\"let column of centerColumns; trackBy: trackByColumnField; let colIndex = index\"\n [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\"\n [class.sort-asc]=\"sortState.field === column.field && sortState.direction === 'asc'\"\n [class.sort-desc]=\"sortState.field === column.field && sortState.direction === 'desc'\"\n [attr.role]=\"'columnheader'\" [attr.aria-sort]=\"getAriaSort(column)\"\n [attr.aria-colindex]=\"leftFrozenColumns.length + colIndex + 1\"\n [attr.aria-label]=\"column.header + (column.sortable ? ' (click to sort)' : '')\" (click)=\"onSort(column)\"\n (keydown.enter)=\"onSort(column)\" (keydown.space)=\"onSort(column); $event.preventDefault()\"\n [tabindex]=\"column.sortable ? '0' : '-1'\" [attr.data-column-id]=\"column.field\">\n <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n <span class=\"header-text\">{{ column.header }}</span>\n </ng-container>\n <ng-template #customHeader>\n <ng-container [ngTemplateOutlet]=\"headerTemplate\"\n [ngTemplateOutletContext]=\"{ column: column }\"></ng-container>\n </ng-template>\n <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n </div>\n </div>\n\n <!-- Right Frozen Columns -->\n <div class=\"frozen-right\" [style.width.px]=\"getRightFrozenWidth()\" *ngIf=\"rightFrozenColumns.length > 0\">\n <div class=\"header-cell\" *ngFor=\"let column of rightFrozenColumns; trackBy: trackByColumnField\"\n [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\" (click)=\"onSort(column)\">\n <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n <span class=\"header-text\">{{ column.header }}</span>\n </ng-container>\n <ng-template #customHeader>\n <ng-container [ngTemplateOutlet]=\"headerTemplate\"\n [ngTemplateOutletContext]=\"{ column: column }\"></ng-container>\n </ng-template>\n <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Table Body with Virtual Scrolling -->\n <div class=\"table-body\" #tableBody [style.height]=\"'calc(100% - ' + headerHeight + 'px)'\"\n (scroll)=\"onBodyScroll($event)\">\n <!-- Virtual spacer for total height -->\n <div class=\"virtual-spacer\" [style.height.px]=\"totalHeight\"></div>\n\n <!-- Visible rows container -->\n <div class=\"visible-rows\" [style.transform]=\"'translateY(' + offsetY + 'px)'\">\n <div class=\"table-row\" *ngFor=\"let row of visibleRows; let i = index; trackBy: trackByRowIndex\"\n [style.height.px]=\"rowHeight\">\n\n <!-- Left Frozen Columns -->\n <div class=\"frozen-left\" [style.width.px]=\"getLeftFrozenWidth()\" *ngIf=\"leftFrozenColumns.length > 0\">\n <div class=\"table-cell\" *ngFor=\"let column of leftFrozenColumns; trackBy: trackByColumnField\"\n [style.width]=\"getColumnWidth(column)\">\n <ng-container *ngIf=\"!cellTemplate; else customCell\">\n {{ getCellValue(row, column) }}\n </ng-container>\n <ng-template #customCell>\n <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n </ng-template>\n </div>\n </div>\n\n <!-- Center Scrollable Columns -->\n <div class=\"center-columns\">\n <div class=\"table-cell\" *ngFor=\"let column of centerColumns; trackBy: trackByColumnField\"\n [style.width]=\"getColumnWidth(column)\">\n <ng-container *ngIf=\"!cellTemplate; else customCell\">\n {{ getCellValue(row, column) }}\n </ng-container>\n <ng-template #customCell>\n <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n </ng-template>\n </div>\n </div>\n\n <!-- Right Frozen Columns -->\n <div class=\"frozen-right\" [style.width.px]=\"getRightFrozenWidth()\" *ngIf=\"rightFrozenColumns.length > 0\">\n <div class=\"table-cell\" *ngFor=\"let column of rightFrozenColumns; trackBy: trackByColumnField\"\n [style.width]=\"getColumnWidth(column)\">\n <ng-container *ngIf=\"!cellTemplate; else customCell\">\n {{ getCellValue(row, column) }}\n </ng-container>\n <ng-template #customCell>\n <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n </ng-template>\n </div>\n </div>\n </div>\n </div>\n </div>\n</div>", styles: [":root{--ngx-simple-dt-bg: #ffffff;--ngx-simple-dt-border: 1px solid #e0e0e0;--ngx-simple-dt-border-radius: 8px;--ngx-simple-dt-box-shadow: 0 2px 8px rgba(0, 0, 0, .1);--ngx-simple-dt-transition: all .2s ease-in-out;--ngx-simple-dt-header-bg: #f8f9fa;--ngx-simple-dt-header-hover-bg: #e9ecef;--ngx-simple-dt-header-border: 1px solid #e0e0e0;--ngx-simple-dt-header-text: #495057;--ngx-simple-dt-header-height: 98px;--ngx-simple-dt-header-font-weight: 600;--ngx-simple-dt-header-padding: 0 16px;--ngx-simple-dt-cell-hover-bg: #f1f3f5;--ngx-simple-dt-cell-active-bg: #e9ecef;--ngx-simple-dt-row-height: 48px;--ngx-simple-dt-cell-padding: 0 16px;--ngx-simple-dt-cell-border: 1px solid #e9ecef;--ngx-simple-dt-cell-font-size: .875rem;--ngx-simple-dt-cell-line-height: 1.5;--ngx-simple-dt-row-bg: #ffffff;--ngx-simple-dt-row-hover-bg: #f8f9fa;--ngx-simple-dt-row-stripe-bg: #f8f9fa;--ngx-simple-dt-row-active-bg: #e9ecef;--ngx-simple-dt-row-border: 1px solid #e9ecef}.dynamic-table-container{width:100%;height:100%;position:relative;display:flex;flex-direction:column;overflow:hidden;background:var(--ngx-simple-dt-bg);border:var(--ngx-simple-dt-border);border-radius:var(--ngx-simple-dt-border-radius);box-shadow:var(--ngx-simple-dt-box-shadow);transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header{position:sticky;top:0;z-index:10;background:var(--ngx-simple-dt-header-bg);border-bottom:var(--ngx-simple-dt-header-border);overflow:hidden;height:var(--ngx-simple-dt-header-height);color:var(--ngx-simple-dt-header-text)}.dynamic-table-container .table-header .header-row{display:flex;min-width:100%;position:relative;height:100%}.dynamic-table-container .table-header .header-row .frozen-left,.dynamic-table-container .table-header .header-row .frozen-right{position:sticky;z-index:12;background:var(--ngx-simple-dt-header-bg);display:flex;box-shadow:2px 0 8px #0000000d;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .frozen-left{left:0;border-right:var(--ngx-simple-dt-header-border)}.dynamic-table-container .table-header .header-row .center-columns{display:flex;flex:1}.dynamic-table-container .table-header .header-row .frozen-right{right:0;border-left:var(--ngx-simple-dt-header-border)}.dynamic-table-container .table-header .header-row .header-cell{display:flex;align-items:center;justify-content:space-between;padding:var(--ngx-simple-dt-cell-padding);border-right:var(--ngx-simple-dt-cell-border);background:inherit;font-weight:var(--ngx-simple-dt-header-font-weight);position:relative;min-width:0;font-size:var(--ngx-simple-dt-cell-font-size);line-height:var(--ngx-simple-dt-cell-line-height);color:inherit;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .header-cell.sortable{cursor:pointer;-webkit-user-select:none;user-select:none;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .header-cell.sortable:hover{background:var(--ngx-simple-dt-header-hover-bg)}.dynamic-table-container .table-header .header-row .header-cell.sortable:active{background:var(--ngx-simple-dt-cell-active-bg)}.dynamic-table-container .table-header .header-row .header-cell .header-text{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.dynamic-table-container .table-header .header-row .header-cell .sort-icon{margin-left:8px;font-size:12px;color:#6c757d;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .header-cell .sort-icon.sort-asc,.dynamic-table-container .table-header .header-row .header-cell .sort-icon.sort-desc{color:#0d6efd}.dynamic-table-container .table-header .header-row .header-cell .resize-handle{position:absolute;right:0;top:0;bottom:0;width:4px;cursor:col-resize;background:transparent;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .header-cell .resize-handle:hover,.dynamic-table-container .table-header .header-row .header-cell .resize-handle:active{background:#0d6efd}.dynamic-table-container .table-body{position:relative;flex:1;overflow:auto;scrollbar-width:thin;scrollbar-color:#ced4da #f8f9fa;-webkit-overflow-scrolling:touch}.dynamic-table-container .table-body::-webkit-scrollbar{width:8px;height:8px}.dynamic-table-container .table-body::-webkit-scrollbar-track{background:#f8f9fa;scroll-margin-left:100px}.dynamic-table-container .table-body::-webkit-scrollbar-thumb{background-color:#ced4da;border-radius:4px}.dynamic-table-container .table-body::-webkit-scrollbar-thumb:hover{background-color:#adb5bd}.dynamic-table-container .table-body .virtual-spacer{position:absolute;top:0;left:0;width:1px;pointer-events:none}.dynamic-table-container .table-body .visible-rows{position:absolute;top:0;left:0;right:0;min-width:fit-content}.dynamic-table-container .table-body .visible-rows .table-row{display:flex;min-width:100%;border-bottom:var(--ngx-simple-dt-row-border);background:var(--ngx-simple-dt-row-bg);transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-body .visible-rows .table-row:nth-child(2n){background:var(--ngx-simple-dt-row-stripe-bg)}.dynamic-table-container .table-body .visible-rows .table-row:hover{background:var(--ngx-simple-dt-row-hover-bg)}.dynamic-table-container .table-body .visible-rows .table-row:active{background:var(--ngx-simple-dt-row-active-bg)}.dynamic-table-container .table-body .visible-rows .table-row .frozen-left{position:sticky;left:0;z-index:5;background:#fff;border-right:2px solid #ccc;display:flex}.table-row:hover .dynamic-table-container .table-body .visible-rows .table-row .frozen-left{background:#f9f9f9}.dynamic-table-container .table-body .visible-rows .table-row .center-columns{display:flex;flex:0 0 auto}.dynamic-table-container .table-body .visible-rows .table-row .frozen-right{position:sticky;right:0;z-index:5;background:#fff;border-left:2px solid #ccc;display:flex}.table-row:hover .dynamic-table-container .table-body .visible-rows .table-row .frozen-right{background:#f9f9f9}.dynamic-table-container .table-body .visible-rows .table-row .table-cell{padding:var(--ngx-simple-dt-cell-padding);border-right:var(--ngx-simple-dt-cell-border);overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:var(--ngx-simple-dt-cell-font-size);line-height:var(--ngx-simple-dt-cell-line-height);display:flex;align-items:center;transition:var(--ngx-simple-dt-transition);min-width:0}.dynamic-table-container .frozen-left .table-cell,.dynamic-table-container .frozen-right .table-cell{background:inherit;box-shadow:2px 0 8px #0000000d}@media (max-width: 768px){.dynamic-table-container{font-size:14px}.dynamic-table-container .header-cell,.dynamic-table-container .table-cell{padding:0 8px}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
447
+ }
448
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: NgxSimpleDatatableComponent, decorators: [{
449
+ type: Component,
450
+ args: [{ selector: "ngx-simple-datatable", standalone: true, imports: [CommonModule], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"dynamic-table-container\" #tableContainer>\n <!-- Sticky Header -->\n <div class=\"table-header\" [style.height.px]=\"headerHeight\">\n <div class=\"header-row\" #headerRow (scroll)=\"onHeaderScroll($event)\">\n <!-- Left Frozen Columns -->\n <div class=\"frozen-left\" [style.width.px]=\"getLeftFrozenWidth()\" *ngIf=\"leftFrozenColumns.length > 0\"\n role=\"presentation\">\n <div class=\"header-cell\"\n *ngFor=\"let column of leftFrozenColumns; trackBy: trackByColumnField; let colIndex = index\"\n [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\"\n [class.sort-asc]=\"sortState.field === column.field && sortState.direction === 'asc'\"\n [class.sort-desc]=\"sortState.field === column.field && sortState.direction === 'desc'\"\n [attr.role]=\"'columnheader'\" [attr.aria-sort]=\"getAriaSort(column)\" [attr.aria-colindex]=\"colIndex + 1\"\n [attr.aria-label]=\"column.header + (column.sortable ? ' (click to sort)' : '')\" (click)=\"onSort(column)\"\n (keydown.enter)=\"onSort(column)\" (keydown.space)=\"onSort(column); $event.preventDefault()\"\n [tabindex]=\"column.sortable ? '0' : '-1'\" [attr.data-column-id]=\"column.field\">\n <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n <span class=\"header-text\">{{ column.header }}</span>\n </ng-container>\n <ng-template #customHeader>\n <ng-container [ngTemplateOutlet]=\"headerTemplate\" [ngTemplateOutletContext]=\"{ column: column }\">\n </ng-container>\n\n </ng-template>\n <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n </div>\n </div>\n\n <!-- Center Scrollable Columns -->\n <div class=\"center-columns\" #headeCenterRow role=\"presentation\" [ngStyle]=\"{'transform': headerTransform}\">\n <div class=\"header-cell\" *ngFor=\"let column of centerColumns; trackBy: trackByColumnField; let colIndex = index\"\n [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\"\n [class.sort-asc]=\"sortState.field === column.field && sortState.direction === 'asc'\"\n [class.sort-desc]=\"sortState.field === column.field && sortState.direction === 'desc'\"\n [attr.role]=\"'columnheader'\" [attr.aria-sort]=\"getAriaSort(column)\"\n [attr.aria-colindex]=\"leftFrozenColumns.length + colIndex + 1\"\n [attr.aria-label]=\"column.header + (column.sortable ? ' (click to sort)' : '')\" (click)=\"onSort(column)\"\n (keydown.enter)=\"onSort(column)\" (keydown.space)=\"onSort(column); $event.preventDefault()\"\n [tabindex]=\"column.sortable ? '0' : '-1'\" [attr.data-column-id]=\"column.field\">\n <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n <span class=\"header-text\">{{ column.header }}</span>\n </ng-container>\n <ng-template #customHeader>\n <ng-container [ngTemplateOutlet]=\"headerTemplate\"\n [ngTemplateOutletContext]=\"{ column: column }\"></ng-container>\n </ng-template>\n <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n </div>\n </div>\n\n <!-- Right Frozen Columns -->\n <div class=\"frozen-right\" [style.width.px]=\"getRightFrozenWidth()\" *ngIf=\"rightFrozenColumns.length > 0\">\n <div class=\"header-cell\" *ngFor=\"let column of rightFrozenColumns; trackBy: trackByColumnField\"\n [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\" (click)=\"onSort(column)\">\n <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n <span class=\"header-text\">{{ column.header }}</span>\n </ng-container>\n <ng-template #customHeader>\n <ng-container [ngTemplateOutlet]=\"headerTemplate\"\n [ngTemplateOutletContext]=\"{ column: column }\"></ng-container>\n </ng-template>\n <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Table Body with Virtual Scrolling -->\n <div class=\"table-body\" #tableBody [style.height]=\"'calc(100% - ' + headerHeight + 'px)'\"\n (scroll)=\"onBodyScroll($event)\">\n <!-- Virtual spacer for total height -->\n <div class=\"virtual-spacer\" [style.height.px]=\"totalHeight\"></div>\n\n <!-- Visible rows container -->\n <div class=\"visible-rows\" [style.transform]=\"'translateY(' + offsetY + 'px)'\">\n <div class=\"table-row\" *ngFor=\"let row of visibleRows; let i = index; trackBy: trackByRowIndex\"\n [style.height.px]=\"rowHeight\">\n\n <!-- Left Frozen Columns -->\n <div class=\"frozen-left\" [style.width.px]=\"getLeftFrozenWidth()\" *ngIf=\"leftFrozenColumns.length > 0\">\n <div class=\"table-cell\" *ngFor=\"let column of leftFrozenColumns; trackBy: trackByColumnField\"\n [style.width]=\"getColumnWidth(column)\">\n <ng-container *ngIf=\"!cellTemplate; else customCell\">\n {{ getCellValue(row, column) }}\n </ng-container>\n <ng-template #customCell>\n <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n </ng-template>\n </div>\n </div>\n\n <!-- Center Scrollable Columns -->\n <div class=\"center-columns\">\n <div class=\"table-cell\" *ngFor=\"let column of centerColumns; trackBy: trackByColumnField\"\n [style.width]=\"getColumnWidth(column)\">\n <ng-container *ngIf=\"!cellTemplate; else customCell\">\n {{ getCellValue(row, column) }}\n </ng-container>\n <ng-template #customCell>\n <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n </ng-template>\n </div>\n </div>\n\n <!-- Right Frozen Columns -->\n <div class=\"frozen-right\" [style.width.px]=\"getRightFrozenWidth()\" *ngIf=\"rightFrozenColumns.length > 0\">\n <div class=\"table-cell\" *ngFor=\"let column of rightFrozenColumns; trackBy: trackByColumnField\"\n [style.width]=\"getColumnWidth(column)\">\n <ng-container *ngIf=\"!cellTemplate; else customCell\">\n {{ getCellValue(row, column) }}\n </ng-container>\n <ng-template #customCell>\n <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n </ng-template>\n </div>\n </div>\n </div>\n </div>\n </div>\n</div>", styles: [":root{--ngx-simple-dt-bg: #ffffff;--ngx-simple-dt-border: 1px solid #e0e0e0;--ngx-simple-dt-border-radius: 8px;--ngx-simple-dt-box-shadow: 0 2px 8px rgba(0, 0, 0, .1);--ngx-simple-dt-transition: all .2s ease-in-out;--ngx-simple-dt-header-bg: #f8f9fa;--ngx-simple-dt-header-hover-bg: #e9ecef;--ngx-simple-dt-header-border: 1px solid #e0e0e0;--ngx-simple-dt-header-text: #495057;--ngx-simple-dt-header-height: 98px;--ngx-simple-dt-header-font-weight: 600;--ngx-simple-dt-header-padding: 0 16px;--ngx-simple-dt-cell-hover-bg: #f1f3f5;--ngx-simple-dt-cell-active-bg: #e9ecef;--ngx-simple-dt-row-height: 48px;--ngx-simple-dt-cell-padding: 0 16px;--ngx-simple-dt-cell-border: 1px solid #e9ecef;--ngx-simple-dt-cell-font-size: .875rem;--ngx-simple-dt-cell-line-height: 1.5;--ngx-simple-dt-row-bg: #ffffff;--ngx-simple-dt-row-hover-bg: #f8f9fa;--ngx-simple-dt-row-stripe-bg: #f8f9fa;--ngx-simple-dt-row-active-bg: #e9ecef;--ngx-simple-dt-row-border: 1px solid #e9ecef}.dynamic-table-container{width:100%;height:100%;position:relative;display:flex;flex-direction:column;overflow:hidden;background:var(--ngx-simple-dt-bg);border:var(--ngx-simple-dt-border);border-radius:var(--ngx-simple-dt-border-radius);box-shadow:var(--ngx-simple-dt-box-shadow);transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header{position:sticky;top:0;z-index:10;background:var(--ngx-simple-dt-header-bg);border-bottom:var(--ngx-simple-dt-header-border);overflow:hidden;height:var(--ngx-simple-dt-header-height);color:var(--ngx-simple-dt-header-text)}.dynamic-table-container .table-header .header-row{display:flex;min-width:100%;position:relative;height:100%}.dynamic-table-container .table-header .header-row .frozen-left,.dynamic-table-container .table-header .header-row .frozen-right{position:sticky;z-index:12;background:var(--ngx-simple-dt-header-bg);display:flex;box-shadow:2px 0 8px #0000000d;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .frozen-left{left:0;border-right:var(--ngx-simple-dt-header-border)}.dynamic-table-container .table-header .header-row .center-columns{display:flex;flex:1}.dynamic-table-container .table-header .header-row .frozen-right{right:0;border-left:var(--ngx-simple-dt-header-border)}.dynamic-table-container .table-header .header-row .header-cell{display:flex;align-items:center;justify-content:space-between;padding:var(--ngx-simple-dt-cell-padding);border-right:var(--ngx-simple-dt-cell-border);background:inherit;font-weight:var(--ngx-simple-dt-header-font-weight);position:relative;min-width:0;font-size:var(--ngx-simple-dt-cell-font-size);line-height:var(--ngx-simple-dt-cell-line-height);color:inherit;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .header-cell.sortable{cursor:pointer;-webkit-user-select:none;user-select:none;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .header-cell.sortable:hover{background:var(--ngx-simple-dt-header-hover-bg)}.dynamic-table-container .table-header .header-row .header-cell.sortable:active{background:var(--ngx-simple-dt-cell-active-bg)}.dynamic-table-container .table-header .header-row .header-cell .header-text{flex:1;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.dynamic-table-container .table-header .header-row .header-cell .sort-icon{margin-left:8px;font-size:12px;color:#6c757d;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .header-cell .sort-icon.sort-asc,.dynamic-table-container .table-header .header-row .header-cell .sort-icon.sort-desc{color:#0d6efd}.dynamic-table-container .table-header .header-row .header-cell .resize-handle{position:absolute;right:0;top:0;bottom:0;width:4px;cursor:col-resize;background:transparent;transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-header .header-row .header-cell .resize-handle:hover,.dynamic-table-container .table-header .header-row .header-cell .resize-handle:active{background:#0d6efd}.dynamic-table-container .table-body{position:relative;flex:1;overflow:auto;scrollbar-width:thin;scrollbar-color:#ced4da #f8f9fa;-webkit-overflow-scrolling:touch}.dynamic-table-container .table-body::-webkit-scrollbar{width:8px;height:8px}.dynamic-table-container .table-body::-webkit-scrollbar-track{background:#f8f9fa;scroll-margin-left:100px}.dynamic-table-container .table-body::-webkit-scrollbar-thumb{background-color:#ced4da;border-radius:4px}.dynamic-table-container .table-body::-webkit-scrollbar-thumb:hover{background-color:#adb5bd}.dynamic-table-container .table-body .virtual-spacer{position:absolute;top:0;left:0;width:1px;pointer-events:none}.dynamic-table-container .table-body .visible-rows{position:absolute;top:0;left:0;right:0;min-width:fit-content}.dynamic-table-container .table-body .visible-rows .table-row{display:flex;min-width:100%;border-bottom:var(--ngx-simple-dt-row-border);background:var(--ngx-simple-dt-row-bg);transition:var(--ngx-simple-dt-transition)}.dynamic-table-container .table-body .visible-rows .table-row:nth-child(2n){background:var(--ngx-simple-dt-row-stripe-bg)}.dynamic-table-container .table-body .visible-rows .table-row:hover{background:var(--ngx-simple-dt-row-hover-bg)}.dynamic-table-container .table-body .visible-rows .table-row:active{background:var(--ngx-simple-dt-row-active-bg)}.dynamic-table-container .table-body .visible-rows .table-row .frozen-left{position:sticky;left:0;z-index:5;background:#fff;border-right:2px solid #ccc;display:flex}.table-row:hover .dynamic-table-container .table-body .visible-rows .table-row .frozen-left{background:#f9f9f9}.dynamic-table-container .table-body .visible-rows .table-row .center-columns{display:flex;flex:0 0 auto}.dynamic-table-container .table-body .visible-rows .table-row .frozen-right{position:sticky;right:0;z-index:5;background:#fff;border-left:2px solid #ccc;display:flex}.table-row:hover .dynamic-table-container .table-body .visible-rows .table-row .frozen-right{background:#f9f9f9}.dynamic-table-container .table-body .visible-rows .table-row .table-cell{padding:var(--ngx-simple-dt-cell-padding);border-right:var(--ngx-simple-dt-cell-border);overflow:hidden;text-overflow:ellipsis;white-space:nowrap;font-size:var(--ngx-simple-dt-cell-font-size);line-height:var(--ngx-simple-dt-cell-line-height);display:flex;align-items:center;transition:var(--ngx-simple-dt-transition);min-width:0}.dynamic-table-container .frozen-left .table-cell,.dynamic-table-container .frozen-right .table-cell{background:inherit;box-shadow:2px 0 8px #0000000d}@media (max-width: 768px){.dynamic-table-container{font-size:14px}.dynamic-table-container .header-cell,.dynamic-table-container .table-cell{padding:0 8px}}\n"] }]
451
+ }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: Object, decorators: [{
452
+ type: Inject,
453
+ args: [PLATFORM_ID]
454
+ }] }, { type: i0.NgZone }], propDecorators: { columns: [{
455
+ type: Input
456
+ }], data: [{
457
+ type: Input
458
+ }], rowHeight: [{
459
+ type: Input
460
+ }], headerHeight: [{
461
+ type: Input
462
+ }], bufferSize: [{
463
+ type: Input
464
+ }], tableContainer: [{
465
+ type: ViewChild,
466
+ args: ["tableContainer", { static: true }]
467
+ }], tableBody: [{
468
+ type: ViewChild,
469
+ args: ["tableBody", { static: true }]
470
+ }], headerRow: [{
471
+ type: ViewChild,
472
+ args: ["headerRow", { static: true }]
473
+ }], headeCenterRow: [{
474
+ type: ViewChild,
475
+ args: ["headeCenterRow", { static: true }]
476
+ }], headerTemplate: [{
477
+ type: ContentChild,
478
+ args: ["headerTemplate", { static: true }]
479
+ }], cellTemplate: [{
480
+ type: ContentChild,
481
+ args: ["cellTemplate", { static: true }]
482
+ }] } });
483
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ngx-simple-datatable.component.js","sourceRoot":"","sources":["../../../../projects/ngx-simple-datatable/src/lib/ngx-simple-datatable.component.ts","../../../../projects/ngx-simple-datatable/src/lib/ngx-simple-datatable.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule, isPlatformBrowser } from \"@angular/common\";\nimport {\n  ChangeDetectionStrategy,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Input,\n  TrackByFunction,\n  ViewChild,\n  Inject,\n  PLATFORM_ID,\n  ContentChild,\n  TemplateRef,\n  AfterContentInit,\n  OnInit,\n  OnDestroy,\n  NgZone,\n  ChangeDetectorRef as CDR,\n} from \"@angular/core\";\nimport { ColumnConfig, SortState } from \"../interfaces/column-config.interface\";\n\n@Component({\n  selector: \"ngx-simple-datatable\",\n  standalone: true,\n  imports: [CommonModule],\n  templateUrl: \"./ngx-simple-datatable.component.html\",\n  styleUrls: [\"./ngx-simple-datatable.component.scss\"],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class NgxSimpleDatatableComponent\n  implements OnInit, OnDestroy, AfterContentInit\n{\n  @Input() columns: ColumnConfig[] = [];\n  @Input() data: any[] = [];\n  @Input() rowHeight = 40;\n  @Input() headerHeight = 50;\n  @Input() bufferSize = 10;\n\n  @ViewChild(\"tableContainer\", { static: true })\n  tableContainer!: ElementRef<HTMLDivElement>;\n  @ViewChild(\"tableBody\", { static: true })\n  tableBody!: ElementRef<HTMLDivElement>;\n  @ViewChild(\"headerRow\", { static: true })\n  headerRow!: ElementRef<HTMLDivElement>;\n  @ViewChild(\"headeCenterRow\", { static: true })\n  headeCenterRow!: ElementRef<HTMLDivElement>;\n\n  @ContentChild(\"headerTemplate\", { static: true })\n  headerTemplate!: TemplateRef<any>;\n  @ContentChild(\"cellTemplate\", { static: true })\n  cellTemplate!: TemplateRef<any>;\n\n  // Virtualization properties\n  visibleStartIndex = 0;\n  visibleEndIndex = 0;\n  visibleRows: any[] = [];\n  totalHeight = 0;\n  offsetY = 0;\n  containerHeight = 0;\n\n  // Column properties\n  leftFrozenColumns: ColumnConfig[] = [];\n  centerColumns: ColumnConfig[] = [];\n  rightFrozenColumns: ColumnConfig[] = [];\n  columnWidths: { [key: string]: number } = {};\n\n  // Sorting\n  sortState: SortState = { field: \"\", direction: null };\n\n  // Resizing\n  private isResizing = false;\n  private resizingColumn = \"\";\n  private startX = 0;\n  private startWidth = 0;\n\n  // Scroll listeners\n  private scrollListener: ((event: Event) => void) | null = null;\n  private resizeListener: ((event: Event) => void) | null = null;\n  private isBrowser: boolean;\n  private storageKey = \"ngx-simple-datatable-column-widths\";\n  private scrollRequestId: number | null = null;\n  private resizeTimer: ReturnType<typeof setTimeout> | null = null;\n  private mouseMoveHandler: ((e: MouseEvent) => void) | null = null;\n  private mouseUpHandler: ((e: MouseEvent) => void) | null = null;\n\n  constructor(\n    private cdr: ChangeDetectorRef,\n    @Inject(PLATFORM_ID) private platformId: Object,\n    private ngZone: NgZone\n  ) {\n    this.isBrowser = isPlatformBrowser(this.platformId);\n  }\n\n  ngOnInit() {\n    this.initializeColumns();\n    if (this.isBrowser) {\n      this.setupEventListeners();\n    }\n    this.calculateVirtualization();\n  }\n\n  ngOnDestroy() {\n    if (this.isBrowser) {\n      this.removeEventListeners();\n\n      // Clean up any pending animations or timeouts\n      if (this.scrollRequestId) {\n        cancelAnimationFrame(this.scrollRequestId);\n      }\n\n      if (this.resizeTimer) {\n        clearTimeout(this.resizeTimer);\n      }\n    }\n  }\n\n  ngAfterContentInit() {\n    // if (!this.headerTemplate) {\n    //   throw new Error(\"ngx-simple-datatable requires a headerTemplate.\");\n    // }\n    // if (!this.cellTemplate) {\n    //   throw new Error(\"ngx-simple-datatable requires a cellTemplate.\");\n    // }\n  }\n\n  private initializeColumns() {\n    this.leftFrozenColumns = this.columns.filter(\n      (col) => col.freeze === \"left\"\n    );\n    this.rightFrozenColumns = this.columns.filter(\n      (col) => col.freeze === \"right\"\n    );\n    this.centerColumns = this.columns.filter((col) => !col.freeze);\n\n    // Load saved column widths if available\n    const savedWidths = this.loadColumnWidths();\n\n    // Initialize column widths\n    this.columns.forEach((col) => {\n      if (col.width) {\n        this.columnWidths[col.field] = Number.parseInt(\n          col.width.replace(\"px\", \"\")\n        );\n      } else if (savedWidths && savedWidths[col.field]) {\n        this.columnWidths[col.field] = savedWidths[col.field];\n      } else {\n        this.columnWidths[col.field] = 150; // default width\n      }\n    });\n  }\n\n  private setupEventListeners() {\n    // Store references for cleanup\n    this.scrollListener = (event: Event) => this.onBodyScroll(event);\n    this.resizeListener = () => this.onResize();\n    this.mouseMoveHandler = (event: MouseEvent) => this.onMouseMove(event);\n    this.mouseUpHandler = (event: MouseEvent) => this.onMouseUp();\n\n    // Add event listeners\n    if (this.isBrowser) {\n      // For the header scroll (horizontal only)\n      if (this.headerRow?.nativeElement) {\n        this.headerRow.nativeElement.addEventListener(\"scroll\", () =>\n          this.onHeaderScroll(null)\n        );\n      }\n\n      // For the body scroll (both vertical and horizontal)\n      if (this.tableBody?.nativeElement && this.scrollListener) {\n        this.tableBody.nativeElement.addEventListener(\n          \"scroll\",\n          this.scrollListener\n        );\n      }\n\n      // Window resize and mouse events\n      if (this.resizeListener) {\n        window.addEventListener(\"resize\", this.resizeListener);\n      }\n\n      if (this.mouseMoveHandler) {\n        document.addEventListener(\"mousemove\", this.mouseMoveHandler);\n      }\n\n      if (this.mouseUpHandler) {\n        document.addEventListener(\"mouseup\", this.mouseUpHandler);\n      }\n    }\n  }\n\n  private removeEventListeners() {\n    if (this.isBrowser) {\n      // Clear any pending animations or timeouts\n      if (this.scrollRequestId) {\n        cancelAnimationFrame(this.scrollRequestId);\n        this.scrollRequestId = null;\n      }\n\n      if (this.resizeTimer) {\n        clearTimeout(this.resizeTimer);\n        this.resizeTimer = null;\n      }\n\n      // Reset all event listeners\n      this.scrollListener = null;\n      this.resizeListener = null;\n      this.mouseMoveHandler = null;\n      this.mouseUpHandler = null;\n    }\n  }\n\n  private calculateVirtualization() {\n    if (!this.tableContainer || !this.tableBody) return;\n\n    // Update container height based on available space\n    const container = this.tableBody.nativeElement;\n    this.containerHeight = container.clientHeight - this.headerHeight;\n    this.totalHeight = this.data.length * this.rowHeight;\n\n    // Get scroll position\n    const scrollTop = container.scrollTop;\n\n    // Calculate visible rows\n    const visibleRowCount =\n      Math.ceil(this.containerHeight / this.rowHeight) + 1;\n\n    // Calculate start and end indices with buffer\n    this.visibleStartIndex = Math.max(\n      0,\n      Math.floor(scrollTop / this.rowHeight) - this.bufferSize\n    );\n\n    this.visibleEndIndex = Math.min(\n      this.data.length - 1,\n      this.visibleStartIndex + visibleRowCount + this.bufferSize * 2\n    );\n\n    // Update visible rows\n    this.visibleRows = this.data.slice(\n      this.visibleStartIndex,\n      this.visibleEndIndex + 1\n    );\n\n    console.log(scrollTop, this.visibleStartIndex, this.visibleEndIndex);\n\n    // Update vertical offset for virtual scrolling\n    this.offsetY = this.visibleStartIndex * this.rowHeight;\n\n    // Trigger change detection in the next tick to avoid ExpressionChangedAfterItHasBeenCheckedError\n    Promise.resolve().then(() => {\n      this.cdr.detectChanges();\n    });\n  }\n\n  // Track last scroll time to prevent feedback loops\n  private lastScrollTime = 0;\n  private readonly SCROLL_DEBOUNCE = 10; // ms\n\n  // Track scroll position for header transform\n  public headerTransform = \"translateX(0)\";\n\n  // Handle body scroll events\n  public onBodyScroll(event: Event) {\n    if (!event) return;\n\n    const now = Date.now();\n    if (now - this.lastScrollTime < this.SCROLL_DEBOUNCE) return;\n    this.lastScrollTime = now;\n\n    const target = event.target as HTMLElement;\n\n    // Sync header scroll with body (horizontally)\n    // if (this.headerRow?.nativeElement) {\n    // this.headerRow.nativeElement.scrollLeft = target.scrollLeft;\n    this.syncHorizontalScroll(target.scrollLeft);\n    // }\n\n    // Handle virtualization (vertical scrolling)\n    if (this.scrollRequestId) {\n      cancelAnimationFrame(this.scrollRequestId);\n    }\n\n    this.scrollRequestId = requestAnimationFrame(() => {\n      this.calculateVirtualization();\n      this.scrollRequestId = null;\n    });\n  }\n\n  // Handle header scroll events\n  public onHeaderScroll(event: Event | null): void {\n    if (!event) return;\n    const now = Date.now();\n    if (now - this.lastScrollTime < this.SCROLL_DEBOUNCE) return;\n    this.lastScrollTime = now;\n\n    // Sync body scroll with header (horizontally)\n    const scrollLeft = (event.target as HTMLElement).scrollLeft;\n    if (this.tableBody?.nativeElement) {\n      this.tableBody.nativeElement.scrollLeft = scrollLeft;\n    }\n  }\n\n  private onResize() {\n    // Debounce resize events for better performance\n    if (this.resizeTimer) {\n      clearTimeout(this.resizeTimer);\n    }\n\n    this.resizeTimer = setTimeout(() => {\n      this.calculateVirtualization();\n      this.resizeTimer = null;\n    }, 50);\n  }\n\n  private syncHorizontalScroll(scrollLeft: number) {\n    if (!this.tableContainer) return;\n\n    // Update the transform value\n    this.headerTransform = `translateX(-${scrollLeft}px)`;\n    // Trigger change detection\n    this.cdr.detectChanges();\n  }\n\n  // Sorting functionality\n  onSort(column: ColumnConfig) {\n    if (!column.sortable) return;\n\n    if (this.sortState.field === column.field) {\n      // Toggle sort direction\n      if (this.sortState.direction === \"asc\") {\n        this.sortState.direction = \"desc\";\n      } else if (this.sortState.direction === \"desc\") {\n        this.sortState.direction = null;\n        this.sortState.field = \"\";\n      } else {\n        this.sortState.direction = \"asc\";\n      }\n    } else {\n      this.sortState.field = column.field;\n      this.sortState.direction = \"asc\";\n    }\n\n    this.applySorting();\n  }\n\n  private applySorting() {\n    if (!this.sortState.direction || !this.sortState.field) {\n      // Reset to original order if no sorting\n      return;\n    }\n\n    this.data.sort((a, b) => {\n      const aValue = a[this.sortState.field];\n      const bValue = b[this.sortState.field];\n\n      let comparison = 0;\n      if (aValue > bValue) comparison = 1;\n      if (aValue < bValue) comparison = -1;\n\n      return this.sortState.direction === \"desc\" ? -comparison : comparison;\n    });\n\n    this.calculateVirtualization();\n  }\n\n  getSortIcon(column: ColumnConfig): string {\n    if (!column.sortable || this.sortState.field !== column.field) return \"\";\n    if (this.sortState.direction === \"asc\") return \"▲\";\n    if (this.sortState.direction === \"desc\") return \"▼\";\n    return \"\";\n  }\n\n  // Column resizing functionality\n  onResizeStart(event: MouseEvent, column: ColumnConfig) {\n    event.preventDefault();\n    this.isResizing = true;\n    this.resizingColumn = column.field;\n    this.startX = event.clientX;\n    this.startWidth = this.columnWidths[column.field];\n\n    if (this.isBrowser) {\n      document.body.style.cursor = \"col-resize\";\n      document.body.style.userSelect = \"none\";\n    }\n  }\n\n  private onMouseMove(event: MouseEvent) {\n    if (!this.isResizing || !this.resizingColumn) return;\n\n    this.ngZone.runOutsideAngular(() => {\n      const deltaX = event.clientX - this.startX;\n      const newWidth = Math.max(50, this.startWidth + deltaX); // Minimum width of 50px\n\n      // Only update if width actually changed\n      if (this.columnWidths[this.resizingColumn] !== newWidth) {\n        this.columnWidths[this.resizingColumn] = newWidth;\n        this.cdr.detectChanges();\n      }\n    });\n  }\n\n  private onMouseUp() {\n    if (this.isResizing) {\n      this.isResizing = false;\n      if (this.isBrowser) {\n        document.body.style.cursor = \"\";\n        document.body.style.userSelect = \"\";\n        // Save column widths when resizing is complete\n        this.saveColumnWidths();\n      }\n      this.resizingColumn = \"\";\n    }\n  }\n\n  // Utility methods\n  getColumnWidth(column: ColumnConfig): string {\n    return `${this.columnWidths[column.field]}px`;\n  }\n\n  getCellValue(row: any, column: ColumnConfig): string {\n    const value = row[column.field];\n    return column.formatter\n      ? column.formatter(value, row)\n      : value?.toString() || \"\";\n  }\n\n  getLeftFrozenWidth(): number {\n    return this.leftFrozenColumns.reduce(\n      (sum, col) => sum + this.columnWidths[col.field],\n      0\n    );\n  }\n\n  getRightFrozenWidth(): number {\n    return this.rightFrozenColumns.reduce(\n      (sum, col) => sum + this.columnWidths[col.field],\n      0\n    );\n  }\n\n  // TrackBy functions for performance\n  trackByRowIndex: TrackByFunction<any> = (index: number) =>\n    this.visibleStartIndex + index;\n  trackByColumnField: TrackByFunction<ColumnConfig> = (\n    index: number,\n    column: ColumnConfig\n  ) => column.field;\n\n  // Accessibility methods\n  getAriaSort(column: ColumnConfig): string | null {\n    if (this.sortState.field !== column.field || !column.sortable) {\n      return null;\n    }\n    return this.sortState.direction === \"asc\" ? \"ascending\" : \"descending\";\n  }\n\n  // Keyboard navigation\n  onKeydown(event: KeyboardEvent): void {\n    if (!this.tableContainer) return;\n\n    const { key, ctrlKey, shiftKey } = event;\n    const scrollAmount = shiftKey ? 100 : 30; // Larger scroll with shift key\n    console.log(key);\n    switch (key) {\n      case \"ArrowLeft\":\n        this.tableContainer.nativeElement.scrollLeft -= scrollAmount;\n        break;\n      case \"ArrowRight\":\n        this.tableContainer.nativeElement.scrollLeft += scrollAmount;\n        break;\n      case \"ArrowUp\":\n        if (ctrlKey) {\n          this.tableContainer.nativeElement.scrollTo({\n            top: 0,\n            behavior: \"smooth\",\n          });\n        } else {\n          this.tableContainer.nativeElement.scrollTop -= scrollAmount;\n        }\n        break;\n      case \"ArrowDown\":\n        if (ctrlKey) {\n          this.tableContainer.nativeElement.scrollTo({\n            top: this.tableContainer.nativeElement.scrollHeight,\n            behavior: \"smooth\",\n          });\n        } else {\n          this.tableContainer.nativeElement.scrollTop += scrollAmount;\n        }\n        break;\n      case \"1\":\n        if (ctrlKey) {\n          this.tableContainer.nativeElement.scrollTo({\n            top: 0,\n            left: 0,\n            behavior: \"smooth\",\n          });\n        } else {\n          this.tableContainer.nativeElement.scrollTo({\n            left: 0,\n            behavior: \"smooth\",\n          });\n        }\n        break;\n      case \"0\":\n        if (ctrlKey) {\n          this.tableContainer.nativeElement.scrollTo({\n            top: this.tableContainer.nativeElement.scrollHeight,\n            left: this.tableContainer.nativeElement.scrollWidth,\n            behavior: \"smooth\",\n          });\n        } else {\n          this.tableContainer.nativeElement.scrollTo({\n            left: this.tableContainer.nativeElement.scrollWidth,\n            behavior: \"smooth\",\n          });\n        }\n        break;\n      default:\n        return; // Exit for other keys\n    }\n\n    event.preventDefault();\n  }\n\n  // Save column widths to localStorage\n  private saveColumnWidths(): void {\n    if (this.isBrowser) {\n      try {\n        localStorage.setItem(\n          this.storageKey,\n          JSON.stringify(this.columnWidths)\n        );\n      } catch (e) {\n        console.warn(\"Failed to save column widths to localStorage\", e);\n      }\n    }\n  }\n\n  // Load column widths from localStorage\n  private loadColumnWidths(): Record<string, number> | null {\n    if (!this.isBrowser) return null;\n\n    try {\n      const saved = localStorage.getItem(this.storageKey);\n      return saved ? JSON.parse(saved) : null;\n    } catch (e) {\n      console.warn(\"Failed to load column widths from localStorage\", e);\n      return null;\n    }\n  }\n\n  // Clear saved column widths\n  public clearSavedWidths(): void {\n    if (this.isBrowser) {\n      try {\n        localStorage.removeItem(this.storageKey);\n        // Reset to default widths\n        this.columns.forEach((col) => {\n          this.columnWidths[col.field] = col.width\n            ? Number.parseInt(col.width.replace(\"px\", \"\"))\n            : 150;\n        });\n        this.cdr.detectChanges();\n      } catch (e) {\n        console.warn(\"Failed to clear saved column widths\", e);\n      }\n    }\n  }\n}\n","<div class=\"dynamic-table-container\" #tableContainer>\n  <!-- Sticky Header -->\n  <div class=\"table-header\" [style.height.px]=\"headerHeight\">\n    <div class=\"header-row\" #headerRow (scroll)=\"onHeaderScroll($event)\">\n      <!-- Left Frozen Columns -->\n      <div class=\"frozen-left\" [style.width.px]=\"getLeftFrozenWidth()\" *ngIf=\"leftFrozenColumns.length > 0\"\n        role=\"presentation\">\n        <div class=\"header-cell\"\n          *ngFor=\"let column of leftFrozenColumns; trackBy: trackByColumnField; let colIndex = index\"\n          [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\"\n          [class.sort-asc]=\"sortState.field === column.field && sortState.direction === 'asc'\"\n          [class.sort-desc]=\"sortState.field === column.field && sortState.direction === 'desc'\"\n          [attr.role]=\"'columnheader'\" [attr.aria-sort]=\"getAriaSort(column)\" [attr.aria-colindex]=\"colIndex + 1\"\n          [attr.aria-label]=\"column.header + (column.sortable ? ' (click to sort)' : '')\" (click)=\"onSort(column)\"\n          (keydown.enter)=\"onSort(column)\" (keydown.space)=\"onSort(column); $event.preventDefault()\"\n          [tabindex]=\"column.sortable ? '0' : '-1'\" [attr.data-column-id]=\"column.field\">\n          <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n            <span class=\"header-text\">{{ column.header }}</span>\n          </ng-container>\n          <ng-template #customHeader>\n            <ng-container [ngTemplateOutlet]=\"headerTemplate\" [ngTemplateOutletContext]=\"{ column: column }\">\n            </ng-container>\n\n          </ng-template>\n          <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n          <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n        </div>\n      </div>\n\n      <!-- Center Scrollable Columns -->\n      <div class=\"center-columns\" #headeCenterRow role=\"presentation\" [ngStyle]=\"{'transform': headerTransform}\">\n        <div class=\"header-cell\" *ngFor=\"let column of centerColumns; trackBy: trackByColumnField; let colIndex = index\"\n          [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\"\n          [class.sort-asc]=\"sortState.field === column.field && sortState.direction === 'asc'\"\n          [class.sort-desc]=\"sortState.field === column.field && sortState.direction === 'desc'\"\n          [attr.role]=\"'columnheader'\" [attr.aria-sort]=\"getAriaSort(column)\"\n          [attr.aria-colindex]=\"leftFrozenColumns.length + colIndex + 1\"\n          [attr.aria-label]=\"column.header + (column.sortable ? ' (click to sort)' : '')\" (click)=\"onSort(column)\"\n          (keydown.enter)=\"onSort(column)\" (keydown.space)=\"onSort(column); $event.preventDefault()\"\n          [tabindex]=\"column.sortable ? '0' : '-1'\" [attr.data-column-id]=\"column.field\">\n          <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n            <span class=\"header-text\">{{ column.header }}</span>\n          </ng-container>\n          <ng-template #customHeader>\n            <ng-container [ngTemplateOutlet]=\"headerTemplate\"\n              [ngTemplateOutletContext]=\"{ column: column }\"></ng-container>\n          </ng-template>\n          <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n          <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n        </div>\n      </div>\n\n      <!-- Right Frozen Columns -->\n      <div class=\"frozen-right\" [style.width.px]=\"getRightFrozenWidth()\" *ngIf=\"rightFrozenColumns.length > 0\">\n        <div class=\"header-cell\" *ngFor=\"let column of rightFrozenColumns; trackBy: trackByColumnField\"\n          [style.width]=\"getColumnWidth(column)\" [class.sortable]=\"column.sortable\" (click)=\"onSort(column)\">\n          <ng-container *ngIf=\"!headerTemplate; else customHeader\">\n            <span class=\"header-text\">{{ column.header }}</span>\n          </ng-container>\n          <ng-template #customHeader>\n            <ng-container [ngTemplateOutlet]=\"headerTemplate\"\n              [ngTemplateOutletContext]=\"{ column: column }\"></ng-container>\n          </ng-template>\n          <span class=\"sort-icon\" *ngIf=\"column.sortable\">{{ getSortIcon(column) }}</span>\n          <div class=\"resize-handle\" (mousedown)=\"onResizeStart($event, column)\"></div>\n        </div>\n      </div>\n    </div>\n  </div>\n\n  <!-- Table Body with Virtual Scrolling -->\n  <div class=\"table-body\" #tableBody [style.height]=\"'calc(100% - ' + headerHeight + 'px)'\"\n    (scroll)=\"onBodyScroll($event)\">\n    <!-- Virtual spacer for total height -->\n    <div class=\"virtual-spacer\" [style.height.px]=\"totalHeight\"></div>\n\n    <!-- Visible rows container -->\n    <div class=\"visible-rows\" [style.transform]=\"'translateY(' + offsetY + 'px)'\">\n      <div class=\"table-row\" *ngFor=\"let row of visibleRows; let i = index; trackBy: trackByRowIndex\"\n        [style.height.px]=\"rowHeight\">\n\n        <!-- Left Frozen Columns -->\n        <div class=\"frozen-left\" [style.width.px]=\"getLeftFrozenWidth()\" *ngIf=\"leftFrozenColumns.length > 0\">\n          <div class=\"table-cell\" *ngFor=\"let column of leftFrozenColumns; trackBy: trackByColumnField\"\n            [style.width]=\"getColumnWidth(column)\">\n            <ng-container *ngIf=\"!cellTemplate; else customCell\">\n              {{ getCellValue(row, column) }}\n            </ng-container>\n            <ng-template #customCell>\n              <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n                [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n            </ng-template>\n          </div>\n        </div>\n\n        <!-- Center Scrollable Columns -->\n        <div class=\"center-columns\">\n          <div class=\"table-cell\" *ngFor=\"let column of centerColumns; trackBy: trackByColumnField\"\n            [style.width]=\"getColumnWidth(column)\">\n            <ng-container *ngIf=\"!cellTemplate; else customCell\">\n              {{ getCellValue(row, column) }}\n            </ng-container>\n            <ng-template #customCell>\n              <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n                [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n            </ng-template>\n          </div>\n        </div>\n\n        <!-- Right Frozen Columns -->\n        <div class=\"frozen-right\" [style.width.px]=\"getRightFrozenWidth()\" *ngIf=\"rightFrozenColumns.length > 0\">\n          <div class=\"table-cell\" *ngFor=\"let column of rightFrozenColumns; trackBy: trackByColumnField\"\n            [style.width]=\"getColumnWidth(column)\">\n            <ng-container *ngIf=\"!cellTemplate; else customCell\">\n              {{ getCellValue(row, column) }}\n            </ng-container>\n            <ng-template #customCell>\n              <ng-container [ngTemplateOutlet]=\"cellTemplate\"\n                [ngTemplateOutletContext]=\"{ row: row, column: column }\"></ng-container>\n            </ng-template>\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>"]}