@c80/ui 1.0.2 → 1.0.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md
CHANGED
|
@@ -24,12 +24,16 @@ Componente para mostrar íconos SVG personalizables, con soporte para distintos
|
|
|
24
24
|
<c80-icon icon="add" [size]="2"></c80-icon>
|
|
25
25
|
```
|
|
26
26
|
|
|
27
|
+
|
|
27
28
|
### `c80-table`
|
|
28
|
-
Componente de tabla editable y dinámica, con soporte para operaciones CRUD
|
|
29
|
+
Componente de tabla editable y dinámica, con soporte para operaciones CRUD, definición de columnas, edición en línea y acciones integradas. Ahora desacoplado: recibe los datos como observable y emite eventos para las acciones.
|
|
29
30
|
|
|
30
31
|
**Props principales:**
|
|
31
|
-
- `
|
|
32
|
+
- `data$`: `Observable<T[]>` (observable con los datos a mostrar)
|
|
32
33
|
- `columns`: `C80TableColDef[]` (definición de columnas: accessor, label, tipo, visible)
|
|
34
|
+
- `create`: `EventEmitter<Partial<T>>` (emite al crear una fila)
|
|
35
|
+
- `update`: `EventEmitter<{ id: number, changes: Partial<T> }>` (emite al editar una fila)
|
|
36
|
+
- `delete`: `EventEmitter<number>` (emite al borrar una fila)
|
|
33
37
|
- `errorEvent`: `EventEmitter<string>` (emite errores de operación)
|
|
34
38
|
|
|
35
39
|
**Definición de columna:**
|
|
@@ -44,7 +48,13 @@ interface C80TableColDef {
|
|
|
44
48
|
|
|
45
49
|
**Ejemplo de uso:**
|
|
46
50
|
```html
|
|
47
|
-
<c80-table
|
|
51
|
+
<c80-table
|
|
52
|
+
[data$]="users$"
|
|
53
|
+
[columns]="userColumns"
|
|
54
|
+
(create)="onCreate($event)"
|
|
55
|
+
(update)="onUpdate($event)"
|
|
56
|
+
(delete)="onDelete($event)"
|
|
57
|
+
(errorEvent)="onError($event)"></c80-table>
|
|
48
58
|
```
|
|
49
59
|
```typescript
|
|
50
60
|
userColumns = [
|
|
@@ -52,6 +62,18 @@ userColumns = [
|
|
|
52
62
|
{ accessor: 'name', label: 'Nombre', type: 'string' },
|
|
53
63
|
{ accessor: 'active', label: 'Activo', type: 'boolean' }
|
|
54
64
|
];
|
|
65
|
+
|
|
66
|
+
users$: Observable<User[]> = this.userService.getAll();
|
|
67
|
+
|
|
68
|
+
onCreate(row: Partial<User>) {
|
|
69
|
+
// Lógica para crear usuario
|
|
70
|
+
}
|
|
71
|
+
onUpdate(evt: { id: number, changes: Partial<User> }) {
|
|
72
|
+
// Lógica para actualizar usuario
|
|
73
|
+
}
|
|
74
|
+
onDelete(id: number) {
|
|
75
|
+
// Lógica para borrar usuario
|
|
76
|
+
}
|
|
55
77
|
```
|
|
56
78
|
|
|
57
79
|
## Instalación y uso
|
|
@@ -50,7 +50,13 @@ export class C80TableComponent {
|
|
|
50
50
|
return;
|
|
51
51
|
}
|
|
52
52
|
if (confirm('¿Está seguro de que desea borrar este elemento?')) {
|
|
53
|
-
this.delete.emit(
|
|
53
|
+
this.delete.emit({
|
|
54
|
+
id,
|
|
55
|
+
done: (success) => {
|
|
56
|
+
if (success)
|
|
57
|
+
this.cancelEdit();
|
|
58
|
+
}
|
|
59
|
+
});
|
|
54
60
|
}
|
|
55
61
|
}
|
|
56
62
|
startCreate() {
|
|
@@ -81,8 +87,12 @@ export class C80TableComponent {
|
|
|
81
87
|
acc[col.accessor] = this.convertCellValue(row[col.accessor], col);
|
|
82
88
|
return acc;
|
|
83
89
|
}, {});
|
|
84
|
-
this.create.emit(
|
|
85
|
-
|
|
90
|
+
this.create.emit({
|
|
91
|
+
row: converted, done: (success) => {
|
|
92
|
+
if (success)
|
|
93
|
+
this.cancelCreate();
|
|
94
|
+
}
|
|
95
|
+
});
|
|
86
96
|
}
|
|
87
97
|
/**
|
|
88
98
|
* Converts a cell value based on column type or sample data.
|
|
@@ -187,8 +197,12 @@ export class C80TableComponent {
|
|
|
187
197
|
acc[col.accessor] = this.convertCellValue(this.editRow()[col.accessor], col);
|
|
188
198
|
return acc;
|
|
189
199
|
}, {});
|
|
190
|
-
this.update.emit({
|
|
191
|
-
|
|
200
|
+
this.update.emit({
|
|
201
|
+
id, changes: converted, done: (success) => {
|
|
202
|
+
if (success)
|
|
203
|
+
this.cancelEdit();
|
|
204
|
+
}
|
|
205
|
+
});
|
|
192
206
|
}
|
|
193
207
|
/**
|
|
194
208
|
* TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).
|
|
@@ -216,4 +230,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
216
230
|
}], errorEvent: [{
|
|
217
231
|
type: Output
|
|
218
232
|
}] } });
|
|
219
|
-
//# 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 { Component, Input, Output, signal, EventEmitter, OnInit, OnDestroy } from '@angular/core';\r\nimport { Observable, Subscription } from 'rxjs';\r\nimport { C80IconComponent } from '../icon/icon.component';\r\n\r\nexport interface C80TableColDef {\r\n  accessor: string;\r\n  label: string;\r\n  visible?: boolean; // Si no se especifica, se asume true\r\n  type?: 'string' | 'number' | 'boolean'; // Tipo de dato para la columna\r\n}\r\n\r\n@Component({\r\n  selector: 'c80-table',\r\n  standalone: true,\r\n  imports: [C80IconComponent],\r\n  templateUrl: './table.component.html',\r\n  styleUrl: './table.component.scss',\r\n})\r\nexport class C80TableComponent<T extends Record<string, unknown>> implements OnInit, OnDestroy {\r\n  @Input() data$!: Observable<T[]>;\r\n  @Input() columns: C80TableColDef[] = [];\r\n  @Output() create = new EventEmitter<Partial<T>>();\r\n  @Output() update = new EventEmitter<{ id: number, changes: Partial<T> }>();\r\n  @Output() delete = new EventEmitter<number>();\r\n  @Output() errorEvent = new EventEmitter<string>();\r\n\r\n  readonly data = signal<T[]>([]);\r\n  readonly keys = signal<string[]>([]);\r\n  creating = signal(false);\r\n  newRow = signal<Partial<T> | null>(null);\r\n  readonly editing = signal<number | null>(null); // id of row being edited\r\n  readonly editRow = signal<Partial<T> | null>(null);\r\n\r\n  private dataSub?: Subscription;\r\n\r\n  ngOnInit() {\r\n    if (!this.data$) return;\r\n    this.dataSub = this.data$.subscribe({\r\n      next: (items) => {\r\n        this.data.set(items);\r\n        // Solo mostrar las columnas visibles\r\n        const visibleColumns = this.columns.filter(col => col?.visible !== false);\r\n        this.keys.set(visibleColumns.map(col => col.accessor));\r\n      },\r\n      error: (err) => this.errorEvent.emit(err?.message || 'Error al cargar datos')\r\n    });\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    this.dataSub?.unsubscribe();\r\n  }\r\n\r\n  onInput(event: Event, key: string, col?: C80TableColDef) {\r\n    if (col?.type === 'boolean') {\r\n      const checked = (event.target as HTMLInputElement).checked;\r\n      this.updateNewRow(key, checked);\r\n    } else {\r\n      const target = (event.target as HTMLInputElement);\r\n      if (target && typeof target.value === 'string') {\r\n        this.updateNewRow(key, target.value);\r\n      }\r\n    }\r\n  }\r\n\r\n  onDelete(row: T) {\r\n    const id = row['id'];\r\n    if (typeof id !== 'number') {\r\n      this.errorEvent.emit('No se puede borrar: id inválido');\r\n      return;\r\n    }\r\n    if (confirm('¿Está seguro de que desea borrar este elemento?')) {\r\n      this.delete.emit(id);\r\n    }\r\n  }\r\n\r\n  startCreate() {\r\n    this.creating.set(true);\r\n    // Inicializa newRow solo con los accessors de columnas visibles\r\n    const row: Partial<T> = {};\r\n    this.columns.filter(col => col.visible !== false).forEach(col => {\r\n      (row as Record<string, unknown>)[col.accessor] = '';\r\n    });\r\n    this.newRow.set(row);\r\n  }\r\n\r\n  cancelCreate() {\r\n    this.creating.set(false);\r\n    this.newRow.set(null);\r\n  }\r\n\r\n  updateNewRow(key: string, value: unknown) {\r\n    const current = this.newRow();\r\n    if (!current) return;\r\n    this.newRow.set({ ...current, [key]: value });\r\n  }\r\n\r\n  saveCreate() {\r\n    const row = this.newRow();\r\n    if (!row) return;\r\n    const visibleColumns = this.columns.filter(col => col.visible !== false);\r\n    const converted = visibleColumns.reduce((acc, col) => {\r\n      acc[col.accessor] = this.convertCellValue(row[col.accessor], col);\r\n      return acc;\r\n    }, {} as Record<string, unknown>) as Partial<T>;\r\n    this.create.emit(converted);\r\n    this.cancelCreate();\r\n  }\r\n\r\n  /**\r\n   * Converts a cell value based on column type or sample data.\r\n   * Handles stringification of objects for string columns.\r\n   */\r\n  /**\r\n   * Converts a cell value based on column type or sample data.\r\n   * Delegates to type-specific helpers for clarity and maintainability.\r\n   */\r\n  private convertCellValue(value: unknown, col: C80TableColDef): unknown {\r\n    if (col.type === 'boolean') return this.toBoolean(value);\r\n    if (col.type === 'number') return this.toNumber(value);\r\n    if (col.type === 'string') return this.toStringValue(value);\r\n\r\n    // Fallback: use sample data if available\r\n    const sample = this.data()[0];\r\n    if (sample && col.accessor in sample) {\r\n      const sampleValue = sample[col.accessor];\r\n      if (typeof sampleValue === 'boolean') return this.toBoolean(value);\r\n      if (typeof sampleValue === 'number') return this.toNumber(value);\r\n      if (typeof sampleValue === 'string') return this.toStringValue(value);\r\n    }\r\n    return value;\r\n  }\r\n\r\n  /** Converts value to boolean using best practices. */\r\n  private toBoolean(value: unknown): boolean {\r\n    if (typeof value === 'boolean') return value;\r\n    if (typeof value === 'string') return value.trim().toLowerCase() === 'true' || value.trim() === '1';\r\n    if (typeof value === 'number') return value === 1;\r\n    return false;\r\n  }\r\n\r\n  /** Converts value to number using best practices. */\r\n  private toNumber(value: unknown): number | undefined {\r\n    if (typeof value === 'number') return value;\r\n    if (typeof value === 'string') return value.trim() === '' ? undefined : Number(value);\r\n    if (typeof value === 'boolean') return value ? 1 : 0;\r\n    return undefined;\r\n  }\r\n\r\n  /** Converts value to string using best practices, always stringifies objects. */\r\n  private toStringValue(value: unknown): string {\r\n    if (value == null) return '';\r\n    if (typeof value === 'string') return value;\r\n    if (typeof value === 'number' || typeof value === 'boolean') return String(value);\r\n    if (typeof value === 'object') {\r\n      try {\r\n        return JSON.stringify(value);\r\n      } catch {\r\n        return '[object Object]';\r\n      }\r\n    }\r\n    // For functions, symbols, undefined, etc., return empty string\r\n    return '';\r\n  }\r\n\r\n  onEdit(row: T) {\r\n    this.editing.set(row['id'] as number);\r\n    const edit: Partial<T> = {};\r\n    this.columns.filter(col => col.visible !== false).forEach(col => {\r\n      (edit as Record<string, unknown>)[col.accessor] = row[col.accessor];\r\n    });\r\n    this.editRow.set(edit);\r\n  }\r\n\r\n  cancelEdit() {\r\n    this.editing.set(null);\r\n    this.editRow.set(null);\r\n  }\r\n\r\n  onEditInput(event: Event, key: string, col?: C80TableColDef) {\r\n    const current = this.editRow();\r\n    if (!current) return;\r\n    if (col?.type === 'boolean') {\r\n      const checked = (event.target as HTMLInputElement).checked;\r\n      this.editRow.set({ ...current, [key]: checked });\r\n    } else {\r\n      const target = (event.target as HTMLInputElement);\r\n      if (target && typeof target.value === 'string') {\r\n        this.editRow.set({ ...current, [key]: target.value });\r\n      }\r\n    }\r\n  }\r\n\r\n  saveEdit(row: T) {\r\n    const id = row['id'];\r\n    if (typeof id !== 'number' || !this.editRow()) return;\r\n    const visibleColumns = this.columns.filter(col => col.visible !== false);\r\n    const converted = visibleColumns.reduce((acc, col) => {\r\n      acc[col.accessor] = this.convertCellValue(this.editRow()![col.accessor], col);\r\n      return acc;\r\n    }, {} as Record<string, unknown>) as Partial<T>;\r\n    this.update.emit({ id, changes: converted });\r\n    this.cancelEdit();\r\n  }\r\n\r\n  /**\r\n   * TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).\r\n   */\r\n  trackById(index: number, row: T): number | string {\r\n    const id = row && typeof row === 'object' && 'id' in row ? (row as Record<string, unknown>)['id'] : undefined;\r\n    return (typeof id === 'string' || typeof id === 'number') ? id : index;\r\n  }\r\n}\r\n","<div class=\"table-responsive\">\n  <table class=\"table table-bordered table-hover align-middle\">\n    <thead class=\"thead table-light\">\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <th class=\"text-center\">{{ col.label }}</th>\n        }\n        @else {\n        <th>{{ col.label }}</th>\n        }\n        }\n        }\n        <th class=\"table-actions-header\">\n          <span>Actions</span>\n          <c80-icon button icon=\"add\" [disabled]=\"creating()\" title=\"Agregar\" class=\"add-action-btn\"\n            (iconClick)=\"startCreate()\"></c80-icon>\n        </th>\n      </tr>\n    </thead>\n    <tbody>\n      @for (row of data(); track trackById(i, row); let i = $index) {\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center\">\n          @if (editing() === row['id']) {\n          <input type=\"checkbox\" [checked]=\"!!editRow()?.[col.accessor]\"\n            (change)=\"onEditInput($event, col.accessor, col)\" [attr.aria-label]=\"col.label\" />\n          }\n          @else {\n          @if (row[col.accessor] === true) {\n          <c80-icon icon=\"check\" [size]=\"1\"></c80-icon>\n          <br />\n          }\n          @else if (row[col.accessor] === false) {\n          <c80-icon icon=\"cancel\"></c80-icon>\n          <br />\n          }\n          }\n        </td>\n        }\n        @else {\n        <td>\n          @if (editing() === row['id']) {\n          <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n            [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n            (input)=\"onEditInput($event, col.accessor, col)\" />\n          }\n          @else {\n          {{ row[col.accessor] }}\n          }\n        </td>\n        }\n        }\n        }\n        <td class=\"text-center\">\n          @if (editing() === row['id']) {\n          <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveEdit(row)\"></c80-icon>\n          <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelEdit()\"></c80-icon>\n          }\n          @else {\n          <c80-icon button icon=\"edit\" title=\"Editar\" (iconClick)=\"onEdit(row)\"></c80-icon>\n          <c80-icon button icon=\"delete\" color=\"warn\" title=\"Borrar\" (iconClick)=\"onDelete(row)\"></c80-icon>\n          }\n        </td>\n      </tr>\n      }\n      @if (creating()) {\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center\">\n          <input type=\"checkbox\" [checked]=\"!!newRow()?.[col.accessor]\" (change)=\"onInput($event, col.accessor, col)\"\n            [attr.aria-label]=\"col.label\" />\n        </td>\n        } @else {\n        <td>\n          <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n            [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n            (input)=\"onInput($event, col.accessor, col)\" />\n        </td>\n        }\n        }\n        }\n        <td class=\"text-center\">\n          <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveCreate()\"></c80-icon>\n          <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelCreate()\"></c80-icon>\n        </td>\n      </tr>\n      }\n    </tbody>\n  </table>\n  @if (data().length === 0) {\n  <div class=\"text-center text-muted py-3\">\n    No hay datos para mostrar.\n  </div>\n  }\n</div>"]}
|
|
233
|
+
//# 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":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,EAAE,YAAY,EAAqB,MAAM,eAAe,CAAC;AAElG,OAAO,EAAE,gBAAgB,EAAE,MAAM,wBAAwB,CAAC;;AAgB1D,MAAM,OAAO,iBAAiB;IACnB,KAAK,CAAmB;IACxB,OAAO,GAAqB,EAAE,CAAC;IAC9B,MAAM,GAAG,IAAI,YAAY,EAAwD,CAAC;IAClF,MAAM,GAAG,IAAI,YAAY,EAAwE,CAAC;IAClG,MAAM,GAAG,IAAI,YAAY,EAAmD,CAAC;IAC7E,UAAU,GAAG,IAAI,YAAY,EAAU,CAAC;IAEzC,IAAI,GAAG,MAAM,CAAM,EAAE,CAAC,CAAC;IACvB,IAAI,GAAG,MAAM,CAAW,EAAE,CAAC,CAAC;IACrC,QAAQ,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;IACzB,MAAM,GAAG,MAAM,CAAoB,IAAI,CAAC,CAAC;IAChC,OAAO,GAAG,MAAM,CAAgB,IAAI,CAAC,CAAC,CAAC,yBAAyB;IAChE,OAAO,GAAG,MAAM,CAAoB,IAAI,CAAC,CAAC;IAE3C,OAAO,CAAgB;IAE/B,QAAQ;QACN,IAAI,CAAC,IAAI,CAAC,KAAK;YAAE,OAAO;QACxB,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,CAAC;YAClC,IAAI,EAAE,CAAC,KAAK,EAAE,EAAE;gBACd,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;gBACrB,qCAAqC;gBACrC,MAAM,cAAc,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,EAAE,OAAO,KAAK,KAAK,CAAC,CAAC;gBAC1E,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,cAAc,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC,CAAC;YACzD,CAAC;YACD,KAAK,EAAE,CAAC,GAAG,EAAE,EAAE,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,GAAG,EAAE,OAAO,IAAI,uBAAuB,CAAC;SAC9E,CAAC,CAAC;IACL,CAAC;IAED,WAAW;QACT,IAAI,CAAC,OAAO,EAAE,WAAW,EAAE,CAAC;IAC9B,CAAC;IAED,OAAO,CAAC,KAAY,EAAE,GAAW,EAAE,GAAoB;QACrD,IAAI,GAAG,EAAE,IAAI,KAAK,SAAS,EAAE,CAAC;YAC5B,MAAM,OAAO,GAAI,KAAK,CAAC,MAA2B,CAAC,OAAO,CAAC;YAC3D,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE,OAAO,CAAC,CAAC;QAClC,CAAC;aAAM,CAAC;YACN,MAAM,MAAM,GAAI,KAAK,CAAC,MAA2B,CAAC;YAClD,IAAI,MAAM,IAAI,OAAO,MAAM,CAAC,KAAK,KAAK,QAAQ,EAAE,CAAC;gBAC/C,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE,MAAM,CAAC,KAAK,CAAC,CAAC;YACvC,CAAC;QACH,CAAC;IACH,CAAC;IAED,QAAQ,CAAC,GAAM;QACb,MAAM,EAAE,GAAG,GAAG,CAAC,IAAI,CAAC,CAAC;QACrB,IAAI,OAAO,EAAE,KAAK,QAAQ,EAAE,CAAC;YAC3B,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,iCAAiC,CAAC,CAAC;YACxD,OAAO;QACT,CAAC;QACD,IAAI,OAAO,CAAC,iDAAiD,CAAC,EAAE,CAAC;YAC/D,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC;gBACf,EAAE;gBACF,IAAI,EAAE,CAAC,OAAgB,EAAE,EAAE;oBACzB,IAAI,OAAO;wBAAE,IAAI,CAAC,UAAU,EAAE,CAAC;gBACjC,CAAC;aACF,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACxB,gEAAgE;QAChE,MAAM,GAAG,GAAe,EAAE,CAAC;QAC3B,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,KAAK,KAAK,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;YAC7D,GAA+B,CAAC,GAAG,CAAC,QAAQ,CAAC,GAAG,EAAE,CAAC;QACtD,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IACvB,CAAC;IAED,YAAY;QACV,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QACzB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACxB,CAAC;IAED,YAAY,CAAC,GAAW,EAAE,KAAc;QACtC,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;QAC9B,IAAI,CAAC,OAAO;YAAE,OAAO;QACrB,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,GAAG,OAAO,EAAE,CAAC,GAAG,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;IAChD,CAAC;IAED,UAAU;QACR,MAAM,GAAG,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC;QAC1B,IAAI,CAAC,GAAG;YAAE,OAAO;QACjB,MAAM,cAAc,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,KAAK,KAAK,CAAC,CAAC;QACzE,MAAM,SAAS,GAAG,cAAc,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;YACnD,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,GAAG,CAAC,CAAC;YAClE,OAAO,GAAG,CAAC;QACb,CAAC,EAAE,EAA6B,CAAe,CAAC;QAChD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC;YACf,GAAG,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,OAAgB,EAAE,EAAE;gBACzC,IAAI,OAAO;oBAAE,IAAI,CAAC,YAAY,EAAE,CAAC;YACnC,CAAC;SACF,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACH;;;OAGG;IACK,gBAAgB,CAAC,KAAc,EAAE,GAAmB;QAC1D,IAAI,GAAG,CAAC,IAAI,KAAK,SAAS;YAAE,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;QACzD,IAAI,GAAG,CAAC,IAAI,KAAK,QAAQ;YAAE,OAAO,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QACvD,IAAI,GAAG,CAAC,IAAI,KAAK,QAAQ;YAAE,OAAO,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;QAE5D,yCAAyC;QACzC,MAAM,MAAM,GAAG,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,CAAC;QAC9B,IAAI,MAAM,IAAI,GAAG,CAAC,QAAQ,IAAI,MAAM,EAAE,CAAC;YACrC,MAAM,WAAW,GAAG,MAAM,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;YACzC,IAAI,OAAO,WAAW,KAAK,SAAS;gBAAE,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YACnE,IAAI,OAAO,WAAW,KAAK,QAAQ;gBAAE,OAAO,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACjE,IAAI,OAAO,WAAW,KAAK,QAAQ;gBAAE,OAAO,IAAI,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;QACxE,CAAC;QACD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,sDAAsD;IAC9C,SAAS,CAAC,KAAc;QAC9B,IAAI,OAAO,KAAK,KAAK,SAAS;YAAE,OAAO,KAAK,CAAC;QAC7C,IAAI,OAAO,KAAK,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAC,IAAI,EAAE,CAAC,WAAW,EAAE,KAAK,MAAM,IAAI,KAAK,CAAC,IAAI,EAAE,KAAK,GAAG,CAAC;QACpG,IAAI,OAAO,KAAK,KAAK,QAAQ;YAAE,OAAO,KAAK,KAAK,CAAC,CAAC;QAClD,OAAO,KAAK,CAAC;IACf,CAAC;IAED,qDAAqD;IAC7C,QAAQ,CAAC,KAAc;QAC7B,IAAI,OAAO,KAAK,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAC;QAC5C,IAAI,OAAO,KAAK,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAC,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC;QACtF,IAAI,OAAO,KAAK,KAAK,SAAS;YAAE,OAAO,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;QACrD,OAAO,SAAS,CAAC;IACnB,CAAC;IAED,iFAAiF;IACzE,aAAa,CAAC,KAAc;QAClC,IAAI,KAAK,IAAI,IAAI;YAAE,OAAO,EAAE,CAAC;QAC7B,IAAI,OAAO,KAAK,KAAK,QAAQ;YAAE,OAAO,KAAK,CAAC;QAC5C,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,OAAO,KAAK,KAAK,SAAS;YAAE,OAAO,MAAM,CAAC,KAAK,CAAC,CAAC;QAClF,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,IAAI,CAAC;gBACH,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YAC/B,CAAC;YAAC,MAAM,CAAC;gBACP,OAAO,iBAAiB,CAAC;YAC3B,CAAC;QACH,CAAC;QACD,+DAA+D;QAC/D,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,MAAM,CAAC,GAAM;QACX,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,IAAI,CAAW,CAAC,CAAC;QACtC,MAAM,IAAI,GAAe,EAAE,CAAC;QAC5B,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,KAAK,KAAK,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE;YAC7D,IAAgC,CAAC,GAAG,CAAC,QAAQ,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QACtE,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACzB,CAAC;IAED,UAAU;QACR,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACvB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;IACzB,CAAC;IAED,WAAW,CAAC,KAAY,EAAE,GAAW,EAAE,GAAoB;QACzD,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;QAC/B,IAAI,CAAC,OAAO;YAAE,OAAO;QACrB,IAAI,GAAG,EAAE,IAAI,KAAK,SAAS,EAAE,CAAC;YAC5B,MAAM,OAAO,GAAI,KAAK,CAAC,MAA2B,CAAC,OAAO,CAAC;YAC3D,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,GAAG,OAAO,EAAE,CAAC,GAAG,CAAC,EAAE,OAAO,EAAE,CAAC,CAAC;QACnD,CAAC;aAAM,CAAC;YACN,MAAM,MAAM,GAAI,KAAK,CAAC,MAA2B,CAAC;YAClD,IAAI,MAAM,IAAI,OAAO,MAAM,CAAC,KAAK,KAAK,QAAQ,EAAE,CAAC;gBAC/C,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,EAAE,GAAG,OAAO,EAAE,CAAC,GAAG,CAAC,EAAE,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC;YACxD,CAAC;QACH,CAAC;IACH,CAAC;IAED,QAAQ,CAAC,GAAM;QACb,MAAM,EAAE,GAAG,GAAG,CAAC,IAAI,CAAC,CAAC;QACrB,IAAI,OAAO,EAAE,KAAK,QAAQ,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YAAE,OAAO;QACtD,MAAM,cAAc,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,OAAO,KAAK,KAAK,CAAC,CAAC;QACzE,MAAM,SAAS,GAAG,cAAc,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,GAAG,EAAE,EAAE;YACnD,GAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,GAAG,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,OAAO,EAAG,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,GAAG,CAAC,CAAC;YAC9E,OAAO,GAAG,CAAC;QACb,CAAC,EAAE,EAA6B,CAAe,CAAC;QAChD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC;YACf,EAAE,EAAE,OAAO,EAAE,SAAS,EAAE,IAAI,EAAE,CAAC,OAAgB,EAAE,EAAE;gBACjD,IAAI,OAAO;oBAAE,IAAI,CAAC,UAAU,EAAE,CAAC;YACjC,CAAC;SACF,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,SAAS,CAAC,KAAa,EAAE,GAAM;QAC7B,MAAM,EAAE,GAAG,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,IAAI,IAAI,GAAG,CAAC,CAAC,CAAE,GAA+B,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAC9G,OAAO,CAAC,OAAO,EAAE,KAAK,QAAQ,IAAI,OAAO,EAAE,KAAK,QAAQ,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC;IACzE,CAAC;wGA3MU,iBAAiB;4FAAjB,iBAAiB,kNClB9B,2rHAqGM,ywCDvFM,gBAAgB;;4FAIf,iBAAiB;kBAP7B,SAAS;+BACE,WAAW,cACT,IAAI,WACP,CAAC,gBAAgB,CAAC;8BAKlB,KAAK;sBAAb,KAAK;gBACG,OAAO;sBAAf,KAAK;gBACI,MAAM;sBAAf,MAAM;gBACG,MAAM;sBAAf,MAAM;gBACG,MAAM;sBAAf,MAAM;gBACG,UAAU;sBAAnB,MAAM","sourcesContent":["import { Component, Input, Output, signal, EventEmitter, OnInit, OnDestroy } from '@angular/core';\r\nimport { Observable, Subscription } from 'rxjs';\r\nimport { C80IconComponent } from '../icon/icon.component';\r\n\r\nexport interface C80TableColDef {\r\n  accessor: string;\r\n  label: string;\r\n  visible?: boolean; // Si no se especifica, se asume true\r\n  type?: 'string' | 'number' | 'boolean'; // Tipo de dato para la columna\r\n}\r\n\r\n@Component({\r\n  selector: 'c80-table',\r\n  standalone: true,\r\n  imports: [C80IconComponent],\r\n  templateUrl: './table.component.html',\r\n  styleUrl: './table.component.scss',\r\n})\r\nexport class C80TableComponent<T extends Record<string, unknown>> implements OnInit, OnDestroy {\r\n  @Input() data$!: Observable<T[]>;\r\n  @Input() columns: C80TableColDef[] = [];\r\n  @Output() create = new EventEmitter<{ row: Partial<T>, done: (result: boolean) => void }>();\r\n  @Output() update = new EventEmitter<{ id: number, changes: Partial<T>, done: (result: boolean) => void }>();\r\n  @Output() delete = new EventEmitter<{ id: number, done: (result: boolean) => void }>();\r\n  @Output() errorEvent = new EventEmitter<string>();\r\n\r\n  readonly data = signal<T[]>([]);\r\n  readonly keys = signal<string[]>([]);\r\n  creating = signal(false);\r\n  newRow = signal<Partial<T> | null>(null);\r\n  readonly editing = signal<number | null>(null); // id of row being edited\r\n  readonly editRow = signal<Partial<T> | null>(null);\r\n\r\n  private dataSub?: Subscription;\r\n\r\n  ngOnInit() {\r\n    if (!this.data$) return;\r\n    this.dataSub = this.data$.subscribe({\r\n      next: (items) => {\r\n        this.data.set(items);\r\n        // Solo mostrar las columnas visibles\r\n        const visibleColumns = this.columns.filter(col => col?.visible !== false);\r\n        this.keys.set(visibleColumns.map(col => col.accessor));\r\n      },\r\n      error: (err) => this.errorEvent.emit(err?.message || 'Error al cargar datos')\r\n    });\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    this.dataSub?.unsubscribe();\r\n  }\r\n\r\n  onInput(event: Event, key: string, col?: C80TableColDef) {\r\n    if (col?.type === 'boolean') {\r\n      const checked = (event.target as HTMLInputElement).checked;\r\n      this.updateNewRow(key, checked);\r\n    } else {\r\n      const target = (event.target as HTMLInputElement);\r\n      if (target && typeof target.value === 'string') {\r\n        this.updateNewRow(key, target.value);\r\n      }\r\n    }\r\n  }\r\n\r\n  onDelete(row: T) {\r\n    const id = row['id'];\r\n    if (typeof id !== 'number') {\r\n      this.errorEvent.emit('No se puede borrar: id inválido');\r\n      return;\r\n    }\r\n    if (confirm('¿Está seguro de que desea borrar este elemento?')) {\r\n      this.delete.emit({\r\n        id,\r\n        done: (success: boolean) => {\r\n          if (success) this.cancelEdit();\r\n        }\r\n      });\r\n    }\r\n  }\r\n\r\n  startCreate() {\r\n    this.creating.set(true);\r\n    // Inicializa newRow solo con los accessors de columnas visibles\r\n    const row: Partial<T> = {};\r\n    this.columns.filter(col => col.visible !== false).forEach(col => {\r\n      (row as Record<string, unknown>)[col.accessor] = '';\r\n    });\r\n    this.newRow.set(row);\r\n  }\r\n\r\n  cancelCreate() {\r\n    this.creating.set(false);\r\n    this.newRow.set(null);\r\n  }\r\n\r\n  updateNewRow(key: string, value: unknown) {\r\n    const current = this.newRow();\r\n    if (!current) return;\r\n    this.newRow.set({ ...current, [key]: value });\r\n  }\r\n\r\n  saveCreate() {\r\n    const row = this.newRow();\r\n    if (!row) return;\r\n    const visibleColumns = this.columns.filter(col => col.visible !== false);\r\n    const converted = visibleColumns.reduce((acc, col) => {\r\n      acc[col.accessor] = this.convertCellValue(row[col.accessor], col);\r\n      return acc;\r\n    }, {} as Record<string, unknown>) as Partial<T>;\r\n    this.create.emit({\r\n      row: converted, done: (success: boolean) => {\r\n        if (success) this.cancelCreate();\r\n      }\r\n    });\r\n  }\r\n\r\n  /**\r\n   * Converts a cell value based on column type or sample data.\r\n   * Handles stringification of objects for string columns.\r\n   */\r\n  /**\r\n   * Converts a cell value based on column type or sample data.\r\n   * Delegates to type-specific helpers for clarity and maintainability.\r\n   */\r\n  private convertCellValue(value: unknown, col: C80TableColDef): unknown {\r\n    if (col.type === 'boolean') return this.toBoolean(value);\r\n    if (col.type === 'number') return this.toNumber(value);\r\n    if (col.type === 'string') return this.toStringValue(value);\r\n\r\n    // Fallback: use sample data if available\r\n    const sample = this.data()[0];\r\n    if (sample && col.accessor in sample) {\r\n      const sampleValue = sample[col.accessor];\r\n      if (typeof sampleValue === 'boolean') return this.toBoolean(value);\r\n      if (typeof sampleValue === 'number') return this.toNumber(value);\r\n      if (typeof sampleValue === 'string') return this.toStringValue(value);\r\n    }\r\n    return value;\r\n  }\r\n\r\n  /** Converts value to boolean using best practices. */\r\n  private toBoolean(value: unknown): boolean {\r\n    if (typeof value === 'boolean') return value;\r\n    if (typeof value === 'string') return value.trim().toLowerCase() === 'true' || value.trim() === '1';\r\n    if (typeof value === 'number') return value === 1;\r\n    return false;\r\n  }\r\n\r\n  /** Converts value to number using best practices. */\r\n  private toNumber(value: unknown): number | undefined {\r\n    if (typeof value === 'number') return value;\r\n    if (typeof value === 'string') return value.trim() === '' ? undefined : Number(value);\r\n    if (typeof value === 'boolean') return value ? 1 : 0;\r\n    return undefined;\r\n  }\r\n\r\n  /** Converts value to string using best practices, always stringifies objects. */\r\n  private toStringValue(value: unknown): string {\r\n    if (value == null) return '';\r\n    if (typeof value === 'string') return value;\r\n    if (typeof value === 'number' || typeof value === 'boolean') return String(value);\r\n    if (typeof value === 'object') {\r\n      try {\r\n        return JSON.stringify(value);\r\n      } catch {\r\n        return '[object Object]';\r\n      }\r\n    }\r\n    // For functions, symbols, undefined, etc., return empty string\r\n    return '';\r\n  }\r\n\r\n  onEdit(row: T) {\r\n    this.editing.set(row['id'] as number);\r\n    const edit: Partial<T> = {};\r\n    this.columns.filter(col => col.visible !== false).forEach(col => {\r\n      (edit as Record<string, unknown>)[col.accessor] = row[col.accessor];\r\n    });\r\n    this.editRow.set(edit);\r\n  }\r\n\r\n  cancelEdit() {\r\n    this.editing.set(null);\r\n    this.editRow.set(null);\r\n  }\r\n\r\n  onEditInput(event: Event, key: string, col?: C80TableColDef) {\r\n    const current = this.editRow();\r\n    if (!current) return;\r\n    if (col?.type === 'boolean') {\r\n      const checked = (event.target as HTMLInputElement).checked;\r\n      this.editRow.set({ ...current, [key]: checked });\r\n    } else {\r\n      const target = (event.target as HTMLInputElement);\r\n      if (target && typeof target.value === 'string') {\r\n        this.editRow.set({ ...current, [key]: target.value });\r\n      }\r\n    }\r\n  }\r\n\r\n  saveEdit(row: T) {\r\n    const id = row['id'];\r\n    if (typeof id !== 'number' || !this.editRow()) return;\r\n    const visibleColumns = this.columns.filter(col => col.visible !== false);\r\n    const converted = visibleColumns.reduce((acc, col) => {\r\n      acc[col.accessor] = this.convertCellValue(this.editRow()![col.accessor], col);\r\n      return acc;\r\n    }, {} as Record<string, unknown>) as Partial<T>;\r\n    this.update.emit({\r\n      id, changes: converted, done: (success: boolean) => {\r\n        if (success) this.cancelEdit();\r\n      }\r\n    });\r\n  }\r\n\r\n  /**\r\n   * TrackBy function for ngFor to avoid DOM re-creation (NG0956 warning).\r\n   */\r\n  trackById(index: number, row: T): number | string {\r\n    const id = row && typeof row === 'object' && 'id' in row ? (row as Record<string, unknown>)['id'] : undefined;\r\n    return (typeof id === 'string' || typeof id === 'number') ? id : index;\r\n  }\r\n}\r\n","<div class=\"table-responsive\">\n  <table class=\"table table-bordered table-hover align-middle\">\n    <thead class=\"thead table-light\">\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <th class=\"text-center\">{{ col.label }}</th>\n        }\n        @else {\n        <th>{{ col.label }}</th>\n        }\n        }\n        }\n        <th class=\"table-actions-header\">\n          <span>Actions</span>\n          <c80-icon button icon=\"add\" [disabled]=\"creating()\" title=\"Agregar\" class=\"add-action-btn\"\n            (iconClick)=\"startCreate()\"></c80-icon>\n        </th>\n      </tr>\n    </thead>\n    <tbody>\n      @for (row of data(); track trackById(i, row); let i = $index) {\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center\">\n          @if (editing() === row['id']) {\n          <input type=\"checkbox\" [checked]=\"!!editRow()?.[col.accessor]\"\n            (change)=\"onEditInput($event, col.accessor, col)\" [attr.aria-label]=\"col.label\" />\n          }\n          @else {\n          @if (row[col.accessor] === true) {\n          <c80-icon icon=\"check\" [size]=\"1\"></c80-icon>\n          <br />\n          }\n          @else if (row[col.accessor] === false) {\n          <c80-icon icon=\"cancel\"></c80-icon>\n          <br />\n          }\n          }\n        </td>\n        }\n        @else {\n        <td>\n          @if (editing() === row['id']) {\n          <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n            [value]=\"editRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n            (input)=\"onEditInput($event, col.accessor, col)\" />\n          }\n          @else {\n          {{ row[col.accessor] }}\n          }\n        </td>\n        }\n        }\n        }\n        <td class=\"text-center\">\n          @if (editing() === row['id']) {\n          <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveEdit(row)\"></c80-icon>\n          <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelEdit()\"></c80-icon>\n          }\n          @else {\n          <c80-icon button icon=\"edit\" title=\"Editar\" (iconClick)=\"onEdit(row)\"></c80-icon>\n          <c80-icon button icon=\"delete\" color=\"warn\" title=\"Borrar\" (iconClick)=\"onDelete(row)\"></c80-icon>\n          }\n        </td>\n      </tr>\n      }\n      @if (creating()) {\n      <tr>\n        @for (col of columns; track col) {\n        @if (col.visible !== false) {\n        @if (col.type === 'boolean') {\n        <td class=\"text-center\">\n          <input type=\"checkbox\" [checked]=\"!!newRow()?.[col.accessor]\" (change)=\"onInput($event, col.accessor, col)\"\n            [attr.aria-label]=\"col.label\" />\n        </td>\n        } @else {\n        <td>\n          <input class=\"form-control form-control-sm\" [type]=\"col.type === 'number' ? 'number' : 'text'\"\n            [value]=\"newRow()?.[col.accessor] ?? ''\" [placeholder]=\"col.label\"\n            (input)=\"onInput($event, col.accessor, col)\" />\n        </td>\n        }\n        }\n        }\n        <td class=\"text-center\">\n          <c80-icon button icon=\"check\" title=\"Guardar\" (iconClick)=\"saveCreate()\"></c80-icon>\n          <c80-icon button icon=\"cancel\" color=\"warn\" title=\"Cancelar\" (iconClick)=\"cancelCreate()\"></c80-icon>\n        </td>\n      </tr>\n      }\n    </tbody>\n  </table>\n  @if (data().length === 0) {\n  <div class=\"text-center text-muted py-3\">\n    No hay datos para mostrar.\n  </div>\n  }\n</div>"]}
|
|
@@ -10,12 +10,19 @@ export interface C80TableColDef {
|
|
|
10
10
|
export declare class C80TableComponent<T extends Record<string, unknown>> implements OnInit, OnDestroy {
|
|
11
11
|
data$: Observable<T[]>;
|
|
12
12
|
columns: C80TableColDef[];
|
|
13
|
-
create: EventEmitter<
|
|
13
|
+
create: EventEmitter<{
|
|
14
|
+
row: Partial<T>;
|
|
15
|
+
done: (result: boolean) => void;
|
|
16
|
+
}>;
|
|
14
17
|
update: EventEmitter<{
|
|
15
18
|
id: number;
|
|
16
19
|
changes: Partial<T>;
|
|
20
|
+
done: (result: boolean) => void;
|
|
21
|
+
}>;
|
|
22
|
+
delete: EventEmitter<{
|
|
23
|
+
id: number;
|
|
24
|
+
done: (result: boolean) => void;
|
|
17
25
|
}>;
|
|
18
|
-
delete: EventEmitter<number>;
|
|
19
26
|
errorEvent: EventEmitter<string>;
|
|
20
27
|
readonly data: import("@angular/core").WritableSignal<T[]>;
|
|
21
28
|
readonly keys: import("@angular/core").WritableSignal<string[]>;
|