ng-prime-tools 1.0.60 → 1.0.63
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/esm2022/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.mjs +128 -45
- package/esm2022/lib/pt-date-input/pt-date-input.component.mjs +3 -3
- package/esm2022/lib/pt-dialog/pt-dialog.component.mjs +2 -2
- package/esm2022/lib/pt-form-builder/pt-form-builder.component.mjs +12 -5
- package/esm2022/lib/pt-side-bar-menu/pt-side-bar-menu.component.mjs +2 -2
- package/fesm2022/ng-prime-tools.mjs +144 -54
- package/fesm2022/ng-prime-tools.mjs.map +1 -1
- package/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.d.ts +24 -3
- package/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.d.ts.map +1 -1
- package/lib/pt-form-builder/pt-form-builder.component.d.ts +8 -1
- package/lib/pt-form-builder/pt-form-builder.component.d.ts.map +1 -1
- package/lib/pt-metric-panel/pt-metric-panel.component.d.ts +1 -1
- package/package.json +1 -1
|
@@ -27,12 +27,18 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
27
27
|
this.hasColumnFilter = false;
|
|
28
28
|
/** paginator UI */
|
|
29
29
|
this.isPaginated = true;
|
|
30
|
-
/**
|
|
30
|
+
/** server-side mode */
|
|
31
31
|
this.isLazy = false;
|
|
32
32
|
this.actions = [];
|
|
33
33
|
this.isSortable = false;
|
|
34
34
|
this.loading = false;
|
|
35
35
|
this.maxHeight = null;
|
|
36
|
+
/**
|
|
37
|
+
* New unified lazy event.
|
|
38
|
+
* Use this in parent components for server-side pagination/filter/sort/search.
|
|
39
|
+
*/
|
|
40
|
+
this.lazyLoad = new EventEmitter();
|
|
41
|
+
/** Backward compatibility */
|
|
36
42
|
this.search = new EventEmitter();
|
|
37
43
|
this.exportExcelEvent = new EventEmitter();
|
|
38
44
|
this.exportPdfEvent = new EventEmitter();
|
|
@@ -48,6 +54,10 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
48
54
|
this.validCurrencyCodes = ['USD', 'EUR', 'MAD'];
|
|
49
55
|
this.iconWidth = 77;
|
|
50
56
|
this.rows = 0;
|
|
57
|
+
this.first = 0;
|
|
58
|
+
this.currentPage = 0;
|
|
59
|
+
this.currentSortField = null;
|
|
60
|
+
this.currentSortOrder = null;
|
|
51
61
|
this.hasGroupedColumns = false;
|
|
52
62
|
this.isDelete = false;
|
|
53
63
|
this.isEdit = false;
|
|
@@ -83,25 +93,45 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
83
93
|
if (col.isEditable === undefined)
|
|
84
94
|
col.isEditable = true;
|
|
85
95
|
if (col.isFilter !== false && col.code !== undefined) {
|
|
86
|
-
this.globalFilterFields.
|
|
96
|
+
if (!this.globalFilterFields.includes(col.code)) {
|
|
97
|
+
this.globalFilterFields.push(col.code);
|
|
98
|
+
}
|
|
87
99
|
}
|
|
88
100
|
if (!col.width)
|
|
89
101
|
col.width = this.calculateColumnWidth(col);
|
|
90
102
|
});
|
|
91
|
-
// ✅ client-side mode: totalRecords = data.length by default
|
|
92
103
|
if (!this.isLazy) {
|
|
93
104
|
this.totalRecords = this.data?.length ?? 0;
|
|
94
105
|
}
|
|
95
106
|
}
|
|
96
107
|
ngOnChanges(changes) {
|
|
97
|
-
// ✅ when data changes in client-side mode, keep paginator consistent
|
|
98
108
|
if (changes['data'] && !this.isLazy) {
|
|
99
109
|
this.totalRecords = this.data?.length ?? 0;
|
|
100
|
-
// if we were on page > 1, go back to first page (avoids empty view)
|
|
101
110
|
if (this.dt)
|
|
102
111
|
this.dt.first = 0;
|
|
103
112
|
}
|
|
104
113
|
}
|
|
114
|
+
// ---------- LAZY PAYLOAD ----------
|
|
115
|
+
emitLazyLoad() {
|
|
116
|
+
const payload = {
|
|
117
|
+
page: this.currentPage,
|
|
118
|
+
rows: this.rows,
|
|
119
|
+
first: this.first,
|
|
120
|
+
search: this.searchValue?.trim() || undefined,
|
|
121
|
+
sortField: this.currentSortField,
|
|
122
|
+
sortOrder: this.currentSortOrder,
|
|
123
|
+
filters: { ...this.latestFilterValues },
|
|
124
|
+
};
|
|
125
|
+
console.log('emitLazyLoad payload =', payload);
|
|
126
|
+
this.lazyLoad.emit(payload);
|
|
127
|
+
}
|
|
128
|
+
resetToFirstPage() {
|
|
129
|
+
this.currentPage = 0;
|
|
130
|
+
this.first = 0;
|
|
131
|
+
if (this.dt) {
|
|
132
|
+
this.dt.first = 0;
|
|
133
|
+
}
|
|
134
|
+
}
|
|
105
135
|
// ---------- HEADER + BODY ALIGNMENT HELPERS ----------
|
|
106
136
|
getHeaderTitleClass(col) {
|
|
107
137
|
const align = col.headerAlign ?? AlignEnum.LEFT;
|
|
@@ -188,7 +218,10 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
188
218
|
initializeComposedFilters(col) {
|
|
189
219
|
col.composedNames?.forEach((composedName) => {
|
|
190
220
|
const code = col.code || '';
|
|
191
|
-
|
|
221
|
+
const composedCode = code + '.' + composedName;
|
|
222
|
+
if (!this.globalFilterFields.includes(composedCode)) {
|
|
223
|
+
this.globalFilterFields.push(composedCode);
|
|
224
|
+
}
|
|
192
225
|
this.filters[composedName] = {
|
|
193
226
|
options: col.filterOptions,
|
|
194
227
|
value: [],
|
|
@@ -212,8 +245,10 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
212
245
|
col.composedNames.forEach((name) => {
|
|
213
246
|
const key = `${col.code}.${name}`;
|
|
214
247
|
delete this.latestFilterValues[key];
|
|
215
|
-
if (this.filters[name])
|
|
248
|
+
if (this.filters[name]) {
|
|
216
249
|
this.filters[name].value = [];
|
|
250
|
+
}
|
|
251
|
+
this.clearedFields.add(key);
|
|
217
252
|
});
|
|
218
253
|
}
|
|
219
254
|
onComposedFilterValueChange(col, composedName, value, filterModel) {
|
|
@@ -250,11 +285,14 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
250
285
|
return;
|
|
251
286
|
const isEmpty = value === null ||
|
|
252
287
|
value === undefined ||
|
|
288
|
+
value === '' ||
|
|
253
289
|
(Array.isArray(value) && value.length === 0);
|
|
254
|
-
if (isEmpty)
|
|
290
|
+
if (isEmpty) {
|
|
255
291
|
delete this.latestFilterValues[field];
|
|
256
|
-
|
|
292
|
+
}
|
|
293
|
+
else {
|
|
257
294
|
this.latestFilterValues[field] = value;
|
|
295
|
+
}
|
|
258
296
|
if (filterModel) {
|
|
259
297
|
filterModel.value = value;
|
|
260
298
|
if (Array.isArray(filterModel.constraints) &&
|
|
@@ -279,13 +317,22 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
279
317
|
return this.columns.find((c) => c.code === field || (c.code && field.startsWith(c.code + '.')));
|
|
280
318
|
}
|
|
281
319
|
filterColumn(event) {
|
|
320
|
+
console.log('filterColumn event =', event);
|
|
321
|
+
console.log('filterColumn event.filters =', event?.filters);
|
|
322
|
+
console.log('filterColumn latestFilterValues BEFORE =', this.latestFilterValues);
|
|
282
323
|
const filters = event?.filters;
|
|
283
324
|
if (!filters) {
|
|
284
|
-
if (this.isLazy)
|
|
325
|
+
if (this.isLazy) {
|
|
326
|
+
this.resetToFirstPage();
|
|
285
327
|
this.onFilterColumn.emit(event);
|
|
328
|
+
this.emitLazyLoad();
|
|
329
|
+
}
|
|
286
330
|
return;
|
|
287
331
|
}
|
|
288
332
|
const isNullish = (v) => v === null || v === undefined || v === '';
|
|
333
|
+
const nextFilterValues = {
|
|
334
|
+
...this.latestFilterValues,
|
|
335
|
+
};
|
|
289
336
|
Object.keys(filters).forEach((field) => {
|
|
290
337
|
const meta = filters[field];
|
|
291
338
|
const normalizeMeta = (m) => Array.isArray(m) && m.length > 0 ? m[0] : m;
|
|
@@ -294,21 +341,48 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
294
341
|
const wasCleared = this.clearedFields.has(field);
|
|
295
342
|
if (!m && !col)
|
|
296
343
|
return;
|
|
297
|
-
//
|
|
344
|
+
// ---- CLEARED FIELD ----
|
|
345
|
+
// Must be handled BEFORE COMPOSED logic
|
|
346
|
+
if (wasCleared) {
|
|
347
|
+
delete nextFilterValues[field];
|
|
348
|
+
if (col?.type === TableTypeEnum.COMPOSED &&
|
|
349
|
+
col.code &&
|
|
350
|
+
col.composedNames) {
|
|
351
|
+
col.composedNames.forEach((name) => {
|
|
352
|
+
delete nextFilterValues[`${col.code}.${name}`];
|
|
353
|
+
});
|
|
354
|
+
}
|
|
355
|
+
if (m) {
|
|
356
|
+
m.value = null;
|
|
357
|
+
if (Array.isArray(m.constraints) && m.constraints.length > 0) {
|
|
358
|
+
m.constraints.forEach((c) => (c.value = null));
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
this.clearedFields.delete(field);
|
|
362
|
+
return;
|
|
363
|
+
}
|
|
364
|
+
// ---- COMPOSED FILTERS ----
|
|
298
365
|
if (col && col.type === TableTypeEnum.COMPOSED) {
|
|
299
366
|
const composedValues = {};
|
|
300
367
|
col.composedNames?.forEach((name) => {
|
|
301
368
|
const key = `${col.code}.${name}`;
|
|
302
|
-
|
|
369
|
+
if (this.clearedFields.has(key)) {
|
|
370
|
+
delete nextFilterValues[key];
|
|
371
|
+
this.clearedFields.delete(key);
|
|
372
|
+
return;
|
|
373
|
+
}
|
|
374
|
+
const val = nextFilterValues[key];
|
|
303
375
|
const empty = isNullish(val) || (Array.isArray(val) && val.length === 0);
|
|
304
|
-
if (!empty)
|
|
376
|
+
if (!empty) {
|
|
305
377
|
composedValues[name] = val;
|
|
378
|
+
}
|
|
306
379
|
});
|
|
307
380
|
if (Object.keys(composedValues).length === 0) {
|
|
308
381
|
if (m) {
|
|
309
382
|
m.value = null;
|
|
310
|
-
if (Array.isArray(m.constraints))
|
|
383
|
+
if (Array.isArray(m.constraints)) {
|
|
311
384
|
m.constraints.forEach((c) => (c.value = null));
|
|
385
|
+
}
|
|
312
386
|
}
|
|
313
387
|
}
|
|
314
388
|
else {
|
|
@@ -318,23 +392,14 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
318
392
|
}
|
|
319
393
|
if (!m)
|
|
320
394
|
return;
|
|
321
|
-
// CLEAR
|
|
322
|
-
if (wasCleared) {
|
|
323
|
-
delete this.latestFilterValues[field];
|
|
324
|
-
m.value = null;
|
|
325
|
-
if (Array.isArray(m.constraints) && m.constraints.length > 0) {
|
|
326
|
-
m.constraints[0].value = null;
|
|
327
|
-
}
|
|
328
|
-
this.clearedFields.delete(field);
|
|
329
|
-
return;
|
|
330
|
-
}
|
|
331
395
|
let value = m.value;
|
|
332
396
|
if (Array.isArray(m.constraints) && m.constraints.length > 0) {
|
|
333
397
|
const cVal = m.constraints[0].value;
|
|
334
|
-
if (!isNullish(cVal))
|
|
398
|
+
if (!isNullish(cVal)) {
|
|
335
399
|
value = cVal;
|
|
400
|
+
}
|
|
336
401
|
}
|
|
337
|
-
const cached =
|
|
402
|
+
const cached = nextFilterValues[field];
|
|
338
403
|
const hasCached = !isNullish(cached) && (!Array.isArray(cached) || cached.length > 0);
|
|
339
404
|
if ((isNullish(value) || (Array.isArray(value) && value.length === 0)) &&
|
|
340
405
|
hasCached) {
|
|
@@ -342,16 +407,14 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
342
407
|
}
|
|
343
408
|
const isEmpty = isNullish(value) || (Array.isArray(value) && value.length === 0);
|
|
344
409
|
if (isEmpty) {
|
|
345
|
-
delete
|
|
410
|
+
delete nextFilterValues[field];
|
|
346
411
|
m.value = null;
|
|
347
412
|
if (Array.isArray(m.constraints) && m.constraints.length > 0) {
|
|
348
413
|
m.constraints[0].value = null;
|
|
349
414
|
}
|
|
350
415
|
return;
|
|
351
416
|
}
|
|
352
|
-
this.latestFilterValues[field] = value;
|
|
353
417
|
let emitValue = value;
|
|
354
|
-
// ✅ DATE / DATETIME normalization for server-side filtering
|
|
355
418
|
if (col &&
|
|
356
419
|
(col.type === TableTypeEnum.DATE || col.type === TableTypeEnum.DATETIME)) {
|
|
357
420
|
const d = this.parseAnyDate(value);
|
|
@@ -360,17 +423,21 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
360
423
|
? this.formatDateWithColumn(d, col)
|
|
361
424
|
: this.formatDateTimeWithColumn(d, col);
|
|
362
425
|
}
|
|
426
|
+
nextFilterValues[field] = emitValue;
|
|
363
427
|
m.value = emitValue;
|
|
364
428
|
if (Array.isArray(m.constraints) && m.constraints.length > 0) {
|
|
365
429
|
m.constraints[0].value = emitValue;
|
|
366
430
|
}
|
|
367
431
|
});
|
|
368
|
-
|
|
432
|
+
console.log('filterColumn nextFilterValues =', nextFilterValues);
|
|
433
|
+
this.latestFilterValues = nextFilterValues;
|
|
434
|
+
console.log('filterColumn latestFilterValues AFTER =', this.latestFilterValues);
|
|
369
435
|
if (this.isLazy) {
|
|
436
|
+
this.resetToFirstPage();
|
|
370
437
|
this.onFilterColumn.emit(event);
|
|
438
|
+
this.emitLazyLoad();
|
|
371
439
|
}
|
|
372
440
|
else {
|
|
373
|
-
// client-side: keep totalRecords in sync with filtered view
|
|
374
441
|
if (this.dt) {
|
|
375
442
|
const current = (this.dt.filteredValue ?? this.dt.value ?? []);
|
|
376
443
|
this.totalRecords = current?.length ?? 0;
|
|
@@ -381,14 +448,16 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
381
448
|
changePage(event) {
|
|
382
449
|
const page = event.page ?? Math.floor((event.first || 0) / event.rows);
|
|
383
450
|
const rows = event.rows;
|
|
451
|
+
const first = event.first ?? page * rows;
|
|
384
452
|
this.rows = rows;
|
|
385
|
-
|
|
453
|
+
this.currentPage = page;
|
|
454
|
+
this.first = first;
|
|
386
455
|
if (this.isLazy) {
|
|
387
456
|
this.onPageChange.emit({ page, rows });
|
|
457
|
+
this.emitLazyLoad();
|
|
388
458
|
}
|
|
389
459
|
}
|
|
390
460
|
sortColumn(event) {
|
|
391
|
-
// ✅ emit only when server-side sorting
|
|
392
461
|
if (!this.isLazy)
|
|
393
462
|
return;
|
|
394
463
|
let field = event.field;
|
|
@@ -397,15 +466,21 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
397
466
|
let textProp;
|
|
398
467
|
if (col.composedNames && col.composedTypes) {
|
|
399
468
|
const idx = col.composedTypes.findIndex((t) => t === TableTypeEnum.STRING);
|
|
400
|
-
if (idx >= 0 && idx < col.composedNames.length)
|
|
469
|
+
if (idx >= 0 && idx < col.composedNames.length) {
|
|
401
470
|
textProp = col.composedNames[idx];
|
|
471
|
+
}
|
|
402
472
|
}
|
|
403
|
-
if (!textProp && col.composedNames?.length)
|
|
473
|
+
if (!textProp && col.composedNames?.length) {
|
|
404
474
|
textProp = col.composedNames[0];
|
|
475
|
+
}
|
|
405
476
|
if (textProp)
|
|
406
477
|
field = `${field}.${textProp}`;
|
|
407
478
|
}
|
|
479
|
+
this.currentSortField = field ?? null;
|
|
480
|
+
this.currentSortOrder = event.order ?? null;
|
|
481
|
+
this.resetToFirstPage();
|
|
408
482
|
this.onSortColumn.emit({ ...event, field });
|
|
483
|
+
this.emitLazyLoad();
|
|
409
484
|
}
|
|
410
485
|
// ---------- EDIT HELPERS ----------
|
|
411
486
|
parseDate_ddMMyyyy(dateString) {
|
|
@@ -496,12 +571,13 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
496
571
|
filterGlobal(event) {
|
|
497
572
|
const target = event.target;
|
|
498
573
|
const value = (target.value || '').toLowerCase();
|
|
499
|
-
// ✅ server-side mode
|
|
500
574
|
if (this.isLazy) {
|
|
575
|
+
this.searchValue = value;
|
|
576
|
+
this.resetToFirstPage();
|
|
501
577
|
this.search.emit(value);
|
|
578
|
+
this.emitLazyLoad();
|
|
502
579
|
return;
|
|
503
580
|
}
|
|
504
|
-
// ✅ client-side mode
|
|
505
581
|
if (!value) {
|
|
506
582
|
this.dt.value = [...(this.data ?? [])];
|
|
507
583
|
this.totalRecords = (this.dt.value ?? []).length;
|
|
@@ -551,7 +627,7 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
551
627
|
}
|
|
552
628
|
return false;
|
|
553
629
|
}
|
|
554
|
-
// ----------
|
|
630
|
+
// ---------- DATE / DATETIME ----------
|
|
555
631
|
parseAnyDate(input) {
|
|
556
632
|
if (input === null || input === undefined || input === '')
|
|
557
633
|
return null;
|
|
@@ -624,16 +700,19 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
624
700
|
table.clear();
|
|
625
701
|
this.searchValue = '';
|
|
626
702
|
this.latestFilterValues = {};
|
|
703
|
+
this.clearedFields.clear();
|
|
704
|
+
this.currentSortField = null;
|
|
705
|
+
this.currentSortOrder = null;
|
|
627
706
|
Object.keys(this.filters).forEach((key) => {
|
|
628
707
|
if (this.filters[key])
|
|
629
708
|
this.filters[key].value = [];
|
|
630
709
|
});
|
|
631
|
-
// ✅ server-side: parent must reload
|
|
632
710
|
if (this.isLazy) {
|
|
711
|
+
this.resetToFirstPage();
|
|
633
712
|
this.onFilterColumn.emit({ cleared: true });
|
|
713
|
+
this.emitLazyLoad();
|
|
634
714
|
return;
|
|
635
715
|
}
|
|
636
|
-
// ✅ client-side: reset paginator totals
|
|
637
716
|
if (this.dt) {
|
|
638
717
|
this.dt.value = [...(this.data ?? [])];
|
|
639
718
|
this.totalRecords = (this.dt.value ?? []).length;
|
|
@@ -646,6 +725,8 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
646
725
|
this.rowsPerPage = [10, 20, 30];
|
|
647
726
|
}
|
|
648
727
|
this.rows = this.rowsPerPage[0];
|
|
728
|
+
this.currentPage = 0;
|
|
729
|
+
this.first = 0;
|
|
649
730
|
}
|
|
650
731
|
}
|
|
651
732
|
getCurrencySymbol(column) {
|
|
@@ -711,21 +792,21 @@ export class PTAdvancedPrimeTableComponent {
|
|
|
711
792
|
isActionVisible(action, row) {
|
|
712
793
|
const v = action?.visible;
|
|
713
794
|
if (v === undefined || v === null)
|
|
714
|
-
return true;
|
|
795
|
+
return true;
|
|
715
796
|
return typeof v === 'function' ? !!v(row) : !!v;
|
|
716
797
|
}
|
|
717
798
|
isActionDisabled(action, row) {
|
|
718
799
|
const d = action.disabled;
|
|
719
800
|
if (d === undefined || d === null)
|
|
720
|
-
return false;
|
|
801
|
+
return false;
|
|
721
802
|
return typeof d === 'function' ? !!d(row) : !!d;
|
|
722
803
|
}
|
|
723
804
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTAdvancedPrimeTableComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
724
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTAdvancedPrimeTableComponent, selector: "pt-advanced-prime-table", inputs: { data: "data", columns: "columns", totalRecords: "totalRecords", rowsPerPage: "rowsPerPage", hasSearchFilter: "hasSearchFilter", hasExportExcel: "hasExportExcel", hasExportPDF: "hasExportPDF", hasColumnFilter: "hasColumnFilter", isPaginated: "isPaginated", isLazy: "isLazy", actions: "actions", isSortable: "isSortable", loading: "loading", maxHeight: "maxHeight" }, outputs: { search: "search", exportExcelEvent: "exportExcelEvent", exportPdfEvent: "exportPdfEvent", onPageChange: "onPageChange", onSortColumn: "onSortColumn", onFilterColumn: "onFilterColumn" }, viewQueries: [{ propertyName: "dt", first: true, predicate: ["dt"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"pt-advanced-prime-table table-container\">\n <p-table\n #dt\n [value]=\"data\"\n [loading]=\"loading\"\n [rows]=\"rows\"\n [paginator]=\"isPaginated\"\n [globalFilterFields]=\"globalFilterFields\"\n [rowsPerPageOptions]=\"rowsPerPage\"\n [totalRecords]=\"totalRecords\"\n [lazy]=\"isLazy\"\n dataKey=\"id\"\n styleClass=\"p-datatable-gridlines p-datatable-striped\"\n [scrollable]=\"true\"\n [scrollHeight]=\"maxHeight !== null ? maxHeight : undefined\"\n (onFilter)=\"filterColumn($event)\"\n (onPage)=\"changePage($event)\"\n (onSort)=\"sortColumn($event)\"\n >\n <!-- COLGROUP: sync header/body widths -->\n <ng-template pTemplate=\"colgroup\" let-columns>\n <colgroup>\n <col\n *ngFor=\"let col of columns\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n />\n </colgroup>\n </ng-template>\n\n <!-- CAPTION -->\n <ng-template pTemplate=\"caption\">\n <div class=\"flex\">\n <div>\n <h3>Total: {{ totalRecords }}</h3>\n </div>\n\n <div>\n <!-- Clear filters -->\n <button\n *ngIf=\"hasSearchFilter\"\n pButton\n icon=\"pi pi-filter-slash\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"clear(dt)\"\n title=\"Clear filters\"\n ></button>\n\n <!-- Export to Excel Button -->\n <button\n *ngIf=\"hasExportExcel\"\n pButton\n icon=\"pi pi-file-excel\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"exportExcel()\"\n title=\"Export to Excel\"\n ></button>\n\n <!-- Export to PDF Button -->\n <button\n *ngIf=\"hasExportPDF\"\n pButton\n icon=\"pi pi-file-pdf\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"exportPdf()\"\n title=\"Export to PDF\"\n ></button>\n </div>\n\n <div class=\"ml-auto\" *ngIf=\"hasSearchFilter\">\n <p-iconField iconPosition=\"left\" class=\"ml-auto\">\n <p-inputIcon>\n <i class=\"pi pi-search\"></i>\n </p-inputIcon>\n <input\n pInputText\n type=\"text\"\n [(ngModel)]=\"searchValue\"\n (input)=\"filterGlobal($event)\"\n placeholder=\"Search keyword\"\n />\n </p-iconField>\n </div>\n </div>\n </ng-template>\n\n <!-- HEADER -->\n <ng-template pTemplate=\"header\">\n <tr class=\"sticky-header\">\n <ng-container *ngFor=\"let col of columns\">\n <th\n *ngIf=\"!col.children; else groupHeader\"\n pSortableColumn=\"{{ col.code }}\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [ngClass]=\"[\n getHeaderAlignClass(col),\n col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n ]\"\n colspan=\"1\"\n >\n <!-- SORTABLE HEADER -->\n <ng-container\n *ngIf=\"isSortable && col.isSortable !== false; else noSortHeader\"\n >\n <div\n class=\"header-container d-flex align-items-center justify-content-between\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [style.margin]=\"'10px'\"\n >\n <span [ngClass]=\"getHeaderTitleClass(col)\">\n {{ col.title }}\n </span>\n <div\n class=\"icons d-flex align-items-center\"\n [style.width]=\"'77px'\"\n >\n <p-sortIcon field=\"{{ col.code }}\" />\n\n <!-- COLUMN FILTERS -->\n <ng-container\n *ngIf=\"hasColumnFilter && col.isFilter !== false\"\n >\n <!-- COMPOSED FILTER (uses built-in Apply/Clear) -->\n <p-columnFilter\n *ngIf=\"col.type === TableTypeEnum.COMPOSED\"\n display=\"menu\"\n [field]=\"col.code\"\n type=\"text\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n (onClear)=\"onComposedColumnClear(col)\"\n >\n <ng-template\n pTemplate=\"filter\"\n let-filter=\"filterCallback\"\n let-filterModel=\"filterModel\"\n >\n <div *ngFor=\"let composedName of col.composedNames\">\n <ng-container\n *ngIf=\"\n getComposedFieldType(col, composedName) ===\n TableTypeEnum.STRING\n \"\n >\n <p-multiSelect\n [options]=\"filters[composedName]?.options\"\n [ngModel]=\"filters[composedName]?.value\"\n (ngModelChange)=\"\n onComposedFilterValueChange(\n col,\n composedName,\n $event,\n filterModel\n )\n \"\n [placeholder]=\"filters[composedName]?.placeholder\"\n display=\"chip\"\n >\n <ng-template let-item pTemplate=\"item\">\n <div class=\"custom-multiselect-item\">\n <img\n *ngIf=\"item.image\"\n [src]=\"item.image\"\n alt=\"icon\"\n class=\"filter-image\"\n />\n <span>{{ item.label }}</span>\n </div>\n </ng-template>\n </p-multiSelect>\n </ng-container>\n </div>\n </ng-template>\n </p-columnFilter>\n\n <!-- OTHER TYPES -->\n <p-columnFilter\n *ngIf=\"col.type !== TableTypeEnum.COMPOSED\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n (onClear)=\"onFilterClear(col.code!)\"\n >\n <!-- NUMBER / AMOUNT: use text input so '.' works -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"\n col.type === TableTypeEnum.NUMBER ||\n col.type === TableTypeEnum.AMOUNT\n \"\n let-value\n >\n <input\n pInputText\n type=\"text\"\n inputmode=\"decimal\"\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onNumberFilterChange(col.code!, $event)\n \"\n placeholder=\"Enter a number\"\n />\n </ng-template>\n\n <!-- DATE -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"col.type === TableTypeEnum.DATE\"\n let-filterModel=\"filterModel\"\n >\n <p-calendar\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n dateFormat=\"dd/mm/yy\"\n placeholder=\"Choose a date\"\n ></p-calendar>\n </ng-template>\n\n <!-- MULTISELECT -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"\n col.type === TableTypeEnum.MULTISELECT &&\n col.filterOptions &&\n col.filterOptions.length > 0\n \"\n let-filterModel=\"filterModel\"\n >\n <p-multiSelect\n [options]=\"col.filterOptions\"\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n display=\"chip\"\n placeholder=\"Choose option\"\n class=\"custom-multiselect\"\n ></p-multiSelect>\n </ng-template>\n </p-columnFilter>\n </ng-container>\n </div>\n </div>\n </ng-container>\n\n <!-- NON-SORTABLE HEADER -->\n <ng-template #noSortHeader>\n <div\n class=\"header-container d-flex align-items-center justify-content-between\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [style.margin]=\"'10px'\"\n >\n <span [ngClass]=\"getHeaderTitleClass(col)\">\n {{ col.title }}\n </span>\n\n <ng-container *ngIf=\"hasColumnFilter && col.isFilter !== false\">\n <!-- AMOUNT behaves like NUMBER -->\n <p-columnFilter\n *ngIf=\"col.type === 'AMOUNT'\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [currency]=\"getCurrencySymbol(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n >\n <ng-template\n pTemplate=\"filter\"\n let-filterModel=\"filterModel\"\n >\n <input\n pInputText\n type=\"text\"\n inputmode=\"decimal\"\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n placeholder=\"Enter an amount\"\n />\n </ng-template>\n </p-columnFilter>\n\n <p-columnFilter\n *ngIf=\"col.type !== 'AMOUNT'\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n >\n <!-- DATE -->\n <ng-template\n pTemplate=\"filter\"\n let-filterModel=\"filterModel\"\n *ngIf=\"getColumnFilterType(col) === 'date'\"\n >\n <p-calendar\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n dateFormat=\"dd/mm/yy\"\n ></p-calendar>\n </ng-template>\n\n <!-- MULTISELECT -->\n <ng-template\n pTemplate=\"filter\"\n let-filterModel=\"filterModel\"\n *ngIf=\"getColumnFilterType(col) === 'multiSelect'\"\n >\n <p-multiSelect\n [options]=\"col.filterOptions\"\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n display=\"chip\"\n placeholder=\"Select\"\n class=\"custom-multiselect\"\n ></p-multiSelect>\n </ng-template>\n </p-columnFilter>\n </ng-container>\n </div>\n </ng-template>\n </th>\n\n <!-- GROUPED HEADER -->\n <ng-template #groupHeader>\n <th\n [attr.colspan]=\"col.children?.length\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"getHeaderAlignClass(col)\"\n >\n <span>{{ col.title }}</span>\n </th>\n </ng-template>\n </ng-container>\n </tr>\n\n <!-- CHILD HEADERS -->\n <tr *ngIf=\"hasGroupedColumns\">\n <ng-container *ngFor=\"let col of columns\">\n <ng-container *ngIf=\"col.children\">\n <th\n *ngFor=\"let child of col.children\"\n [style.width]=\"child.width || getHeaderWidth(child)\"\n [style.padding]=\"'0px'\"\n ></th>\n </ng-container>\n </ng-container>\n </tr>\n </ng-template>\n\n <!-- EMPTY MESSAGE -->\n <ng-template pTemplate=\"emptymessage\">\n <tr class=\"p-datatable-emptymessage\">\n <!-- span all columns -->\n <td class=\"empty-message-cell\" [attr.colspan]=\"columns.length || 1\">\n <div class=\"empty-message-wrapper\">\n <div class=\"empty-message\">\n <i class=\"pi pi-info-circle\"></i>\n <p>No records available to display.</p>\n </div>\n </div>\n </td>\n </tr>\n </ng-template>\n\n <!-- BODY -->\n <ng-template\n pTemplate=\"body\"\n let-data\n let-editing=\"editing\"\n let-ri=\"rowIndex\"\n >\n <tr *ngIf=\"!loading\" [pEditableRow]=\"isEdit ? data : null\">\n <ng-container *ngFor=\"let col of columns\">\n <!-- SIMPLE COLUMNS (no children) -->\n <ng-container *ngIf=\"!col.children; else childColumns\">\n <!-- EDITABLE CELL -->\n <td\n *ngIf=\"\n isEditable(col.code!) && col.type !== TableTypeEnum.ACTION;\n else normalTD\n \"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"getDataAlignClass(col)\"\n >\n <!-- Editable input for NUMBER/DATE/STRING/MULTISELECT -->\n <ng-container *ngIf=\"isMultiSelect(col.code); else datePicker\">\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <p-multiSelect\n appendTo=\"body\"\n [ngModel]=\"data[col.code!]\"\n [style]=\"{ width: '100%' }\"\n (ngModelChange)=\"changeHandler(data.id, col.code, $event)\"\n [options]=\"optionValues\"\n ></p-multiSelect>\n </ng-template>\n <ng-template pTemplate=\"output\">\n <div class=\"multi-select-container\">\n <ng-container *ngFor=\"let rec of data[col.code!]\">\n <p-tag [value]=\"rec\"></p-tag>\n </ng-container>\n </div>\n </ng-template>\n </p-cellEditor>\n </ng-container>\n\n <ng-template #datePicker>\n <ng-container *ngIf=\"isDatePicker(col.code); else normalInput\">\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <p-calendar\n [inputId]=\"data[col.code!]\"\n [ngModel]=\"data[col.code!]\"\n (ngModelChange)=\"\n changeHandler(data.id, col.code, $event)\n \"\n [dateFormat]=\"'dd/mm/yy'\"\n ></p-calendar>\n </ng-template>\n <ng-template pTemplate=\"output\">\n {{ data[col.code!] | customDate }}\n </ng-template>\n </p-cellEditor>\n </ng-container>\n </ng-template>\n\n <ng-template #normalInput>\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <input\n pInputText\n type=\"text\"\n [ngModel]=\"data[col.code!]\"\n (change)=\"onChange($event, data.id, col.code)\"\n />\n </ng-template>\n <ng-template pTemplate=\"output\">\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.AMOUNT;\n else normalOutput\n \"\n >\n {{\n data[col.code!]\n | customCurrency\n : getCurrencySymbol(col)\n : col.decimalPlaces\n : col.thousandSeparator\n : col.decimalSeparator\n }}\n </ng-container>\n <ng-template #normalOutput>\n {{ data[col.code!] }}\n </ng-template>\n </ng-template>\n </p-cellEditor>\n </ng-template>\n </td>\n\n <!-- NON-EDITABLE CELL (ALWAYS RENDERED) -->\n <ng-template #normalTD>\n <td\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"[\n getDataAlignClass(col),\n col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n ]\"\n >\n <!-- ACTION column: built-in edit/delete + custom actions -->\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.ACTION; else nonActionCell\"\n >\n <div class=\"action-buttons-container\">\n <!-- built-in delete -->\n <button\n *ngIf=\"isDelete\"\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-trash\"\n (click)=\"Delete(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n\n <!-- built-in inline edit -->\n <div *ngIf=\"isEdit\">\n <button\n pInitEditableRow\n *ngIf=\"!editing\"\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-pencil\"\n (click)=\"initEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n <button\n *ngIf=\"editing\"\n pSaveEditableRow\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-check\"\n (click)=\"saveEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n <button\n *ngIf=\"editing\"\n pCancelEditableRow\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-times\"\n (click)=\"cancelEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n </div>\n\n <!-- custom actions -->\n <ng-container *ngFor=\"let act of customActions\">\n <button\n *ngIf=\"isActionVisible(act, data)\"\n pButton\n pRipple\n type=\"button\"\n class=\"p-button-rounded p-button-text\"\n [icon]=\"act.icon || 'pi pi-ellipsis-h'\"\n [ngClass]=\"act.styleClass\"\n [disabled]=\"isActionDisabled(act, data)\"\n (click)=\"onCustomActionClick(act, data)\"\n ></button>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- NON-ACTION cells -->\n <ng-template #nonActionCell>\n <!-- COMPOSED -->\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.COMPOSED;\n else nonComposed\n \"\n >\n <div class=\"composed-cell\">\n <ng-container\n *ngFor=\"\n let composedName of col.composedNames;\n let i = index\n \"\n >\n <!-- IMAGE -->\n <ng-container\n *ngIf=\"\n col.composedTypes &&\n col.composedTypes[i] === TableTypeEnum.IMAGE\n \"\n >\n <img\n [src]=\"data[col.code!]?.[composedName]\"\n alt=\"composed-img\"\n class=\"composed-image\"\n [ngStyle]=\"\n getImageStyle(col.composedStyles?.[composedName])\n \"\n />\n </ng-container>\n\n <!-- STRING -->\n <ng-container\n *ngIf=\"\n col.composedTypes &&\n col.composedTypes[i] === TableTypeEnum.STRING\n \"\n >\n <span\n class=\"composed-text\"\n [ngStyle]=\"\n getTitleStyle(col.composedStyles?.[composedName])\n \"\n >\n {{ data[col.code!]?.[composedName] }}\n </span>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n\n <ng-template #nonComposed>\n <!-- AMOUNT-->\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.AMOUNT; else nonAmount\"\n >\n {{\n data[col.code!]\n | customCurrency\n : getCurrencySymbol(col)\n : col.decimalPlaces\n : col.thousandSeparator\n : col.decimalSeparator\n }}\n </ng-container>\n\n <ng-template #nonAmount>\n <!-- NUMBER-->\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.NUMBER;\n else nonNumber\n \"\n >\n {{\n formatNumber(\n data[col.code!],\n col.decimalPlaces,\n col.thousandSeparator,\n col.decimalSeparator\n )\n }}\n </ng-container>\n\n <ng-template #nonNumber>\n <!-- DATE -->\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.DATE ||\n col.type === TableTypeEnum.DATETIME;\n else normalTypes\n \"\n >\n <ng-container *ngIf=\"col.type === TableTypeEnum.DATE\">\n {{\n formatDateWithColumn(\n parseAnyDate(data[col.code!]),\n col\n )\n }}\n </ng-container>\n\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.DATETIME\"\n >\n {{\n formatDateTimeWithColumn(\n parseAnyDate(data[col.code!]),\n col\n )\n }}\n </ng-container>\n </ng-container>\n\n <ng-template #normalTypes>\n <!-- STRING, MULTISELECT-->\n <ng-container\n *ngIf=\"\n [\n TableTypeEnum.STRING,\n TableTypeEnum.MULTISELECT,\n ].includes(col.type!)\n \"\n >\n {{ data[col.code!] }}\n </ng-container>\n </ng-template>\n </ng-template>\n </ng-template>\n </ng-template>\n </ng-template>\n </td>\n </ng-template>\n </ng-container>\n\n <!-- CHILD COLUMNS -->\n <ng-template #childColumns>\n <ng-container *ngFor=\"let child of col.children\">\n <td\n [style.width]=\"child.width || getHeaderWidth(child)\"\n [ngClass]=\"getDataAlignClass(child)\"\n >\n <ng-container\n *ngIf=\"isEditable(child.code); else childNormalTD\"\n >\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <input\n pInputText\n type=\"text\"\n [ngModel]=\"child.code ? data[child.code] : null\"\n (change)=\"onChange($event, data.id, child.code)\"\n />\n </ng-template>\n <ng-template pTemplate=\"output\">\n {{ child.code ? data[child.code] : \"\" }}\n </ng-template>\n </p-cellEditor>\n </ng-container>\n\n <ng-template #childNormalTD>\n {{ child.code ? data[child.code] : \"\" }}\n </ng-template>\n </td>\n </ng-container>\n </ng-template>\n </ng-container>\n </tr>\n </ng-template>\n </p-table>\n</div>\n", styles: [".pt-advanced-prime-table .bread-crumb{margin-bottom:15px}.pt-advanced-prime-table .date{width:100%;height:5rem;display:grid;justify-items:start;align-items:center}.pt-advanced-prime-table .filter-container{width:100%;display:flex;justify-content:space-between;align-items:center}.pt-advanced-prime-table .settings{display:flex;gap:1rem}.pt-advanced-prime-table .multi-select-container{display:flex;justify-content:center;align-items:center;gap:.3rem}.pt-advanced-prime-table ::ng-deep p-table{min-width:50rem}.pt-advanced-prime-table ::ng-deep .custom-multiselect .p-hidden-accessible input{display:none}.pt-advanced-prime-table ::ng-deep .p-datatable .p-sortable-column.p-highlight:hover{background:none}.pt-advanced-prime-table ::ng-deep .p-datatable .p-sortable-column:focus{box-shadow:none;outline:0 none}.pt-advanced-prime-table ::ng-deep .header-container{display:flex;justify-content:space-between;align-items:center;width:100%}.pt-advanced-prime-table ::ng-deep .header-title-left,.pt-advanced-prime-table ::ng-deep .header-title-center,.pt-advanced-prime-table ::ng-deep .header-title-right{flex:1}.pt-advanced-prime-table ::ng-deep .header-title-left{text-align:left}.pt-advanced-prime-table ::ng-deep .header-title-center{text-align:center}.pt-advanced-prime-table ::ng-deep .header-title-right{text-align:right}.pt-advanced-prime-table ::ng-deep .header-align-left{text-align:left}.pt-advanced-prime-table ::ng-deep .header-align-center{text-align:center}.pt-advanced-prime-table ::ng-deep .header-align-right{text-align:right}.pt-advanced-prime-table ::ng-deep p-columnfilter.p-element.ng-star-inserted{margin-top:4px}.pt-advanced-prime-table .flex{display:flex;justify-content:space-between;align-items:center}.pt-advanced-prime-table .ml-auto{margin-left:auto}.pt-advanced-prime-table ::ng-deep p-inputicon{margin-right:-1.5rem;z-index:2;position:relative}.pt-advanced-prime-table ::ng-deep .p-inputtext{padding-left:1.7rem}.pt-advanced-prime-table ::ng-deep .bt-filter-btn button{cursor:pointer;margin-left:1rem}.pt-advanced-prime-table ::ng-deep .p-icon-field-left .p-input-icon:first-of-type{left:-1rem}.pt-advanced-prime-table .table-row{text-align:center;display:flex;gap:1rem;justify-content:center}.pt-advanced-prime-table ::ng-deep span.p-button-icon.pi.pi-file-excel{font-size:1.25em;color:green}.pt-advanced-prime-table ::ng-deep span.p-button-icon.pi.pi-file-pdf{font-size:1.25em;color:red}.pt-advanced-prime-table .table-container{display:block;width:100%}.pt-advanced-prime-table ::ng-deep .p-datatable-emptymessage>td.empty-message-cell{padding:0!important}.pt-advanced-prime-table .empty-message-wrapper{width:100%;height:100%;min-height:180px;display:flex;align-items:center;justify-content:center}.pt-advanced-prime-table .empty-message{text-align:center;color:#888;font-size:1.2rem}.pt-advanced-prime-table .empty-message i{display:block;font-size:2rem;margin-bottom:.5rem}.pt-advanced-prime-table th{white-space:normal;word-wrap:break-word}.filter-image{width:22px;height:14px;margin-right:5px}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-left{text-align:left!important}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-center{text-align:center!important}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-right{text-align:right!important}.pt-advanced-prime-table ::ng-deep th.action-column,.pt-advanced-prime-table ::ng-deep td.action-column{text-align:center!important;justify-content:center;align-items:center;overflow:hidden;white-space:nowrap;padding:0}.pt-advanced-prime-table ::ng-deep .action-buttons-container{width:100%;display:flex;justify-content:center!important;align-items:center!important;gap:.35rem}.pt-advanced-prime-table ::ng-deep .action-buttons-container .p-button.p-button-rounded.p-button-text{padding:.25rem!important;min-width:22px!important;height:22px!important}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i2.Table, selector: "p-table", inputs: ["frozenColumns", "frozenValue", "style", "styleClass", "tableStyle", "tableStyleClass", "paginator", "pageLinks", "rowsPerPageOptions", "alwaysShowPaginator", "paginatorPosition", "paginatorStyleClass", "paginatorDropdownAppendTo", "paginatorDropdownScrollHeight", "currentPageReportTemplate", "showCurrentPageReport", "showJumpToPageDropdown", "showJumpToPageInput", "showFirstLastIcon", "showPageLinks", "defaultSortOrder", "sortMode", "resetPageOnSort", "selectionMode", "selectionPageOnly", "contextMenuSelection", "contextMenuSelectionMode", "dataKey", "metaKeySelection", "rowSelectable", "rowTrackBy", "lazy", "lazyLoadOnInit", "compareSelectionBy", "csvSeparator", "exportFilename", "filters", "globalFilterFields", "filterDelay", "filterLocale", "expandedRowKeys", "editingRowKeys", "rowExpandMode", "scrollable", "scrollDirection", "rowGroupMode", "scrollHeight", "virtualScroll", "virtualScrollItemSize", "virtualScrollOptions", "virtualScrollDelay", "frozenWidth", "responsive", "contextMenu", "resizableColumns", "columnResizeMode", "reorderableColumns", "loading", "loadingIcon", "showLoader", "rowHover", "customSort", "showInitialSortBadge", "autoLayout", "exportFunction", "exportHeader", "stateKey", "stateStorage", "editMode", "groupRowsBy", "groupRowsByOrder", "responsiveLayout", "breakpoint", "paginatorLocale", "value", "columns", "first", "rows", "totalRecords", "sortField", "sortOrder", "multiSortMeta", "selection", "selectAll", "virtualRowHeight"], outputs: ["contextMenuSelectionChange", "selectAllChange", "selectionChange", "onRowSelect", "onRowUnselect", "onPage", "onSort", "onFilter", "onLazyLoad", "onRowExpand", "onRowCollapse", "onContextMenuSelect", "onColResize", "onColReorder", "onRowReorder", "onEditInit", "onEditComplete", "onEditCancel", "onHeaderCheckboxToggle", "sortFunction", "firstChange", "rowsChange", "onStateSave", "onStateRestore"] }, { kind: "directive", type: i3.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "directive", type: i2.SortableColumn, selector: "[pSortableColumn]", inputs: ["pSortableColumn", "pSortableColumnDisabled"] }, { kind: "component", type: i2.CellEditor, selector: "p-cellEditor" }, { kind: "component", type: i2.SortIcon, selector: "p-sortIcon", inputs: ["field"] }, { kind: "directive", type: i2.EditableRow, selector: "[pEditableRow]", inputs: ["pEditableRow", "pEditableRowDisabled"] }, { kind: "directive", type: i2.InitEditableRow, selector: "[pInitEditableRow]" }, { kind: "directive", type: i2.SaveEditableRow, selector: "[pSaveEditableRow]" }, { kind: "directive", type: i2.CancelEditableRow, selector: "[pCancelEditableRow]" }, { kind: "component", type: i2.ColumnFilter, selector: "p-columnFilter", inputs: ["field", "type", "display", "showMenu", "matchMode", "operator", "showOperator", "showClearButton", "showApplyButton", "showMatchModes", "showAddButton", "hideOnClear", "placeholder", "matchModeOptions", "maxConstraints", "minFractionDigits", "maxFractionDigits", "prefix", "suffix", "locale", "localeMatcher", "currency", "currencyDisplay", "useGrouping", "showButtons", "ariaLabel"], outputs: ["onShow", "onHide"] }, { kind: "directive", type: i4.InputText, selector: "[pInputText]", inputs: ["variant"] }, { kind: "directive", type: i5.ButtonDirective, selector: "[pButton]", inputs: ["iconPos", "loadingIcon", "label", "icon", "loading", "severity", "raised", "rounded", "text", "outlined", "size", "plain"] }, { kind: "component", type: i6.Calendar, selector: "p-calendar", inputs: ["iconDisplay", "style", "styleClass", "inputStyle", "inputId", "name", "inputStyleClass", "placeholder", "ariaLabelledBy", "ariaLabel", "iconAriaLabel", "disabled", "dateFormat", "multipleSeparator", "rangeSeparator", "inline", "showOtherMonths", "selectOtherMonths", "showIcon", "icon", "appendTo", "readonlyInput", "shortYearCutoff", "monthNavigator", "yearNavigator", "hourFormat", "timeOnly", "stepYearPicker", "stepHour", "stepMinute", "stepSecond", "showSeconds", "required", "showOnFocus", "showWeek", "startWeekFromFirstDayOfYear", "showClear", "dataType", "selectionMode", "maxDateCount", "showButtonBar", "todayButtonStyleClass", "clearButtonStyleClass", "autofocus", "autoZIndex", "baseZIndex", "panelStyleClass", "panelStyle", "keepInvalid", "hideOnDateTimeSelect", "touchUI", "timeSeparator", "focusTrap", "showTransitionOptions", "hideTransitionOptions", "tabindex", "variant", "minDate", "maxDate", "disabledDates", "disabledDays", "yearRange", "showTime", "responsiveOptions", "numberOfMonths", "firstDayOfWeek", "locale", "view", "defaultDate"], outputs: ["onFocus", "onBlur", "onClose", "onSelect", "onClear", "onInput", "onTodayClick", "onClearClick", "onMonthChange", "onYearChange", "onClickOutside", "onShow"] }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i8.MultiSelect, selector: "p-multiSelect", inputs: ["id", "ariaLabel", "style", "styleClass", "panelStyle", "panelStyleClass", "inputId", "disabled", "readonly", "group", "filter", "filterPlaceHolder", "filterLocale", "overlayVisible", "tabindex", "variant", "appendTo", "dataKey", "name", "ariaLabelledBy", "displaySelectedLabel", "maxSelectedLabels", "selectionLimit", "selectedItemsLabel", "showToggleAll", "emptyFilterMessage", "emptyMessage", "resetFilterOnHide", "dropdownIcon", "optionLabel", "optionValue", "optionDisabled", "optionGroupLabel", "optionGroupChildren", "showHeader", "filterBy", "scrollHeight", "lazy", "virtualScroll", "loading", "virtualScrollItemSize", "loadingIcon", "virtualScrollOptions", "overlayOptions", "ariaFilterLabel", "filterMatchMode", "tooltip", "tooltipPosition", "tooltipPositionStyle", "tooltipStyleClass", "autofocusFilter", "display", "autocomplete", "showClear", "autofocus", "autoZIndex", "baseZIndex", "showTransitionOptions", "hideTransitionOptions", "defaultLabel", "placeholder", "options", "filterValue", "itemSize", "selectAll", "focusOnHover", "filterFields", "selectOnFocus", "autoOptionFocus"], outputs: ["onChange", "onFilter", "onFocus", "onBlur", "onClick", "onClear", "onPanelShow", "onPanelHide", "onLazyLoad", "onRemove", "onSelectAllChange"] }, { kind: "component", type: i9.Tag, selector: "p-tag", inputs: ["style", "styleClass", "severity", "value", "icon", "rounded"] }, { kind: "component", type: i10.IconField, selector: "p-iconField", inputs: ["iconPosition"] }, { kind: "component", type: i11.InputIcon, selector: "p-inputIcon", inputs: ["styleClass"] }, { kind: "pipe", type: i12.CustomCurrencyPipe, name: "customCurrency" }, { kind: "pipe", type: i13.CustomDatePipe, name: "customDate" }] }); }
|
|
805
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.11", type: PTAdvancedPrimeTableComponent, selector: "pt-advanced-prime-table", inputs: { data: "data", columns: "columns", totalRecords: "totalRecords", rowsPerPage: "rowsPerPage", hasSearchFilter: "hasSearchFilter", hasExportExcel: "hasExportExcel", hasExportPDF: "hasExportPDF", hasColumnFilter: "hasColumnFilter", isPaginated: "isPaginated", isLazy: "isLazy", actions: "actions", isSortable: "isSortable", loading: "loading", maxHeight: "maxHeight" }, outputs: { lazyLoad: "lazyLoad", search: "search", exportExcelEvent: "exportExcelEvent", exportPdfEvent: "exportPdfEvent", onPageChange: "onPageChange", onSortColumn: "onSortColumn", onFilterColumn: "onFilterColumn" }, viewQueries: [{ propertyName: "dt", first: true, predicate: ["dt"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"pt-advanced-prime-table table-container\">\n <p-table\n #dt\n [value]=\"data\"\n [loading]=\"loading\"\n [rows]=\"rows\"\n [paginator]=\"isPaginated\"\n [globalFilterFields]=\"globalFilterFields\"\n [rowsPerPageOptions]=\"rowsPerPage\"\n [totalRecords]=\"totalRecords\"\n [lazy]=\"isLazy\"\n [filterDelay]=\"0\"\n dataKey=\"id\"\n styleClass=\"p-datatable-gridlines p-datatable-striped\"\n [scrollable]=\"true\"\n [scrollHeight]=\"maxHeight !== null ? maxHeight : undefined\"\n (onPage)=\"changePage($event)\"\n (onSort)=\"sortColumn($event)\"\n (onFilter)=\"filterColumn($event)\"\n >\n <ng-template pTemplate=\"colgroup\" let-columns>\n <colgroup>\n <col\n *ngFor=\"let col of columns\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n />\n </colgroup>\n </ng-template>\n\n <ng-template pTemplate=\"caption\">\n <div class=\"flex\">\n <div>\n <h3>Total: {{ totalRecords }}</h3>\n </div>\n\n <div>\n <button\n *ngIf=\"hasSearchFilter\"\n pButton\n icon=\"pi pi-filter-slash\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"clear(dt)\"\n title=\"Clear filters\"\n ></button>\n\n <button\n *ngIf=\"hasExportExcel\"\n pButton\n icon=\"pi pi-file-excel\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"exportExcel()\"\n title=\"Export to Excel\"\n ></button>\n\n <button\n *ngIf=\"hasExportPDF\"\n pButton\n icon=\"pi pi-file-pdf\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"exportPdf()\"\n title=\"Export to PDF\"\n ></button>\n </div>\n\n <div class=\"ml-auto\" *ngIf=\"hasSearchFilter\">\n <p-iconField iconPosition=\"left\" class=\"ml-auto\">\n <p-inputIcon>\n <i class=\"pi pi-search\"></i>\n </p-inputIcon>\n <input\n pInputText\n type=\"text\"\n [(ngModel)]=\"searchValue\"\n (input)=\"filterGlobal($event)\"\n placeholder=\"Search keyword\"\n />\n </p-iconField>\n </div>\n </div>\n </ng-template>\n\n <ng-template pTemplate=\"header\">\n <tr class=\"sticky-header\">\n <ng-container *ngFor=\"let col of columns\">\n <th\n *ngIf=\"!col.children; else groupHeader\"\n [pSortableColumn]=\"col.code\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [ngClass]=\"[\n getHeaderAlignClass(col),\n col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n ]\"\n colspan=\"1\"\n >\n <ng-container\n *ngIf=\"isSortable && col.isSortable !== false; else noSortHeader\"\n >\n <div\n class=\"header-container d-flex align-items-center justify-content-between\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [style.margin]=\"'10px'\"\n >\n <span [ngClass]=\"getHeaderTitleClass(col)\">\n {{ col.title }}\n </span>\n\n <div\n class=\"icons d-flex align-items-center\"\n [style.width]=\"'77px'\"\n >\n <p-sortIcon [field]=\"col.code\" />\n\n <ng-container\n *ngIf=\"hasColumnFilter && col.isFilter !== false\"\n >\n <!-- COMPOSED FILTER -->\n <p-columnFilter\n *ngIf=\"col.type === TableTypeEnum.COMPOSED\"\n display=\"menu\"\n [field]=\"col.code\"\n type=\"text\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n [showMatchModes]=\"false\"\n [showOperator]=\"false\"\n (onClear)=\"onComposedColumnClear(col)\"\n >\n <ng-template\n pTemplate=\"filter\"\n let-filter=\"filterCallback\"\n >\n <div *ngFor=\"let composedName of col.composedNames\">\n <ng-container\n *ngIf=\"\n getComposedFieldType(col, composedName) ===\n TableTypeEnum.STRING\n \"\n >\n <p-multiSelect\n [options]=\"filters[composedName]?.options\"\n optionLabel=\"label\"\n optionValue=\"value\"\n [ngModel]=\"filters[composedName]?.value ?? []\"\n (ngModelChange)=\"\n onComposedFilterValueChange(\n col,\n composedName,\n $event\n );\n filter($event ?? [])\n \"\n [placeholder]=\"filters[composedName]?.placeholder\"\n display=\"chip\"\n >\n <ng-template let-item pTemplate=\"item\">\n <div class=\"custom-multiselect-item\">\n <img\n *ngIf=\"item.image\"\n [src]=\"item.image\"\n alt=\"icon\"\n class=\"filter-image\"\n />\n <span>{{ item.label }}</span>\n </div>\n </ng-template>\n </p-multiSelect>\n </ng-container>\n </div>\n </ng-template>\n </p-columnFilter>\n\n <!-- OTHER TYPES -->\n <p-columnFilter\n *ngIf=\"col.type !== TableTypeEnum.COMPOSED\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n [showMatchModes]=\"col.type !== TableTypeEnum.MULTISELECT\"\n [showOperator]=\"col.type !== TableTypeEnum.MULTISELECT\"\n [matchMode]=\"\n col.type === TableTypeEnum.MULTISELECT\n ? 'in'\n : undefined\n \"\n (onClear)=\"onFilterClear(col.code!)\"\n >\n <!-- NUMBER / AMOUNT -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"\n col.type === TableTypeEnum.NUMBER ||\n col.type === TableTypeEnum.AMOUNT\n \"\n let-value\n >\n <input\n pInputText\n type=\"text\"\n inputmode=\"decimal\"\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onNumberFilterChange(col.code!, $event)\n \"\n placeholder=\"Enter a number\"\n />\n </ng-template>\n\n <!-- DATE -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"col.type === TableTypeEnum.DATE\"\n let-filter=\"filterCallback\"\n let-value\n >\n <p-calendar\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event);\n filter($event)\n \"\n dateFormat=\"dd/mm/yy\"\n placeholder=\"Choose a date\"\n ></p-calendar>\n </ng-template>\n\n <!-- MULTISELECT -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"\n col.type === TableTypeEnum.MULTISELECT &&\n col.filterOptions &&\n col.filterOptions.length > 0\n \"\n let-filter=\"filterCallback\"\n let-value\n >\n <p-multiSelect\n [options]=\"col.filterOptions\"\n optionLabel=\"label\"\n optionValue=\"value\"\n [ngModel]=\"\n latestFilterValues[col.code!] ?? value ?? []\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event);\n filter($event ?? [])\n \"\n display=\"chip\"\n placeholder=\"Choose option\"\n class=\"custom-multiselect\"\n ></p-multiSelect>\n </ng-template>\n </p-columnFilter>\n </ng-container>\n </div>\n </div>\n </ng-container>\n\n <ng-template #noSortHeader>\n <div\n class=\"header-container d-flex align-items-center justify-content-between\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [style.margin]=\"'10px'\"\n >\n <span [ngClass]=\"getHeaderTitleClass(col)\">\n {{ col.title }}\n </span>\n\n <ng-container *ngIf=\"hasColumnFilter && col.isFilter !== false\">\n <p-columnFilter\n *ngIf=\"col.type === 'AMOUNT'\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [currency]=\"getCurrencySymbol(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n >\n <ng-template pTemplate=\"filter\" let-value>\n <input\n pInputText\n type=\"text\"\n inputmode=\"decimal\"\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event)\n \"\n placeholder=\"Enter an amount\"\n />\n </ng-template>\n </p-columnFilter>\n\n <p-columnFilter\n *ngIf=\"col.type !== 'AMOUNT'\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n [showMatchModes]=\"\n getColumnFilterType(col) !== 'multiSelect'\n \"\n [showOperator]=\"getColumnFilterType(col) !== 'multiSelect'\"\n [matchMode]=\"\n getColumnFilterType(col) === 'multiSelect'\n ? 'in'\n : undefined\n \"\n (onClear)=\"onFilterClear(col.code!)\"\n >\n <!-- DATE -->\n <ng-template\n pTemplate=\"filter\"\n let-filter=\"filterCallback\"\n let-value\n *ngIf=\"getColumnFilterType(col) === 'date'\"\n >\n <p-calendar\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event);\n filter($event)\n \"\n dateFormat=\"dd/mm/yy\"\n ></p-calendar>\n </ng-template>\n\n <!-- MULTISELECT -->\n <ng-template\n pTemplate=\"filter\"\n let-filter=\"filterCallback\"\n let-value\n *ngIf=\"getColumnFilterType(col) === 'multiSelect'\"\n >\n <p-multiSelect\n [options]=\"col.filterOptions\"\n optionLabel=\"label\"\n optionValue=\"value\"\n [ngModel]=\"latestFilterValues[col.code!] ?? value ?? []\"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event);\n filter($event ?? [])\n \"\n display=\"chip\"\n placeholder=\"Select\"\n class=\"custom-multiselect\"\n ></p-multiSelect>\n </ng-template>\n </p-columnFilter>\n </ng-container>\n </div>\n </ng-template>\n </th>\n\n <ng-template #groupHeader>\n <th\n [attr.colspan]=\"col.children?.length\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"getHeaderAlignClass(col)\"\n >\n <span>{{ col.title }}</span>\n </th>\n </ng-template>\n </ng-container>\n </tr>\n\n <tr *ngIf=\"hasGroupedColumns\">\n <ng-container *ngFor=\"let col of columns\">\n <ng-container *ngIf=\"col.children\">\n <th\n *ngFor=\"let child of col.children\"\n [style.width]=\"child.width || getHeaderWidth(child)\"\n [style.padding]=\"'0px'\"\n ></th>\n </ng-container>\n </ng-container>\n </tr>\n </ng-template>\n\n <ng-template pTemplate=\"emptymessage\">\n <tr class=\"p-datatable-emptymessage\">\n <td class=\"empty-message-cell\" [attr.colspan]=\"columns.length || 1\">\n <div class=\"empty-message-wrapper\">\n <div class=\"empty-message\">\n <i class=\"pi pi-info-circle\"></i>\n <p>No records available to display.</p>\n </div>\n </div>\n </td>\n </tr>\n </ng-template>\n\n <ng-template\n pTemplate=\"body\"\n let-data\n let-editing=\"editing\"\n let-ri=\"rowIndex\"\n >\n <tr *ngIf=\"!loading\" [pEditableRow]=\"isEdit ? data : null\">\n <ng-container *ngFor=\"let col of columns\">\n <ng-container *ngIf=\"!col.children; else childColumns\">\n <td\n *ngIf=\"\n isEditable(col.code!) && col.type !== TableTypeEnum.ACTION;\n else normalTD\n \"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"getDataAlignClass(col)\"\n >\n <ng-container *ngIf=\"isMultiSelect(col.code); else datePicker\">\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <p-multiSelect\n appendTo=\"body\"\n [ngModel]=\"data[col.code!]\"\n [style]=\"{ width: '100%' }\"\n (ngModelChange)=\"changeHandler(data.id, col.code, $event)\"\n [options]=\"optionValues\"\n ></p-multiSelect>\n </ng-template>\n <ng-template pTemplate=\"output\">\n <div class=\"multi-select-container\">\n <ng-container *ngFor=\"let rec of data[col.code!]\">\n <p-tag [value]=\"rec\"></p-tag>\n </ng-container>\n </div>\n </ng-template>\n </p-cellEditor>\n </ng-container>\n\n <ng-template #datePicker>\n <ng-container *ngIf=\"isDatePicker(col.code); else normalInput\">\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <p-calendar\n [inputId]=\"data[col.code!]\"\n [ngModel]=\"data[col.code!]\"\n (ngModelChange)=\"\n changeHandler(data.id, col.code, $event)\n \"\n [dateFormat]=\"'dd/mm/yy'\"\n ></p-calendar>\n </ng-template>\n <ng-template pTemplate=\"output\">\n {{ data[col.code!] | customDate }}\n </ng-template>\n </p-cellEditor>\n </ng-container>\n </ng-template>\n\n <ng-template #normalInput>\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <input\n pInputText\n type=\"text\"\n [ngModel]=\"data[col.code!]\"\n (change)=\"onChange($event, data.id, col.code)\"\n />\n </ng-template>\n <ng-template pTemplate=\"output\">\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.AMOUNT;\n else normalOutput\n \"\n >\n {{\n data[col.code!]\n | customCurrency\n : getCurrencySymbol(col)\n : col.decimalPlaces\n : col.thousandSeparator\n : col.decimalSeparator\n }}\n </ng-container>\n <ng-template #normalOutput>\n {{ data[col.code!] }}\n </ng-template>\n </ng-template>\n </p-cellEditor>\n </ng-template>\n </td>\n\n <ng-template #normalTD>\n <td\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"[\n getDataAlignClass(col),\n col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n ]\"\n >\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.ACTION; else nonActionCell\"\n >\n <div class=\"action-buttons-container\">\n <button\n *ngIf=\"isDelete\"\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-trash\"\n (click)=\"Delete(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n\n <div *ngIf=\"isEdit\">\n <button\n pInitEditableRow\n *ngIf=\"!editing\"\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-pencil\"\n (click)=\"initEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n <button\n *ngIf=\"editing\"\n pSaveEditableRow\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-check\"\n (click)=\"saveEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n <button\n *ngIf=\"editing\"\n pCancelEditableRow\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-times\"\n (click)=\"cancelEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n </div>\n\n <ng-container *ngFor=\"let act of customActions\">\n <button\n *ngIf=\"isActionVisible(act, data)\"\n pButton\n pRipple\n type=\"button\"\n class=\"p-button-rounded p-button-text\"\n [icon]=\"act.icon || 'pi pi-ellipsis-h'\"\n [ngClass]=\"act.styleClass\"\n [disabled]=\"isActionDisabled(act, data)\"\n (click)=\"onCustomActionClick(act, data)\"\n ></button>\n </ng-container>\n </div>\n </ng-container>\n\n <ng-template #nonActionCell>\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.COMPOSED;\n else nonComposed\n \"\n >\n <div class=\"composed-cell\">\n <ng-container\n *ngFor=\"\n let composedName of col.composedNames;\n let i = index\n \"\n >\n <ng-container\n *ngIf=\"\n col.composedTypes &&\n col.composedTypes[i] === TableTypeEnum.IMAGE\n \"\n >\n <img\n [src]=\"data[col.code!]?.[composedName]\"\n alt=\"composed-img\"\n class=\"composed-image\"\n [ngStyle]=\"\n getImageStyle(col.composedStyles?.[composedName])\n \"\n />\n </ng-container>\n\n <ng-container\n *ngIf=\"\n col.composedTypes &&\n col.composedTypes[i] === TableTypeEnum.STRING\n \"\n >\n <span\n class=\"composed-text\"\n [ngStyle]=\"\n getTitleStyle(col.composedStyles?.[composedName])\n \"\n >\n {{ data[col.code!]?.[composedName] }}\n </span>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n\n <ng-template #nonComposed>\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.AMOUNT; else nonAmount\"\n >\n {{\n data[col.code!]\n | customCurrency\n : getCurrencySymbol(col)\n : col.decimalPlaces\n : col.thousandSeparator\n : col.decimalSeparator\n }}\n </ng-container>\n\n <ng-template #nonAmount>\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.NUMBER;\n else nonNumber\n \"\n >\n {{\n formatNumber(\n data[col.code!],\n col.decimalPlaces,\n col.thousandSeparator,\n col.decimalSeparator\n )\n }}\n </ng-container>\n\n <ng-template #nonNumber>\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.DATE ||\n col.type === TableTypeEnum.DATETIME;\n else normalTypes\n \"\n >\n <ng-container *ngIf=\"col.type === TableTypeEnum.DATE\">\n {{\n formatDateWithColumn(\n parseAnyDate(data[col.code!]),\n col\n )\n }}\n </ng-container>\n\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.DATETIME\"\n >\n {{\n formatDateTimeWithColumn(\n parseAnyDate(data[col.code!]),\n col\n )\n }}\n </ng-container>\n </ng-container>\n\n <ng-template #normalTypes>\n <ng-container\n *ngIf=\"\n [\n TableTypeEnum.STRING,\n TableTypeEnum.MULTISELECT,\n ].includes(col.type!)\n \"\n >\n {{ data[col.code!] }}\n </ng-container>\n </ng-template>\n </ng-template>\n </ng-template>\n </ng-template>\n </ng-template>\n </td>\n </ng-template>\n </ng-container>\n\n <ng-template #childColumns>\n <ng-container *ngFor=\"let child of col.children\">\n <td\n [style.width]=\"child.width || getHeaderWidth(child)\"\n [ngClass]=\"getDataAlignClass(child)\"\n >\n <ng-container\n *ngIf=\"isEditable(child.code); else childNormalTD\"\n >\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <input\n pInputText\n type=\"text\"\n [ngModel]=\"child.code ? data[child.code] : null\"\n (change)=\"onChange($event, data.id, child.code)\"\n />\n </ng-template>\n <ng-template pTemplate=\"output\">\n {{ child.code ? data[child.code] : \"\" }}\n </ng-template>\n </p-cellEditor>\n </ng-container>\n\n <ng-template #childNormalTD>\n {{ child.code ? data[child.code] : \"\" }}\n </ng-template>\n </td>\n </ng-container>\n </ng-template>\n </ng-container>\n </tr>\n </ng-template>\n </p-table>\n</div>\n", styles: [".pt-advanced-prime-table .bread-crumb{margin-bottom:15px}.pt-advanced-prime-table .date{width:100%;height:5rem;display:grid;justify-items:start;align-items:center}.pt-advanced-prime-table .filter-container{width:100%;display:flex;justify-content:space-between;align-items:center}.pt-advanced-prime-table .settings{display:flex;gap:1rem}.pt-advanced-prime-table .multi-select-container{display:flex;justify-content:center;align-items:center;gap:.3rem}.pt-advanced-prime-table ::ng-deep p-table{min-width:50rem}.pt-advanced-prime-table ::ng-deep .custom-multiselect .p-hidden-accessible input{display:none}.pt-advanced-prime-table ::ng-deep .p-datatable .p-sortable-column.p-highlight:hover{background:none}.pt-advanced-prime-table ::ng-deep .p-datatable .p-sortable-column:focus{box-shadow:none;outline:0 none}.pt-advanced-prime-table ::ng-deep .header-container{display:flex;justify-content:space-between;align-items:center;width:100%}.pt-advanced-prime-table ::ng-deep .header-title-left,.pt-advanced-prime-table ::ng-deep .header-title-center,.pt-advanced-prime-table ::ng-deep .header-title-right{flex:1}.pt-advanced-prime-table ::ng-deep .header-title-left{text-align:left}.pt-advanced-prime-table ::ng-deep .header-title-center{text-align:center}.pt-advanced-prime-table ::ng-deep .header-title-right{text-align:right}.pt-advanced-prime-table ::ng-deep .header-align-left{text-align:left}.pt-advanced-prime-table ::ng-deep .header-align-center{text-align:center}.pt-advanced-prime-table ::ng-deep .header-align-right{text-align:right}.pt-advanced-prime-table ::ng-deep p-columnfilter.p-element.ng-star-inserted{margin-top:4px}.pt-advanced-prime-table .flex{display:flex;justify-content:space-between;align-items:center}.pt-advanced-prime-table .ml-auto{margin-left:auto}.pt-advanced-prime-table ::ng-deep p-inputicon{margin-right:-1.5rem;z-index:2;position:relative}.pt-advanced-prime-table ::ng-deep .p-inputtext{padding-left:1.7rem}.pt-advanced-prime-table ::ng-deep .bt-filter-btn button{cursor:pointer;margin-left:1rem}.pt-advanced-prime-table ::ng-deep .p-icon-field-left .p-input-icon:first-of-type{left:-1rem}.pt-advanced-prime-table .table-row{text-align:center;display:flex;gap:1rem;justify-content:center}.pt-advanced-prime-table ::ng-deep span.p-button-icon.pi.pi-file-excel{font-size:1.25em;color:green}.pt-advanced-prime-table ::ng-deep span.p-button-icon.pi.pi-file-pdf{font-size:1.25em;color:red}.pt-advanced-prime-table .table-container{display:block;width:100%}.pt-advanced-prime-table ::ng-deep .p-datatable-emptymessage>td.empty-message-cell{padding:0!important}.pt-advanced-prime-table .empty-message-wrapper{width:100%;height:100%;min-height:180px;display:flex;align-items:center;justify-content:center}.pt-advanced-prime-table .empty-message{text-align:center;color:#888;font-size:1.2rem}.pt-advanced-prime-table .empty-message i{display:block;font-size:2rem;margin-bottom:.5rem}.pt-advanced-prime-table th{white-space:normal;word-wrap:break-word}.filter-image{width:22px;height:14px;margin-right:5px}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-left{text-align:left!important}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-center{text-align:center!important}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-right{text-align:right!important}.pt-advanced-prime-table ::ng-deep th.action-column,.pt-advanced-prime-table ::ng-deep td.action-column{text-align:center!important;justify-content:center;align-items:center;overflow:hidden;white-space:nowrap;padding:0}.pt-advanced-prime-table ::ng-deep .action-buttons-container{width:100%;display:flex;justify-content:center!important;align-items:center!important;gap:.35rem}.pt-advanced-prime-table ::ng-deep .action-buttons-container .p-button.p-button-rounded.p-button-text{padding:.25rem!important;min-width:22px!important;height:22px!important}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i2.Table, selector: "p-table", inputs: ["frozenColumns", "frozenValue", "style", "styleClass", "tableStyle", "tableStyleClass", "paginator", "pageLinks", "rowsPerPageOptions", "alwaysShowPaginator", "paginatorPosition", "paginatorStyleClass", "paginatorDropdownAppendTo", "paginatorDropdownScrollHeight", "currentPageReportTemplate", "showCurrentPageReport", "showJumpToPageDropdown", "showJumpToPageInput", "showFirstLastIcon", "showPageLinks", "defaultSortOrder", "sortMode", "resetPageOnSort", "selectionMode", "selectionPageOnly", "contextMenuSelection", "contextMenuSelectionMode", "dataKey", "metaKeySelection", "rowSelectable", "rowTrackBy", "lazy", "lazyLoadOnInit", "compareSelectionBy", "csvSeparator", "exportFilename", "filters", "globalFilterFields", "filterDelay", "filterLocale", "expandedRowKeys", "editingRowKeys", "rowExpandMode", "scrollable", "scrollDirection", "rowGroupMode", "scrollHeight", "virtualScroll", "virtualScrollItemSize", "virtualScrollOptions", "virtualScrollDelay", "frozenWidth", "responsive", "contextMenu", "resizableColumns", "columnResizeMode", "reorderableColumns", "loading", "loadingIcon", "showLoader", "rowHover", "customSort", "showInitialSortBadge", "autoLayout", "exportFunction", "exportHeader", "stateKey", "stateStorage", "editMode", "groupRowsBy", "groupRowsByOrder", "responsiveLayout", "breakpoint", "paginatorLocale", "value", "columns", "first", "rows", "totalRecords", "sortField", "sortOrder", "multiSortMeta", "selection", "selectAll", "virtualRowHeight"], outputs: ["contextMenuSelectionChange", "selectAllChange", "selectionChange", "onRowSelect", "onRowUnselect", "onPage", "onSort", "onFilter", "onLazyLoad", "onRowExpand", "onRowCollapse", "onContextMenuSelect", "onColResize", "onColReorder", "onRowReorder", "onEditInit", "onEditComplete", "onEditCancel", "onHeaderCheckboxToggle", "sortFunction", "firstChange", "rowsChange", "onStateSave", "onStateRestore"] }, { kind: "directive", type: i3.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "directive", type: i2.SortableColumn, selector: "[pSortableColumn]", inputs: ["pSortableColumn", "pSortableColumnDisabled"] }, { kind: "component", type: i2.CellEditor, selector: "p-cellEditor" }, { kind: "component", type: i2.SortIcon, selector: "p-sortIcon", inputs: ["field"] }, { kind: "directive", type: i2.EditableRow, selector: "[pEditableRow]", inputs: ["pEditableRow", "pEditableRowDisabled"] }, { kind: "directive", type: i2.InitEditableRow, selector: "[pInitEditableRow]" }, { kind: "directive", type: i2.SaveEditableRow, selector: "[pSaveEditableRow]" }, { kind: "directive", type: i2.CancelEditableRow, selector: "[pCancelEditableRow]" }, { kind: "component", type: i2.ColumnFilter, selector: "p-columnFilter", inputs: ["field", "type", "display", "showMenu", "matchMode", "operator", "showOperator", "showClearButton", "showApplyButton", "showMatchModes", "showAddButton", "hideOnClear", "placeholder", "matchModeOptions", "maxConstraints", "minFractionDigits", "maxFractionDigits", "prefix", "suffix", "locale", "localeMatcher", "currency", "currencyDisplay", "useGrouping", "showButtons", "ariaLabel"], outputs: ["onShow", "onHide"] }, { kind: "directive", type: i4.InputText, selector: "[pInputText]", inputs: ["variant"] }, { kind: "directive", type: i5.ButtonDirective, selector: "[pButton]", inputs: ["iconPos", "loadingIcon", "label", "icon", "loading", "severity", "raised", "rounded", "text", "outlined", "size", "plain"] }, { kind: "component", type: i6.Calendar, selector: "p-calendar", inputs: ["iconDisplay", "style", "styleClass", "inputStyle", "inputId", "name", "inputStyleClass", "placeholder", "ariaLabelledBy", "ariaLabel", "iconAriaLabel", "disabled", "dateFormat", "multipleSeparator", "rangeSeparator", "inline", "showOtherMonths", "selectOtherMonths", "showIcon", "icon", "appendTo", "readonlyInput", "shortYearCutoff", "monthNavigator", "yearNavigator", "hourFormat", "timeOnly", "stepYearPicker", "stepHour", "stepMinute", "stepSecond", "showSeconds", "required", "showOnFocus", "showWeek", "startWeekFromFirstDayOfYear", "showClear", "dataType", "selectionMode", "maxDateCount", "showButtonBar", "todayButtonStyleClass", "clearButtonStyleClass", "autofocus", "autoZIndex", "baseZIndex", "panelStyleClass", "panelStyle", "keepInvalid", "hideOnDateTimeSelect", "touchUI", "timeSeparator", "focusTrap", "showTransitionOptions", "hideTransitionOptions", "tabindex", "variant", "minDate", "maxDate", "disabledDates", "disabledDays", "yearRange", "showTime", "responsiveOptions", "numberOfMonths", "firstDayOfWeek", "locale", "view", "defaultDate"], outputs: ["onFocus", "onBlur", "onClose", "onSelect", "onClear", "onInput", "onTodayClick", "onClearClick", "onMonthChange", "onYearChange", "onClickOutside", "onShow"] }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i8.MultiSelect, selector: "p-multiSelect", inputs: ["id", "ariaLabel", "style", "styleClass", "panelStyle", "panelStyleClass", "inputId", "disabled", "readonly", "group", "filter", "filterPlaceHolder", "filterLocale", "overlayVisible", "tabindex", "variant", "appendTo", "dataKey", "name", "ariaLabelledBy", "displaySelectedLabel", "maxSelectedLabels", "selectionLimit", "selectedItemsLabel", "showToggleAll", "emptyFilterMessage", "emptyMessage", "resetFilterOnHide", "dropdownIcon", "optionLabel", "optionValue", "optionDisabled", "optionGroupLabel", "optionGroupChildren", "showHeader", "filterBy", "scrollHeight", "lazy", "virtualScroll", "loading", "virtualScrollItemSize", "loadingIcon", "virtualScrollOptions", "overlayOptions", "ariaFilterLabel", "filterMatchMode", "tooltip", "tooltipPosition", "tooltipPositionStyle", "tooltipStyleClass", "autofocusFilter", "display", "autocomplete", "showClear", "autofocus", "autoZIndex", "baseZIndex", "showTransitionOptions", "hideTransitionOptions", "defaultLabel", "placeholder", "options", "filterValue", "itemSize", "selectAll", "focusOnHover", "filterFields", "selectOnFocus", "autoOptionFocus"], outputs: ["onChange", "onFilter", "onFocus", "onBlur", "onClick", "onClear", "onPanelShow", "onPanelHide", "onLazyLoad", "onRemove", "onSelectAllChange"] }, { kind: "component", type: i9.Tag, selector: "p-tag", inputs: ["style", "styleClass", "severity", "value", "icon", "rounded"] }, { kind: "component", type: i10.IconField, selector: "p-iconField", inputs: ["iconPosition"] }, { kind: "component", type: i11.InputIcon, selector: "p-inputIcon", inputs: ["styleClass"] }, { kind: "pipe", type: i12.CustomCurrencyPipe, name: "customCurrency" }, { kind: "pipe", type: i13.CustomDatePipe, name: "customDate" }] }); }
|
|
725
806
|
}
|
|
726
807
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImport: i0, type: PTAdvancedPrimeTableComponent, decorators: [{
|
|
727
808
|
type: Component,
|
|
728
|
-
args: [{ selector: 'pt-advanced-prime-table', template: "<div class=\"pt-advanced-prime-table table-container\">\n <p-table\n #dt\n [value]=\"data\"\n [loading]=\"loading\"\n [rows]=\"rows\"\n [paginator]=\"isPaginated\"\n [globalFilterFields]=\"globalFilterFields\"\n [rowsPerPageOptions]=\"rowsPerPage\"\n [totalRecords]=\"totalRecords\"\n [lazy]=\"isLazy\"\n dataKey=\"id\"\n styleClass=\"p-datatable-gridlines p-datatable-striped\"\n [scrollable]=\"true\"\n [scrollHeight]=\"maxHeight !== null ? maxHeight : undefined\"\n (onFilter)=\"filterColumn($event)\"\n (onPage)=\"changePage($event)\"\n (onSort)=\"sortColumn($event)\"\n >\n <!-- COLGROUP: sync header/body widths -->\n <ng-template pTemplate=\"colgroup\" let-columns>\n <colgroup>\n <col\n *ngFor=\"let col of columns\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n />\n </colgroup>\n </ng-template>\n\n <!-- CAPTION -->\n <ng-template pTemplate=\"caption\">\n <div class=\"flex\">\n <div>\n <h3>Total: {{ totalRecords }}</h3>\n </div>\n\n <div>\n <!-- Clear filters -->\n <button\n *ngIf=\"hasSearchFilter\"\n pButton\n icon=\"pi pi-filter-slash\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"clear(dt)\"\n title=\"Clear filters\"\n ></button>\n\n <!-- Export to Excel Button -->\n <button\n *ngIf=\"hasExportExcel\"\n pButton\n icon=\"pi pi-file-excel\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"exportExcel()\"\n title=\"Export to Excel\"\n ></button>\n\n <!-- Export to PDF Button -->\n <button\n *ngIf=\"hasExportPDF\"\n pButton\n icon=\"pi pi-file-pdf\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"exportPdf()\"\n title=\"Export to PDF\"\n ></button>\n </div>\n\n <div class=\"ml-auto\" *ngIf=\"hasSearchFilter\">\n <p-iconField iconPosition=\"left\" class=\"ml-auto\">\n <p-inputIcon>\n <i class=\"pi pi-search\"></i>\n </p-inputIcon>\n <input\n pInputText\n type=\"text\"\n [(ngModel)]=\"searchValue\"\n (input)=\"filterGlobal($event)\"\n placeholder=\"Search keyword\"\n />\n </p-iconField>\n </div>\n </div>\n </ng-template>\n\n <!-- HEADER -->\n <ng-template pTemplate=\"header\">\n <tr class=\"sticky-header\">\n <ng-container *ngFor=\"let col of columns\">\n <th\n *ngIf=\"!col.children; else groupHeader\"\n pSortableColumn=\"{{ col.code }}\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [ngClass]=\"[\n getHeaderAlignClass(col),\n col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n ]\"\n colspan=\"1\"\n >\n <!-- SORTABLE HEADER -->\n <ng-container\n *ngIf=\"isSortable && col.isSortable !== false; else noSortHeader\"\n >\n <div\n class=\"header-container d-flex align-items-center justify-content-between\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [style.margin]=\"'10px'\"\n >\n <span [ngClass]=\"getHeaderTitleClass(col)\">\n {{ col.title }}\n </span>\n <div\n class=\"icons d-flex align-items-center\"\n [style.width]=\"'77px'\"\n >\n <p-sortIcon field=\"{{ col.code }}\" />\n\n <!-- COLUMN FILTERS -->\n <ng-container\n *ngIf=\"hasColumnFilter && col.isFilter !== false\"\n >\n <!-- COMPOSED FILTER (uses built-in Apply/Clear) -->\n <p-columnFilter\n *ngIf=\"col.type === TableTypeEnum.COMPOSED\"\n display=\"menu\"\n [field]=\"col.code\"\n type=\"text\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n (onClear)=\"onComposedColumnClear(col)\"\n >\n <ng-template\n pTemplate=\"filter\"\n let-filter=\"filterCallback\"\n let-filterModel=\"filterModel\"\n >\n <div *ngFor=\"let composedName of col.composedNames\">\n <ng-container\n *ngIf=\"\n getComposedFieldType(col, composedName) ===\n TableTypeEnum.STRING\n \"\n >\n <p-multiSelect\n [options]=\"filters[composedName]?.options\"\n [ngModel]=\"filters[composedName]?.value\"\n (ngModelChange)=\"\n onComposedFilterValueChange(\n col,\n composedName,\n $event,\n filterModel\n )\n \"\n [placeholder]=\"filters[composedName]?.placeholder\"\n display=\"chip\"\n >\n <ng-template let-item pTemplate=\"item\">\n <div class=\"custom-multiselect-item\">\n <img\n *ngIf=\"item.image\"\n [src]=\"item.image\"\n alt=\"icon\"\n class=\"filter-image\"\n />\n <span>{{ item.label }}</span>\n </div>\n </ng-template>\n </p-multiSelect>\n </ng-container>\n </div>\n </ng-template>\n </p-columnFilter>\n\n <!-- OTHER TYPES -->\n <p-columnFilter\n *ngIf=\"col.type !== TableTypeEnum.COMPOSED\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n (onClear)=\"onFilterClear(col.code!)\"\n >\n <!-- NUMBER / AMOUNT: use text input so '.' works -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"\n col.type === TableTypeEnum.NUMBER ||\n col.type === TableTypeEnum.AMOUNT\n \"\n let-value\n >\n <input\n pInputText\n type=\"text\"\n inputmode=\"decimal\"\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onNumberFilterChange(col.code!, $event)\n \"\n placeholder=\"Enter a number\"\n />\n </ng-template>\n\n <!-- DATE -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"col.type === TableTypeEnum.DATE\"\n let-filterModel=\"filterModel\"\n >\n <p-calendar\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n dateFormat=\"dd/mm/yy\"\n placeholder=\"Choose a date\"\n ></p-calendar>\n </ng-template>\n\n <!-- MULTISELECT -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"\n col.type === TableTypeEnum.MULTISELECT &&\n col.filterOptions &&\n col.filterOptions.length > 0\n \"\n let-filterModel=\"filterModel\"\n >\n <p-multiSelect\n [options]=\"col.filterOptions\"\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n display=\"chip\"\n placeholder=\"Choose option\"\n class=\"custom-multiselect\"\n ></p-multiSelect>\n </ng-template>\n </p-columnFilter>\n </ng-container>\n </div>\n </div>\n </ng-container>\n\n <!-- NON-SORTABLE HEADER -->\n <ng-template #noSortHeader>\n <div\n class=\"header-container d-flex align-items-center justify-content-between\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [style.margin]=\"'10px'\"\n >\n <span [ngClass]=\"getHeaderTitleClass(col)\">\n {{ col.title }}\n </span>\n\n <ng-container *ngIf=\"hasColumnFilter && col.isFilter !== false\">\n <!-- AMOUNT behaves like NUMBER -->\n <p-columnFilter\n *ngIf=\"col.type === 'AMOUNT'\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [currency]=\"getCurrencySymbol(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n >\n <ng-template\n pTemplate=\"filter\"\n let-filterModel=\"filterModel\"\n >\n <input\n pInputText\n type=\"text\"\n inputmode=\"decimal\"\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n placeholder=\"Enter an amount\"\n />\n </ng-template>\n </p-columnFilter>\n\n <p-columnFilter\n *ngIf=\"col.type !== 'AMOUNT'\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n >\n <!-- DATE -->\n <ng-template\n pTemplate=\"filter\"\n let-filterModel=\"filterModel\"\n *ngIf=\"getColumnFilterType(col) === 'date'\"\n >\n <p-calendar\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n dateFormat=\"dd/mm/yy\"\n ></p-calendar>\n </ng-template>\n\n <!-- MULTISELECT -->\n <ng-template\n pTemplate=\"filter\"\n let-filterModel=\"filterModel\"\n *ngIf=\"getColumnFilterType(col) === 'multiSelect'\"\n >\n <p-multiSelect\n [options]=\"col.filterOptions\"\n [ngModel]=\"\n latestFilterValues[col.code!] ?? filterModel?.value\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, filterModel, $event)\n \"\n display=\"chip\"\n placeholder=\"Select\"\n class=\"custom-multiselect\"\n ></p-multiSelect>\n </ng-template>\n </p-columnFilter>\n </ng-container>\n </div>\n </ng-template>\n </th>\n\n <!-- GROUPED HEADER -->\n <ng-template #groupHeader>\n <th\n [attr.colspan]=\"col.children?.length\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"getHeaderAlignClass(col)\"\n >\n <span>{{ col.title }}</span>\n </th>\n </ng-template>\n </ng-container>\n </tr>\n\n <!-- CHILD HEADERS -->\n <tr *ngIf=\"hasGroupedColumns\">\n <ng-container *ngFor=\"let col of columns\">\n <ng-container *ngIf=\"col.children\">\n <th\n *ngFor=\"let child of col.children\"\n [style.width]=\"child.width || getHeaderWidth(child)\"\n [style.padding]=\"'0px'\"\n ></th>\n </ng-container>\n </ng-container>\n </tr>\n </ng-template>\n\n <!-- EMPTY MESSAGE -->\n <ng-template pTemplate=\"emptymessage\">\n <tr class=\"p-datatable-emptymessage\">\n <!-- span all columns -->\n <td class=\"empty-message-cell\" [attr.colspan]=\"columns.length || 1\">\n <div class=\"empty-message-wrapper\">\n <div class=\"empty-message\">\n <i class=\"pi pi-info-circle\"></i>\n <p>No records available to display.</p>\n </div>\n </div>\n </td>\n </tr>\n </ng-template>\n\n <!-- BODY -->\n <ng-template\n pTemplate=\"body\"\n let-data\n let-editing=\"editing\"\n let-ri=\"rowIndex\"\n >\n <tr *ngIf=\"!loading\" [pEditableRow]=\"isEdit ? data : null\">\n <ng-container *ngFor=\"let col of columns\">\n <!-- SIMPLE COLUMNS (no children) -->\n <ng-container *ngIf=\"!col.children; else childColumns\">\n <!-- EDITABLE CELL -->\n <td\n *ngIf=\"\n isEditable(col.code!) && col.type !== TableTypeEnum.ACTION;\n else normalTD\n \"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"getDataAlignClass(col)\"\n >\n <!-- Editable input for NUMBER/DATE/STRING/MULTISELECT -->\n <ng-container *ngIf=\"isMultiSelect(col.code); else datePicker\">\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <p-multiSelect\n appendTo=\"body\"\n [ngModel]=\"data[col.code!]\"\n [style]=\"{ width: '100%' }\"\n (ngModelChange)=\"changeHandler(data.id, col.code, $event)\"\n [options]=\"optionValues\"\n ></p-multiSelect>\n </ng-template>\n <ng-template pTemplate=\"output\">\n <div class=\"multi-select-container\">\n <ng-container *ngFor=\"let rec of data[col.code!]\">\n <p-tag [value]=\"rec\"></p-tag>\n </ng-container>\n </div>\n </ng-template>\n </p-cellEditor>\n </ng-container>\n\n <ng-template #datePicker>\n <ng-container *ngIf=\"isDatePicker(col.code); else normalInput\">\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <p-calendar\n [inputId]=\"data[col.code!]\"\n [ngModel]=\"data[col.code!]\"\n (ngModelChange)=\"\n changeHandler(data.id, col.code, $event)\n \"\n [dateFormat]=\"'dd/mm/yy'\"\n ></p-calendar>\n </ng-template>\n <ng-template pTemplate=\"output\">\n {{ data[col.code!] | customDate }}\n </ng-template>\n </p-cellEditor>\n </ng-container>\n </ng-template>\n\n <ng-template #normalInput>\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <input\n pInputText\n type=\"text\"\n [ngModel]=\"data[col.code!]\"\n (change)=\"onChange($event, data.id, col.code)\"\n />\n </ng-template>\n <ng-template pTemplate=\"output\">\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.AMOUNT;\n else normalOutput\n \"\n >\n {{\n data[col.code!]\n | customCurrency\n : getCurrencySymbol(col)\n : col.decimalPlaces\n : col.thousandSeparator\n : col.decimalSeparator\n }}\n </ng-container>\n <ng-template #normalOutput>\n {{ data[col.code!] }}\n </ng-template>\n </ng-template>\n </p-cellEditor>\n </ng-template>\n </td>\n\n <!-- NON-EDITABLE CELL (ALWAYS RENDERED) -->\n <ng-template #normalTD>\n <td\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"[\n getDataAlignClass(col),\n col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n ]\"\n >\n <!-- ACTION column: built-in edit/delete + custom actions -->\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.ACTION; else nonActionCell\"\n >\n <div class=\"action-buttons-container\">\n <!-- built-in delete -->\n <button\n *ngIf=\"isDelete\"\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-trash\"\n (click)=\"Delete(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n\n <!-- built-in inline edit -->\n <div *ngIf=\"isEdit\">\n <button\n pInitEditableRow\n *ngIf=\"!editing\"\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-pencil\"\n (click)=\"initEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n <button\n *ngIf=\"editing\"\n pSaveEditableRow\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-check\"\n (click)=\"saveEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n <button\n *ngIf=\"editing\"\n pCancelEditableRow\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-times\"\n (click)=\"cancelEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n </div>\n\n <!-- custom actions -->\n <ng-container *ngFor=\"let act of customActions\">\n <button\n *ngIf=\"isActionVisible(act, data)\"\n pButton\n pRipple\n type=\"button\"\n class=\"p-button-rounded p-button-text\"\n [icon]=\"act.icon || 'pi pi-ellipsis-h'\"\n [ngClass]=\"act.styleClass\"\n [disabled]=\"isActionDisabled(act, data)\"\n (click)=\"onCustomActionClick(act, data)\"\n ></button>\n </ng-container>\n </div>\n </ng-container>\n\n <!-- NON-ACTION cells -->\n <ng-template #nonActionCell>\n <!-- COMPOSED -->\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.COMPOSED;\n else nonComposed\n \"\n >\n <div class=\"composed-cell\">\n <ng-container\n *ngFor=\"\n let composedName of col.composedNames;\n let i = index\n \"\n >\n <!-- IMAGE -->\n <ng-container\n *ngIf=\"\n col.composedTypes &&\n col.composedTypes[i] === TableTypeEnum.IMAGE\n \"\n >\n <img\n [src]=\"data[col.code!]?.[composedName]\"\n alt=\"composed-img\"\n class=\"composed-image\"\n [ngStyle]=\"\n getImageStyle(col.composedStyles?.[composedName])\n \"\n />\n </ng-container>\n\n <!-- STRING -->\n <ng-container\n *ngIf=\"\n col.composedTypes &&\n col.composedTypes[i] === TableTypeEnum.STRING\n \"\n >\n <span\n class=\"composed-text\"\n [ngStyle]=\"\n getTitleStyle(col.composedStyles?.[composedName])\n \"\n >\n {{ data[col.code!]?.[composedName] }}\n </span>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n\n <ng-template #nonComposed>\n <!-- AMOUNT-->\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.AMOUNT; else nonAmount\"\n >\n {{\n data[col.code!]\n | customCurrency\n : getCurrencySymbol(col)\n : col.decimalPlaces\n : col.thousandSeparator\n : col.decimalSeparator\n }}\n </ng-container>\n\n <ng-template #nonAmount>\n <!-- NUMBER-->\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.NUMBER;\n else nonNumber\n \"\n >\n {{\n formatNumber(\n data[col.code!],\n col.decimalPlaces,\n col.thousandSeparator,\n col.decimalSeparator\n )\n }}\n </ng-container>\n\n <ng-template #nonNumber>\n <!-- DATE -->\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.DATE ||\n col.type === TableTypeEnum.DATETIME;\n else normalTypes\n \"\n >\n <ng-container *ngIf=\"col.type === TableTypeEnum.DATE\">\n {{\n formatDateWithColumn(\n parseAnyDate(data[col.code!]),\n col\n )\n }}\n </ng-container>\n\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.DATETIME\"\n >\n {{\n formatDateTimeWithColumn(\n parseAnyDate(data[col.code!]),\n col\n )\n }}\n </ng-container>\n </ng-container>\n\n <ng-template #normalTypes>\n <!-- STRING, MULTISELECT-->\n <ng-container\n *ngIf=\"\n [\n TableTypeEnum.STRING,\n TableTypeEnum.MULTISELECT,\n ].includes(col.type!)\n \"\n >\n {{ data[col.code!] }}\n </ng-container>\n </ng-template>\n </ng-template>\n </ng-template>\n </ng-template>\n </ng-template>\n </td>\n </ng-template>\n </ng-container>\n\n <!-- CHILD COLUMNS -->\n <ng-template #childColumns>\n <ng-container *ngFor=\"let child of col.children\">\n <td\n [style.width]=\"child.width || getHeaderWidth(child)\"\n [ngClass]=\"getDataAlignClass(child)\"\n >\n <ng-container\n *ngIf=\"isEditable(child.code); else childNormalTD\"\n >\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <input\n pInputText\n type=\"text\"\n [ngModel]=\"child.code ? data[child.code] : null\"\n (change)=\"onChange($event, data.id, child.code)\"\n />\n </ng-template>\n <ng-template pTemplate=\"output\">\n {{ child.code ? data[child.code] : \"\" }}\n </ng-template>\n </p-cellEditor>\n </ng-container>\n\n <ng-template #childNormalTD>\n {{ child.code ? data[child.code] : \"\" }}\n </ng-template>\n </td>\n </ng-container>\n </ng-template>\n </ng-container>\n </tr>\n </ng-template>\n </p-table>\n</div>\n", styles: [".pt-advanced-prime-table .bread-crumb{margin-bottom:15px}.pt-advanced-prime-table .date{width:100%;height:5rem;display:grid;justify-items:start;align-items:center}.pt-advanced-prime-table .filter-container{width:100%;display:flex;justify-content:space-between;align-items:center}.pt-advanced-prime-table .settings{display:flex;gap:1rem}.pt-advanced-prime-table .multi-select-container{display:flex;justify-content:center;align-items:center;gap:.3rem}.pt-advanced-prime-table ::ng-deep p-table{min-width:50rem}.pt-advanced-prime-table ::ng-deep .custom-multiselect .p-hidden-accessible input{display:none}.pt-advanced-prime-table ::ng-deep .p-datatable .p-sortable-column.p-highlight:hover{background:none}.pt-advanced-prime-table ::ng-deep .p-datatable .p-sortable-column:focus{box-shadow:none;outline:0 none}.pt-advanced-prime-table ::ng-deep .header-container{display:flex;justify-content:space-between;align-items:center;width:100%}.pt-advanced-prime-table ::ng-deep .header-title-left,.pt-advanced-prime-table ::ng-deep .header-title-center,.pt-advanced-prime-table ::ng-deep .header-title-right{flex:1}.pt-advanced-prime-table ::ng-deep .header-title-left{text-align:left}.pt-advanced-prime-table ::ng-deep .header-title-center{text-align:center}.pt-advanced-prime-table ::ng-deep .header-title-right{text-align:right}.pt-advanced-prime-table ::ng-deep .header-align-left{text-align:left}.pt-advanced-prime-table ::ng-deep .header-align-center{text-align:center}.pt-advanced-prime-table ::ng-deep .header-align-right{text-align:right}.pt-advanced-prime-table ::ng-deep p-columnfilter.p-element.ng-star-inserted{margin-top:4px}.pt-advanced-prime-table .flex{display:flex;justify-content:space-between;align-items:center}.pt-advanced-prime-table .ml-auto{margin-left:auto}.pt-advanced-prime-table ::ng-deep p-inputicon{margin-right:-1.5rem;z-index:2;position:relative}.pt-advanced-prime-table ::ng-deep .p-inputtext{padding-left:1.7rem}.pt-advanced-prime-table ::ng-deep .bt-filter-btn button{cursor:pointer;margin-left:1rem}.pt-advanced-prime-table ::ng-deep .p-icon-field-left .p-input-icon:first-of-type{left:-1rem}.pt-advanced-prime-table .table-row{text-align:center;display:flex;gap:1rem;justify-content:center}.pt-advanced-prime-table ::ng-deep span.p-button-icon.pi.pi-file-excel{font-size:1.25em;color:green}.pt-advanced-prime-table ::ng-deep span.p-button-icon.pi.pi-file-pdf{font-size:1.25em;color:red}.pt-advanced-prime-table .table-container{display:block;width:100%}.pt-advanced-prime-table ::ng-deep .p-datatable-emptymessage>td.empty-message-cell{padding:0!important}.pt-advanced-prime-table .empty-message-wrapper{width:100%;height:100%;min-height:180px;display:flex;align-items:center;justify-content:center}.pt-advanced-prime-table .empty-message{text-align:center;color:#888;font-size:1.2rem}.pt-advanced-prime-table .empty-message i{display:block;font-size:2rem;margin-bottom:.5rem}.pt-advanced-prime-table th{white-space:normal;word-wrap:break-word}.filter-image{width:22px;height:14px;margin-right:5px}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-left{text-align:left!important}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-center{text-align:center!important}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-right{text-align:right!important}.pt-advanced-prime-table ::ng-deep th.action-column,.pt-advanced-prime-table ::ng-deep td.action-column{text-align:center!important;justify-content:center;align-items:center;overflow:hidden;white-space:nowrap;padding:0}.pt-advanced-prime-table ::ng-deep .action-buttons-container{width:100%;display:flex;justify-content:center!important;align-items:center!important;gap:.35rem}.pt-advanced-prime-table ::ng-deep .action-buttons-container .p-button.p-button-rounded.p-button-text{padding:.25rem!important;min-width:22px!important;height:22px!important}\n"] }]
|
|
809
|
+
args: [{ selector: 'pt-advanced-prime-table', template: "<div class=\"pt-advanced-prime-table table-container\">\n <p-table\n #dt\n [value]=\"data\"\n [loading]=\"loading\"\n [rows]=\"rows\"\n [paginator]=\"isPaginated\"\n [globalFilterFields]=\"globalFilterFields\"\n [rowsPerPageOptions]=\"rowsPerPage\"\n [totalRecords]=\"totalRecords\"\n [lazy]=\"isLazy\"\n [filterDelay]=\"0\"\n dataKey=\"id\"\n styleClass=\"p-datatable-gridlines p-datatable-striped\"\n [scrollable]=\"true\"\n [scrollHeight]=\"maxHeight !== null ? maxHeight : undefined\"\n (onPage)=\"changePage($event)\"\n (onSort)=\"sortColumn($event)\"\n (onFilter)=\"filterColumn($event)\"\n >\n <ng-template pTemplate=\"colgroup\" let-columns>\n <colgroup>\n <col\n *ngFor=\"let col of columns\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n />\n </colgroup>\n </ng-template>\n\n <ng-template pTemplate=\"caption\">\n <div class=\"flex\">\n <div>\n <h3>Total: {{ totalRecords }}</h3>\n </div>\n\n <div>\n <button\n *ngIf=\"hasSearchFilter\"\n pButton\n icon=\"pi pi-filter-slash\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"clear(dt)\"\n title=\"Clear filters\"\n ></button>\n\n <button\n *ngIf=\"hasExportExcel\"\n pButton\n icon=\"pi pi-file-excel\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"exportExcel()\"\n title=\"Export to Excel\"\n ></button>\n\n <button\n *ngIf=\"hasExportPDF\"\n pButton\n icon=\"pi pi-file-pdf\"\n class=\"p-button-rounded p-button-text\"\n (click)=\"exportPdf()\"\n title=\"Export to PDF\"\n ></button>\n </div>\n\n <div class=\"ml-auto\" *ngIf=\"hasSearchFilter\">\n <p-iconField iconPosition=\"left\" class=\"ml-auto\">\n <p-inputIcon>\n <i class=\"pi pi-search\"></i>\n </p-inputIcon>\n <input\n pInputText\n type=\"text\"\n [(ngModel)]=\"searchValue\"\n (input)=\"filterGlobal($event)\"\n placeholder=\"Search keyword\"\n />\n </p-iconField>\n </div>\n </div>\n </ng-template>\n\n <ng-template pTemplate=\"header\">\n <tr class=\"sticky-header\">\n <ng-container *ngFor=\"let col of columns\">\n <th\n *ngIf=\"!col.children; else groupHeader\"\n [pSortableColumn]=\"col.code\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [ngClass]=\"[\n getHeaderAlignClass(col),\n col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n ]\"\n colspan=\"1\"\n >\n <ng-container\n *ngIf=\"isSortable && col.isSortable !== false; else noSortHeader\"\n >\n <div\n class=\"header-container d-flex align-items-center justify-content-between\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [style.margin]=\"'10px'\"\n >\n <span [ngClass]=\"getHeaderTitleClass(col)\">\n {{ col.title }}\n </span>\n\n <div\n class=\"icons d-flex align-items-center\"\n [style.width]=\"'77px'\"\n >\n <p-sortIcon [field]=\"col.code\" />\n\n <ng-container\n *ngIf=\"hasColumnFilter && col.isFilter !== false\"\n >\n <!-- COMPOSED FILTER -->\n <p-columnFilter\n *ngIf=\"col.type === TableTypeEnum.COMPOSED\"\n display=\"menu\"\n [field]=\"col.code\"\n type=\"text\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n [showMatchModes]=\"false\"\n [showOperator]=\"false\"\n (onClear)=\"onComposedColumnClear(col)\"\n >\n <ng-template\n pTemplate=\"filter\"\n let-filter=\"filterCallback\"\n >\n <div *ngFor=\"let composedName of col.composedNames\">\n <ng-container\n *ngIf=\"\n getComposedFieldType(col, composedName) ===\n TableTypeEnum.STRING\n \"\n >\n <p-multiSelect\n [options]=\"filters[composedName]?.options\"\n optionLabel=\"label\"\n optionValue=\"value\"\n [ngModel]=\"filters[composedName]?.value ?? []\"\n (ngModelChange)=\"\n onComposedFilterValueChange(\n col,\n composedName,\n $event\n );\n filter($event ?? [])\n \"\n [placeholder]=\"filters[composedName]?.placeholder\"\n display=\"chip\"\n >\n <ng-template let-item pTemplate=\"item\">\n <div class=\"custom-multiselect-item\">\n <img\n *ngIf=\"item.image\"\n [src]=\"item.image\"\n alt=\"icon\"\n class=\"filter-image\"\n />\n <span>{{ item.label }}</span>\n </div>\n </ng-template>\n </p-multiSelect>\n </ng-container>\n </div>\n </ng-template>\n </p-columnFilter>\n\n <!-- OTHER TYPES -->\n <p-columnFilter\n *ngIf=\"col.type !== TableTypeEnum.COMPOSED\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n [showMatchModes]=\"col.type !== TableTypeEnum.MULTISELECT\"\n [showOperator]=\"col.type !== TableTypeEnum.MULTISELECT\"\n [matchMode]=\"\n col.type === TableTypeEnum.MULTISELECT\n ? 'in'\n : undefined\n \"\n (onClear)=\"onFilterClear(col.code!)\"\n >\n <!-- NUMBER / AMOUNT -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"\n col.type === TableTypeEnum.NUMBER ||\n col.type === TableTypeEnum.AMOUNT\n \"\n let-value\n >\n <input\n pInputText\n type=\"text\"\n inputmode=\"decimal\"\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onNumberFilterChange(col.code!, $event)\n \"\n placeholder=\"Enter a number\"\n />\n </ng-template>\n\n <!-- DATE -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"col.type === TableTypeEnum.DATE\"\n let-filter=\"filterCallback\"\n let-value\n >\n <p-calendar\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event);\n filter($event)\n \"\n dateFormat=\"dd/mm/yy\"\n placeholder=\"Choose a date\"\n ></p-calendar>\n </ng-template>\n\n <!-- MULTISELECT -->\n <ng-template\n pTemplate=\"filter\"\n *ngIf=\"\n col.type === TableTypeEnum.MULTISELECT &&\n col.filterOptions &&\n col.filterOptions.length > 0\n \"\n let-filter=\"filterCallback\"\n let-value\n >\n <p-multiSelect\n [options]=\"col.filterOptions\"\n optionLabel=\"label\"\n optionValue=\"value\"\n [ngModel]=\"\n latestFilterValues[col.code!] ?? value ?? []\n \"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event);\n filter($event ?? [])\n \"\n display=\"chip\"\n placeholder=\"Choose option\"\n class=\"custom-multiselect\"\n ></p-multiSelect>\n </ng-template>\n </p-columnFilter>\n </ng-container>\n </div>\n </div>\n </ng-container>\n\n <ng-template #noSortHeader>\n <div\n class=\"header-container d-flex align-items-center justify-content-between\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [style.padding]=\"'0px'\"\n [style.margin]=\"'10px'\"\n >\n <span [ngClass]=\"getHeaderTitleClass(col)\">\n {{ col.title }}\n </span>\n\n <ng-container *ngIf=\"hasColumnFilter && col.isFilter !== false\">\n <p-columnFilter\n *ngIf=\"col.type === 'AMOUNT'\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [currency]=\"getCurrencySymbol(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n >\n <ng-template pTemplate=\"filter\" let-value>\n <input\n pInputText\n type=\"text\"\n inputmode=\"decimal\"\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event)\n \"\n placeholder=\"Enter an amount\"\n />\n </ng-template>\n </p-columnFilter>\n\n <p-columnFilter\n *ngIf=\"col.type !== 'AMOUNT'\"\n display=\"menu\"\n [field]=\"col.code\"\n [type]=\"getColumnFilterType(col)\"\n [showApplyButton]=\"true\"\n [showClearButton]=\"true\"\n [showMatchModes]=\"\n getColumnFilterType(col) !== 'multiSelect'\n \"\n [showOperator]=\"getColumnFilterType(col) !== 'multiSelect'\"\n [matchMode]=\"\n getColumnFilterType(col) === 'multiSelect'\n ? 'in'\n : undefined\n \"\n (onClear)=\"onFilterClear(col.code!)\"\n >\n <!-- DATE -->\n <ng-template\n pTemplate=\"filter\"\n let-filter=\"filterCallback\"\n let-value\n *ngIf=\"getColumnFilterType(col) === 'date'\"\n >\n <p-calendar\n [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event);\n filter($event)\n \"\n dateFormat=\"dd/mm/yy\"\n ></p-calendar>\n </ng-template>\n\n <!-- MULTISELECT -->\n <ng-template\n pTemplate=\"filter\"\n let-filter=\"filterCallback\"\n let-value\n *ngIf=\"getColumnFilterType(col) === 'multiSelect'\"\n >\n <p-multiSelect\n [options]=\"col.filterOptions\"\n optionLabel=\"label\"\n optionValue=\"value\"\n [ngModel]=\"latestFilterValues[col.code!] ?? value ?? []\"\n (ngModelChange)=\"\n onFilterValueChange(col.code!, null, $event);\n filter($event ?? [])\n \"\n display=\"chip\"\n placeholder=\"Select\"\n class=\"custom-multiselect\"\n ></p-multiSelect>\n </ng-template>\n </p-columnFilter>\n </ng-container>\n </div>\n </ng-template>\n </th>\n\n <ng-template #groupHeader>\n <th\n [attr.colspan]=\"col.children?.length\"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"getHeaderAlignClass(col)\"\n >\n <span>{{ col.title }}</span>\n </th>\n </ng-template>\n </ng-container>\n </tr>\n\n <tr *ngIf=\"hasGroupedColumns\">\n <ng-container *ngFor=\"let col of columns\">\n <ng-container *ngIf=\"col.children\">\n <th\n *ngFor=\"let child of col.children\"\n [style.width]=\"child.width || getHeaderWidth(child)\"\n [style.padding]=\"'0px'\"\n ></th>\n </ng-container>\n </ng-container>\n </tr>\n </ng-template>\n\n <ng-template pTemplate=\"emptymessage\">\n <tr class=\"p-datatable-emptymessage\">\n <td class=\"empty-message-cell\" [attr.colspan]=\"columns.length || 1\">\n <div class=\"empty-message-wrapper\">\n <div class=\"empty-message\">\n <i class=\"pi pi-info-circle\"></i>\n <p>No records available to display.</p>\n </div>\n </div>\n </td>\n </tr>\n </ng-template>\n\n <ng-template\n pTemplate=\"body\"\n let-data\n let-editing=\"editing\"\n let-ri=\"rowIndex\"\n >\n <tr *ngIf=\"!loading\" [pEditableRow]=\"isEdit ? data : null\">\n <ng-container *ngFor=\"let col of columns\">\n <ng-container *ngIf=\"!col.children; else childColumns\">\n <td\n *ngIf=\"\n isEditable(col.code!) && col.type !== TableTypeEnum.ACTION;\n else normalTD\n \"\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"getDataAlignClass(col)\"\n >\n <ng-container *ngIf=\"isMultiSelect(col.code); else datePicker\">\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <p-multiSelect\n appendTo=\"body\"\n [ngModel]=\"data[col.code!]\"\n [style]=\"{ width: '100%' }\"\n (ngModelChange)=\"changeHandler(data.id, col.code, $event)\"\n [options]=\"optionValues\"\n ></p-multiSelect>\n </ng-template>\n <ng-template pTemplate=\"output\">\n <div class=\"multi-select-container\">\n <ng-container *ngFor=\"let rec of data[col.code!]\">\n <p-tag [value]=\"rec\"></p-tag>\n </ng-container>\n </div>\n </ng-template>\n </p-cellEditor>\n </ng-container>\n\n <ng-template #datePicker>\n <ng-container *ngIf=\"isDatePicker(col.code); else normalInput\">\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <p-calendar\n [inputId]=\"data[col.code!]\"\n [ngModel]=\"data[col.code!]\"\n (ngModelChange)=\"\n changeHandler(data.id, col.code, $event)\n \"\n [dateFormat]=\"'dd/mm/yy'\"\n ></p-calendar>\n </ng-template>\n <ng-template pTemplate=\"output\">\n {{ data[col.code!] | customDate }}\n </ng-template>\n </p-cellEditor>\n </ng-container>\n </ng-template>\n\n <ng-template #normalInput>\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <input\n pInputText\n type=\"text\"\n [ngModel]=\"data[col.code!]\"\n (change)=\"onChange($event, data.id, col.code)\"\n />\n </ng-template>\n <ng-template pTemplate=\"output\">\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.AMOUNT;\n else normalOutput\n \"\n >\n {{\n data[col.code!]\n | customCurrency\n : getCurrencySymbol(col)\n : col.decimalPlaces\n : col.thousandSeparator\n : col.decimalSeparator\n }}\n </ng-container>\n <ng-template #normalOutput>\n {{ data[col.code!] }}\n </ng-template>\n </ng-template>\n </p-cellEditor>\n </ng-template>\n </td>\n\n <ng-template #normalTD>\n <td\n [style.width]=\"col.width || getHeaderWidth(col)\"\n [ngClass]=\"[\n getDataAlignClass(col),\n col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n ]\"\n >\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.ACTION; else nonActionCell\"\n >\n <div class=\"action-buttons-container\">\n <button\n *ngIf=\"isDelete\"\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-trash\"\n (click)=\"Delete(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n\n <div *ngIf=\"isEdit\">\n <button\n pInitEditableRow\n *ngIf=\"!editing\"\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-pencil\"\n (click)=\"initEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n <button\n *ngIf=\"editing\"\n pSaveEditableRow\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-check\"\n (click)=\"saveEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n <button\n *ngIf=\"editing\"\n pCancelEditableRow\n pButton\n pRipple\n type=\"button\"\n icon=\"pi pi-times\"\n (click)=\"cancelEditableRow(data)\"\n class=\"p-button-rounded p-button-text\"\n ></button>\n </div>\n\n <ng-container *ngFor=\"let act of customActions\">\n <button\n *ngIf=\"isActionVisible(act, data)\"\n pButton\n pRipple\n type=\"button\"\n class=\"p-button-rounded p-button-text\"\n [icon]=\"act.icon || 'pi pi-ellipsis-h'\"\n [ngClass]=\"act.styleClass\"\n [disabled]=\"isActionDisabled(act, data)\"\n (click)=\"onCustomActionClick(act, data)\"\n ></button>\n </ng-container>\n </div>\n </ng-container>\n\n <ng-template #nonActionCell>\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.COMPOSED;\n else nonComposed\n \"\n >\n <div class=\"composed-cell\">\n <ng-container\n *ngFor=\"\n let composedName of col.composedNames;\n let i = index\n \"\n >\n <ng-container\n *ngIf=\"\n col.composedTypes &&\n col.composedTypes[i] === TableTypeEnum.IMAGE\n \"\n >\n <img\n [src]=\"data[col.code!]?.[composedName]\"\n alt=\"composed-img\"\n class=\"composed-image\"\n [ngStyle]=\"\n getImageStyle(col.composedStyles?.[composedName])\n \"\n />\n </ng-container>\n\n <ng-container\n *ngIf=\"\n col.composedTypes &&\n col.composedTypes[i] === TableTypeEnum.STRING\n \"\n >\n <span\n class=\"composed-text\"\n [ngStyle]=\"\n getTitleStyle(col.composedStyles?.[composedName])\n \"\n >\n {{ data[col.code!]?.[composedName] }}\n </span>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n\n <ng-template #nonComposed>\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.AMOUNT; else nonAmount\"\n >\n {{\n data[col.code!]\n | customCurrency\n : getCurrencySymbol(col)\n : col.decimalPlaces\n : col.thousandSeparator\n : col.decimalSeparator\n }}\n </ng-container>\n\n <ng-template #nonAmount>\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.NUMBER;\n else nonNumber\n \"\n >\n {{\n formatNumber(\n data[col.code!],\n col.decimalPlaces,\n col.thousandSeparator,\n col.decimalSeparator\n )\n }}\n </ng-container>\n\n <ng-template #nonNumber>\n <ng-container\n *ngIf=\"\n col.type === TableTypeEnum.DATE ||\n col.type === TableTypeEnum.DATETIME;\n else normalTypes\n \"\n >\n <ng-container *ngIf=\"col.type === TableTypeEnum.DATE\">\n {{\n formatDateWithColumn(\n parseAnyDate(data[col.code!]),\n col\n )\n }}\n </ng-container>\n\n <ng-container\n *ngIf=\"col.type === TableTypeEnum.DATETIME\"\n >\n {{\n formatDateTimeWithColumn(\n parseAnyDate(data[col.code!]),\n col\n )\n }}\n </ng-container>\n </ng-container>\n\n <ng-template #normalTypes>\n <ng-container\n *ngIf=\"\n [\n TableTypeEnum.STRING,\n TableTypeEnum.MULTISELECT,\n ].includes(col.type!)\n \"\n >\n {{ data[col.code!] }}\n </ng-container>\n </ng-template>\n </ng-template>\n </ng-template>\n </ng-template>\n </ng-template>\n </td>\n </ng-template>\n </ng-container>\n\n <ng-template #childColumns>\n <ng-container *ngFor=\"let child of col.children\">\n <td\n [style.width]=\"child.width || getHeaderWidth(child)\"\n [ngClass]=\"getDataAlignClass(child)\"\n >\n <ng-container\n *ngIf=\"isEditable(child.code); else childNormalTD\"\n >\n <p-cellEditor>\n <ng-template pTemplate=\"input\">\n <input\n pInputText\n type=\"text\"\n [ngModel]=\"child.code ? data[child.code] : null\"\n (change)=\"onChange($event, data.id, child.code)\"\n />\n </ng-template>\n <ng-template pTemplate=\"output\">\n {{ child.code ? data[child.code] : \"\" }}\n </ng-template>\n </p-cellEditor>\n </ng-container>\n\n <ng-template #childNormalTD>\n {{ child.code ? data[child.code] : \"\" }}\n </ng-template>\n </td>\n </ng-container>\n </ng-template>\n </ng-container>\n </tr>\n </ng-template>\n </p-table>\n</div>\n", styles: [".pt-advanced-prime-table .bread-crumb{margin-bottom:15px}.pt-advanced-prime-table .date{width:100%;height:5rem;display:grid;justify-items:start;align-items:center}.pt-advanced-prime-table .filter-container{width:100%;display:flex;justify-content:space-between;align-items:center}.pt-advanced-prime-table .settings{display:flex;gap:1rem}.pt-advanced-prime-table .multi-select-container{display:flex;justify-content:center;align-items:center;gap:.3rem}.pt-advanced-prime-table ::ng-deep p-table{min-width:50rem}.pt-advanced-prime-table ::ng-deep .custom-multiselect .p-hidden-accessible input{display:none}.pt-advanced-prime-table ::ng-deep .p-datatable .p-sortable-column.p-highlight:hover{background:none}.pt-advanced-prime-table ::ng-deep .p-datatable .p-sortable-column:focus{box-shadow:none;outline:0 none}.pt-advanced-prime-table ::ng-deep .header-container{display:flex;justify-content:space-between;align-items:center;width:100%}.pt-advanced-prime-table ::ng-deep .header-title-left,.pt-advanced-prime-table ::ng-deep .header-title-center,.pt-advanced-prime-table ::ng-deep .header-title-right{flex:1}.pt-advanced-prime-table ::ng-deep .header-title-left{text-align:left}.pt-advanced-prime-table ::ng-deep .header-title-center{text-align:center}.pt-advanced-prime-table ::ng-deep .header-title-right{text-align:right}.pt-advanced-prime-table ::ng-deep .header-align-left{text-align:left}.pt-advanced-prime-table ::ng-deep .header-align-center{text-align:center}.pt-advanced-prime-table ::ng-deep .header-align-right{text-align:right}.pt-advanced-prime-table ::ng-deep p-columnfilter.p-element.ng-star-inserted{margin-top:4px}.pt-advanced-prime-table .flex{display:flex;justify-content:space-between;align-items:center}.pt-advanced-prime-table .ml-auto{margin-left:auto}.pt-advanced-prime-table ::ng-deep p-inputicon{margin-right:-1.5rem;z-index:2;position:relative}.pt-advanced-prime-table ::ng-deep .p-inputtext{padding-left:1.7rem}.pt-advanced-prime-table ::ng-deep .bt-filter-btn button{cursor:pointer;margin-left:1rem}.pt-advanced-prime-table ::ng-deep .p-icon-field-left .p-input-icon:first-of-type{left:-1rem}.pt-advanced-prime-table .table-row{text-align:center;display:flex;gap:1rem;justify-content:center}.pt-advanced-prime-table ::ng-deep span.p-button-icon.pi.pi-file-excel{font-size:1.25em;color:green}.pt-advanced-prime-table ::ng-deep span.p-button-icon.pi.pi-file-pdf{font-size:1.25em;color:red}.pt-advanced-prime-table .table-container{display:block;width:100%}.pt-advanced-prime-table ::ng-deep .p-datatable-emptymessage>td.empty-message-cell{padding:0!important}.pt-advanced-prime-table .empty-message-wrapper{width:100%;height:100%;min-height:180px;display:flex;align-items:center;justify-content:center}.pt-advanced-prime-table .empty-message{text-align:center;color:#888;font-size:1.2rem}.pt-advanced-prime-table .empty-message i{display:block;font-size:2rem;margin-bottom:.5rem}.pt-advanced-prime-table th{white-space:normal;word-wrap:break-word}.filter-image{width:22px;height:14px;margin-right:5px}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-left{text-align:left!important}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-center{text-align:center!important}.pt-advanced-prime-table ::ng-deep .p-datatable-tbody>tr>td.cell-align-right{text-align:right!important}.pt-advanced-prime-table ::ng-deep th.action-column,.pt-advanced-prime-table ::ng-deep td.action-column{text-align:center!important;justify-content:center;align-items:center;overflow:hidden;white-space:nowrap;padding:0}.pt-advanced-prime-table ::ng-deep .action-buttons-container{width:100%;display:flex;justify-content:center!important;align-items:center!important;gap:.35rem}.pt-advanced-prime-table ::ng-deep .action-buttons-container .p-button.p-button-rounded.p-button-text{padding:.25rem!important;min-width:22px!important;height:22px!important}\n"] }]
|
|
729
810
|
}], propDecorators: { data: [{
|
|
730
811
|
type: Input
|
|
731
812
|
}], columns: [{
|
|
@@ -754,6 +835,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
|
|
|
754
835
|
type: Input
|
|
755
836
|
}], maxHeight: [{
|
|
756
837
|
type: Input
|
|
838
|
+
}], lazyLoad: [{
|
|
839
|
+
type: Output
|
|
757
840
|
}], search: [{
|
|
758
841
|
type: Output
|
|
759
842
|
}], exportExcelEvent: [{
|
|
@@ -770,4 +853,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.11", ngImpo
|
|
|
770
853
|
type: ViewChild,
|
|
771
854
|
args: ['dt', { static: false }]
|
|
772
855
|
}] } });
|
|
773
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-advanced-prime-table.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { Table } from 'primeng/table';\nimport {\n  AlignEnum,\n  ImageStyle,\n  TableColumn,\n  TableTypeEnum,\n  TitleStyle,\n} from '../../public-api';\nimport { calculateTextWidth } from '../utils/text.util';\n\nexport interface PTTableActionConfig {\n  code: string;\n  icon?: string;\n  styleClass?: string;\n  tooltip?: string;\n  action: any;\n  visible?: boolean | ((row: any) => boolean);\n  disabled?: boolean | ((row: any) => boolean);\n}\n\n@Component({\n  selector: 'pt-advanced-prime-table',\n  templateUrl: './pt-advanced-prime-table.component.html',\n  styleUrl: './pt-advanced-prime-table.component.css',\n})\nexport class PTAdvancedPrimeTableComponent implements OnInit, OnChanges {\n  @Input() data: any[] = [];\n  @Input() columns: TableColumn[] = [];\n  @Input() totalRecords = 0;\n  @Input() rowsPerPage: number[] = [10, 20, 30];\n  @Input() hasSearchFilter = false;\n  @Input() hasExportExcel = false;\n  @Input() hasExportPDF = false;\n  @Input() hasColumnFilter = false;\n\n  /** paginator UI */\n  @Input() isPaginated = true;\n\n  /** ✅ server-side mode (PrimeNG lazy = true) */\n  @Input() isLazy = false;\n\n  @Input() actions: PTTableActionConfig[] = [];\n  @Input() isSortable = false;\n  @Input() loading = false;\n  @Input() maxHeight: string | null = null;\n\n  @Output() search = new EventEmitter<string>();\n  @Output() exportExcelEvent = new EventEmitter<void>();\n  @Output() exportPdfEvent = new EventEmitter<void>();\n  @Output() onPageChange = new EventEmitter<{ page: number; rows: number }>();\n  @Output() onSortColumn = new EventEmitter<any>();\n  @Output() onFilterColumn = new EventEmitter<any>();\n\n  @ViewChild('dt', { static: false }) dt!: Table;\n\n  public TableTypeEnum = TableTypeEnum;\n  public AlignEnum = AlignEnum;\n\n  searchValue = '';\n\n  public filters: { [key: string]: any } = {};\n  latestFilterValues: { [field: string]: any } = {};\n  private clearedFields = new Set<string>();\n\n  private validCurrencyCodes = ['USD', 'EUR', 'MAD'];\n  iconWidth = 77;\n\n  rows = 0;\n  hasGroupedColumns = false;\n\n  isDelete = false;\n  isEdit = false;\n\n  Delete: (value: any) => void = () => {};\n  initEditableRow: (data: any) => void = () => {};\n  saveEditableRow: (data: any) => void = () => {};\n  cancelEditableRow: (item: any) => void = () => {};\n\n  customActions: PTTableActionConfig[] = [];\n\n  dataMap = new Map<any, any>();\n  map = new Map<any, Map<any, any>>();\n  optionEntries = new Map<string, any[]>();\n  optionValues: any[] = [];\n  globalFilterFields: string[] = [];\n\n  ngOnInit(): void {\n    this.hasGroupedColumns = this.columns.some(\n      (col) => col.children && col.children.length > 0,\n    );\n\n    this.globalFilterFields = this.columns\n      .filter((col) => col.code !== undefined && col.isFilter !== false)\n      .map((col) => col.code as string);\n\n    this.initializePagination();\n    this.initializeActions();\n\n    this.columns.forEach((col) => {\n      if (col.type === TableTypeEnum.ACTION) {\n        col.isEditable = false;\n        col.isFilter = false;\n        col.isSortable = false;\n      }\n\n      if (col.type === TableTypeEnum.COMPOSED) {\n        this.initializeComposedFilters(col);\n      }\n\n      if (col.isSortable === undefined) col.isSortable = true;\n      if (col.isEditable === undefined) col.isEditable = true;\n\n      if (col.isFilter !== false && col.code !== undefined) {\n        this.globalFilterFields.push(col.code);\n      }\n\n      if (!col.width) col.width = this.calculateColumnWidth(col);\n    });\n\n    // ✅ client-side mode: totalRecords = data.length by default\n    if (!this.isLazy) {\n      this.totalRecords = this.data?.length ?? 0;\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // ✅ when data changes in client-side mode, keep paginator consistent\n    if (changes['data'] && !this.isLazy) {\n      this.totalRecords = this.data?.length ?? 0;\n\n      // if we were on page > 1, go back to first page (avoids empty view)\n      if (this.dt) this.dt.first = 0;\n    }\n  }\n\n  // ---------- HEADER + BODY ALIGNMENT HELPERS ----------\n\n  getHeaderTitleClass(col: TableColumn): string {\n    const align = col.headerAlign ?? AlignEnum.LEFT;\n    switch (align) {\n      case AlignEnum.CENTER:\n        return 'header-title-center';\n      case AlignEnum.RIGHT:\n        return 'header-title-right';\n      default:\n        return 'header-title-left';\n    }\n  }\n\n  getHeaderAlignClass(col: TableColumn): string {\n    const align = col.headerAlign ?? AlignEnum.LEFT;\n    switch (align) {\n      case AlignEnum.CENTER:\n        return 'header-align-center';\n      case AlignEnum.RIGHT:\n        return 'header-align-right';\n      default:\n        return 'header-align-left';\n    }\n  }\n\n  getDataAlignClass(col: TableColumn): string {\n    const effectiveAlign =\n      col.dataAlign ??\n      (col.type === TableTypeEnum.NUMBER || col.type === TableTypeEnum.AMOUNT\n        ? AlignEnum.RIGHT\n        : AlignEnum.LEFT);\n\n    switch (effectiveAlign) {\n      case AlignEnum.CENTER:\n        return 'cell-align-center';\n      case AlignEnum.RIGHT:\n        return 'cell-align-right';\n      default:\n        return 'cell-align-left';\n    }\n  }\n\n  // ---------- ACTIONS INITIALIZATION ----------\n\n  private initializeActions(): void {\n    this.isDelete = false;\n    this.isEdit = false;\n    this.Delete = () => {};\n    this.initEditableRow = () => {};\n    this.saveEditableRow = () => {};\n    this.cancelEditableRow = () => {};\n    this.customActions = [];\n\n    if (!this.actions || this.actions.length === 0) return;\n\n    this.actions.forEach((action) => {\n      switch (action.code) {\n        case 'delete':\n          this.isDelete = true;\n          this.Delete = (row: any) => action.action(row);\n          break;\n        case 'edit':\n          this.initializeEditActions(action);\n          break;\n        default:\n          this.customActions.push(action);\n          break;\n      }\n    });\n  }\n\n  private initializeEditActions(action: PTTableActionConfig): void {\n    this.isEdit = true;\n    this.initEditableRow = (data: any) => action.action.init(data);\n    this.saveEditableRow = (data: any) => {\n      const record = this.map.get(data.id);\n      action.action.save(data, record);\n      this.dataMap.clear();\n    };\n    this.cancelEditableRow = (item: any) => console.log(item);\n  }\n\n  onCustomActionClick(action: PTTableActionConfig, row: any): void {\n    if (!this.isActionVisible(action, row)) return;\n    if (this.isActionDisabled(action, row)) return;\n\n    if (action && typeof action.action === 'function') {\n      setTimeout(() => action.action(row), 0);\n    }\n  }\n\n  // ---------- FILTERING / SORTING / PAGINATION ----------\n\n  private initializeComposedFilters(col: TableColumn): void {\n    col.composedNames?.forEach((composedName) => {\n      const code = (col.code as string) || '';\n      this.globalFilterFields.push(code + '.' + composedName);\n      this.filters[composedName] = {\n        options: col.filterOptions,\n        value: [],\n        label: 'Filter by ' + composedName,\n        placeholder: 'Select option',\n      };\n    });\n  }\n\n  getComposedFieldType(\n    col: TableColumn,\n    composedName: string,\n  ): TableTypeEnum | undefined {\n    if (col.composedNames && col.composedTypes) {\n      const index = col.composedNames.indexOf(composedName);\n      if (index >= 0 && index < col.composedTypes.length) {\n        return col.composedTypes[index];\n      }\n    }\n    return undefined;\n  }\n\n  onComposedColumnClear(col: TableColumn): void {\n    if (!col.code || !col.composedNames) return;\n\n    col.composedNames.forEach((name) => {\n      const key = `${col.code}.${name}`;\n      delete this.latestFilterValues[key];\n      if (this.filters[name]) this.filters[name].value = [];\n    });\n  }\n\n  onComposedFilterValueChange(\n    col: TableColumn,\n    composedName: string,\n    value: any[],\n    filterModel?: any,\n  ): void {\n    const key = `${col.code}.${composedName}`;\n\n    if (filterModel) {\n      filterModel.value = value;\n      if (\n        Array.isArray(filterModel.constraints) &&\n        filterModel.constraints.length > 0\n      ) {\n        filterModel.constraints[0].value = value;\n      }\n    }\n\n    if (!this.filters[composedName]) {\n      this.filters[composedName] = {\n        options: col.filterOptions,\n        value: [],\n        placeholder: 'Select option',\n      };\n    }\n    this.filters[composedName].value = value || [];\n\n    const isEmpty = !value || (Array.isArray(value) && value.length === 0);\n    if (isEmpty) delete this.latestFilterValues[key];\n    else this.latestFilterValues[key] = value;\n  }\n\n  onFilterClear(field: string | undefined): void {\n    if (!field) return;\n    delete this.latestFilterValues[field];\n    this.clearedFields.add(field);\n  }\n\n  onFilterValueChange(\n    field: string | undefined,\n    filterModel: any,\n    value: any,\n  ): void {\n    if (!field) return;\n\n    const isEmpty =\n      value === null ||\n      value === undefined ||\n      (Array.isArray(value) && value.length === 0);\n\n    if (isEmpty) delete this.latestFilterValues[field];\n    else this.latestFilterValues[field] = value;\n\n    if (filterModel) {\n      filterModel.value = value;\n      if (\n        Array.isArray(filterModel.constraints) &&\n        filterModel.constraints.length > 0\n      ) {\n        filterModel.constraints[0].value = value;\n      }\n    }\n  }\n\n  onNumberFilterChange(field: string | undefined, value: any): void {\n    if (!field) return;\n\n    const isEmpty =\n      value === null ||\n      value === undefined ||\n      value === '' ||\n      (Array.isArray(value) && value.length === 0);\n\n    if (isEmpty) delete this.latestFilterValues[field];\n    else this.latestFilterValues[field] = value;\n  }\n\n  private findColumnByField(field: string): TableColumn | undefined {\n    return this.columns.find(\n      (c) => c.code === field || (c.code && field.startsWith(c.code + '.')),\n    );\n  }\n\n  filterColumn(event: any): void {\n    const filters = event?.filters;\n    if (!filters) {\n      if (this.isLazy) this.onFilterColumn.emit(event);\n      return;\n    }\n\n    const isNullish = (v: any) => v === null || v === undefined || v === '';\n\n    Object.keys(filters).forEach((field) => {\n      const meta = filters[field];\n      const normalizeMeta = (m: any) =>\n        Array.isArray(m) && m.length > 0 ? m[0] : m;\n      const m = normalizeMeta(meta);\n      const col = this.findColumnByField(field);\n      const wasCleared = this.clearedFields.has(field);\n\n      if (!m && !col) return;\n\n      // COMPOSED\n      if (col && col.type === TableTypeEnum.COMPOSED) {\n        const composedValues: any = {};\n\n        col.composedNames?.forEach((name) => {\n          const key = `${col.code}.${name}`;\n          const val = this.latestFilterValues[key];\n          const empty =\n            isNullish(val) || (Array.isArray(val) && val.length === 0);\n          if (!empty) composedValues[name] = val;\n        });\n\n        if (Object.keys(composedValues).length === 0) {\n          if (m) {\n            m.value = null;\n            if (Array.isArray(m.constraints))\n              m.constraints.forEach((c: any) => (c.value = null));\n          }\n        } else {\n          m.value = composedValues;\n        }\n        return;\n      }\n\n      if (!m) return;\n\n      // CLEAR\n      if (wasCleared) {\n        delete this.latestFilterValues[field];\n        m.value = null;\n        if (Array.isArray(m.constraints) && m.constraints.length > 0) {\n          m.constraints[0].value = null;\n        }\n        this.clearedFields.delete(field);\n        return;\n      }\n\n      let value: any = m.value;\n      if (Array.isArray(m.constraints) && m.constraints.length > 0) {\n        const cVal = m.constraints[0].value;\n        if (!isNullish(cVal)) value = cVal;\n      }\n\n      const cached = this.latestFilterValues[field];\n      const hasCached =\n        !isNullish(cached) && (!Array.isArray(cached) || cached.length > 0);\n\n      if (\n        (isNullish(value) || (Array.isArray(value) && value.length === 0)) &&\n        hasCached\n      ) {\n        value = cached;\n      }\n\n      const isEmpty =\n        isNullish(value) || (Array.isArray(value) && value.length === 0);\n\n      if (isEmpty) {\n        delete this.latestFilterValues[field];\n        m.value = null;\n        if (Array.isArray(m.constraints) && m.constraints.length > 0) {\n          m.constraints[0].value = null;\n        }\n        return;\n      }\n\n      this.latestFilterValues[field] = value;\n\n      let emitValue: any = value;\n\n      // ✅ DATE / DATETIME normalization for server-side filtering\n      if (\n        col &&\n        (col.type === TableTypeEnum.DATE || col.type === TableTypeEnum.DATETIME)\n      ) {\n        const d = this.parseAnyDate(value);\n        emitValue =\n          col.type === TableTypeEnum.DATE\n            ? this.formatDateWithColumn(d, col)\n            : this.formatDateTimeWithColumn(d, col);\n      }\n\n      m.value = emitValue;\n      if (Array.isArray(m.constraints) && m.constraints.length > 0) {\n        m.constraints[0].value = emitValue;\n      }\n    });\n\n    // ✅ emit only in server-side mode\n    if (this.isLazy) {\n      this.onFilterColumn.emit(event);\n    } else {\n      // client-side: keep totalRecords in sync with filtered view\n      if (this.dt) {\n        const current = (this.dt.filteredValue ?? this.dt.value ?? []) as any[];\n        this.totalRecords = current?.length ?? 0;\n        this.dt.first = 0;\n      }\n    }\n  }\n\n  changePage(event: any): void {\n    const page = event.page ?? Math.floor((event.first || 0) / event.rows);\n    const rows = event.rows;\n    this.rows = rows;\n\n    // ✅ emit only when server-side pagination\n    if (this.isLazy) {\n      this.onPageChange.emit({ page, rows });\n    }\n  }\n\n  sortColumn(event: any): void {\n    // ✅ emit only when server-side sorting\n    if (!this.isLazy) return;\n\n    let field = event.field;\n    const col = this.columns.find((c) => c.code === field);\n\n    if (col && col.type === TableTypeEnum.COMPOSED) {\n      let textProp: string | undefined;\n\n      if (col.composedNames && col.composedTypes) {\n        const idx = col.composedTypes.findIndex(\n          (t) => t === TableTypeEnum.STRING,\n        );\n        if (idx >= 0 && idx < col.composedNames.length)\n          textProp = col.composedNames[idx];\n      }\n\n      if (!textProp && col.composedNames?.length)\n        textProp = col.composedNames[0];\n\n      if (textProp) field = `${field}.${textProp}`;\n    }\n\n    this.onSortColumn.emit({ ...event, field });\n  }\n\n  // ---------- EDIT HELPERS ----------\n\n  private parseDate_ddMMyyyy(dateString: string): Date | null {\n    const parts = dateString.split('/');\n    if (parts.length === 3) {\n      const day = parseInt(parts[0], 10);\n      const month = parseInt(parts[1], 10) - 1;\n      const year = parseInt(parts[2], 10);\n      const date = new Date(year, month, day);\n      return isNaN(date.getTime()) ? null : date;\n    }\n    return null;\n  }\n\n  onChange(event: Event, id: number, key: any) {\n    const target = event.target as HTMLInputElement;\n    this.changeHandler(id, key, target.value);\n  }\n\n  changeHandler(id: number, key: any, value: any) {\n    const column = this.columns.find((item) => item.code === key);\n\n    if (!this.map.get(id)) {\n      if (column?.type === TableTypeEnum.DATE) {\n        this.dataMap.set(key, this.parseDate_ddMMyyyy(value));\n      } else {\n        this.dataMap.set(key, value);\n      }\n      this.map.set(id, new Map(this.dataMap));\n    } else {\n      const mapItem = this.map.get(id) as Map<any, any>;\n      if (column?.type === TableTypeEnum.DATE) {\n        mapItem.set(key, this.parseDate_ddMMyyyy(value));\n      } else {\n        mapItem.set(key, value);\n      }\n    }\n  }\n\n  // ---------- TYPES ----------\n\n  getColumnFilterType(column: TableColumn): string {\n    switch (column.type) {\n      case TableTypeEnum.STRING:\n        return 'text';\n      case TableTypeEnum.AMOUNT:\n      case TableTypeEnum.NUMBER:\n        return 'numeric';\n      case TableTypeEnum.DATE:\n      case TableTypeEnum.DATETIME:\n        return 'date';\n      case TableTypeEnum.MULTISELECT:\n        return 'multiSelect';\n      case TableTypeEnum.BOOLEAN:\n        return 'boolean';\n      case TableTypeEnum.COMPOSED:\n        return 'composed';\n      default:\n        return 'text';\n    }\n  }\n\n  isEditable(key: string | undefined): boolean {\n    if (!key) return false;\n    const column = this.columns.find((item) => item.code === key);\n    return column?.isEditable !== false;\n  }\n\n  isMultiSelect(key: any): boolean {\n    const column = this.columns.find((item) => item.code === key);\n    if (\n      column?.type === TableTypeEnum.MULTISELECT &&\n      column.options &&\n      column.code !== undefined\n    ) {\n      this.optionEntries = new Map([\n        [column.code as string, Object.values(column.options)],\n      ]);\n      this.optionValues = this.optionEntries.get(key) || [];\n      return true;\n    }\n    return false;\n  }\n\n  isDatePicker(key: any): boolean {\n    const t = this.columns.find((item) => item.code === key)?.type;\n    return t === TableTypeEnum.DATE || t === TableTypeEnum.DATETIME;\n  }\n\n  isDateTimePicker(key: any): boolean {\n    const t = this.columns.find((item) => item.code === key)?.type;\n    return t === TableTypeEnum.DATETIME;\n  }\n\n  // ---------- SEARCH ----------\n\n  filterGlobal(event: Event): void {\n    const target = event.target as HTMLInputElement;\n    const value = (target.value || '').toLowerCase();\n\n    // ✅ server-side mode\n    if (this.isLazy) {\n      this.search.emit(value);\n      return;\n    }\n\n    // ✅ client-side mode\n    if (!value) {\n      this.dt.value = [...(this.data ?? [])];\n      this.totalRecords = (this.dt.value ?? []).length;\n      this.dt.first = 0;\n      return;\n    }\n\n    const filteredData = (this.data ?? []).filter((item) =>\n      (this.globalFilterFields ?? []).some((field) => {\n        const column = this.columns?.find((col) => col.code === field);\n        if (!column) return false;\n\n        const cell = item?.[field];\n\n        if (column.type === TableTypeEnum.DATE) {\n          return this.formatDateWithColumn(this.parseAnyDate(cell), column)\n            .toLowerCase()\n            .includes(value);\n        }\n\n        if (column.type === TableTypeEnum.DATETIME) {\n          return this.formatDateTimeWithColumn(this.parseAnyDate(cell), column)\n            .toLowerCase()\n            .includes(value);\n        }\n\n        if (\n          column.type === TableTypeEnum.AMOUNT ||\n          column.type === TableTypeEnum.NUMBER\n        ) {\n          return String(cell ?? '')\n            .toLowerCase()\n            .includes(value);\n        }\n\n        if (column.type === TableTypeEnum.COMPOSED) {\n          return this.filterComposedColumn(cell, value);\n        }\n\n        return String(cell ?? '')\n          .toLowerCase()\n          .includes(value);\n      }),\n    );\n\n    this.dt.value = filteredData;\n    this.totalRecords = filteredData.length;\n    this.dt.first = 0;\n  }\n\n  private filterComposedColumn(composedData: any, value: string): boolean {\n    if (composedData) {\n      return Object.keys(composedData).some((key) => {\n        const cellValue = composedData[key];\n        return typeof cellValue === 'string'\n          ? cellValue.toLowerCase().includes(value)\n          : false;\n      });\n    }\n    return false;\n  }\n\n  // ---------- ✅ DATE / DATETIME PARSING + FORMATTING ----------\n\n  public parseAnyDate(input: any): Date | null {\n    if (input === null || input === undefined || input === '') return null;\n    if (input instanceof Date) return isNaN(input.getTime()) ? null : input;\n\n    if (typeof input === 'number') {\n      const d = new Date(input);\n      return isNaN(d.getTime()) ? null : d;\n    }\n\n    if (typeof input === 'string') {\n      const s = input.trim();\n      const isoTry = new Date(s);\n      if (!isNaN(isoTry.getTime())) return isoTry;\n\n      if (s.includes('/')) {\n        const parts = s.split(' ')[0].split('/');\n        if (parts.length === 3) {\n          const day = parseInt(parts[0], 10);\n          const month = parseInt(parts[1], 10) - 1;\n          const year = parseInt(parts[2], 10);\n          const d = new Date(year, month, day);\n          return isNaN(d.getTime()) ? null : d;\n        }\n      }\n    }\n\n    return null;\n  }\n\n  private formatWithPattern(d: Date | null, pattern: string): string {\n    if (!d) return '';\n    const pad2 = (n: number) => String(n).padStart(2, '0');\n    const map: Record<string, string> = {\n      dd: pad2(d.getDate()),\n      MM: pad2(d.getMonth() + 1),\n      yyyy: String(d.getFullYear()),\n      HH: pad2(d.getHours()),\n      mm: pad2(d.getMinutes()),\n      ss: pad2(d.getSeconds()),\n    };\n\n    return pattern\n      .replace(/yyyy/g, map.yyyy)\n      .replace(/dd/g, map.dd)\n      .replace(/MM/g, map.MM)\n      .replace(/HH/g, map.HH)\n      .replace(/mm/g, map.mm)\n      .replace(/ss/g, map.ss);\n  }\n\n  formatDate(date: any): string {\n    const d = this.parseAnyDate(date);\n    return this.formatWithPattern(d, 'dd/MM/yyyy');\n  }\n\n  formatDateWithColumn(d: Date | null, col: TableColumn): string {\n    const fmt = col.dateFormat?.trim() || 'dd/MM/yyyy';\n    return this.formatWithPattern(d, fmt);\n  }\n\n  formatDateTimeWithColumn(d: Date | null, col: TableColumn): string {\n    const fmt = col.dateTimeFormat?.trim() || 'dd/MM/yyyy HH:mm:ss';\n    return this.formatWithPattern(d, fmt);\n  }\n\n  // ---------- MISC ----------\n\n  private calculateColumnWidth(col: TableColumn): string {\n    const calculatedWidth = calculateTextWidth(col, col.title);\n    const totalWidth = calculatedWidth + this.iconWidth + 20;\n    return `${totalWidth}px`;\n  }\n\n  getHeaderWidth(col: TableColumn): string {\n    return col.width ? col.width : this.calculateColumnWidth(col);\n  }\n\n  clear(table: Table) {\n    table.clear();\n    this.searchValue = '';\n    this.latestFilterValues = {};\n\n    Object.keys(this.filters).forEach((key) => {\n      if (this.filters[key]) this.filters[key].value = [];\n    });\n\n    // ✅ server-side: parent must reload\n    if (this.isLazy) {\n      this.onFilterColumn.emit({ cleared: true });\n      return;\n    }\n\n    // ✅ client-side: reset paginator totals\n    if (this.dt) {\n      this.dt.value = [...(this.data ?? [])];\n      this.totalRecords = (this.dt.value ?? []).length;\n      this.dt.first = 0;\n    }\n  }\n\n  private initializePagination(): void {\n    if (this.isPaginated) {\n      if (!this.rowsPerPage || this.rowsPerPage.length === 0) {\n        this.rowsPerPage = [10, 20, 30];\n      }\n      this.rows = this.rowsPerPage[0];\n    }\n  }\n\n  getCurrencySymbol(column: TableColumn): string | undefined {\n    return column.type === TableTypeEnum.AMOUNT &&\n      column.currency &&\n      this.isValidCurrencyCode(column.currency)\n      ? column.currency\n      : undefined;\n  }\n\n  private isValidCurrencyCode(currencyCode: string): boolean {\n    return this.validCurrencyCodes.includes(currencyCode);\n  }\n\n  exportExcel() {\n    this.exportExcelEvent.emit();\n  }\n\n  exportPdf() {\n    this.exportPdfEvent.emit();\n  }\n\n  getImageStyle(style: TitleStyle | ImageStyle | undefined): {\n    [key: string]: string;\n  } {\n    if (!style) return {};\n    const imageStyle: { [key: string]: string } = {\n      width: (style as ImageStyle).width || 'auto',\n      height: (style as ImageStyle).height || 'auto',\n    };\n    if ((style as ImageStyle).margin)\n      imageStyle.margin = (style as ImageStyle).margin!;\n    if ((style as ImageStyle).marginLeft)\n      imageStyle.marginLeft = (style as ImageStyle).marginLeft!;\n    if ((style as ImageStyle).marginRight)\n      imageStyle.marginRight = (style as ImageStyle).marginRight!;\n    if ((style as ImageStyle).marginTop)\n      imageStyle.marginTop = (style as ImageStyle).marginTop!;\n    if ((style as ImageStyle).marginBottom)\n      imageStyle.marginBottom = (style as ImageStyle).marginBottom!;\n    return imageStyle;\n  }\n\n  getTitleStyle(style: TitleStyle | ImageStyle | undefined): {\n    [key: string]: string;\n  } {\n    if (!style) return {};\n    return {\n      color: style.color || 'inherit',\n      fontSize: (style as TitleStyle).fontSize || 'inherit',\n      textAlign: (style as TitleStyle).position || 'left',\n    };\n  }\n\n  formatNumber(\n    value: number,\n    decimalPlaces?: number,\n    thousandSeparator: 'comma' | 'space' = 'comma',\n    decimalSeparator: 'comma' | 'dot' = 'dot',\n  ): string {\n    if (value === null || value === undefined || isNaN(value)) return '';\n\n    let formattedNumber =\n      decimalPlaces !== undefined\n        ? value.toFixed(decimalPlaces)\n        : value.toString();\n\n    if (decimalSeparator === 'comma') {\n      formattedNumber = formattedNumber.replace('.', ',');\n    }\n\n    if (thousandSeparator && Math.abs(value) >= 1000) {\n      const parts = formattedNumber.split(\n        decimalSeparator === 'comma' ? ',' : '.',\n      );\n      parts[0] = parts[0].replace(\n        /\\B(?=(\\d{3})+(?!\\d))/g,\n        thousandSeparator === 'comma' ? ',' : ' ',\n      );\n      formattedNumber = parts.join(decimalSeparator === 'comma' ? ',' : '.');\n    }\n\n    return formattedNumber;\n  }\n\n  isActionVisible(action: PTTableActionConfig, row: any): boolean {\n    const v = action?.visible;\n    if (v === undefined || v === null) return true; // default: visible\n    return typeof v === 'function' ? !!v(row) : !!v;\n  }\n\n  isActionDisabled(action: PTTableActionConfig, row: any): boolean {\n    const d = action.disabled;\n    if (d === undefined || d === null) return false; // default: enabled\n    return typeof d === 'function' ? !!d(row) : !!d;\n  }\n}\n","<div class=\"pt-advanced-prime-table table-container\">\n  <p-table\n    #dt\n    [value]=\"data\"\n    [loading]=\"loading\"\n    [rows]=\"rows\"\n    [paginator]=\"isPaginated\"\n    [globalFilterFields]=\"globalFilterFields\"\n    [rowsPerPageOptions]=\"rowsPerPage\"\n    [totalRecords]=\"totalRecords\"\n    [lazy]=\"isLazy\"\n    dataKey=\"id\"\n    styleClass=\"p-datatable-gridlines p-datatable-striped\"\n    [scrollable]=\"true\"\n    [scrollHeight]=\"maxHeight !== null ? maxHeight : undefined\"\n    (onFilter)=\"filterColumn($event)\"\n    (onPage)=\"changePage($event)\"\n    (onSort)=\"sortColumn($event)\"\n  >\n    <!-- COLGROUP: sync header/body widths -->\n    <ng-template pTemplate=\"colgroup\" let-columns>\n      <colgroup>\n        <col\n          *ngFor=\"let col of columns\"\n          [style.width]=\"col.width || getHeaderWidth(col)\"\n        />\n      </colgroup>\n    </ng-template>\n\n    <!-- CAPTION -->\n    <ng-template pTemplate=\"caption\">\n      <div class=\"flex\">\n        <div>\n          <h3>Total: {{ totalRecords }}</h3>\n        </div>\n\n        <div>\n          <!-- Clear filters -->\n          <button\n            *ngIf=\"hasSearchFilter\"\n            pButton\n            icon=\"pi pi-filter-slash\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"clear(dt)\"\n            title=\"Clear filters\"\n          ></button>\n\n          <!-- Export to Excel Button -->\n          <button\n            *ngIf=\"hasExportExcel\"\n            pButton\n            icon=\"pi pi-file-excel\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"exportExcel()\"\n            title=\"Export to Excel\"\n          ></button>\n\n          <!-- Export to PDF Button -->\n          <button\n            *ngIf=\"hasExportPDF\"\n            pButton\n            icon=\"pi pi-file-pdf\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"exportPdf()\"\n            title=\"Export to PDF\"\n          ></button>\n        </div>\n\n        <div class=\"ml-auto\" *ngIf=\"hasSearchFilter\">\n          <p-iconField iconPosition=\"left\" class=\"ml-auto\">\n            <p-inputIcon>\n              <i class=\"pi pi-search\"></i>\n            </p-inputIcon>\n            <input\n              pInputText\n              type=\"text\"\n              [(ngModel)]=\"searchValue\"\n              (input)=\"filterGlobal($event)\"\n              placeholder=\"Search keyword\"\n            />\n          </p-iconField>\n        </div>\n      </div>\n    </ng-template>\n\n    <!-- HEADER -->\n    <ng-template pTemplate=\"header\">\n      <tr class=\"sticky-header\">\n        <ng-container *ngFor=\"let col of columns\">\n          <th\n            *ngIf=\"!col.children; else groupHeader\"\n            pSortableColumn=\"{{ col.code }}\"\n            [style.width]=\"col.width || getHeaderWidth(col)\"\n            [style.padding]=\"'0px'\"\n            [ngClass]=\"[\n              getHeaderAlignClass(col),\n              col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n            ]\"\n            colspan=\"1\"\n          >\n            <!-- SORTABLE HEADER -->\n            <ng-container\n              *ngIf=\"isSortable && col.isSortable !== false; else noSortHeader\"\n            >\n              <div\n                class=\"header-container d-flex align-items-center justify-content-between\"\n                [style.width]=\"col.width || getHeaderWidth(col)\"\n                [style.padding]=\"'0px'\"\n                [style.margin]=\"'10px'\"\n              >\n                <span [ngClass]=\"getHeaderTitleClass(col)\">\n                  {{ col.title }}\n                </span>\n                <div\n                  class=\"icons d-flex align-items-center\"\n                  [style.width]=\"'77px'\"\n                >\n                  <p-sortIcon field=\"{{ col.code }}\" />\n\n                  <!-- COLUMN FILTERS -->\n                  <ng-container\n                    *ngIf=\"hasColumnFilter && col.isFilter !== false\"\n                  >\n                    <!-- COMPOSED FILTER (uses built-in Apply/Clear) -->\n                    <p-columnFilter\n                      *ngIf=\"col.type === TableTypeEnum.COMPOSED\"\n                      display=\"menu\"\n                      [field]=\"col.code\"\n                      type=\"text\"\n                      [showApplyButton]=\"true\"\n                      [showClearButton]=\"true\"\n                      (onClear)=\"onComposedColumnClear(col)\"\n                    >\n                      <ng-template\n                        pTemplate=\"filter\"\n                        let-filter=\"filterCallback\"\n                        let-filterModel=\"filterModel\"\n                      >\n                        <div *ngFor=\"let composedName of col.composedNames\">\n                          <ng-container\n                            *ngIf=\"\n                              getComposedFieldType(col, composedName) ===\n                              TableTypeEnum.STRING\n                            \"\n                          >\n                            <p-multiSelect\n                              [options]=\"filters[composedName]?.options\"\n                              [ngModel]=\"filters[composedName]?.value\"\n                              (ngModelChange)=\"\n                                onComposedFilterValueChange(\n                                  col,\n                                  composedName,\n                                  $event,\n                                  filterModel\n                                )\n                              \"\n                              [placeholder]=\"filters[composedName]?.placeholder\"\n                              display=\"chip\"\n                            >\n                              <ng-template let-item pTemplate=\"item\">\n                                <div class=\"custom-multiselect-item\">\n                                  <img\n                                    *ngIf=\"item.image\"\n                                    [src]=\"item.image\"\n                                    alt=\"icon\"\n                                    class=\"filter-image\"\n                                  />\n                                  <span>{{ item.label }}</span>\n                                </div>\n                              </ng-template>\n                            </p-multiSelect>\n                          </ng-container>\n                        </div>\n                      </ng-template>\n                    </p-columnFilter>\n\n                    <!-- OTHER TYPES -->\n                    <p-columnFilter\n                      *ngIf=\"col.type !== TableTypeEnum.COMPOSED\"\n                      display=\"menu\"\n                      [field]=\"col.code\"\n                      [type]=\"getColumnFilterType(col)\"\n                      [showApplyButton]=\"true\"\n                      [showClearButton]=\"true\"\n                      (onClear)=\"onFilterClear(col.code!)\"\n                    >\n                      <!-- NUMBER / AMOUNT: use text input so '.' works -->\n                      <ng-template\n                        pTemplate=\"filter\"\n                        *ngIf=\"\n                          col.type === TableTypeEnum.NUMBER ||\n                          col.type === TableTypeEnum.AMOUNT\n                        \"\n                        let-value\n                      >\n                        <input\n                          pInputText\n                          type=\"text\"\n                          inputmode=\"decimal\"\n                          [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n                          (ngModelChange)=\"\n                            onNumberFilterChange(col.code!, $event)\n                          \"\n                          placeholder=\"Enter a number\"\n                        />\n                      </ng-template>\n\n                      <!-- DATE -->\n                      <ng-template\n                        pTemplate=\"filter\"\n                        *ngIf=\"col.type === TableTypeEnum.DATE\"\n                        let-filterModel=\"filterModel\"\n                      >\n                        <p-calendar\n                          [ngModel]=\"\n                            latestFilterValues[col.code!] ?? filterModel?.value\n                          \"\n                          (ngModelChange)=\"\n                            onFilterValueChange(col.code!, filterModel, $event)\n                          \"\n                          dateFormat=\"dd/mm/yy\"\n                          placeholder=\"Choose a date\"\n                        ></p-calendar>\n                      </ng-template>\n\n                      <!-- MULTISELECT -->\n                      <ng-template\n                        pTemplate=\"filter\"\n                        *ngIf=\"\n                          col.type === TableTypeEnum.MULTISELECT &&\n                          col.filterOptions &&\n                          col.filterOptions.length > 0\n                        \"\n                        let-filterModel=\"filterModel\"\n                      >\n                        <p-multiSelect\n                          [options]=\"col.filterOptions\"\n                          [ngModel]=\"\n                            latestFilterValues[col.code!] ?? filterModel?.value\n                          \"\n                          (ngModelChange)=\"\n                            onFilterValueChange(col.code!, filterModel, $event)\n                          \"\n                          display=\"chip\"\n                          placeholder=\"Choose option\"\n                          class=\"custom-multiselect\"\n                        ></p-multiSelect>\n                      </ng-template>\n                    </p-columnFilter>\n                  </ng-container>\n                </div>\n              </div>\n            </ng-container>\n\n            <!-- NON-SORTABLE HEADER -->\n            <ng-template #noSortHeader>\n              <div\n                class=\"header-container d-flex align-items-center justify-content-between\"\n                [style.width]=\"col.width || getHeaderWidth(col)\"\n                [style.padding]=\"'0px'\"\n                [style.margin]=\"'10px'\"\n              >\n                <span [ngClass]=\"getHeaderTitleClass(col)\">\n                  {{ col.title }}\n                </span>\n\n                <ng-container *ngIf=\"hasColumnFilter && col.isFilter !== false\">\n                  <!-- AMOUNT behaves like NUMBER -->\n                  <p-columnFilter\n                    *ngIf=\"col.type === 'AMOUNT'\"\n                    display=\"menu\"\n                    [field]=\"col.code\"\n                    [type]=\"getColumnFilterType(col)\"\n                    [currency]=\"getCurrencySymbol(col)\"\n                    [showApplyButton]=\"true\"\n                    [showClearButton]=\"true\"\n                  >\n                    <ng-template\n                      pTemplate=\"filter\"\n                      let-filterModel=\"filterModel\"\n                    >\n                      <input\n                        pInputText\n                        type=\"text\"\n                        inputmode=\"decimal\"\n                        [ngModel]=\"\n                          latestFilterValues[col.code!] ?? filterModel?.value\n                        \"\n                        (ngModelChange)=\"\n                          onFilterValueChange(col.code!, filterModel, $event)\n                        \"\n                        placeholder=\"Enter an amount\"\n                      />\n                    </ng-template>\n                  </p-columnFilter>\n\n                  <p-columnFilter\n                    *ngIf=\"col.type !== 'AMOUNT'\"\n                    display=\"menu\"\n                    [field]=\"col.code\"\n                    [type]=\"getColumnFilterType(col)\"\n                    [showApplyButton]=\"true\"\n                    [showClearButton]=\"true\"\n                  >\n                    <!-- DATE -->\n                    <ng-template\n                      pTemplate=\"filter\"\n                      let-filterModel=\"filterModel\"\n                      *ngIf=\"getColumnFilterType(col) === 'date'\"\n                    >\n                      <p-calendar\n                        [ngModel]=\"\n                          latestFilterValues[col.code!] ?? filterModel?.value\n                        \"\n                        (ngModelChange)=\"\n                          onFilterValueChange(col.code!, filterModel, $event)\n                        \"\n                        dateFormat=\"dd/mm/yy\"\n                      ></p-calendar>\n                    </ng-template>\n\n                    <!-- MULTISELECT -->\n                    <ng-template\n                      pTemplate=\"filter\"\n                      let-filterModel=\"filterModel\"\n                      *ngIf=\"getColumnFilterType(col) === 'multiSelect'\"\n                    >\n                      <p-multiSelect\n                        [options]=\"col.filterOptions\"\n                        [ngModel]=\"\n                          latestFilterValues[col.code!] ?? filterModel?.value\n                        \"\n                        (ngModelChange)=\"\n                          onFilterValueChange(col.code!, filterModel, $event)\n                        \"\n                        display=\"chip\"\n                        placeholder=\"Select\"\n                        class=\"custom-multiselect\"\n                      ></p-multiSelect>\n                    </ng-template>\n                  </p-columnFilter>\n                </ng-container>\n              </div>\n            </ng-template>\n          </th>\n\n          <!-- GROUPED HEADER -->\n          <ng-template #groupHeader>\n            <th\n              [attr.colspan]=\"col.children?.length\"\n              [style.width]=\"col.width || getHeaderWidth(col)\"\n              [ngClass]=\"getHeaderAlignClass(col)\"\n            >\n              <span>{{ col.title }}</span>\n            </th>\n          </ng-template>\n        </ng-container>\n      </tr>\n\n      <!-- CHILD HEADERS -->\n      <tr *ngIf=\"hasGroupedColumns\">\n        <ng-container *ngFor=\"let col of columns\">\n          <ng-container *ngIf=\"col.children\">\n            <th\n              *ngFor=\"let child of col.children\"\n              [style.width]=\"child.width || getHeaderWidth(child)\"\n              [style.padding]=\"'0px'\"\n            ></th>\n          </ng-container>\n        </ng-container>\n      </tr>\n    </ng-template>\n\n    <!-- EMPTY MESSAGE -->\n    <ng-template pTemplate=\"emptymessage\">\n      <tr class=\"p-datatable-emptymessage\">\n        <!-- span all columns -->\n        <td class=\"empty-message-cell\" [attr.colspan]=\"columns.length || 1\">\n          <div class=\"empty-message-wrapper\">\n            <div class=\"empty-message\">\n              <i class=\"pi pi-info-circle\"></i>\n              <p>No records available to display.</p>\n            </div>\n          </div>\n        </td>\n      </tr>\n    </ng-template>\n\n    <!-- BODY -->\n    <ng-template\n      pTemplate=\"body\"\n      let-data\n      let-editing=\"editing\"\n      let-ri=\"rowIndex\"\n    >\n      <tr *ngIf=\"!loading\" [pEditableRow]=\"isEdit ? data : null\">\n        <ng-container *ngFor=\"let col of columns\">\n          <!-- SIMPLE COLUMNS (no children) -->\n          <ng-container *ngIf=\"!col.children; else childColumns\">\n            <!-- EDITABLE CELL -->\n            <td\n              *ngIf=\"\n                isEditable(col.code!) && col.type !== TableTypeEnum.ACTION;\n                else normalTD\n              \"\n              [style.width]=\"col.width || getHeaderWidth(col)\"\n              [ngClass]=\"getDataAlignClass(col)\"\n            >\n              <!-- Editable input for NUMBER/DATE/STRING/MULTISELECT -->\n              <ng-container *ngIf=\"isMultiSelect(col.code); else datePicker\">\n                <p-cellEditor>\n                  <ng-template pTemplate=\"input\">\n                    <p-multiSelect\n                      appendTo=\"body\"\n                      [ngModel]=\"data[col.code!]\"\n                      [style]=\"{ width: '100%' }\"\n                      (ngModelChange)=\"changeHandler(data.id, col.code, $event)\"\n                      [options]=\"optionValues\"\n                    ></p-multiSelect>\n                  </ng-template>\n                  <ng-template pTemplate=\"output\">\n                    <div class=\"multi-select-container\">\n                      <ng-container *ngFor=\"let rec of data[col.code!]\">\n                        <p-tag [value]=\"rec\"></p-tag>\n                      </ng-container>\n                    </div>\n                  </ng-template>\n                </p-cellEditor>\n              </ng-container>\n\n              <ng-template #datePicker>\n                <ng-container *ngIf=\"isDatePicker(col.code); else normalInput\">\n                  <p-cellEditor>\n                    <ng-template pTemplate=\"input\">\n                      <p-calendar\n                        [inputId]=\"data[col.code!]\"\n                        [ngModel]=\"data[col.code!]\"\n                        (ngModelChange)=\"\n                          changeHandler(data.id, col.code, $event)\n                        \"\n                        [dateFormat]=\"'dd/mm/yy'\"\n                      ></p-calendar>\n                    </ng-template>\n                    <ng-template pTemplate=\"output\">\n                      {{ data[col.code!] | customDate }}\n                    </ng-template>\n                  </p-cellEditor>\n                </ng-container>\n              </ng-template>\n\n              <ng-template #normalInput>\n                <p-cellEditor>\n                  <ng-template pTemplate=\"input\">\n                    <input\n                      pInputText\n                      type=\"text\"\n                      [ngModel]=\"data[col.code!]\"\n                      (change)=\"onChange($event, data.id, col.code)\"\n                    />\n                  </ng-template>\n                  <ng-template pTemplate=\"output\">\n                    <ng-container\n                      *ngIf=\"\n                        col.type === TableTypeEnum.AMOUNT;\n                        else normalOutput\n                      \"\n                    >\n                      {{\n                        data[col.code!]\n                          | customCurrency\n                            : getCurrencySymbol(col)\n                            : col.decimalPlaces\n                            : col.thousandSeparator\n                            : col.decimalSeparator\n                      }}\n                    </ng-container>\n                    <ng-template #normalOutput>\n                      {{ data[col.code!] }}\n                    </ng-template>\n                  </ng-template>\n                </p-cellEditor>\n              </ng-template>\n            </td>\n\n            <!-- NON-EDITABLE CELL (ALWAYS RENDERED) -->\n            <ng-template #normalTD>\n              <td\n                [style.width]=\"col.width || getHeaderWidth(col)\"\n                [ngClass]=\"[\n                  getDataAlignClass(col),\n                  col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n                ]\"\n              >\n                <!-- ACTION column: built-in edit/delete + custom actions -->\n                <ng-container\n                  *ngIf=\"col.type === TableTypeEnum.ACTION; else nonActionCell\"\n                >\n                  <div class=\"action-buttons-container\">\n                    <!-- built-in delete -->\n                    <button\n                      *ngIf=\"isDelete\"\n                      pButton\n                      pRipple\n                      type=\"button\"\n                      icon=\"pi pi-trash\"\n                      (click)=\"Delete(data)\"\n                      class=\"p-button-rounded p-button-text\"\n                    ></button>\n\n                    <!-- built-in inline edit -->\n                    <div *ngIf=\"isEdit\">\n                      <button\n                        pInitEditableRow\n                        *ngIf=\"!editing\"\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        icon=\"pi pi-pencil\"\n                        (click)=\"initEditableRow(data)\"\n                        class=\"p-button-rounded p-button-text\"\n                      ></button>\n                      <button\n                        *ngIf=\"editing\"\n                        pSaveEditableRow\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        icon=\"pi pi-check\"\n                        (click)=\"saveEditableRow(data)\"\n                        class=\"p-button-rounded p-button-text\"\n                      ></button>\n                      <button\n                        *ngIf=\"editing\"\n                        pCancelEditableRow\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        icon=\"pi pi-times\"\n                        (click)=\"cancelEditableRow(data)\"\n                        class=\"p-button-rounded p-button-text\"\n                      ></button>\n                    </div>\n\n                    <!-- custom actions -->\n                    <ng-container *ngFor=\"let act of customActions\">\n                      <button\n                        *ngIf=\"isActionVisible(act, data)\"\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        class=\"p-button-rounded p-button-text\"\n                        [icon]=\"act.icon || 'pi pi-ellipsis-h'\"\n                        [ngClass]=\"act.styleClass\"\n                        [disabled]=\"isActionDisabled(act, data)\"\n                        (click)=\"onCustomActionClick(act, data)\"\n                      ></button>\n                    </ng-container>\n                  </div>\n                </ng-container>\n\n                <!-- NON-ACTION cells -->\n                <ng-template #nonActionCell>\n                  <!-- COMPOSED -->\n                  <ng-container\n                    *ngIf=\"\n                      col.type === TableTypeEnum.COMPOSED;\n                      else nonComposed\n                    \"\n                  >\n                    <div class=\"composed-cell\">\n                      <ng-container\n                        *ngFor=\"\n                          let composedName of col.composedNames;\n                          let i = index\n                        \"\n                      >\n                        <!-- IMAGE -->\n                        <ng-container\n                          *ngIf=\"\n                            col.composedTypes &&\n                            col.composedTypes[i] === TableTypeEnum.IMAGE\n                          \"\n                        >\n                          <img\n                            [src]=\"data[col.code!]?.[composedName]\"\n                            alt=\"composed-img\"\n                            class=\"composed-image\"\n                            [ngStyle]=\"\n                              getImageStyle(col.composedStyles?.[composedName])\n                            \"\n                          />\n                        </ng-container>\n\n                        <!-- STRING -->\n                        <ng-container\n                          *ngIf=\"\n                            col.composedTypes &&\n                            col.composedTypes[i] === TableTypeEnum.STRING\n                          \"\n                        >\n                          <span\n                            class=\"composed-text\"\n                            [ngStyle]=\"\n                              getTitleStyle(col.composedStyles?.[composedName])\n                            \"\n                          >\n                            {{ data[col.code!]?.[composedName] }}\n                          </span>\n                        </ng-container>\n                      </ng-container>\n                    </div>\n                  </ng-container>\n\n                  <ng-template #nonComposed>\n                    <!-- AMOUNT-->\n                    <ng-container\n                      *ngIf=\"col.type === TableTypeEnum.AMOUNT; else nonAmount\"\n                    >\n                      {{\n                        data[col.code!]\n                          | customCurrency\n                            : getCurrencySymbol(col)\n                            : col.decimalPlaces\n                            : col.thousandSeparator\n                            : col.decimalSeparator\n                      }}\n                    </ng-container>\n\n                    <ng-template #nonAmount>\n                      <!-- NUMBER-->\n                      <ng-container\n                        *ngIf=\"\n                          col.type === TableTypeEnum.NUMBER;\n                          else nonNumber\n                        \"\n                      >\n                        {{\n                          formatNumber(\n                            data[col.code!],\n                            col.decimalPlaces,\n                            col.thousandSeparator,\n                            col.decimalSeparator\n                          )\n                        }}\n                      </ng-container>\n\n                      <ng-template #nonNumber>\n                        <!-- DATE -->\n                        <ng-container\n                          *ngIf=\"\n                            col.type === TableTypeEnum.DATE ||\n                              col.type === TableTypeEnum.DATETIME;\n                            else normalTypes\n                          \"\n                        >\n                          <ng-container *ngIf=\"col.type === TableTypeEnum.DATE\">\n                            {{\n                              formatDateWithColumn(\n                                parseAnyDate(data[col.code!]),\n                                col\n                              )\n                            }}\n                          </ng-container>\n\n                          <ng-container\n                            *ngIf=\"col.type === TableTypeEnum.DATETIME\"\n                          >\n                            {{\n                              formatDateTimeWithColumn(\n                                parseAnyDate(data[col.code!]),\n                                col\n                              )\n                            }}\n                          </ng-container>\n                        </ng-container>\n\n                        <ng-template #normalTypes>\n                          <!-- STRING, MULTISELECT-->\n                          <ng-container\n                            *ngIf=\"\n                              [\n                                TableTypeEnum.STRING,\n                                TableTypeEnum.MULTISELECT,\n                              ].includes(col.type!)\n                            \"\n                          >\n                            {{ data[col.code!] }}\n                          </ng-container>\n                        </ng-template>\n                      </ng-template>\n                    </ng-template>\n                  </ng-template>\n                </ng-template>\n              </td>\n            </ng-template>\n          </ng-container>\n\n          <!-- CHILD COLUMNS -->\n          <ng-template #childColumns>\n            <ng-container *ngFor=\"let child of col.children\">\n              <td\n                [style.width]=\"child.width || getHeaderWidth(child)\"\n                [ngClass]=\"getDataAlignClass(child)\"\n              >\n                <ng-container\n                  *ngIf=\"isEditable(child.code); else childNormalTD\"\n                >\n                  <p-cellEditor>\n                    <ng-template pTemplate=\"input\">\n                      <input\n                        pInputText\n                        type=\"text\"\n                        [ngModel]=\"child.code ? data[child.code] : null\"\n                        (change)=\"onChange($event, data.id, child.code)\"\n                      />\n                    </ng-template>\n                    <ng-template pTemplate=\"output\">\n                      {{ child.code ? data[child.code] : \"\" }}\n                    </ng-template>\n                  </p-cellEditor>\n                </ng-container>\n\n                <ng-template #childNormalTD>\n                  {{ child.code ? data[child.code] : \"\" }}\n                </ng-template>\n              </td>\n            </ng-container>\n          </ng-template>\n        </ng-container>\n      </tr>\n    </ng-template>\n  </p-table>\n</div>\n"]}
|
|
856
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"pt-advanced-prime-table.component.js","sourceRoot":"","sources":["../../../../../projects/ng-prime-tools/src/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.ts","../../../../../projects/ng-prime-tools/src/lib/pt-advanced-prime-table/pt-advanced-prime-table.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { Table } from 'primeng/table';\nimport {\n  AlignEnum,\n  ImageStyle,\n  TableColumn,\n  TableTypeEnum,\n  TitleStyle,\n} from '../../public-api';\nimport { calculateTextWidth } from '../utils/text.util';\n\nexport interface PTTableActionConfig {\n  code: string;\n  icon?: string;\n  styleClass?: string;\n  tooltip?: string;\n  action: any;\n  visible?: boolean | ((row: any) => boolean);\n  disabled?: boolean | ((row: any) => boolean);\n}\n\nexport interface PTTableLazyLoadEvent {\n  page: number;\n  rows: number;\n  first: number;\n  search?: string;\n  sortField?: string | null;\n  sortOrder?: 1 | -1 | null;\n  filters?: Record<string, any>;\n}\n\n@Component({\n  selector: 'pt-advanced-prime-table',\n  templateUrl: './pt-advanced-prime-table.component.html',\n  styleUrl: './pt-advanced-prime-table.component.css',\n})\nexport class PTAdvancedPrimeTableComponent implements OnInit, OnChanges {\n  @Input() data: any[] = [];\n  @Input() columns: TableColumn[] = [];\n  @Input() totalRecords = 0;\n  @Input() rowsPerPage: number[] = [10, 20, 30];\n  @Input() hasSearchFilter = false;\n  @Input() hasExportExcel = false;\n  @Input() hasExportPDF = false;\n  @Input() hasColumnFilter = false;\n\n  /** paginator UI */\n  @Input() isPaginated = true;\n\n  /** server-side mode */\n  @Input() isLazy = false;\n\n  @Input() actions: PTTableActionConfig[] = [];\n  @Input() isSortable = false;\n  @Input() loading = false;\n  @Input() maxHeight: string | null = null;\n\n  /**\n   * New unified lazy event.\n   * Use this in parent components for server-side pagination/filter/sort/search.\n   */\n  @Output() lazyLoad = new EventEmitter<PTTableLazyLoadEvent>();\n\n  /** Backward compatibility */\n  @Output() search = new EventEmitter<string>();\n  @Output() exportExcelEvent = new EventEmitter<void>();\n  @Output() exportPdfEvent = new EventEmitter<void>();\n  @Output() onPageChange = new EventEmitter<{ page: number; rows: number }>();\n  @Output() onSortColumn = new EventEmitter<any>();\n  @Output() onFilterColumn = new EventEmitter<any>();\n\n  @ViewChild('dt', { static: false }) dt!: Table;\n\n  public TableTypeEnum = TableTypeEnum;\n  public AlignEnum = AlignEnum;\n\n  searchValue = '';\n\n  public filters: { [key: string]: any } = {};\n  latestFilterValues: { [field: string]: any } = {};\n  private clearedFields = new Set<string>();\n\n  private validCurrencyCodes = ['USD', 'EUR', 'MAD'];\n  iconWidth = 77;\n\n  rows = 0;\n  first = 0;\n  currentPage = 0;\n  currentSortField: string | null = null;\n  currentSortOrder: 1 | -1 | null = null;\n\n  hasGroupedColumns = false;\n\n  isDelete = false;\n  isEdit = false;\n\n  Delete: (value: any) => void = () => {};\n  initEditableRow: (data: any) => void = () => {};\n  saveEditableRow: (data: any) => void = () => {};\n  cancelEditableRow: (item: any) => void = () => {};\n\n  customActions: PTTableActionConfig[] = [];\n\n  dataMap = new Map<any, any>();\n  map = new Map<any, Map<any, any>>();\n  optionEntries = new Map<string, any[]>();\n  optionValues: any[] = [];\n  globalFilterFields: string[] = [];\n\n  ngOnInit(): void {\n    this.hasGroupedColumns = this.columns.some(\n      (col) => col.children && col.children.length > 0,\n    );\n\n    this.globalFilterFields = this.columns\n      .filter((col) => col.code !== undefined && col.isFilter !== false)\n      .map((col) => col.code as string);\n\n    this.initializePagination();\n    this.initializeActions();\n\n    this.columns.forEach((col) => {\n      if (col.type === TableTypeEnum.ACTION) {\n        col.isEditable = false;\n        col.isFilter = false;\n        col.isSortable = false;\n      }\n\n      if (col.type === TableTypeEnum.COMPOSED) {\n        this.initializeComposedFilters(col);\n      }\n\n      if (col.isSortable === undefined) col.isSortable = true;\n      if (col.isEditable === undefined) col.isEditable = true;\n\n      if (col.isFilter !== false && col.code !== undefined) {\n        if (!this.globalFilterFields.includes(col.code)) {\n          this.globalFilterFields.push(col.code);\n        }\n      }\n\n      if (!col.width) col.width = this.calculateColumnWidth(col);\n    });\n\n    if (!this.isLazy) {\n      this.totalRecords = this.data?.length ?? 0;\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['data'] && !this.isLazy) {\n      this.totalRecords = this.data?.length ?? 0;\n\n      if (this.dt) this.dt.first = 0;\n    }\n  }\n\n  // ---------- LAZY PAYLOAD ----------\n\n  private emitLazyLoad(): void {\n    const payload: PTTableLazyLoadEvent = {\n      page: this.currentPage,\n      rows: this.rows,\n      first: this.first,\n      search: this.searchValue?.trim() || undefined,\n      sortField: this.currentSortField,\n      sortOrder: this.currentSortOrder,\n      filters: { ...this.latestFilterValues },\n    };\n    console.log('emitLazyLoad payload =', payload);\n    this.lazyLoad.emit(payload);\n  }\n\n  private resetToFirstPage(): void {\n    this.currentPage = 0;\n    this.first = 0;\n\n    if (this.dt) {\n      this.dt.first = 0;\n    }\n  }\n\n  // ---------- HEADER + BODY ALIGNMENT HELPERS ----------\n\n  getHeaderTitleClass(col: TableColumn): string {\n    const align = col.headerAlign ?? AlignEnum.LEFT;\n    switch (align) {\n      case AlignEnum.CENTER:\n        return 'header-title-center';\n      case AlignEnum.RIGHT:\n        return 'header-title-right';\n      default:\n        return 'header-title-left';\n    }\n  }\n\n  getHeaderAlignClass(col: TableColumn): string {\n    const align = col.headerAlign ?? AlignEnum.LEFT;\n    switch (align) {\n      case AlignEnum.CENTER:\n        return 'header-align-center';\n      case AlignEnum.RIGHT:\n        return 'header-align-right';\n      default:\n        return 'header-align-left';\n    }\n  }\n\n  getDataAlignClass(col: TableColumn): string {\n    const effectiveAlign =\n      col.dataAlign ??\n      (col.type === TableTypeEnum.NUMBER || col.type === TableTypeEnum.AMOUNT\n        ? AlignEnum.RIGHT\n        : AlignEnum.LEFT);\n\n    switch (effectiveAlign) {\n      case AlignEnum.CENTER:\n        return 'cell-align-center';\n      case AlignEnum.RIGHT:\n        return 'cell-align-right';\n      default:\n        return 'cell-align-left';\n    }\n  }\n\n  // ---------- ACTIONS INITIALIZATION ----------\n\n  private initializeActions(): void {\n    this.isDelete = false;\n    this.isEdit = false;\n    this.Delete = () => {};\n    this.initEditableRow = () => {};\n    this.saveEditableRow = () => {};\n    this.cancelEditableRow = () => {};\n    this.customActions = [];\n\n    if (!this.actions || this.actions.length === 0) return;\n\n    this.actions.forEach((action) => {\n      switch (action.code) {\n        case 'delete':\n          this.isDelete = true;\n          this.Delete = (row: any) => action.action(row);\n          break;\n        case 'edit':\n          this.initializeEditActions(action);\n          break;\n        default:\n          this.customActions.push(action);\n          break;\n      }\n    });\n  }\n\n  private initializeEditActions(action: PTTableActionConfig): void {\n    this.isEdit = true;\n    this.initEditableRow = (data: any) => action.action.init(data);\n    this.saveEditableRow = (data: any) => {\n      const record = this.map.get(data.id);\n      action.action.save(data, record);\n      this.dataMap.clear();\n    };\n    this.cancelEditableRow = (item: any) => console.log(item);\n  }\n\n  onCustomActionClick(action: PTTableActionConfig, row: any): void {\n    if (!this.isActionVisible(action, row)) return;\n    if (this.isActionDisabled(action, row)) return;\n\n    if (action && typeof action.action === 'function') {\n      setTimeout(() => action.action(row), 0);\n    }\n  }\n\n  // ---------- FILTERING / SORTING / PAGINATION ----------\n\n  private initializeComposedFilters(col: TableColumn): void {\n    col.composedNames?.forEach((composedName) => {\n      const code = (col.code as string) || '';\n      const composedCode = code + '.' + composedName;\n\n      if (!this.globalFilterFields.includes(composedCode)) {\n        this.globalFilterFields.push(composedCode);\n      }\n\n      this.filters[composedName] = {\n        options: col.filterOptions,\n        value: [],\n        label: 'Filter by ' + composedName,\n        placeholder: 'Select option',\n      };\n    });\n  }\n\n  getComposedFieldType(\n    col: TableColumn,\n    composedName: string,\n  ): TableTypeEnum | undefined {\n    if (col.composedNames && col.composedTypes) {\n      const index = col.composedNames.indexOf(composedName);\n      if (index >= 0 && index < col.composedTypes.length) {\n        return col.composedTypes[index];\n      }\n    }\n    return undefined;\n  }\n\n  onComposedColumnClear(col: TableColumn): void {\n    if (!col.code || !col.composedNames) return;\n\n    col.composedNames.forEach((name) => {\n      const key = `${col.code}.${name}`;\n      delete this.latestFilterValues[key];\n\n      if (this.filters[name]) {\n        this.filters[name].value = [];\n      }\n\n      this.clearedFields.add(key);\n    });\n  }\n\n  onComposedFilterValueChange(\n    col: TableColumn,\n    composedName: string,\n    value: any[],\n    filterModel?: any,\n  ): void {\n    const key = `${col.code}.${composedName}`;\n\n    if (filterModel) {\n      filterModel.value = value;\n      if (\n        Array.isArray(filterModel.constraints) &&\n        filterModel.constraints.length > 0\n      ) {\n        filterModel.constraints[0].value = value;\n      }\n    }\n\n    if (!this.filters[composedName]) {\n      this.filters[composedName] = {\n        options: col.filterOptions,\n        value: [],\n        placeholder: 'Select option',\n      };\n    }\n    this.filters[composedName].value = value || [];\n\n    const isEmpty = !value || (Array.isArray(value) && value.length === 0);\n    if (isEmpty) delete this.latestFilterValues[key];\n    else this.latestFilterValues[key] = value;\n  }\n\n  onFilterClear(field: string | undefined): void {\n    if (!field) return;\n    delete this.latestFilterValues[field];\n    this.clearedFields.add(field);\n  }\n\n  onFilterValueChange(\n    field: string | undefined,\n    filterModel: any | null,\n    value: any,\n  ): void {\n    if (!field) return;\n\n    const isEmpty =\n      value === null ||\n      value === undefined ||\n      value === '' ||\n      (Array.isArray(value) && value.length === 0);\n\n    if (isEmpty) {\n      delete this.latestFilterValues[field];\n    } else {\n      this.latestFilterValues[field] = value;\n    }\n\n    if (filterModel) {\n      filterModel.value = value;\n\n      if (\n        Array.isArray(filterModel.constraints) &&\n        filterModel.constraints.length > 0\n      ) {\n        filterModel.constraints[0].value = value;\n      }\n    }\n  }\n\n  onNumberFilterChange(field: string | undefined, value: any): void {\n    if (!field) return;\n\n    const isEmpty =\n      value === null ||\n      value === undefined ||\n      value === '' ||\n      (Array.isArray(value) && value.length === 0);\n\n    if (isEmpty) delete this.latestFilterValues[field];\n    else this.latestFilterValues[field] = value;\n  }\n\n  private findColumnByField(field: string): TableColumn | undefined {\n    return this.columns.find(\n      (c) => c.code === field || (c.code && field.startsWith(c.code + '.')),\n    );\n  }\n\n  filterColumn(event: any): void {\n    console.log('filterColumn event =', event);\n    console.log('filterColumn event.filters =', event?.filters);\n    console.log(\n      'filterColumn latestFilterValues BEFORE =',\n      this.latestFilterValues,\n    );\n\n    const filters = event?.filters;\n\n    if (!filters) {\n      if (this.isLazy) {\n        this.resetToFirstPage();\n        this.onFilterColumn.emit(event);\n        this.emitLazyLoad();\n      }\n      return;\n    }\n\n    const isNullish = (v: any) => v === null || v === undefined || v === '';\n    const nextFilterValues: { [field: string]: any } = {\n      ...this.latestFilterValues,\n    };\n\n    Object.keys(filters).forEach((field) => {\n      const meta = filters[field];\n      const normalizeMeta = (m: any) =>\n        Array.isArray(m) && m.length > 0 ? m[0] : m;\n\n      const m = normalizeMeta(meta);\n      const col = this.findColumnByField(field);\n      const wasCleared = this.clearedFields.has(field);\n\n      if (!m && !col) return;\n\n      // ---- CLEARED FIELD ----\n      // Must be handled BEFORE COMPOSED logic\n      if (wasCleared) {\n        delete nextFilterValues[field];\n\n        if (\n          col?.type === TableTypeEnum.COMPOSED &&\n          col.code &&\n          col.composedNames\n        ) {\n          col.composedNames.forEach((name) => {\n            delete nextFilterValues[`${col.code}.${name}`];\n          });\n        }\n\n        if (m) {\n          m.value = null;\n\n          if (Array.isArray(m.constraints) && m.constraints.length > 0) {\n            m.constraints.forEach((c: any) => (c.value = null));\n          }\n        }\n\n        this.clearedFields.delete(field);\n        return;\n      }\n\n      // ---- COMPOSED FILTERS ----\n      if (col && col.type === TableTypeEnum.COMPOSED) {\n        const composedValues: any = {};\n\n        col.composedNames?.forEach((name) => {\n          const key = `${col.code}.${name}`;\n\n          if (this.clearedFields.has(key)) {\n            delete nextFilterValues[key];\n            this.clearedFields.delete(key);\n            return;\n          }\n\n          const val = nextFilterValues[key];\n          const empty =\n            isNullish(val) || (Array.isArray(val) && val.length === 0);\n\n          if (!empty) {\n            composedValues[name] = val;\n          }\n        });\n\n        if (Object.keys(composedValues).length === 0) {\n          if (m) {\n            m.value = null;\n            if (Array.isArray(m.constraints)) {\n              m.constraints.forEach((c: any) => (c.value = null));\n            }\n          }\n        } else {\n          m.value = composedValues;\n        }\n\n        return;\n      }\n\n      if (!m) return;\n\n      let value: any = m.value;\n\n      if (Array.isArray(m.constraints) && m.constraints.length > 0) {\n        const cVal = m.constraints[0].value;\n        if (!isNullish(cVal)) {\n          value = cVal;\n        }\n      }\n\n      const cached = nextFilterValues[field];\n      const hasCached =\n        !isNullish(cached) && (!Array.isArray(cached) || cached.length > 0);\n\n      if (\n        (isNullish(value) || (Array.isArray(value) && value.length === 0)) &&\n        hasCached\n      ) {\n        value = cached;\n      }\n\n      const isEmpty =\n        isNullish(value) || (Array.isArray(value) && value.length === 0);\n\n      if (isEmpty) {\n        delete nextFilterValues[field];\n        m.value = null;\n\n        if (Array.isArray(m.constraints) && m.constraints.length > 0) {\n          m.constraints[0].value = null;\n        }\n\n        return;\n      }\n\n      let emitValue: any = value;\n\n      if (\n        col &&\n        (col.type === TableTypeEnum.DATE || col.type === TableTypeEnum.DATETIME)\n      ) {\n        const d = this.parseAnyDate(value);\n        emitValue =\n          col.type === TableTypeEnum.DATE\n            ? this.formatDateWithColumn(d, col)\n            : this.formatDateTimeWithColumn(d, col);\n      }\n\n      nextFilterValues[field] = emitValue;\n      m.value = emitValue;\n\n      if (Array.isArray(m.constraints) && m.constraints.length > 0) {\n        m.constraints[0].value = emitValue;\n      }\n    });\n\n    console.log('filterColumn nextFilterValues =', nextFilterValues);\n    this.latestFilterValues = nextFilterValues;\n    console.log(\n      'filterColumn latestFilterValues AFTER =',\n      this.latestFilterValues,\n    );\n\n    if (this.isLazy) {\n      this.resetToFirstPage();\n      this.onFilterColumn.emit(event);\n      this.emitLazyLoad();\n    } else {\n      if (this.dt) {\n        const current = (this.dt.filteredValue ?? this.dt.value ?? []) as any[];\n        this.totalRecords = current?.length ?? 0;\n        this.dt.first = 0;\n      }\n    }\n  }\n\n  changePage(event: any): void {\n    const page = event.page ?? Math.floor((event.first || 0) / event.rows);\n    const rows = event.rows;\n    const first = event.first ?? page * rows;\n\n    this.rows = rows;\n    this.currentPage = page;\n    this.first = first;\n\n    if (this.isLazy) {\n      this.onPageChange.emit({ page, rows });\n      this.emitLazyLoad();\n    }\n  }\n\n  sortColumn(event: any): void {\n    if (!this.isLazy) return;\n\n    let field = event.field;\n    const col = this.columns.find((c) => c.code === field);\n\n    if (col && col.type === TableTypeEnum.COMPOSED) {\n      let textProp: string | undefined;\n\n      if (col.composedNames && col.composedTypes) {\n        const idx = col.composedTypes.findIndex(\n          (t) => t === TableTypeEnum.STRING,\n        );\n        if (idx >= 0 && idx < col.composedNames.length) {\n          textProp = col.composedNames[idx];\n        }\n      }\n\n      if (!textProp && col.composedNames?.length) {\n        textProp = col.composedNames[0];\n      }\n\n      if (textProp) field = `${field}.${textProp}`;\n    }\n\n    this.currentSortField = field ?? null;\n    this.currentSortOrder = (event.order as 1 | -1) ?? null;\n\n    this.resetToFirstPage();\n    this.onSortColumn.emit({ ...event, field });\n    this.emitLazyLoad();\n  }\n\n  // ---------- EDIT HELPERS ----------\n\n  private parseDate_ddMMyyyy(dateString: string): Date | null {\n    const parts = dateString.split('/');\n    if (parts.length === 3) {\n      const day = parseInt(parts[0], 10);\n      const month = parseInt(parts[1], 10) - 1;\n      const year = parseInt(parts[2], 10);\n      const date = new Date(year, month, day);\n      return isNaN(date.getTime()) ? null : date;\n    }\n    return null;\n  }\n\n  onChange(event: Event, id: number, key: any) {\n    const target = event.target as HTMLInputElement;\n    this.changeHandler(id, key, target.value);\n  }\n\n  changeHandler(id: number, key: any, value: any) {\n    const column = this.columns.find((item) => item.code === key);\n\n    if (!this.map.get(id)) {\n      if (column?.type === TableTypeEnum.DATE) {\n        this.dataMap.set(key, this.parseDate_ddMMyyyy(value));\n      } else {\n        this.dataMap.set(key, value);\n      }\n      this.map.set(id, new Map(this.dataMap));\n    } else {\n      const mapItem = this.map.get(id) as Map<any, any>;\n      if (column?.type === TableTypeEnum.DATE) {\n        mapItem.set(key, this.parseDate_ddMMyyyy(value));\n      } else {\n        mapItem.set(key, value);\n      }\n    }\n  }\n\n  // ---------- TYPES ----------\n\n  getColumnFilterType(column: TableColumn): string {\n    switch (column.type) {\n      case TableTypeEnum.STRING:\n        return 'text';\n      case TableTypeEnum.AMOUNT:\n      case TableTypeEnum.NUMBER:\n        return 'numeric';\n      case TableTypeEnum.DATE:\n      case TableTypeEnum.DATETIME:\n        return 'date';\n      case TableTypeEnum.MULTISELECT:\n        return 'multiSelect';\n      case TableTypeEnum.BOOLEAN:\n        return 'boolean';\n      case TableTypeEnum.COMPOSED:\n        return 'composed';\n      default:\n        return 'text';\n    }\n  }\n\n  isEditable(key: string | undefined): boolean {\n    if (!key) return false;\n    const column = this.columns.find((item) => item.code === key);\n    return column?.isEditable !== false;\n  }\n\n  isMultiSelect(key: any): boolean {\n    const column = this.columns.find((item) => item.code === key);\n    if (\n      column?.type === TableTypeEnum.MULTISELECT &&\n      column.options &&\n      column.code !== undefined\n    ) {\n      this.optionEntries = new Map([\n        [column.code as string, Object.values(column.options)],\n      ]);\n      this.optionValues = this.optionEntries.get(key) || [];\n      return true;\n    }\n    return false;\n  }\n\n  isDatePicker(key: any): boolean {\n    const t = this.columns.find((item) => item.code === key)?.type;\n    return t === TableTypeEnum.DATE || t === TableTypeEnum.DATETIME;\n  }\n\n  isDateTimePicker(key: any): boolean {\n    const t = this.columns.find((item) => item.code === key)?.type;\n    return t === TableTypeEnum.DATETIME;\n  }\n\n  // ---------- SEARCH ----------\n\n  filterGlobal(event: Event): void {\n    const target = event.target as HTMLInputElement;\n    const value = (target.value || '').toLowerCase();\n\n    if (this.isLazy) {\n      this.searchValue = value;\n      this.resetToFirstPage();\n      this.search.emit(value);\n      this.emitLazyLoad();\n      return;\n    }\n\n    if (!value) {\n      this.dt.value = [...(this.data ?? [])];\n      this.totalRecords = (this.dt.value ?? []).length;\n      this.dt.first = 0;\n      return;\n    }\n\n    const filteredData = (this.data ?? []).filter((item) =>\n      (this.globalFilterFields ?? []).some((field) => {\n        const column = this.columns?.find((col) => col.code === field);\n        if (!column) return false;\n\n        const cell = item?.[field];\n\n        if (column.type === TableTypeEnum.DATE) {\n          return this.formatDateWithColumn(this.parseAnyDate(cell), column)\n            .toLowerCase()\n            .includes(value);\n        }\n\n        if (column.type === TableTypeEnum.DATETIME) {\n          return this.formatDateTimeWithColumn(this.parseAnyDate(cell), column)\n            .toLowerCase()\n            .includes(value);\n        }\n\n        if (\n          column.type === TableTypeEnum.AMOUNT ||\n          column.type === TableTypeEnum.NUMBER\n        ) {\n          return String(cell ?? '')\n            .toLowerCase()\n            .includes(value);\n        }\n\n        if (column.type === TableTypeEnum.COMPOSED) {\n          return this.filterComposedColumn(cell, value);\n        }\n\n        return String(cell ?? '')\n          .toLowerCase()\n          .includes(value);\n      }),\n    );\n\n    this.dt.value = filteredData;\n    this.totalRecords = filteredData.length;\n    this.dt.first = 0;\n  }\n\n  private filterComposedColumn(composedData: any, value: string): boolean {\n    if (composedData) {\n      return Object.keys(composedData).some((key) => {\n        const cellValue = composedData[key];\n        return typeof cellValue === 'string'\n          ? cellValue.toLowerCase().includes(value)\n          : false;\n      });\n    }\n    return false;\n  }\n\n  // ---------- DATE / DATETIME ----------\n\n  public parseAnyDate(input: any): Date | null {\n    if (input === null || input === undefined || input === '') return null;\n    if (input instanceof Date) return isNaN(input.getTime()) ? null : input;\n\n    if (typeof input === 'number') {\n      const d = new Date(input);\n      return isNaN(d.getTime()) ? null : d;\n    }\n\n    if (typeof input === 'string') {\n      const s = input.trim();\n      const isoTry = new Date(s);\n      if (!isNaN(isoTry.getTime())) return isoTry;\n\n      if (s.includes('/')) {\n        const parts = s.split(' ')[0].split('/');\n        if (parts.length === 3) {\n          const day = parseInt(parts[0], 10);\n          const month = parseInt(parts[1], 10) - 1;\n          const year = parseInt(parts[2], 10);\n          const d = new Date(year, month, day);\n          return isNaN(d.getTime()) ? null : d;\n        }\n      }\n    }\n\n    return null;\n  }\n\n  private formatWithPattern(d: Date | null, pattern: string): string {\n    if (!d) return '';\n    const pad2 = (n: number) => String(n).padStart(2, '0');\n    const map: Record<string, string> = {\n      dd: pad2(d.getDate()),\n      MM: pad2(d.getMonth() + 1),\n      yyyy: String(d.getFullYear()),\n      HH: pad2(d.getHours()),\n      mm: pad2(d.getMinutes()),\n      ss: pad2(d.getSeconds()),\n    };\n\n    return pattern\n      .replace(/yyyy/g, map.yyyy)\n      .replace(/dd/g, map.dd)\n      .replace(/MM/g, map.MM)\n      .replace(/HH/g, map.HH)\n      .replace(/mm/g, map.mm)\n      .replace(/ss/g, map.ss);\n  }\n\n  formatDate(date: any): string {\n    const d = this.parseAnyDate(date);\n    return this.formatWithPattern(d, 'dd/MM/yyyy');\n  }\n\n  formatDateWithColumn(d: Date | null, col: TableColumn): string {\n    const fmt = col.dateFormat?.trim() || 'dd/MM/yyyy';\n    return this.formatWithPattern(d, fmt);\n  }\n\n  formatDateTimeWithColumn(d: Date | null, col: TableColumn): string {\n    const fmt = col.dateTimeFormat?.trim() || 'dd/MM/yyyy HH:mm:ss';\n    return this.formatWithPattern(d, fmt);\n  }\n\n  // ---------- MISC ----------\n\n  private calculateColumnWidth(col: TableColumn): string {\n    const calculatedWidth = calculateTextWidth(col, col.title);\n    const totalWidth = calculatedWidth + this.iconWidth + 20;\n    return `${totalWidth}px`;\n  }\n\n  getHeaderWidth(col: TableColumn): string {\n    return col.width ? col.width : this.calculateColumnWidth(col);\n  }\n\n  clear(table: Table) {\n    table.clear();\n    this.searchValue = '';\n    this.latestFilterValues = {};\n    this.clearedFields.clear();\n    this.currentSortField = null;\n    this.currentSortOrder = null;\n\n    Object.keys(this.filters).forEach((key) => {\n      if (this.filters[key]) this.filters[key].value = [];\n    });\n\n    if (this.isLazy) {\n      this.resetToFirstPage();\n      this.onFilterColumn.emit({ cleared: true });\n      this.emitLazyLoad();\n      return;\n    }\n\n    if (this.dt) {\n      this.dt.value = [...(this.data ?? [])];\n      this.totalRecords = (this.dt.value ?? []).length;\n      this.dt.first = 0;\n    }\n  }\n\n  private initializePagination(): void {\n    if (this.isPaginated) {\n      if (!this.rowsPerPage || this.rowsPerPage.length === 0) {\n        this.rowsPerPage = [10, 20, 30];\n      }\n      this.rows = this.rowsPerPage[0];\n      this.currentPage = 0;\n      this.first = 0;\n    }\n  }\n\n  getCurrencySymbol(column: TableColumn): string | undefined {\n    return column.type === TableTypeEnum.AMOUNT &&\n      column.currency &&\n      this.isValidCurrencyCode(column.currency)\n      ? column.currency\n      : undefined;\n  }\n\n  private isValidCurrencyCode(currencyCode: string): boolean {\n    return this.validCurrencyCodes.includes(currencyCode);\n  }\n\n  exportExcel() {\n    this.exportExcelEvent.emit();\n  }\n\n  exportPdf() {\n    this.exportPdfEvent.emit();\n  }\n\n  getImageStyle(style: TitleStyle | ImageStyle | undefined): {\n    [key: string]: string;\n  } {\n    if (!style) return {};\n    const imageStyle: { [key: string]: string } = {\n      width: (style as ImageStyle).width || 'auto',\n      height: (style as ImageStyle).height || 'auto',\n    };\n    if ((style as ImageStyle).margin)\n      imageStyle.margin = (style as ImageStyle).margin!;\n    if ((style as ImageStyle).marginLeft)\n      imageStyle.marginLeft = (style as ImageStyle).marginLeft!;\n    if ((style as ImageStyle).marginRight)\n      imageStyle.marginRight = (style as ImageStyle).marginRight!;\n    if ((style as ImageStyle).marginTop)\n      imageStyle.marginTop = (style as ImageStyle).marginTop!;\n    if ((style as ImageStyle).marginBottom)\n      imageStyle.marginBottom = (style as ImageStyle).marginBottom!;\n    return imageStyle;\n  }\n\n  getTitleStyle(style: TitleStyle | ImageStyle | undefined): {\n    [key: string]: string;\n  } {\n    if (!style) return {};\n    return {\n      color: style.color || 'inherit',\n      fontSize: (style as TitleStyle).fontSize || 'inherit',\n      textAlign: (style as TitleStyle).position || 'left',\n    };\n  }\n\n  formatNumber(\n    value: number,\n    decimalPlaces?: number,\n    thousandSeparator: 'comma' | 'space' = 'comma',\n    decimalSeparator: 'comma' | 'dot' = 'dot',\n  ): string {\n    if (value === null || value === undefined || isNaN(value)) return '';\n\n    let formattedNumber =\n      decimalPlaces !== undefined\n        ? value.toFixed(decimalPlaces)\n        : value.toString();\n\n    if (decimalSeparator === 'comma') {\n      formattedNumber = formattedNumber.replace('.', ',');\n    }\n\n    if (thousandSeparator && Math.abs(value) >= 1000) {\n      const parts = formattedNumber.split(\n        decimalSeparator === 'comma' ? ',' : '.',\n      );\n      parts[0] = parts[0].replace(\n        /\\B(?=(\\d{3})+(?!\\d))/g,\n        thousandSeparator === 'comma' ? ',' : ' ',\n      );\n      formattedNumber = parts.join(decimalSeparator === 'comma' ? ',' : '.');\n    }\n\n    return formattedNumber;\n  }\n\n  isActionVisible(action: PTTableActionConfig, row: any): boolean {\n    const v = action?.visible;\n    if (v === undefined || v === null) return true;\n    return typeof v === 'function' ? !!v(row) : !!v;\n  }\n\n  isActionDisabled(action: PTTableActionConfig, row: any): boolean {\n    const d = action.disabled;\n    if (d === undefined || d === null) return false;\n    return typeof d === 'function' ? !!d(row) : !!d;\n  }\n}\n","<div class=\"pt-advanced-prime-table table-container\">\n  <p-table\n    #dt\n    [value]=\"data\"\n    [loading]=\"loading\"\n    [rows]=\"rows\"\n    [paginator]=\"isPaginated\"\n    [globalFilterFields]=\"globalFilterFields\"\n    [rowsPerPageOptions]=\"rowsPerPage\"\n    [totalRecords]=\"totalRecords\"\n    [lazy]=\"isLazy\"\n    [filterDelay]=\"0\"\n    dataKey=\"id\"\n    styleClass=\"p-datatable-gridlines p-datatable-striped\"\n    [scrollable]=\"true\"\n    [scrollHeight]=\"maxHeight !== null ? maxHeight : undefined\"\n    (onPage)=\"changePage($event)\"\n    (onSort)=\"sortColumn($event)\"\n    (onFilter)=\"filterColumn($event)\"\n  >\n    <ng-template pTemplate=\"colgroup\" let-columns>\n      <colgroup>\n        <col\n          *ngFor=\"let col of columns\"\n          [style.width]=\"col.width || getHeaderWidth(col)\"\n        />\n      </colgroup>\n    </ng-template>\n\n    <ng-template pTemplate=\"caption\">\n      <div class=\"flex\">\n        <div>\n          <h3>Total: {{ totalRecords }}</h3>\n        </div>\n\n        <div>\n          <button\n            *ngIf=\"hasSearchFilter\"\n            pButton\n            icon=\"pi pi-filter-slash\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"clear(dt)\"\n            title=\"Clear filters\"\n          ></button>\n\n          <button\n            *ngIf=\"hasExportExcel\"\n            pButton\n            icon=\"pi pi-file-excel\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"exportExcel()\"\n            title=\"Export to Excel\"\n          ></button>\n\n          <button\n            *ngIf=\"hasExportPDF\"\n            pButton\n            icon=\"pi pi-file-pdf\"\n            class=\"p-button-rounded p-button-text\"\n            (click)=\"exportPdf()\"\n            title=\"Export to PDF\"\n          ></button>\n        </div>\n\n        <div class=\"ml-auto\" *ngIf=\"hasSearchFilter\">\n          <p-iconField iconPosition=\"left\" class=\"ml-auto\">\n            <p-inputIcon>\n              <i class=\"pi pi-search\"></i>\n            </p-inputIcon>\n            <input\n              pInputText\n              type=\"text\"\n              [(ngModel)]=\"searchValue\"\n              (input)=\"filterGlobal($event)\"\n              placeholder=\"Search keyword\"\n            />\n          </p-iconField>\n        </div>\n      </div>\n    </ng-template>\n\n    <ng-template pTemplate=\"header\">\n      <tr class=\"sticky-header\">\n        <ng-container *ngFor=\"let col of columns\">\n          <th\n            *ngIf=\"!col.children; else groupHeader\"\n            [pSortableColumn]=\"col.code\"\n            [style.width]=\"col.width || getHeaderWidth(col)\"\n            [style.padding]=\"'0px'\"\n            [ngClass]=\"[\n              getHeaderAlignClass(col),\n              col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n            ]\"\n            colspan=\"1\"\n          >\n            <ng-container\n              *ngIf=\"isSortable && col.isSortable !== false; else noSortHeader\"\n            >\n              <div\n                class=\"header-container d-flex align-items-center justify-content-between\"\n                [style.width]=\"col.width || getHeaderWidth(col)\"\n                [style.padding]=\"'0px'\"\n                [style.margin]=\"'10px'\"\n              >\n                <span [ngClass]=\"getHeaderTitleClass(col)\">\n                  {{ col.title }}\n                </span>\n\n                <div\n                  class=\"icons d-flex align-items-center\"\n                  [style.width]=\"'77px'\"\n                >\n                  <p-sortIcon [field]=\"col.code\" />\n\n                  <ng-container\n                    *ngIf=\"hasColumnFilter && col.isFilter !== false\"\n                  >\n                    <!-- COMPOSED FILTER -->\n                    <p-columnFilter\n                      *ngIf=\"col.type === TableTypeEnum.COMPOSED\"\n                      display=\"menu\"\n                      [field]=\"col.code\"\n                      type=\"text\"\n                      [showApplyButton]=\"true\"\n                      [showClearButton]=\"true\"\n                      [showMatchModes]=\"false\"\n                      [showOperator]=\"false\"\n                      (onClear)=\"onComposedColumnClear(col)\"\n                    >\n                      <ng-template\n                        pTemplate=\"filter\"\n                        let-filter=\"filterCallback\"\n                      >\n                        <div *ngFor=\"let composedName of col.composedNames\">\n                          <ng-container\n                            *ngIf=\"\n                              getComposedFieldType(col, composedName) ===\n                              TableTypeEnum.STRING\n                            \"\n                          >\n                            <p-multiSelect\n                              [options]=\"filters[composedName]?.options\"\n                              optionLabel=\"label\"\n                              optionValue=\"value\"\n                              [ngModel]=\"filters[composedName]?.value ?? []\"\n                              (ngModelChange)=\"\n                                onComposedFilterValueChange(\n                                  col,\n                                  composedName,\n                                  $event\n                                );\n                                filter($event ?? [])\n                              \"\n                              [placeholder]=\"filters[composedName]?.placeholder\"\n                              display=\"chip\"\n                            >\n                              <ng-template let-item pTemplate=\"item\">\n                                <div class=\"custom-multiselect-item\">\n                                  <img\n                                    *ngIf=\"item.image\"\n                                    [src]=\"item.image\"\n                                    alt=\"icon\"\n                                    class=\"filter-image\"\n                                  />\n                                  <span>{{ item.label }}</span>\n                                </div>\n                              </ng-template>\n                            </p-multiSelect>\n                          </ng-container>\n                        </div>\n                      </ng-template>\n                    </p-columnFilter>\n\n                    <!-- OTHER TYPES -->\n                    <p-columnFilter\n                      *ngIf=\"col.type !== TableTypeEnum.COMPOSED\"\n                      display=\"menu\"\n                      [field]=\"col.code\"\n                      [type]=\"getColumnFilterType(col)\"\n                      [showApplyButton]=\"true\"\n                      [showClearButton]=\"true\"\n                      [showMatchModes]=\"col.type !== TableTypeEnum.MULTISELECT\"\n                      [showOperator]=\"col.type !== TableTypeEnum.MULTISELECT\"\n                      [matchMode]=\"\n                        col.type === TableTypeEnum.MULTISELECT\n                          ? 'in'\n                          : undefined\n                      \"\n                      (onClear)=\"onFilterClear(col.code!)\"\n                    >\n                      <!-- NUMBER / AMOUNT -->\n                      <ng-template\n                        pTemplate=\"filter\"\n                        *ngIf=\"\n                          col.type === TableTypeEnum.NUMBER ||\n                          col.type === TableTypeEnum.AMOUNT\n                        \"\n                        let-value\n                      >\n                        <input\n                          pInputText\n                          type=\"text\"\n                          inputmode=\"decimal\"\n                          [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n                          (ngModelChange)=\"\n                            onNumberFilterChange(col.code!, $event)\n                          \"\n                          placeholder=\"Enter a number\"\n                        />\n                      </ng-template>\n\n                      <!-- DATE -->\n                      <ng-template\n                        pTemplate=\"filter\"\n                        *ngIf=\"col.type === TableTypeEnum.DATE\"\n                        let-filter=\"filterCallback\"\n                        let-value\n                      >\n                        <p-calendar\n                          [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n                          (ngModelChange)=\"\n                            onFilterValueChange(col.code!, null, $event);\n                            filter($event)\n                          \"\n                          dateFormat=\"dd/mm/yy\"\n                          placeholder=\"Choose a date\"\n                        ></p-calendar>\n                      </ng-template>\n\n                      <!-- MULTISELECT -->\n                      <ng-template\n                        pTemplate=\"filter\"\n                        *ngIf=\"\n                          col.type === TableTypeEnum.MULTISELECT &&\n                          col.filterOptions &&\n                          col.filterOptions.length > 0\n                        \"\n                        let-filter=\"filterCallback\"\n                        let-value\n                      >\n                        <p-multiSelect\n                          [options]=\"col.filterOptions\"\n                          optionLabel=\"label\"\n                          optionValue=\"value\"\n                          [ngModel]=\"\n                            latestFilterValues[col.code!] ?? value ?? []\n                          \"\n                          (ngModelChange)=\"\n                            onFilterValueChange(col.code!, null, $event);\n                            filter($event ?? [])\n                          \"\n                          display=\"chip\"\n                          placeholder=\"Choose option\"\n                          class=\"custom-multiselect\"\n                        ></p-multiSelect>\n                      </ng-template>\n                    </p-columnFilter>\n                  </ng-container>\n                </div>\n              </div>\n            </ng-container>\n\n            <ng-template #noSortHeader>\n              <div\n                class=\"header-container d-flex align-items-center justify-content-between\"\n                [style.width]=\"col.width || getHeaderWidth(col)\"\n                [style.padding]=\"'0px'\"\n                [style.margin]=\"'10px'\"\n              >\n                <span [ngClass]=\"getHeaderTitleClass(col)\">\n                  {{ col.title }}\n                </span>\n\n                <ng-container *ngIf=\"hasColumnFilter && col.isFilter !== false\">\n                  <p-columnFilter\n                    *ngIf=\"col.type === 'AMOUNT'\"\n                    display=\"menu\"\n                    [field]=\"col.code\"\n                    [type]=\"getColumnFilterType(col)\"\n                    [currency]=\"getCurrencySymbol(col)\"\n                    [showApplyButton]=\"true\"\n                    [showClearButton]=\"true\"\n                  >\n                    <ng-template pTemplate=\"filter\" let-value>\n                      <input\n                        pInputText\n                        type=\"text\"\n                        inputmode=\"decimal\"\n                        [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n                        (ngModelChange)=\"\n                          onFilterValueChange(col.code!, null, $event)\n                        \"\n                        placeholder=\"Enter an amount\"\n                      />\n                    </ng-template>\n                  </p-columnFilter>\n\n                  <p-columnFilter\n                    *ngIf=\"col.type !== 'AMOUNT'\"\n                    display=\"menu\"\n                    [field]=\"col.code\"\n                    [type]=\"getColumnFilterType(col)\"\n                    [showApplyButton]=\"true\"\n                    [showClearButton]=\"true\"\n                    [showMatchModes]=\"\n                      getColumnFilterType(col) !== 'multiSelect'\n                    \"\n                    [showOperator]=\"getColumnFilterType(col) !== 'multiSelect'\"\n                    [matchMode]=\"\n                      getColumnFilterType(col) === 'multiSelect'\n                        ? 'in'\n                        : undefined\n                    \"\n                    (onClear)=\"onFilterClear(col.code!)\"\n                  >\n                    <!-- DATE -->\n                    <ng-template\n                      pTemplate=\"filter\"\n                      let-filter=\"filterCallback\"\n                      let-value\n                      *ngIf=\"getColumnFilterType(col) === 'date'\"\n                    >\n                      <p-calendar\n                        [ngModel]=\"latestFilterValues[col.code!] ?? value\"\n                        (ngModelChange)=\"\n                          onFilterValueChange(col.code!, null, $event);\n                          filter($event)\n                        \"\n                        dateFormat=\"dd/mm/yy\"\n                      ></p-calendar>\n                    </ng-template>\n\n                    <!-- MULTISELECT -->\n                    <ng-template\n                      pTemplate=\"filter\"\n                      let-filter=\"filterCallback\"\n                      let-value\n                      *ngIf=\"getColumnFilterType(col) === 'multiSelect'\"\n                    >\n                      <p-multiSelect\n                        [options]=\"col.filterOptions\"\n                        optionLabel=\"label\"\n                        optionValue=\"value\"\n                        [ngModel]=\"latestFilterValues[col.code!] ?? value ?? []\"\n                        (ngModelChange)=\"\n                          onFilterValueChange(col.code!, null, $event);\n                          filter($event ?? [])\n                        \"\n                        display=\"chip\"\n                        placeholder=\"Select\"\n                        class=\"custom-multiselect\"\n                      ></p-multiSelect>\n                    </ng-template>\n                  </p-columnFilter>\n                </ng-container>\n              </div>\n            </ng-template>\n          </th>\n\n          <ng-template #groupHeader>\n            <th\n              [attr.colspan]=\"col.children?.length\"\n              [style.width]=\"col.width || getHeaderWidth(col)\"\n              [ngClass]=\"getHeaderAlignClass(col)\"\n            >\n              <span>{{ col.title }}</span>\n            </th>\n          </ng-template>\n        </ng-container>\n      </tr>\n\n      <tr *ngIf=\"hasGroupedColumns\">\n        <ng-container *ngFor=\"let col of columns\">\n          <ng-container *ngIf=\"col.children\">\n            <th\n              *ngFor=\"let child of col.children\"\n              [style.width]=\"child.width || getHeaderWidth(child)\"\n              [style.padding]=\"'0px'\"\n            ></th>\n          </ng-container>\n        </ng-container>\n      </tr>\n    </ng-template>\n\n    <ng-template pTemplate=\"emptymessage\">\n      <tr class=\"p-datatable-emptymessage\">\n        <td class=\"empty-message-cell\" [attr.colspan]=\"columns.length || 1\">\n          <div class=\"empty-message-wrapper\">\n            <div class=\"empty-message\">\n              <i class=\"pi pi-info-circle\"></i>\n              <p>No records available to display.</p>\n            </div>\n          </div>\n        </td>\n      </tr>\n    </ng-template>\n\n    <ng-template\n      pTemplate=\"body\"\n      let-data\n      let-editing=\"editing\"\n      let-ri=\"rowIndex\"\n    >\n      <tr *ngIf=\"!loading\" [pEditableRow]=\"isEdit ? data : null\">\n        <ng-container *ngFor=\"let col of columns\">\n          <ng-container *ngIf=\"!col.children; else childColumns\">\n            <td\n              *ngIf=\"\n                isEditable(col.code!) && col.type !== TableTypeEnum.ACTION;\n                else normalTD\n              \"\n              [style.width]=\"col.width || getHeaderWidth(col)\"\n              [ngClass]=\"getDataAlignClass(col)\"\n            >\n              <ng-container *ngIf=\"isMultiSelect(col.code); else datePicker\">\n                <p-cellEditor>\n                  <ng-template pTemplate=\"input\">\n                    <p-multiSelect\n                      appendTo=\"body\"\n                      [ngModel]=\"data[col.code!]\"\n                      [style]=\"{ width: '100%' }\"\n                      (ngModelChange)=\"changeHandler(data.id, col.code, $event)\"\n                      [options]=\"optionValues\"\n                    ></p-multiSelect>\n                  </ng-template>\n                  <ng-template pTemplate=\"output\">\n                    <div class=\"multi-select-container\">\n                      <ng-container *ngFor=\"let rec of data[col.code!]\">\n                        <p-tag [value]=\"rec\"></p-tag>\n                      </ng-container>\n                    </div>\n                  </ng-template>\n                </p-cellEditor>\n              </ng-container>\n\n              <ng-template #datePicker>\n                <ng-container *ngIf=\"isDatePicker(col.code); else normalInput\">\n                  <p-cellEditor>\n                    <ng-template pTemplate=\"input\">\n                      <p-calendar\n                        [inputId]=\"data[col.code!]\"\n                        [ngModel]=\"data[col.code!]\"\n                        (ngModelChange)=\"\n                          changeHandler(data.id, col.code, $event)\n                        \"\n                        [dateFormat]=\"'dd/mm/yy'\"\n                      ></p-calendar>\n                    </ng-template>\n                    <ng-template pTemplate=\"output\">\n                      {{ data[col.code!] | customDate }}\n                    </ng-template>\n                  </p-cellEditor>\n                </ng-container>\n              </ng-template>\n\n              <ng-template #normalInput>\n                <p-cellEditor>\n                  <ng-template pTemplate=\"input\">\n                    <input\n                      pInputText\n                      type=\"text\"\n                      [ngModel]=\"data[col.code!]\"\n                      (change)=\"onChange($event, data.id, col.code)\"\n                    />\n                  </ng-template>\n                  <ng-template pTemplate=\"output\">\n                    <ng-container\n                      *ngIf=\"\n                        col.type === TableTypeEnum.AMOUNT;\n                        else normalOutput\n                      \"\n                    >\n                      {{\n                        data[col.code!]\n                          | customCurrency\n                            : getCurrencySymbol(col)\n                            : col.decimalPlaces\n                            : col.thousandSeparator\n                            : col.decimalSeparator\n                      }}\n                    </ng-container>\n                    <ng-template #normalOutput>\n                      {{ data[col.code!] }}\n                    </ng-template>\n                  </ng-template>\n                </p-cellEditor>\n              </ng-template>\n            </td>\n\n            <ng-template #normalTD>\n              <td\n                [style.width]=\"col.width || getHeaderWidth(col)\"\n                [ngClass]=\"[\n                  getDataAlignClass(col),\n                  col.type === TableTypeEnum.ACTION ? 'action-column' : '',\n                ]\"\n              >\n                <ng-container\n                  *ngIf=\"col.type === TableTypeEnum.ACTION; else nonActionCell\"\n                >\n                  <div class=\"action-buttons-container\">\n                    <button\n                      *ngIf=\"isDelete\"\n                      pButton\n                      pRipple\n                      type=\"button\"\n                      icon=\"pi pi-trash\"\n                      (click)=\"Delete(data)\"\n                      class=\"p-button-rounded p-button-text\"\n                    ></button>\n\n                    <div *ngIf=\"isEdit\">\n                      <button\n                        pInitEditableRow\n                        *ngIf=\"!editing\"\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        icon=\"pi pi-pencil\"\n                        (click)=\"initEditableRow(data)\"\n                        class=\"p-button-rounded p-button-text\"\n                      ></button>\n                      <button\n                        *ngIf=\"editing\"\n                        pSaveEditableRow\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        icon=\"pi pi-check\"\n                        (click)=\"saveEditableRow(data)\"\n                        class=\"p-button-rounded p-button-text\"\n                      ></button>\n                      <button\n                        *ngIf=\"editing\"\n                        pCancelEditableRow\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        icon=\"pi pi-times\"\n                        (click)=\"cancelEditableRow(data)\"\n                        class=\"p-button-rounded p-button-text\"\n                      ></button>\n                    </div>\n\n                    <ng-container *ngFor=\"let act of customActions\">\n                      <button\n                        *ngIf=\"isActionVisible(act, data)\"\n                        pButton\n                        pRipple\n                        type=\"button\"\n                        class=\"p-button-rounded p-button-text\"\n                        [icon]=\"act.icon || 'pi pi-ellipsis-h'\"\n                        [ngClass]=\"act.styleClass\"\n                        [disabled]=\"isActionDisabled(act, data)\"\n                        (click)=\"onCustomActionClick(act, data)\"\n                      ></button>\n                    </ng-container>\n                  </div>\n                </ng-container>\n\n                <ng-template #nonActionCell>\n                  <ng-container\n                    *ngIf=\"\n                      col.type === TableTypeEnum.COMPOSED;\n                      else nonComposed\n                    \"\n                  >\n                    <div class=\"composed-cell\">\n                      <ng-container\n                        *ngFor=\"\n                          let composedName of col.composedNames;\n                          let i = index\n                        \"\n                      >\n                        <ng-container\n                          *ngIf=\"\n                            col.composedTypes &&\n                            col.composedTypes[i] === TableTypeEnum.IMAGE\n                          \"\n                        >\n                          <img\n                            [src]=\"data[col.code!]?.[composedName]\"\n                            alt=\"composed-img\"\n                            class=\"composed-image\"\n                            [ngStyle]=\"\n                              getImageStyle(col.composedStyles?.[composedName])\n                            \"\n                          />\n                        </ng-container>\n\n                        <ng-container\n                          *ngIf=\"\n                            col.composedTypes &&\n                            col.composedTypes[i] === TableTypeEnum.STRING\n                          \"\n                        >\n                          <span\n                            class=\"composed-text\"\n                            [ngStyle]=\"\n                              getTitleStyle(col.composedStyles?.[composedName])\n                            \"\n                          >\n                            {{ data[col.code!]?.[composedName] }}\n                          </span>\n                        </ng-container>\n                      </ng-container>\n                    </div>\n                  </ng-container>\n\n                  <ng-template #nonComposed>\n                    <ng-container\n                      *ngIf=\"col.type === TableTypeEnum.AMOUNT; else nonAmount\"\n                    >\n                      {{\n                        data[col.code!]\n                          | customCurrency\n                            : getCurrencySymbol(col)\n                            : col.decimalPlaces\n                            : col.thousandSeparator\n                            : col.decimalSeparator\n                      }}\n                    </ng-container>\n\n                    <ng-template #nonAmount>\n                      <ng-container\n                        *ngIf=\"\n                          col.type === TableTypeEnum.NUMBER;\n                          else nonNumber\n                        \"\n                      >\n                        {{\n                          formatNumber(\n                            data[col.code!],\n                            col.decimalPlaces,\n                            col.thousandSeparator,\n                            col.decimalSeparator\n                          )\n                        }}\n                      </ng-container>\n\n                      <ng-template #nonNumber>\n                        <ng-container\n                          *ngIf=\"\n                            col.type === TableTypeEnum.DATE ||\n                              col.type === TableTypeEnum.DATETIME;\n                            else normalTypes\n                          \"\n                        >\n                          <ng-container *ngIf=\"col.type === TableTypeEnum.DATE\">\n                            {{\n                              formatDateWithColumn(\n                                parseAnyDate(data[col.code!]),\n                                col\n                              )\n                            }}\n                          </ng-container>\n\n                          <ng-container\n                            *ngIf=\"col.type === TableTypeEnum.DATETIME\"\n                          >\n                            {{\n                              formatDateTimeWithColumn(\n                                parseAnyDate(data[col.code!]),\n                                col\n                              )\n                            }}\n                          </ng-container>\n                        </ng-container>\n\n                        <ng-template #normalTypes>\n                          <ng-container\n                            *ngIf=\"\n                              [\n                                TableTypeEnum.STRING,\n                                TableTypeEnum.MULTISELECT,\n                              ].includes(col.type!)\n                            \"\n                          >\n                            {{ data[col.code!] }}\n                          </ng-container>\n                        </ng-template>\n                      </ng-template>\n                    </ng-template>\n                  </ng-template>\n                </ng-template>\n              </td>\n            </ng-template>\n          </ng-container>\n\n          <ng-template #childColumns>\n            <ng-container *ngFor=\"let child of col.children\">\n              <td\n                [style.width]=\"child.width || getHeaderWidth(child)\"\n                [ngClass]=\"getDataAlignClass(child)\"\n              >\n                <ng-container\n                  *ngIf=\"isEditable(child.code); else childNormalTD\"\n                >\n                  <p-cellEditor>\n                    <ng-template pTemplate=\"input\">\n                      <input\n                        pInputText\n                        type=\"text\"\n                        [ngModel]=\"child.code ? data[child.code] : null\"\n                        (change)=\"onChange($event, data.id, child.code)\"\n                      />\n                    </ng-template>\n                    <ng-template pTemplate=\"output\">\n                      {{ child.code ? data[child.code] : \"\" }}\n                    </ng-template>\n                  </p-cellEditor>\n                </ng-container>\n\n                <ng-template #childNormalTD>\n                  {{ child.code ? data[child.code] : \"\" }}\n                </ng-template>\n              </td>\n            </ng-container>\n          </ng-template>\n        </ng-container>\n      </tr>\n    </ng-template>\n  </p-table>\n</div>\n"]}
|