@c80/ui 1.0.41 → 1.0.44

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.
@@ -1,8 +1,9 @@
1
1
  export * from './table.types';
2
2
  export * from './table.component';
3
+ export * from './table.utils';
3
4
  export * from './table-data-converter.service';
4
5
  export * from './table-column-visibility.service';
5
6
  export * from './table-data-utils.service';
6
7
  export * from './table-selection.service';
7
8
  export * from './table-crud-state.service';
8
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3VpL3NyYy9saWIvdGFibGUvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxtQkFBbUIsQ0FBQztBQUNsQyxjQUFjLGdDQUFnQyxDQUFDO0FBQy9DLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYyw0QkFBNEIsQ0FBQztBQUMzQyxjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3RhYmxlLnR5cGVzJztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUtZGF0YS1jb252ZXJ0ZXIuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLWNvbHVtbi12aXNpYmlsaXR5LnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS1kYXRhLXV0aWxzLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS1zZWxlY3Rpb24uc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLWNydWQtc3RhdGUuc2VydmljZSc7XG4iXX0=
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3VpL3NyYy9saWIvdGFibGUvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxtQkFBbUIsQ0FBQztBQUNsQyxjQUFjLGVBQWUsQ0FBQztBQUM5QixjQUFjLGdDQUFnQyxDQUFDO0FBQy9DLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYyw0QkFBNEIsQ0FBQztBQUMzQyxjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3RhYmxlLnR5cGVzJztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUudXRpbHMnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS1kYXRhLWNvbnZlcnRlci5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUtY29sdW1uLXZpc2liaWxpdHkuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLWRhdGEtdXRpbHMuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLXNlbGVjdGlvbi5zZXJ2aWNlJztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUtY3J1ZC1zdGF0ZS5zZXJ2aWNlJztcbiJdfQ==
@@ -1,5 +1,6 @@
1
1
  import { Injectable, inject } from '@angular/core';
2
2
  import { TableDataConverterService } from './table-data-converter.service';
3
+ import { TableDataUtilsService } from './table-data-utils.service';
3
4
  import * as i0 from "@angular/core";
4
5
  /**
5
6
  * Servicio para gestionar la lógica de visibilidad de columnas en tablas C80
@@ -11,6 +12,7 @@ import * as i0 from "@angular/core";
11
12
  */
