ngx-simple-datatables 1.17.0 → 2.0.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.
- package/README.md +86 -52
- package/esm2022/interfaces/column-config.interface.mjs +1 -1
- package/esm2022/lib/ngx-simple-datatables.component.mjs +483 -0
- package/esm2022/lib/ngx-simple-datatables.module.mjs +20 -0
- package/esm2022/lib/ngx-simple-datatables.service.mjs +14 -0
- package/esm2022/ngx-simple-datatables.mjs +1 -1
- package/esm2022/public-api.mjs +5 -4
- package/fesm2022/ngx-simple-datatables.mjs +32 -16
- package/fesm2022/ngx-simple-datatables.mjs.map +1 -1
- package/lib/{ngx-simple-datatable.component.d.ts → ngx-simple-datatables.component.d.ts} +3 -3
- package/lib/ngx-simple-datatables.module.d.ts +8 -0
- package/lib/ngx-simple-datatables.service.d.ts +6 -0
- package/package.json +6 -6
- package/public-api.d.ts +3 -2
- package/esm2022/lib/ngx-simple-datatable.component.mjs +0 -483
- package/esm2022/lib/ngx-simple-datatable.service.mjs +0 -14
- package/lib/ngx-simple-datatable.service.d.ts +0 -6
|
@@ -0,0 +1,483 @@
|
|
|
1
|
+
import { 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 NgxSimpleDatatablesComponent {
|
|
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-datatables-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-datatables requires a headerTemplate.");
|
|
74
|
+
// }
|
|
75
|
+
// if (!this.cellTemplate) {
|
|
76
|
+
// throw new Error("ngx-simple-datatables 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: NgxSimpleDatatablesComponent, 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: NgxSimpleDatatablesComponent, selector: "ngx-simple-datatables", 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: "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: NgxSimpleDatatablesComponent, decorators: [{
|
|
449
|
+
type: Component,
|
|
450
|
+
args: [{ selector: "ngx-simple-datatables", 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-datatables.component.js","sourceRoot":"","sources":["../../../../projects/ngx-simple-datatables/src/lib/ngx-simple-datatables.component.ts","../../../../projects/ngx-simple-datatables/src/lib/ngx-simple-datatables.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-datatables\",\n  templateUrl: \"./ngx-simple-datatables.component.html\",\n  styleUrls: [\"./ngx-simple-datatables.component.scss\"],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class NgxSimpleDatatablesComponent\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-datatables-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-datatables requires a headerTemplate.\");\n    // }\n    // if (!this.cellTemplate) {\n    //   throw new Error(\"ngx-simple-datatables 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>"]}
|