@c80/ui 1.0.34 → 1.0.37

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.
@@ -13,4 +13,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
13
13
  }], propDecorators: { cards: [{
14
14
  type: Input
15
15
  }] } });
16
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdC1jYXJkLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvc3JjL2xpYi9zdGF0LWNhcmQvc3RhdC1jYXJkLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvc3JjL2xpYi9zdGF0LWNhcmQvc3RhdC1jYXJkLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2pELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQVEsZ0JBQWdCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQzs7QUFpQmhFLE1BQU0sT0FBTyxvQkFBb0I7SUFDdEIsS0FBSyxHQUFjLEVBQUUsQ0FBQzt3R0FEcEIsb0JBQW9COzRGQUFwQixvQkFBb0IscUdDbkJqQyxtakJBY00sc3ZCRENNLFlBQVksK0JBQUUsZ0JBQWdCOzs0RkFJN0Isb0JBQW9CO2tCQVJoQyxTQUFTOytCQUVFLGVBQWUsY0FDYixJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsZ0JBQWdCLENBQUM7OEJBS2hDLEtBQUs7c0JBQWIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgSWNvbiwgQzgwSWNvbkNvbXBvbmVudCB9IGZyb20gJy4uL2ljb24vaWNvbi5jb21wb25lbnQnO1xyXG5cclxuZXhwb3J0IGludGVyZmFjZSBDYXJkRGVmIHtcclxuICBjb2xvcjogc3RyaW5nOyAvLyBBY2VwdGEgY3VhbHF1aWVyIGNvbG9yIENTU1xyXG4gIGljb246IEljb247XHJcbiAgdGV4dDogc3RyaW5nO1xyXG4gIGNvdW50OiBudW1iZXI7XHJcbn1cclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAYW5ndWxhci1lc2xpbnQvY29tcG9uZW50LXNlbGVjdG9yXHJcbiAgc2VsZWN0b3I6ICdjODAtc3RhdC1jYXJkJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIEM4MEljb25Db21wb25lbnRdLFxyXG4gIHRlbXBsYXRlVXJsOiAnLi9zdGF0LWNhcmQuY29tcG9uZW50Lmh0bWwnLFxyXG4gIHN0eWxlVXJsczogWycuL3N0YXQtY2FyZC5jb21wb25lbnQuc2NzcyddLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgQzgwU3RhdENhcmRDb21wb25lbnQge1xyXG4gIEBJbnB1dCgpIGNhcmRzOiBDYXJkRGVmW10gPSBbXTtcclxufVxyXG4iLCI8ZGl2IGNsYXNzPVwic3RhdHMtc2VjdGlvbiBweS00XCI+XHJcbiAgQGZvciAoY2FyZCBvZiBjYXJkczsgdHJhY2sgY2FyZC50ZXh0KSB7XHJcbiAgPGRpdiBjbGFzcz1cInN0YXQtY2FyZFwiIFtzdHlsZS5ib3JkZXItbGVmdC1jb2xvcl09XCJjYXJkLmNvbG9yXCI+XHJcbiAgICA8ZGl2IGNsYXNzPVwic3RhdC1jYXJkLWNvbnRlbnRcIj5cclxuICAgICAgPGRpdiBjbGFzcz1cInN0YXQtY29udGVudFwiPlxyXG4gICAgICAgIDxjODAtaWNvbiBbaWNvbl09XCJjYXJkLmljb25cIiBbY29sb3JdPVwiJ3NlY29uZGFyeSdcIiBbc2l6ZV09XCIxLjFcIj48L2M4MC1pY29uPlxyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJzdGF0LWluZm9cIj5cclxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJzdGF0LW51bWJlclwiPnt7IGNhcmQuY291bnQgfX08L2Rpdj5cclxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJzdGF0LWxhYmVsXCI+e3sgY2FyZC50ZXh0IH19PC9kaXY+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICAgIDwvZGl2PlxyXG4gICAgPC9kaXY+XHJcbiAgPC9kaXY+XHJcbiAgfVxyXG48L2Rpdj4iXX0=
16
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3RhdC1jYXJkLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvc3JjL2xpYi9zdGF0LWNhcmQvc3RhdC1jYXJkLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvc3JjL2xpYi9zdGF0LWNhcmQvc3RhdC1jYXJkLmNvbXBvbmVudC5odG1sIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ2pELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSx3QkFBd0IsQ0FBQzs7QUFrQjFELE1BQU0sT0FBTyxvQkFBb0I7SUFDdEIsS0FBSyxHQUFjLEVBQUUsQ0FBQzt3R0FEcEIsb0JBQW9COzRGQUFwQixvQkFBb0IscUdDcEJqQyxtakJBY00sc3ZCREVNLFlBQVksK0JBQUUsZ0JBQWdCOzs0RkFJN0Isb0JBQW9CO2tCQVJoQyxTQUFTOytCQUVFLGVBQWUsY0FDYixJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsZ0JBQWdCLENBQUM7OEJBS2hDLEtBQUs7c0JBQWIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbXBvbmVudCwgSW5wdXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgQzgwSWNvbkNvbXBvbmVudCB9IGZyb20gJy4uL2ljb24vaWNvbi5jb21wb25lbnQnO1xyXG5pbXBvcnQgeyBJY29uVHlwZSB9IGZyb20gJy4uL2ljb24vaWNvbi50eXBlcyc7XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIENhcmREZWYge1xyXG4gIGNvbG9yOiBzdHJpbmc7IC8vIEFjZXB0YSBjdWFscXVpZXIgY29sb3IgQ1NTXHJcbiAgaWNvbjogSWNvblR5cGU7XHJcbiAgdGV4dDogc3RyaW5nO1xyXG4gIGNvdW50OiBudW1iZXI7XHJcbn1cclxuXHJcbkBDb21wb25lbnQoe1xyXG4gIC8vIGVzbGludC1kaXNhYmxlLW5leHQtbGluZSBAYW5ndWxhci1lc2xpbnQvY29tcG9uZW50LXNlbGVjdG9yXHJcbiAgc2VsZWN0b3I6ICdjODAtc3RhdC1jYXJkJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGUsIEM4MEljb25Db21wb25lbnRdLFxyXG4gIHRlbXBsYXRlVXJsOiAnLi9zdGF0LWNhcmQuY29tcG9uZW50Lmh0bWwnLFxyXG4gIHN0eWxlVXJsczogWycuL3N0YXQtY2FyZC5jb21wb25lbnQuc2NzcyddLFxyXG59KVxyXG5leHBvcnQgY2xhc3MgQzgwU3RhdENhcmRDb21wb25lbnQge1xyXG4gIEBJbnB1dCgpIGNhcmRzOiBDYXJkRGVmW10gPSBbXTtcclxufVxyXG4iLCI8ZGl2IGNsYXNzPVwic3RhdHMtc2VjdGlvbiBweS00XCI+XHJcbiAgQGZvciAoY2FyZCBvZiBjYXJkczsgdHJhY2sgY2FyZC50ZXh0KSB7XHJcbiAgPGRpdiBjbGFzcz1cInN0YXQtY2FyZFwiIFtzdHlsZS5ib3JkZXItbGVmdC1jb2xvcl09XCJjYXJkLmNvbG9yXCI+XHJcbiAgICA8ZGl2IGNsYXNzPVwic3RhdC1jYXJkLWNvbnRlbnRcIj5cclxuICAgICAgPGRpdiBjbGFzcz1cInN0YXQtY29udGVudFwiPlxyXG4gICAgICAgIDxjODAtaWNvbiBbaWNvbl09XCJjYXJkLmljb25cIiBbY29sb3JdPVwiJ3NlY29uZGFyeSdcIiBbc2l6ZV09XCIxLjFcIj48L2M4MC1pY29uPlxyXG4gICAgICAgIDxkaXYgY2xhc3M9XCJzdGF0LWluZm9cIj5cclxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJzdGF0LW51bWJlclwiPnt7IGNhcmQuY291bnQgfX08L2Rpdj5cclxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJzdGF0LWxhYmVsXCI+e3sgY2FyZC50ZXh0IH19PC9kaXY+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICAgIDwvZGl2PlxyXG4gICAgPC9kaXY+XHJcbiAgPC9kaXY+XHJcbiAgfVxyXG48L2Rpdj4iXX0=
@@ -153,4 +153,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
153
153
  providedIn: 'root'
