@c80/ui 1.0.32 → 1.0.34
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/card-level/card-level.component.mjs +34 -5
- package/esm2022/lib/table/index.mjs +7 -1
- package/esm2022/lib/table/table-column-visibility.service.mjs +156 -0
- package/esm2022/lib/table/table-crud-state.service.mjs +165 -0
- package/esm2022/lib/table/table-data-converter.service.mjs +149 -0
- package/esm2022/lib/table/table-data-utils.service.mjs +166 -0
- package/esm2022/lib/table/table-selection.service.mjs +151 -0
- package/esm2022/lib/table/table.component.mjs +59 -500
- package/esm2022/lib/table/table.types.mjs +5 -0
- package/lib/card-level/card-level.component.d.ts +4 -0
- package/lib/table/index.d.ts +6 -0
- package/lib/table/table-column-visibility.service.d.ts +71 -0
- package/lib/table/table-crud-state.service.d.ts +44 -0
- package/lib/table/table-data-converter.service.d.ts +50 -0
- package/lib/table/table-data-utils.service.d.ts +70 -0
- package/lib/table/table-selection.service.d.ts +39 -0
- package/lib/table/table.component.d.ts +14 -99
- package/lib/table/table.types.d.ts +16 -0
- package/package.json +1 -1
|
@@ -0,0 +1,149 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Servicio para conversión y validación de tipos de datos en tablas C80
|
|
5
|
+
*
|
|
6
|
+
* Maneja la conversión entre diferentes tipos de datos y la validación de valores vacíos.
|
|
7
|
+
* Soporta los tipos: string, number, integer, boolean, password, enum
|
|
8
|
+
*/
|
|
9
|
+
export class TableDataConverterService {
|
|
10
|
+
/**
|
|
11
|
+
* Convierte un valor de celda basándose en el tipo de columna o datos de muestra
|
|
12
|
+
* @param value - Valor a convertir
|
|
13
|
+
* @param col - Definición de la columna
|
|
14
|
+
* @param sampleValue - Valor de muestra para inferir el tipo (opcional)
|
|
15
|
+
* @returns Valor convertido al tipo apropiado
|
|
16
|
+
*/
|
|
17
|
+
convertCellValue(value, col, sampleValue) {
|
|
18
|
+
if (col.type === 'boolean')
|
|
19
|
+
return this.toBoolean(value);
|
|
20
|
+
if (col.type === 'number')
|
|
21
|
+
return this.toNumber(value);
|
|
22
|
+
if (col.type === 'integer')
|
|
23
|
+
return this.toInteger(value);
|
|
24
|
+
if (col.type === 'string' || col.type === 'password')
|
|
25
|
+
return this.toStringValue(value);
|
|
26
|
+
// Fallback: usar datos de muestra si están disponibles
|
|
27
|
+
if (sampleValue) {
|
|
28
|
+
if (typeof sampleValue === 'boolean')
|
|
29
|
+
return this.toBoolean(value);
|
|
30
|
+
if (typeof sampleValue === 'number')
|
|
31
|
+
return this.toNumber(value);
|
|
32
|
+
if (typeof sampleValue === 'string')
|
|
33
|
+
return this.toStringValue(value);
|
|
34
|
+
}
|
|
35
|
+
return value;
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Convierte un valor a boolean usando mejores prácticas
|
|
39
|
+
* @param value - Valor a convertir
|
|
40
|
+
* @returns Valor booleano
|
|
41
|
+
*/
|
|
42
|
+
toBoolean(value) {
|
|
43
|
+
if (typeof value === 'boolean')
|
|
44
|
+
return value;
|
|
45
|
+
if (typeof value === 'string')
|
|
46
|
+
return value.trim().toLowerCase() === 'true' || value.trim() === '1';
|
|
47
|
+
if (typeof value === 'number')
|
|
48
|
+
return value === 1;
|
|
49
|
+
return false;
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Convierte un valor a number usando mejores prácticas
|
|
53
|
+
* @param value - Valor a convertir
|
|
54
|
+
* @returns Número o undefined si no es válido
|
|
55
|
+
*/
|
|
56
|
+
toNumber(value) {
|
|
57
|
+
if (typeof value === 'number')
|
|
58
|
+
return value;
|
|
59
|
+
if (typeof value === 'string') {
|
|
60
|
+
const trimmed = value.trim();
|
|
61
|
+
if (trimmed === '')
|
|
62
|
+
return undefined;
|
|
63
|
+
const num = Number(trimmed);
|
|
64
|
+
return isNaN(num) ? undefined : num;
|
|
65
|
+
}
|
|
66
|
+
if (typeof value === 'boolean')
|
|
67
|
+
return value ? 1 : 0;
|
|
68
|
+
return undefined;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Convierte un valor a integer usando mejores prácticas
|
|
72
|
+
* @param value - Valor a convertir
|
|
73
|
+
* @returns Entero o undefined si no es válido
|
|
74
|
+
*/
|
|
75
|
+
toInteger(value) {
|
|
76
|
+
if (typeof value === 'number')
|
|
77
|
+
return Math.floor(value);
|
|
78
|
+
if (typeof value === 'string') {
|
|
79
|
+
const trimmed = value.trim();
|
|
80
|
+
if (trimmed === '')
|
|
81
|
+
return undefined;
|
|
82
|
+
const num = Number(trimmed);
|
|
83
|
+
return isNaN(num) ? undefined : Math.floor(num);
|
|
84
|
+
}
|
|
85
|
+
if (typeof value === 'boolean')
|
|
86
|
+
return value ? 1 : 0;
|
|
87
|
+
return undefined;
|
|
88
|
+
}
|
|
89
|
+
/**
|
|
90
|
+
* Convierte un valor a string usando mejores prácticas, siempre stringifica objetos
|
|
91
|
+
* @param value - Valor a convertir
|
|
92
|
+
* @returns Cadena de texto
|
|
93
|
+
*/
|
|
94
|
+
toStringValue(value) {
|
|
95
|
+
if (value == null)
|
|
96
|
+
return '';
|
|
97
|
+
if (typeof value === 'string')
|
|
98
|
+
return value;
|
|
99
|
+
if (typeof value === 'number' || typeof value === 'boolean')
|
|
100
|
+
return String(value);
|
|
101
|
+
if (typeof value === 'object') {
|
|
102
|
+
try {
|
|
103
|
+
return JSON.stringify(value);
|
|
104
|
+
}
|
|
105
|
+
catch {
|
|
106
|
+
return '[object Object]';
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
// Para funciones, símbolos, undefined, etc., devolver cadena vacía
|
|
110
|
+
return '';
|
|
111
|
+
}
|
|
112
|
+
/**
|
|
113
|
+
* Verifica si un valor individual está vacío/nulo
|
|
114
|
+
* @param value - Valor a verificar
|
|
115
|
+
* @returns true si el valor se considera vacío
|
|
116
|
+
*/
|
|
117
|
+
isValueEmpty(value) {
|
|
118
|
+
// Considerar vacío: null, undefined, ''
|
|
119
|
+
if (value === null || value === undefined || value === '') {
|
|
120
|
+
return true;
|
|
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') {
|
|
128
|
+
return false;
|
|
129
|
+
}
|
|
130
|
+
// Para arrays vacíos
|
|
131
|
+
if (Array.isArray(value) && value.length === 0) {
|
|
132
|
+
return true;
|
|
133
|
+
}
|
|
134
|
+
// Para objetos vacíos
|
|
135
|
+
if (typeof value === 'object' && value !== null && Object.keys(value).length === 0) {
|
|
136
|
+
return true;
|
|
137
|
+
}
|
|
138
|
+
return false;
|
|
139
|
+
}
|
|
140
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableDataConverterService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
141
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableDataConverterService, providedIn: 'root' });
|
|
142
|
+
}
|
|
143
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableDataConverterService, decorators: [{
|
|
144
|
+
type: Injectable,
|
|
145
|
+
args: [{
|
|
146
|
+
providedIn: 'root'
|
|
147
|
+
}]
|
|
148
|
+
}] });
|
|
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"]}
|
|
@@ -0,0 +1,166 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Servicio para utilidades de manipulación de datos en tablas C80
|
|
5
|
+
*
|
|
6
|
+
* Proporciona métodos para:
|
|
7
|
+
* - Acceso a valores de celdas (incluyendo propiedades anidadas)
|
|
8
|
+
* - Formateo de valores para display
|
|
9
|
+
* - Gestión de enums y colores
|
|
10
|
+
* - Ordenamiento de datos
|
|
11
|
+
*/
|
|
12
|
+
export class TableDataUtilsService {
|
|
13
|
+
/**
|
|
14
|
+
* Obtiene el valor de una celda usando el accessor, soportando notación de punto para propiedades anidadas
|
|
15
|
+
* @param row - La fila de datos
|
|
16
|
+
* @param accessor - El accessor de la columna
|
|
17
|
+
* @returns El valor de la celda
|
|
18
|
+
*/
|
|
19
|
+
getCellValue(row, accessor) {
|
|
20
|
+
if (accessor.includes('.')) {
|
|
21
|
+
return this.getNestedValue(row, accessor);
|
|
22
|
+
}
|
|
23
|
+
return row[accessor];
|
|
24
|
+
}
|
|
25
|
+
/**
|
|
26
|
+
* Obtiene el valor de un objeto usando notación de punto (ej: 'task.name')
|
|
27
|
+
* @param obj - Objeto del cual obtener el valor
|
|
28
|
+
* @param accessor - Ruta de acceso usando notación de punto
|
|
29
|
+
* @returns El valor anidado o undefined
|
|
30
|
+
*/
|
|
31
|
+
getNestedValue(obj, accessor) {
|
|
32
|
+
return accessor.split('.').reduce((current, key) => {
|
|
33
|
+
return current && typeof current === 'object' ? current[key] : undefined;
|
|
34
|
+
}, obj);
|
|
35
|
+
}
|
|
36
|
+
/**
|
|
37
|
+
* Devuelve el valor de display para una celda, mostrando '-' para valores falsy excepto 0, false y objetos/arrays vacíos
|
|
38
|
+
* @param value - Valor a formatear
|
|
39
|
+
* @returns Cadena formateada para display
|
|
40
|
+
*/
|
|
41
|
+
getDisplayValue(value) {
|
|
42
|
+
// Si el valor es 0, mostrarlo
|
|
43
|
+
if (value === 0) {
|
|
44
|
+
return '0';
|
|
45
|
+
}
|
|
46
|
+
// Si el valor es false, debe ser manejado por lógica booleana en template, no aquí
|
|
47
|
+
if (value === false) {
|
|
48
|
+
return 'false';
|
|
49
|
+
}
|
|
50
|
+
// Manejar objetos y arrays (incluyendo los vacíos)
|
|
51
|
+
if (typeof value === 'object' && value !== null) {
|
|
52
|
+
try {
|
|
53
|
+
return JSON.stringify(value);
|
|
54
|
+
}
|
|
55
|
+
catch {
|
|
56
|
+
return '[object Object]';
|
|
57
|
+
}
|
|
58
|
+
}
|
|
59
|
+
// Si el valor son otros falsy values (null, undefined, ''), mostrar '-'
|
|
60
|
+
if (!value) {
|
|
61
|
+
return '-';
|
|
62
|
+
}
|
|
63
|
+
// Manejar otros tipos explícitamente
|
|
64
|
+
if (typeof value === 'string' ||
|
|
65
|
+
typeof value === 'number' ||
|
|
66
|
+
typeof value === 'boolean') {
|
|
67
|
+
return String(value);
|
|
68
|
+
}
|
|
69
|
+
// Para cualquier otro tipo (función, símbolo, etc.), devolver un fallback seguro
|
|
70
|
+
return '-';
|
|
71
|
+
}
|
|
72
|
+
/**
|
|
73
|
+
* Obtiene el texto de display para un valor enum
|
|
74
|
+
* @param value - Valor del enum
|
|
75
|
+
* @param col - Definición de la columna
|
|
76
|
+
* @returns Texto formateado del enum
|
|
77
|
+
*/
|
|
78
|
+
getEnumDisplayValue(value, col) {
|
|
79
|
+
if (!col.enum || (value !== 0 && !value)) {
|
|
80
|
+
return '-';
|
|
81
|
+
}
|
|
82
|
+
const displayValue = col.enum[value];
|
|
83
|
+
return (displayValue ||
|
|
84
|
+
(typeof value === 'string' || typeof value === 'number'
|
|
85
|
+
? String(value)
|
|
86
|
+
: '-'));
|
|
87
|
+
}
|
|
88
|
+
/**
|
|
89
|
+
* Obtiene las opciones del enum como array para dropdowns select
|
|
90
|
+
* @param col - Definición de la columna
|
|
91
|
+
* @returns Array de opciones con value y label
|
|
92
|
+
*/
|
|
93
|
+
getEnumOptions(col) {
|
|
94
|
+
if (!col.enum)
|
|
95
|
+
return [];
|
|
96
|
+
return Object.entries(col.enum).map(([value, label]) => ({
|
|
97
|
+
value: isNaN(Number(value)) ? value : Number(value),
|
|
98
|
+
label,
|
|
99
|
+
}));
|
|
100
|
+
}
|
|
101
|
+
/**
|
|
102
|
+
* Obtiene el color CSS para un valor de celda basado en la configuración de color de la columna
|
|
103
|
+
* @param value - Valor de la celda
|
|
104
|
+
* @param col - Definición de la columna
|
|
105
|
+
* @returns Color CSS o undefined si no hay configuración
|
|
106
|
+
*/
|
|
107
|
+
getCellColor(value, col) {
|
|
108
|
+
if (!col.color || (value !== 0 && !value)) {
|
|
109
|
+
return undefined;
|
|
110
|
+
}
|
|
111
|
+
return col.color[value];
|
|
112
|
+
}
|
|
113
|
+
/**
|
|
114
|
+
* Aplica ordenamiento a los items basándose en las columnas con configuración de orden
|
|
115
|
+
* @param items - Array de items a ordenar (se modifica in-place)
|
|
116
|
+
* @param columns - Definiciones de columnas
|
|
117
|
+
*/
|
|
118
|
+
applySorting(items, columns) {
|
|
119
|
+
const orderedColumns = columns.filter((col) => col.order);
|
|
120
|
+
if (orderedColumns.length > 0) {
|
|
121
|
+
orderedColumns.forEach((col) => {
|
|
122
|
+
items.sort((a, b) => {
|
|
123
|
+
const valueA = this.getCellValue(a, col.accessor);
|
|
124
|
+
const valueB = this.getCellValue(b, col.accessor);
|
|
125
|
+
if (col.order === 'ASC') {
|
|
126
|
+
if (valueA > valueB)
|
|
127
|
+
return 1;
|
|
128
|
+
if (valueA < valueB)
|
|
129
|
+
return -1;
|
|
130
|
+
return 0;
|
|
131
|
+
}
|
|
132
|
+
else {
|
|
133
|
+
if (valueA < valueB)
|
|
134
|
+
return 1;
|
|
135
|
+
if (valueA > valueB)
|
|
136
|
+
return -1;
|
|
137
|
+
return 0;
|
|
138
|
+
}
|
|
139
|
+
});
|
|
140
|
+
});
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
/**
|
|
144
|
+
* Calcula el max-height de la tabla basado en el tamaño
|
|
145
|
+
* @param size - Tamaño configurado (0 = sin límite)
|
|
146
|
+
* @returns String CSS para max-height o undefined si sin límite
|
|
147
|
+
*/
|
|
148
|
+
getTableMaxHeight(size) {
|
|
149
|
+
if (size <= 0) {
|
|
150
|
+
return undefined; // Sin límite de altura
|
|
151
|
+
}
|
|
152
|
+
// Altura base de 400px * size
|
|
153
|
+
const baseHeight = 400;
|
|
154
|
+
const maxHeight = Math.round(baseHeight * size);
|
|
155
|
+
return `${maxHeight}px`;
|
|
156
|
+
}
|
|
157
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableDataUtilsService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
158
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableDataUtilsService, providedIn: 'root' });
|
|
159
|
+
}
|
|
160
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableDataUtilsService, decorators: [{
|
|
161
|
+
type: Injectable,
|
|
162
|
+
args: [{
|
|
163
|
+
providedIn: 'root'
|
|
164
|
+
}]
|
|
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":"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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: 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"]}
|
|
@@ -0,0 +1,151 @@
|
|
|
1
|
+
import { Injectable, signal } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* Servicio para gestionar la selección de elementos en tablas C80
|
|
5
|
+
*
|
|
6
|
+
* Maneja:
|
|
7
|
+
* - Selección simple y múltiple
|
|
8
|
+
* - Estado de selección completa
|
|
9
|
+
* - Preservación de selección tras actualizaciones de datos
|
|
10
|
+
* - Emisión de eventos de selección
|
|
11
|
+
*/
|
|
12
|
+
export class TableSelectionService {
|
|
13
|
+
/**
|
|
14
|
+
* Inicializa el estado de selección para una nueva tabla
|
|
15
|
+
* @returns Objeto con signals y métodos de selección
|
|
16
|
+
*/
|
|
17
|
+
createSelectionState() {
|
|
18
|
+
const selectedItems = signal(new Set());
|
|
19
|
+
const selectAllChecked = signal(false);
|
|
20
|
+
const selectAllIndeterminate = signal(false);
|
|
21
|
+
const clearSelection = () => {
|
|
22
|
+
selectedItems.set(new Set());
|
|
23
|
+
updateSelectAllState();
|
|
24
|
+
};
|
|
25
|
+
const updateSelectAllState = () => {
|
|
26
|
+
// Esta función será actualizada cuando se tengan los datos disponibles
|
|
27
|
+
selectAllChecked.set(false);
|
|
28
|
+
selectAllIndeterminate.set(false);
|
|
29
|
+
};
|
|
30
|
+
const toggleSelectAll = (allItems) => {
|
|
31
|
+
const currentSelection = selectedItems();
|
|
32
|
+
if (currentSelection.size === allItems.length) {
|
|
33
|
+
// Desmarcar todos los elementos
|
|
34
|
+
clearSelection();
|
|
35
|
+
}
|
|
36
|
+
else {
|
|
37
|
+
// Marcar todos los elementos
|
|
38
|
+
const allIds = new Set(allItems.map(item => item['id']));
|
|
39
|
+
selectedItems.set(allIds);
|
|
40
|
+
updateSelectAllStateWithData(allItems);
|
|
41
|
+
}
|
|
42
|
+
};
|
|
43
|
+
const toggleItemSelection = (item, multiple) => {
|
|
44
|
+
const id = item['id'];
|
|
45
|
+
const currentSelection = new Set(selectedItems());
|
|
46
|
+
if (!multiple) {
|
|
47
|
+
// Selección simple: solo permitir un elemento seleccionado
|
|
48
|
+
if (currentSelection.has(id)) {
|
|
49
|
+
// Deseleccionar el elemento actual
|
|
50
|
+
currentSelection.clear();
|
|
51
|
+
}
|
|
52
|
+
else {
|
|
53
|
+
// Seleccionar solo este elemento
|
|
54
|
+
currentSelection.clear();
|
|
55
|
+
currentSelection.add(id);
|
|
56
|
+
}
|
|
57
|
+
}
|
|
58
|
+
else if (currentSelection.has(id)) {
|
|
59
|
+
// Selección múltiple: deseleccionar elemento existente
|
|
60
|
+
currentSelection.delete(id);
|
|
61
|
+
}
|
|
62
|
+
else {
|
|
63
|
+
// Selección múltiple: agregar nuevo elemento
|
|
64
|
+
currentSelection.add(id);
|
|
65
|
+
}
|
|
66
|
+
selectedItems.set(currentSelection);
|
|
67
|
+
};
|
|
68
|
+
const isItemSelected = (item) => {
|
|
69
|
+
const id = item['id'];
|
|
70
|
+
return selectedItems().has(id);
|
|
71
|
+
};
|
|
72
|
+
const updateSelectAllStateWithData = (allItems) => {
|
|
73
|
+
const selectedCount = selectedItems().size;
|
|
74
|
+
const totalCount = allItems.length;
|
|
75
|
+
if (selectedCount === 0) {
|
|
76
|
+
selectAllChecked.set(false);
|
|
77
|
+
selectAllIndeterminate.set(false);
|
|
78
|
+
}
|
|
79
|
+
else if (selectedCount === totalCount) {
|
|
80
|
+
selectAllChecked.set(true);
|
|
81
|
+
selectAllIndeterminate.set(false);
|
|
82
|
+
}
|
|
83
|
+
else {
|
|
84
|
+
selectAllChecked.set(false);
|
|
85
|
+
selectAllIndeterminate.set(true);
|
|
86
|
+
}
|
|
87
|
+
};
|
|
88
|
+
const preserveSelection = (newData) => {
|
|
89
|
+
const currentSelection = selectedItems();
|
|
90
|
+
if (currentSelection.size === 0 || newData.length === 0) {
|
|
91
|
+
// Si no hay selección o no hay datos, limpiar la selección
|
|
92
|
+
clearSelection();
|
|
93
|
+
return;
|
|
94
|
+
}
|
|
95
|
+
// Obtener los IDs disponibles en los nuevos datos
|
|
96
|
+
const availableIds = new Set(newData.map(item => item['id']));
|
|
97
|
+
// Filtrar la selección actual para mantener solo los IDs que aún existen
|
|
98
|
+
const preservedSelection = new Set(Array.from(currentSelection).filter(id => availableIds.has(id)));
|
|
99
|
+
// Actualizar la selección con los IDs preservados
|
|
100
|
+
selectedItems.set(preservedSelection);
|
|
101
|
+
updateSelectAllStateWithData(newData);
|
|
102
|
+
};
|
|
103
|
+
const getSelectedItems = (allData) => {
|
|
104
|
+
const selectedIds = selectedItems();
|
|
105
|
+
return allData.filter(item => selectedIds.has(item['id']));
|
|
106
|
+
};
|
|
107
|
+
// Redefinir updateSelectAllState con acceso a los datos
|
|
108
|
+
const originalUpdateSelectAllState = updateSelectAllState;
|
|
109
|
+
const updateSelectAllStateWrapper = (allItems) => {
|
|
110
|
+
if (allItems) {
|
|
111
|
+
updateSelectAllStateWithData(allItems);
|
|
112
|
+
}
|
|
113
|
+
else {
|
|
114
|
+
originalUpdateSelectAllState();
|
|
115
|
+
}
|
|
116
|
+
};
|
|
117
|
+
return {
|
|
118
|
+
// Signals
|
|
119
|
+
selectedItems: selectedItems.asReadonly(),
|
|
120
|
+
selectAllChecked: selectAllChecked.asReadonly(),
|
|
121
|
+
selectAllIndeterminate: selectAllIndeterminate.asReadonly(),
|
|
122
|
+
// Métodos
|
|
123
|
+
clearSelection,
|
|
124
|
+
toggleSelectAll,
|
|
125
|
+
toggleItemSelection,
|
|
126
|
+
isItemSelected,
|
|
127
|
+
preserveSelection,
|
|
128
|
+
getSelectedItems,
|
|
129
|
+
updateSelectAllState: updateSelectAllStateWrapper,
|
|
130
|
+
};
|
|
131
|
+
}
|
|
132
|
+
/**
|
|
133
|
+
* Emite los elementos seleccionados a través del EventEmitter proporcionado
|
|
134
|
+
* @param selectionState - Estado de selección
|
|
135
|
+
* @param allData - Todos los datos de la tabla
|
|
136
|
+
* @param selectableEmitter - EventEmitter para emitir la selección
|
|
137
|
+
*/
|
|
138
|
+
emitSelection(selectionState, allData, selectableEmitter) {
|
|
139
|
+
const selectedItems = selectionState.getSelectedItems(allData);
|
|
140
|
+
selectableEmitter.emit(selectedItems);
|
|
141
|
+
}
|
|
142
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableSelectionService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
143
|
+
static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableSelectionService, providedIn: 'root' });
|
|
144
|
+
}
|
|
145
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableSelectionService, decorators: [{
|
|
146
|
+
type: Injectable,
|
|
147
|
+
args: [{
|
|
148
|
+
providedIn: 'root'
|
|
149
|
+
}]
|
|
150
|
+
}] });
|
|
151
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-selection.service.js","sourceRoot":"","sources":["../../../../../libs/ui/src/lib/table/table-selection.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, signal, EventEmitter } from '@angular/core';\nimport { ID } from './table.types';\n\n/**\n * Servicio para gestionar la selección de elementos en tablas C80\n *\n * Maneja:\n * - Selección simple y múltiple\n * - Estado de selección completa\n * - Preservación de selección tras actualizaciones de datos\n * - Emisión de eventos de selección\n */\n@Injectable({\n    providedIn: 'root'\n})\nexport class TableSelectionService {\n\n    /**\n     * Inicializa el estado de selección para una nueva tabla\n     * @returns Objeto con signals y métodos de selección\n     */\n    createSelectionState<T extends Record<string, unknown>>() {\n        const selectedItems = signal<Set<ID>>(new Set());\n        const selectAllChecked = signal(false);\n        const selectAllIndeterminate = signal(false);\n\n        const clearSelection = (): void => {\n            selectedItems.set(new Set());\n            updateSelectAllState();\n        };\n\n        const updateSelectAllState = (): void => {\n            // Esta función será actualizada cuando se tengan los datos disponibles\n            selectAllChecked.set(false);\n            selectAllIndeterminate.set(false);\n        };\n\n        const toggleSelectAll = (allItems: T[]): void => {\n            const currentSelection = selectedItems();\n\n            if (currentSelection.size === allItems.length) {\n                // Desmarcar todos los elementos\n                clearSelection();\n            } else {\n                // Marcar todos los elementos\n                const allIds = new Set(allItems.map(item => item['id'] as ID));\n                selectedItems.set(allIds);\n                updateSelectAllStateWithData(allItems);\n            }\n        };\n\n        const toggleItemSelection = (item: T, multiple: boolean): void => {\n            const id = item['id'] as ID;\n            const currentSelection = new Set(selectedItems());\n\n            if (!multiple) {\n                // Selección simple: solo permitir un elemento seleccionado\n                if (currentSelection.has(id)) {\n                    // Deseleccionar el elemento actual\n                    currentSelection.clear();\n                } else {\n                    // Seleccionar solo este elemento\n                    currentSelection.clear();\n                    currentSelection.add(id);\n                }\n            } else if (currentSelection.has(id)) {\n                // Selección múltiple: deseleccionar elemento existente\n                currentSelection.delete(id);\n            } else {\n                // Selección múltiple: agregar nuevo elemento\n                currentSelection.add(id);\n            }\n\n            selectedItems.set(currentSelection);\n        };\n\n        const isItemSelected = (item: T): boolean => {\n            const id = item['id'] as ID;\n            return selectedItems().has(id);\n        };\n\n        const updateSelectAllStateWithData = (allItems: T[]): void => {\n            const selectedCount = selectedItems().size;\n            const totalCount = allItems.length;\n\n            if (selectedCount === 0) {\n                selectAllChecked.set(false);\n                selectAllIndeterminate.set(false);\n            } else if (selectedCount === totalCount) {\n                selectAllChecked.set(true);\n                selectAllIndeterminate.set(false);\n            } else {\n                selectAllChecked.set(false);\n                selectAllIndeterminate.set(true);\n            }\n        };\n\n        const preserveSelection = (newData: T[]): void => {\n            const currentSelection = selectedItems();\n\n            if (currentSelection.size === 0 || newData.length === 0) {\n                // Si no hay selección o no hay datos, limpiar la selección\n                clearSelection();\n                return;\n            }\n\n            // Obtener los IDs disponibles en los nuevos datos\n            const availableIds = new Set(newData.map(item => item['id'] as ID));\n\n            // Filtrar la selección actual para mantener solo los IDs que aún existen\n            const preservedSelection = new Set(\n                Array.from(currentSelection).filter(id => availableIds.has(id))\n            );\n\n            // Actualizar la selección con los IDs preservados\n            selectedItems.set(preservedSelection);\n            updateSelectAllStateWithData(newData);\n        };\n\n        const getSelectedItems = (allData: T[]): T[] => {\n            const selectedIds = selectedItems();\n            return allData.filter(item => selectedIds.has(item['id'] as ID));\n        };\n\n        // Redefinir updateSelectAllState con acceso a los datos\n        const originalUpdateSelectAllState = updateSelectAllState;\n        const updateSelectAllStateWrapper = (allItems?: T[]): void => {\n            if (allItems) {\n                updateSelectAllStateWithData(allItems);\n            } else {\n                originalUpdateSelectAllState();\n            }\n        };\n\n        return {\n            // Signals\n            selectedItems: selectedItems.asReadonly(),\n            selectAllChecked: selectAllChecked.asReadonly(),\n            selectAllIndeterminate: selectAllIndeterminate.asReadonly(),\n\n            // Métodos\n            clearSelection,\n            toggleSelectAll,\n            toggleItemSelection,\n            isItemSelected,\n            preserveSelection,\n            getSelectedItems,\n            updateSelectAllState: updateSelectAllStateWrapper,\n        };\n    }\n\n    /**\n     * Emite los elementos seleccionados a través del EventEmitter proporcionado\n     * @param selectionState - Estado de selección\n     * @param allData - Todos los datos de la tabla\n     * @param selectableEmitter - EventEmitter para emitir la selección\n     */\n    emitSelection<T extends Record<string, unknown>>(\n        selectionState: ReturnType<typeof this.createSelectionState<T>>,\n        allData: T[],\n        selectableEmitter: EventEmitter<T[]>\n    ): void {\n        const selectedItems = selectionState.getSelectedItems(allData);\n        selectableEmitter.emit(selectedItems);\n    }\n}\n"]}
|