12
13
  export class TableColumnVisibilityService {
13
14
  dataConverter = inject(TableDataConverterService);
15
+ dataUtils = inject(TableDataUtilsService);
14
16
  /**
15
17
  * Actualiza las keys de columnas visibles basándose en el estado actual
16
18
  * @param columns - Definiciones de columnas
@@ -25,125 +27,72 @@ export class TableColumnVisibilityService {
25
27
  }
26
28
  /**
27
29
  * Determina si una columna debe ser visible basándose en su configuración y datos
30
+ * Método unificado que maneja todos los casos de visibilidad
28
31
  * @param column - La definición de la columna
29
32
  * @param data - Datos actuales de la tabla
30
- * @param forceShowInCreation - Si es true, ignora hideIfAllValuesAreNull (usado en modo creación)
33
+ * @param options - Opciones de contexto (creación, edición, fila específica)
31
34
  * @returns true si la columna debe ser visible
32
35
  */
33
- isColumnVisible(column, data, forceShowInCreation = false) {
36
+ isColumnVisible(column, data, options = {}) {
34
37
  // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar
35
38
  if (column.visible === false) {
36
39
  return false;
37
40
  }
38
- // PRIORIDAD 2: Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
39
- // EXCEPCIÓN: En modo creación, mostramos estas columnas para permitir entrada de datos
40
- if (column.hideIfAllValuesAreNull === true && !forceShowInCreation && this.areAllColumnValuesEmpty(column, data)) {
41
- return false;
42
- }
43
- // Por defecto: mostrar
44
- return true;
45
- }
46
- /**
47
- * Determina si una columna debe ser visible en los headers
48
- * @param column - La definición de la columna
49
- * @param data - Datos actuales de la tabla
50
- * @param creating - Si está en modo creación
51
- * @param editing - ID de fila en edición (null si no hay edición)
52
- * @returns true si el header debe ser visible
53
- */
54
- isColumnVisibleInHeader(column, data, creating, editing) {
55
- // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar
56
- if (column.visible === false) {
57
- return false;
58
- }
59
- // Si estamos en modo creación, mostrar columnas con hideIfAllValuesAreNull
60
- if (creating && column.hideIfAllValuesAreNull === true) {
41
+ // Si no tiene hideIfAllValuesAreNull, mostrar siempre
42
+ if (!column.hideIfAllValuesAreNull) {
61
43
  return true;
62
44
  }
63
- // Si hay una fila en modo edición y la columna tiene hideIfAllValuesAreNull
64
- if (editing !== null && column.hideIfAllValuesAreNull === true) {
65
- // Buscar la fila que se está editando
66
- const editingRow = data.find(row => row['id'] === editing);
67
- if (editingRow) {
68
- // Solo mostrar el header si la fila en edición tiene valor en esta columna
69
- const cellValue = this.getCellValue(editingRow, column.accessor);
45
+ // PRIORIDAD 2: Para columnas con hideIfAllValuesAreNull, ocultar si el valor es falsy (excepto 0)
46
+ // Esto aplica tanto en modo creación como edición
47
+ // CONTEXTO: Modo creación
48
+ if (options.creating) {
49
+ // En creación, verificar el valor de la fila en creación si existe
50
+ const creatingRow = options.row;
51
+ if (creatingRow) {
52
+ const cellValue = this.dataUtils.getCellValue(creatingRow, column.accessor);
70
53
  return !this.dataConverter.isValueEmpty(cellValue);
71
54
  }
72
- // Si no se encuentra la fila, no mostrar el header
55
+ // Si no hay fila, ocultar (valor vacío por defecto)
73
56
  return false;
74
57
  }
75
- // Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
76
- if (column.hideIfAllValuesAreNull === true && this.areAllColumnValuesEmpty(column, data)) {
58
+ // CONTEXTO: Fila en edición
59
+ if (options.editing !== undefined && options.editing !== null) {
60
+ const editingRow = options.row ?? data.find(r => r['id'] === options.editing);
61
+ if (editingRow) {
62
+ const cellValue = this.dataUtils.getCellValue(editingRow, column.accessor);
63
+ return !this.dataConverter.isValueEmpty(cellValue);
64
+ }
77
65
  return false;
78
66
  }
79
- // Por defecto: mostrar
80
- return true;
67
+ // CONTEXTO: Vista normal - verificar si todos los valores están vacíos
68
+ return !this.areAllColumnValuesEmpty(column, data);
81
69
  }
82
70
  /**
83
- * Determina si una columna debe ser visible en una fila específica considerando el modo edición
84
- * @param column - La definición de la columna
85
- * @param row - La fila actual
86
- * @param data - Datos actuales de la tabla
87
- * @param editing - ID de fila en edición (null si no hay edición)
88
- * @returns true si la columna debe ser visible para esta fila
71
+ * Determina si una columna debe ser visible en los headers
72
+ * Wrapper para mantener compatibilidad con API existente
73
+ */
74
+ isColumnVisibleInHeader(column, data, creating, editing) {
75
+ return this.isColumnVisible(column, data, { creating, editing });
76
+ }
77
+ /**
78
+ * Determina si una columna debe ser visible en una fila específica
79
+ * Wrapper para mantener compatibilidad con API existente
89
80
  */
90
81
  isColumnVisibleForRow(column, row, data, editing) {
91
- // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar
92
- if (column.visible === false) {
93
- return false;
94
- }
95
- // PRIORIDAD 2: Si esta fila específica está en modo edición y la columna tiene hideIfAllValuesAreNull
96
- const isEditingThisRow = editing === row['id'];
97
- if (isEditingThisRow && column.hideIfAllValuesAreNull === true) {
98
- // En edición, solo mostrar si esta fila específica tiene valor en esta columna
99
- const cellValue = this.getCellValue(row, column.accessor);
100
- return !this.dataConverter.isValueEmpty(cellValue);
101
- }
102
- // PRIORIDAD 3: Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
103
- if (column.hideIfAllValuesAreNull === true && this.areAllColumnValuesEmpty(column, data)) {
104
- return false;
105
- }
106
- // Por defecto: mostrar
107
- return true;
82
+ return this.isColumnVisible(column, data, { editing, row });
108
83
  }
109
84
  /**
110
85
  * Verifica si todos los valores de una columna están vacíos/nulos
111
- * @param column - La definición de la columna
112
- * @param data - Datos actuales de la tabla
113
- * @returns true si todos los valores están vacíos
114
86
  */
115
87
  areAllColumnValuesEmpty(column, data) {
116
88
  if (data.length === 0) {
117
- return true; // Si no hay datos, consideramos la columna como vacía
89
+ return true;
118
90
  }
119
91
  return data.every(row => {
120
- const value = this.getCellValue(row, column.accessor);
92
+ const value = this.dataUtils.getCellValue(row, column.accessor);
121
93
  return this.dataConverter.isValueEmpty(value);
122
94
  });
123
95
  }
124
- /**
125
- * Obtiene el valor de una celda usando el accessor, soportando notación de punto para propiedades anidadas
126
- * @param row - La fila de datos
127
- * @param accessor - El accessor de la columna
128
- * @returns El valor de la celda
129
- */
130
- getCellValue(row, accessor) {
131
- if (accessor.includes('.')) {
132
- return this.getNestedValue(row, accessor);
133
- }
134
- return row[accessor];
135
- }
136
- /**
137
- * Obtiene el valor de un objeto usando notación de punto (ej: 'task.name')
138
- * @param obj - Objeto del cual obtener el valor
139
- * @param accessor - Ruta de acceso usando notación de punto
140
- * @returns El valor anidado o undefined
141
- */
142
- getNestedValue(obj, accessor) {
143
- return accessor.split('.').reduce((current, key) => {
144
- return current && typeof current === 'object' ? current[key] : undefined;
145
- }, obj);
146
- }
147
96
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableColumnVisibilityService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
148
97
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableColumnVisibilityService, providedIn: 'root' });
149
98
  }
@@ -153,4 +102,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
153
102
  providedIn: 'root'
154
103
  }]
155
104
  }] });