154
154
  }]
155
155
  }] });
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: 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"]}
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"]}
@@ -14,6 +14,46 @@ import * as i0 from "@angular/core";
14
14
  export class TableCrudStateService {
15
15
  visibilityService = inject(TableColumnVisibilityService);
16
16
  dataConverter = inject(TableDataConverterService);
17
+ /**
18
+ * Filtra columnas visibles y editables
19
+ */
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
+ }, {});
34
+ }
35
+ /**
36
+ * Inicializa row para creación con valores por defecto
37
+ */
38
+ initializeNewRow(columns, data) {
39
+ const row = {};
40
+ for (const col of this.getEditableColumns(columns, data, true)) {
41
+ const defaultValue = col.default === undefined ? '' : col.default;
42
+ row[col.accessor] = defaultValue;
43
+ }
44
+ return row;
45
+ }
46
+ /**
47
+ * Inicializa row para edición con valores actuales
48
+ */
49
+ initializeEditRow(row, columns, data) {
50
+ const edit = {};
51
+ for (const col of this.getEditableColumns(columns, data, true)) {
52
+ const value = this.visibilityService['getCellValue'](row, col.accessor);
53
+ edit[col.accessor] = value;
54
+ }
55
+ return edit;
56
+ }
17
57
  /**
18
58
  * Inicializa el estado CRUD para una nueva tabla
19
59
  * @returns Objeto con signals y métodos para gestión CRUD
@@ -23,135 +63,116 @@ export class TableCrudStateService {
23
63
  const newRow = signal(null);
24
64
  const editing = signal(null);
25
65
  const editRow = signal(null);
26
- const startCreate = (columns, data) => {
66
+ return {
67
+ creating: creating.asReadonly(),
68
+ newRow: newRow.asReadonly(),
69
+ editing: editing.asReadonly(),
70
+ 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),
80
+ };
81
+ }
82
+ createStartCreateFn(creating, newRow) {
83
+ return (columns, data) => {
27
84
  creating.set(true);
28
- // Inicializar newRow solo con columnas visibles y no readOnly
29
- // En modo creación, mostramos columnas con hideIfAllValuesAreNull para permitir entrada de datos
30
- const row = {};
31
- columns
32
- .filter((col) => this.visibilityService.isColumnVisible(col, data, true) && !col.readOnly)
33
- .forEach((col) => {
34
- // Si la columna tiene un valor por defecto, usarlo; si no, usar cadena vacía
35
- const defaultValue = col.default !== undefined ? col.default : '';
36
- row[col.accessor] = defaultValue;
37
- });
38
- newRow.set(row);
85
+ newRow.set(this.initializeNewRow(columns, data));
39
86
  };
40
- const cancelCreate = () => {
87
+ }
88
+ createCancelCreateFn(creating, newRow) {
89
+ return () => {
41
90
  creating.set(false);
42
91
  newRow.set(null);
43
92
  };
44
- const updateNewRow = (key, value) => {
93
+ }
94
+ createUpdateNewRowFn(newRow) {
95
+ return (key, value) => {
45
96
  const current = newRow();
46
97
  if (!current)
47
98
  return;
48
99
  newRow.set({ ...current, [key]: value });
49
100
  };
50
- const saveCreate = (columns, data, createEmitter) => {
101
+ }
102
+ createSaveCreateFn(newRow, creating) {
103
+ return (columns, data, createEmitter) => {
51
104
  const row = newRow();
52
105
  if (!row)
53
106
  return;
54
- // Validar campos requeridos antes de crear - solo columnas visibles y no readOnly
55
- // En modo creación, incluimos columnas con hideIfAllValuesAreNull
56
- const visibleColumns = columns.filter((col) => this.visibilityService.isColumnVisible(col, data, true) && !col.readOnly);
57
- const converted = visibleColumns.reduce((acc, col) => {
58
- const sampleValue = data.length > 0
59
- ? this.visibilityService['getCellValue'](data[0], col.accessor)
60
- : undefined;
61
- acc[col.accessor] = this.dataConverter.convertCellValue(row[col.accessor], col, sampleValue);
62
- return acc;
63
- }, {});
107
+ const visibleColumns = this.getEditableColumns(columns, data, true);
108
+ const converted = this.convertRowValues(row, visibleColumns, data);
64
109
  createEmitter.emit({
65
110
  row: converted,
66
111
  done: (success) => {
67
112
  if (success) {
68
- cancelCreate();
113
+ creating.set(false);
114
+ newRow.set(null);
69
115
  }
70
116
  },
71
117
  });
72
118
  };
73
- const startEdit = (row, columns, data) => {
119
+ }
120
+ createStartEditFn(editing, editRow) {
121
+ return (row, columns, data) => {
74
122
  editing.set(row['id']);
75
- const edit = {};
76
- columns
77
- .filter((col) => this.visibilityService.isColumnVisible(col, data, true) && !col.readOnly)
78
- .forEach((col) => {
79
- const value = this.visibilityService['getCellValue'](row, col.accessor);
80
- edit[col.accessor] = value;
81
- });
82
- editRow.set(edit);
123
+ editRow.set(this.initializeEditRow(row, columns, data));
83
124
  };
84
- const cancelEdit = () => {
125
+ }
126
+ createCancelEditFn(editing, editRow) {
127
+ return () => {
85
128
  editing.set(null);
86
129
  editRow.set(null);
87
130
  };
88
- const updateEditRow = (key, value) => {
131
+ }
132
+ createUpdateEditRowFn(editRow) {
133
+ return (key, value) => {
89
134
  const current = editRow();
90
135
  if (!current)
91
136
  return;
92
137
  editRow.set({ ...current, [key]: value });
93
138
  };
94
- const saveEdit = (rowId, columns, data, updateEmitter) => {
139
+ }
140
+ createSaveEditFn(editRow, editing) {
141
+ return (rowId, columns, data, updateEmitter) => {
95
142
  const currentEditRow = editRow();
96
143
  if (!currentEditRow)
97
144
  return;
98
- const visibleColumns = columns.filter((col) => this.visibilityService.isColumnVisible(col, data, true) && !col.readOnly);
99
- const converted = visibleColumns.reduce((acc, col) => {
100
- const sampleValue = data.length > 0
101
- ? this.visibilityService['getCellValue'](data[0], col.accessor)
102
- : undefined;
103
- acc[col.accessor] = this.dataConverter.convertCellValue(currentEditRow[col.accessor], col, sampleValue);
104
- return acc;
105
- }, {});
145
+ const visibleColumns = this.getEditableColumns(columns, data, true);
146
+ const converted = this.convertRowValues(currentEditRow, visibleColumns, data);
106
147
  updateEmitter.emit({
107
148
  id: rowId,
108
149
  changes: converted,
109
150
  done: (success) => {
110
151
  if (success) {
111
- cancelEdit();
152
+ editing.set(null);
153
+ editRow.set(null);
112
154
  }
113
155
  },
114
156
  });
115
157
  };
116
- const applyInputValues = (partialValues) => {
117
- // Solo aplicar valores si estamos en modo creación o edición
118
- if (!creating() && editing() === null) {
158
+ }
159
+ createApplyInputValuesFn(creating, newRow, editing, editRow) {
160
+ return (partialValues) => {
161
+ if (!creating() && editing() === null)
119
162
  return;
120
- }
121
- // Aplicar valores en modo creación
122
163
  if (creating()) {
123
164
  const currentRow = newRow();
124
165
  if (currentRow) {
125
- const updatedRow = { ...currentRow, ...partialValues };
126
- newRow.set(updatedRow);
166
+ newRow.set({ ...currentRow, ...partialValues });
127
167
  }
128
168
  }
129
- // Aplicar valores en modo edición
130
- if (editing() !== null) {
169
+ else {
131
170
  const currentEditRow = editRow();
132
171
  if (currentEditRow) {
133
- const updatedEditRow = { ...currentEditRow, ...partialValues };
134
- editRow.set(updatedEditRow);
172
+ editRow.set({ ...currentEditRow, ...partialValues });
135
173
  }
136
174
  }
137
175
  };
138
- return {
139
- // Signals de solo lectura
140
- creating: creating.asReadonly(),
141
- newRow: newRow.asReadonly(),
142
- editing: editing.asReadonly(),
143
- editRow: editRow.asReadonly(),
144
- // Métodos
145
- startCreate,
146
- cancelCreate,
147
- updateNewRow,
148
- saveCreate,
149
- startEdit,
150
- cancelEdit,
151
- updateEditRow,
152
- saveEdit,
153
- applyInputValues,
154
- };
155
176
  }
156
177
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableCrudStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
157
178
  static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableCrudStateService, providedIn: 'root' });
@@ -162,4 +183,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
162
183
  providedIn: 'root'
163
184
  }]
164
185
  }] });
165
- //# 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     * 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        const startCreate = (columns: C80TableColDef[], data: T[]): void => {\n            creating.set(true);\n\n            // Inicializar 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            columns\n                .filter((col) => this.visibilityService.isColumnVisible(col, data, 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            newRow.set(row);\n        };\n\n        const cancelCreate = (): void => {\n            creating.set(false);\n            newRow.set(null);\n        };\n\n        const updateNewRow = (key: string, value: unknown): void => {\n            const current = newRow();\n            if (!current) return;\n            newRow.set({ ...current, [key]: value });\n        };\n\n        const saveCreate = (\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            // Validar campos requeridos antes de crear - solo columnas visibles y no readOnly\n            // En modo creación, incluimos columnas con hideIfAllValuesAreNull\n            const visibleColumns = columns.filter((col) =>\n                this.visibilityService.isColumnVisible(col, data, true) && !col.readOnly\n            );\n\n            const converted = visibleColumns.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            createEmitter.emit({\n                row: converted,\n                done: (success: boolean) => {\n                    if (success) {\n                        cancelCreate();\n                    }\n                },\n            });\n        };\n\n        const startEdit = (row: T, columns: C80TableColDef[], data: T[]): void => {\n            editing.set(row['id'] as ID);\n\n            const edit: Partial<T> = {};\n            columns\n                .filter((col) => this.visibilityService.isColumnVisible(col, data, true) && !col.readOnly)\n                .forEach((col) => {\n                    const value = this.visibilityService['getCellValue'](row, col.accessor);\n                    (edit as Record<string, unknown>)[col.accessor] = value;\n                });\n            editRow.set(edit);\n        };\n\n        const cancelEdit = (): void => {\n            editing.set(null);\n            editRow.set(null);\n        };\n\n        const updateEditRow = (key: string, value: unknown): void => {\n            const current = editRow();\n            if (!current) return;\n            editRow.set({ ...current, [key]: value });\n        };\n\n        const saveEdit = (\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 = columns.filter(\n                (col) => this.visibilityService.isColumnVisible(col, data, true) && !col.readOnly\n            );\n\n            const converted = visibleColumns.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(\n                    currentEditRow[col.accessor],\n                    col,\n                    sampleValue\n                );\n                return acc;\n            }, {} as Record<string, unknown>) as Partial<T>;\n\n            updateEmitter.emit({\n                id: rowId,\n                changes: converted,\n                done: (success: boolean) => {\n                    if (success) {\n                        cancelEdit();\n                    }\n                },\n            });\n        };\n\n        const applyInputValues = (partialValues: Partial<T>): void => {\n            // Solo aplicar valores si estamos en modo creación o edición\n            if (!creating() && editing() === null) {\n                return;\n            }\n\n            // Aplicar valores en modo creación\n            if (creating()) {\n                const currentRow = newRow();\n                if (currentRow) {\n                    const updatedRow = { ...currentRow, ...partialValues };\n                    newRow.set(updatedRow);\n                }\n            }\n\n            // Aplicar valores en modo edición\n            if (editing() !== null) {\n                const currentEditRow = editRow();\n                if (currentEditRow) {\n                    const updatedEditRow = { ...currentEditRow, ...partialValues };\n                    editRow.set(updatedEditRow);\n                }\n            }\n        };\n\n        return {\n            // Signals de solo lectura\n            creating: creating.asReadonly(),\n            newRow: newRow.asReadonly(),\n            editing: editing.asReadonly(),\n            editRow: editRow.asReadonly(),\n\n            // Métodos\n            startCreate,\n            cancelCreate,\n            updateNewRow,\n            saveCreate,\n            startEdit,\n            cancelEdit,\n            updateEditRow,\n            saveEdit,\n            applyInputValues,\n        };\n    }\n}\n"]}
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"]}
@@ -61,7 +61,7 @@ export class TableDataConverterService {
61
61
  if (trimmed === '')
62
62
  return undefined;
63
63
  const num = Number(trimmed);
64
- return isNaN(num) ? undefined : num;
64
+ return Number.isNaN(num) ? undefined : num;
65
65
  }
66
66
  if (typeof value === 'boolean')
67
67
  return value ? 1 : 0;
@@ -80,7 +80,7 @@ export class TableDataConverterService {
80
80
  if (trimmed === '')
81
81
  return undefined;
82
82
  const num = Number(trimmed);
83
- return isNaN(num) ? undefined : Math.floor(num);
83
+ return Number.isNaN(num) ? undefined : Math.floor(num);
84
84
  }
85
85
  if (typeof value === 'boolean')
86
86
  return value ? 1 : 0;
@@ -119,21 +119,17 @@ export class TableDataConverterService {
119
119
  if (value === null || value === undefined || value === '') {
120
120
  return true;
121
121
  }
122
- // Para números, 0 se considera como valor válido, no vacío
123
- if (typeof value === 'number') {
124
- return false;
125
- }
126
- // Para booleanos, false se considera como valor válido, no vacío
127
- if (typeof value === 'boolean') {
122
+ // Para números y booleanos, siempre son valores válidos
123
+ if (typeof value === 'number' || typeof value === 'boolean') {
128
124
  return false;
129
125
  }
130
126
  // Para arrays vacíos
131
- if (Array.isArray(value) && value.length === 0) {
132
- return true;
127
+ if (Array.isArray(value)) {
128
+ return value.length === 0;
133
129
  }
134
130
  // Para objetos vacíos
135
- if (typeof value === 'object' && value !== null && Object.keys(value).length === 0) {
136
- return true;
131
+ if (typeof value === 'object' && value !== null) {
132
+ return Object.keys(value).length === 0;
137
133
  }
138
134
  return false;
139
135
  }
@@ -146,4 +142,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
146
142
  providedIn: 'root'
147
143
  }]
148
144
  }] });
149
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-data-converter.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-data-converter.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { C80TableColDef } from './table.types';\n\n/**\n * Servicio para conversión y validación de tipos de datos en tablas C80\n *\n * Maneja la conversión entre diferentes tipos de datos y la validación de valores vacíos.\n * Soporta los tipos: string, number, integer, boolean, password, enum\n */\n@Injectable({\n    providedIn: 'root'\n})\nexport class TableDataConverterService {\n\n    /**\n     * Convierte un valor de celda basándose en el tipo de columna o datos de muestra\n     * @param value - Valor a convertir\n     * @param col - Definición de la columna\n     * @param sampleValue - Valor de muestra para inferir el tipo (opcional)\n     * @returns Valor convertido al tipo apropiado\n     */\n    convertCellValue(value: unknown, col: C80TableColDef, sampleValue?: unknown): 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: usar datos de muestra si están disponibles\n        if (sampleValue) {\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\n        return value;\n    }\n\n    /**\n     * Convierte un valor a boolean usando mejores prácticas\n     * @param value - Valor a convertir\n     * @returns Valor booleano\n     */\n    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    /**\n     * Convierte un valor a number usando mejores prácticas\n     * @param value - Valor a convertir\n     * @returns Número o undefined si no es válido\n     */\n    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    /**\n     * Convierte un valor a integer usando mejores prácticas\n     * @param value - Valor a convertir\n     * @returns Entero o undefined si no es válido\n     */\n    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    /**\n     * Convierte un valor a string usando mejores prácticas, siempre stringifica objetos\n     * @param value - Valor a convertir\n     * @returns Cadena de texto\n     */\n    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        // Para funciones, símbolos, undefined, etc., devolver cadena vacía\n        return '';\n    }\n\n    /**\n     * Verifica si un valor individual está vacío/nulo\n     * @param value - Valor a verificar\n     * @returns true si el valor se considera vacío\n     */\n    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"]}
145
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-data-converter.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-data-converter.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { C80TableColDef } from './table.types';\n\n/**\n * Servicio para conversión y validación de tipos de datos en tablas C80\n *\n * Maneja la conversión entre diferentes tipos de datos y la validación de valores vacíos.\n * Soporta los tipos: string, number, integer, boolean, password, enum\n */\n@Injectable({\n    providedIn: 'root'\n})\nexport class TableDataConverterService {\n\n    /**\n     * Convierte un valor de celda basándose en el tipo de columna o datos de muestra\n     * @param value - Valor a convertir\n     * @param col - Definición de la columna\n     * @param sampleValue - Valor de muestra para inferir el tipo (opcional)\n     * @returns Valor convertido al tipo apropiado\n     */\n    convertCellValue(value: unknown, col: C80TableColDef, sampleValue?: unknown): 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: usar datos de muestra si están disponibles\n        if (sampleValue) {\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\n        return value;\n    }\n\n    /**\n     * Convierte un valor a boolean usando mejores prácticas\n     * @param value - Valor a convertir\n     * @returns Valor booleano\n     */\n    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    /**\n     * Convierte un valor a number usando mejores prácticas\n     * @param value - Valor a convertir\n     * @returns Número o undefined si no es válido\n     */\n    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 Number.isNaN(num) ? undefined : num;\n        }\n        if (typeof value === 'boolean') return value ? 1 : 0;\n        return undefined;\n    }\n\n    /**\n     * Convierte un valor a integer usando mejores prácticas\n     * @param value - Valor a convertir\n     * @returns Entero o undefined si no es válido\n     */\n    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 Number.isNaN(num) ? undefined : Math.floor(num);\n        }\n        if (typeof value === 'boolean') return value ? 1 : 0;\n        return undefined;\n    }\n\n    /**\n     * Convierte un valor a string usando mejores prácticas, siempre stringifica objetos\n     * @param value - Valor a convertir\n     * @returns Cadena de texto\n     */\n    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        // Para funciones, símbolos, undefined, etc., devolver cadena vacía\n        return '';\n    }\n\n    /**\n     * Verifica si un valor individual está vacío/nulo\n     * @param value - Valor a verificar\n     * @returns true si el valor se considera vacío\n     */\n    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 y booleanos, siempre son valores válidos\n        if (typeof value === 'number' || typeof value === 'boolean') {\n            return false;\n        }\n\n        // Para arrays vacíos\n        if (Array.isArray(value)) {\n            return value.length === 0;\n        }\n\n        // Para objetos vacíos\n        if (typeof value === 'object' && value !== null) {\n            return Object.keys(value).length === 0;\n        }\n\n        return false;\n    }\n}\n"]}
@@ -94,7 +94,7 @@ export class TableDataUtilsService {
94
94
  if (!col.enum)
95
95
  return [];
96
96
  return Object.entries(col.enum).map(([value, label]) => ({
97
- value: isNaN(Number(value)) ? value : Number(value),
97
+ value: Number.isNaN(Number(value)) ? value : Number(value),
98
98
  label,
99
99
  }));
100
100
  }
