@litigiovirtual/ius-design-components 1.0.73 → 1.0.75

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.
@@ -112,4 +112,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
112
112
  }], buttonClicked: [{
113
113
  type: Output
114
114
  }] } });
115
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button-dynamic.component.js","sourceRoot":"","sources":["../../../../../projects/ius-design-components/src/lib/button-dynamic/button-dynamic.component.ts","../../../../../projects/ius-design-components/src/lib/button-dynamic/button-dynamic.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, EventEmitter, Input, OnChanges, OnDestroy, Output, SimpleChanges } from '@angular/core';\r\nimport { IconMdComponent } from '../icon-md/icon-md.component';\r\nimport { LoadingCircleComponent } from '../loading-circle/loading-circle.component';\r\n\r\ntype ButtonResult = 'success' | 'error' | undefined;\r\ntype ButtonState = 'default' | 'loading' | 'success' | 'error' | 'disabled';\r\n\r\n@Component({\r\n  selector: 'ius-button-dynamic',\r\n  standalone: true,\r\n  imports: [CommonModule, LoadingCircleComponent, IconMdComponent],\r\n  templateUrl: './button-dynamic.component.html',\r\n  styleUrls: ['./button-dynamic.component.scss'],\r\n})\r\nexport class ButtonDynamicComponent implements OnChanges, OnDestroy {\r\n\r\n  @Input() labelDefault = 'Label';\r\n  @Input() labelSuccess?: string;\r\n  @Input() labelError?: string;\r\n\r\n  @Input() disabled = false;\r\n  @Input() loading = false;\r\n  @Input() result: ButtonResult;\r\n\r\n  @Input() autoReset = true;\r\n  @Input() autoResetDelay = 1500;\r\n\r\n  @Output() buttonClicked = new EventEmitter<void>();\r\n\r\n  state: ButtonState = 'default';\r\n  private resetTimer: any;\r\n\r\n  ngOnInit() {\r\n    if (!this.labelSuccess) {\r\n      this.labelSuccess = this.labelDefault;\r\n    }\r\n    if (!this.labelError) {\r\n      this.labelError = this.labelDefault;\r\n    }\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    // 1) disabled manda todo a 'disabled'\r\n    if (this.disabled) {\r\n      this.setState('disabled');\r\n      return;\r\n    }\r\n\r\n    // 2) loading gobierna mientras es true\r\n    if (changes['loading']) {\r\n      if (this.loading) {\r\n        this.clearTimer();\r\n        this.setState('loading');\r\n        return;\r\n      } else {\r\n        // loading → false: mirar result\r\n        this.applyResultState();\r\n        return;\r\n      }\r\n    }\r\n\r\n    // 3) Si cambió 'result' y no estamos cargando, aplicar\r\n    if (changes['result'] && !this.loading) {\r\n      this.applyResultState();\r\n    }\r\n\r\n    // 4) fallback\r\n    if (!this.loading && !this.result) {\r\n      this.setState('default');\r\n    }\r\n  }\r\n\r\n  applyResultState(): void {\r\n    if (this.result === 'success') {\r\n      this.setState('success');\r\n      this.armAutoReset();\r\n    } else if (this.result === 'error') {\r\n      this.setState('error');\r\n      this.armAutoReset();\r\n    } else {\r\n      this.setState('default');\r\n    }\r\n  }\r\n\r\n  armAutoReset(): void {\r\n    this.clearTimer();\r\n    if (!this.autoReset) return;\r\n    this.resetTimer = setTimeout(() => {\r\n      this.setState('default');\r\n    }, this.autoResetDelay);\r\n  }\r\n\r\n  clearTimer(): void {\r\n    if (this.resetTimer) {\r\n      clearTimeout(this.resetTimer);\r\n      this.resetTimer = null;\r\n    }\r\n  }\r\n\r\n  setState(s: ButtonState) {\r\n    this.state = s;\r\n  }\r\n\r\n  onClick() {\r\n    if (this.state === 'loading' || this.state === 'disabled') return;\r\n    this.buttonClicked.emit();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.clearTimer();\r\n  }\r\n}\r\n","<button class=\"ius-btn\" [attr.data-state]=\"state\" [disabled]=\"state === 'disabled' || state === 'loading'\"\r\n  (click)=\"onClick()\" aria-live=\"polite\">\r\n  @switch (state) {\r\n  @case ('loading') {\r\n  <ius-loading-circle></ius-loading-circle>\r\n  }\r\n  @case ('success') {\r\n  <ius-icon-md iconName=\"icon-check-circle\"></ius-icon-md>\r\n  <span>{{ labelSuccess }}</span>\r\n  }\r\n  @case ('error') {\r\n  <ius-icon-md iconName=\"icon-error\"></ius-icon-md>\r\n  <span>{{ labelError }}</span>\r\n  }\r\n  @default {\r\n  <span>{{ labelDefault }}</span>\r\n  }\r\n  }\r\n</button>"]}
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"button-dynamic.component.js","sourceRoot":"","sources":["../../../../../projects/ius-design-components/src/lib/button-dynamic/button-dynamic.component.ts","../../../../../projects/ius-design-components/src/lib/button-dynamic/button-dynamic.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport { Component, EventEmitter, Input, OnChanges, OnDestroy, Output, SimpleChanges } from '@angular/core';\r\nimport { IconMdComponent } from '../icon-md/icon-md.component';\r\nimport { LoadingCircleComponent } from '../loading-circle/loading-circle.component';\r\n\r\nexport type ButtonResult = 'success' | 'error' | undefined;\r\ntype ButtonState = 'default' | 'loading' | 'success' | 'error' | 'disabled';\r\n\r\n@Component({\r\n  selector: 'ius-button-dynamic',\r\n  standalone: true,\r\n  imports: [CommonModule, LoadingCircleComponent, IconMdComponent],\r\n  templateUrl: './button-dynamic.component.html',\r\n  styleUrls: ['./button-dynamic.component.scss'],\r\n})\r\nexport class ButtonDynamicComponent implements OnChanges, OnDestroy {\r\n\r\n  @Input() labelDefault = 'Label';\r\n  @Input() labelSuccess?: string;\r\n  @Input() labelError?: string;\r\n\r\n  @Input() disabled = false;\r\n  @Input() loading = false;\r\n  @Input() result: ButtonResult;\r\n\r\n  @Input() autoReset = true;\r\n  @Input() autoResetDelay = 1500;\r\n\r\n  @Output() buttonClicked = new EventEmitter<void>();\r\n\r\n  state: ButtonState = 'default';\r\n  private resetTimer: any;\r\n\r\n  ngOnInit() {\r\n    if (!this.labelSuccess) {\r\n      this.labelSuccess = this.labelDefault;\r\n    }\r\n    if (!this.labelError) {\r\n      this.labelError = this.labelDefault;\r\n    }\r\n  }\r\n\r\n  ngOnChanges(changes: SimpleChanges): void {\r\n    // 1) disabled manda todo a 'disabled'\r\n    if (this.disabled) {\r\n      this.setState('disabled');\r\n      return;\r\n    }\r\n\r\n    // 2) loading gobierna mientras es true\r\n    if (changes['loading']) {\r\n      if (this.loading) {\r\n        this.clearTimer();\r\n        this.setState('loading');\r\n        return;\r\n      } else {\r\n        // loading → false: mirar result\r\n        this.applyResultState();\r\n        return;\r\n      }\r\n    }\r\n\r\n    // 3) Si cambió 'result' y no estamos cargando, aplicar\r\n    if (changes['result'] && !this.loading) {\r\n      this.applyResultState();\r\n    }\r\n\r\n    // 4) fallback\r\n    if (!this.loading && !this.result) {\r\n      this.setState('default');\r\n    }\r\n  }\r\n\r\n  applyResultState(): void {\r\n    if (this.result === 'success') {\r\n      this.setState('success');\r\n      this.armAutoReset();\r\n    } else if (this.result === 'error') {\r\n      this.setState('error');\r\n      this.armAutoReset();\r\n    } else {\r\n      this.setState('default');\r\n    }\r\n  }\r\n\r\n  armAutoReset(): void {\r\n    this.clearTimer();\r\n    if (!this.autoReset) return;\r\n    this.resetTimer = setTimeout(() => {\r\n      this.setState('default');\r\n    }, this.autoResetDelay);\r\n  }\r\n\r\n  clearTimer(): void {\r\n    if (this.resetTimer) {\r\n      clearTimeout(this.resetTimer);\r\n      this.resetTimer = null;\r\n    }\r\n  }\r\n\r\n  setState(s: ButtonState) {\r\n    this.state = s;\r\n  }\r\n\r\n  onClick() {\r\n    if (this.state === 'loading' || this.state === 'disabled') return;\r\n    this.buttonClicked.emit();\r\n  }\r\n\r\n  ngOnDestroy(): void {\r\n    this.clearTimer();\r\n  }\r\n}\r\n","<button class=\"ius-btn\" [attr.data-state]=\"state\" [disabled]=\"state === 'disabled' || state === 'loading'\"\r\n  (click)=\"onClick()\" aria-live=\"polite\">\r\n  @switch (state) {\r\n  @case ('loading') {\r\n  <ius-loading-circle></ius-loading-circle>\r\n  }\r\n  @case ('success') {\r\n  <ius-icon-md iconName=\"icon-check-circle\"></ius-icon-md>\r\n  <span>{{ labelSuccess }}</span>\r\n  }\r\n  @case ('error') {\r\n  <ius-icon-md iconName=\"icon-error\"></ius-icon-md>\r\n  <span>{{ labelError }}</span>\r\n  }\r\n  @default {\r\n  <span>{{ labelDefault }}</span>\r\n  }\r\n  }\r\n</button>"]}
@@ -0,0 +1,234 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, Input, Output, EventEmitter, computed, signal, ViewChild } from '@angular/core';
3
+ import { PopoverDirective } from '../directives/popover.directive';
4
+ import * as i0 from "@angular/core";
5
+ export class IusChartsBarGroupedComponent {
6
+ constructor(zone) {
7
+ this.zone = zone;
8
+ this.data = [];
9
+ this.legendPosition = 'bottom';
10
+ // Márgenes (left = ancho panel eje Y)
11
+ this.margin = { top: 24, right: 16, bottom: 34, left: 40 };
12
+ // Layout X fijo
13
+ this.barWidth = 11;
14
+ this.barGap = 2;
15
+ this.groupGap = 40;
16
+ this.outerGap = 18;
17
+ // Estilo barra
18
+ this.barRadius = 5;
19
+ this.ticks = 5;
20
+ // Fallbacks si no hay tamaño CSS externo
21
+ this.width = 640;
22
+ this.size = 320;
23
+ this.maxLabelChars = 10;
24
+ this.barClick = new EventEmitter();
25
+ // Estado
26
+ this._bars = signal([]);
27
+ this._groups = signal([]);
28
+ this._series = signal([]);
29
+ this._colors = signal({});
30
+ this._yMax = signal(0);
31
+ this._paneH = signal(0); // alto real del panel superior
32
+ this._plotW = signal(0); // ancho visible del plot
33
+ this._plotWidth = signal(0); // ancho virtual del SVG derecho
34
+ this._groupCenters = signal([]);
35
+ this._barStripes = signal([]);
36
+ this.bars = computed(() => this._bars());
37
+ this.groups = computed(() => this._groups());
38
+ this.series = computed(() => this._series());
39
+ this.colors = computed(() => this._colors());
40
+ this.yScaleMax = computed(() => this._yMax());
41
+ this.chartHeight = computed(() => this._paneH() || this.size); // ← altura para ambos SVG
42
+ this.plotWidth = computed(() => this._plotWidth());
43
+ this.groupCenters = computed(() => this._groupCenters());
44
+ this.barStripes = computed(() => this._barStripes());
45
+ this.innerHeight = computed(() => Math.max(this.chartHeight() - this.margin.top - this.margin.bottom, 0));
46
+ // viewBox por SVG
47
+ this.axisViewBox = computed(() => `0 0 ${this.margin.left} ${this.chartHeight()}`);
48
+ this.plotViewBox = computed(() => `0 0 ${this.plotWidth()} ${this.chartHeight()}`);
49
+ this.Math = Math;
50
+ }
51
+ ngAfterViewInit() {
52
+ // Medición inicial SIN esperar al primer tick del RO
53
+ this.measureNow();
54
+ // Observa alto (panes) y ancho visible (plotCtn)
55
+ this.ro = new ResizeObserver(() => {
56
+ this.zone.run(() => {
57
+ const changed = this.measureNow();
58
+ if (changed)
59
+ this.recalc();
60
+ });
61
+ });
62
+ this.ro.observe(this.panesRef.nativeElement);
63
+ this.ro.observe(this.plotCtn.nativeElement);
64
+ // Primer cálculo
65
+ this.recalc();
66
+ }
67
+ ngOnDestroy() { this.ro?.disconnect(); }
68
+ ngOnChanges() { this.recalc(); }
69
+ // Devuelve true si cambió algo.
70
+ measureNow() {
71
+ const paneH = Math.floor(this.panesRef?.nativeElement?.clientHeight || 0);
72
+ const plotW = Math.floor(this.plotCtn?.nativeElement?.clientWidth || 0);
73
+ let changed = false;
74
+ if (paneH && paneH !== this._paneH()) {
75
+ this._paneH.set(paneH);
76
+ changed = true;
77
+ }
78
+ if (plotW && plotW !== this._plotW()) {
79
+ this._plotW.set(plotW);
80
+ changed = true;
81
+ }
82
+ // Si no hay valores aún (por render temprano), usa fallbacks para no quedarte en 0
83
+ if (!this._paneH()) {
84
+ this._paneH.set(this.size);
85
+ changed = true;
86
+ }
87
+ if (!this._plotW()) {
88
+ this._plotW.set(this.width - this.margin.left);
89
+ changed = true;
90
+ }
91
+ return changed;
92
+ }
93
+ recalc() {
94
+ const cleaned = (this.data ?? []).map(g => ({
95
+ label: g.label,
96
+ items: (g.items ?? []).filter(i => Number(i?.value) >= 0)
97
+ })).filter(x => x.items.length > 0);
98
+ const groups = cleaned.map(g => g.label);
99
+ // series + orden
100
+ const seriesSet = new Set();
101
+ for (const g of cleaned)
102
+ for (const it of g.items)
103
+ seriesSet.add(it.label);
104
+ let series = Array.from(seriesSet);
105
+ if (this.seriesOrder?.length) {
106
+ const order = new Map(this.seriesOrder.map((k, idx) => [k, idx]));
107
+ series.sort((a, b) => (order.get(a) ?? 1e9) - (order.get(b) ?? 1e9));
108
+ }
109
+ // colores por serie
110
+ const palette = [
111
+ '#2167FF', // Info-500
112
+ '#0A2893', // Info-800
113
+ '#40CAE9', // Primary-400
114
+ '#5892FF', // Info-400
115
+ '#08A6DB', // Primary-500
116
+ '#C4DFFF', // Info-200
117
+ ];
118
+ const colorMap = {};
119
+ series.forEach((s, i) => colorMap[s] = palette[i % palette.length]);
120
+ // normalizar
121
+ const normalized = cleaned.map(g => {
122
+ const map = new Map(g.items.map(i => [i.label, i]));
123
+ const items = series.map(s => {
124
+ const i = map.get(s);
125
+ return { label: s, value: i?.value ?? 0, color: i?.color ?? colorMap[s] };
126
+ });
127
+ return { label: g.label, items };
128
+ });
129
+ // Y máx
130
+ const maxValue = Math.max(0, ...normalized.flatMap(g => g.items.map(i => i.value)));
131
+ const yMax = this.yMax && this.yMax > 0 ? this.yMax : this.niceMax(maxValue);
132
+ this._yMax.set(yMax);
133
+ // dimensiones visibles del panel derecho
134
+ const visibleW = this._plotW() || (this.width - this.margin.left);
135
+ // ===== Layout X con outerGap =====
136
+ const G = normalized.length;
137
+ const S = series.length;
138
+ const groupInnerWidth = S * this.barWidth + Math.max(0, S - 1) * this.barGap;
139
+ const innerNaturalWidth = Math.max(0, (this.outerGap * 2) + G * groupInnerWidth + Math.max(0, G - 1) * this.groupGap);
140
+ const innerVisibleWidth = Math.max(visibleW - this.margin.right, 0);
141
+ const innerVirtualWidth = Math.max(innerVisibleWidth, innerNaturalWidth);
142
+ const plotWidth = innerVirtualWidth + this.margin.right; // solo margen derecho en el SVG derecho
143
+ this._plotWidth.set(plotWidth);
144
+ // rects + franjas
145
+ const ih = this.innerHeight();
146
+ const bars = [];
147
+ const centers = [];
148
+ const stripes = [];
149
+ let cursor = this.outerGap;
150
+ normalized.forEach(g => {
151
+ const startX = cursor;
152
+ const center = startX + groupInnerWidth / 2;
153
+ centers.push(center);
154
+ g.items.forEach((it, si) => {
155
+ const x = startX + si * (this.barWidth + this.barGap);
156
+ // franja por barra
157
+ stripes.push({ x, width: this.barWidth });
158
+ const h = it.value <= 0 ? 0 : (it.value / yMax) * ih;
159
+ const y = this.margin.top + ih - h;
160
+ bars.push({
161
+ groupLabel: g.label,
162
+ seriesLabel: it.label,
163
+ value: it.value,
164
+ x, y,
165
+ width: this.barWidth,
166
+ height: h,
167
+ color: it.color ?? colorMap[it.label]
168
+ });
169
+ });
170
+ cursor += groupInnerWidth + this.groupGap;
171
+ });
172
+ this._bars.set(bars);
173
+ this._groups.set(groups);
174
+ this._series.set(series);
175
+ this._colors.set(colorMap);
176
+ this._groupCenters.set(centers);
177
+ this._barStripes.set(stripes);
178
+ }
179
+ niceMax(maxVal) {
180
+ if (maxVal <= 0)
181
+ return 1;
182
+ const exp = Math.floor(Math.log10(maxVal));
183
+ const base = Math.pow(10, exp);
184
+ const n = Math.ceil(maxVal / base);
185
+ const nice = n <= 2 ? 2 : n <= 5 ? 5 : 10;
186
+ return nice * base;
187
+ }
188
+ getY(value) {
189
+ const ih = this.innerHeight();
190
+ const max = this.yScaleMax();
191
+ return this.margin.top + ih - (value / max) * ih;
192
+ }
193
+ get yTicksArr() {
194
+ const max = this.yScaleMax();
195
+ const n = Math.max(2, this.ticks | 0);
196
+ const step = max / n;
197
+ const arr = [];
198
+ for (let i = 0; i <= n; i++)
199
+ arr.push(+((i * step)).toFixed(2));
200
+ return arr;
201
+ }
202
+ shortLabel(lbl) {
203
+ return lbl.length > this.maxLabelChars ? (lbl.slice(0, this.maxLabelChars - 1) + '…') : lbl;
204
+ }
205
+ formatTooltip(b) {
206
+ return `${b.groupLabel} — ${b.seriesLabel}: ${b.value}`;
207
+ }
208
+ onBarClick(b) {
209
+ this.barClick.emit({ group: b.groupLabel, series: b.seriesLabel, value: b.value });
210
+ }
211
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: IusChartsBarGroupedComponent, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component }); }
212
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: IusChartsBarGroupedComponent, isStandalone: true, selector: "ius-charts-bar-grouped", inputs: { data: "data", legendPosition: "legendPosition", seriesOrder: "seriesOrder", maxLabelChars: "maxLabelChars" }, outputs: { barClick: "barClick" }, viewQueries: [{ propertyName: "panesRef", first: true, predicate: ["panes"], descendants: true, static: true }, { propertyName: "plotCtn", first: true, predicate: ["plotCtn"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<div class=\"chart-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n <div class=\"panes\" #panes>\r\n <div class=\"axis-ctn\">\r\n <svg class=\"chart-axis\"\r\n [attr.viewBox]=\"axisViewBox()\"\r\n preserveAspectRatio=\"xMinYMin meet\"\r\n [style.width.px]=\"margin.left\"\r\n [style.height.px]=\"chartHeight()\">\r\n @for (t of yTicksArr; track t) {\r\n <text class=\"y-tick body-sm\"\r\n [attr.x]=\"margin.left - 6\"\r\n [attr.y]=\"getY(t)\"\r\n text-anchor=\"end\"\r\n dominant-baseline=\"middle\">\r\n {{ t }}\r\n </text>\r\n }\r\n </svg>\r\n </div>\r\n\r\n <div class=\"plot-ctn scrollable-small\" #plotCtn>\r\n <svg class=\"chart\"\r\n [attr.viewBox]=\"plotViewBox()\"\r\n preserveAspectRatio=\"xMinYMin meet\"\r\n [style.width.px]=\"plotWidth()\"\r\n [style.height.px]=\"chartHeight()\">\r\n\r\n @for (s of barStripes(); track s.x) {\r\n <rect class=\"bar-stripe\"\r\n [attr.x]=\"s.x\"\r\n [attr.y]=\"margin.top\"\r\n [attr.width]=\"s.width\"\r\n [attr.height]=\"innerHeight()\"></rect>\r\n }\r\n\r\n @for (t of yTicksArr; track t) {\r\n <line class=\"grid-line\"\r\n [attr.x1]=\"0\"\r\n [attr.x2]=\"plotWidth() - margin.right\"\r\n [attr.y1]=\"getY(t)\" [attr.y2]=\"getY(t)\"/>\r\n }\r\n\r\n @for (g of groups(); track g; let i = $index) {\r\n <text class=\"x-label body-sm\"\r\n [attr.x]=\"groupCenters()[i]\"\r\n [attr.y]=\"chartHeight() - margin.bottom + 22\"\r\n text-anchor=\"middle\">\r\n {{ shortLabel(g) }} <title>{{ g }}</title>\r\n </text>\r\n }\r\n\r\n @for (b of bars(); track b.groupLabel + '_' + b.seriesLabel) {\r\n <rect class=\"bar\"\r\n [attr.x]=\"b.x\"\r\n [attr.y]=\"b.y\"\r\n [attr.width]=\"b.width\"\r\n [attr.height]=\"Math.max(b.height, 0)\"\r\n [attr.fill]=\"b.color\"\r\n [attr.rx]=\"barRadius\"\r\n [attr.ry]=\"barRadius\"\r\n [iusPopoverContent]=\"formatTooltip(b)\"\r\n (click)=\"onBarClick(b)\"/>\r\n }\r\n </svg>\r\n </div>\r\n </div>\r\n\r\n <!-- Leyenda -->\r\n <div class=\"legend\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n @for (s of series(); track s) {\r\n <div class=\"legend-item\">\r\n <span class=\"dot\" [style.background]=\"colors()[s]\"></span>\r\n <span class=\"label caption-sm\">{{ s }}</span>\r\n </div>\r\n }\r\n </div>\r\n</div>\r\n", styles: [".h1{font-family:Roboto,sans-serif;font-size:2.375rem;font-weight:500;line-height:46px}.h2{font-family:Roboto,sans-serif;font-size:1.875rem;font-weight:700;line-height:38px}.h3{font-family:Roboto,sans-serif;font-size:1.5rem;font-weight:500;line-height:32px}.h4{font-family:Roboto,sans-serif;font-size:1.25rem;font-weight:700;line-height:26px}.h5{font-family:Roboto,sans-serif;font-size:1.125rem;font-weight:500;line-height:24px;letter-spacing:.18px}.label-large{font-family:Roboto,sans-serif;font-size:1rem;font-weight:500;line-height:22px}.body-large{font-family:Rubik,sans-serif;font-size:1rem;font-weight:400;line-height:22px}.label-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:500;line-height:20px;letter-spacing:.28px}.body-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:400;line-height:20px;font-style:italic;letter-spacing:.28px}.body-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;letter-spacing:.28px}.body-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;font-style:italic;letter-spacing:.28px}.caption-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;letter-spacing:.28px}.caption-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;font-style:italic;letter-spacing:.28px}:host{display:block;max-width:100%;min-height:0}.chart-wrapper{display:grid;grid-template-columns:1fr auto;column-gap:16px;height:100%;min-height:0;align-items:start}.chart-wrapper.bottom{grid-template-columns:1fr;grid-template-rows:1fr auto;row-gap:12px}.panes{display:grid;grid-template-columns:auto 1fr;height:100%;min-height:0}.axis-ctn{height:100%;min-height:0;overflow:hidden}.plot-ctn{width:100%;height:100%;min-height:0;overflow-x:auto;overflow-y:hidden}.chart-wrapper,.panes,.axis-ctn,.plot-ctn{min-height:0}.chart,.chart-axis{display:block}.grid-line{stroke:#1f293729;stroke-width:1;stroke-dasharray:3 3}.y-tick,.x-label{fill:#000000b3}.bar-stripe{fill:#d6dbed66;pointer-events:none}.bar{cursor:pointer;transition:opacity .15s ease,transform .15s ease;pointer-events:all}.bar:hover{opacity:.9}.bar:focus{outline:none;filter:drop-shadow(0 0 2px rgba(0,0,0,.2))}.legend{display:flex;justify-content:center;flex-direction:column;gap:8px;min-width:0}.legend.bottom{flex-direction:row;flex-wrap:wrap}.legend .legend-item{display:flex;align-items:center;gap:8px;min-width:0;white-space:nowrap}.legend .dot{width:8px;height:8px;border-radius:8px;flex:0 0 8px}.legend .label{color:#595959;min-width:0;overflow:hidden;text-overflow:ellipsis}.plot-ctn.scrollable-small::-webkit-scrollbar{-webkit-appearance:none}.plot-ctn.scrollable-small::-webkit-scrollbar:vertical{width:4px;height:32px;padding-top:2px;padding-bottom:2px}.plot-ctn.scrollable-small::-webkit-scrollbar:horizontal{height:5px;padding-top:2px;padding-bottom:2px}.plot-ctn.scrollable-small::-webkit-scrollbar-thumb{background-color:#bfbfbf;border-radius:4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: PopoverDirective, selector: "[iusPopover], [iusPopoverTitle], [iusPopoverContent]", inputs: ["iusPopover", "iusPopoverTitle", "iusPopoverContent", "iusPopoverPosition", "iusPopoverOpenDelay", "iusPopoverCloseDelay"] }] }); }
213
+ }
214
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: IusChartsBarGroupedComponent, decorators: [{
215
+ type: Component,
216
+ args: [{ selector: 'ius-charts-bar-grouped', standalone: true, imports: [CommonModule, PopoverDirective], template: "<div class=\"chart-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n <div class=\"panes\" #panes>\r\n <div class=\"axis-ctn\">\r\n <svg class=\"chart-axis\"\r\n [attr.viewBox]=\"axisViewBox()\"\r\n preserveAspectRatio=\"xMinYMin meet\"\r\n [style.width.px]=\"margin.left\"\r\n [style.height.px]=\"chartHeight()\">\r\n @for (t of yTicksArr; track t) {\r\n <text class=\"y-tick body-sm\"\r\n [attr.x]=\"margin.left - 6\"\r\n [attr.y]=\"getY(t)\"\r\n text-anchor=\"end\"\r\n dominant-baseline=\"middle\">\r\n {{ t }}\r\n </text>\r\n }\r\n </svg>\r\n </div>\r\n\r\n <div class=\"plot-ctn scrollable-small\" #plotCtn>\r\n <svg class=\"chart\"\r\n [attr.viewBox]=\"plotViewBox()\"\r\n preserveAspectRatio=\"xMinYMin meet\"\r\n [style.width.px]=\"plotWidth()\"\r\n [style.height.px]=\"chartHeight()\">\r\n\r\n @for (s of barStripes(); track s.x) {\r\n <rect class=\"bar-stripe\"\r\n [attr.x]=\"s.x\"\r\n [attr.y]=\"margin.top\"\r\n [attr.width]=\"s.width\"\r\n [attr.height]=\"innerHeight()\"></rect>\r\n }\r\n\r\n @for (t of yTicksArr; track t) {\r\n <line class=\"grid-line\"\r\n [attr.x1]=\"0\"\r\n [attr.x2]=\"plotWidth() - margin.right\"\r\n [attr.y1]=\"getY(t)\" [attr.y2]=\"getY(t)\"/>\r\n }\r\n\r\n @for (g of groups(); track g; let i = $index) {\r\n <text class=\"x-label body-sm\"\r\n [attr.x]=\"groupCenters()[i]\"\r\n [attr.y]=\"chartHeight() - margin.bottom + 22\"\r\n text-anchor=\"middle\">\r\n {{ shortLabel(g) }} <title>{{ g }}</title>\r\n </text>\r\n }\r\n\r\n @for (b of bars(); track b.groupLabel + '_' + b.seriesLabel) {\r\n <rect class=\"bar\"\r\n [attr.x]=\"b.x\"\r\n [attr.y]=\"b.y\"\r\n [attr.width]=\"b.width\"\r\n [attr.height]=\"Math.max(b.height, 0)\"\r\n [attr.fill]=\"b.color\"\r\n [attr.rx]=\"barRadius\"\r\n [attr.ry]=\"barRadius\"\r\n [iusPopoverContent]=\"formatTooltip(b)\"\r\n (click)=\"onBarClick(b)\"/>\r\n }\r\n </svg>\r\n </div>\r\n </div>\r\n\r\n <!-- Leyenda -->\r\n <div class=\"legend\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n @for (s of series(); track s) {\r\n <div class=\"legend-item\">\r\n <span class=\"dot\" [style.background]=\"colors()[s]\"></span>\r\n <span class=\"label caption-sm\">{{ s }}</span>\r\n </div>\r\n }\r\n </div>\r\n</div>\r\n", styles: [".h1{font-family:Roboto,sans-serif;font-size:2.375rem;font-weight:500;line-height:46px}.h2{font-family:Roboto,sans-serif;font-size:1.875rem;font-weight:700;line-height:38px}.h3{font-family:Roboto,sans-serif;font-size:1.5rem;font-weight:500;line-height:32px}.h4{font-family:Roboto,sans-serif;font-size:1.25rem;font-weight:700;line-height:26px}.h5{font-family:Roboto,sans-serif;font-size:1.125rem;font-weight:500;line-height:24px;letter-spacing:.18px}.label-large{font-family:Roboto,sans-serif;font-size:1rem;font-weight:500;line-height:22px}.body-large{font-family:Rubik,sans-serif;font-size:1rem;font-weight:400;line-height:22px}.label-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:500;line-height:20px;letter-spacing:.28px}.body-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:400;line-height:20px;font-style:italic;letter-spacing:.28px}.body-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;letter-spacing:.28px}.body-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;font-style:italic;letter-spacing:.28px}.caption-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;letter-spacing:.28px}.caption-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;font-style:italic;letter-spacing:.28px}:host{display:block;max-width:100%;min-height:0}.chart-wrapper{display:grid;grid-template-columns:1fr auto;column-gap:16px;height:100%;min-height:0;align-items:start}.chart-wrapper.bottom{grid-template-columns:1fr;grid-template-rows:1fr auto;row-gap:12px}.panes{display:grid;grid-template-columns:auto 1fr;height:100%;min-height:0}.axis-ctn{height:100%;min-height:0;overflow:hidden}.plot-ctn{width:100%;height:100%;min-height:0;overflow-x:auto;overflow-y:hidden}.chart-wrapper,.panes,.axis-ctn,.plot-ctn{min-height:0}.chart,.chart-axis{display:block}.grid-line{stroke:#1f293729;stroke-width:1;stroke-dasharray:3 3}.y-tick,.x-label{fill:#000000b3}.bar-stripe{fill:#d6dbed66;pointer-events:none}.bar{cursor:pointer;transition:opacity .15s ease,transform .15s ease;pointer-events:all}.bar:hover{opacity:.9}.bar:focus{outline:none;filter:drop-shadow(0 0 2px rgba(0,0,0,.2))}.legend{display:flex;justify-content:center;flex-direction:column;gap:8px;min-width:0}.legend.bottom{flex-direction:row;flex-wrap:wrap}.legend .legend-item{display:flex;align-items:center;gap:8px;min-width:0;white-space:nowrap}.legend .dot{width:8px;height:8px;border-radius:8px;flex:0 0 8px}.legend .label{color:#595959;min-width:0;overflow:hidden;text-overflow:ellipsis}.plot-ctn.scrollable-small::-webkit-scrollbar{-webkit-appearance:none}.plot-ctn.scrollable-small::-webkit-scrollbar:vertical{width:4px;height:32px;padding-top:2px;padding-bottom:2px}.plot-ctn.scrollable-small::-webkit-scrollbar:horizontal{height:5px;padding-top:2px;padding-bottom:2px}.plot-ctn.scrollable-small::-webkit-scrollbar-thumb{background-color:#bfbfbf;border-radius:4px}\n"] }]
217
+ }], ctorParameters: () => [{ type: i0.NgZone }], propDecorators: { data: [{
218
+ type: Input
219
+ }], legendPosition: [{
220
+ type: Input
221
+ }], seriesOrder: [{
222
+ type: Input
223
+ }], maxLabelChars: [{
224
+ type: Input
225
+ }], barClick: [{
226
+ type: Output
227
+ }], panesRef: [{
228
+ type: ViewChild,
229
+ args: ['panes', { static: true }]
230
+ }], plotCtn: [{
231
+ type: ViewChild,
232
+ args: ['plotCtn', { static: true }]
233
+ }] } });
234
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"charts-bar-grouped.component.js","sourceRoot":"","sources":["../../../../../projects/ius-design-components/src/lib/charts-bar-grouped/charts-bar-grouped.component.ts","../../../../../projects/ius-design-components/src/lib/charts-bar-grouped/charts-bar-grouped.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,YAAY,EAC3B,QAAQ,EAAE,MAAM,EAAE,SAAS,EACvC,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,gBAAgB,EAAE,MAAM,iCAAiC,CAAC;;AA2BnE,MAAM,OAAO,4BAA4B;IAoEvC,YAAoB,IAAY;QAAZ,SAAI,GAAJ,IAAI,CAAQ;QAnEvB,SAAI,GAAe,EAAE,CAAC;QACtB,mBAAc,GAAuB,QAAQ,CAAC;QAGvD,sCAAsC;QACtC,WAAM,GAAG,EAAE,GAAG,EAAE,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,IAAI,EAAE,EAAE,EAAE,CAAC;QAEtD,gBAAgB;QAChB,aAAQ,GAAG,EAAE,CAAC;QACd,WAAM,GAAG,CAAC,CAAC;QACX,aAAQ,GAAG,EAAE,CAAC;QACd,aAAQ,GAAG,EAAE,CAAC;QAEd,eAAe;QACf,cAAS,GAAG,CAAC,CAAC;QAId,UAAK,GAAG,CAAC,CAAC;QAEV,yCAAyC;QACzC,UAAK,GAAG,GAAG,CAAC;QACZ,SAAI,GAAG,GAAG,CAAC;QAEF,kBAAa,GAAG,EAAE,CAAC;QAElB,aAAQ,GAAG,IAAI,YAAY,EAAoD,CAAC;QAO1F,SAAS;QACD,UAAK,GAAG,MAAM,CAAY,EAAE,CAAC,CAAC;QAC9B,YAAO,GAAG,MAAM,CAAW,EAAE,CAAC,CAAC;QAC/B,YAAO,GAAG,MAAM,CAAW,EAAE,CAAC,CAAC;QAC/B,YAAO,GAAG,MAAM,CAAiB,EAAE,CAAC,CAAC;QACrC,UAAK,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;QAElB,WAAM,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,CAAK,+BAA+B;QACvD,WAAM,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,CAAK,yBAAyB;QACjD,eAAU,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC,gCAAgC;QACxD,kBAAa,GAAG,MAAM,CAAW,EAAE,CAAC,CAAC;QACrC,gBAAW,GAAG,MAAM,CAAc,EAAE,CAAC,CAAC;QAErC,SAAI,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QACpC,WAAM,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QACxC,WAAM,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QACxC,WAAM,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC;QACxC,cAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,KAAK,EAAE,CAAC,CAAC;QAEzC,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,0BAA0B;QACpF,cAAS,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC;QAC9C,iBAAY,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC,CAAC;QACpD,eAAU,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,CAAC;QAEhD,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE,CACnC,IAAI,CAAC,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC,CAAC,CACvE,CAAC;QAEF,kBAAkB;QACT,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,OAAO,IAAI,CAAC,MAAM,CAAC,IAAI,IAAI,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;QAC9E,gBAAW,GAAG,QAAQ,CAAC,GAAG,EAAE,CAAC,OAAO,IAAI,CAAC,SAAS,EAAE,IAAI,IAAI,CAAC,WAAW,EAAE,EAAE,CAAC,CAAC;QAE9E,SAAI,GAAG,IAAI,CAAC;IAEe,CAAC;IAErC,eAAe;QACb,qDAAqD;QACrD,IAAI,CAAC,UAAU,EAAE,CAAC;QAElB,iDAAiD;QACjD,IAAI,CAAC,EAAE,GAAG,IAAI,cAAc,CAAC,GAAG,EAAE;YAChC,IAAI,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE;gBACjB,MAAM,OAAO,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;gBAClC,IAAI,OAAO;oBAAE,IAAI,CAAC,MAAM,EAAE,CAAC;YAC7B,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,CAAC;QAC7C,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,CAAC,CAAC;QAE5C,iBAAiB;QACjB,IAAI,CAAC,MAAM,EAAE,CAAC;IAChB,CAAC;IAED,WAAW,KAAW,IAAI,CAAC,EAAE,EAAE,UAAU,EAAE,CAAC,CAAC,CAAC;IAE9C,WAAW,KAAW,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC,CAAC;IAEtC,gCAAgC;IACxB,UAAU;QAChB,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,EAAE,aAAa,EAAE,YAAY,IAAI,CAAC,CAAC,CAAC;QAC1E,MAAM,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,EAAE,aAAa,EAAE,WAAW,IAAI,CAAC,CAAC,CAAC;QAExE,IAAI,OAAO,GAAG,KAAK,CAAC;QACpB,IAAI,KAAK,IAAI,KAAK,KAAK,IAAI,CAAC,MAAM,EAAE,EAAE,CAAC;YAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAAC,OAAO,GAAG,IAAI,CAAC;QAAC,CAAC;QACjF,IAAI,KAAK,IAAI,KAAK,KAAK,IAAI,CAAC,MAAM,EAAE,EAAE,CAAC;YAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC;YAAC,OAAO,GAAG,IAAI,CAAC;QAAC,CAAC;QAEjF,mFAAmF;QACnF,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,CAAC;YAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAAC,OAAO,GAAG,IAAI,CAAC;QAAC,CAAC;QACnE,IAAI,CAAC,IAAI,CAAC,MAAM,EAAE,EAAE,CAAC;YAAC,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;YAAC,OAAO,GAAG,IAAI,CAAC;QAAC,CAAC;QAEvF,OAAO,OAAO,CAAC;IACjB,CAAC;IAEO,MAAM;QACZ,MAAM,OAAO,GAAG,CAAC,IAAI,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;YAC1C,KAAK,EAAE,CAAC,CAAC,KAAK;YACd,KAAK,EAAE,CAAC,CAAC,CAAC,KAAK,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;SAC1D,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAEpC,MAAM,MAAM,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC;QAEzC,iBAAiB;QACjB,MAAM,SAAS,GAAG,IAAI,GAAG,EAAU,CAAC;QACpC,KAAK,MAAM,CAAC,IAAI,OAAO;YAAE,KAAK,MAAM,EAAE,IAAI,CAAC,CAAC,KAAK;gBAAE,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;QAC3E,IAAI,MAAM,GAAG,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACnC,IAAI,IAAI,CAAC,WAAW,EAAE,MAAM,EAAE,CAAC;YAC7B,MAAM,KAAK,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,GAAG,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC;YAClE,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,GAAG,CAAC,GAAG,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC;QACvE,CAAC;QAED,oBAAoB;QACpB,MAAM,OAAO,GAAG;YACd,SAAS,EAAE,WAAW;YACtB,SAAS,EAAE,WAAW;YACtB,SAAS,EAAE,cAAc;YACzB,SAAS,EAAE,WAAW;YACtB,SAAS,EAAE,cAAc;YACzB,SAAS,EAAE,WAAW;SACvB,CAAC;QACF,MAAM,QAAQ,GAAmB,EAAE,CAAC;QACpC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC,GAAG,OAAO,CAAC,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,CAAC;QAEpE,aAAa;QACb,MAAM,UAAU,GAAG,OAAO,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;YACjC,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;YACpD,MAAM,KAAK,GAAG,MAAM,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE;gBAC3B,MAAM,CAAC,GAAG,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC;gBACrB,OAAO,EAAE,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,KAAK,IAAI,QAAQ,CAAC,CAAC,CAAC,EAAE,CAAC;YAC5E,CAAC,CAAC,CAAC;YACH,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,CAAC;QACnC,CAAC,CAAC,CAAC;QAEH,QAAQ;QACR,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,GAAG,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;QACpF,MAAM,IAAI,GAAG,IAAI,CAAC,IAAI,IAAI,IAAI,CAAC,IAAI,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAC7E,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAErB,yCAAyC;QACzC,MAAM,QAAQ,GAAG,IAAI,CAAC,MAAM,EAAE,IAAI,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QAElE,oCAAoC;QACpC,MAAM,CAAC,GAAG,UAAU,CAAC,MAAM,CAAC;QAC5B,MAAM,CAAC,GAAG,MAAM,CAAC,MAAM,CAAC;QACxB,MAAM,eAAe,GAAG,CAAC,GAAG,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC;QAC7E,MAAM,iBAAiB,GACrB,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,QAAQ,GAAG,CAAC,CAAC,GAAG,CAAC,GAAG,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,CAAC;QAE9F,MAAM,iBAAiB,GAAG,IAAI,CAAC,GAAG,CAAC,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;QACpE,MAAM,iBAAiB,GAAG,IAAI,CAAC,GAAG,CAAC,iBAAiB,EAAE,iBAAiB,CAAC,CAAC;QACzE,MAAM,SAAS,GAAG,iBAAiB,GAAG,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,wCAAwC;QACjG,IAAI,CAAC,UAAU,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAE/B,kBAAkB;QAClB,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC9B,MAAM,IAAI,GAAc,EAAE,CAAC;QAC3B,MAAM,OAAO,GAAa,EAAE,CAAC;QAC7B,MAAM,OAAO,GAAgB,EAAE,CAAC;QAEhC,IAAI,MAAM,GAAG,IAAI,CAAC,QAAQ,CAAC;QAC3B,UAAU,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE;YACrB,MAAM,MAAM,GAAG,MAAM,CAAC;YACtB,MAAM,MAAM,GAAG,MAAM,GAAG,eAAe,GAAG,CAAC,CAAC;YAC5C,OAAO,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;YAErB,CAAC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE,EAAE,EAAE;gBACzB,MAAM,CAAC,GAAG,MAAM,GAAG,EAAE,GAAG,CAAC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,MAAM,CAAC,CAAC;gBAEtD,mBAAmB;gBACnB,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE,KAAK,EAAE,IAAI,CAAC,QAAQ,EAAE,CAAC,CAAC;gBAE1C,MAAM,CAAC,GAAG,EAAE,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;gBACrD,MAAM,CAAC,GAAG,IAAI,CAAC,MAAM,CAAC,GAAG,GAAG,EAAE,GAAG,CAAC,CAAC;gBACnC,IAAI,CAAC,IAAI,CAAC;oBACR,UAAU,EAAE,CAAC,CAAC,KAAK;oBACnB,WAAW,EAAE,EAAE,CAAC,KAAK;oBACrB,KAAK,EAAE,EAAE,CAAC,KAAK;oBACf,CAAC,EAAE,CAAC;oBACJ,KAAK,EAAE,IAAI,CAAC,QAAQ;oBACpB,MAAM,EAAE,CAAC;oBACT,KAAK,EAAE,EAAE,CAAC,KAAK,IAAI,QAAQ,CAAC,EAAE,CAAC,KAAK,CAAC;iBACtC,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YAEH,MAAM,IAAI,eAAe,GAAG,IAAI,CAAC,QAAQ,CAAC;QAC5C,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QACrB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACzB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACzB,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAC3B,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;QAChC,IAAI,CAAC,WAAW,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;IAChC,CAAC;IAEO,OAAO,CAAC,MAAc;QAC5B,IAAI,MAAM,IAAI,CAAC;YAAE,OAAO,CAAC,CAAC;QAC1B,MAAM,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;QAC3C,MAAM,IAAI,GAAG,IAAI,CAAC,GAAG,CAAC,EAAE,EAAE,GAAG,CAAC,CAAC;QAC/B,MAAM,CAAC,GAAG,IAAI,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,CAAC;QACnC,MAAM,IAAI,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAC1C,OAAO,IAAI,GAAG,IAAI,CAAC;IACrB,CAAC;IAED,IAAI,CAAC,KAAa;QAChB,MAAM,EAAE,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC;QAC9B,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;QAC7B,OAAO,IAAI,CAAC,MAAM,CAAC,GAAG,GAAG,EAAE,GAAG,CAAC,KAAK,GAAG,GAAG,CAAC,GAAG,EAAE,CAAC;IACnD,CAAC;IAED,IAAI,SAAS;QACX,MAAM,GAAG,GAAG,IAAI,CAAC,SAAS,EAAE,CAAC;QAC7B,MAAM,CAAC,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,EAAE,IAAI,CAAC,KAAK,GAAG,CAAC,CAAC,CAAC;QACtC,MAAM,IAAI,GAAG,GAAG,GAAG,CAAC,CAAC;QACrB,MAAM,GAAG,GAAa,EAAE,CAAC;QACzB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE;YAAE,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,CAAC;QAChE,OAAO,GAAG,CAAC;IACb,CAAC;IAED,UAAU,CAAC,GAAW;QACpB,OAAO,GAAG,CAAC,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE,IAAI,CAAC,aAAa,GAAG,CAAC,CAAC,GAAG,GAAG,CAAC,CAAC,CAAC,CAAC,GAAG,CAAC;IAC9F,CAAC;IAED,aAAa,CAAC,CAAU;QACtB,OAAO,GAAG,CAAC,CAAC,UAAU,MAAM,CAAC,CAAC,WAAW,KAAK,CAAC,CAAC,KAAK,EAAE,CAAC;IAC1D,CAAC;IAED,UAAU,CAAC,CAAU;QACnB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC,UAAU,EAAE,MAAM,EAAE,CAAC,CAAC,WAAW,EAAE,KAAK,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;IACrF,CAAC;+GAnPU,4BAA4B;mGAA5B,4BAA4B,sdChCzC,4tGA6EA,o+FDjDY,YAAY,+BAAE,gBAAgB;;4FAI7B,4BAA4B;kBAPxC,SAAS;+BACE,wBAAwB,cACtB,IAAI,WACP,CAAC,YAAY,EAAE,gBAAgB,CAAC;2EAKhC,IAAI;sBAAZ,KAAK;gBACG,cAAc;sBAAtB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBAsBG,aAAa;sBAArB,KAAK;gBAEI,QAAQ;sBAAjB,MAAM;gBAG+B,QAAQ;sBAA7C,SAAS;uBAAC,OAAO,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBACI,OAAO;sBAA9C,SAAS;uBAAC,SAAS,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE","sourcesContent":["import { CommonModule } from '@angular/common';\r\nimport {\r\n  Component, Input, Output, EventEmitter,\r\n  OnChanges, computed, signal, ViewChild, ElementRef, AfterViewInit, OnDestroy, NgZone\r\n} from '@angular/core';\r\nimport { PopoverDirective } from '../directives/popover.directive';\r\n\r\nexport type BarSeriesItem = { label: string; value: number; color?: string };\r\nexport type BarGroup = { label: string; items: BarSeriesItem[] };\r\n\r\ntype SeriesColorMap = Record<string, string>;\r\n\r\ntype BarRect = {\r\n  groupLabel: string;\r\n  seriesLabel: string;\r\n  value: number;\r\n  x: number;\r\n  y: number;\r\n  width: number;\r\n  height: number;\r\n  color: string;\r\n};\r\n\r\ntype BarStripe = { x: number; width: number };\r\n\r\n@Component({\r\n  selector: 'ius-charts-bar-grouped',\r\n  standalone: true,\r\n  imports: [CommonModule, PopoverDirective],\r\n  templateUrl: './charts-bar-grouped.component.html',\r\n  styleUrls: ['./charts-bar-grouped.component.scss'],\r\n})\r\nexport class IusChartsBarGroupedComponent implements OnChanges, AfterViewInit, OnDestroy {\r\n  @Input() data: BarGroup[] = [];\r\n  @Input() legendPosition: 'right' | 'bottom' = 'bottom';\r\n  @Input() seriesOrder?: string[];\r\n\r\n  // Márgenes (left = ancho panel eje Y)\r\n  margin = { top: 24, right: 16, bottom: 34, left: 40 };\r\n\r\n  // Layout X fijo\r\n  barWidth = 11;\r\n  barGap = 2;\r\n  groupGap = 40;\r\n  outerGap = 18;\r\n\r\n  // Estilo barra\r\n  barRadius = 5;\r\n\r\n  // Escala Y\r\n  yMax?: number;\r\n  ticks = 5;\r\n\r\n  // Fallbacks si no hay tamaño CSS externo\r\n  width = 640;\r\n  size = 320;\r\n\r\n  @Input() maxLabelChars = 10;\r\n\r\n  @Output() barClick = new EventEmitter<{ group: string; series: string; value: number }>();\r\n\r\n  // Refs\r\n  @ViewChild('panes', { static: true }) panesRef!: ElementRef<HTMLElement>;   // ← alto real (ejeY+plot)\r\n  @ViewChild('plotCtn', { static: true }) plotCtn!: ElementRef<HTMLElement>;     // ← ancho visible (scroll)\r\n  private ro?: ResizeObserver;\r\n\r\n  // Estado\r\n  private _bars = signal<BarRect[]>([]);\r\n  private _groups = signal<string[]>([]);\r\n  private _series = signal<string[]>([]);\r\n  private _colors = signal<SeriesColorMap>({});\r\n  private _yMax = signal(0);\r\n\r\n  private _paneH = signal(0);     // alto real del panel superior\r\n  private _plotW = signal(0);     // ancho visible del plot\r\n  private _plotWidth = signal(0); // ancho virtual del SVG derecho\r\n  private _groupCenters = signal<number[]>([]);\r\n  private _barStripes = signal<BarStripe[]>([]);\r\n\r\n  readonly bars = computed(() => this._bars());\r\n  readonly groups = computed(() => this._groups());\r\n  readonly series = computed(() => this._series());\r\n  readonly colors = computed(() => this._colors());\r\n  readonly yScaleMax = computed(() => this._yMax());\r\n\r\n  readonly chartHeight = computed(() => this._paneH() || this.size); // ← altura para ambos SVG\r\n  readonly plotWidth = computed(() => this._plotWidth());\r\n  readonly groupCenters = computed(() => this._groupCenters());\r\n  readonly barStripes = computed(() => this._barStripes());\r\n\r\n  readonly innerHeight = computed(() =>\r\n    Math.max(this.chartHeight() - this.margin.top - this.margin.bottom, 0)\r\n  );\r\n\r\n  // viewBox por SVG\r\n  readonly axisViewBox = computed(() => `0 0 ${this.margin.left} ${this.chartHeight()}`);\r\n  readonly plotViewBox = computed(() => `0 0 ${this.plotWidth()} ${this.chartHeight()}`);\r\n\r\n  readonly Math = Math;\r\n\r\n  constructor(private zone: NgZone) { }\r\n\r\n  ngAfterViewInit(): void {\r\n    // Medición inicial SIN esperar al primer tick del RO\r\n    this.measureNow();\r\n\r\n    // Observa alto (panes) y ancho visible (plotCtn)\r\n    this.ro = new ResizeObserver(() => {\r\n      this.zone.run(() => {\r\n        const changed = this.measureNow();\r\n        if (changed) this.recalc();\r\n      });\r\n    });\r\n    this.ro.observe(this.panesRef.nativeElement);\r\n    this.ro.observe(this.plotCtn.nativeElement);\r\n\r\n    // Primer cálculo\r\n    this.recalc();\r\n  }\r\n\r\n  ngOnDestroy(): void { this.ro?.disconnect(); }\r\n\r\n  ngOnChanges(): void { this.recalc(); }\r\n\r\n  // Devuelve true si cambió algo.\r\n  private measureNow(): boolean {\r\n    const paneH = Math.floor(this.panesRef?.nativeElement?.clientHeight || 0);\r\n    const plotW = Math.floor(this.plotCtn?.nativeElement?.clientWidth || 0);\r\n\r\n    let changed = false;\r\n    if (paneH && paneH !== this._paneH()) { this._paneH.set(paneH); changed = true; }\r\n    if (plotW && plotW !== this._plotW()) { this._plotW.set(plotW); changed = true; }\r\n\r\n    // Si no hay valores aún (por render temprano), usa fallbacks para no quedarte en 0\r\n    if (!this._paneH()) { this._paneH.set(this.size); changed = true; }\r\n    if (!this._plotW()) { this._plotW.set(this.width - this.margin.left); changed = true; }\r\n\r\n    return changed;\r\n  }\r\n\r\n  private recalc(): void {\r\n    const cleaned = (this.data ?? []).map(g => ({\r\n      label: g.label,\r\n      items: (g.items ?? []).filter(i => Number(i?.value) >= 0)\r\n    })).filter(x => x.items.length > 0);\r\n\r\n    const groups = cleaned.map(g => g.label);\r\n\r\n    // series + orden\r\n    const seriesSet = new Set<string>();\r\n    for (const g of cleaned) for (const it of g.items) seriesSet.add(it.label);\r\n    let series = Array.from(seriesSet);\r\n    if (this.seriesOrder?.length) {\r\n      const order = new Map(this.seriesOrder.map((k, idx) => [k, idx]));\r\n      series.sort((a, b) => (order.get(a) ?? 1e9) - (order.get(b) ?? 1e9));\r\n    }\r\n\r\n    // colores por serie\r\n    const palette = [\r\n      '#2167FF', // Info-500\r\n      '#0A2893', // Info-800\r\n      '#40CAE9', // Primary-400\r\n      '#5892FF', // Info-400\r\n      '#08A6DB', // Primary-500\r\n      '#C4DFFF', // Info-200\r\n    ];\r\n    const colorMap: SeriesColorMap = {};\r\n    series.forEach((s, i) => colorMap[s] = palette[i % palette.length]);\r\n\r\n    // normalizar\r\n    const normalized = cleaned.map(g => {\r\n      const map = new Map(g.items.map(i => [i.label, i]));\r\n      const items = series.map(s => {\r\n        const i = map.get(s);\r\n        return { label: s, value: i?.value ?? 0, color: i?.color ?? colorMap[s] };\r\n      });\r\n      return { label: g.label, items };\r\n    });\r\n\r\n    // Y máx\r\n    const maxValue = Math.max(0, ...normalized.flatMap(g => g.items.map(i => i.value)));\r\n    const yMax = this.yMax && this.yMax > 0 ? this.yMax : this.niceMax(maxValue);\r\n    this._yMax.set(yMax);\r\n\r\n    // dimensiones visibles del panel derecho\r\n    const visibleW = this._plotW() || (this.width - this.margin.left);\r\n\r\n    // ===== Layout X con outerGap =====\r\n    const G = normalized.length;\r\n    const S = series.length;\r\n    const groupInnerWidth = S * this.barWidth + Math.max(0, S - 1) * this.barGap;\r\n    const innerNaturalWidth =\r\n      Math.max(0, (this.outerGap * 2) + G * groupInnerWidth + Math.max(0, G - 1) * this.groupGap);\r\n\r\n    const innerVisibleWidth = Math.max(visibleW - this.margin.right, 0);\r\n    const innerVirtualWidth = Math.max(innerVisibleWidth, innerNaturalWidth);\r\n    const plotWidth = innerVirtualWidth + this.margin.right; // solo margen derecho en el SVG derecho\r\n    this._plotWidth.set(plotWidth);\r\n\r\n    // rects + franjas\r\n    const ih = this.innerHeight();\r\n    const bars: BarRect[] = [];\r\n    const centers: number[] = [];\r\n    const stripes: BarStripe[] = [];\r\n\r\n    let cursor = this.outerGap;\r\n    normalized.forEach(g => {\r\n      const startX = cursor;\r\n      const center = startX + groupInnerWidth / 2;\r\n      centers.push(center);\r\n\r\n      g.items.forEach((it, si) => {\r\n        const x = startX + si * (this.barWidth + this.barGap);\r\n\r\n        // franja por barra\r\n        stripes.push({ x, width: this.barWidth });\r\n\r\n        const h = it.value <= 0 ? 0 : (it.value / yMax) * ih;\r\n        const y = this.margin.top + ih - h;\r\n        bars.push({\r\n          groupLabel: g.label,\r\n          seriesLabel: it.label,\r\n          value: it.value,\r\n          x, y,\r\n          width: this.barWidth,\r\n          height: h,\r\n          color: it.color ?? colorMap[it.label]\r\n        });\r\n      });\r\n\r\n      cursor += groupInnerWidth + this.groupGap;\r\n    });\r\n\r\n    this._bars.set(bars);\r\n    this._groups.set(groups);\r\n    this._series.set(series);\r\n    this._colors.set(colorMap);\r\n    this._groupCenters.set(centers);\r\n    this._barStripes.set(stripes);\r\n  }\r\n\r\n  private niceMax(maxVal: number): number {\r\n    if (maxVal <= 0) return 1;\r\n    const exp = Math.floor(Math.log10(maxVal));\r\n    const base = Math.pow(10, exp);\r\n    const n = Math.ceil(maxVal / base);\r\n    const nice = n <= 2 ? 2 : n <= 5 ? 5 : 10;\r\n    return nice * base;\r\n  }\r\n\r\n  getY(value: number): number {\r\n    const ih = this.innerHeight();\r\n    const max = this.yScaleMax();\r\n    return this.margin.top + ih - (value / max) * ih;\r\n  }\r\n\r\n  get yTicksArr(): number[] {\r\n    const max = this.yScaleMax();\r\n    const n = Math.max(2, this.ticks | 0);\r\n    const step = max / n;\r\n    const arr: number[] = [];\r\n    for (let i = 0; i <= n; i++) arr.push(+((i * step)).toFixed(2));\r\n    return arr;\r\n  }\r\n\r\n  shortLabel(lbl: string): string {\r\n    return lbl.length > this.maxLabelChars ? (lbl.slice(0, this.maxLabelChars - 1) + '…') : lbl;\r\n  }\r\n\r\n  formatTooltip(b: BarRect): string {\r\n    return `${b.groupLabel} — ${b.seriesLabel}: ${b.value}`;\r\n  }\r\n\r\n  onBarClick(b: BarRect) {\r\n    this.barClick.emit({ group: b.groupLabel, series: b.seriesLabel, value: b.value });\r\n  }\r\n}\r\n","<div class=\"chart-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n    <div class=\"panes\" #panes>\r\n        <div class=\"axis-ctn\">\r\n            <svg class=\"chart-axis\"\r\n                [attr.viewBox]=\"axisViewBox()\"\r\n                preserveAspectRatio=\"xMinYMin meet\"\r\n                [style.width.px]=\"margin.left\"\r\n                [style.height.px]=\"chartHeight()\">\r\n                @for (t of yTicksArr; track t) {\r\n                <text class=\"y-tick body-sm\"\r\n                        [attr.x]=\"margin.left - 6\"\r\n                        [attr.y]=\"getY(t)\"\r\n                        text-anchor=\"end\"\r\n                        dominant-baseline=\"middle\">\r\n                    {{ t }}\r\n                </text>\r\n                }\r\n            </svg>\r\n        </div>\r\n\r\n        <div class=\"plot-ctn scrollable-small\" #plotCtn>\r\n            <svg class=\"chart\"\r\n                [attr.viewBox]=\"plotViewBox()\"\r\n                preserveAspectRatio=\"xMinYMin meet\"\r\n                [style.width.px]=\"plotWidth()\"\r\n                [style.height.px]=\"chartHeight()\">\r\n\r\n                @for (s of barStripes(); track s.x) {\r\n                <rect class=\"bar-stripe\"\r\n                        [attr.x]=\"s.x\"\r\n                        [attr.y]=\"margin.top\"\r\n                        [attr.width]=\"s.width\"\r\n                        [attr.height]=\"innerHeight()\"></rect>\r\n                }\r\n\r\n                @for (t of yTicksArr; track t) {\r\n                <line class=\"grid-line\"\r\n                        [attr.x1]=\"0\"\r\n                        [attr.x2]=\"plotWidth() - margin.right\"\r\n                        [attr.y1]=\"getY(t)\" [attr.y2]=\"getY(t)\"/>\r\n                }\r\n\r\n                @for (g of groups(); track g; let i = $index) {\r\n                <text class=\"x-label body-sm\"\r\n                        [attr.x]=\"groupCenters()[i]\"\r\n                        [attr.y]=\"chartHeight() - margin.bottom + 22\"\r\n                        text-anchor=\"middle\">\r\n                    {{ shortLabel(g) }} <title>{{ g }}</title>\r\n                </text>\r\n                }\r\n\r\n                @for (b of bars(); track b.groupLabel + '_' + b.seriesLabel) {\r\n                <rect class=\"bar\"\r\n                        [attr.x]=\"b.x\"\r\n                        [attr.y]=\"b.y\"\r\n                        [attr.width]=\"b.width\"\r\n                        [attr.height]=\"Math.max(b.height, 0)\"\r\n                        [attr.fill]=\"b.color\"\r\n                        [attr.rx]=\"barRadius\"\r\n                        [attr.ry]=\"barRadius\"\r\n                        [iusPopoverContent]=\"formatTooltip(b)\"\r\n                        (click)=\"onBarClick(b)\"/>\r\n                }\r\n            </svg>\r\n        </div>\r\n    </div>\r\n\r\n  <!-- Leyenda -->\r\n    <div class=\"legend\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n        @for (s of series(); track s) {\r\n        <div class=\"legend-item\">\r\n            <span class=\"dot\" [style.background]=\"colors()[s]\"></span>\r\n            <span class=\"label caption-sm\">{{ s }}</span>\r\n        </div>\r\n        }\r\n    </div>\r\n</div>\r\n"]}
@@ -0,0 +1,2 @@
1
+ export * from './charts-bar-grouped.component';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9pdXMtZGVzaWduLWNvbXBvbmVudHMvc3JjL2xpYi9jaGFydHMtYmFyLWdyb3VwZWQvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxnQ0FBZ0MsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vY2hhcnRzLWJhci1ncm91cGVkLmNvbXBvbmVudCc7XHJcblxyXG4iXX0=
@@ -110,11 +110,11 @@ export class ChartsDonutComponent {
110
110
  return `Gráfica de dona. Total ${this.total()}. ${parts.join('. ')}`;
111
111
  }
112
112
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChartsDonutComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
113
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: ChartsDonutComponent, isStandalone: true, selector: "ius-charts-donut", inputs: { data: "data", size: "size", thickness: "thickness", gapDegrees: "gapDegrees", showLegend: "showLegend", legendPosition: "legendPosition", centerText: "centerText", sort: "sort" }, outputs: { sliceClick: "sliceClick" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"donut-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n <svg\r\n class=\"donut\"\r\n role=\"img\"\r\n [attr.aria-label]=\"ariaLabel\"\r\n [attr.width]=\"size\"\r\n [attr.height]=\"size\"\r\n [attr.viewBox]=\"viewBox()\">\r\n <!-- pista -->\r\n <circle\r\n class=\"track\"\r\n [attr.cx]=\"size/2\"\r\n [attr.cy]=\"size/2\"\r\n [attr.r]=\"radius()\"\r\n [attr.stroke-width]=\"thickness\"\r\n ></circle>\r\n\r\n <!-- slices como PATH (cada uno es su arco real) -->\r\n @for (s of slices(); track s.label) {\r\n <path\r\n class=\"slice\"\r\n [attr.d]=\"s.pathD\"\r\n fill=\"none\"\r\n [attr.stroke]=\"s.color\"\r\n [attr.stroke-width]=\"thickness\"\r\n stroke-linecap=\"butt\"\r\n [iusPopoverContent]=\"getPopoverText(s)\"\r\n (click)=\"onSliceClick(s)\"\r\n tabindex=\"0\"\r\n >\r\n </path>\r\n }\r\n\r\n <!-- centro -->\r\n <g>\r\n <text\r\n [attr.x]=\"size/2\"\r\n [attr.y]=\"size/2\"\r\n text-anchor=\"middle\"\r\n alignment-baseline=\"middle\"\r\n class=\"center-text\">\r\n {{ centerLabel }}\r\n </text>\r\n </g>\r\n </svg>\r\n\r\n @if (showLegend && slices().length) {\r\n <div class=\"legend scrollable-small\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n @for (s of slices(); track s.label) {\r\n <div class=\"legend-item\">\r\n <span class=\"dot\" [style.background]=\"s.color\"></span>\r\n <span class=\"label caption-sm\" [iusPopoverContent]=\"s.value.toString() + ' - ' + s.label\">{{s.value}} - {{s.label}}</span>\r\n </div>\r\n }\r\n </div>\r\n }\r\n</div>\r\n", styles: [".h1{font-family:Roboto,sans-serif;font-size:2.375rem;font-weight:500;line-height:46px}.h2{font-family:Roboto,sans-serif;font-size:1.875rem;font-weight:700;line-height:38px}.h3{font-family:Roboto,sans-serif;font-size:1.5rem;font-weight:500;line-height:32px}.h4{font-family:Roboto,sans-serif;font-size:1.25rem;font-weight:700;line-height:26px}.h5{font-family:Roboto,sans-serif;font-size:1.125rem;font-weight:500;line-height:24px;letter-spacing:.18px}.label-large{font-family:Roboto,sans-serif;font-size:1rem;font-weight:500;line-height:22px}.body-large{font-family:Rubik,sans-serif;font-size:1rem;font-weight:400;line-height:22px}.label-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:500;line-height:20px;letter-spacing:.28px}.body-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:400;line-height:20px;font-style:italic;letter-spacing:.28px}.body-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;letter-spacing:.28px}.body-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;font-style:italic;letter-spacing:.28px}.caption-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;letter-spacing:.28px}.caption-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;font-style:italic;letter-spacing:.28px}:host{display:block;max-width:100%;height:100%}.donut-wrapper{display:grid;grid-template-columns:auto 1fr;column-gap:16px;height:100%;align-items:stretch}.donut-wrapper.bottom{grid-template-columns:1fr;grid-template-rows:auto 1fr;row-gap:12px;align-items:stretch}.donut{display:block;max-width:100%;height:auto;align-self:center}.track{fill:none;stroke:#eef2f6}.slice{pointer-events:stroke;cursor:pointer;transition:opacity .15s ease,transform .15s ease}.slice:focus{outline:none;filter:drop-shadow(0 0 2px rgba(0,0,0,.2))}.slice:hover{opacity:.9}.center-text{fill:#595959;font-family:Rubik,sans-serif;font-size:1.5rem;font-weight:600}.legend{display:flex;flex-direction:column;gap:8px;padding-right:4px;min-width:0;overflow-x:hidden;overflow-y:auto;align-self:stretch;max-height:100%;min-height:0}.legend.bottom{flex-direction:row;flex-wrap:wrap;overflow-y:auto}.legend .legend-item{display:flex;align-items:center;gap:8px;min-width:0;white-space:nowrap}.legend .dot{width:8px;height:8px;border-radius:8px;flex:0 0 8px}.legend .label{color:#595959;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.scrollable-small::-webkit-scrollbar{-webkit-appearance:none}.scrollable-small::-webkit-scrollbar:vertical{width:4px;height:32px;padding-top:2px;padding-bottom:2px}.scrollable-small::-webkit-scrollbar:horizontal{height:5px;padding-top:2px;padding-bottom:2px}.scrollable-small::-webkit-scrollbar-thumb{background-color:#08a6db;border-radius:4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: PopoverDirective, selector: "[iusPopover], [iusPopoverTitle], [iusPopoverContent]", inputs: ["iusPopover", "iusPopoverTitle", "iusPopoverContent", "iusPopoverPosition", "iusPopoverOpenDelay", "iusPopoverCloseDelay"] }] }); }
113
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: ChartsDonutComponent, isStandalone: true, selector: "ius-charts-donut", inputs: { data: "data", size: "size", thickness: "thickness", gapDegrees: "gapDegrees", showLegend: "showLegend", legendPosition: "legendPosition", centerText: "centerText", sort: "sort" }, outputs: { sliceClick: "sliceClick" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"donut-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n <svg\r\n class=\"donut\"\r\n role=\"img\"\r\n [attr.aria-label]=\"ariaLabel\"\r\n [attr.width]=\"size\"\r\n [attr.height]=\"size\"\r\n [attr.viewBox]=\"viewBox()\">\r\n <!-- pista -->\r\n <circle\r\n class=\"track\"\r\n [attr.cx]=\"size/2\"\r\n [attr.cy]=\"size/2\"\r\n [attr.r]=\"radius()\"\r\n [attr.stroke-width]=\"thickness\"\r\n ></circle>\r\n\r\n <!-- slices como PATH (cada uno es su arco real) -->\r\n @for (s of slices(); track s.label) {\r\n <path\r\n class=\"slice\"\r\n [attr.d]=\"s.pathD\"\r\n fill=\"none\"\r\n [attr.stroke]=\"s.color\"\r\n [attr.stroke-width]=\"thickness\"\r\n stroke-linecap=\"butt\"\r\n [iusPopoverContent]=\"getPopoverText(s)\"\r\n (click)=\"onSliceClick(s)\"\r\n tabindex=\"0\"\r\n >\r\n </path>\r\n }\r\n\r\n <!-- centro -->\r\n <g>\r\n <text\r\n [attr.x]=\"size/2\"\r\n [attr.y]=\"size/2\"\r\n text-anchor=\"middle\"\r\n alignment-baseline=\"middle\"\r\n class=\"center-text\">\r\n {{ centerLabel }}\r\n </text>\r\n </g>\r\n </svg>\r\n\r\n @if (showLegend && slices().length) {\r\n <div class=\"legend scrollable-small\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n @for (s of slices(); track s.label) {\r\n <div class=\"legend-item\">\r\n <span class=\"dot\" [style.background]=\"s.color\"></span>\r\n <span class=\"label caption-sm\" [iusPopoverContent]=\"s.value.toString() + ' - ' + s.label\">{{s.value}} - {{s.label}}</span>\r\n </div>\r\n }\r\n </div>\r\n }\r\n</div>\r\n", styles: [".h1{font-family:Roboto,sans-serif;font-size:2.375rem;font-weight:500;line-height:46px}.h2{font-family:Roboto,sans-serif;font-size:1.875rem;font-weight:700;line-height:38px}.h3{font-family:Roboto,sans-serif;font-size:1.5rem;font-weight:500;line-height:32px}.h4{font-family:Roboto,sans-serif;font-size:1.25rem;font-weight:700;line-height:26px}.h5{font-family:Roboto,sans-serif;font-size:1.125rem;font-weight:500;line-height:24px;letter-spacing:.18px}.label-large{font-family:Roboto,sans-serif;font-size:1rem;font-weight:500;line-height:22px}.body-large{font-family:Rubik,sans-serif;font-size:1rem;font-weight:400;line-height:22px}.label-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:500;line-height:20px;letter-spacing:.28px}.body-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:400;line-height:20px;font-style:italic;letter-spacing:.28px}.body-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;letter-spacing:.28px}.body-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;font-style:italic;letter-spacing:.28px}.caption-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;letter-spacing:.28px}.caption-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;font-style:italic;letter-spacing:.28px}:host{display:block;max-width:100%;height:100%}.donut-wrapper{display:grid;grid-template-columns:auto 1fr;column-gap:16px;height:100%;align-items:stretch}.donut-wrapper.bottom{grid-template-columns:1fr;grid-template-rows:auto 1fr;row-gap:12px;align-items:stretch}.donut{display:block;max-width:100%;height:auto;align-self:center}.track{fill:none;stroke:#eef2f6}.slice{pointer-events:stroke;cursor:pointer;transition:opacity .15s ease,transform .15s ease}.slice:focus{outline:none;filter:drop-shadow(0 0 2px rgba(0,0,0,.2))}.slice:hover{opacity:.9}.center-text{fill:#595959;font-family:Rubik,sans-serif;font-size:1.5rem;font-weight:600}.legend{display:flex;justify-content:center;flex-direction:column;gap:8px;padding-right:4px;min-width:0;overflow-x:hidden;overflow-y:auto;align-self:stretch;max-height:100%;min-height:0}.legend.bottom{flex-direction:row;flex-wrap:wrap;overflow-y:auto}.legend .legend-item{display:flex;align-items:center;gap:8px;min-width:0;white-space:nowrap}.legend .dot{width:8px;height:8px;border-radius:8px;flex:0 0 8px}.legend .label{color:#595959;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.scrollable-small::-webkit-scrollbar{-webkit-appearance:none}.scrollable-small::-webkit-scrollbar:vertical{width:4px;height:32px;padding-top:2px;padding-bottom:2px}.scrollable-small::-webkit-scrollbar:horizontal{height:5px;padding-top:2px;padding-bottom:2px}.scrollable-small::-webkit-scrollbar-thumb{background-color:#08a6db;border-radius:4px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: PopoverDirective, selector: "[iusPopover], [iusPopoverTitle], [iusPopoverContent]", inputs: ["iusPopover", "iusPopoverTitle", "iusPopoverContent", "iusPopoverPosition", "iusPopoverOpenDelay", "iusPopoverCloseDelay"] }] }); }
114
114
  }
115
115
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChartsDonutComponent, decorators: [{
116
116
  type: Component,
117
- args: [{ selector: 'ius-charts-donut', standalone: true, imports: [CommonModule, PopoverDirective], template: "<div class=\"donut-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n <svg\r\n class=\"donut\"\r\n role=\"img\"\r\n [attr.aria-label]=\"ariaLabel\"\r\n [attr.width]=\"size\"\r\n [attr.height]=\"size\"\r\n [attr.viewBox]=\"viewBox()\">\r\n <!-- pista -->\r\n <circle\r\n class=\"track\"\r\n [attr.cx]=\"size/2\"\r\n [attr.cy]=\"size/2\"\r\n [attr.r]=\"radius()\"\r\n [attr.stroke-width]=\"thickness\"\r\n ></circle>\r\n\r\n <!-- slices como PATH (cada uno es su arco real) -->\r\n @for (s of slices(); track s.label) {\r\n <path\r\n class=\"slice\"\r\n [attr.d]=\"s.pathD\"\r\n fill=\"none\"\r\n [attr.stroke]=\"s.color\"\r\n [attr.stroke-width]=\"thickness\"\r\n stroke-linecap=\"butt\"\r\n [iusPopoverContent]=\"getPopoverText(s)\"\r\n (click)=\"onSliceClick(s)\"\r\n tabindex=\"0\"\r\n >\r\n </path>\r\n }\r\n\r\n <!-- centro -->\r\n <g>\r\n <text\r\n [attr.x]=\"size/2\"\r\n [attr.y]=\"size/2\"\r\n text-anchor=\"middle\"\r\n alignment-baseline=\"middle\"\r\n class=\"center-text\">\r\n {{ centerLabel }}\r\n </text>\r\n </g>\r\n </svg>\r\n\r\n @if (showLegend && slices().length) {\r\n <div class=\"legend scrollable-small\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n @for (s of slices(); track s.label) {\r\n <div class=\"legend-item\">\r\n <span class=\"dot\" [style.background]=\"s.color\"></span>\r\n <span class=\"label caption-sm\" [iusPopoverContent]=\"s.value.toString() + ' - ' + s.label\">{{s.value}} - {{s.label}}</span>\r\n </div>\r\n }\r\n </div>\r\n }\r\n</div>\r\n", styles: [".h1{font-family:Roboto,sans-serif;font-size:2.375rem;font-weight:500;line-height:46px}.h2{font-family:Roboto,sans-serif;font-size:1.875rem;font-weight:700;line-height:38px}.h3{font-family:Roboto,sans-serif;font-size:1.5rem;font-weight:500;line-height:32px}.h4{font-family:Roboto,sans-serif;font-size:1.25rem;font-weight:700;line-height:26px}.h5{font-family:Roboto,sans-serif;font-size:1.125rem;font-weight:500;line-height:24px;letter-spacing:.18px}.label-large{font-family:Roboto,sans-serif;font-size:1rem;font-weight:500;line-height:22px}.body-large{font-family:Rubik,sans-serif;font-size:1rem;font-weight:400;line-height:22px}.label-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:500;line-height:20px;letter-spacing:.28px}.body-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:400;line-height:20px;font-style:italic;letter-spacing:.28px}.body-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;letter-spacing:.28px}.body-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;font-style:italic;letter-spacing:.28px}.caption-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;letter-spacing:.28px}.caption-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;font-style:italic;letter-spacing:.28px}:host{display:block;max-width:100%;height:100%}.donut-wrapper{display:grid;grid-template-columns:auto 1fr;column-gap:16px;height:100%;align-items:stretch}.donut-wrapper.bottom{grid-template-columns:1fr;grid-template-rows:auto 1fr;row-gap:12px;align-items:stretch}.donut{display:block;max-width:100%;height:auto;align-self:center}.track{fill:none;stroke:#eef2f6}.slice{pointer-events:stroke;cursor:pointer;transition:opacity .15s ease,transform .15s ease}.slice:focus{outline:none;filter:drop-shadow(0 0 2px rgba(0,0,0,.2))}.slice:hover{opacity:.9}.center-text{fill:#595959;font-family:Rubik,sans-serif;font-size:1.5rem;font-weight:600}.legend{display:flex;flex-direction:column;gap:8px;padding-right:4px;min-width:0;overflow-x:hidden;overflow-y:auto;align-self:stretch;max-height:100%;min-height:0}.legend.bottom{flex-direction:row;flex-wrap:wrap;overflow-y:auto}.legend .legend-item{display:flex;align-items:center;gap:8px;min-width:0;white-space:nowrap}.legend .dot{width:8px;height:8px;border-radius:8px;flex:0 0 8px}.legend .label{color:#595959;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.scrollable-small::-webkit-scrollbar{-webkit-appearance:none}.scrollable-small::-webkit-scrollbar:vertical{width:4px;height:32px;padding-top:2px;padding-bottom:2px}.scrollable-small::-webkit-scrollbar:horizontal{height:5px;padding-top:2px;padding-bottom:2px}.scrollable-small::-webkit-scrollbar-thumb{background-color:#08a6db;border-radius:4px}\n"] }]
117
+ args: [{ selector: 'ius-charts-donut', standalone: true, imports: [CommonModule, PopoverDirective], template: "<div class=\"donut-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n <svg\r\n class=\"donut\"\r\n role=\"img\"\r\n [attr.aria-label]=\"ariaLabel\"\r\n [attr.width]=\"size\"\r\n [attr.height]=\"size\"\r\n [attr.viewBox]=\"viewBox()\">\r\n <!-- pista -->\r\n <circle\r\n class=\"track\"\r\n [attr.cx]=\"size/2\"\r\n [attr.cy]=\"size/2\"\r\n [attr.r]=\"radius()\"\r\n [attr.stroke-width]=\"thickness\"\r\n ></circle>\r\n\r\n <!-- slices como PATH (cada uno es su arco real) -->\r\n @for (s of slices(); track s.label) {\r\n <path\r\n class=\"slice\"\r\n [attr.d]=\"s.pathD\"\r\n fill=\"none\"\r\n [attr.stroke]=\"s.color\"\r\n [attr.stroke-width]=\"thickness\"\r\n stroke-linecap=\"butt\"\r\n [iusPopoverContent]=\"getPopoverText(s)\"\r\n (click)=\"onSliceClick(s)\"\r\n tabindex=\"0\"\r\n >\r\n </path>\r\n }\r\n\r\n <!-- centro -->\r\n <g>\r\n <text\r\n [attr.x]=\"size/2\"\r\n [attr.y]=\"size/2\"\r\n text-anchor=\"middle\"\r\n alignment-baseline=\"middle\"\r\n class=\"center-text\">\r\n {{ centerLabel }}\r\n </text>\r\n </g>\r\n </svg>\r\n\r\n @if (showLegend && slices().length) {\r\n <div class=\"legend scrollable-small\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n @for (s of slices(); track s.label) {\r\n <div class=\"legend-item\">\r\n <span class=\"dot\" [style.background]=\"s.color\"></span>\r\n <span class=\"label caption-sm\" [iusPopoverContent]=\"s.value.toString() + ' - ' + s.label\">{{s.value}} - {{s.label}}</span>\r\n </div>\r\n }\r\n </div>\r\n }\r\n</div>\r\n", styles: [".h1{font-family:Roboto,sans-serif;font-size:2.375rem;font-weight:500;line-height:46px}.h2{font-family:Roboto,sans-serif;font-size:1.875rem;font-weight:700;line-height:38px}.h3{font-family:Roboto,sans-serif;font-size:1.5rem;font-weight:500;line-height:32px}.h4{font-family:Roboto,sans-serif;font-size:1.25rem;font-weight:700;line-height:26px}.h5{font-family:Roboto,sans-serif;font-size:1.125rem;font-weight:500;line-height:24px;letter-spacing:.18px}.label-large{font-family:Roboto,sans-serif;font-size:1rem;font-weight:500;line-height:22px}.body-large{font-family:Rubik,sans-serif;font-size:1rem;font-weight:400;line-height:22px}.label-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:500;line-height:20px;letter-spacing:.28px}.body-base{font-family:Rubik,sans-serif;font-size:.875rem;font-weight:400;line-height:20px;font-style:italic;letter-spacing:.28px}.body-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;letter-spacing:.28px}.body-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:400;line-height:16px;font-style:italic;letter-spacing:.28px}.caption-sm{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;letter-spacing:.28px}.caption-sm-italic{font-family:Rubik,sans-serif;font-size:.75rem;font-weight:500;line-height:16px;font-style:italic;letter-spacing:.28px}:host{display:block;max-width:100%;height:100%}.donut-wrapper{display:grid;grid-template-columns:auto 1fr;column-gap:16px;height:100%;align-items:stretch}.donut-wrapper.bottom{grid-template-columns:1fr;grid-template-rows:auto 1fr;row-gap:12px;align-items:stretch}.donut{display:block;max-width:100%;height:auto;align-self:center}.track{fill:none;stroke:#eef2f6}.slice{pointer-events:stroke;cursor:pointer;transition:opacity .15s ease,transform .15s ease}.slice:focus{outline:none;filter:drop-shadow(0 0 2px rgba(0,0,0,.2))}.slice:hover{opacity:.9}.center-text{fill:#595959;font-family:Rubik,sans-serif;font-size:1.5rem;font-weight:600}.legend{display:flex;justify-content:center;flex-direction:column;gap:8px;padding-right:4px;min-width:0;overflow-x:hidden;overflow-y:auto;align-self:stretch;max-height:100%;min-height:0}.legend.bottom{flex-direction:row;flex-wrap:wrap;overflow-y:auto}.legend .legend-item{display:flex;align-items:center;gap:8px;min-width:0;white-space:nowrap}.legend .dot{width:8px;height:8px;border-radius:8px;flex:0 0 8px}.legend .label{color:#595959;min-width:0;overflow:hidden;text-overflow:ellipsis;white-space:nowrap}.scrollable-small::-webkit-scrollbar{-webkit-appearance:none}.scrollable-small::-webkit-scrollbar:vertical{width:4px;height:32px;padding-top:2px;padding-bottom:2px}.scrollable-small::-webkit-scrollbar:horizontal{height:5px;padding-top:2px;padding-bottom:2px}.scrollable-small::-webkit-scrollbar-thumb{background-color:#08a6db;border-radius:4px}\n"] }]
118
118
  }], propDecorators: { data: [{
119
119
  type: Input
120
120
  }], size: [{
@@ -134,4 +134,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
134
134
  }], sliceClick: [{
135
135
  type: Output
136
136
  }] } });
137
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"charts-donut.component.js","sourceRoot":"","sources":["../../../../../projects/ius-design-components/src/lib/charts-donut/charts-donut.component.ts","../../../../../projects/ius-design-components/src/lib/charts-donut/charts-donut.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, computed, EventEmitter, Input, Output, signal, OnChanges } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { PopoverDirective } from '../directives/popover.directive';\r\n\r\nexport type DonutData = { label: string; value: number; color?: string };\r\n\r\ntype SliceComputed = DonutData & {\r\n  percentage: number;\r\n  pathD: string;   // path del arco\r\n  midX: number;    // punto medio del arco (por si se quiere anclar ahí)\r\n  midY: number;\r\n};\r\n\r\n@Component({\r\n  selector: 'ius-charts-donut',\r\n  standalone: true,\r\n  imports: [CommonModule, PopoverDirective],\r\n  templateUrl: './charts-donut.component.html',\r\n  styleUrls: ['./charts-donut.component.scss']\r\n})\r\nexport class ChartsDonutComponent implements OnChanges {\r\n  @Input() data: DonutData[] = [];\r\n  @Input() size = 180;                 // width/height del SVG\r\n  @Input() thickness = 18;             // grosor del anillo\r\n  @Input() gapDegrees = 0;             // separación entre segmentos (grados)\r\n  @Input() showLegend = true;\r\n  @Input() legendPosition: 'right' | 'bottom' = 'right';\r\n  @Input() centerText?: string | number; // sobrescribe el texto central\r\n  @Input() sort = false;               // ordena segmentos por valor (desc)\r\n\r\n  @Output() sliceClick = new EventEmitter<DonutData>();\r\n\r\n  // señales para recalcular\r\n  private _slices = signal<SliceComputed[]>([]);\r\n  private _total = signal(0);\r\n  private _circ = signal(0);\r\n  private _radius = signal(0);\r\n\r\n  readonly slices = computed(() => this._slices());\r\n  readonly total = computed(() => this._total());\r\n  readonly radius = computed(() => this._radius());\r\n  readonly circumference = computed(() => this._circ());\r\n  readonly viewBox = computed(() => `0 0 ${this.size} ${this.size}`);\r\n\r\n  ngOnChanges(): void {\r\n    const cleaned = (this.data ?? []).filter(d => Number(d?.value) > 0);\r\n    const total = cleaned.reduce((a, b) => a + b.value, 0);\r\n\r\n    const radius = this.size / 2 - this.thickness / 2;\r\n    const circumference = 2 * Math.PI * radius;\r\n    this._radius.set(radius);\r\n    this._circ.set(circumference);\r\n\r\n    // sin datos válidos\r\n    if (total <= 0) {\r\n      this._slices.set([]);\r\n      this._total.set(0);\r\n      return;\r\n    }\r\n\r\n    // paleta (rota si no alcanza)\r\n    const palette = [\r\n      '#2167FF', // Info-500\r\n      '#0A2893', // Info-800\r\n      '#40CAE9', // Primary-400\r\n      '#5892FF', // Info-400\r\n      '#08A6DB', // Primary-500\r\n      '#C4DFFF', // Info-200\r\n    ];\r\n    let pIndex = 0;\r\n    const base = cleaned.map(d => ({\r\n      ...d,\r\n      color: d.color ?? palette[pIndex++ % palette.length]\r\n    }));\r\n\r\n    // orden opcional\r\n    const dataOrdered = this.sort ? [...base].sort((a, b) => b.value - a.value) : base;\r\n\r\n    // geometría angular\r\n    const cx = this.size / 2;\r\n    const cy = this.size / 2;\r\n    const n = dataOrdered.length;\r\n    const gap = Math.max(this.gapDegrees, 0);\r\n    const usefulDeg = Math.max(360 - n * gap, 0); // evita negativos\r\n\r\n    let current = -90; // arranca a las 12 en punto\r\n    const slices: SliceComputed[] = dataOrdered.map(d => {\r\n      const fraction = d.value / total;\r\n      const sweep = usefulDeg * fraction;\r\n\r\n      // dejamos medio gap antes y después del arco\r\n      const startDeg = current + gap / 2;\r\n      const endDeg = startDeg + sweep;\r\n\r\n      const pathD = this.arcPath(cx, cy, radius, startDeg, endDeg);\r\n      const midDeg = startDeg + sweep / 2;\r\n      const mid = this.polarToCartesian(cx, cy, radius, midDeg);\r\n\r\n      // avanza el cursor angular para el siguiente slice\r\n      current = endDeg + gap / 2;\r\n\r\n      return {\r\n        ...d,\r\n        percentage: +(fraction * 100).toFixed(2),\r\n        pathD,\r\n        midX: mid.x,\r\n        midY: mid.y\r\n      };\r\n    });\r\n\r\n    this._slices.set(slices);\r\n    this._total.set(total);\r\n  }\r\n\r\n  // === helpers de geometría ===\r\n  private polarToCartesian(cx: number, cy: number, r: number, deg: number) {\r\n    const rad = (deg * Math.PI) / 180;\r\n    return { x: cx + r * Math.cos(rad), y: cy + r * Math.sin(rad) };\r\n  }\r\n\r\n  private arcPath(cx: number, cy: number, r: number, startDeg: number, endDeg: number): string {\r\n    const start = this.polarToCartesian(cx, cy, r, startDeg);\r\n    const end = this.polarToCartesian(cx, cy, r, endDeg);\r\n    const sweep = endDeg - startDeg;\r\n    const largeArcFlag = sweep > 180 ? 1 : 0;\r\n    // arco horario (sweep-flag=1)\r\n    return `M ${start.x} ${start.y} A ${r} ${r} 0 ${largeArcFlag} 1 ${end.x} ${end.y}`;\r\n  }\r\n\r\n  // eventos / utilidades\r\n  onSliceClick(s: SliceComputed) {\r\n    this.sliceClick.emit({ label: s.label, value: s.value, color: s.color });\r\n  }\r\n\r\n  get centerLabel(): string {\r\n    return `${this.centerText ?? this.total()}`;\r\n  }\r\n\r\n  getPopoverText(s: { label: string; value: number; percentage: number }): string {\r\n    return `${s.label}: ${s.value} (${s.percentage}%)`;\r\n  }\r\n\r\n  // accesibilidad: describe la dona\r\n  get ariaLabel(): string {\r\n    const parts = this.slices().map(s => `${s.label}: ${s.value} (${s.percentage}%)`);\r\n    return `Gráfica de dona. Total ${this.total()}. ${parts.join('. ')}`;\r\n  }\r\n}\r\n","<div class=\"donut-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n    <svg\r\n        class=\"donut\"\r\n        role=\"img\"\r\n        [attr.aria-label]=\"ariaLabel\"\r\n        [attr.width]=\"size\"\r\n        [attr.height]=\"size\"\r\n        [attr.viewBox]=\"viewBox()\">\r\n        <!-- pista -->\r\n        <circle\r\n        class=\"track\"\r\n        [attr.cx]=\"size/2\"\r\n        [attr.cy]=\"size/2\"\r\n        [attr.r]=\"radius()\"\r\n        [attr.stroke-width]=\"thickness\"\r\n        ></circle>\r\n\r\n        <!-- slices como PATH (cada uno es su arco real) -->\r\n        @for (s of slices(); track s.label) {\r\n        <path\r\n            class=\"slice\"\r\n            [attr.d]=\"s.pathD\"\r\n            fill=\"none\"\r\n            [attr.stroke]=\"s.color\"\r\n            [attr.stroke-width]=\"thickness\"\r\n            stroke-linecap=\"butt\"\r\n            [iusPopoverContent]=\"getPopoverText(s)\"\r\n            (click)=\"onSliceClick(s)\"\r\n            tabindex=\"0\"\r\n        >\r\n        </path>\r\n        }\r\n\r\n        <!-- centro -->\r\n        <g>\r\n            <text\r\n                [attr.x]=\"size/2\"\r\n                [attr.y]=\"size/2\"\r\n                text-anchor=\"middle\"\r\n                alignment-baseline=\"middle\"\r\n                class=\"center-text\">\r\n                {{ centerLabel }}\r\n            </text>\r\n        </g>\r\n    </svg>\r\n\r\n    @if (showLegend && slices().length) {\r\n    <div class=\"legend scrollable-small\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n      @for (s of slices(); track s.label) {\r\n        <div class=\"legend-item\">\r\n          <span class=\"dot\" [style.background]=\"s.color\"></span>\r\n          <span class=\"label caption-sm\" [iusPopoverContent]=\"s.value.toString() + ' - ' + s.label\">{{s.value}} - {{s.label}}</span>\r\n        </div>\r\n      }\r\n    </div>\r\n    }\r\n</div>\r\n"]}
137
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"charts-donut.component.js","sourceRoot":"","sources":["../../../../../projects/ius-design-components/src/lib/charts-donut/charts-donut.component.ts","../../../../../projects/ius-design-components/src/lib/charts-donut/charts-donut.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, computed, EventEmitter, Input, Output, signal, OnChanges } from '@angular/core';\r\nimport { CommonModule } from '@angular/common';\r\nimport { PopoverDirective } from '../directives/popover.directive';\r\n\r\nexport type DonutData = { label: string; value: number; color?: string };\r\n\r\ntype SliceComputed = DonutData & {\r\n  percentage: number;\r\n  pathD: string;   // path del arco\r\n  midX: number;    // punto medio del arco (por si se quiere anclar ahí)\r\n  midY: number;\r\n};\r\n\r\n@Component({\r\n  selector: 'ius-charts-donut',\r\n  standalone: true,\r\n  imports: [CommonModule, PopoverDirective],\r\n  templateUrl: './charts-donut.component.html',\r\n  styleUrls: ['./charts-donut.component.scss']\r\n})\r\nexport class ChartsDonutComponent implements OnChanges {\r\n  @Input() data: DonutData[] = [];\r\n  @Input() size = 180;                 // width/height del SVG\r\n  @Input() thickness = 18;             // grosor del anillo\r\n  @Input() gapDegrees = 0;             // separación entre segmentos (grados)\r\n  @Input() showLegend = true;\r\n  @Input() legendPosition: 'right' | 'bottom' = 'right';\r\n  @Input() centerText?: string | number; // sobrescribe el texto central\r\n  @Input() sort = false;               // ordena segmentos por valor (desc)\r\n\r\n  @Output() sliceClick = new EventEmitter<DonutData>();\r\n\r\n  // señales para recalcular\r\n  private _slices = signal<SliceComputed[]>([]);\r\n  private _total = signal(0);\r\n  private _circ = signal(0);\r\n  private _radius = signal(0);\r\n\r\n  readonly slices = computed(() => this._slices());\r\n  readonly total = computed(() => this._total());\r\n  readonly radius = computed(() => this._radius());\r\n  readonly circumference = computed(() => this._circ());\r\n  readonly viewBox = computed(() => `0 0 ${this.size} ${this.size}`);\r\n\r\n  ngOnChanges(): void {\r\n    const cleaned = (this.data ?? []).filter(d => Number(d?.value) > 0);\r\n    const total = cleaned.reduce((a, b) => a + b.value, 0);\r\n\r\n    const radius = this.size / 2 - this.thickness / 2;\r\n    const circumference = 2 * Math.PI * radius;\r\n    this._radius.set(radius);\r\n    this._circ.set(circumference);\r\n\r\n    // sin datos válidos\r\n    if (total <= 0) {\r\n      this._slices.set([]);\r\n      this._total.set(0);\r\n      return;\r\n    }\r\n\r\n    // paleta (rota si no alcanza)\r\n    const palette = [\r\n      '#2167FF', // Info-500\r\n      '#0A2893', // Info-800\r\n      '#40CAE9', // Primary-400\r\n      '#5892FF', // Info-400\r\n      '#08A6DB', // Primary-500\r\n      '#C4DFFF', // Info-200\r\n    ];\r\n    let pIndex = 0;\r\n    const base = cleaned.map(d => ({\r\n      ...d,\r\n      color: d.color ?? palette[pIndex++ % palette.length]\r\n    }));\r\n\r\n    // orden opcional\r\n    const dataOrdered = this.sort ? [...base].sort((a, b) => b.value - a.value) : base;\r\n\r\n    // geometría angular\r\n    const cx = this.size / 2;\r\n    const cy = this.size / 2;\r\n    const n = dataOrdered.length;\r\n    const gap = Math.max(this.gapDegrees, 0);\r\n    const usefulDeg = Math.max(360 - n * gap, 0); // evita negativos\r\n\r\n    let current = -90; // arranca a las 12 en punto\r\n    const slices: SliceComputed[] = dataOrdered.map(d => {\r\n      const fraction = d.value / total;\r\n      const sweep = usefulDeg * fraction;\r\n\r\n      // dejamos medio gap antes y después del arco\r\n      const startDeg = current + gap / 2;\r\n      const endDeg = startDeg + sweep;\r\n\r\n      const pathD = this.arcPath(cx, cy, radius, startDeg, endDeg);\r\n      const midDeg = startDeg + sweep / 2;\r\n      const mid = this.polarToCartesian(cx, cy, radius, midDeg);\r\n\r\n      // avanza el cursor angular para el siguiente slice\r\n      current = endDeg + gap / 2;\r\n\r\n      return {\r\n        ...d,\r\n        percentage: +(fraction * 100).toFixed(2),\r\n        pathD,\r\n        midX: mid.x,\r\n        midY: mid.y\r\n      };\r\n    });\r\n\r\n    this._slices.set(slices);\r\n    this._total.set(total);\r\n  }\r\n\r\n  // === helpers de geometría ===\r\n  private polarToCartesian(cx: number, cy: number, r: number, deg: number) {\r\n    const rad = (deg * Math.PI) / 180;\r\n    return { x: cx + r * Math.cos(rad), y: cy + r * Math.sin(rad) };\r\n  }\r\n\r\n  private arcPath(cx: number, cy: number, r: number, startDeg: number, endDeg: number): string {\r\n    const start = this.polarToCartesian(cx, cy, r, startDeg);\r\n    const end = this.polarToCartesian(cx, cy, r, endDeg);\r\n    const sweep = endDeg - startDeg;\r\n    const largeArcFlag = sweep > 180 ? 1 : 0;\r\n    // arco horario (sweep-flag=1)\r\n    return `M ${start.x} ${start.y} A ${r} ${r} 0 ${largeArcFlag} 1 ${end.x} ${end.y}`;\r\n  }\r\n\r\n  // eventos / utilidades\r\n  onSliceClick(s: SliceComputed) {\r\n    this.sliceClick.emit({ label: s.label, value: s.value, color: s.color });\r\n  }\r\n\r\n  get centerLabel(): string {\r\n    return `${this.centerText ?? this.total()}`;\r\n  }\r\n\r\n  getPopoverText(s: { label: string; value: number; percentage: number }): string {\r\n    return `${s.label}: ${s.value} (${s.percentage}%)`;\r\n  }\r\n\r\n  // accesibilidad: describe la dona\r\n  get ariaLabel(): string {\r\n    const parts = this.slices().map(s => `${s.label}: ${s.value} (${s.percentage}%)`);\r\n    return `Gráfica de dona. Total ${this.total()}. ${parts.join('. ')}`;\r\n  }\r\n}\r\n","<div class=\"donut-wrapper\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n    <svg\r\n        class=\"donut\"\r\n        role=\"img\"\r\n        [attr.aria-label]=\"ariaLabel\"\r\n        [attr.width]=\"size\"\r\n        [attr.height]=\"size\"\r\n        [attr.viewBox]=\"viewBox()\">\r\n        <!-- pista -->\r\n        <circle\r\n        class=\"track\"\r\n        [attr.cx]=\"size/2\"\r\n        [attr.cy]=\"size/2\"\r\n        [attr.r]=\"radius()\"\r\n        [attr.stroke-width]=\"thickness\"\r\n        ></circle>\r\n\r\n        <!-- slices como PATH (cada uno es su arco real) -->\r\n        @for (s of slices(); track s.label) {\r\n        <path\r\n            class=\"slice\"\r\n            [attr.d]=\"s.pathD\"\r\n            fill=\"none\"\r\n            [attr.stroke]=\"s.color\"\r\n            [attr.stroke-width]=\"thickness\"\r\n            stroke-linecap=\"butt\"\r\n            [iusPopoverContent]=\"getPopoverText(s)\"\r\n            (click)=\"onSliceClick(s)\"\r\n            tabindex=\"0\"\r\n        >\r\n        </path>\r\n        }\r\n\r\n        <!-- centro -->\r\n        <g>\r\n            <text\r\n                [attr.x]=\"size/2\"\r\n                [attr.y]=\"size/2\"\r\n                text-anchor=\"middle\"\r\n                alignment-baseline=\"middle\"\r\n                class=\"center-text\">\r\n                {{ centerLabel }}\r\n            </text>\r\n        </g>\r\n    </svg>\r\n\r\n    @if (showLegend && slices().length) {\r\n    <div class=\"legend scrollable-small\" [class.bottom]=\"legendPosition === 'bottom'\">\r\n      @for (s of slices(); track s.label) {\r\n        <div class=\"legend-item\">\r\n          <span class=\"dot\" [style.background]=\"s.color\"></span>\r\n          <span class=\"label caption-sm\" [iusPopoverContent]=\"s.value.toString() + ' - ' + s.label\">{{s.value}} - {{s.label}}</span>\r\n        </div>\r\n      }\r\n    </div>\r\n    }\r\n</div>\r\n"]}