@c80/ui 1.0.32 → 1.0.33
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/table/index.mjs +7 -1
- package/esm2022/lib/table/table-column-visibility.service.mjs +156 -0
- package/esm2022/lib/table/table-crud-state.service.mjs +165 -0
- package/esm2022/lib/table/table-data-converter.service.mjs +149 -0
- package/esm2022/lib/table/table-data-utils.service.mjs +166 -0
- package/esm2022/lib/table/table-selection.service.mjs +151 -0
- package/esm2022/lib/table/table.component.mjs +59 -500
- package/esm2022/lib/table/table.types.mjs +5 -0
- package/lib/table/index.d.ts +6 -0
- package/lib/table/table-column-visibility.service.d.ts +71 -0
- package/lib/table/table-crud-state.service.d.ts +44 -0
- package/lib/table/table-data-converter.service.d.ts +50 -0
- package/lib/table/table-data-utils.service.d.ts +70 -0
- package/lib/table/table-selection.service.d.ts +39 -0
- package/lib/table/table.component.d.ts +14 -99
- package/lib/table/table.types.d.ts +16 -0
- package/package.json +1 -1
|
@@ -1,6 +1,10 @@
|
|
|
1
1
|
import { Component, Input, Output, signal, EventEmitter, inject, } from '@angular/core';
|
|
2
2
|
import { C80IconComponent } from '../icon/icon.component';
|
|
3
3
|
import { C80ModalComponent, ModalService } from '../modal';
|
|
4
|
+
import { TableColumnVisibilityService } from './table-column-visibility.service';
|
|
5
|
+
import { TableDataUtilsService } from './table-data-utils.service';
|
|
6
|
+
import { TableSelectionService } from './table-selection.service';
|
|
7
|
+
import { TableCrudStateService } from './table-crud-state.service';
|
|
4
8
|
import * as i0 from "@angular/core";
|
|
5
9
|
/**
|
|
6
10
|
* C80TableComponent - Componente de tabla avanzado con funcionalidades CRUD
|
|
@@ -58,13 +62,20 @@ import * as i0 from "@angular/core";
|
|
|
58
62
|
* - { accessor: 'name' } → Siempre visible, sin valor por defecto (cadena vacía)
|
|
59
63
|
* - inputValues$ emite { motorPos: 5, weight: 2.3 } → actualiza inputs dinámicamente
|
|
60
64
|
*/ export class C80TableComponent {
|
|
65
|
+
// Servicios inyectados
|
|
61
66
|
modalService = inject(ModalService);
|
|
67
|
+
visibilityService = inject(TableColumnVisibilityService);
|
|
68
|
+
dataUtils = inject(TableDataUtilsService);
|
|
69
|
+
selectionService = inject(TableSelectionService);
|
|
70
|
+
crudService = inject(TableCrudStateService);
|
|
71
|
+
// Inputs
|
|
62
72
|
data$;
|
|
63
73
|
columns = [];
|
|
64
74
|
inputValues$; // Observable para actualizar valores de inputs dinámicamente en creación/edición
|
|
65
75
|
size = 0; // Tamaño de la tabla (0 = sin límite, > 0 aplica max-height)
|
|
66
76
|
multiple = true; // Permite selección múltiple por defecto
|
|
67
77
|
noConfirm = false; // Si es true, no muestra confirmaciones modales
|
|
78
|
+
// Outputs
|
|
68
79
|
createAction = new EventEmitter();
|
|
69
80
|
updateAction = new EventEmitter();
|
|
70
81
|
deleteAction = new EventEmitter();
|
|
@@ -78,16 +89,21 @@ import * as i0 from "@angular/core";
|
|
|
78
89
|
searchTerm = new EventEmitter();
|
|
79
90
|
errorEvent = new EventEmitter();
|
|
80
91
|
selectable = new EventEmitter();
|
|
92
|
+
// Estado principal
|
|
81
93
|
data = signal([]);
|
|
82
94
|
keys = signal([]);
|
|
83
|
-
creating = signal(false);
|
|
84
|
-
newRow = signal(null);
|
|
85
|
-
editing = signal(null); // id of row being edited
|
|
86
|
-
editRow = signal(null);
|
|
87
|
-
selectedItems = signal(new Set());
|
|
88
|
-
selectAllChecked = signal(false);
|
|
89
|
-
selectAllIndeterminate = signal(false);
|
|
90
95
|
searchValue = signal('');
|
|
96
|
+
// Estado de selección (delegado a servicio)
|
|
97
|
+
selectionState = this.selectionService.createSelectionState();
|
|
98
|
+
selectedItems = this.selectionState.selectedItems;
|
|
99
|
+
selectAllChecked = this.selectionState.selectAllChecked;
|
|
100
|
+
selectAllIndeterminate = this.selectionState.selectAllIndeterminate;
|
|
101
|
+
// Estado CRUD (delegado a servicio)
|
|
102
|
+
crudState = this.crudService.createCrudState();
|
|
103
|
+
creating = this.crudState.creating;
|
|
104
|
+
newRow = this.crudState.newRow;
|
|
105
|
+
editing = this.crudState.editing;
|
|
106
|
+
editRow = this.crudState.editRow;
|
|
91
107
|
// Flags to check if events have listeners
|
|
92
108
|
hasCreateActionListener = signal(false);
|
|
93
109
|
hasUpdateActionListener = signal(false);
|
|
@@ -110,34 +126,15 @@ import * as i0 from "@angular/core";
|
|
|
110
126
|
* Actualiza las keys de columnas visibles basándose en el estado actual
|
|
111
127
|
*/
|
|
112
128
|
updateVisibleKeys() {
|
|
113
|
-
const
|
|
114
|
-
this.keys.set(
|
|
129
|
+
const visibleKeys = this.visibilityService.updateVisibleKeys(this.columns, this.data(), this.creating(), this.editing());
|
|
130
|
+
this.keys.set(visibleKeys);
|
|
115
131
|
}
|
|
116
132
|
/**
|
|
117
133
|
* Aplica valores parciales a los inputs en modo creación o edición.
|
|
118
134
|
* Solo actualiza si estamos en modo creación (creating = true) o edición (editing != null)
|
|
119
135
|
*/
|
|
120
136
|
applyInputValues(partialValues) {
|
|
121
|
-
|
|
122
|
-
if (!this.creating() && this.editing() === null) {
|
|
123
|
-
return;
|
|
124
|
-
}
|
|
125
|
-
// Aplicar valores en modo creación
|
|
126
|
-
if (this.creating()) {
|
|
127
|
-
const currentRow = this.newRow();
|
|
128
|
-
if (currentRow) {
|
|
129
|
-
const updatedRow = { ...currentRow, ...partialValues };
|
|
130
|
-
this.newRow.set(updatedRow);
|
|
131
|
-
}
|
|
132
|
-
}
|
|
133
|
-
// Aplicar valores en modo edición
|
|
134
|
-
if (this.editing() !== null) {
|
|
135
|
-
const currentEditRow = this.editRow();
|
|
136
|
-
if (currentEditRow) {
|
|
137
|
-
const updatedEditRow = { ...currentEditRow, ...partialValues };
|
|
138
|
-
this.editRow.set(updatedEditRow);
|
|
139
|
-
}
|
|
140
|
-
}
|
|
137
|
+
this.crudState.applyInputValues(partialValues);
|
|
141
138
|
}
|
|
142
139
|
ngOnInit() {
|
|
143
140
|
// Check if the outputs have listeners
|
|
@@ -157,7 +154,7 @@ import * as i0 from "@angular/core";
|
|
|
157
154
|
return;
|
|
158
155
|
this.dataSub = this.data$.subscribe({
|
|
159
156
|
next: (items) => {
|
|
160
|
-
this.applySorting(items);
|
|
157
|
+
this.dataUtils.applySorting(items, this.columns);
|
|
161
158
|
this.data.set(items);
|
|
162
159
|
// Actualizar las columnas visibles basándose en el estado actual
|
|
163
160
|
this.updateVisibleKeys();
|
|
@@ -231,198 +228,50 @@ import * as i0 from "@angular/core";
|
|
|
231
228
|
}
|
|
232
229
|
}
|
|
233
230
|
startCreate() {
|
|
234
|
-
this.
|
|
231
|
+
this.crudState.startCreate(this.columns, this.data());
|
|
235
232
|
// Actualizar keys de columnas visibles cuando entramos en modo creación
|
|
236
233
|
this.updateVisibleKeys();
|
|
237
|
-
// Inicializa newRow solo con columnas visibles y no readOnly
|
|
238
|
-
// En modo creación, mostramos columnas con hideIfAllValuesAreNull para permitir entrada de datos
|
|
239
|
-
const row = {};
|
|
240
|
-
this.columns
|
|
241
|
-
.filter((col) => this.isColumnVisible(col, true) && !col.readOnly)
|
|
242
|
-
.forEach((col) => {
|
|
243
|
-
// Si la columna tiene un valor por defecto, usarlo; si no, usar cadena vacía
|
|
244
|
-
const defaultValue = col.default !== undefined ? col.default : '';
|
|
245
|
-
row[col.accessor] = defaultValue;
|
|
246
|
-
});
|
|
247
|
-
this.newRow.set(row);
|
|
248
234
|
}
|
|
249
235
|
cancelCreate() {
|
|
250
|
-
this.
|
|
251
|
-
this.newRow.set(null);
|
|
236
|
+
this.crudState.cancelCreate();
|
|
252
237
|
// Actualizar keys de columnas visibles cuando salimos del modo creación
|
|
253
238
|
this.updateVisibleKeys();
|
|
254
239
|
}
|
|
255
240
|
updateNewRow(key, value) {
|
|
256
|
-
|
|
257
|
-
if (!current)
|
|
258
|
-
return;
|
|
259
|
-
this.newRow.set({ ...current, [key]: value });
|
|
241
|
+
this.crudState.updateNewRow(key, value);
|
|
260
242
|
}
|
|
261
243
|
saveCreate() {
|
|
262
|
-
|
|
263
|
-
if (!row)
|
|
264
|
-
return;
|
|
265
|
-
// Validar campos requeridos antes de crear - solo columnas visibles y no readOnly
|
|
266
|
-
// En modo creación, incluimos columnas con hideIfAllValuesAreNull
|
|
267
|
-
const visibleColumns = this.columns.filter((col) => this.isColumnVisible(col, true) && !col.readOnly);
|
|
268
|
-
const converted = visibleColumns.reduce((acc, col) => {
|
|
269
|
-
acc[col.accessor] = this.convertCellValue(row[col.accessor], col);
|
|
270
|
-
return acc;
|
|
271
|
-
}, {});
|
|
272
|
-
this.createAction.emit({
|
|
273
|
-
row: converted,
|
|
274
|
-
done: (success) => {
|
|
275
|
-
if (success) {
|
|
276
|
-
this.cancelCreate();
|
|
277
|
-
}
|
|
278
|
-
},
|
|
279
|
-
});
|
|
280
|
-
}
|
|
281
|
-
/**
|
|
282
|
-
* Converts a cell value based on column type or sample data.
|
|
283
|
-
* Handles stringification of objects for string columns.
|
|
284
|
-
*/
|
|
285
|
-
/**
|
|
286
|
-
* Converts a cell value based on column type or sample data.
|
|
287
|
-
* Delegates to type-specific helpers for clarity and maintainability.
|
|
288
|
-
*/
|
|
289
|
-
convertCellValue(value, col) {
|
|
290
|
-
if (col.type === 'boolean')
|
|
291
|
-
return this.toBoolean(value);
|
|
292
|
-
if (col.type === 'number')
|
|
293
|
-
return this.toNumber(value);
|
|
294
|
-
if (col.type === 'integer')
|
|
295
|
-
return this.toInteger(value);
|
|
296
|
-
if (col.type === 'string' || col.type === 'password')
|
|
297
|
-
return this.toStringValue(value);
|
|
298
|
-
// Fallback: use sample data if available
|
|
299
|
-
const sample = this.data()[0];
|
|
300
|
-
if (sample) {
|
|
301
|
-
const sampleValue = this.getCellValue(sample, col.accessor);
|
|
302
|
-
if (typeof sampleValue === 'boolean')
|
|
303
|
-
return this.toBoolean(value);
|
|
304
|
-
if (typeof sampleValue === 'number')
|
|
305
|
-
return this.toNumber(value);
|
|
306
|
-
if (typeof sampleValue === 'string')
|
|
307
|
-
return this.toStringValue(value);
|
|
308
|
-
}
|
|
309
|
-
return value;
|
|
310
|
-
}
|
|
311
|
-
/** Converts value to boolean using best practices. */
|
|
312
|
-
toBoolean(value) {
|
|
313
|
-
if (typeof value === 'boolean')
|
|
314
|
-
return value;
|
|
315
|
-
if (typeof value === 'string')
|
|
316
|
-
return value.trim().toLowerCase() === 'true' || value.trim() === '1';
|
|
317
|
-
if (typeof value === 'number')
|
|
318
|
-
return value === 1;
|
|
319
|
-
return false;
|
|
320
|
-
}
|
|
321
|
-
/** Converts value to number using best practices. */
|
|
322
|
-
toNumber(value) {
|
|
323
|
-
if (typeof value === 'number')
|
|
324
|
-
return value;
|
|
325
|
-
if (typeof value === 'string') {
|
|
326
|
-
const trimmed = value.trim();
|
|
327
|
-
if (trimmed === '')
|
|
328
|
-
return undefined;
|
|
329
|
-
const num = Number(trimmed);
|
|
330
|
-
return isNaN(num) ? undefined : num;
|
|
331
|
-
}
|
|
332
|
-
if (typeof value === 'boolean')
|
|
333
|
-
return value ? 1 : 0;
|
|
334
|
-
return undefined;
|
|
335
|
-
}
|
|
336
|
-
/** Converts value to integer using best practices. */
|
|
337
|
-
toInteger(value) {
|
|
338
|
-
if (typeof value === 'number')
|
|
339
|
-
return Math.floor(value);
|
|
340
|
-
if (typeof value === 'string') {
|
|
341
|
-
const trimmed = value.trim();
|
|
342
|
-
if (trimmed === '')
|
|
343
|
-
return undefined;
|
|
344
|
-
const num = Number(trimmed);
|
|
345
|
-
return isNaN(num) ? undefined : Math.floor(num);
|
|
346
|
-
}
|
|
347
|
-
if (typeof value === 'boolean')
|
|
348
|
-
return value ? 1 : 0;
|
|
349
|
-
return undefined;
|
|
350
|
-
}
|
|
351
|
-
/** Converts value to string using best practices, always stringifies objects. */
|
|
352
|
-
toStringValue(value) {
|
|
353
|
-
if (value == null)
|
|
354
|
-
return '';
|
|
355
|
-
if (typeof value === 'string')
|
|
356
|
-
return value;
|
|
357
|
-
if (typeof value === 'number' || typeof value === 'boolean')
|
|
358
|
-
return String(value);
|
|
359
|
-
if (typeof value === 'object') {
|
|
360
|
-
try {
|
|
361
|
-
return JSON.stringify(value);
|
|
362
|
-
}
|
|
363
|
-
catch {
|
|
364
|
-
return '[object Object]';
|
|
365
|
-
}
|
|
366
|
-
}
|
|
367
|
-
// For functions, symbols, undefined, etc., return empty string
|
|
368
|
-
return '';
|
|
244
|
+
this.crudState.saveCreate(this.columns, this.data(), this.createAction);
|
|
369
245
|
}
|
|
370
246
|
onEdit(row) {
|
|
371
|
-
this.
|
|
247
|
+
this.crudState.startEdit(row, this.columns, this.data());
|
|
372
248
|
// Actualizar keys de columnas visibles cuando entramos en modo edición
|
|
373
249
|
this.updateVisibleKeys();
|
|
374
|
-
const edit = {};
|
|
375
|
-
this.columns
|
|
376
|
-
.filter((col) => this.isColumnVisible(col, true) && !col.readOnly)
|
|
377
|
-
.forEach((col) => {
|
|
378
|
-
const value = this.getCellValue(row, col.accessor);
|
|
379
|
-
edit[col.accessor] = value;
|
|
380
|
-
});
|
|
381
|
-
this.editRow.set(edit);
|
|
382
250
|
}
|
|
383
251
|
cancelEdit() {
|
|
384
|
-
this.
|
|
385
|
-
this.editRow.set(null);
|
|
252
|
+
this.crudState.cancelEdit();
|
|
386
253
|
// Actualizar keys de columnas visibles cuando salimos del modo edición
|
|
387
254
|
this.updateVisibleKeys();
|
|
388
255
|
}
|
|
389
256
|
onEditInput(event, key, col) {
|
|
390
|
-
const current = this.editRow();
|
|
391
|
-
if (!current)
|
|
392
|
-
return;
|
|
393
257
|
if (col?.type === 'boolean') {
|
|
394
258
|
const checked = event.target.checked;
|
|
395
|
-
this.
|
|
259
|
+
this.crudState.updateEditRow(key, checked);
|
|
396
260
|
}
|
|
397
261
|
else if (col?.type === 'enum') {
|
|
398
262
|
const value = event.target.value;
|
|
399
|
-
this.
|
|
263
|
+
this.crudState.updateEditRow(key, value);
|
|
400
264
|
}
|
|
401
265
|
else {
|
|
402
266
|
const target = event.target;
|
|
403
267
|
if (target && typeof target.value === 'string') {
|
|
404
|
-
this.
|
|
268
|
+
this.crudState.updateEditRow(key, target.value);
|
|
405
269
|
}
|
|
406
270
|
}
|
|
407
271
|
}
|
|
408
272
|
saveEdit(row) {
|
|
409
273
|
const id = row['id'];
|
|
410
|
-
|
|
411
|
-
return;
|
|
412
|
-
const visibleColumns = this.columns.filter((col) => this.isColumnVisible(col, true) && !col.readOnly);
|
|
413
|
-
const converted = visibleColumns.reduce((acc, col) => {
|
|
414
|
-
acc[col.accessor] = this.convertCellValue(this.editRow()[col.accessor], col);
|
|
415
|
-
return acc;
|
|
416
|
-
}, {});
|
|
417
|
-
this.updateAction.emit({
|
|
418
|
-
id,
|
|
419
|
-
changes: converted,
|
|
420
|
-
done: (success) => {
|
|
421
|
-
if (success) {
|
|
422
|
-
this.cancelEdit();
|
|
423
|
-
}
|
|
424
|
-
},
|
|
425
|
-
});
|
|
274
|
+
this.crudState.saveEdit(id, this.columns, this.data(), this.updateAction);
|
|
426
275
|
}
|
|
427
276
|
/**
|
|
428
277
|
* TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).
|
|
@@ -502,351 +351,61 @@ import * as i0 from "@angular/core";
|
|
|
502
351
|
this.searchValue.set('');
|
|
503
352
|
this.searchTerm.emit('');
|
|
504
353
|
}
|
|
505
|
-
|
|
506
|
-
* Gets the value from an object using dot notation (e.g., 'task.name')
|
|
507
|
-
*/
|
|
508
|
-
getNestedValue(obj, accessor) {
|
|
509
|
-
return accessor.split('.').reduce((current, key) => {
|
|
510
|
-
return current && typeof current === 'object' ? current[key] : undefined;
|
|
511
|
-
}, obj);
|
|
512
|
-
}
|
|
513
|
-
/**
|
|
514
|
-
* Gets the value for a cell using the accessor, supporting dot notation for nested properties
|
|
515
|
-
*/
|
|
354
|
+
// Métodos delegados a servicios
|
|
516
355
|
getCellValue(row, accessor) {
|
|
517
|
-
|
|
518
|
-
return this.getNestedValue(row, accessor);
|
|
519
|
-
}
|
|
520
|
-
return row[accessor];
|
|
356
|
+
return this.dataUtils.getCellValue(row, accessor);
|
|
521
357
|
}
|
|
522
|
-
/**
|
|
523
|
-
* Returns the display value for a cell, showing '-' for falsy values except 0, false, and empty objects/arrays
|
|
524
|
-
*/
|
|
525
358
|
getDisplayValue(value) {
|
|
526
|
-
|
|
527
|
-
if (value === 0) {
|
|
528
|
-
return '0';
|
|
529
|
-
}
|
|
530
|
-
// If value is false, it should be handled by boolean logic in template, not here
|
|
531
|
-
if (value === false) {
|
|
532
|
-
return 'false';
|
|
533
|
-
}
|
|
534
|
-
// Handle objects and arrays (including empty ones)
|
|
535
|
-
if (typeof value === 'object' && value !== null) {
|
|
536
|
-
try {
|
|
537
|
-
return JSON.stringify(value);
|
|
538
|
-
}
|
|
539
|
-
catch {
|
|
540
|
-
return '[object Object]';
|
|
541
|
-
}
|
|
542
|
-
}
|
|
543
|
-
// If value is other falsy values (null, undefined, ''), show '-'
|
|
544
|
-
if (!value) {
|
|
545
|
-
return '-';
|
|
546
|
-
}
|
|
547
|
-
// Handle other types explicitly
|
|
548
|
-
if (typeof value === 'string' ||
|
|
549
|
-
typeof value === 'number' ||
|
|
550
|
-
typeof value === 'boolean') {
|
|
551
|
-
return String(value);
|
|
552
|
-
}
|
|
553
|
-
// For any other type (function, symbol, etc.), return a safe fallback
|
|
554
|
-
return '-';
|
|
359
|
+
return this.dataUtils.getDisplayValue(value);
|
|
555
360
|
}
|
|
556
|
-
/**
|
|
557
|
-
* Gets the display text for an enum value
|
|
558
|
-
*/
|
|
559
361
|
getEnumDisplayValue(value, col) {
|
|
560
|
-
|
|
561
|
-
return '-';
|
|
562
|
-
}
|
|
563
|
-
const displayValue = col.enum[value];
|
|
564
|
-
return (displayValue ||
|
|
565
|
-
(typeof value === 'string' || typeof value === 'number'
|
|
566
|
-
? String(value)
|
|
567
|
-
: '-'));
|
|
362
|
+
return this.dataUtils.getEnumDisplayValue(value, col);
|
|
568
363
|
}
|
|
569
|
-
/**
|
|
570
|
-
* Gets the enum options as an array for select dropdowns
|
|
571
|
-
*/
|
|
572
364
|
getEnumOptions(col) {
|
|
573
|
-
|
|
574
|
-
return [];
|
|
575
|
-
return Object.entries(col.enum).map(([value, label]) => ({
|
|
576
|
-
value: isNaN(Number(value)) ? value : Number(value),
|
|
577
|
-
label,
|
|
578
|
-
}));
|
|
365
|
+
return this.dataUtils.getEnumOptions(col);
|
|
579
366
|
}
|
|
580
|
-
/**
|
|
581
|
-
* Gets the CSS color for a cell value based on the column's color configuration
|
|
582
|
-
*/
|
|
583
367
|
getCellColor(value, col) {
|
|
584
|
-
|
|
585
|
-
return undefined;
|
|
586
|
-
}
|
|
587
|
-
return col.color[value];
|
|
368
|
+
return this.dataUtils.getCellColor(value, col);
|
|
588
369
|
}
|
|
589
|
-
/**
|
|
590
|
-
* Calcula el max-height de la tabla basado en el tamaño
|
|
591
|
-
* Si size es 0, retorna undefined (sin límite de altura)
|
|
592
|
-
*/
|
|
593
370
|
getTableMaxHeight() {
|
|
594
|
-
|
|
595
|
-
return undefined; // Sin límite de altura
|
|
596
|
-
}
|
|
597
|
-
// Altura base de 400px * size
|
|
598
|
-
const baseHeight = 400;
|
|
599
|
-
const maxHeight = Math.round(baseHeight * this.size);
|
|
600
|
-
return `${maxHeight}px`;
|
|
371
|
+
return this.dataUtils.getTableMaxHeight(this.size);
|
|
601
372
|
}
|
|
602
373
|
/**
|
|
603
374
|
* Selection methods
|
|
604
375
|
*/
|
|
605
376
|
clearSelection() {
|
|
606
|
-
this.
|
|
607
|
-
this.
|
|
608
|
-
this.emitSelection();
|
|
377
|
+
this.selectionState.clearSelection();
|
|
378
|
+
this.selectionService.emitSelection(this.selectionState, this.data(), this.selectable);
|
|
609
379
|
}
|
|
610
380
|
/**
|
|
611
381
|
* Mantiene la selección existente después de actualizar los datos,
|
|
612
382
|
* eliminando solo los IDs que ya no existen en los nuevos datos
|
|
613
383
|
*/
|
|
614
384
|
preserveSelection() {
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
if (currentSelection.size === 0 || currentData.length === 0) {
|
|
618
|
-
// Si no hay selección o no hay datos, limpiar la selección
|
|
619
|
-
this.clearSelection();
|
|
620
|
-
return;
|
|
621
|
-
}
|
|
622
|
-
// Obtener los IDs disponibles en los nuevos datos
|
|
623
|
-
const availableIds = new Set(currentData.map(item => item['id']));
|
|
624
|
-
// Filtrar la selección actual para mantener solo los IDs que aún existen
|
|
625
|
-
const preservedSelection = new Set(Array.from(currentSelection).filter(id => availableIds.has(id)));
|
|
626
|
-
// Actualizar la selección con los IDs preservados
|
|
627
|
-
this.selectedItems.set(preservedSelection);
|
|
628
|
-
this.updateSelectAllState();
|
|
629
|
-
this.emitSelection();
|
|
385
|
+
this.selectionState.preserveSelection(this.data());
|
|
386
|
+
this.selectionService.emitSelection(this.selectionState, this.data(), this.selectable);
|
|
630
387
|
}
|
|
631
388
|
toggleSelectAll() {
|
|
632
|
-
|
|
633
|
-
|
|
634
|
-
if (currentSelection.size === allItems.length) {
|
|
635
|
-
// Deselect all
|
|
636
|
-
this.clearSelection();
|
|
637
|
-
}
|
|
638
|
-
else {
|
|
639
|
-
// Select all
|
|
640
|
-
const allIds = new Set(allItems.map(item => item['id']));
|
|
641
|
-
this.selectedItems.set(allIds);
|
|
642
|
-
this.updateSelectAllState();
|
|
643
|
-
this.emitSelection();
|
|
644
|
-
}
|
|
389
|
+
this.selectionState.toggleSelectAll(this.data());
|
|
390
|
+
this.selectionService.emitSelection(this.selectionState, this.data(), this.selectable);
|
|
645
391
|
}
|
|
646
392
|
toggleItemSelection(item) {
|
|
647
|
-
|
|
648
|
-
|
|
649
|
-
|
|
650
|
-
// Selección simple: solo permitir un elemento seleccionado
|
|
651
|
-
if (currentSelection.has(id)) {
|
|
652
|
-
// Deseleccionar el elemento actual
|
|
653
|
-
currentSelection.clear();
|
|
654
|
-
}
|
|
655
|
-
else {
|
|
656
|
-
// Seleccionar solo este elemento
|
|
657
|
-
currentSelection.clear();
|
|
658
|
-
currentSelection.add(id);
|
|
659
|
-
}
|
|
660
|
-
}
|
|
661
|
-
else if (currentSelection.has(id)) {
|
|
662
|
-
// Selección múltiple: deseleccionar elemento existente
|
|
663
|
-
currentSelection.delete(id);
|
|
664
|
-
}
|
|
665
|
-
else {
|
|
666
|
-
// Selección múltiple: agregar nuevo elemento
|
|
667
|
-
currentSelection.add(id);
|
|
668
|
-
}
|
|
669
|
-
this.selectedItems.set(currentSelection);
|
|
670
|
-
this.updateSelectAllState();
|
|
671
|
-
this.emitSelection();
|
|
393
|
+
this.selectionState.toggleItemSelection(item, this.multiple);
|
|
394
|
+
this.selectionState.updateSelectAllState(this.data());
|
|
395
|
+
this.selectionService.emitSelection(this.selectionState, this.data(), this.selectable);
|
|
672
396
|
}
|
|
673
397
|
isItemSelected(item) {
|
|
674
|
-
|
|
675
|
-
return this.selectedItems().has(id);
|
|
676
|
-
}
|
|
677
|
-
updateSelectAllState() {
|
|
678
|
-
const selectedCount = this.selectedItems().size;
|
|
679
|
-
const totalCount = this.data().length;
|
|
680
|
-
if (selectedCount === 0) {
|
|
681
|
-
this.selectAllChecked.set(false);
|
|
682
|
-
this.selectAllIndeterminate.set(false);
|
|
683
|
-
}
|
|
684
|
-
else if (selectedCount === totalCount) {
|
|
685
|
-
this.selectAllChecked.set(true);
|
|
686
|
-
this.selectAllIndeterminate.set(false);
|
|
687
|
-
}
|
|
688
|
-
else {
|
|
689
|
-
this.selectAllChecked.set(false);
|
|
690
|
-
this.selectAllIndeterminate.set(true);
|
|
691
|
-
}
|
|
692
|
-
}
|
|
693
|
-
emitSelection() {
|
|
694
|
-
const selectedIds = this.selectedItems();
|
|
695
|
-
const selectedItems = this.data().filter(item => selectedIds.has(item['id']));
|
|
696
|
-
this.selectable.emit(selectedItems);
|
|
697
|
-
}
|
|
698
|
-
/**
|
|
699
|
-
* Verifica si un valor individual está vacío/nulo
|
|
700
|
-
*/
|
|
701
|
-
isValueEmpty(value) {
|
|
702
|
-
// Considerar vacío: null, undefined, ''
|
|
703
|
-
if (value === null || value === undefined || value === '') {
|
|
704
|
-
return true;
|
|
705
|
-
}
|
|
706
|
-
// Para números, 0 se considera como valor válido, no vacío
|
|
707
|
-
if (typeof value === 'number') {
|
|
708
|
-
return false;
|
|
709
|
-
}
|
|
710
|
-
// Para booleanos, false se considera como valor válido, no vacío
|
|
711
|
-
if (typeof value === 'boolean') {
|
|
712
|
-
return false;
|
|
713
|
-
}
|
|
714
|
-
// Para arrays vacíos
|
|
715
|
-
if (Array.isArray(value) && value.length === 0) {
|
|
716
|
-
return true;
|
|
717
|
-
}
|
|
718
|
-
// Para objetos vacíos
|
|
719
|
-
if (typeof value === 'object' && value !== null && Object.keys(value).length === 0) {
|
|
720
|
-
return true;
|
|
721
|
-
}
|
|
722
|
-
return false;
|
|
398
|
+
return this.selectionState.isItemSelected(item);
|
|
723
399
|
}
|
|
724
|
-
|
|
725
|
-
* Verifica si todos los valores de una columna están vacíos/nulos
|
|
726
|
-
*/
|
|
727
|
-
areAllColumnValuesEmpty(column) {
|
|
728
|
-
const currentData = this.data();
|
|
729
|
-
if (currentData.length === 0) {
|
|
730
|
-
return true; // Si no hay datos, consideramos la columna como vacía
|
|
731
|
-
}
|
|
732
|
-
return currentData.every(row => {
|
|
733
|
-
const value = this.getCellValue(row, column.accessor);
|
|
734
|
-
return this.isValueEmpty(value);
|
|
735
|
-
});
|
|
736
|
-
}
|
|
737
|
-
/**
|
|
738
|
-
* Determina si una columna debe ser visible basándose en su configuración y datos.
|
|
739
|
-
*
|
|
740
|
-
* Orden de prioridad:
|
|
741
|
-
* 1. Si visible === false: SIEMPRE se oculta (máxima prioridad)
|
|
742
|
-
* 2. Si hideIfAllValuesAreNull === true Y todos los valores están vacíos: se oculta
|
|
743
|
-
* EXCEPCIÓN: En modo creación (creating === true), estas columnas se muestran para permitir entrada de datos
|
|
744
|
-
* 3. Por defecto: se muestra
|
|
745
|
-
*
|
|
746
|
-
* @param column - La definición de la columna
|
|
747
|
-
* @param forceShowInCreation - Si es true, ignora hideIfAllValuesAreNull (usado en modo creación)
|
|
748
|
-
*/
|
|
400
|
+
// Métodos de visibilidad delegados a servicio
|
|
749
401
|
isColumnVisible(column, forceShowInCreation = false) {
|
|
750
|
-
|
|
751
|
-
if (column.visible === false) {
|
|
752
|
-
return false;
|
|
753
|
-
}
|
|
754
|
-
// PRIORIDAD 2: Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
|
|
755
|
-
// EXCEPCIÓN: En modo creación, mostramos estas columnas para permitir entrada de datos
|
|
756
|
-
if (column.hideIfAllValuesAreNull === true && !forceShowInCreation && this.areAllColumnValuesEmpty(column)) {
|
|
757
|
-
return false;
|
|
758
|
-
}
|
|
759
|
-
// Por defecto: mostrar
|
|
760
|
-
return true;
|
|
402
|
+
return this.visibilityService.isColumnVisible(column, this.data(), forceShowInCreation);
|
|
761
403
|
}
|
|
762
|
-
/**
|
|
763
|
-
* Determina si una columna debe ser visible en los headers.
|
|
764
|
-
* Los headers se muestran si la columna es visible en cualquiera de estos casos:
|
|
765
|
-
* - Visualización normal
|
|
766
|
-
* - Modo creación
|
|
767
|
-
* - Hay una fila en modo edición y esa fila tiene valor en la columna
|
|
768
|
-
*
|
|
769
|
-
* @param column - La definición de la columna
|
|
770
|
-
*/
|
|
771
404
|
isColumnVisibleInHeader(column) {
|
|
772
|
-
|
|
773
|
-
if (column.visible === false) {
|
|
774
|
-
return false;
|
|
775
|
-
}
|
|
776
|
-
// Si estamos en modo creación, mostrar columnas con hideIfAllValuesAreNull
|
|
777
|
-
if (this.creating() && column.hideIfAllValuesAreNull === true) {
|
|
778
|
-
return true;
|
|
779
|
-
}
|
|
780
|
-
// Si hay una fila en modo edición y la columna tiene hideIfAllValuesAreNull
|
|
781
|
-
if (this.editing() !== null && column.hideIfAllValuesAreNull === true) {
|
|
782
|
-
// Buscar la fila que se está editando
|
|
783
|
-
const editingRowId = this.editing();
|
|
784
|
-
const editingRow = this.data().find(row => row['id'] === editingRowId);
|
|
785
|
-
if (editingRow) {
|
|
786
|
-
// Solo mostrar el header si la fila en edición tiene valor en esta columna
|
|
787
|
-
const cellValue = this.getCellValue(editingRow, column.accessor);
|
|
788
|
-
return !this.isValueEmpty(cellValue);
|
|
789
|
-
}
|
|
790
|
-
// Si no se encuentra la fila, no mostrar el header
|
|
791
|
-
return false;
|
|
792
|
-
}
|
|
793
|
-
// Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
|
|
794
|
-
if (column.hideIfAllValuesAreNull === true && this.areAllColumnValuesEmpty(column)) {
|
|
795
|
-
return false;
|
|
796
|
-
}
|
|
797
|
-
// Por defecto: mostrar
|
|
798
|
-
return true;
|
|
405
|
+
return this.visibilityService.isColumnVisibleInHeader(column, this.data(), this.creating(), this.editing());
|
|
799
406
|
}
|
|
800
|
-
/**
|
|
801
|
-
* Determina si una columna debe ser visible en una fila específica considerando el modo edición.
|
|
802
|
-
* En modo edición, las columnas con hideIfAllValuesAreNull se muestran solo si tienen valor.
|
|
803
|
-
*
|
|
804
|
-
* @param column - La definición de la columna
|
|
805
|
-
* @param row - La fila actual
|
|
806
|
-
*/
|
|
807
407
|
isColumnVisibleForRow(column, row) {
|
|
808
|
-
|
|
809
|
-
if (column.visible === false) {
|
|
810
|
-
return false;
|
|
811
|
-
}
|
|
812
|
-
// PRIORIDAD 2: Si esta fila específica está en modo edición y la columna tiene hideIfAllValuesAreNull
|
|
813
|
-
const isEditingThisRow = this.editing() === row['id'];
|
|
814
|
-
if (isEditingThisRow && column.hideIfAllValuesAreNull === true) {
|
|
815
|
-
// En edición, solo mostrar si esta fila específica tiene valor en esta columna
|
|
816
|
-
const cellValue = this.getCellValue(row, column.accessor);
|
|
817
|
-
return !this.isValueEmpty(cellValue);
|
|
818
|
-
}
|
|
819
|
-
// PRIORIDAD 3: Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
|
|
820
|
-
if (column.hideIfAllValuesAreNull === true && this.areAllColumnValuesEmpty(column)) {
|
|
821
|
-
return false;
|
|
822
|
-
}
|
|
823
|
-
// Por defecto: mostrar
|
|
824
|
-
return true;
|
|
825
|
-
}
|
|
826
|
-
applySorting(items) {
|
|
827
|
-
const orderedColumns = this.columns.filter((col) => col.order);
|
|
828
|
-
if (orderedColumns.length > 0) {
|
|
829
|
-
orderedColumns.forEach((col) => {
|
|
830
|
-
items.sort((a, b) => {
|
|
831
|
-
const valueA = this.getCellValue(a, col.accessor);
|
|
832
|
-
const valueB = this.getCellValue(b, col.accessor);
|
|
833
|
-
if (col.order === 'ASC') {
|
|
834
|
-
if (valueA > valueB)
|
|
835
|
-
return 1;
|
|
836
|
-
if (valueA < valueB)
|
|
837
|
-
return -1;
|
|
838
|
-
return 0;
|
|
839
|
-
}
|
|
840
|
-
else {
|
|
841
|
-
if (valueA < valueB)
|
|
842
|
-
return 1;
|
|
843
|
-
if (valueA > valueB)
|
|
844
|
-
return -1;
|
|
845
|
-
return 0;
|
|
846
|
-
}
|
|
847
|
-
});
|
|
848
|
-
});
|
|
849
|
-
}
|
|
408
|
+
return this.visibilityService.isColumnVisibleForRow(column, row, this.data(), this.editing());
|
|
850
409
|
}
|
|
851
410
|
onUpload(row) {
|
|
852
411
|
this.uploadAction.emit(row);
|
|
@@ -897,4 +456,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
897
456
|
}], selectable: [{
|
|
898
457
|
type: Output
|
|
899
458
|
}] } });
|
|
900
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table.component.ts","../../../../../libs/ui/src/lib/table/table.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  signal,\n  EventEmitter,\n  OnInit,\n  OnDestroy,\n  inject,\n} from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\nimport { C80IconComponent } from '../icon/icon.component';\nimport { C80ModalComponent, ModalService } from '../modal';\n\nexport interface C80TableColDef {\n  accessor: string;\n  label: string;\n  visible?: boolean; // Si no se especifica, se asume true. Si es false, la columna se oculta SIEMPRE (máxima prioridad)\n  type?: 'string' | 'number' | 'integer' | 'boolean' | 'password' | 'enum'; // Tipo de dato para la columna\n  order?: 'ASC' | 'DESC'; // Ordenamiento de la columna\n  readOnly?: boolean; // Si no se especifica, se asume false\n  enum?: Record<string | number, string>; // Opciones del enum para type 'enum'\n  color?: Record<string | number, string>; // Colores CSS basados en el valor de la celda\n  hideIfAllValuesAreNull?: boolean; // Si es true, oculta la columna cuando todos los valores están vacíos (default: false).\n  // Solo aplica si visible !== false. EXCEPCIÓN: En modo creación Y edición siempre se muestra.\n  default?: unknown; // Valor por defecto para usar en modo creación. Puede ser string, number, boolean, etc.\n}\n\nexport type ID = number | string;\n\n/**\n * C80TableComponent - Componente de tabla avanzado con funcionalidades CRUD\n *\n * COMPORTAMIENTO DE VISIBILIDAD DE COLUMNAS:\n * ========================================\n *\n * 1. PRIORIDAD MÁXIMA - visible: false\n *    - La columna se oculta SIEMPRE, sin excepciones\n *    - Ni en modo creación ni en modo edición se muestra\n *\n * 2. OCULTACIÓN AUTOMÁTICA - hideIfAllValuesAreNull: true\n *    - La columna se oculta solo si TODOS los valores actuales están vacíos\n *    - Valores considerados vacíos: null, undefined, '', [], {}\n *    - Valores NO vacíos: 0, false (son valores válidos)\n *    - EXCEPCIONES IMPORTANTES:\n *      * En modo creación (creating=true) estas columnas SÍ se muestran\n *      * En modo edición de una fila específica se muestran SOLO si esa fila tiene valor\n *\n * 3. POR DEFECTO - Columnas normales\n *    - Se muestran siempre (visualización, creación y edición)\n *\n * VALORES POR DEFECTO EN CREACIÓN:\n * ===============================\n *\n * - Propiedad `default?: unknown` permite definir valores por defecto para modo creación\n * - Solo se aplica cuando se inicia el modo creación (startCreate)\n * - Puede ser cualquier tipo: string, number, boolean, etc.\n * - Si no se especifica `default`, se usa cadena vacía ('')\n *\n * VALORES DINÁMICOS EN CREACIÓN Y EDICIÓN:\n * =======================================\n *\n * - Input `inputValues$?: Observable<Partial<T>>` permite actualizar valores dinámicamente\n * - Solo se aplica durante modo creación (creating=true) o edición (editing!=null)\n * - Los valores se aplican automáticamente cuando el Observable emite\n * - Permite cambios múltiples durante el mismo modo de creación/edición\n * - Los nuevos valores sobrescriben los existentes (spread operator)\n *\n * TIPOS DE DATOS SOPORTADOS:\n * =========================\n *\n * - 'string': Texto normal (default)\n * - 'number': Números decimales (preserva decimales: 5.3 → 5.3)\n * - 'integer': Números enteros (convierte a entero: 5.3 → 5)\n * - 'boolean': Verdadero/Falso (checkbox)\n * - 'password': Texto oculto (input type=\"password\")\n * - 'enum': Lista de opciones predefinidas (select)\n *\n * EJEMPLOS:\n * - { accessor: 'id', visible: false } → NUNCA se muestra\n * - { accessor: 'motorPos', hideIfAllValuesAreNull: true, default: 0, type: 'integer' } → Entero (5.7 → 5)\n * - { accessor: 'weight', type: 'number', default: 2.5 } → Decimal preservado (5.7 → 5.7)\n * - { accessor: 'status', default: 'active' } → Se autorellena con 'active' en creación\n * - { accessor: 'name' } → Siempre visible, sin valor por defecto (cadena vacía)\n * - inputValues$ emite { motorPos: 5, weight: 2.3 } → actualiza inputs dinámicamente\n */@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'c80-table',\n  standalone: true,\n  imports: [C80IconComponent, C80ModalComponent],\n  templateUrl: './table.component.html',\n  styleUrl: './table.component.scss',\n})\nexport class C80TableComponent<T extends Record<string, unknown>> implements OnInit, OnDestroy {\n  private readonly modalService: ModalService = inject(ModalService);\n  @Input() data$!: Observable<T[]>;\n  @Input() columns: C80TableColDef[] = [];\n  @Input() inputValues$?: Observable<Partial<T>>; // Observable para actualizar valores de inputs dinámicamente en creación/edición\n  @Input() size = 0; // Tamaño de la tabla (0 = sin límite, > 0 aplica max-height)\n  @Input() multiple = true; // Permite selección múltiple por defecto\n  @Input({ transform: (value: boolean | string) => value === '' || value === true || value === 'true' })\n  noConfirm = false; // Si es true, no muestra confirmaciones modales\n  @Output() createAction = new EventEmitter<{ row: Partial<T>; done: (result: boolean) => void; }>();\n  @Output() updateAction = new EventEmitter<{ id: ID; changes: Partial<T>; done: (result: boolean) => void; }>();\n  @Output() deleteAction = new EventEmitter<{ id: ID; done: (result: boolean) => void; }>();\n  @Output() cancelAction = new EventEmitter<{ id: ID; done: (result: boolean) => void; }>();\n  @Output() viewAction = new EventEmitter<T>();\n  @Output() uploadAction = new EventEmitter<T>();\n  @Output() getRowButtonAction = new EventEmitter<T>();\n  @Output() moveUpAction = new EventEmitter<{ id: ID; done: (result: boolean) => void; }>();\n  @Output() moveDownAction = new EventEmitter<{ id: ID; done: (result: boolean) => void; }>();\n  @Output() enableAction = new EventEmitter<{ id: ID; enabled: boolean; done: (result: boolean) => void; }>();\n  @Output() searchTerm = new EventEmitter<string>();\n\n  @Output() errorEvent = new EventEmitter<string>();\n  @Output() selectable = new EventEmitter<T[]>();\n\n  readonly data = signal<T[]>([]);\n  readonly keys = signal<string[]>([]);\n  creating = signal(false);\n  newRow = signal<Partial<T> | null>(null);\n  readonly editing = signal<ID | null>(null); // id of row being edited\n  readonly editRow = signal<Partial<T> | null>(null);\n  readonly selectedItems = signal<Set<ID>>(new Set());\n  readonly selectAllChecked = signal(false);\n  readonly selectAllIndeterminate = signal(false);\n  readonly searchValue = signal<string>('');\n\n  // Flags to check if events have listeners\n  readonly hasCreateActionListener = signal(false);\n  readonly hasUpdateActionListener = signal(false);\n  readonly hasDeleteActionListener = signal(false);\n  readonly hasCancelActionListener = signal(false);\n  readonly hasViewActionListener = signal(false);\n  readonly hasUploadActionListener = signal(false);\n  readonly hasGetRowButtonListener = signal(false);\n  readonly hasMoveUpActionListener = signal(false);\n  readonly hasMoveDownActionListener = signal(false);\n  readonly hasEnableActionListener = signal(false);\n  readonly hasSearchTermListener = signal(false);\n  readonly hasSelectableListener = signal(false);\n\n  private dataSub?: Subscription;\n  private inputValuesSub?: Subscription;\n\n  private getErrorMessage(err: unknown): string {\n    return (err as any)?.message || 'Error al cargar datos';\n  }\n\n  /**\n   * Actualiza las keys de columnas visibles basándose en el estado actual\n   */\n  private updateVisibleKeys(): void {\n    const visibleColumns = this.columns.filter(col => this.isColumnVisibleInHeader(col));\n    this.keys.set(visibleColumns.map((col) => col.accessor));\n  }\n\n  /**\n   * Aplica valores parciales a los inputs en modo creación o edición.\n   * Solo actualiza si estamos en modo creación (creating = true) o edición (editing != null)\n   */\n  private applyInputValues(partialValues: Partial<T>): void {\n    // Solo aplicar valores si estamos en modo creación o edición\n    if (!this.creating() && this.editing() === null) {\n      return;\n    }\n\n    // Aplicar valores en modo creación\n    if (this.creating()) {\n      const currentRow = this.newRow();\n      if (currentRow) {\n        const updatedRow = { ...currentRow, ...partialValues };\n        this.newRow.set(updatedRow);\n      }\n    }\n\n    // Aplicar valores en modo edición\n    if (this.editing() !== null) {\n      const currentEditRow = this.editRow();\n      if (currentEditRow) {\n        const updatedEditRow = { ...currentEditRow, ...partialValues };\n        this.editRow.set(updatedEditRow);\n      }\n    }\n  }\n\n  ngOnInit() {\n    // Check if the outputs have listeners\n    this.hasCreateActionListener.set(this.createAction.observed);\n    this.hasUpdateActionListener.set(this.updateAction.observed);\n    this.hasDeleteActionListener.set(this.deleteAction.observed);\n    this.hasCancelActionListener.set(this.cancelAction.observed);\n    this.hasViewActionListener.set(this.viewAction.observed);\n    this.hasUploadActionListener.set(this.uploadAction.observed);\n    this.hasGetRowButtonListener.set(this.getRowButtonAction.observed);\n    this.hasMoveUpActionListener.set(this.moveUpAction.observed);\n    this.hasMoveDownActionListener.set(this.moveDownAction.observed);\n    this.hasEnableActionListener.set(this.enableAction.observed);\n    this.hasSearchTermListener.set(this.searchTerm.observed);\n    this.hasSelectableListener.set(this.selectable.observed);\n\n    if (!this.data$) return;\n    this.dataSub = this.data$.subscribe({\n      next: (items) => {\n        this.applySorting(items);\n        this.data.set(items);\n        // Actualizar las columnas visibles basándose en el estado actual\n        this.updateVisibleKeys();\n        // Mantener la selección existente después de actualizar los datos\n        this.preserveSelection();\n      },\n      error: (err: any) =>\n        this.errorEvent.emit(this.getErrorMessage(err)),\n    });\n\n    // Suscribirse a inputValues$ para actualizar valores dinámicamente en creación/edición\n    if (this.inputValues$) {\n      this.inputValuesSub = this.inputValues$.subscribe({\n        next: (partialValues) => {\n          this.applyInputValues(partialValues);\n        },\n        error: (err: any) =>\n          console.warn('Error en inputValues$:', this.getErrorMessage(err)),\n      });\n    }\n  }\n\n  ngOnDestroy() {\n    this.dataSub?.unsubscribe();\n    this.inputValuesSub?.unsubscribe();\n\n    // Close any open modal when component is destroyed\n    this.modalService.closeModal();\n  }\n\n  onInput(event: Event, key: string, col?: C80TableColDef) {\n    if (col?.type === 'boolean') {\n      const checked = (event.target as HTMLInputElement).checked;\n      this.updateNewRow(key, checked);\n    } else if (col?.type === 'enum') {\n      const value = (event.target as HTMLSelectElement).value;\n      this.updateNewRow(key, value);\n    } else {\n      const target = event.target as HTMLInputElement;\n      if (target && typeof target.value === 'string') {\n        this.updateNewRow(key, target.value);\n      }\n    }\n  }\n\n  async onDelete(row: T) {\n    const id = row['id'] as ID;\n\n    let confirmed = true;\n    if (!this.noConfirm) {\n      confirmed = await this.modalService.confirm(\n        'Confirmar eliminación',\n        '¿Está seguro de que desea eliminar este elemento? Esta acción no se puede deshacer.',\n        'Eliminar',\n        'Cancelar'\n      );\n    }\n\n    if (confirmed) {\n      this.deleteAction.emit({\n        id,\n        done: (success: boolean) => {\n          if (success) this.cancelEdit();\n        },\n      });\n    }\n  }\n\n  async onCancel(row: T) {\n    const id = row['id'] as ID;\n\n    let confirmed = true;\n    if (!this.noConfirm) {\n      confirmed = await this.modalService.confirm(\n        'Confirmar cancelación',\n        '¿Está seguro de que desea cancelar este elemento?',\n        'Cancelar elemento',\n        'No cancelar'\n      );\n    }\n\n    if (confirmed) {\n      this.cancelAction.emit({\n        id,\n        done: (success: boolean) => {\n          if (success) this.cancelEdit();\n        },\n      });\n    }\n  }\n\n  startCreate() {\n    this.creating.set(true);\n    // Actualizar keys de columnas visibles cuando entramos en modo creación\n    this.updateVisibleKeys();\n    // Inicializa newRow solo con columnas visibles y no readOnly\n    // En modo creación, mostramos columnas con hideIfAllValuesAreNull para permitir entrada de datos\n    const row: Partial<T> = {};\n    this.columns\n      .filter((col) => this.isColumnVisible(col, true) && !col.readOnly)\n      .forEach((col) => {\n        // Si la columna tiene un valor por defecto, usarlo; si no, usar cadena vacía\n        const defaultValue = col.default !== undefined ? col.default : '';\n        (row as Record<string, unknown>)[col.accessor] = defaultValue;\n      });\n    this.newRow.set(row);\n  }\n\n  cancelCreate() {\n    this.creating.set(false);\n    this.newRow.set(null);\n    // Actualizar keys de columnas visibles cuando salimos del modo creación\n    this.updateVisibleKeys();\n  }\n\n  updateNewRow(key: string, value: unknown) {\n    const current = this.newRow();\n    if (!current) return;\n    this.newRow.set({ ...current, [key]: value });\n  }\n\n  saveCreate() {\n    const row = this.newRow();\n    if (!row) return;\n\n    // Validar campos requeridos antes de crear - solo columnas visibles y no readOnly\n    // En modo creación, incluimos columnas con hideIfAllValuesAreNull\n    const visibleColumns = this.columns.filter((col) => this.isColumnVisible(col, true) && !col.readOnly);\n    const converted = visibleColumns.reduce((acc, col) => {\n      acc[col.accessor] = this.convertCellValue(row[col.accessor], col);\n      return acc;\n    }, {} as Record<string, unknown>) as Partial<T>;\n\n    this.createAction.emit({\n      row: converted,\n      done: (success: boolean) => {\n        if (success) {\n          this.cancelCreate();\n        }\n      },\n    });\n  }\n\n  /**\n   * Converts a cell value based on column type or sample data.\n   * Handles stringification of objects for string columns.\n   */\n  /**\n   * Converts a cell value based on column type or sample data.\n   * Delegates to type-specific helpers for clarity and maintainability.\n   */\n  private convertCellValue(value: unknown, col: C80TableColDef): unknown {\n    if (col.type === 'boolean') return this.toBoolean(value);\n    if (col.type === 'number') return this.toNumber(value);\n    if (col.type === 'integer') return this.toInteger(value);\n    if (col.type === 'string' || col.type === 'password')\n      return this.toStringValue(value);\n\n    // Fallback: use sample data if available\n    const sample = this.data()[0];\n    if (sample) {\n      const sampleValue = this.getCellValue(sample, col.accessor);\n      if (typeof sampleValue === 'boolean') return this.toBoolean(value);\n      if (typeof sampleValue === 'number') return this.toNumber(value);\n      if (typeof sampleValue === 'string') return this.toStringValue(value);\n    }\n    return value;\n  }\n\n  /** Converts value to boolean using best practices. */\n  private toBoolean(value: unknown): boolean {\n    if (typeof value === 'boolean') return value;\n    if (typeof value === 'string')\n      return value.trim().toLowerCase() === 'true' || value.trim() === '1';\n    if (typeof value === 'number') return value === 1;\n    return false;\n  }\n\n  /** Converts value to number using best practices. */\n  private toNumber(value: unknown): number | undefined {\n    if (typeof value === 'number') return value;\n    if (typeof value === 'string') {\n      const trimmed = value.trim();\n      if (trimmed === '') return undefined;\n      const num = Number(trimmed);\n      return isNaN(num) ? undefined : num;\n    }\n    if (typeof value === 'boolean') return value ? 1 : 0;\n    return undefined;\n  }\n\n  /** Converts value to integer using best practices. */\n  private toInteger(value: unknown): number | undefined {\n    if (typeof value === 'number') return Math.floor(value);\n    if (typeof value === 'string') {\n      const trimmed = value.trim();\n      if (trimmed === '') return undefined;\n      const num = Number(trimmed);\n      return isNaN(num) ? undefined : Math.floor(num);\n    }\n    if (typeof value === 'boolean') return value ? 1 : 0;\n    return undefined;\n  }\n\n  /** Converts value to string using best practices, always stringifies objects. */\n  private toStringValue(value: unknown): string {\n    if (value == null) return '';\n    if (typeof value === 'string') return value;\n    if (typeof value === 'number' || typeof value === 'boolean')\n      return String(value);\n    if (typeof value === 'object') {\n      try {\n        return JSON.stringify(value);\n      } catch {\n        return '[object Object]';\n      }\n    }\n    // For functions, symbols, undefined, etc., return empty string\n    return '';\n  }\n\n  onEdit(row: T) {\n    this.editing.set(row['id'] as ID);\n    // Actualizar keys de columnas visibles cuando entramos en modo edición\n    this.updateVisibleKeys();\n    const edit: Partial<T> = {};\n    this.columns\n      .filter((col) => this.isColumnVisible(col, true) && !col.readOnly)\n      .forEach((col) => {\n        const value = this.getCellValue(row, col.accessor);\n        (edit as Record<string, unknown>)[col.accessor] = value;\n      });\n    this.editRow.set(edit);\n  }\n\n  cancelEdit() {\n    this.editing.set(null);\n    this.editRow.set(null);\n    // Actualizar keys de columnas visibles cuando salimos del modo edición\n    this.updateVisibleKeys();\n  }\n\n  onEditInput(event: Event, key: string, col?: C80TableColDef) {\n    const current = this.editRow();\n    if (!current) return;\n    if (col?.type === 'boolean') {\n      const checked = (event.target as HTMLInputElement).checked;\n      this.editRow.set({ ...current, [key]: checked });\n    } else if (col?.type === 'enum') {\n      const value = (event.target as HTMLSelectElement).value;\n      this.editRow.set({ ...current, [key]: value });\n    } else {\n      const target = event.target as HTMLInputElement;\n      if (target && typeof target.value === 'string') {\n        this.editRow.set({ ...current, [key]: target.value });\n      }\n    }\n  }\n\n  saveEdit(row: T) {\n    const id = row['id'];\n    if ((typeof id !== 'string' && typeof id !== 'number') || !this.editRow())\n      return;\n    const visibleColumns = this.columns.filter(\n      (col) => this.isColumnVisible(col, true) && !col.readOnly\n    );\n    const converted = visibleColumns.reduce((acc, col) => {\n      acc[col.accessor] = this.convertCellValue(\n        this.editRow()![col.accessor],\n        col\n      );\n      return acc;\n    }, {} as Record<string, unknown>) as Partial<T>;\n\n    this.updateAction.emit({\n      id,\n      changes: converted,\n      done: (success: boolean) => {\n        if (success) {\n          this.cancelEdit();\n        }\n      },\n    });\n  }\n\n  /**\n   * TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).\n   */\n  trackById(index: number, row: T): number | string {\n    const id =\n      row && typeof row === 'object' && 'id' in row\n        ? (row as Record<string, unknown>)['id']\n        : undefined;\n    return typeof id === 'string' || typeof id === 'number' ? id : index;\n  }\n\n  /**\n   * Emits the view event with the entire row data\n   */\n  onView(row: T): void {\n    this.viewAction.emit(row);\n  }\n\n  /**\n   * Emits the getRowButtonAction event with the entire row data\n   */\n  onGetRowButton(row: T): void {\n    this.getRowButtonAction.emit(row);\n  }\n\n  /**\n   * Emits the moveUpAction event with the row id\n   */\n  async onMoveUp(row: T) {\n    const id = row['id'] as ID;\n    this.moveUpAction.emit({\n      id,\n      done: (success: boolean) => {\n        if (success) this.cancelEdit();\n      },\n    });\n  }\n\n  /**\n   * Emits the moveDownAction event with the row id\n   */\n  async onMoveDown(row: T) {\n    const id = row['id'] as ID;\n    this.moveDownAction.emit({\n      id,\n      done: (success: boolean) => {\n        if (success) this.cancelEdit();\n      },\n    });\n  }\n\n  /**\n   * Emits the enableAction event with the row id and current enabled status\n   */\n  async onEnable(row: T) {\n    const id = row['id'] as ID;\n    const currentEnabled = row['enabled'] as boolean;\n    this.enableAction.emit({\n      id,\n      enabled: !currentEnabled, // Toggle the current state\n      done: (success: boolean) => {\n        if (success) this.cancelEdit();\n      },\n    });\n  }\n\n  /**\n   * Handles search input changes with debouncing\n   */\n  onSearchInput(event: Event): void {\n    const target = event.target as HTMLInputElement;\n    const value = target.value;\n    this.searchValue.set(value);\n    this.searchTerm.emit(value);\n  }\n\n  /**\n   * Clears the search input\n   */\n  clearSearch(): void {\n    this.searchValue.set('');\n    this.searchTerm.emit('');\n  }\n\n  /**\n   * Gets the value from an object using dot notation (e.g., 'task.name')\n   */\n  private getNestedValue(obj: any, accessor: string): unknown {\n    return accessor.split('.').reduce((current, key) => {\n      return current && typeof current === 'object' ? current[key] : undefined;\n    }, obj);\n  }\n\n  /**\n   * Gets the value for a cell using the accessor, supporting dot notation for nested properties\n   */\n  getCellValue(row: T, accessor: string): unknown {\n    if (accessor.includes('.')) {\n      return this.getNestedValue(row, accessor);\n    }\n    return (row as Record<string, unknown>)[accessor];\n  }\n\n  /**\n   * Returns the display value for a cell, showing '-' for falsy values except 0, false, and empty objects/arrays\n   */\n  getDisplayValue(value: unknown): string {\n    // If value is 0, show it\n    if (value === 0) {\n      return '0';\n    }\n    // If value is false, it should be handled by boolean logic in template, not here\n    if (value === false) {\n      return 'false';\n    }\n    // Handle objects and arrays (including empty ones)\n    if (typeof value === 'object' && value !== null) {\n      try {\n        return JSON.stringify(value);\n      } catch {\n        return '[object Object]';\n      }\n    }\n    // If value is other falsy values (null, undefined, ''), show '-'\n    if (!value) {\n      return '-';\n    }\n    // Handle other types explicitly\n    if (\n      typeof value === 'string' ||\n      typeof value === 'number' ||\n      typeof value === 'boolean'\n    ) {\n      return String(value);\n    }\n    // For any other type (function, symbol, etc.), return a safe fallback\n    return '-';\n  }\n\n  /**\n   * Gets the display text for an enum value\n   */\n  getEnumDisplayValue(value: unknown, col: C80TableColDef): string {\n    if (!col.enum || (value !== 0 && !value)) {\n      return '-';\n    }\n    const displayValue = col.enum[value as string | number];\n    return (\n      displayValue ||\n      (typeof value === 'string' || typeof value === 'number'\n        ? String(value)\n        : '-')\n    );\n  }\n\n  /**\n   * Gets the enum options as an array for select dropdowns\n   */\n  getEnumOptions(\n    col: C80TableColDef\n  ): { value: string | number; label: string }[] {\n    if (!col.enum) return [];\n    return Object.entries(col.enum).map(([value, label]) => ({\n      value: isNaN(Number(value)) ? value : Number(value),\n      label,\n    }));\n  }\n\n  /**\n   * Gets the CSS color for a cell value based on the column's color configuration\n   */\n  getCellColor(value: unknown, col: C80TableColDef): string | undefined {\n    if (!col.color || (value !== 0 && !value)) {\n      return undefined;\n    }\n    return col.color[value as string | number];\n  }\n\n  /**\n   * Calcula el max-height de la tabla basado en el tamaño\n   * Si size es 0, retorna undefined (sin límite de altura)\n   */\n  getTableMaxHeight(): string | undefined {\n    if (this.size <= 0) {\n      return undefined; // Sin límite de altura\n    }\n    // Altura base de 400px * size\n    const baseHeight = 400;\n    const maxHeight = Math.round(baseHeight * this.size);\n    return `${maxHeight}px`;\n  }\n\n  /**\n   * Selection methods\n   */\n  clearSelection(): void {\n    this.selectedItems.set(new Set());\n    this.updateSelectAllState();\n    this.emitSelection();\n  }\n\n  /**\n   * Mantiene la selección existente después de actualizar los datos,\n   * eliminando solo los IDs que ya no existen en los nuevos datos\n   */\n  private preserveSelection(): void {\n    const currentSelection = this.selectedItems();\n    const currentData = this.data();\n\n    if (currentSelection.size === 0 || currentData.length === 0) {\n      // Si no hay selección o no hay datos, limpiar la selección\n      this.clearSelection();\n      return;\n    }\n\n    // Obtener los IDs disponibles en los nuevos datos\n    const availableIds = new Set(currentData.map(item => item['id'] as ID));\n\n    // Filtrar la selección actual para mantener solo los IDs que aún existen\n    const preservedSelection = new Set(\n      Array.from(currentSelection).filter(id => availableIds.has(id))\n    );\n\n    // Actualizar la selección con los IDs preservados\n    this.selectedItems.set(preservedSelection);\n    this.updateSelectAllState();\n    this.emitSelection();\n  }\n\n  toggleSelectAll(): void {\n    const currentSelection = this.selectedItems();\n    const allItems = this.data();\n\n    if (currentSelection.size === allItems.length) {\n      // Deselect all\n      this.clearSelection();\n    } else {\n      // Select all\n      const allIds = new Set(allItems.map(item => item['id'] as ID));\n      this.selectedItems.set(allIds);\n      this.updateSelectAllState();\n      this.emitSelection();\n    }\n  }\n\n  toggleItemSelection(item: T): void {\n    const id = item['id'] as ID;\n    const currentSelection = new Set(this.selectedItems());\n\n    if (!this.multiple) {\n      // Selección simple: solo permitir un elemento seleccionado\n      if (currentSelection.has(id)) {\n        // Deseleccionar el elemento actual\n        currentSelection.clear();\n      } else {\n        // Seleccionar solo este elemento\n        currentSelection.clear();\n        currentSelection.add(id);\n      }\n    } else if (currentSelection.has(id)) {\n      // Selección múltiple: deseleccionar elemento existente\n      currentSelection.delete(id);\n    } else {\n      // Selección múltiple: agregar nuevo elemento\n      currentSelection.add(id);\n    }\n\n    this.selectedItems.set(currentSelection);\n    this.updateSelectAllState();\n    this.emitSelection();\n  }\n\n  isItemSelected(item: T): boolean {\n    const id = item['id'] as ID;\n    return this.selectedItems().has(id);\n  }\n\n  private updateSelectAllState(): void {\n    const selectedCount = this.selectedItems().size;\n    const totalCount = this.data().length;\n\n    if (selectedCount === 0) {\n      this.selectAllChecked.set(false);\n      this.selectAllIndeterminate.set(false);\n    } else if (selectedCount === totalCount) {\n      this.selectAllChecked.set(true);\n      this.selectAllIndeterminate.set(false);\n    } else {\n      this.selectAllChecked.set(false);\n      this.selectAllIndeterminate.set(true);\n    }\n  }\n\n  private emitSelection(): void {\n    const selectedIds = this.selectedItems();\n    const selectedItems = this.data().filter(item =>\n      selectedIds.has(item['id'] as ID)\n    );\n    this.selectable.emit(selectedItems);\n  }\n\n  /**\n   * Verifica si un valor individual está vacío/nulo\n   */\n  private isValueEmpty(value: unknown): boolean {\n    // Considerar vacío: null, undefined, ''\n    if (value === null || value === undefined || value === '') {\n      return true;\n    }\n\n    // Para números, 0 se considera como valor válido, no vacío\n    if (typeof value === 'number') {\n      return false;\n    }\n\n    // Para booleanos, false se considera como valor válido, no vacío\n    if (typeof value === 'boolean') {\n      return false;\n    }\n\n    // Para arrays vacíos\n    if (Array.isArray(value) && value.length === 0) {\n      return true;\n    }\n\n    // Para objetos vacíos\n    if (typeof value === 'object' && value !== null && Object.keys(value).length === 0) {\n      return true;\n    }\n\n    return false;\n  }\n\n  /**\n   * Verifica si todos los valores de una columna están vacíos/nulos\n   */\n  private areAllColumnValuesEmpty(column: C80TableColDef): boolean {\n    const currentData = this.data();\n\n    if (currentData.length === 0) {\n      return true; // Si no hay datos, consideramos la columna como vacía\n    }\n\n    return currentData.every(row => {\n      const value = this.getCellValue(row, column.accessor);\n      return this.isValueEmpty(value);\n    });\n  }\n\n  /**\n   * Determina si una columna debe ser visible basándose en su configuración y datos.\n   *\n   * Orden de prioridad:\n   * 1. Si visible === false: SIEMPRE se oculta (máxima prioridad)\n   * 2. Si hideIfAllValuesAreNull === true Y todos los valores están vacíos: se oculta\n   *    EXCEPCIÓN: En modo creación (creating === true), estas columnas se muestran para permitir entrada de datos\n   * 3. Por defecto: se muestra\n   *\n   * @param column - La definición de la columna\n   * @param forceShowInCreation - Si es true, ignora hideIfAllValuesAreNull (usado en modo creación)\n   */\n  isColumnVisible(column: C80TableColDef, forceShowInCreation = false): boolean {\n    // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar\n    if (column.visible === false) {\n      return false;\n    }\n\n    // PRIORIDAD 2: Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar\n    // EXCEPCIÓN: En modo creación, mostramos estas columnas para permitir entrada de datos\n    if (column.hideIfAllValuesAreNull === true && !forceShowInCreation && this.areAllColumnValuesEmpty(column)) {\n      return false;\n    }\n\n    // Por defecto: mostrar\n    return true;\n  }\n\n  /**\n   * Determina si una columna debe ser visible en los headers.\n   * Los headers se muestran si la columna es visible en cualquiera de estos casos:\n   * - Visualización normal\n   * - Modo creación\n   * - Hay una fila en modo edición y esa fila tiene valor en la columna\n   *\n   * @param column - La definición de la columna\n   */\n  isColumnVisibleInHeader(column: C80TableColDef): boolean {\n    // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar\n    if (column.visible === false) {\n      return false;\n    }\n\n    // Si estamos en modo creación, mostrar columnas con hideIfAllValuesAreNull\n    if (this.creating() && column.hideIfAllValuesAreNull === true) {\n      return true;\n    }\n\n    // Si hay una fila en modo edición y la columna tiene hideIfAllValuesAreNull\n    if (this.editing() !== null && column.hideIfAllValuesAreNull === true) {\n      // Buscar la fila que se está editando\n      const editingRowId = this.editing();\n      const editingRow = this.data().find(row => row['id'] === editingRowId);\n\n      if (editingRow) {\n        // Solo mostrar el header si la fila en edición tiene valor en esta columna\n        const cellValue = this.getCellValue(editingRow, column.accessor);\n        return !this.isValueEmpty(cellValue);\n      }\n\n      // Si no se encuentra la fila, no mostrar el header\n      return false;\n    }\n\n    // Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar\n    if (column.hideIfAllValuesAreNull === true && this.areAllColumnValuesEmpty(column)) {\n      return false;\n    }\n\n    // Por defecto: mostrar\n    return true;\n  }\n\n  /**\n   * Determina si una columna debe ser visible en una fila específica considerando el modo edición.\n   * En modo edición, las columnas con hideIfAllValuesAreNull se muestran solo si tienen valor.\n   *\n   * @param column - La definición de la columna\n   * @param row - La fila actual\n   */\n  isColumnVisibleForRow(column: C80TableColDef, row: T): boolean {\n    // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar\n    if (column.visible === false) {\n      return false;\n    }\n\n    // PRIORIDAD 2: Si esta fila específica está en modo edición y la columna tiene hideIfAllValuesAreNull\n    const isEditingThisRow = this.editing() === row['id'];\n    if (isEditingThisRow && column.hideIfAllValuesAreNull === true) {\n      // En edición, solo mostrar si esta fila específica tiene valor en esta columna\n      const cellValue = this.getCellValue(row, column.accessor);\n      return !this.isValueEmpty(cellValue);\n    }\n\n    // PRIORIDAD 3: Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar\n    if (column.hideIfAllValuesAreNull === true && this.areAllColumnValuesEmpty(column)) {\n      return false;\n    }\n\n    // Por defecto: mostrar\n    return true;\n  }\n\n  private applySorting(items: T[]): void {\n    const orderedColumns = this.columns.filter((col) => col.order);\n    if (orderedColumns.length > 0) {\n      orderedColumns.forEach((col) => {\n        items.sort((a, b) => {\n          const valueA = this.getCellValue(a, col.accessor) as string | number;\n          const valueB = this.getCellValue(b, col.accessor) as string | number;\n\n          if (col.order === 'ASC') {\n            if (valueA > valueB) return 1;\n            if (valueA < valueB) return -1;\n            return 0;\n          } else {\n            if (valueA < valueB) return 1;\n            if (valueA > valueB) return -1;\n            return 0;\n          }\n        });\n      });\n    }\n  }\n\n  onUpload(row: T): void {\n    this.uploadAction.emit(row);\n  }\n}\n","<div class=\"table-responsive\" [style.max-height]=\"getTableMaxHeight()\" [style.overflow-y]=\"size > 0 ? 'auto' : 'visible'\">\n  <!-- Search Bar -->\n  @if (hasSearchTermListener()) {\n  <div class=\"search-container\">\n    <div class=\"search-input-wrapper\">\n      <div class=\"input-group\">\n        <span class=\"input-group-text\">\n          <c80-icon icon=\"search\" [size]=\".8\"></c80-icon>\n        </span>\n        <input type=\"text\" class=\"form-control search-input\" placeholder=\"Buscar...\" [value]=\"searchValue()\" (input)=\"onSearchInput($event)\" aria-label=\"Buscar en la tabla\" />\n        @if (searchValue()) {\n        <button class=\"btn btn-outline-secondary btn-borrar\" type=\"button\" (click)=\"clearSearch()\" title=\"Limpiar búsqueda\">\n          <c80-icon icon=\"cancel\" [size]=\".7\"></c80-icon>\n        </button>\n        }\n      </div>\n    </div>\n  </div>\n  }\n\n  <table class=\"table table-bordered table-hover align-middle\">\n    <thead class=\"thead table-light sticky-header\">\n      <tr>\n        @if (hasSelectableListener() && data().length !== 0) {\n        <th class=\"text-center selection-column\">\n          @if (multiple) {\n          <input type=\"checkbox\" [checked]=\"selectAllChecked()\" [indeterminate]=\"selectAllIndeterminate()\" (change)=\"toggleSelectAll()\" aria-label=\"Seleccionar todo\" />\n          }\n        </th>\n        }\n        @for (col of columns; track col) {\n        @if (isColumnVisibleInHeader(col)) {\n        @if (col.type === 'boolean') {\n        <th class=\"text-center boolean-column\">{{ col.label }}</th>\n        }\n        @else if (col.type === 'number') {\n        <th class=\"text-center number-column\">{{ col.label }}</th>\n        }\n        @else {\n        <th>{{ col.label }}</th>\n        }\n        }\n        }\n        @if (hasCreateActionListener() || hasUpdateActionListener() || hasDeleteActionListener() ||\n        hasCancelActionListener() || hasViewActionListener() || hasUploadActionListener() || hasGetRowButtonListener() ||\n        hasMoveUpActionListener() || hasMoveDownActionListener()) {\n        <th class=\"table-actions-header\">\n          <div class=\"actions-wrapper\">\n            <span>Actions</span>\n            @if (hasCreateActionListener()) {\n            <c80-icon button icon=\"add\" [disabled]=\"creating()\" title=\"Agregar\" [size]=\".6\" (iconClick)=\"startCreate()\"></c80-icon>\n            }\n          </div>\n        </th>\n        }\n      </tr>\n    </thead>\n    <tbody>\n      @for (row of data(); track trackById(i, row); let i = $index) {\n      <tr>\n        @if (hasSelectableListener() && data().length !== 0) {\n        <td class=\"text-center selection-column\">\n          <input type=\"checkbox\" [checked]=\"isItemSelected(row)\" (change)=\"toggleItemSelection(row)\" [attr.aria-label]=\"'Seleccionar fila ' + (i + 1)\" />\n        </td>\n        }\n        @for (col of columns; track col) {\n        @if (isColumnVisibleForRow(col, row)) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center boolean-column\">\n          @if (editing() === row['id'] && !col.readOnly) {\n          <input type=\"checkbox\" [checked]=\"!!editRow()?.[col.accessor]\" (change)=\"onEditInput($event, col.accessor, col)\" [attr.aria-label]=\"col.label\" />\n          }\n          @else {\n          @if (getCellValue(row, col.accessor) === true) {\n          <c80-icon icon=\"check\" [size]=\".7\"></c80-icon>\n          <br />\n          }\n          @else if (getCellValue(row, col.accessor) === false) {\n          <c80-icon icon=\"cancel\" [size]=\".7\"></c80-icon>\n          <br />\n          }\n          }\n        </td>\n        }\n        @else if (col.type === 'number') {\n        <td class=\"text-center number-column\">\n          @if (editing() === row['id'] && !col.readOnly) {\n          <input class=\"form-control form-control-sm\" type=\"number\" [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\" (input)=\"onEditInput($event, col.accessor, col)\" />\n          }\n          @else {\n          <span [style.color]=\"getCellColor(getCellValue(row, col.accessor), col)\">{{ getDisplayValue(getCellValue(row,\n            col.accessor)) }}</span>\n          }\n        </td>\n        }\n        @else {\n        <td>\n          @if (editing() === row['id'] && !col.readOnly) {\n          @if (col.type === 'enum') {\n          <select class=\"form-control form-control-sm\" [value]=\"editRow()?.[col.accessor] ?? ''\" (change)=\"onEditInput($event, col.accessor, col)\">\n            <option value=\"\">{{ col.label }}</option>\n            @for (option of getEnumOptions(col); track option.value) {\n            <option [value]=\"option.value\">{{ option.label }}</option>\n            }\n          </select>\n          }\n          @else {\n          <input class=\"form-control form-control-sm\" [type]=\"col.type === 'password' ? 'password' : 'text'\" [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\" (input)=\"onEditInput($event, col.accessor, col)\" />\n          }\n          }\n          @else {\n          @if (col.type === 'password') {\n          <span [style.color]=\"getCellColor(getCellValue(row, col.accessor), col)\">******</span>\n          }\n          @else if (col.type === 'enum') {\n          <span [style.color]=\"getCellColor(getCellValue(row, col.accessor), col)\">{{\n            getEnumDisplayValue(getCellValue(row, col.accessor), col)\n            }}</span>\n          }\n          @else if (getCellValue(row, col.accessor) === true) {\n          <c80-icon icon=\"check\" [size]=\".7\"></c80-icon>\n          }\n          @else if (getCellValue(row, col.accessor) === false) {\n          <c80-icon icon=\"cancel\" [size]=\".7\"></c80-icon>\n          }\n          @else {\n          <span [style.color]=\"getCellColor(getCellValue(row, col.accessor), col)\">{{ getDisplayValue(getCellValue(row,\n            col.accessor)) }}</span>\n          }\n          }\n        </td>\n        }\n        }\n        }\n        @if (hasCreateActionListener() || hasUpdateActionListener() || hasDeleteActionListener() ||\n        hasCancelActionListener() || hasViewActionListener() || hasUploadActionListener() || hasGetRowButtonListener() ||\n        hasMoveUpActionListener() || hasMoveDownActionListener()) {\n        <td class=\"text-center actions-cell\">\n          <div class=\"actions-container\">\n            @if (editing() === row['id']) {\n            <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveEdit(row)\" [size]=\".7\"></c80-icon>\n            <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelEdit()\" [size]=\".7\"></c80-icon>\n            }\n            @else {\n            @if (hasUpdateActionListener()) {\n            <c80-icon button icon=\"edit\" title=\"Editar\" (iconClick)=\"onEdit(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasCancelActionListener()) {\n            <c80-icon button icon=\"cancel_circle\" color=\"secondary\" title=\"Cancelar\" (iconClick)=\"onCancel(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasViewActionListener()) {\n            <c80-icon button icon=\"view\" color=\"primary\" title=\"Ver\" (iconClick)=\"onView(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasUploadActionListener()) {\n            <c80-icon button icon=\"upload\" color=\"primary\" title=\"Subir\" (iconClick)=\"onUpload(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasGetRowButtonListener()) {\n            <c80-icon button icon=\"get\" color=\"success\" title=\"Vincular\" (iconClick)=\"onGetRowButton(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasMoveUpActionListener()) {\n            <c80-icon button icon=\"arrow_up\" color=\"secondary\" title=\"Mover arriba\" (iconClick)=\"onMoveUp(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasMoveDownActionListener()) {\n            <c80-icon button icon=\"arrow_down\" color=\"secondary\" title=\"Mover abajo\" (iconClick)=\"onMoveDown(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasEnableActionListener()) {\n            <c80-icon button [icon]=\"getCellValue(row, 'enabled') ? 'toggle_on' : 'toggle_off'\" [color]=\"getCellValue(row, 'enabled') ? 'success' : 'secondary'\" [title]=\"getCellValue(row, 'enabled') ? 'Deshabilitar' : 'Habilitar'\"\n              (iconClick)=\"onEnable(row)\" [size]=\".7\">\n            </c80-icon>\n            }\n            @if (hasDeleteActionListener()) {\n            <c80-icon button icon=\"delete\" color=\"warn\" title=\"Borrar\" (iconClick)=\"onDelete(row)\" [size]=\".7\"></c80-icon>\n            }\n            }\n          </div>\n        </td>\n        }\n      </tr>\n      }\n      @if (creating() && hasCreateActionListener()) {\n      <tr>\n        @if (hasSelectableListener() && data().length !== 0) {\n        <td class=\"text-center selection-column\">\n          <!-- Empty cell for alignment -->\n        </td>\n        }\n        @for (col of columns; track col) {\n        @if (isColumnVisible(col, true)) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center\">\n          @if (!col.readOnly) {\n          <input type=\"checkbox\" [checked]=\"!!newRow()?.[col.accessor]\" (change)=\"onInput($event, col.accessor, col)\" [attr.aria-label]=\"col.label\" />\n          }\n          @else {\n          <!-- ReadOnly boolean column in create mode shows empty -->\n          <span class=\"text-muted\">-</span>\n          }\n        </td>\n        } @else if (col.type === 'number') {\n        <td class=\"text-center number-column\">\n          @if (!col.readOnly) {\n          <input class=\"form-control form-control-sm\" type=\"number\" [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\" (input)=\"onInput($event, col.accessor, col)\" />\n          }\n          @else {\n          <!-- ReadOnly number column in create mode shows empty -->\n          <span class=\"text-muted\">-</span>\n          }\n        </td>\n        } @else {\n        <td>\n          @if (!col.readOnly) {\n          @if (col.type === 'enum') {\n          <select class=\"form-control form-control-sm\" [value]=\"newRow()?.[col.accessor] ?? ''\" (change)=\"onInput($event, col.accessor, col)\">\n            <option value=\"\">{{ col.label }}</option>\n            @for (option of getEnumOptions(col); track option.value) {\n            <option [value]=\"option.value\">{{ option.label }}</option>\n            }\n          </select>\n          }\n          @else {\n          <input class=\"form-control form-control-sm\" type=\"text\" [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\" (input)=\"onInput($event, col.accessor, col)\" />\n          }\n          }\n          @else {\n          <!-- ReadOnly column in create mode shows empty -->\n          <span class=\"text-muted\">-</span>\n          }\n        </td>\n        }\n        }\n        }\n        <td class=\"text-center actions-cell\">\n          <div class=\"actions-container\">\n            <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveCreate()\" [size]=\".7\"></c80-icon>\n            <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelCreate()\" [size]=\".7\"></c80-icon>\n          </div>\n        </td>\n      </tr>\n      }\n    </tbody>\n  </table>\n  @if (data().length === 0 && !creating()) {\n  <div class=\"text-center text-muted py-3 small\">\n    No hay datos para mostrar.\n  </div>\n  }\n</div>\n\n<c80-modal></c80-modal>"]}
|
|
459
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table.component.ts","../../../../../libs/ui/src/lib/table/table.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  signal,\n  EventEmitter,\n  OnInit,\n  OnDestroy,\n  inject,\n} from '@angular/core';\nimport { Observable, Subscription } from 'rxjs';\nimport { C80IconComponent } from '../icon/icon.component';\nimport { C80ModalComponent, ModalService } from '../modal';\nimport { TableColumnVisibilityService } from './table-column-visibility.service';\nimport { TableDataUtilsService } from './table-data-utils.service';\nimport { TableSelectionService } from './table-selection.service';\nimport { TableCrudStateService } from './table-crud-state.service';\nimport { C80TableColDef, ID } from './table.types';\n\n/**\n * C80TableComponent - Componente de tabla avanzado con funcionalidades CRUD\n *\n * COMPORTAMIENTO DE VISIBILIDAD DE COLUMNAS:\n * ========================================\n *\n * 1. PRIORIDAD MÁXIMA - visible: false\n *    - La columna se oculta SIEMPRE, sin excepciones\n *    - Ni en modo creación ni en modo edición se muestra\n *\n * 2. OCULTACIÓN AUTOMÁTICA - hideIfAllValuesAreNull: true\n *    - La columna se oculta solo si TODOS los valores actuales están vacíos\n *    - Valores considerados vacíos: null, undefined, '', [], {}\n *    - Valores NO vacíos: 0, false (son valores válidos)\n *    - EXCEPCIONES IMPORTANTES:\n *      * En modo creación (creating=true) estas columnas SÍ se muestran\n *      * En modo edición de una fila específica se muestran SOLO si esa fila tiene valor\n *\n * 3. POR DEFECTO - Columnas normales\n *    - Se muestran siempre (visualización, creación y edición)\n *\n * VALORES POR DEFECTO EN CREACIÓN:\n * ===============================\n *\n * - Propiedad `default?: unknown` permite definir valores por defecto para modo creación\n * - Solo se aplica cuando se inicia el modo creación (startCreate)\n * - Puede ser cualquier tipo: string, number, boolean, etc.\n * - Si no se especifica `default`, se usa cadena vacía ('')\n *\n * VALORES DINÁMICOS EN CREACIÓN Y EDICIÓN:\n * =======================================\n *\n * - Input `inputValues$?: Observable<Partial<T>>` permite actualizar valores dinámicamente\n * - Solo se aplica durante modo creación (creating=true) o edición (editing!=null)\n * - Los valores se aplican automáticamente cuando el Observable emite\n * - Permite cambios múltiples durante el mismo modo de creación/edición\n * - Los nuevos valores sobrescriben los existentes (spread operator)\n *\n * TIPOS DE DATOS SOPORTADOS:\n * =========================\n *\n * - 'string': Texto normal (default)\n * - 'number': Números decimales (preserva decimales: 5.3 → 5.3)\n * - 'integer': Números enteros (convierte a entero: 5.3 → 5)\n * - 'boolean': Verdadero/Falso (checkbox)\n * - 'password': Texto oculto (input type=\"password\")\n * - 'enum': Lista de opciones predefinidas (select)\n *\n * EJEMPLOS:\n * - { accessor: 'id', visible: false } → NUNCA se muestra\n * - { accessor: 'motorPos', hideIfAllValuesAreNull: true, default: 0, type: 'integer' } → Entero (5.7 → 5)\n * - { accessor: 'weight', type: 'number', default: 2.5 } → Decimal preservado (5.7 → 5.7)\n * - { accessor: 'status', default: 'active' } → Se autorellena con 'active' en creación\n * - { accessor: 'name' } → Siempre visible, sin valor por defecto (cadena vacía)\n * - inputValues$ emite { motorPos: 5, weight: 2.3 } → actualiza inputs dinámicamente\n */@Component({\n  // eslint-disable-next-line @angular-eslint/component-selector\n  selector: 'c80-table',\n  standalone: true,\n  imports: [C80IconComponent, C80ModalComponent],\n  templateUrl: './table.component.html',\n  styleUrl: './table.component.scss',\n})\nexport class C80TableComponent<T extends Record<string, unknown>> implements OnInit, OnDestroy {\n  // Servicios inyectados\n  private readonly modalService = inject(ModalService);\n  private readonly visibilityService = inject(TableColumnVisibilityService);\n  private readonly dataUtils = inject(TableDataUtilsService);\n  private readonly selectionService = inject(TableSelectionService);\n  private readonly crudService = inject(TableCrudStateService);\n\n  // Inputs\n  @Input() data$!: Observable<T[]>;\n  @Input() columns: C80TableColDef[] = [];\n  @Input() inputValues$?: Observable<Partial<T>>; // Observable para actualizar valores de inputs dinámicamente en creación/edición\n  @Input() size = 0; // Tamaño de la tabla (0 = sin límite, > 0 aplica max-height)\n  @Input() multiple = true; // Permite selección múltiple por defecto\n  @Input({ transform: (value: boolean | string) => value === '' || value === true || value === 'true' })\n  noConfirm = false; // Si es true, no muestra confirmaciones modales\n\n  // Outputs\n  @Output() createAction = new EventEmitter<{ row: Partial<T>; done: (result: boolean) => void; }>();\n  @Output() updateAction = new EventEmitter<{ id: ID; changes: Partial<T>; done: (result: boolean) => void; }>();\n  @Output() deleteAction = new EventEmitter<{ id: ID; done: (result: boolean) => void; }>();\n  @Output() cancelAction = new EventEmitter<{ id: ID; done: (result: boolean) => void; }>();\n  @Output() viewAction = new EventEmitter<T>();\n  @Output() uploadAction = new EventEmitter<T>();\n  @Output() getRowButtonAction = new EventEmitter<T>();\n  @Output() moveUpAction = new EventEmitter<{ id: ID; done: (result: boolean) => void; }>();\n  @Output() moveDownAction = new EventEmitter<{ id: ID; done: (result: boolean) => void; }>();\n  @Output() enableAction = new EventEmitter<{ id: ID; enabled: boolean; done: (result: boolean) => void; }>();\n  @Output() searchTerm = new EventEmitter<string>();\n  @Output() errorEvent = new EventEmitter<string>();\n  @Output() selectable = new EventEmitter<T[]>();\n\n  // Estado principal\n  readonly data = signal<T[]>([]);\n  readonly keys = signal<string[]>([]);\n  readonly searchValue = signal<string>('');\n\n  // Estado de selección (delegado a servicio)\n  private readonly selectionState = this.selectionService.createSelectionState<T>();\n  readonly selectedItems = this.selectionState.selectedItems;\n  readonly selectAllChecked = this.selectionState.selectAllChecked;\n  readonly selectAllIndeterminate = this.selectionState.selectAllIndeterminate;\n\n  // Estado CRUD (delegado a servicio)\n  private readonly crudState = this.crudService.createCrudState<T>();\n  readonly creating = this.crudState.creating;\n  readonly newRow = this.crudState.newRow;\n  readonly editing = this.crudState.editing;\n  readonly editRow = this.crudState.editRow;\n\n  // Flags to check if events have listeners\n  readonly hasCreateActionListener = signal(false);\n  readonly hasUpdateActionListener = signal(false);\n  readonly hasDeleteActionListener = signal(false);\n  readonly hasCancelActionListener = signal(false);\n  readonly hasViewActionListener = signal(false);\n  readonly hasUploadActionListener = signal(false);\n  readonly hasGetRowButtonListener = signal(false);\n  readonly hasMoveUpActionListener = signal(false);\n  readonly hasMoveDownActionListener = signal(false);\n  readonly hasEnableActionListener = signal(false);\n  readonly hasSearchTermListener = signal(false);\n  readonly hasSelectableListener = signal(false);\n\n  private dataSub?: Subscription;\n  private inputValuesSub?: Subscription;\n\n  private getErrorMessage(err: unknown): string {\n    return (err as any)?.message || 'Error al cargar datos';\n  }\n\n  /**\n   * Actualiza las keys de columnas visibles basándose en el estado actual\n   */\n  private updateVisibleKeys(): void {\n    const visibleKeys = this.visibilityService.updateVisibleKeys(\n      this.columns,\n      this.data(),\n      this.creating(),\n      this.editing()\n    );\n    this.keys.set(visibleKeys);\n  }\n\n  /**\n   * Aplica valores parciales a los inputs en modo creación o edición.\n   * Solo actualiza si estamos en modo creación (creating = true) o edición (editing != null)\n   */\n  private applyInputValues(partialValues: Partial<T>): void {\n    this.crudState.applyInputValues(partialValues);\n  }\n\n  ngOnInit() {\n    // Check if the outputs have listeners\n    this.hasCreateActionListener.set(this.createAction.observed);\n    this.hasUpdateActionListener.set(this.updateAction.observed);\n    this.hasDeleteActionListener.set(this.deleteAction.observed);\n    this.hasCancelActionListener.set(this.cancelAction.observed);\n    this.hasViewActionListener.set(this.viewAction.observed);\n    this.hasUploadActionListener.set(this.uploadAction.observed);\n    this.hasGetRowButtonListener.set(this.getRowButtonAction.observed);\n    this.hasMoveUpActionListener.set(this.moveUpAction.observed);\n    this.hasMoveDownActionListener.set(this.moveDownAction.observed);\n    this.hasEnableActionListener.set(this.enableAction.observed);\n    this.hasSearchTermListener.set(this.searchTerm.observed);\n    this.hasSelectableListener.set(this.selectable.observed);\n\n    if (!this.data$) return;\n    this.dataSub = this.data$.subscribe({\n      next: (items) => {\n        this.dataUtils.applySorting(items, this.columns);\n        this.data.set(items);\n        // Actualizar las columnas visibles basándose en el estado actual\n        this.updateVisibleKeys();\n        // Mantener la selección existente después de actualizar los datos\n        this.preserveSelection();\n      },\n      error: (err: any) =>\n        this.errorEvent.emit(this.getErrorMessage(err)),\n    });\n\n    // Suscribirse a inputValues$ para actualizar valores dinámicamente en creación/edición\n    if (this.inputValues$) {\n      this.inputValuesSub = this.inputValues$.subscribe({\n        next: (partialValues) => {\n          this.applyInputValues(partialValues);\n        },\n        error: (err: any) =>\n          console.warn('Error en inputValues$:', this.getErrorMessage(err)),\n      });\n    }\n  }\n\n  ngOnDestroy() {\n    this.dataSub?.unsubscribe();\n    this.inputValuesSub?.unsubscribe();\n\n    // Close any open modal when component is destroyed\n    this.modalService.closeModal();\n  }\n\n  onInput(event: Event, key: string, col?: C80TableColDef) {\n    if (col?.type === 'boolean') {\n      const checked = (event.target as HTMLInputElement).checked;\n      this.updateNewRow(key, checked);\n    } else if (col?.type === 'enum') {\n      const value = (event.target as HTMLSelectElement).value;\n      this.updateNewRow(key, value);\n    } else {\n      const target = event.target as HTMLInputElement;\n      if (target && typeof target.value === 'string') {\n        this.updateNewRow(key, target.value);\n      }\n    }\n  }\n\n  async onDelete(row: T) {\n    const id = row['id'] as ID;\n\n    let confirmed = true;\n    if (!this.noConfirm) {\n      confirmed = await this.modalService.confirm(\n        'Confirmar eliminación',\n        '¿Está seguro de que desea eliminar este elemento? Esta acción no se puede deshacer.',\n        'Eliminar',\n        'Cancelar'\n      );\n    }\n\n    if (confirmed) {\n      this.deleteAction.emit({\n        id,\n        done: (success: boolean) => {\n          if (success) this.cancelEdit();\n        },\n      });\n    }\n  }\n\n  async onCancel(row: T) {\n    const id = row['id'] as ID;\n\n    let confirmed = true;\n    if (!this.noConfirm) {\n      confirmed = await this.modalService.confirm(\n        'Confirmar cancelación',\n        '¿Está seguro de que desea cancelar este elemento?',\n        'Cancelar elemento',\n        'No cancelar'\n      );\n    }\n\n    if (confirmed) {\n      this.cancelAction.emit({\n        id,\n        done: (success: boolean) => {\n          if (success) this.cancelEdit();\n        },\n      });\n    }\n  }\n\n  startCreate() {\n    this.crudState.startCreate(this.columns, this.data());\n    // Actualizar keys de columnas visibles cuando entramos en modo creación\n    this.updateVisibleKeys();\n  }\n\n  cancelCreate() {\n    this.crudState.cancelCreate();\n    // Actualizar keys de columnas visibles cuando salimos del modo creación\n    this.updateVisibleKeys();\n  }\n\n  updateNewRow(key: string, value: unknown) {\n    this.crudState.updateNewRow(key, value);\n  }\n\n  saveCreate() {\n    this.crudState.saveCreate(this.columns, this.data(), this.createAction);\n  }\n\n\n\n  onEdit(row: T) {\n    this.crudState.startEdit(row, this.columns, this.data());\n    // Actualizar keys de columnas visibles cuando entramos en modo edición\n    this.updateVisibleKeys();\n  }\n\n  cancelEdit() {\n    this.crudState.cancelEdit();\n    // Actualizar keys de columnas visibles cuando salimos del modo edición\n    this.updateVisibleKeys();\n  }\n\n  onEditInput(event: Event, key: string, col?: C80TableColDef) {\n    if (col?.type === 'boolean') {\n      const checked = (event.target as HTMLInputElement).checked;\n      this.crudState.updateEditRow(key, checked);\n    } else if (col?.type === 'enum') {\n      const value = (event.target as HTMLSelectElement).value;\n      this.crudState.updateEditRow(key, value);\n    } else {\n      const target = event.target as HTMLInputElement;\n      if (target && typeof target.value === 'string') {\n        this.crudState.updateEditRow(key, target.value);\n      }\n    }\n  }\n\n  saveEdit(row: T) {\n    const id = row['id'] as ID;\n    this.crudState.saveEdit(id, this.columns, this.data(), this.updateAction);\n  }\n\n  /**\n   * TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).\n   */\n  trackById(index: number, row: T): number | string {\n    const id =\n      row && typeof row === 'object' && 'id' in row\n        ? (row as Record<string, unknown>)['id']\n        : undefined;\n    return typeof id === 'string' || typeof id === 'number' ? id : index;\n  }\n\n  /**\n   * Emits the view event with the entire row data\n   */\n  onView(row: T): void {\n    this.viewAction.emit(row);\n  }\n\n  /**\n   * Emits the getRowButtonAction event with the entire row data\n   */\n  onGetRowButton(row: T): void {\n    this.getRowButtonAction.emit(row);\n  }\n\n  /**\n   * Emits the moveUpAction event with the row id\n   */\n  async onMoveUp(row: T) {\n    const id = row['id'] as ID;\n    this.moveUpAction.emit({\n      id,\n      done: (success: boolean) => {\n        if (success) this.cancelEdit();\n      },\n    });\n  }\n\n  /**\n   * Emits the moveDownAction event with the row id\n   */\n  async onMoveDown(row: T) {\n    const id = row['id'] as ID;\n    this.moveDownAction.emit({\n      id,\n      done: (success: boolean) => {\n        if (success) this.cancelEdit();\n      },\n    });\n  }\n\n  /**\n   * Emits the enableAction event with the row id and current enabled status\n   */\n  async onEnable(row: T) {\n    const id = row['id'] as ID;\n    const currentEnabled = row['enabled'] as boolean;\n    this.enableAction.emit({\n      id,\n      enabled: !currentEnabled, // Toggle the current state\n      done: (success: boolean) => {\n        if (success) this.cancelEdit();\n      },\n    });\n  }\n\n  /**\n   * Handles search input changes with debouncing\n   */\n  onSearchInput(event: Event): void {\n    const target = event.target as HTMLInputElement;\n    const value = target.value;\n    this.searchValue.set(value);\n    this.searchTerm.emit(value);\n  }\n\n  /**\n   * Clears the search input\n   */\n  clearSearch(): void {\n    this.searchValue.set('');\n    this.searchTerm.emit('');\n  }\n\n  // Métodos delegados a servicios\n  getCellValue(row: T, accessor: string): unknown {\n    return this.dataUtils.getCellValue(row, accessor);\n  }\n\n  getDisplayValue(value: unknown): string {\n    return this.dataUtils.getDisplayValue(value);\n  }\n\n  getEnumDisplayValue(value: unknown, col: C80TableColDef): string {\n    return this.dataUtils.getEnumDisplayValue(value, col);\n  }\n\n  getEnumOptions(col: C80TableColDef): { value: string | number; label: string }[] {\n    return this.dataUtils.getEnumOptions(col);\n  }\n\n  getCellColor(value: unknown, col: C80TableColDef): string | undefined {\n    return this.dataUtils.getCellColor(value, col);\n  }\n\n  getTableMaxHeight(): string | undefined {\n    return this.dataUtils.getTableMaxHeight(this.size);\n  }\n\n  /**\n   * Selection methods\n   */\n  clearSelection(): void {\n    this.selectionState.clearSelection();\n    this.selectionService.emitSelection(this.selectionState, this.data(), this.selectable);\n  }\n\n  /**\n   * Mantiene la selección existente después de actualizar los datos,\n   * eliminando solo los IDs que ya no existen en los nuevos datos\n   */\n  private preserveSelection(): void {\n    this.selectionState.preserveSelection(this.data());\n    this.selectionService.emitSelection(this.selectionState, this.data(), this.selectable);\n  }\n\n  toggleSelectAll(): void {\n    this.selectionState.toggleSelectAll(this.data());\n    this.selectionService.emitSelection(this.selectionState, this.data(), this.selectable);\n  }\n\n  toggleItemSelection(item: T): void {\n    this.selectionState.toggleItemSelection(item, this.multiple);\n    this.selectionState.updateSelectAllState(this.data());\n    this.selectionService.emitSelection(this.selectionState, this.data(), this.selectable);\n  }\n\n  isItemSelected(item: T): boolean {\n    return this.selectionState.isItemSelected(item);\n  }\n\n\n\n  // Métodos de visibilidad delegados a servicio\n  isColumnVisible(column: C80TableColDef, forceShowInCreation = false): boolean {\n    return this.visibilityService.isColumnVisible(column, this.data(), forceShowInCreation);\n  }\n\n  isColumnVisibleInHeader(column: C80TableColDef): boolean {\n    return this.visibilityService.isColumnVisibleInHeader(\n      column,\n      this.data(),\n      this.creating(),\n      this.editing()\n    );\n  }\n\n  isColumnVisibleForRow(column: C80TableColDef, row: T): boolean {\n    return this.visibilityService.isColumnVisibleForRow(\n      column,\n      row,\n      this.data(),\n      this.editing()\n    );\n  }\n\n  onUpload(row: T): void {\n    this.uploadAction.emit(row);\n  }\n}\n","<div class=\"table-responsive\" [style.max-height]=\"getTableMaxHeight()\" [style.overflow-y]=\"size > 0 ? 'auto' : 'visible'\">\n  <!-- Search Bar -->\n  @if (hasSearchTermListener()) {\n  <div class=\"search-container\">\n    <div class=\"search-input-wrapper\">\n      <div class=\"input-group\">\n        <span class=\"input-group-text\">\n          <c80-icon icon=\"search\" [size]=\".8\"></c80-icon>\n        </span>\n        <input type=\"text\" class=\"form-control search-input\" placeholder=\"Buscar...\" [value]=\"searchValue()\" (input)=\"onSearchInput($event)\" aria-label=\"Buscar en la tabla\" />\n        @if (searchValue()) {\n        <button class=\"btn btn-outline-secondary btn-borrar\" type=\"button\" (click)=\"clearSearch()\" title=\"Limpiar búsqueda\">\n          <c80-icon icon=\"cancel\" [size]=\".7\"></c80-icon>\n        </button>\n        }\n      </div>\n    </div>\n  </div>\n  }\n\n  <table class=\"table table-bordered table-hover align-middle\">\n    <thead class=\"thead table-light sticky-header\">\n      <tr>\n        @if (hasSelectableListener() && data().length !== 0) {\n        <th class=\"text-center selection-column\">\n          @if (multiple) {\n          <input type=\"checkbox\" [checked]=\"selectAllChecked()\" [indeterminate]=\"selectAllIndeterminate()\" (change)=\"toggleSelectAll()\" aria-label=\"Seleccionar todo\" />\n          }\n        </th>\n        }\n        @for (col of columns; track col) {\n        @if (isColumnVisibleInHeader(col)) {\n        @if (col.type === 'boolean') {\n        <th class=\"text-center boolean-column\">{{ col.label }}</th>\n        }\n        @else if (col.type === 'number') {\n        <th class=\"text-center number-column\">{{ col.label }}</th>\n        }\n        @else {\n        <th>{{ col.label }}</th>\n        }\n        }\n        }\n        @if (hasCreateActionListener() || hasUpdateActionListener() || hasDeleteActionListener() ||\n        hasCancelActionListener() || hasViewActionListener() || hasUploadActionListener() || hasGetRowButtonListener() ||\n        hasMoveUpActionListener() || hasMoveDownActionListener()) {\n        <th class=\"table-actions-header\">\n          <div class=\"actions-wrapper\">\n            <span>Actions</span>\n            @if (hasCreateActionListener()) {\n            <c80-icon button icon=\"add\" [disabled]=\"creating()\" title=\"Agregar\" [size]=\".6\" (iconClick)=\"startCreate()\"></c80-icon>\n            }\n          </div>\n        </th>\n        }\n      </tr>\n    </thead>\n    <tbody>\n      @for (row of data(); track trackById(i, row); let i = $index) {\n      <tr>\n        @if (hasSelectableListener() && data().length !== 0) {\n        <td class=\"text-center selection-column\">\n          <input type=\"checkbox\" [checked]=\"isItemSelected(row)\" (change)=\"toggleItemSelection(row)\" [attr.aria-label]=\"'Seleccionar fila ' + (i + 1)\" />\n        </td>\n        }\n        @for (col of columns; track col) {\n        @if (isColumnVisibleForRow(col, row)) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center boolean-column\">\n          @if (editing() === row['id'] && !col.readOnly) {\n          <input type=\"checkbox\" [checked]=\"!!editRow()?.[col.accessor]\" (change)=\"onEditInput($event, col.accessor, col)\" [attr.aria-label]=\"col.label\" />\n          }\n          @else {\n          @if (getCellValue(row, col.accessor) === true) {\n          <c80-icon icon=\"check\" [size]=\".7\"></c80-icon>\n          <br />\n          }\n          @else if (getCellValue(row, col.accessor) === false) {\n          <c80-icon icon=\"cancel\" [size]=\".7\"></c80-icon>\n          <br />\n          }\n          }\n        </td>\n        }\n        @else if (col.type === 'number') {\n        <td class=\"text-center number-column\">\n          @if (editing() === row['id'] && !col.readOnly) {\n          <input class=\"form-control form-control-sm\" type=\"number\" [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\" (input)=\"onEditInput($event, col.accessor, col)\" />\n          }\n          @else {\n          <span [style.color]=\"getCellColor(getCellValue(row, col.accessor), col)\">{{ getDisplayValue(getCellValue(row,\n            col.accessor)) }}</span>\n          }\n        </td>\n        }\n        @else {\n        <td>\n          @if (editing() === row['id'] && !col.readOnly) {\n          @if (col.type === 'enum') {\n          <select class=\"form-control form-control-sm\" [value]=\"editRow()?.[col.accessor] ?? ''\" (change)=\"onEditInput($event, col.accessor, col)\">\n            <option value=\"\">{{ col.label }}</option>\n            @for (option of getEnumOptions(col); track option.value) {\n            <option [value]=\"option.value\">{{ option.label }}</option>\n            }\n          </select>\n          }\n          @else {\n          <input class=\"form-control form-control-sm\" [type]=\"col.type === 'password' ? 'password' : 'text'\" [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\" (input)=\"onEditInput($event, col.accessor, col)\" />\n          }\n          }\n          @else {\n          @if (col.type === 'password') {\n          <span [style.color]=\"getCellColor(getCellValue(row, col.accessor), col)\">******</span>\n          }\n          @else if (col.type === 'enum') {\n          <span [style.color]=\"getCellColor(getCellValue(row, col.accessor), col)\">{{\n            getEnumDisplayValue(getCellValue(row, col.accessor), col)\n            }}</span>\n          }\n          @else if (getCellValue(row, col.accessor) === true) {\n          <c80-icon icon=\"check\" [size]=\".7\"></c80-icon>\n          }\n          @else if (getCellValue(row, col.accessor) === false) {\n          <c80-icon icon=\"cancel\" [size]=\".7\"></c80-icon>\n          }\n          @else {\n          <span [style.color]=\"getCellColor(getCellValue(row, col.accessor), col)\">{{ getDisplayValue(getCellValue(row,\n            col.accessor)) }}</span>\n          }\n          }\n        </td>\n        }\n        }\n        }\n        @if (hasCreateActionListener() || hasUpdateActionListener() || hasDeleteActionListener() ||\n        hasCancelActionListener() || hasViewActionListener() || hasUploadActionListener() || hasGetRowButtonListener() ||\n        hasMoveUpActionListener() || hasMoveDownActionListener()) {\n        <td class=\"text-center actions-cell\">\n          <div class=\"actions-container\">\n            @if (editing() === row['id']) {\n            <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveEdit(row)\" [size]=\".7\"></c80-icon>\n            <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelEdit()\" [size]=\".7\"></c80-icon>\n            }\n            @else {\n            @if (hasUpdateActionListener()) {\n            <c80-icon button icon=\"edit\" title=\"Editar\" (iconClick)=\"onEdit(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasCancelActionListener()) {\n            <c80-icon button icon=\"cancel_circle\" color=\"secondary\" title=\"Cancelar\" (iconClick)=\"onCancel(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasViewActionListener()) {\n            <c80-icon button icon=\"view\" color=\"primary\" title=\"Ver\" (iconClick)=\"onView(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasUploadActionListener()) {\n            <c80-icon button icon=\"upload\" color=\"primary\" title=\"Subir\" (iconClick)=\"onUpload(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasGetRowButtonListener()) {\n            <c80-icon button icon=\"get\" color=\"success\" title=\"Vincular\" (iconClick)=\"onGetRowButton(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasMoveUpActionListener()) {\n            <c80-icon button icon=\"arrow_up\" color=\"secondary\" title=\"Mover arriba\" (iconClick)=\"onMoveUp(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasMoveDownActionListener()) {\n            <c80-icon button icon=\"arrow_down\" color=\"secondary\" title=\"Mover abajo\" (iconClick)=\"onMoveDown(row)\" [size]=\".7\"></c80-icon>\n            }\n            @if (hasEnableActionListener()) {\n            <c80-icon button [icon]=\"getCellValue(row, 'enabled') ? 'toggle_on' : 'toggle_off'\" [color]=\"getCellValue(row, 'enabled') ? 'success' : 'secondary'\" [title]=\"getCellValue(row, 'enabled') ? 'Deshabilitar' : 'Habilitar'\"\n              (iconClick)=\"onEnable(row)\" [size]=\".7\">\n            </c80-icon>\n            }\n            @if (hasDeleteActionListener()) {\n            <c80-icon button icon=\"delete\" color=\"warn\" title=\"Borrar\" (iconClick)=\"onDelete(row)\" [size]=\".7\"></c80-icon>\n            }\n            }\n          </div>\n        </td>\n        }\n      </tr>\n      }\n      @if (creating() && hasCreateActionListener()) {\n      <tr>\n        @if (hasSelectableListener() && data().length !== 0) {\n        <td class=\"text-center selection-column\">\n          <!-- Empty cell for alignment -->\n        </td>\n        }\n        @for (col of columns; track col) {\n        @if (isColumnVisible(col, true)) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center\">\n          @if (!col.readOnly) {\n          <input type=\"checkbox\" [checked]=\"!!newRow()?.[col.accessor]\" (change)=\"onInput($event, col.accessor, col)\" [attr.aria-label]=\"col.label\" />\n          }\n          @else {\n          <!-- ReadOnly boolean column in create mode shows empty -->\n          <span class=\"text-muted\">-</span>\n          }\n        </td>\n        } @else if (col.type === 'number') {\n        <td class=\"text-center number-column\">\n          @if (!col.readOnly) {\n          <input class=\"form-control form-control-sm\" type=\"number\" [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\" (input)=\"onInput($event, col.accessor, col)\" />\n          }\n          @else {\n          <!-- ReadOnly number column in create mode shows empty -->\n          <span class=\"text-muted\">-</span>\n          }\n        </td>\n        } @else {\n        <td>\n          @if (!col.readOnly) {\n          @if (col.type === 'enum') {\n          <select class=\"form-control form-control-sm\" [value]=\"newRow()?.[col.accessor] ?? ''\" (change)=\"onInput($event, col.accessor, col)\">\n            <option value=\"\">{{ col.label }}</option>\n            @for (option of getEnumOptions(col); track option.value) {\n            <option [value]=\"option.value\">{{ option.label }}</option>\n            }\n          </select>\n          }\n          @else {\n          <input class=\"form-control form-control-sm\" type=\"text\" [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\" (input)=\"onInput($event, col.accessor, col)\" />\n          }\n          }\n          @else {\n          <!-- ReadOnly column in create mode shows empty -->\n          <span class=\"text-muted\">-</span>\n          }\n        </td>\n        }\n        }\n        }\n        <td class=\"text-center actions-cell\">\n          <div class=\"actions-container\">\n            <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveCreate()\" [size]=\".7\"></c80-icon>\n            <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelCreate()\" [size]=\".7\"></c80-icon>\n          </div>\n        </td>\n      </tr>\n      }\n    </tbody>\n  </table>\n  @if (data().length === 0 && !creating()) {\n  <div class=\"text-center text-muted py-3 small\">\n    No hay datos para mostrar.\n  </div>\n  }\n</div>\n\n<c80-modal></c80-modal>"]}
|