156
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-column-visibility.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-column-visibility.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { C80TableColDef, ID } from './table.types';\nimport { TableDataConverterService } from './table-data-converter.service';\n\n/**\n * Servicio para gestionar la lógica de visibilidad de columnas en tablas C80\n *\n * Maneja las reglas complejas de visibilidad basadas en:\n * - Configuración explícita (visible: false)\n * - Ocultación automática (hideIfAllValuesAreNull)\n * - Estado de creación y edición\n */\n@Injectable({\n    providedIn: 'root'\n})\nexport class TableColumnVisibilityService {\n\n    private readonly dataConverter = inject(TableDataConverterService);\n\n    /**\n     * Actualiza las keys de columnas visibles basándose en el estado actual\n     * @param columns - Definiciones de columnas\n     * @param data - Datos actuales de la tabla\n     * @param creating - Si está en modo creación\n     * @param editing - ID de fila en edición (null si no hay edición)\n     * @returns Array de accessors de columnas visibles\n     */\n    updateVisibleKeys<T extends Record<string, unknown>>(\n        columns: C80TableColDef[],\n        data: T[],\n        creating: boolean,\n        editing: ID | null\n    ): string[] {\n        const visibleColumns = columns.filter(col =>\n            this.isColumnVisibleInHeader(col, data, creating, editing)\n        );\n        return visibleColumns.map(col => col.accessor);\n    }\n\n    /**\n     * Determina si una columna debe ser visible basándose en su configuración y datos\n     * @param column - La definición de la columna\n     * @param data - Datos actuales de la tabla\n     * @param forceShowInCreation - Si es true, ignora hideIfAllValuesAreNull (usado en modo creación)\n     * @returns true si la columna debe ser visible\n     */\n    isColumnVisible<T extends Record<string, unknown>>(\n        column: C80TableColDef,\n        data: T[],\n        forceShowInCreation = false\n    ): 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, data)) {\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     * @param column - La definición de la columna\n     * @param data - Datos actuales de la tabla\n     * @param creating - Si está en modo creación\n     * @param editing - ID de fila en edición (null si no hay edición)\n     * @returns true si el header debe ser visible\n     */\n    isColumnVisibleInHeader<T extends Record<string, unknown>>(\n        column: C80TableColDef,\n        data: T[],\n        creating: boolean,\n        editing: ID | null\n    ): 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 (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 (editing !== null && column.hideIfAllValuesAreNull === true) {\n            // Buscar la fila que se está editando\n            const editingRow = data.find(row => row['id'] === editing);\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.dataConverter.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, data)) {\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     * @param column - La definición de la columna\n     * @param row - La fila actual\n     * @param data - Datos actuales de la tabla\n     * @param editing - ID de fila en edición (null si no hay edición)\n     * @returns true si la columna debe ser visible para esta fila\n     */\n    isColumnVisibleForRow<T extends Record<string, unknown>>(\n        column: C80TableColDef,\n        row: T,\n        data: T[],\n        editing: ID | null\n    ): 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 = 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.dataConverter.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, data)) {\n            return false;\n        }\n\n        // Por defecto: mostrar\n        return true;\n    }\n\n    /**\n     * Verifica si todos los valores de una columna están vacíos/nulos\n     * @param column - La definición de la columna\n     * @param data - Datos actuales de la tabla\n     * @returns true si todos los valores están vacíos\n     */\n    areAllColumnValuesEmpty<T extends Record<string, unknown>>(\n        column: C80TableColDef,\n        data: T[]\n    ): boolean {\n        if (data.length === 0) {\n            return true; // Si no hay datos, consideramos la columna como vacía\n        }\n\n        return data.every(row => {\n            const value = this.getCellValue(row, column.accessor);\n            return this.dataConverter.isValueEmpty(value);\n        });\n    }\n\n    /**\n     * Obtiene el valor de una celda usando el accessor, soportando notación de punto para propiedades anidadas\n     * @param row - La fila de datos\n     * @param accessor - El accessor de la columna\n     * @returns El valor de la celda\n     */\n    private getCellValue<T extends Record<string, unknown>>(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     * Obtiene el valor de un objeto usando notación de punto (ej: 'task.name')\n     * @param obj - Objeto del cual obtener el valor\n     * @param accessor - Ruta de acceso usando notación de punto\n     * @returns El valor anidado o undefined\n     */\n    private getNestedValue(obj: Record<string, unknown>, accessor: string): unknown {\n        return accessor.split('.').reduce((current, key) => {\n            return current && typeof current === 'object' ? (current as Record<string, unknown>)[key] : undefined;\n        }, obj as unknown);\n    }\n}\n"]}
105
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-column-visibility.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-column-visibility.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, inject } from '@angular/core';\nimport { C80TableColDef } from './table.types';\nimport { TableDataConverterService } from './table-data-converter.service';\nimport { TableDataUtilsService } from './table-data-utils.service';\n\n/**\n * Servicio para gestionar la lógica de visibilidad de columnas en tablas C80\n *\n * Maneja las reglas complejas de visibilidad basadas en:\n * - Configuración explícita (visible: false)\n * - Ocultación automática (hideIfAllValuesAreNull)\n * - Estado de creación y edición\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class TableColumnVisibilityService {\n\n  private readonly dataConverter = inject(TableDataConverterService);\n  private readonly dataUtils = inject(TableDataUtilsService);\n\n  /**\n   * Actualiza las keys de columnas visibles basándose en el estado actual\n   * @param columns - Definiciones de columnas\n   * @param data - Datos actuales de la tabla\n   * @param creating - Si está en modo creación\n   * @param editing - ID de fila en edición (null si no hay edición)\n   * @returns Array de accessors de columnas visibles\n   */\n  updateVisibleKeys<T extends Record<string, unknown>>(\n    columns: C80TableColDef[],\n    data: T[],\n    creating: boolean,\n    editing: string | null\n  ): string[] {\n    const visibleColumns = columns.filter(col =>\n      this.isColumnVisibleInHeader(col, data, creating, editing)\n    );\n    return visibleColumns.map(col => col.accessor);\n  }\n\n  /**\n   * Determina si una columna debe ser visible basándose en su configuración y datos\n   * Método unificado que maneja todos los casos de visibilidad\n   * @param column - La definición de la columna\n   * @param data - Datos actuales de la tabla\n   * @param options - Opciones de contexto (creación, edición, fila específica)\n   * @returns true si la columna debe ser visible\n   */\n  isColumnVisible<T extends Record<string, unknown>>(\n    column: C80TableColDef,\n    data: T[],\n    options: {\n      forceShowInCreation?: boolean;\n      creating?: boolean;\n      editing?: string | null;\n      row?: T;\n    } = {}\n  ): 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 no tiene hideIfAllValuesAreNull, mostrar siempre\n    if (!column.hideIfAllValuesAreNull) {\n      return true;\n    }\n\n    // PRIORIDAD 2: Para columnas con hideIfAllValuesAreNull, ocultar si el valor es falsy (excepto 0)\n    // Esto aplica tanto en modo creación como edición\n\n    // CONTEXTO: Modo creación\n    if (options.creating) {\n      // En creación, verificar el valor de la fila en creación si existe\n      const creatingRow = options.row;\n      if (creatingRow) {\n        const cellValue = this.dataUtils.getCellValue(creatingRow, column.accessor);\n        return !this.dataConverter.isValueEmpty(cellValue);\n      }\n      // Si no hay fila, ocultar (valor vacío por defecto)\n      return false;\n    }\n\n    // CONTEXTO: Fila en edición\n    if (options.editing !== undefined && options.editing !== null) {\n      const editingRow = options.row ?? data.find(r => r['id'] === options.editing);\n      if (editingRow) {\n        const cellValue = this.dataUtils.getCellValue(editingRow, column.accessor);\n        return !this.dataConverter.isValueEmpty(cellValue);\n      }\n      return false;\n    }\n\n    // CONTEXTO: Vista normal - verificar si todos los valores están vacíos\n    return !this.areAllColumnValuesEmpty(column, data);\n  }\n\n  /**\n   * Determina si una columna debe ser visible en los headers\n   * Wrapper para mantener compatibilidad con API existente\n   */\n  isColumnVisibleInHeader<T extends Record<string, unknown>>(\n    column: C80TableColDef,\n    data: T[],\n    creating: boolean,\n    editing: string | null\n  ): boolean {\n    return this.isColumnVisible(column, data, { creating, editing });\n  }\n\n  /**\n   * Determina si una columna debe ser visible en una fila específica\n   * Wrapper para mantener compatibilidad con API existente\n   */\n  isColumnVisibleForRow<T extends Record<string, unknown>>(\n    column: C80TableColDef,\n    row: T,\n    data: T[],\n    editing: string | null\n  ): boolean {\n    return this.isColumnVisible(column, data, { editing, row });\n  }\n\n  /**\n   * Verifica si todos los valores de una columna están vacíos/nulos\n   */\n  private areAllColumnValuesEmpty<T extends Record<string, unknown>>(\n    column: C80TableColDef,\n    data: T[]\n  ): boolean {\n    if (data.length === 0) {\n      return true;\n    }\n\n    return data.every(row => {\n      const value = this.dataUtils.getCellValue(row, column.accessor);\n      return this.dataConverter.isValueEmpty(value);\n    });\n  }\n}\n"]}
@@ -1,6 +1,6 @@
1
1
  import { Injectable, signal, inject } from '@angular/core';
2
2
  import { TableColumnVisibilityService } from './table-column-visibility.service';
3
- import { TableDataConverterService } from './table-data-converter.service';
3
+ import { TableDataUtilsService } from './table-data-utils.service';
4
4
  import * as i0 from "@angular/core";
5
5
  /**
6
6
  * Servicio para gestionar el estado CRUD (Crear, Leer, Actualizar, Eliminar) en tablas C80
@@ -9,28 +9,16 @@ import * as i0 from "@angular/core";
9
9
  * - Estado de creación y edición
10
10
  * - Gestión de datos temporales (newRow, editRow)
11
11
  * - Aplicación de valores dinámicos
12
- * - Validación y conversión de datos antes de emisión de eventos
12
+ * - Validación y conversión de datos
13
13
  */
14
14
  export class TableCrudStateService {
15
15
  visibilityService = inject(TableColumnVisibilityService);
16
- dataConverter = inject(TableDataConverterService);
16
+ dataUtils = inject(TableDataUtilsService);
17
17
  /**
18
18
  * Filtra columnas visibles y editables
19
19
  */
20
20
  getEditableColumns(columns, data, includeHiddenIfNull) {
21
- return columns.filter(col => this.visibilityService.isColumnVisible(col, data, includeHiddenIfNull) && !col.readOnly);
22
- }
23
- /**
24
- * Convierte valores de row según definición de columnas
25
- */
26
- convertRowValues(row, columns, data) {
27
- return columns.reduce((acc, col) => {
28
- const sampleValue = data.length > 0
29
- ? this.visibilityService['getCellValue'](data[0], col.accessor)
30
- : undefined;
31
- acc[col.accessor] = this.dataConverter.convertCellValue(row[col.accessor], col, sampleValue);
32
- return acc;
33
- }, {});
21
+ return columns.filter(col => this.visibilityService.isColumnVisible(col, data, { forceShowInCreation: includeHiddenIfNull }) && !col.readOnly);
34
22
  }
35
23
  /**
36
24
  * Inicializa row para creación con valores por defecto
@@ -49,7 +37,7 @@ export class TableCrudStateService {
49
37
  initializeEditRow(row, columns, data) {
50
38
  const edit = {};
51
39
  for (const col of this.getEditableColumns(columns, data, true)) {
52
- const value = this.visibilityService['getCellValue'](row, col.accessor);
40
+ const value = this.dataUtils.getCellValue(row, col.accessor);
53
41
  edit[col.accessor] = value;
54
42
  }
55
43
  return edit;
@@ -68,111 +56,52 @@ export class TableCrudStateService {
68
56
  newRow: newRow.asReadonly(),
69
57
  editing: editing.asReadonly(),
70
58
  editRow: editRow.asReadonly(),
71
- startCreate: this.createStartCreateFn(creating, newRow),
72
- cancelCreate: this.createCancelCreateFn(creating, newRow),
73
- updateNewRow: this.createUpdateNewRowFn(newRow),
74
- saveCreate: this.createSaveCreateFn(newRow, creating),
75
- startEdit: this.createStartEditFn(editing, editRow),
76
- cancelEdit: this.createCancelEditFn(editing, editRow),
77
- updateEditRow: this.createUpdateEditRowFn(editRow),
78
- saveEdit: this.createSaveEditFn(editRow, editing),
79
- applyInputValues: this.createApplyInputValuesFn(creating, newRow, editing, editRow),
59
+ startCreate: (columns, data) => this.startCreate(creating, newRow, columns, data),
60
+ cancelCreate: () => this.cancelCreate(creating, newRow),
61
+ updateNewRow: (key, value) => this.updateRow(newRow, key, value),
62
+ startEdit: (row, columns, data) => this.startEdit({ editing, editRow }, row, columns, data),
63
+ cancelEdit: () => this.cancelEdit(editing, editRow),
64
+ updateEditRow: (key, value) => this.updateRow(editRow, key, value),
65
+ applyInputValues: (partialValues) => this.applyInputValues({ creating, newRow, editing, editRow }, partialValues),
80
66
  };
81
67
  }
82
- createStartCreateFn(creating, newRow) {
83
- return (columns, data) => {
84
- creating.set(true);
85
- newRow.set(this.initializeNewRow(columns, data));
86
- };
68
+ startCreate(creating, newRow, columns, data) {
69
+ creating.set(true);
70
+ newRow.set(this.initializeNewRow(columns, data));
87
71
  }
88
- createCancelCreateFn(creating, newRow) {
89
- return () => {
90
- creating.set(false);
91
- newRow.set(null);
92
- };
72
+ cancelCreate(creating, newRow) {
73
+ creating.set(false);
74
+ newRow.set(null);
93
75
  }
94
- createUpdateNewRowFn(newRow) {
95
- return (key, value) => {
96
- const current = newRow();
97
- if (!current)
98
- return;
99
- newRow.set({ ...current, [key]: value });
100
- };
76
+ updateRow(rowSignal, key, value) {
77
+ const current = rowSignal();
78
+ if (!current)
79
+ return;
80
+ rowSignal.set({ ...current, [key]: value });
101
81
  }
102
- createSaveCreateFn(newRow, creating) {
103
- return (columns, data, createEmitter) => {
104
- const row = newRow();
105
- if (!row)
106
- return;
107
- const visibleColumns = this.getEditableColumns(columns, data, true);
108
- const converted = this.convertRowValues(row, visibleColumns, data);
109
- createEmitter.emit({
110
- row: converted,
111
- done: (success) => {
112
- if (success) {
113
- creating.set(false);
114
- newRow.set(null);
115
- }
116
- },
117
- });
118
- };
82
+ startEdit(signals, row, columns, data) {
83
+ signals.editing.set(row['id']);
84
+ signals.editRow.set(this.initializeEditRow(row, columns, data));
119
85
  }
120
- createStartEditFn(editing, editRow) {
121
- return (row, columns, data) => {
122
- editing.set(row['id']);
123
- editRow.set(this.initializeEditRow(row, columns, data));
124
- };
86
+ cancelEdit(editing, editRow) {
87
+ editing.set(null);
88
+ editRow.set(null);
125
89
  }
126
- createCancelEditFn(editing, editRow) {
127
- return () => {
128
- editing.set(null);
129
- editRow.set(null);
130
- };
131
- }
132
- createUpdateEditRowFn(editRow) {
133
- return (key, value) => {
134
- const current = editRow();
135
- if (!current)
136
- return;
137
- editRow.set({ ...current, [key]: value });
138
- };
139
- }
140
- createSaveEditFn(editRow, editing) {
141
- return (rowId, columns, data, updateEmitter) => {
142
- const currentEditRow = editRow();
143
- if (!currentEditRow)
144
- return;
145
- const visibleColumns = this.getEditableColumns(columns, data, true);
146
- const converted = this.convertRowValues(currentEditRow, visibleColumns, data);
147
- updateEmitter.emit({
148
- id: rowId,
149
- changes: converted,
150
- done: (success) => {
151
- if (success) {
152
- editing.set(null);
153
- editRow.set(null);
154
- }
155
- },
156
- });
157
- };
158
- }
159
- createApplyInputValuesFn(creating, newRow, editing, editRow) {
160
- return (partialValues) => {
161
- if (!creating() && editing() === null)
162
- return;
163
- if (creating()) {
164
- const currentRow = newRow();
165
- if (currentRow) {
166
- newRow.set({ ...currentRow, ...partialValues });
167
- }
90
+ applyInputValues(signals, partialValues) {
91
+ if (!signals.creating() && signals.editing() === null)
92
+ return;
93
+ if (signals.creating()) {
94
+ const currentRow = signals.newRow();
95
+ if (currentRow) {
96
+ signals.newRow.set({ ...currentRow, ...partialValues });
168
97
  }
169
- else {
170
- const currentEditRow = editRow();
171
- if (currentEditRow) {
172
- editRow.set({ ...currentEditRow, ...partialValues });
173
- }
98
+ }
99
+ else {
100
+ const currentEditRow = signals.editRow();
101
+ if (currentEditRow) {
102
+ signals.editRow.set({ ...currentEditRow, ...partialValues });
174
103
  }
175
- };
104
+ }
176
105
  }
177
106
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableCrudStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
178
107
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableCrudStateService, providedIn: 'root' });
@@ -183,4 +112,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
183
112
  providedIn: 'root'
184
113
  }]
185
114
  }] });
186
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-crud-state.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-crud-state.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, EventEmitter, inject } from '@angular/core';\nimport { C80TableColDef, ID } from './table.types';\nimport { TableColumnVisibilityService } from './table-column-visibility.service';\nimport { TableDataConverterService } from './table-data-converter.service';\n\n/**\n * Servicio para gestionar el estado CRUD (Crear, Leer, Actualizar, Eliminar) en tablas C80\n *\n * Maneja:\n * - Estado de creación y edición\n * - Gestión de datos temporales (newRow, editRow)\n * - Aplicación de valores dinámicos\n * - Validación y conversión de datos antes de emisión de eventos\n */\n@Injectable({\n    providedIn: 'root'\n})\nexport class TableCrudStateService {\n\n    private readonly visibilityService = inject(TableColumnVisibilityService);\n    private readonly dataConverter = inject(TableDataConverterService);\n\n    /**\n     * Filtra columnas visibles y editables\n     */\n    private getEditableColumns<T extends Record<string, unknown>>(\n        columns: C80TableColDef[],\n        data: T[],\n        includeHiddenIfNull: boolean\n    ): C80TableColDef[] {\n        return columns.filter(\n            col => this.visibilityService.isColumnVisible(col, data, includeHiddenIfNull) && !col.readOnly\n        );\n    }\n\n    /**\n     * Convierte valores de row según definición de columnas\n     */\n    private convertRowValues<T extends Record<string, unknown>>(\n        row: Partial<T>,\n        columns: C80TableColDef[],\n        data: T[]\n    ): Partial<T> {\n        return columns.reduce((acc, col) => {\n            const sampleValue = data.length > 0\n                ? this.visibilityService['getCellValue'](data[0], col.accessor)\n                : undefined;\n            acc[col.accessor] = this.dataConverter.convertCellValue(row[col.accessor], col, sampleValue);\n            return acc;\n        }, {} as Record<string, unknown>) as Partial<T>;\n    }\n\n    /**\n     * Inicializa row para creación con valores por defecto\n     */\n    private initializeNewRow<T extends Record<string, unknown>>(\n        columns: C80TableColDef[],\n        data: T[]\n    ): Partial<T> {\n        const row: Partial<T> = {};\n        for (const col of this.getEditableColumns(columns, data, true)) {\n            const defaultValue = col.default === undefined ? '' : col.default;\n            (row as Record<string, unknown>)[col.accessor] = defaultValue;\n        }\n        return row;\n    }\n\n    /**\n     * Inicializa row para edición con valores actuales\n     */\n    private initializeEditRow<T extends Record<string, unknown>>(\n        row: T,\n        columns: C80TableColDef[],\n        data: T[]\n    ): Partial<T> {\n        const edit: Partial<T> = {};\n        for (const col of this.getEditableColumns(columns, data, true)) {\n            const value = this.visibilityService['getCellValue'](row, col.accessor);\n            (edit as Record<string, unknown>)[col.accessor] = value;\n        }\n        return edit;\n    }\n\n    /**\n     * Inicializa el estado CRUD para una nueva tabla\n     * @returns Objeto con signals y métodos para gestión CRUD\n     */\n    createCrudState<T extends Record<string, unknown>>() {\n        const creating = signal(false);\n        const newRow = signal<Partial<T> | null>(null);\n        const editing = signal<ID | null>(null);\n        const editRow = signal<Partial<T> | null>(null);\n\n        return {\n            creating: creating.asReadonly(),\n            newRow: newRow.asReadonly(),\n            editing: editing.asReadonly(),\n            editRow: editRow.asReadonly(),\n            startCreate: this.createStartCreateFn(creating, newRow),\n            cancelCreate: this.createCancelCreateFn(creating, newRow),\n            updateNewRow: this.createUpdateNewRowFn(newRow),\n            saveCreate: this.createSaveCreateFn(newRow, creating),\n            startEdit: this.createStartEditFn(editing, editRow),\n            cancelEdit: this.createCancelEditFn(editing, editRow),\n            updateEditRow: this.createUpdateEditRowFn(editRow),\n            saveEdit: this.createSaveEditFn(editRow, editing),\n            applyInputValues: this.createApplyInputValuesFn(creating, newRow, editing, editRow),\n        };\n    }\n\n    private createStartCreateFn<T extends Record<string, unknown>>(\n        creating: ReturnType<typeof signal<boolean>>,\n        newRow: ReturnType<typeof signal<Partial<T> | null>>\n    ) {\n        return (columns: C80TableColDef[], data: T[]): void => {\n            creating.set(true);\n            newRow.set(this.initializeNewRow(columns, data));\n        };\n    }\n\n    private createCancelCreateFn<T extends Record<string, unknown>>(\n        creating: ReturnType<typeof signal<boolean>>,\n        newRow: ReturnType<typeof signal<Partial<T> | null>>\n    ) {\n        return (): void => {\n            creating.set(false);\n            newRow.set(null);\n        };\n    }\n\n    private createUpdateNewRowFn<T extends Record<string, unknown>>(\n        newRow: ReturnType<typeof signal<Partial<T> | null>>\n    ) {\n        return (key: string, value: unknown): void => {\n            const current = newRow();\n            if (!current) return;\n            newRow.set({ ...current, [key]: value });\n        };\n    }\n\n    private createSaveCreateFn<T extends Record<string, unknown>>(\n        newRow: ReturnType<typeof signal<Partial<T> | null>>,\n        creating: ReturnType<typeof signal<boolean>>\n    ) {\n        return (\n            columns: C80TableColDef[],\n            data: T[],\n            createEmitter: EventEmitter<{ row: Partial<T>; done: (result: boolean) => void; }>\n        ): void => {\n            const row = newRow();\n            if (!row) return;\n\n            const visibleColumns = this.getEditableColumns(columns, data, true);\n            const converted = this.convertRowValues(row, visibleColumns, data);\n\n            createEmitter.emit({\n                row: converted,\n                done: (success: boolean) => {\n                    if (success) {\n                        creating.set(false);\n                        newRow.set(null);\n                    }\n                },\n            });\n        };\n    }\n\n    private createStartEditFn<T extends Record<string, unknown>>(\n        editing: ReturnType<typeof signal<ID | null>>,\n        editRow: ReturnType<typeof signal<Partial<T> | null>>\n    ) {\n        return (row: T, columns: C80TableColDef[], data: T[]): void => {\n            editing.set(row['id'] as ID);\n            editRow.set(this.initializeEditRow(row, columns, data));\n        };\n    }\n\n    private createCancelEditFn<T extends Record<string, unknown>>(\n        editing: ReturnType<typeof signal<ID | null>>,\n        editRow: ReturnType<typeof signal<Partial<T> | null>>\n    ) {\n        return (): void => {\n            editing.set(null);\n            editRow.set(null);\n        };\n    }\n\n    private createUpdateEditRowFn<T extends Record<string, unknown>>(\n        editRow: ReturnType<typeof signal<Partial<T> | null>>\n    ) {\n        return (key: string, value: unknown): void => {\n            const current = editRow();\n            if (!current) return;\n            editRow.set({ ...current, [key]: value });\n        };\n    }\n\n    private createSaveEditFn<T extends Record<string, unknown>>(\n        editRow: ReturnType<typeof signal<Partial<T> | null>>,\n        editing: ReturnType<typeof signal<ID | null>>\n    ) {\n        return (\n            rowId: ID,\n            columns: C80TableColDef[],\n            data: T[],\n            updateEmitter: EventEmitter<{ id: ID; changes: Partial<T>; done: (result: boolean) => void; }>\n        ): void => {\n            const currentEditRow = editRow();\n            if (!currentEditRow) return;\n\n            const visibleColumns = this.getEditableColumns(columns, data, true);\n            const converted = this.convertRowValues(currentEditRow, visibleColumns, data);\n\n            updateEmitter.emit({\n                id: rowId,\n                changes: converted,\n                done: (success: boolean) => {\n                    if (success) {\n                        editing.set(null);\n                        editRow.set(null);\n                    }\n                },\n            });\n        };\n    }\n\n    private createApplyInputValuesFn<T extends Record<string, unknown>>(\n        creating: ReturnType<typeof signal<boolean>>,\n        newRow: ReturnType<typeof signal<Partial<T> | null>>,\n        editing: ReturnType<typeof signal<ID | null>>,\n        editRow: ReturnType<typeof signal<Partial<T> | null>>\n    ) {\n        return (partialValues: Partial<T>): void => {\n            if (!creating() && editing() === null) return;\n\n            if (creating()) {\n                const currentRow = newRow();\n                if (currentRow) {\n                    newRow.set({ ...currentRow, ...partialValues });\n                }\n            } else {\n                const currentEditRow = editRow();\n                if (currentEditRow) {\n                    editRow.set({ ...currentEditRow, ...partialValues });\n                }\n            }\n        };\n    }\n}\n"]}
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-crud-state.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-crud-state.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, inject } from '@angular/core';\nimport { C80TableColDef } from './table.types';\nimport { TableColumnVisibilityService } from './table-column-visibility.service';\nimport { TableDataUtilsService } from './table-data-utils.service';\n\n/**\n * Servicio para gestionar el estado CRUD (Crear, Leer, Actualizar, Eliminar) en tablas C80\n *\n * Maneja:\n * - Estado de creación y edición\n * - Gestión de datos temporales (newRow, editRow)\n * - Aplicación de valores dinámicos\n * - Validación y conversión de datos\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class TableCrudStateService {\n\n  private readonly visibilityService = inject(TableColumnVisibilityService);\n  private readonly dataUtils = inject(TableDataUtilsService);\n\n  /**\n   * Filtra columnas visibles y editables\n   */\n  private getEditableColumns<T extends Record<string, unknown>>(\n    columns: C80TableColDef[],\n    data: T[],\n    includeHiddenIfNull: boolean\n  ): C80TableColDef[] {\n    return columns.filter(\n      col => this.visibilityService.isColumnVisible(col, data, { forceShowInCreation: includeHiddenIfNull }) && !col.readOnly\n    );\n  }\n\n  /**\n   * Inicializa row para creación con valores por defecto\n   */\n  private initializeNewRow<T extends Record<string, unknown>>(\n    columns: C80TableColDef[],\n    data: T[]\n  ): Partial<T> {\n    const row: Partial<T> = {};\n    for (const col of this.getEditableColumns(columns, data, true)) {\n      const defaultValue = col.default === undefined ? '' : col.default;\n      (row as Record<string, unknown>)[col.accessor] = defaultValue;\n    }\n    return row;\n  }\n\n  /**\n   * Inicializa row para edición con valores actuales\n   */\n  private initializeEditRow<T extends Record<string, unknown>>(\n    row: T,\n    columns: C80TableColDef[],\n    data: T[]\n  ): Partial<T> {\n    const edit: Partial<T> = {};\n    for (const col of this.getEditableColumns(columns, data, true)) {\n      const value = this.dataUtils.getCellValue(row, col.accessor);\n      (edit as Record<string, unknown>)[col.accessor] = value;\n    }\n    return edit;\n  }\n\n  /**\n   * Inicializa el estado CRUD para una nueva tabla\n   * @returns Objeto con signals y métodos para gestión CRUD\n   */\n  createCrudState<T extends Record<string, unknown>>() {\n    const creating = signal(false);\n    const newRow = signal<Partial<T> | null>(null);\n    const editing = signal<string | null>(null);\n    const editRow = signal<Partial<T> | null>(null);\n\n    return {\n      creating: creating.asReadonly(),\n      newRow: newRow.asReadonly(),\n      editing: editing.asReadonly(),\n      editRow: editRow.asReadonly(),\n      startCreate: (columns: C80TableColDef[], data: T[]) => this.startCreate(creating, newRow, columns, data),\n      cancelCreate: () => this.cancelCreate(creating, newRow),\n      updateNewRow: (key: string, value: unknown) => this.updateRow(newRow, key, value),\n      startEdit: (row: T, columns: C80TableColDef[], data: T[]) => this.startEdit({ editing, editRow }, row, columns, data),\n      cancelEdit: () => this.cancelEdit(editing, editRow),\n      updateEditRow: (key: string, value: unknown) => this.updateRow(editRow, key, value),\n      applyInputValues: (partialValues: Partial<T>) => this.applyInputValues({ creating, newRow, editing, editRow }, partialValues),\n    };\n  }\n\n  private startCreate<T extends Record<string, unknown>>(\n    creating: ReturnType<typeof signal<boolean>>,\n    newRow: ReturnType<typeof signal<Partial<T> | null>>,\n    columns: C80TableColDef[],\n    data: T[]\n  ): void {\n    creating.set(true);\n    newRow.set(this.initializeNewRow(columns, data));\n  }\n\n  private cancelCreate<T extends Record<string, unknown>>(\n    creating: ReturnType<typeof signal<boolean>>,\n    newRow: ReturnType<typeof signal<Partial<T> | null>>\n  ): void {\n    creating.set(false);\n    newRow.set(null);\n  }\n\n  private updateRow<T extends Record<string, unknown>>(\n    rowSignal: ReturnType<typeof signal<Partial<T> | null>>,\n    key: string,\n    value: unknown\n  ): void {\n    const current = rowSignal();\n    if (!current) return;\n    rowSignal.set({ ...current, [key]: value });\n  }\n\n  private startEdit<T extends Record<string, unknown>>(\n    signals: {\n      editing: ReturnType<typeof signal<string | null>>;\n      editRow: ReturnType<typeof signal<Partial<T> | null>>;\n    },\n    row: T,\n    columns: C80TableColDef[],\n    data: T[]\n  ): void {\n    signals.editing.set(row['id'] as string);\n    signals.editRow.set(this.initializeEditRow(row, columns, data));\n  }\n\n  private cancelEdit<T extends Record<string, unknown>>(\n    editing: ReturnType<typeof signal<string | null>>,\n    editRow: ReturnType<typeof signal<Partial<T> | null>>\n  ): void {\n    editing.set(null);\n    editRow.set(null);\n  }\n\n  private applyInputValues<T extends Record<string, unknown>>(\n    signals: {\n      creating: ReturnType<typeof signal<boolean>>;\n      newRow: ReturnType<typeof signal<Partial<T> | null>>;\n      editing: ReturnType<typeof signal<string | null>>;\n      editRow: ReturnType<typeof signal<Partial<T> | null>>;\n    },\n    partialValues: Partial<T>\n  ): void {\n    if (!signals.creating() && signals.editing() === null) return;\n\n    if (signals.creating()) {\n      const currentRow = signals.newRow();\n      if (currentRow) {\n        signals.newRow.set({ ...currentRow, ...partialValues });\n      }\n    } else {\n      const currentEditRow = signals.editRow();\n      if (currentEditRow) {\n        signals.editRow.set({ ...currentEditRow, ...partialValues });\n      }\n    }\n  }\n}\n"]}