@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.
@@ -1,4 +1,4 @@
1
- import { Component, input } from '@angular/core';
1
+ import { Component, input, computed } from '@angular/core';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import * as i0 from "@angular/core";
4
4
  export class CardLevelComponent {
@@ -7,7 +7,7 @@ export class CardLevelComponent {
7
7
  _generatedId;
8
8
  get generatedId() {
9
9
  if (!this._generatedId) {
10
- const label = this.cardLevelData().label.toLowerCase().replace(/\s+/g, '-');
10
+ const label = this.cardLevelData().label.toLowerCase().replaceAll(/\s+/g, '-');
11
11
  const randomNum = Math.floor(Math.random() * 10000);
12
12
  this._generatedId = `${label}-${randomNum}`;
13
13
  }
@@ -18,11 +18,40 @@ export class CardLevelComponent {
18
18
  const calculatedWidth = baseWidth * this.size();
19
19
  return `${calculatedWidth}px`;
20
20
  }
21
+ // Detecta si el rango es bidireccional (min negativo, max positivo)
22
+ isBidirectional = computed(() => {
23
+ const data = this.cardLevelData();
24
+ return data.min < 0 && data.max > 0;
25
+ });
26
+ // Calcula el porcentaje de fill para barras bidireccionales
27
+ bidirectionalFillPercent = computed(() => {
28
+ const data = this.cardLevelData();
29
+ const totalRange = data.max - data.min;
30
+ return Math.abs(data.value / totalRange) * 100;
31
+ });
32
+ // Determina la dirección del fill (left/right)
33
+ fillDirection = computed(() => {
34
+ return this.cardLevelData().value >= 0 ? 'right' : 'left';
35
+ });
36
+ // Color según el valor y umbrales
37
+ fillColor = computed(() => {
38
+ const data = this.cardLevelData();
39
+ const absValue = Math.abs(data.value);
40
+ const absHigh = Math.abs(data.high);
41
+ const absLow = Math.abs(data.low);
42
+ if (absValue >= absHigh) {
43
+ return 'var(--color-danger)';
44
+ }
45
+ if (absValue >= absLow) {
46
+ return 'var(--color-warning)';
47
+ }
48
+ return 'var(--color-success)';
49
+ });
21
50
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CardLevelComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
22
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.1.0", version: "18.2.13", type: CardLevelComponent, isStandalone: true, selector: "c80-card-level", inputs: { cardLevelData: { classPropertyName: "cardLevelData", publicName: "cardLevelData", isSignal: true, isRequired: true, transformFunction: null }, size: { classPropertyName: "size", publicName: "size", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<div class=\"card-level-container\" [style.width]=\"cardWidth\">\r\n\r\n <label [for]=\"generatedId\">\r\n\r\n <strong>{{ cardLevelData().label }}</strong>\r\n\r\n <span class=\"value-display\">\r\n\r\n <meter [id]=\"generatedId\" [optimum]=\"cardLevelData().optimum\" [min]=\"cardLevelData().min\" [max]=\"cardLevelData().max\" [low]=\"cardLevelData().low\" [high]=\"cardLevelData().high\" [value]=\"cardLevelData().value\"> </meter>\r\n\r\n {{ cardLevelData().value }}{{ cardLevelData().unit }}\r\n\r\n </span>\r\n\r\n </label>\r\n\r\n</div>", styles: [".card-level-container{background:#fff;border-radius:5px;padding:8px;margin:4px;box-shadow:0 4px 6px #0000001a,0 1px 3px #00000014;min-width:100px;flex:1;transition:all .8s ease}.card-level-container label{display:flex;flex-direction:column;margin:0}.card-level-container label strong{font-size:12px;font-weight:600;color:#2d3748;text-transform:uppercase;letter-spacing:.5px;margin-bottom:4px}.card-level-container label .value-display{font-size:12px;font-weight:600;color:#1a202c;text-align:center;padding:0 5px;background:#4299e11a;border-radius:5px}.card-level-container label meter{width:100%;border-radius:5px;border:none;background:#e2e8f0}.card-level-container label meter::-webkit-meter-bar{background:#e2e8f0;border-radius:12px;box-shadow:inset 0 2px 4px #0000001a}.card-level-container label meter::-webkit-meter-optimum-value{background:linear-gradient(90deg,#48bb78,#38a169);border-radius:12px;transition:all .3s ease}.card-level-container label meter::-webkit-meter-suboptimum-value{background:linear-gradient(90deg,#ed8936,#dd6b20);border-radius:12px;transition:all .3s ease}.card-level-container label meter::-webkit-meter-even-less-good-value{background:linear-gradient(90deg,#f56565,#e53e3e);border-radius:12px;transition:all .3s ease}.card-level-container label meter::-moz-meter-bar{background:linear-gradient(90deg,#48bb78,#38a169);border-radius:12px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] });
51
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: CardLevelComponent, isStandalone: true, selector: "c80-card-level", inputs: { cardLevelData: { classPropertyName: "cardLevelData", publicName: "cardLevelData", isSignal: true, isRequired: true, transformFunction: null }, size: { classPropertyName: "size", publicName: "size", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<div class=\"card-level-container\" [style.width]=\"cardWidth\">\r\n <label [for]=\"generatedId\">\r\n <strong>{{ cardLevelData().label }}</strong>\r\n\r\n <span class=\"value-display\">\r\n @if (isBidirectional()) {\r\n <!-- Barra bidireccional desde el centro -->\r\n <div class=\"bidirectional-meter\">\r\n <div class=\"meter-track\">\r\n <div class=\"center-line\"></div>\r\n <div class=\"meter-fill\" [class.fill-left]=\"fillDirection() === 'left'\" [class.fill-right]=\"fillDirection() === 'right'\" [style.width.%]=\"bidirectionalFillPercent()\" [style.background]=\"fillColor()\">\r\n </div>\r\n </div>\r\n </div>\r\n } @else {\r\n <!-- Meter est\u00E1ndar para rangos unidireccionales -->\r\n <meter [id]=\"generatedId\" [optimum]=\"cardLevelData().optimum\" [min]=\"cardLevelData().min\" [max]=\"cardLevelData().max\" [low]=\"cardLevelData().low\" [high]=\"cardLevelData().high\" [value]=\"cardLevelData().value\">\r\n </meter>\r\n }\r\n\r\n {{ cardLevelData().value }}{{ cardLevelData().unit }}\r\n </span>\r\n </label>\r\n</div>", styles: [".card-level-container{background:#fff;border-radius:5px;padding:8px;margin:4px;box-shadow:0 4px 6px #0000001a,0 1px 3px #00000014;min-width:100px;flex:1}.card-level-container label{display:flex;flex-direction:column;margin:0}.card-level-container label strong{font-size:12px;font-weight:600;color:#2d3748;text-transform:uppercase;letter-spacing:.5px;margin-bottom:4px}.card-level-container label .value-display{font-size:12px;font-weight:600;color:#1a202c;text-align:center;padding:0 5px;background:#4299e11a;border-radius:5px}.card-level-container label .bidirectional-meter{width:100%;margin-bottom:4px}.card-level-container label .bidirectional-meter .meter-track{position:relative;width:100%;height:10px;background:#e2e8f0;border-radius:12px;box-shadow:inset 0 2px 4px #0000001a;overflow:hidden}.card-level-container label .bidirectional-meter .meter-track .center-line{position:absolute;left:50%;top:0;bottom:0;width:4px;background:#4a5568;transform:translate(-50%);z-index:1}.card-level-container label .bidirectional-meter .meter-track .meter-fill{position:absolute;top:0;bottom:0;border-radius:5px}.card-level-container label .bidirectional-meter .meter-track .meter-fill.fill-right{left:50%}.card-level-container label .bidirectional-meter .meter-track .meter-fill.fill-left{right:50%}.card-level-container label meter{width:100%;border-radius:12px;border:none;background:#e2e8f0;margin-bottom:4px}.card-level-container label meter::-webkit-meter-bar{background:#e2e8f0;border-radius:12px;box-shadow:inset 0 2px 4px #0000001a}.card-level-container label meter::-webkit-meter-optimum-value{background:linear-gradient(90deg,#48bb78,#38a169);border-radius:12px}.card-level-container label meter::-webkit-meter-suboptimum-value{background:linear-gradient(90deg,#ed8936,#dd6b20);border-radius:12px}.card-level-container label meter::-webkit-meter-even-less-good-value{background:linear-gradient(90deg,#f56565,#e53e3e);border-radius:12px}.card-level-container label meter::-moz-meter-bar{background:linear-gradient(90deg,#48bb78,#38a169);border-radius:12px}:host{--color-success: linear-gradient(90deg, #48bb78, #38a169);--color-warning: linear-gradient(90deg, #ed8936, #dd6b20);--color-danger: linear-gradient(90deg, #f56565, #e53e3e)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }] });
23
52
  }
24
53
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: CardLevelComponent, decorators: [{
25
54
  type: Component,
26
- args: [{ selector: 'c80-card-level', standalone: true, imports: [CommonModule], template: "<div class=\"card-level-container\" [style.width]=\"cardWidth\">\r\n\r\n <label [for]=\"generatedId\">\r\n\r\n <strong>{{ cardLevelData().label }}</strong>\r\n\r\n <span class=\"value-display\">\r\n\r\n <meter [id]=\"generatedId\" [optimum]=\"cardLevelData().optimum\" [min]=\"cardLevelData().min\" [max]=\"cardLevelData().max\" [low]=\"cardLevelData().low\" [high]=\"cardLevelData().high\" [value]=\"cardLevelData().value\"> </meter>\r\n\r\n {{ cardLevelData().value }}{{ cardLevelData().unit }}\r\n\r\n </span>\r\n\r\n </label>\r\n\r\n</div>", styles: [".card-level-container{background:#fff;border-radius:5px;padding:8px;margin:4px;box-shadow:0 4px 6px #0000001a,0 1px 3px #00000014;min-width:100px;flex:1;transition:all .8s ease}.card-level-container label{display:flex;flex-direction:column;margin:0}.card-level-container label strong{font-size:12px;font-weight:600;color:#2d3748;text-transform:uppercase;letter-spacing:.5px;margin-bottom:4px}.card-level-container label .value-display{font-size:12px;font-weight:600;color:#1a202c;text-align:center;padding:0 5px;background:#4299e11a;border-radius:5px}.card-level-container label meter{width:100%;border-radius:5px;border:none;background:#e2e8f0}.card-level-container label meter::-webkit-meter-bar{background:#e2e8f0;border-radius:12px;box-shadow:inset 0 2px 4px #0000001a}.card-level-container label meter::-webkit-meter-optimum-value{background:linear-gradient(90deg,#48bb78,#38a169);border-radius:12px;transition:all .3s ease}.card-level-container label meter::-webkit-meter-suboptimum-value{background:linear-gradient(90deg,#ed8936,#dd6b20);border-radius:12px;transition:all .3s ease}.card-level-container label meter::-webkit-meter-even-less-good-value{background:linear-gradient(90deg,#f56565,#e53e3e);border-radius:12px;transition:all .3s ease}.card-level-container label meter::-moz-meter-bar{background:linear-gradient(90deg,#48bb78,#38a169);border-radius:12px}\n"] }]
55
+ args: [{ selector: 'c80-card-level', standalone: true, imports: [CommonModule], template: "<div class=\"card-level-container\" [style.width]=\"cardWidth\">\r\n <label [for]=\"generatedId\">\r\n <strong>{{ cardLevelData().label }}</strong>\r\n\r\n <span class=\"value-display\">\r\n @if (isBidirectional()) {\r\n <!-- Barra bidireccional desde el centro -->\r\n <div class=\"bidirectional-meter\">\r\n <div class=\"meter-track\">\r\n <div class=\"center-line\"></div>\r\n <div class=\"meter-fill\" [class.fill-left]=\"fillDirection() === 'left'\" [class.fill-right]=\"fillDirection() === 'right'\" [style.width.%]=\"bidirectionalFillPercent()\" [style.background]=\"fillColor()\">\r\n </div>\r\n </div>\r\n </div>\r\n } @else {\r\n <!-- Meter est\u00E1ndar para rangos unidireccionales -->\r\n <meter [id]=\"generatedId\" [optimum]=\"cardLevelData().optimum\" [min]=\"cardLevelData().min\" [max]=\"cardLevelData().max\" [low]=\"cardLevelData().low\" [high]=\"cardLevelData().high\" [value]=\"cardLevelData().value\">\r\n </meter>\r\n }\r\n\r\n {{ cardLevelData().value }}{{ cardLevelData().unit }}\r\n </span>\r\n </label>\r\n</div>", styles: [".card-level-container{background:#fff;border-radius:5px;padding:8px;margin:4px;box-shadow:0 4px 6px #0000001a,0 1px 3px #00000014;min-width:100px;flex:1}.card-level-container label{display:flex;flex-direction:column;margin:0}.card-level-container label strong{font-size:12px;font-weight:600;color:#2d3748;text-transform:uppercase;letter-spacing:.5px;margin-bottom:4px}.card-level-container label .value-display{font-size:12px;font-weight:600;color:#1a202c;text-align:center;padding:0 5px;background:#4299e11a;border-radius:5px}.card-level-container label .bidirectional-meter{width:100%;margin-bottom:4px}.card-level-container label .bidirectional-meter .meter-track{position:relative;width:100%;height:10px;background:#e2e8f0;border-radius:12px;box-shadow:inset 0 2px 4px #0000001a;overflow:hidden}.card-level-container label .bidirectional-meter .meter-track .center-line{position:absolute;left:50%;top:0;bottom:0;width:4px;background:#4a5568;transform:translate(-50%);z-index:1}.card-level-container label .bidirectional-meter .meter-track .meter-fill{position:absolute;top:0;bottom:0;border-radius:5px}.card-level-container label .bidirectional-meter .meter-track .meter-fill.fill-right{left:50%}.card-level-container label .bidirectional-meter .meter-track .meter-fill.fill-left{right:50%}.card-level-container label meter{width:100%;border-radius:12px;border:none;background:#e2e8f0;margin-bottom:4px}.card-level-container label meter::-webkit-meter-bar{background:#e2e8f0;border-radius:12px;box-shadow:inset 0 2px 4px #0000001a}.card-level-container label meter::-webkit-meter-optimum-value{background:linear-gradient(90deg,#48bb78,#38a169);border-radius:12px}.card-level-container label meter::-webkit-meter-suboptimum-value{background:linear-gradient(90deg,#ed8936,#dd6b20);border-radius:12px}.card-level-container label meter::-webkit-meter-even-less-good-value{background:linear-gradient(90deg,#f56565,#e53e3e);border-radius:12px}.card-level-container label meter::-moz-meter-bar{background:linear-gradient(90deg,#48bb78,#38a169);border-radius:12px}:host{--color-success: linear-gradient(90deg, #48bb78, #38a169);--color-warning: linear-gradient(90deg, #ed8936, #dd6b20);--color-danger: linear-gradient(90deg, #f56565, #e53e3e)}\n"] }]
27
56
  }] });
28
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FyZC1sZXZlbC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3VpL3NyYy9saWIvY2FyZC1sZXZlbC9jYXJkLWxldmVsLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvc3JjL2xpYi9jYXJkLWxldmVsL2NhcmQtbGV2ZWwuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDakQsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDOztBQVUvQyxNQUFNLE9BQU8sa0JBQWtCO0lBQzdCLGFBQWEsR0FBRyxLQUFLLENBQUMsUUFBUSxFQUFpQixDQUFDO0lBQ2hELElBQUksR0FBRyxLQUFLLENBQVMsQ0FBQyxDQUFDLENBQUMsQ0FBQyx3Q0FBd0M7SUFFekQsWUFBWSxDQUFVO0lBRTlCLElBQUksV0FBVztRQUNiLElBQUksQ0FBQyxJQUFJLENBQUMsWUFBWSxFQUFFLENBQUM7WUFDdkIsTUFBTSxLQUFLLEdBQUcsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDLEtBQUssQ0FBQyxXQUFXLEVBQUUsQ0FBQyxPQUFPLENBQUMsTUFBTSxFQUFFLEdBQUcsQ0FBQyxDQUFDO1lBQzVFLE1BQU0sU0FBUyxHQUFHLElBQUksQ0FBQyxLQUFLLENBQUMsSUFBSSxDQUFDLE1BQU0sRUFBRSxHQUFHLEtBQUssQ0FBQyxDQUFDO1lBQ3BELElBQUksQ0FBQyxZQUFZLEdBQUcsR0FBRyxLQUFLLElBQUksU0FBUyxFQUFFLENBQUM7UUFDOUMsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLFlBQVksQ0FBQztJQUMzQixDQUFDO0lBRUQsSUFBSSxTQUFTO1FBQ1gsTUFBTSxTQUFTLEdBQUcsR0FBRyxDQUFDO1FBQ3RCLE1BQU0sZUFBZSxHQUFHLFNBQVMsR0FBRyxJQUFJLENBQUMsSUFBSSxFQUFFLENBQUM7UUFDaEQsT0FBTyxHQUFHLGVBQWUsSUFBSSxDQUFDO0lBQ2hDLENBQUM7d0dBbkJVLGtCQUFrQjs0RkFBbEIsa0JBQWtCLDBWQ1gvQixzakJBZ0JNLHE1Q0RUTSxZQUFZOzs0RkFJWCxrQkFBa0I7a0JBUDlCLFNBQVM7K0JBQ0UsZ0JBQWdCLGNBQ2QsSUFBSSxXQUNQLENBQUMsWUFBWSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBpbnB1dCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xyXG5pbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xyXG5pbXBvcnQgeyBDYXJkTGV2ZWxEYXRhIH0gZnJvbSAnLi9jYXJkLWxldmVsLmludGVyZmFjZSc7XHJcblxyXG5AQ29tcG9uZW50KHtcclxuICBzZWxlY3RvcjogJ2M4MC1jYXJkLWxldmVsJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdLFxyXG4gIHRlbXBsYXRlVXJsOiAnLi9jYXJkLWxldmVsLmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybDogJy4vY2FyZC1sZXZlbC5jb21wb25lbnQuc2NzcydcclxufSlcclxuZXhwb3J0IGNsYXNzIENhcmRMZXZlbENvbXBvbmVudCB7XHJcbiAgY2FyZExldmVsRGF0YSA9IGlucHV0LnJlcXVpcmVkPENhcmRMZXZlbERhdGE+KCk7XHJcbiAgc2l6ZSA9IGlucHV0PG51bWJlcj4oMSk7IC8vIE11bHRpcGxpY2Fkb3IgZGVsIHRhbWHDsW8gYmFzZSAoMjIwcHgpXHJcblxyXG4gIHByaXZhdGUgX2dlbmVyYXRlZElkPzogc3RyaW5nO1xyXG5cclxuICBnZXQgZ2VuZXJhdGVkSWQoKTogc3RyaW5nIHtcclxuICAgIGlmICghdGhpcy5fZ2VuZXJhdGVkSWQpIHtcclxuICAgICAgY29uc3QgbGFiZWwgPSB0aGlzLmNhcmRMZXZlbERhdGEoKS5sYWJlbC50b0xvd2VyQ2FzZSgpLnJlcGxhY2UoL1xccysvZywgJy0nKTtcclxuICAgICAgY29uc3QgcmFuZG9tTnVtID0gTWF0aC5mbG9vcihNYXRoLnJhbmRvbSgpICogMTAwMDApO1xyXG4gICAgICB0aGlzLl9nZW5lcmF0ZWRJZCA9IGAke2xhYmVsfS0ke3JhbmRvbU51bX1gO1xyXG4gICAgfVxyXG4gICAgcmV0dXJuIHRoaXMuX2dlbmVyYXRlZElkO1xyXG4gIH1cclxuXHJcbiAgZ2V0IGNhcmRXaWR0aCgpOiBzdHJpbmcge1xyXG4gICAgY29uc3QgYmFzZVdpZHRoID0gMTQwO1xyXG4gICAgY29uc3QgY2FsY3VsYXRlZFdpZHRoID0gYmFzZVdpZHRoICogdGhpcy5zaXplKCk7XHJcbiAgICByZXR1cm4gYCR7Y2FsY3VsYXRlZFdpZHRofXB4YDtcclxuICB9XHJcbn1cclxuIiwiPGRpdiBjbGFzcz1cImNhcmQtbGV2ZWwtY29udGFpbmVyXCIgW3N0eWxlLndpZHRoXT1cImNhcmRXaWR0aFwiPlxyXG5cclxuICA8bGFiZWwgW2Zvcl09XCJnZW5lcmF0ZWRJZFwiPlxyXG5cclxuICAgIDxzdHJvbmc+e3sgY2FyZExldmVsRGF0YSgpLmxhYmVsIH19PC9zdHJvbmc+XHJcblxyXG4gICAgPHNwYW4gY2xhc3M9XCJ2YWx1ZS1kaXNwbGF5XCI+XHJcblxyXG4gICAgICA8bWV0ZXIgW2lkXT1cImdlbmVyYXRlZElkXCIgW29wdGltdW1dPVwiY2FyZExldmVsRGF0YSgpLm9wdGltdW1cIiBbbWluXT1cImNhcmRMZXZlbERhdGEoKS5taW5cIiBbbWF4XT1cImNhcmRMZXZlbERhdGEoKS5tYXhcIiBbbG93XT1cImNhcmRMZXZlbERhdGEoKS5sb3dcIiBbaGlnaF09XCJjYXJkTGV2ZWxEYXRhKCkuaGlnaFwiIFt2YWx1ZV09XCJjYXJkTGV2ZWxEYXRhKCkudmFsdWVcIj4gPC9tZXRlcj5cclxuXHJcbiAgICAgIHt7IGNhcmRMZXZlbERhdGEoKS52YWx1ZSB9fXt7IGNhcmRMZXZlbERhdGEoKS51bml0IH19XHJcblxyXG4gICAgPC9zcGFuPlxyXG5cclxuICA8L2xhYmVsPlxyXG5cclxuPC9kaXY+Il19
57
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FyZC1sZXZlbC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3VpL3NyYy9saWIvY2FyZC1sZXZlbC9jYXJkLWxldmVsLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL2xpYnMvdWkvc3JjL2xpYi9jYXJkLWxldmVsL2NhcmQtbGV2ZWwuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQzs7QUFXL0MsTUFBTSxPQUFPLGtCQUFrQjtJQUM3QixhQUFhLEdBQUcsS0FBSyxDQUFDLFFBQVEsRUFBaUIsQ0FBQztJQUNoRCxJQUFJLEdBQUcsS0FBSyxDQUFTLENBQUMsQ0FBQyxDQUFDLENBQUMsd0NBQXdDO0lBRXpELFlBQVksQ0FBVTtJQUU5QixJQUFJLFdBQVc7UUFDYixJQUFJLENBQUMsSUFBSSxDQUFDLFlBQVksRUFBRSxDQUFDO1lBQ3ZCLE1BQU0sS0FBSyxHQUFHLElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQyxLQUFLLENBQUMsV0FBVyxFQUFFLENBQUMsVUFBVSxDQUFDLE1BQU0sRUFBRSxHQUFHLENBQUMsQ0FBQztZQUMvRSxNQUFNLFNBQVMsR0FBRyxJQUFJLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxNQUFNLEVBQUUsR0FBRyxLQUFLLENBQUMsQ0FBQztZQUNwRCxJQUFJLENBQUMsWUFBWSxHQUFHLEdBQUcsS0FBSyxJQUFJLFNBQVMsRUFBRSxDQUFDO1FBQzlDLENBQUM7UUFDRCxPQUFPLElBQUksQ0FBQyxZQUFZLENBQUM7SUFDM0IsQ0FBQztJQUVELElBQUksU0FBUztRQUNYLE1BQU0sU0FBUyxHQUFHLEdBQUcsQ0FBQztRQUN0QixNQUFNLGVBQWUsR0FBRyxTQUFTLEdBQUcsSUFBSSxDQUFDLElBQUksRUFBRSxDQUFDO1FBQ2hELE9BQU8sR0FBRyxlQUFlLElBQUksQ0FBQztJQUNoQyxDQUFDO0lBRUQsb0VBQW9FO0lBQ3BFLGVBQWUsR0FBRyxRQUFRLENBQUMsR0FBRyxFQUFFO1FBQzlCLE1BQU0sSUFBSSxHQUFHLElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztRQUNsQyxPQUFPLElBQUksQ0FBQyxHQUFHLEdBQUcsQ0FBQyxJQUFJLElBQUksQ0FBQyxHQUFHLEdBQUcsQ0FBQyxDQUFDO0lBQ3RDLENBQUMsQ0FBQyxDQUFDO0lBRUgsNERBQTREO0lBQzVELHdCQUF3QixHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUU7UUFDdkMsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO1FBQ2xDLE1BQU0sVUFBVSxHQUFHLElBQUksQ0FBQyxHQUFHLEdBQUcsSUFBSSxDQUFDLEdBQUcsQ0FBQztRQUN2QyxPQUFPLElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLEtBQUssR0FBRyxVQUFVLENBQUMsR0FBRyxHQUFHLENBQUM7SUFDakQsQ0FBQyxDQUFDLENBQUM7SUFFSCwrQ0FBK0M7SUFDL0MsYUFBYSxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUU7UUFDNUIsT0FBTyxJQUFJLENBQUMsYUFBYSxFQUFFLENBQUMsS0FBSyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsT0FBTyxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUM7SUFDNUQsQ0FBQyxDQUFDLENBQUM7SUFFSCxrQ0FBa0M7SUFDbEMsU0FBUyxHQUFHLFFBQVEsQ0FBQyxHQUFHLEVBQUU7UUFDeEIsTUFBTSxJQUFJLEdBQUcsSUFBSSxDQUFDLGFBQWEsRUFBRSxDQUFDO1FBQ2xDLE1BQU0sUUFBUSxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3RDLE1BQU0sT0FBTyxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDO1FBQ3BDLE1BQU0sTUFBTSxHQUFHLElBQUksQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDO1FBRWxDLElBQUksUUFBUSxJQUFJLE9BQU8sRUFBRSxDQUFDO1lBQ3hCLE9BQU8scUJBQXFCLENBQUM7UUFDL0IsQ0FBQztRQUNELElBQUksUUFBUSxJQUFJLE1BQU0sRUFBRSxDQUFDO1lBQ3ZCLE9BQU8sc0JBQXNCLENBQUM7UUFDaEMsQ0FBQztRQUNELE9BQU8sc0JBQXNCLENBQUM7SUFDaEMsQ0FBQyxDQUFDLENBQUM7d0dBckRRLGtCQUFrQjs0RkFBbEIsa0JBQWtCLDBWQ1ovQix3bkNBdUJNLDB2RURmTSxZQUFZOzs0RkFJWCxrQkFBa0I7a0JBUjlCLFNBQVM7K0JBRUUsZ0JBQWdCLGNBQ2QsSUFBSSxXQUNQLENBQUMsWUFBWSxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBpbnB1dCwgY29tcHV0ZWQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcclxuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcclxuaW1wb3J0IHsgQ2FyZExldmVsRGF0YSB9IGZyb20gJy4vY2FyZC1sZXZlbC5pbnRlcmZhY2UnO1xyXG5cclxuQENvbXBvbmVudCh7XHJcbiAgLy8gZXNsaW50LWRpc2FibGUtbmV4dC1saW5lIEBhbmd1bGFyLWVzbGludC9jb21wb25lbnQtc2VsZWN0b3JcclxuICBzZWxlY3RvcjogJ2M4MC1jYXJkLWxldmVsJyxcclxuICBzdGFuZGFsb25lOiB0cnVlLFxyXG4gIGltcG9ydHM6IFtDb21tb25Nb2R1bGVdLFxyXG4gIHRlbXBsYXRlVXJsOiAnLi9jYXJkLWxldmVsLmNvbXBvbmVudC5odG1sJyxcclxuICBzdHlsZVVybDogJy4vY2FyZC1sZXZlbC5jb21wb25lbnQuc2NzcydcclxufSlcclxuZXhwb3J0IGNsYXNzIENhcmRMZXZlbENvbXBvbmVudCB7XHJcbiAgY2FyZExldmVsRGF0YSA9IGlucHV0LnJlcXVpcmVkPENhcmRMZXZlbERhdGE+KCk7XHJcbiAgc2l6ZSA9IGlucHV0PG51bWJlcj4oMSk7IC8vIE11bHRpcGxpY2Fkb3IgZGVsIHRhbWHDsW8gYmFzZSAoMjIwcHgpXHJcblxyXG4gIHByaXZhdGUgX2dlbmVyYXRlZElkPzogc3RyaW5nO1xyXG5cclxuICBnZXQgZ2VuZXJhdGVkSWQoKTogc3RyaW5nIHtcclxuICAgIGlmICghdGhpcy5fZ2VuZXJhdGVkSWQpIHtcclxuICAgICAgY29uc3QgbGFiZWwgPSB0aGlzLmNhcmRMZXZlbERhdGEoKS5sYWJlbC50b0xvd2VyQ2FzZSgpLnJlcGxhY2VBbGwoL1xccysvZywgJy0nKTtcclxuICAgICAgY29uc3QgcmFuZG9tTnVtID0gTWF0aC5mbG9vcihNYXRoLnJhbmRvbSgpICogMTAwMDApO1xyXG4gICAgICB0aGlzLl9nZW5lcmF0ZWRJZCA9IGAke2xhYmVsfS0ke3JhbmRvbU51bX1gO1xyXG4gICAgfVxyXG4gICAgcmV0dXJuIHRoaXMuX2dlbmVyYXRlZElkO1xyXG4gIH1cclxuXHJcbiAgZ2V0IGNhcmRXaWR0aCgpOiBzdHJpbmcge1xyXG4gICAgY29uc3QgYmFzZVdpZHRoID0gMTQwO1xyXG4gICAgY29uc3QgY2FsY3VsYXRlZFdpZHRoID0gYmFzZVdpZHRoICogdGhpcy5zaXplKCk7XHJcbiAgICByZXR1cm4gYCR7Y2FsY3VsYXRlZFdpZHRofXB4YDtcclxuICB9XHJcblxyXG4gIC8vIERldGVjdGEgc2kgZWwgcmFuZ28gZXMgYmlkaXJlY2Npb25hbCAobWluIG5lZ2F0aXZvLCBtYXggcG9zaXRpdm8pXHJcbiAgaXNCaWRpcmVjdGlvbmFsID0gY29tcHV0ZWQoKCkgPT4ge1xyXG4gICAgY29uc3QgZGF0YSA9IHRoaXMuY2FyZExldmVsRGF0YSgpO1xyXG4gICAgcmV0dXJuIGRhdGEubWluIDwgMCAmJiBkYXRhLm1heCA+IDA7XHJcbiAgfSk7XHJcblxyXG4gIC8vIENhbGN1bGEgZWwgcG9yY2VudGFqZSBkZSBmaWxsIHBhcmEgYmFycmFzIGJpZGlyZWNjaW9uYWxlc1xyXG4gIGJpZGlyZWN0aW9uYWxGaWxsUGVyY2VudCA9IGNvbXB1dGVkKCgpID0+IHtcclxuICAgIGNvbnN0IGRhdGEgPSB0aGlzLmNhcmRMZXZlbERhdGEoKTtcclxuICAgIGNvbnN0IHRvdGFsUmFuZ2UgPSBkYXRhLm1heCAtIGRhdGEubWluO1xyXG4gICAgcmV0dXJuIE1hdGguYWJzKGRhdGEudmFsdWUgLyB0b3RhbFJhbmdlKSAqIDEwMDtcclxuICB9KTtcclxuXHJcbiAgLy8gRGV0ZXJtaW5hIGxhIGRpcmVjY2nDs24gZGVsIGZpbGwgKGxlZnQvcmlnaHQpXHJcbiAgZmlsbERpcmVjdGlvbiA9IGNvbXB1dGVkKCgpID0+IHtcclxuICAgIHJldHVybiB0aGlzLmNhcmRMZXZlbERhdGEoKS52YWx1ZSA+PSAwID8gJ3JpZ2h0JyA6ICdsZWZ0JztcclxuICB9KTtcclxuXHJcbiAgLy8gQ29sb3Igc2Vnw7puIGVsIHZhbG9yIHkgdW1icmFsZXNcclxuICBmaWxsQ29sb3IgPSBjb21wdXRlZCgoKSA9PiB7XHJcbiAgICBjb25zdCBkYXRhID0gdGhpcy5jYXJkTGV2ZWxEYXRhKCk7XHJcbiAgICBjb25zdCBhYnNWYWx1ZSA9IE1hdGguYWJzKGRhdGEudmFsdWUpO1xyXG4gICAgY29uc3QgYWJzSGlnaCA9IE1hdGguYWJzKGRhdGEuaGlnaCk7XHJcbiAgICBjb25zdCBhYnNMb3cgPSBNYXRoLmFicyhkYXRhLmxvdyk7XHJcblxyXG4gICAgaWYgKGFic1ZhbHVlID49IGFic0hpZ2gpIHtcclxuICAgICAgcmV0dXJuICd2YXIoLS1jb2xvci1kYW5nZXIpJztcclxuICAgIH1cclxuICAgIGlmIChhYnNWYWx1ZSA+PSBhYnNMb3cpIHtcclxuICAgICAgcmV0dXJuICd2YXIoLS1jb2xvci13YXJuaW5nKSc7XHJcbiAgICB9XHJcbiAgICByZXR1cm4gJ3ZhcigtLWNvbG9yLXN1Y2Nlc3MpJztcclxuICB9KTtcclxufVxyXG4iLCI8ZGl2IGNsYXNzPVwiY2FyZC1sZXZlbC1jb250YWluZXJcIiBbc3R5bGUud2lkdGhdPVwiY2FyZFdpZHRoXCI+XHJcbiAgPGxhYmVsIFtmb3JdPVwiZ2VuZXJhdGVkSWRcIj5cclxuICAgIDxzdHJvbmc+e3sgY2FyZExldmVsRGF0YSgpLmxhYmVsIH19PC9zdHJvbmc+XHJcblxyXG4gICAgPHNwYW4gY2xhc3M9XCJ2YWx1ZS1kaXNwbGF5XCI+XHJcbiAgICAgIEBpZiAoaXNCaWRpcmVjdGlvbmFsKCkpIHtcclxuICAgICAgPCEtLSBCYXJyYSBiaWRpcmVjY2lvbmFsIGRlc2RlIGVsIGNlbnRybyAtLT5cclxuICAgICAgPGRpdiBjbGFzcz1cImJpZGlyZWN0aW9uYWwtbWV0ZXJcIj5cclxuICAgICAgICA8ZGl2IGNsYXNzPVwibWV0ZXItdHJhY2tcIj5cclxuICAgICAgICAgIDxkaXYgY2xhc3M9XCJjZW50ZXItbGluZVwiPjwvZGl2PlxyXG4gICAgICAgICAgPGRpdiBjbGFzcz1cIm1ldGVyLWZpbGxcIiBbY2xhc3MuZmlsbC1sZWZ0XT1cImZpbGxEaXJlY3Rpb24oKSA9PT0gJ2xlZnQnXCIgW2NsYXNzLmZpbGwtcmlnaHRdPVwiZmlsbERpcmVjdGlvbigpID09PSAncmlnaHQnXCIgW3N0eWxlLndpZHRoLiVdPVwiYmlkaXJlY3Rpb25hbEZpbGxQZXJjZW50KClcIiBbc3R5bGUuYmFja2dyb3VuZF09XCJmaWxsQ29sb3IoKVwiPlxyXG4gICAgICAgICAgPC9kaXY+XHJcbiAgICAgICAgPC9kaXY+XHJcbiAgICAgIDwvZGl2PlxyXG4gICAgICB9IEBlbHNlIHtcclxuICAgICAgPCEtLSBNZXRlciBlc3TDoW5kYXIgcGFyYSByYW5nb3MgdW5pZGlyZWNjaW9uYWxlcyAtLT5cclxuICAgICAgPG1ldGVyIFtpZF09XCJnZW5lcmF0ZWRJZFwiIFtvcHRpbXVtXT1cImNhcmRMZXZlbERhdGEoKS5vcHRpbXVtXCIgW21pbl09XCJjYXJkTGV2ZWxEYXRhKCkubWluXCIgW21heF09XCJjYXJkTGV2ZWxEYXRhKCkubWF4XCIgW2xvd109XCJjYXJkTGV2ZWxEYXRhKCkubG93XCIgW2hpZ2hdPVwiY2FyZExldmVsRGF0YSgpLmhpZ2hcIiBbdmFsdWVdPVwiY2FyZExldmVsRGF0YSgpLnZhbHVlXCI+XHJcbiAgICAgIDwvbWV0ZXI+XHJcbiAgICAgIH1cclxuXHJcbiAgICAgIHt7IGNhcmRMZXZlbERhdGEoKS52YWx1ZSB9fXt7IGNhcmRMZXZlbERhdGEoKS51bml0IH19XHJcbiAgICA8L3NwYW4+XHJcbiAgPC9sYWJlbD5cclxuPC9kaXY+Il19
@@ -1,2 +1,8 @@
1
+ export * from './table.types';
1
2
  export * from './table.component';
2
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3VpL3NyYy9saWIvdGFibGUvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxtQkFBbUIsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vdGFibGUuY29tcG9uZW50JztcbiJdfQ==
3
+ export * from './table-data-converter.service';
4
+ export * from './table-column-visibility.service';
5
+ export * from './table-data-utils.service';
6
+ export * from './table-selection.service';
7
+ export * from './table-crud-state.service';
8
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3VpL3NyYy9saWIvdGFibGUvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxlQUFlLENBQUM7QUFDOUIsY0FBYyxtQkFBbUIsQ0FBQztBQUNsQyxjQUFjLGdDQUFnQyxDQUFDO0FBQy9DLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYyw0QkFBNEIsQ0FBQztBQUMzQyxjQUFjLDJCQUEyQixDQUFDO0FBQzFDLGNBQWMsNEJBQTRCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL3RhYmxlLnR5cGVzJztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUtZGF0YS1jb252ZXJ0ZXIuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLWNvbHVtbi12aXNpYmlsaXR5LnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS1kYXRhLXV0aWxzLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS1zZWxlY3Rpb24uc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLWNydWQtc3RhdGUuc2VydmljZSc7XG4iXX0=
@@ -0,0 +1,156 @@
1
+ import { Injectable, inject } from '@angular/core';
2
+ import { TableDataConverterService } from './table-data-converter.service';
3
+ import * as i0 from "@angular/core";
4
+ /**
5
+ * Servicio para gestionar la lógica de visibilidad de columnas en tablas C80
6
+ *
7
+ * Maneja las reglas complejas de visibilidad basadas en:
8
+ * - Configuración explícita (visible: false)
9
+ * - Ocultación automática (hideIfAllValuesAreNull)
10
+ * - Estado de creación y edición
11
+ */
12
+ export class TableColumnVisibilityService {
13
+ dataConverter = inject(TableDataConverterService);
14
+ /**
15
+ * Actualiza las keys de columnas visibles basándose en el estado actual
16
+ * @param columns - Definiciones de columnas
17
+ * @param data - Datos actuales de la tabla
18
+ * @param creating - Si está en modo creación
19
+ * @param editing - ID de fila en edición (null si no hay edición)
20
+ * @returns Array de accessors de columnas visibles
21
+ */
22
+ updateVisibleKeys(columns, data, creating, editing) {
23
+ const visibleColumns = columns.filter(col => this.isColumnVisibleInHeader(col, data, creating, editing));
24
+ return visibleColumns.map(col => col.accessor);
25
+ }
26
+ /**
27
+ * Determina si una columna debe ser visible basándose en su configuración y datos
28
+ * @param column - La definición de la columna
29
+ * @param data - Datos actuales de la tabla
30
+ * @param forceShowInCreation - Si es true, ignora hideIfAllValuesAreNull (usado en modo creación)
31
+ * @returns true si la columna debe ser visible
32
+ */
33
+ isColumnVisible(column, data, forceShowInCreation = false) {
34
+ // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar
35
+ if (column.visible === false) {
36
+ return false;
37
+ }
38
+ // PRIORIDAD 2: Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
39
+ // EXCEPCIÓN: En modo creación, mostramos estas columnas para permitir entrada de datos
40
+ if (column.hideIfAllValuesAreNull === true && !forceShowInCreation && this.areAllColumnValuesEmpty(column, data)) {
41
+ return false;
42
+ }
43
+ // Por defecto: mostrar
44
+ return true;
45
+ }
46
+ /**
47
+ * Determina si una columna debe ser visible en los headers
48
+ * @param column - La definición de la columna
49
+ * @param data - Datos actuales de la tabla
50
+ * @param creating - Si está en modo creación
51
+ * @param editing - ID de fila en edición (null si no hay edición)
52
+ * @returns true si el header debe ser visible
53
+ */
54
+ isColumnVisibleInHeader(column, data, creating, editing) {
55
+ // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar
56
+ if (column.visible === false) {
57
+ return false;
58
+ }
59
+ // Si estamos en modo creación, mostrar columnas con hideIfAllValuesAreNull
60
+ if (creating && column.hideIfAllValuesAreNull === true) {
61
+ return true;
62
+ }
63
+ // Si hay una fila en modo edición y la columna tiene hideIfAllValuesAreNull
64
+ if (editing !== null && column.hideIfAllValuesAreNull === true) {
65
+ // Buscar la fila que se está editando
66
+ const editingRow = data.find(row => row['id'] === editing);
67
+ if (editingRow) {
68
+ // Solo mostrar el header si la fila en edición tiene valor en esta columna
69
+ const cellValue = this.getCellValue(editingRow, column.accessor);
70
+ return !this.dataConverter.isValueEmpty(cellValue);
71
+ }
72
+ // Si no se encuentra la fila, no mostrar el header
73
+ return false;
74
+ }
75
+ // Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
76
+ if (column.hideIfAllValuesAreNull === true && this.areAllColumnValuesEmpty(column, data)) {
77
+ return false;
78
+ }
79
+ // Por defecto: mostrar
80
+ return true;
81
+ }
82
+ /**
83
+ * Determina si una columna debe ser visible en una fila específica considerando el modo edición
84
+ * @param column - La definición de la columna
85
+ * @param row - La fila actual
86
+ * @param data - Datos actuales de la tabla
87
+ * @param editing - ID de fila en edición (null si no hay edición)
88
+ * @returns true si la columna debe ser visible para esta fila
89
+ */
90
+ isColumnVisibleForRow(column, row, data, editing) {
91
+ // PRIORIDAD 1: Si visible está explícitamente establecido en false, SIEMPRE ocultar
92
+ if (column.visible === false) {
93
+ return false;
94
+ }
95
+ // PRIORIDAD 2: Si esta fila específica está en modo edición y la columna tiene hideIfAllValuesAreNull
96
+ const isEditingThisRow = editing === row['id'];
97
+ if (isEditingThisRow && column.hideIfAllValuesAreNull === true) {
98
+ // En edición, solo mostrar si esta fila específica tiene valor en esta columna
99
+ const cellValue = this.getCellValue(row, column.accessor);
100
+ return !this.dataConverter.isValueEmpty(cellValue);
101
+ }
102
+ // PRIORIDAD 3: Si hideIfAllValuesAreNull es true y todos los valores están vacíos, ocultar
103
+ if (column.hideIfAllValuesAreNull === true && this.areAllColumnValuesEmpty(column, data)) {
104
+ return false;
105
+ }
106
+ // Por defecto: mostrar
107
+ return true;
108
+ }
109
+ /**
110
+ * Verifica si todos los valores de una columna están vacíos/nulos
111
+ * @param column - La definición de la columna
112
+ * @param data - Datos actuales de la tabla
113
+ * @returns true si todos los valores están vacíos
114
+ */
115
+ areAllColumnValuesEmpty(column, data) {
116
+ if (data.length === 0) {
117
+ return true; // Si no hay datos, consideramos la columna como vacía
118
+ }
119
+ return data.every(row => {
120
+ const value = this.getCellValue(row, column.accessor);
121
+ return this.dataConverter.isValueEmpty(value);
122
+ });
123
+ }
124
+ /**
125
+ * Obtiene el valor de una celda usando el accessor, soportando notación de punto para propiedades anidadas
126
+ * @param row - La fila de datos
127
+ * @param accessor - El accessor de la columna
128
+ * @returns El valor de la celda
129
+ */
130
+ getCellValue(row, accessor) {
131
+ if (accessor.includes('.')) {
132
+ return this.getNestedValue(row, accessor);
133
+ }
134
+ return row[accessor];
135
+ }
136
+ /**
137
+ * Obtiene el valor de un objeto usando notación de punto (ej: 'task.name')
138
+ * @param obj - Objeto del cual obtener el valor
139
+ * @param accessor - Ruta de acceso usando notación de punto
140
+ * @returns El valor anidado o undefined
141
+ */
142
+ getNestedValue(obj, accessor) {
143
+ return accessor.split('.').reduce((current, key) => {
144
+ return current && typeof current === 'object' ? current[key] : undefined;
145
+ }, obj);
146
+ }
147
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableColumnVisibilityService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
148
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableColumnVisibilityService, providedIn: 'root' });
149
+ }
150
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableColumnVisibilityService, decorators: [{
151
+ type: Injectable,
152
+ args: [{
153
+ providedIn: 'root'
154
+ }]
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":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAEnD,OAAO,EAAE,yBAAyB,EAAE,MAAM,gCAAgC,CAAC;;AAE3E;;;;;;;GAOG;AAIH,MAAM,OAAO,4BAA4B;IAEpB,aAAa,GAAG,MAAM,CAAC,yBAAyB,CAAC,CAAC;IAEnE;;;;;;;OAOG;IACH,iBAAiB,CACb,OAAyB,EACzB,IAAS,EACT,QAAiB,EACjB,OAAkB;QAElB,MAAM,cAAc,GAAG,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE,CACxC,IAAI,CAAC,uBAAuB,CAAC,GAAG,EAAE,IAAI,EAAE,QAAQ,EAAE,OAAO,CAAC,CAC7D,CAAC;QACF,OAAO,cAAc,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IACnD,CAAC;IAED;;;;;;OAMG;IACH,eAAe,CACX,MAAsB,EACtB,IAAS,EACT,mBAAmB,GAAG,KAAK;QAE3B,oFAAoF;QACpF,IAAI,MAAM,CAAC,OAAO,KAAK,KAAK,EAAE,CAAC;YAC3B,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,2FAA2F;QAC3F,uFAAuF;QACvF,IAAI,MAAM,CAAC,sBAAsB,KAAK,IAAI,IAAI,CAAC,mBAAmB,IAAI,IAAI,CAAC,uBAAuB,CAAC,MAAM,EAAE,IAAI,CAAC,EAAE,CAAC;YAC/G,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,uBAAuB;QACvB,OAAO,IAAI,CAAC;IAChB,CAAC;IAED;;;;;;;OAOG;IACH,uBAAuB,CACnB,MAAsB,EACtB,IAAS,EACT,QAAiB,EACjB,OAAkB;QAElB,oFAAoF;QACpF,IAAI,MAAM,CAAC,OAAO,KAAK,KAAK,EAAE,CAAC;YAC3B,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,2EAA2E;QAC3E,IAAI,QAAQ,IAAI,MAAM,CAAC,sBAAsB,KAAK,IAAI,EAAE,CAAC;YACrD,OAAO,IAAI,CAAC;QAChB,CAAC;QAED,4EAA4E;QAC5E,IAAI,OAAO,KAAK,IAAI,IAAI,MAAM,CAAC,sBAAsB,KAAK,IAAI,EAAE,CAAC;YAC7D,sCAAsC;YACtC,MAAM,UAAU,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,OAAO,CAAC,CAAC;YAE3D,IAAI,UAAU,EAAE,CAAC;gBACb,2EAA2E;gBAC3E,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,UAAU,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;gBACjE,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;YACvD,CAAC;YAED,mDAAmD;YACnD,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,8EAA8E;QAC9E,IAAI,MAAM,CAAC,sBAAsB,KAAK,IAAI,IAAI,IAAI,CAAC,uBAAuB,CAAC,MAAM,EAAE,IAAI,CAAC,EAAE,CAAC;YACvF,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,uBAAuB;QACvB,OAAO,IAAI,CAAC;IAChB,CAAC;IAED;;;;;;;OAOG;IACH,qBAAqB,CACjB,MAAsB,EACtB,GAAM,EACN,IAAS,EACT,OAAkB;QAElB,oFAAoF;QACpF,IAAI,MAAM,CAAC,OAAO,KAAK,KAAK,EAAE,CAAC;YAC3B,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,sGAAsG;QACtG,MAAM,gBAAgB,GAAG,OAAO,KAAK,GAAG,CAAC,IAAI,CAAC,CAAC;QAC/C,IAAI,gBAAgB,IAAI,MAAM,CAAC,sBAAsB,KAAK,IAAI,EAAE,CAAC;YAC7D,+EAA+E;YAC/E,MAAM,SAAS,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;YAC1D,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;QACvD,CAAC;QAED,2FAA2F;QAC3F,IAAI,MAAM,CAAC,sBAAsB,KAAK,IAAI,IAAI,IAAI,CAAC,uBAAuB,CAAC,MAAM,EAAE,IAAI,CAAC,EAAE,CAAC;YACvF,OAAO,KAAK,CAAC;QACjB,CAAC;QAED,uBAAuB;QACvB,OAAO,IAAI,CAAC;IAChB,CAAC;IAED;;;;;OAKG;IACH,uBAAuB,CACnB,MAAsB,EACtB,IAAS;QAET,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACpB,OAAO,IAAI,CAAC,CAAC,sDAAsD;QACvE,CAAC;QAED,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE;YACpB,MAAM,KAAK,GAAG,IAAI,CAAC,YAAY,CAAC,GAAG,EAAE,MAAM,CAAC,QAAQ,CAAC,CAAC;YACtD,OAAO,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAClD,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;;;;OAKG;IACK,YAAY,CAAoC,GAAM,EAAE,QAAgB;QAC5E,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;IACK,cAAc,CAAC,GAAQ,EAAE,QAAgB;QAC7C,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;wGAlLQ,4BAA4B;4GAA5B,4BAA4B,cAFzB,MAAM;;4FAET,4BAA4B;kBAHxC,UAAU;mBAAC;oBACR,UAAU,EAAE,MAAM;iBACrB","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"]}
@@ -0,0 +1,165 @@
1
+ import { Injectable, signal, inject } from '@angular/core';
2
+ import { TableColumnVisibilityService } from './table-column-visibility.service';
3
+ import { TableDataConverterService } from './table-data-converter.service';
4
+ import * as i0 from "@angular/core";
5
+ /**
6
+ * Servicio para gestionar el estado CRUD (Crear, Leer, Actualizar, Eliminar) en tablas C80
7
+ *
8
+ * Maneja:
9
+ * - Estado de creación y edición
10
+ * - Gestión de datos temporales (newRow, editRow)
11
+ * - Aplicación de valores dinámicos
12
+ * - Validación y conversión de datos antes de emisión de eventos
13
+ */
14
+ export class TableCrudStateService {
15
+ visibilityService = inject(TableColumnVisibilityService);
16
+ dataConverter = inject(TableDataConverterService);
17
+ /**
18
+ * Inicializa el estado CRUD para una nueva tabla
19
+ * @returns Objeto con signals y métodos para gestión CRUD
20
+ */
21
+ createCrudState() {
22
+ const creating = signal(false);
23
+ const newRow = signal(null);
24
+ const editing = signal(null);
25
+ const editRow = signal(null);
26
+ const startCreate = (columns, data) => {
27
+ 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);
39
+ };
40
+ const cancelCreate = () => {
41
+ creating.set(false);
42
+ newRow.set(null);
43
+ };
44
+ const updateNewRow = (key, value) => {
45
+ const current = newRow();
46
+ if (!current)
47
+ return;
48
+ newRow.set({ ...current, [key]: value });
49
+ };
50
+ const saveCreate = (columns, data, createEmitter) => {
51
+ const row = newRow();
52
+ if (!row)
53
+ 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
+ }, {});
64
+ createEmitter.emit({
65
+ row: converted,
66
+ done: (success) => {
67
+ if (success) {
68
+ cancelCreate();
69
+ }
70
+ },
71
+ });
72
+ };
73
+ const startEdit = (row, columns, data) => {
74
+ 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);
83
+ };
84
+ const cancelEdit = () => {
85
+ editing.set(null);
86
+ editRow.set(null);
87
+ };
88
+ const updateEditRow = (key, value) => {
89
+ const current = editRow();
90
+ if (!current)
91
+ return;
92
+ editRow.set({ ...current, [key]: value });
93
+ };
94
+ const saveEdit = (rowId, columns, data, updateEmitter) => {
95
+ const currentEditRow = editRow();
96
+ if (!currentEditRow)
97
+ 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
+ }, {});
106
+ updateEmitter.emit({
107
+ id: rowId,
108
+ changes: converted,
109
+ done: (success) => {
110
+ if (success) {
111
+ cancelEdit();
112
+ }
113
+ },
114
+ });
115
+ };
116
+ const applyInputValues = (partialValues) => {
117
+ // Solo aplicar valores si estamos en modo creación o edición
118
+ if (!creating() && editing() === null) {
119
+ return;
120
+ }
121
+ // Aplicar valores en modo creación
122
+ if (creating()) {
123
+ const currentRow = newRow();
124
+ if (currentRow) {
125
+ const updatedRow = { ...currentRow, ...partialValues };
126
+ newRow.set(updatedRow);
127
+ }
128
+ }
129
+ // Aplicar valores en modo edición
130
+ if (editing() !== null) {
131
+ const currentEditRow = editRow();
132
+ if (currentEditRow) {
133
+ const updatedEditRow = { ...currentEditRow, ...partialValues };
134
+ editRow.set(updatedEditRow);
135
+ }
136
+ }
137
+ };
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
+ }
156
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableCrudStateService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
157
+ static ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableCrudStateService, providedIn: 'root' });
158
+ }
159
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: TableCrudStateService, decorators: [{
160
+ type: Injectable,
161
+ args: [{
162
+ providedIn: 'root'
163
+ }]
164
+ }] });
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"]}