@@ -118,7 +118,7 @@ export class TableDataUtilsService {
118
118
  applySorting(items, columns) {
119
119
  const orderedColumns = columns.filter((col) => col.order);
120
120
  if (orderedColumns.length > 0) {
121
- orderedColumns.forEach((col) => {
121
+ for (const col of orderedColumns) {
122
122
  items.sort((a, b) => {
123
123
  const valueA = this.getCellValue(a, col.accessor);
124
124
  const valueB = this.getCellValue(b, col.accessor);
@@ -137,7 +137,7 @@ export class TableDataUtilsService {
137
137
  return 0;
138
138
  }
139
139
  });
140
- });
140
+ }
141
141
  }
142
142
  }
143
143
  /**
@@ -163,4 +163,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
163
163
  providedIn: 'root'
164
164
  }]
165
165
  }] });
166
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-data-utils.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-data-utils.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;;AAG3C;;;;;;;;GAQG;AAIH,MAAM,OAAO,qBAAqB;IAE9B;;;;;OAKG;IACH,YAAY,CAAoC,GAAM,EAAE,QAAgB;QACpE,IAAI,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;YACzB,OAAO,IAAI,CAAC,cAAc,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;QAC9C,CAAC;QACD,OAAQ,GAA+B,CAAC,QAAQ,CAAC,CAAC;IACtD,CAAC;IAED;;;;;OAKG;IACH,cAAc,CAAC,GAAQ,EAAE,QAAgB;QACrC,OAAO,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,CAAC,OAAO,EAAE,GAAG,EAAE,EAAE;YAC/C,OAAO,OAAO,IAAI,OAAO,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAC7E,CAAC,EAAE,GAAG,CAAC,CAAC;IACZ,CAAC;IAED;;;;OAIG;IACH,eAAe,CAAC,KAAc;QAC1B,8BAA8B;QAC9B,IAAI,KAAK,KAAK,CAAC,EAAE,CAAC;YACd,OAAO,GAAG,CAAC;QACf,CAAC;QACD,mFAAmF;QACnF,IAAI,KAAK,KAAK,KAAK,EAAE,CAAC;YAClB,OAAO,OAAO,CAAC;QACnB,CAAC;QACD,mDAAmD;QACnD,IAAI,OAAO,KAAK,KAAK,QAAQ,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;YAC9C,IAAI,CAAC;gBACD,OAAO,IAAI,CAAC,SAAS,CAAC,KAAK,CAAC,CAAC;YACjC,CAAC;YAAC,MAAM,CAAC;gBACL,OAAO,iBAAiB,CAAC;YAC7B,CAAC;QACL,CAAC;QACD,wEAAwE;QACxE,IAAI,CAAC,KAAK,EAAE,CAAC;YACT,OAAO,GAAG,CAAC;QACf,CAAC;QACD,qCAAqC;QACrC,IACI,OAAO,KAAK,KAAK,QAAQ;YACzB,OAAO,KAAK,KAAK,QAAQ;YACzB,OAAO,KAAK,KAAK,SAAS,EAC5B,CAAC;YACC,OAAO,MAAM,CAAC,KAAK,CAAC,CAAC;QACzB,CAAC;QACD,iFAAiF;QACjF,OAAO,GAAG,CAAC;IACf,CAAC;IAED;;;;;OAKG;IACH,mBAAmB,CAAC,KAAc,EAAE,GAAmB;QACnD,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,KAAK,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;YACvC,OAAO,GAAG,CAAC;QACf,CAAC;QACD,MAAM,YAAY,GAAG,GAAG,CAAC,IAAI,CAAC,KAAwB,CAAC,CAAC;QACxD,OAAO,CACH,YAAY;YACZ,CAAC,OAAO,KAAK,KAAK,QAAQ,IAAI,OAAO,KAAK,KAAK,QAAQ;gBACnD,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;gBACf,CAAC,CAAC,GAAG,CAAC,CACb,CAAC;IACN,CAAC;IAED;;;;OAIG;IACH,cAAc,CAAC,GAAmB;QAC9B,IAAI,CAAC,GAAG,CAAC,IAAI;YAAE,OAAO,EAAE,CAAC;QACzB,OAAO,MAAM,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,KAAK,EAAE,KAAK,CAAC,EAAE,EAAE,CAAC,CAAC;YACrD,KAAK,EAAE,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC;YACnD,KAAK;SACR,CAAC,CAAC,CAAC;IACR,CAAC;IAED;;;;;OAKG;IACH,YAAY,CAAC,KAAc,EAAE,GAAmB;QAC5C,IAAI,CAAC,GAAG,CAAC,KAAK,IAAI,CAAC,KAAK,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,EAAE,CAAC;YACxC,OAAO,SAAS,CAAC;QACrB,CAAC;QACD,OAAO,GAAG,CAAC,KAAK,CAAC,KAAwB,CAAC,CAAC;IAC/C,CAAC;IAED;;;;OAIG;IACH,YAAY,CAAoC,KAAU,EAAE,OAAyB;QACjF,MAAM,cAAc,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;QAC1D,IAAI,cAAc,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC5B,cAAc,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;gBAC3B,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE;oBAChB,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC,EAAE,GAAG,CAAC,QAAQ,CAAoB,CAAC;oBACrE,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC,CAAC,EAAE,GAAG,CAAC,QAAQ,CAAoB,CAAC;oBAErE,IAAI,GAAG,CAAC,KAAK,KAAK,KAAK,EAAE,CAAC;wBACtB,IAAI,MAAM,GAAG,MAAM;4BAAE,OAAO,CAAC,CAAC;wBAC9B,IAAI,MAAM,GAAG,MAAM;4BAAE,OAAO,CAAC,CAAC,CAAC;wBAC/B,OAAO,CAAC,CAAC;oBACb,CAAC;yBAAM,CAAC;wBACJ,IAAI,MAAM,GAAG,MAAM;4BAAE,OAAO,CAAC,CAAC;wBAC9B,IAAI,MAAM,GAAG,MAAM;4BAAE,OAAO,CAAC,CAAC,CAAC;wBAC/B,OAAO,CAAC,CAAC;oBACb,CAAC;gBACL,CAAC,CAAC,CAAC;YACP,CAAC,CAAC,CAAC;QACP,CAAC;IACL,CAAC;IAED;;;;OAIG;IACH,iBAAiB,CAAC,IAAY;QAC1B,IAAI,IAAI,IAAI,CAAC,EAAE,CAAC;YACZ,OAAO,SAAS,CAAC,CAAC,uBAAuB;QAC7C,CAAC;QACD,8BAA8B;QAC9B,MAAM,UAAU,GAAG,GAAG,CAAC;QACvB,MAAM,SAAS,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,GAAG,IAAI,CAAC,CAAC;QAChD,OAAO,GAAG,SAAS,IAAI,CAAC;IAC5B,CAAC;wGAtJQ,qBAAqB;4GAArB,qBAAqB,cAFlB,MAAM;;4FAET,qBAAqB;kBAHjC,UAAU;mBAAC;oBACR,UAAU,EAAE,MAAM;iBACrB","sourcesContent":["import { Injectable } from '@angular/core';\nimport { C80TableColDef } from './table.types';\n\n/**\n * Servicio para utilidades de manipulación de datos en tablas C80\n *\n * Proporciona métodos para:\n * - Acceso a valores de celdas (incluyendo propiedades anidadas)\n * - Formateo de valores para display\n * - Gestión de enums y colores\n * - Ordenamiento de datos\n */\n@Injectable({\n    providedIn: 'root'\n})\nexport class TableDataUtilsService {\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    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    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     * Devuelve el valor de display para una celda, mostrando '-' para valores falsy excepto 0, false y objetos/arrays vacíos\n     * @param value - Valor a formatear\n     * @returns Cadena formateada para display\n     */\n    getDisplayValue(value: unknown): string {\n        // Si el valor es 0, mostrarlo\n        if (value === 0) {\n            return '0';\n        }\n        // Si el valor es false, debe ser manejado por lógica booleana en template, no aquí\n        if (value === false) {\n            return 'false';\n        }\n        // Manejar objetos y arrays (incluyendo los vacíos)\n        if (typeof value === 'object' && value !== null) {\n            try {\n                return JSON.stringify(value);\n            } catch {\n                return '[object Object]';\n            }\n        }\n        // Si el valor son otros falsy values (null, undefined, ''), mostrar '-'\n        if (!value) {\n            return '-';\n        }\n        // Manejar otros tipos explícitamente\n        if (\n            typeof value === 'string' ||\n            typeof value === 'number' ||\n            typeof value === 'boolean'\n        ) {\n            return String(value);\n        }\n        // Para cualquier otro tipo (función, símbolo, etc.), devolver un fallback seguro\n        return '-';\n    }\n\n    /**\n     * Obtiene el texto de display para un valor enum\n     * @param value - Valor del enum\n     * @param col - Definición de la columna\n     * @returns Texto formateado del enum\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     * Obtiene las opciones del enum como array para dropdowns select\n     * @param col - Definición de la columna\n     * @returns Array de opciones con value y label\n     */\n    getEnumOptions(col: C80TableColDef): { 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     * Obtiene el color CSS para un valor de celda basado en la configuración de color de la columna\n     * @param value - Valor de la celda\n     * @param col - Definición de la columna\n     * @returns Color CSS o undefined si no hay configuración\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     * Aplica ordenamiento a los items basándose en las columnas con configuración de orden\n     * @param items - Array de items a ordenar (se modifica in-place)\n     * @param columns - Definiciones de columnas\n     */\n    applySorting<T extends Record<string, unknown>>(items: T[], columns: C80TableColDef[]): void {\n        const orderedColumns = 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    /**\n     * Calcula el max-height de la tabla basado en el tamaño\n     * @param size - Tamaño configurado (0 = sin límite)\n     * @returns String CSS para max-height o undefined si sin límite\n     */\n    getTableMaxHeight(size: number): string | undefined {\n        if (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 * size);\n        return `${maxHeight}px`;\n    }\n}\n"]}
166
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-data-utils.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-data-utils.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { C80TableColDef } from './table.types';\n\n/**\n * Servicio para utilidades de manipulación de datos en tablas C80\n *\n * Proporciona métodos para:\n * - Acceso a valores de celdas (incluyendo propiedades anidadas)\n * - Formateo de valores para display\n * - Gestión de enums y colores\n * - Ordenamiento de datos\n */\n@Injectable({\n    providedIn: 'root'\n})\nexport class TableDataUtilsService {\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    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    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    /**\n     * Devuelve el valor de display para una celda, mostrando '-' para valores falsy excepto 0, false y objetos/arrays vacíos\n     * @param value - Valor a formatear\n     * @returns Cadena formateada para display\n     */\n    getDisplayValue(value: unknown): string {\n        // Si el valor es 0, mostrarlo\n        if (value === 0) {\n            return '0';\n        }\n        // Si el valor es false, debe ser manejado por lógica booleana en template, no aquí\n        if (value === false) {\n            return 'false';\n        }\n        // Manejar objetos y arrays (incluyendo los vacíos)\n        if (typeof value === 'object' && value !== null) {\n            try {\n                return JSON.stringify(value);\n            } catch {\n                return '[object Object]';\n            }\n        }\n        // Si el valor son otros falsy values (null, undefined, ''), mostrar '-'\n        if (!value) {\n            return '-';\n        }\n        // Manejar otros tipos explícitamente\n        if (\n            typeof value === 'string' ||\n            typeof value === 'number' ||\n            typeof value === 'boolean'\n        ) {\n            return String(value);\n        }\n        // Para cualquier otro tipo (función, símbolo, etc.), devolver un fallback seguro\n        return '-';\n    }\n\n    /**\n     * Obtiene el texto de display para un valor enum\n     * @param value - Valor del enum\n     * @param col - Definición de la columna\n     * @returns Texto formateado del enum\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     * Obtiene las opciones del enum como array para dropdowns select\n     * @param col - Definición de la columna\n     * @returns Array de opciones con value y label\n     */\n    getEnumOptions(col: C80TableColDef): { value: string | number; label: string }[] {\n        if (!col.enum) return [];\n        return Object.entries(col.enum).map(([value, label]) => ({\n            value: Number.isNaN(Number(value)) ? value : Number(value),\n            label,\n        }));\n    }\n\n    /**\n     * Obtiene el color CSS para un valor de celda basado en la configuración de color de la columna\n     * @param value - Valor de la celda\n     * @param col - Definición de la columna\n     * @returns Color CSS o undefined si no hay configuración\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     * Aplica ordenamiento a los items basándose en las columnas con configuración de orden\n     * @param items - Array de items a ordenar (se modifica in-place)\n     * @param columns - Definiciones de columnas\n     */\n    applySorting<T extends Record<string, unknown>>(items: T[], columns: C80TableColDef[]): void {\n        const orderedColumns = columns.filter((col) => col.order);\n        if (orderedColumns.length > 0) {\n            for (const col of orderedColumns) {\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    /**\n     * Calcula el max-height de la tabla basado en el tamaño\n     * @param size - Tamaño configurado (0 = sin límite)\n     * @returns String CSS para max-height o undefined si sin límite\n     */\n    getTableMaxHeight(size: number): string | undefined {\n        if (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 * size);\n        return `${maxHeight}px`;\n    }\n}\n"]}