raain-app 1.6.21 → 1.6.24

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 { ChangeDetectionStrategy, Component, Input, } from '@angular/core';
1
+ import { ChangeDetectionStrategy, Component, Input, ViewChild, } from '@angular/core';
2
2
  import { CartesianMeasureValue, TeamNode } from 'raain-model';
3
3
  import { DateRange, MapLatLng, TimeframeContainers } from 'raain-ui';
4
4
  import { mapDateRangeToString } from '../tools';
@@ -11,6 +11,7 @@ import * as i5 from "../raain-map/raain-map.component";
11
11
  import * as i6 from "../raain-compare-stack/raain-compare-stack.component";
12
12
  import * as i7 from "../raain-date-focus/raain-date-focus.component";
13
13
  import * as i8 from "../raain-date-dynamic/raain-date-dynamic.component";
14
+ import * as i9 from "../cumulative-selector/cumulative-selector.component";
14
15
  let TEST_DETECTION = 0;
15
16
  export class RaainDetailsComponent {
16
17
  constructor(storage, cdr) {
@@ -21,6 +22,7 @@ export class RaainDetailsComponent {
21
22
  this.showPixelMarkers = false;
22
23
  this.qualityIndicators = [];
23
24
  this.qualityIndicatorsLoading = false;
25
+ this.showCumulativeSelector = false;
24
26
  // Cached computed values (to avoid method calls in templates)
25
27
  this.percentOfComputations = 0;
26
28
  this.percentOfImages = 0;
@@ -256,14 +258,73 @@ export class RaainDetailsComponent {
256
258
  onTogglePixelMarkers() {
257
259
  // Toggle is bound to showPixelMarkers, raain-map handles marker display
258
260
  }
259
- async toggleCumulativeChanged(_$event) {
260
- this.storage.set('raain-toggleCumulative', this.toggleCumulative);
261
+ async onCumulativeToggleClick($event) {
262
+ console.log('[CumulativeToggle] click event', {
263
+ currentValue: this.toggleCumulative,
264
+ eventType: $event.type,
265
+ target: $event.target?.tagName,
266
+ });
267
+ $event.preventDefault();
268
+ $event.stopPropagation();
269
+ if (!this.toggleCumulative) {
270
+ console.log('[CumulativeToggle] showing popup, toggle stays OFF');
271
+ // Currently OFF, user wants to enable - show selector popup
272
+ this.showCumulativeSelector = true;
273
+ // Force reset the toggle visual state after ion-toggle's internal handler
274
+ setTimeout(() => {
275
+ if (this.cumulativeToggleRef) {
276
+ this.cumulativeToggleRef.checked = false;
277
+ console.log('[CumulativeToggle] force reset toggle to OFF');
278
+ }
279
+ }, 0);
280
+ this.cdr.markForCheck();
281
+ }
282
+ else {
283
+ console.log('[CumulativeToggle] turning OFF');
284
+ // Currently ON, user wants to disable - direct toggle off
285
+ this.toggleCumulative = false;
286
+ this.storage.set('raain-toggleCumulative', false);
287
+ this.dateShown = this.getDateBasedOnCumulativeMode(this.timeframeDates);
288
+ this.updateCumulativeDurationInMinutes();
289
+ if (this.toggleMap) {
290
+ this.updateRefreshManagerPeriod();
291
+ await this.refreshManager.refresh(false, this.toggleAdmin);
292
+ }
293
+ this.cdr.markForCheck();
294
+ }
295
+ console.log('[CumulativeToggle] after handler, toggleCumulative =', this.toggleCumulative);
296
+ }
297
+ async onCumulativePeriodSelected(selection) {
298
+ this.showCumulativeSelector = false;
299
+ // Update period to match selection
300
+ this.periodBegin = selection.periodBegin;
301
+ this.periodEnd = selection.periodEnd;
302
+ this.periodBeginAsString = RaainDetailsComponent.PeriodDisplay(this.periodBegin);
303
+ this.periodEndAsString = RaainDetailsComponent.PeriodDisplay(this.periodEnd);
304
+ // Calculate duration in hours
305
+ const durationMs = this.periodEnd.getTime() - this.periodBegin.getTime();
306
+ const durationHours = durationMs / RaainDetailsComponent.HOUR_MS;
307
+ this.periodDurationAsString = String(durationHours);
308
+ // Store cumulative period
309
+ this.storage.set('raain-periodBegin', this.periodBegin);
310
+ this.storage.set('raain-periodEnd', this.periodEnd);
311
+ this.storage.set('raain-periodDurationInHours', durationHours);
312
+ // Enable cumulative mode
313
+ this.toggleCumulative = true;
314
+ this.storage.set('raain-toggleCumulative', true);
261
315
  this.dateShown = this.getDateBasedOnCumulativeMode(this.timeframeDates);
262
316
  this.updateCumulativeDurationInMinutes();
317
+ // Refresh map
263
318
  if (this.toggleMap) {
264
- this.updateRefreshManagerPeriod(); // Update cumulative flag before refresh
319
+ this.updateRefreshManagerPeriod();
265
320
  await this.refreshManager.refresh(false, this.toggleAdmin);
266
321
  }
322
+ this.cdr.markForCheck();
323
+ }
324
+ onCumulativeSelectorCancelled() {
325
+ console.log('[CumulativeToggle] cancelled, toggleCumulative =', this.toggleCumulative);
326
+ this.showCumulativeSelector = false;
327
+ this.cdr.markForCheck();
267
328
  }
268
329
  updateCumulativeDurationInMinutes() {
269
330
  if (this.toggleCumulative) {
@@ -464,6 +525,8 @@ export class RaainDetailsComponent {
464
525
  const center = this.rainNode.getCenter();
465
526
  this.coordinates = new MapLatLng(center.lat, center.lng);
466
527
  this.teamNode = await this.profileService.getTeam(this.rainNode.getLink(TeamNode.TYPE).getId());
528
+ // Load all gauges linked to the rainNode on map
529
+ await this.compareManager.setGaugesInMap();
467
530
  if (this.periodBegin && this.periodEnd) {
468
531
  this.updateRefreshManagerPeriod();
469
532
  await this.refreshManager.refresh(false, this.toggleAdmin);
@@ -501,10 +564,10 @@ export class RaainDetailsComponent {
501
564
  RaainDetailsComponent.HOUR_MS = 60 * 60000;
502
565
  RaainDetailsComponent.DAY_MS = 24 * 60 * 60 * 1000;
503
566
  RaainDetailsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RaainDetailsComponent, deps: [{ token: i1.Storage }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
504
- RaainDetailsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: RaainDetailsComponent, selector: "raain-details", inputs: { toggleAdmin: "toggleAdmin", rainNode: "rainNode", compareManager: "compareManager", refreshManager: "refreshManager", profileService: "profileService", radarService: "radarService" }, usesOnChanges: true, ngImport: i0, template: "<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n <!-- Period selection section -->\n <ion-card class=\"period-card\">\n <ion-card-content>\n <div class=\"period-controls\">\n <div class=\"period-row\">\n\n <ion-button (click)=\"toggleHistory = !toggleHistory; onEnableCountHistoryTab(rainNode)\"\n fill=\"outline\">\n <ion-icon name=\"calendar-clear-outline\" slot=\"start\"></ion-icon>\n <ion-icon [name]=\"toggleHistory ? 'chevron-down' : 'chevron-forward'\" slot=\"end\"></ion-icon>\n </ion-button>\n\n <div class=\"period-start ion-hide-md-down\">\n <input (change)=\"onPeriodBeginChange($event)\"\n [disabled]=\"toggleCumulative\"\n [value]=\"periodBeginAsString\"\n class=\"datetime-input\"\n type=\"datetime-local\">\n </div>\n\n <div class=\"period-duration ion-hide-md-down\">\n <ion-select (ionDismiss)=\"onPeriodDurationChange($event)\"\n [(ngModel)]=\"periodDurationAsString\"\n class=\"duration-select\"\n id=\"periodDuration\"\n interface=\"popover\">\n <ion-select-option value=\"0.25\">15 minutes</ion-select-option>\n <ion-select-option value=\"0.5\">30 minutes</ion-select-option>\n <ion-select-option value=\"1\">1 hour</ion-select-option>\n <ion-select-option value=\"2\">2 hours</ion-select-option>\n <ion-select-option value=\"3\">3 hours</ion-select-option>\n <ion-select-option value=\"4\">4 hours</ion-select-option>\n <ion-select-option value=\"5\">5 hours</ion-select-option>\n <ion-select-option value=\"6\">6 hours</ion-select-option>\n <ion-select-option value=\"8\">8 hours</ion-select-option>\n <ion-select-option value=\"10\">10 hours</ion-select-option>\n <ion-select-option value=\"12\">12 hours</ion-select-option>\n <ion-select-option *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n </ion-select>\n </div>\n\n <div class=\"period-start ion-hide-md-down\">\n <input (change)=\"onPeriodEndChange($event)\"\n [disabled]=\"!toggleCumulative\"\n [value]=\"periodEndAsString\"\n class=\"datetime-input\"\n type=\"datetime-local\">\n </div>\n\n <div class=\"toggle-cumulative\">\n <ion-label [class.text-primary]=\"toggleCumulative\">\n {{ toggleCumulative ? 'Cumulative' : 'Granular' }}\n </ion-label>\n <ion-toggle (ionChange)=\"toggleCumulativeChanged($event)\"\n [(ngModel)]=\"toggleCumulative\"\n [checked]=\"toggleCumulative\">\n </ion-toggle>\n </div>\n </div>\n\n <!-- Hidden label for change detection (uncomment to debug)\n <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\n -->\n </div>\n\n <!-- Historical map section -->\n <div *ngIf=\"toggleHistory\" class=\"period-controls\">\n <raain-date-dynamic (changedDate)=\"onDateChangeInCount($event)\"\n [currentHeight]=\"300\"\n [fetchData]=\"fetchDataWrapper\"\n [points]=\"countPoints\">\n </raain-date-dynamic>\n </div>\n </ion-card-content>\n </ion-card>\n\n <!-- Map performance -->\n <ion-grid class=\"map-performance\">\n <ion-row id=\"progressAndRefresh\">\n <ion-col class=\"provider-selection\" size=\"12\" size-md=\"6\">\n <ion-button (click)=\"openQualityModal()\" class=\"quality-info-button\" fill=\"clear\">\n {{ refreshManager.rainComputationMapVersion }}\n <ion-icon name=\"help-circle-outline\" slot=\"end\"></ion-icon>\n </ion-button>\n <div *ngIf=\"availableProviders.length > 0 || availableTimeSteps.length > 0\" class=\"selection-row\">\n <ion-select (ionChange)=\"onProviderChanged($event)\"\n [value]=\"selectedProvider\"\n interface=\"popover\"\n label=\"Provider\"\n placeholder=\"Select Provider\">\n <ion-select-option *ngFor=\"let provider of availableProviders\" [value]=\"provider\">\n {{ provider }}\n </ion-select-option>\n </ion-select>\n\n <ion-select (ionChange)=\"onTimeStepChanged($event)\"\n [value]=\"selectedTimeStep\"\n interface=\"popover\"\n label=\"Time Step\"\n placeholder=\"Select Time Step\">\n <ion-select-option *ngFor=\"let step of availableTimeSteps\" [value]=\"step\">\n {{ step }} min\n </ion-select-option>\n </ion-select>\n </div>\n </ion-col>\n <ion-col class=\"ion-text-right\" size=\"12\" size-md=\"6\">\n <ion-label class=\"ion-margin-end\">\n <span *ngIf=\"percentOfComputations\">\n {{ percentOfComputations }}% Images\n <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n In Progress: {{ countsPeriod.progress }}...\n </i>\n </span>\n <span *ngIf=\"!percentOfComputations\">\n No image available\n </span>\n\n </ion-label>\n\n <ion-button (click)=\"refreshMap()\" [disabled]=\"refreshInProgress\" class=\"refresh-button\">\n Refresh Map\n </ion-button>\n </ion-col>\n </ion-row>\n\n <!-- status update row -->\n <ion-row>\n <!-- Progress col -->\n <ion-progress-bar\n [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n [style.visibility]=\"refreshInProgress ? 'visible' : 'hidden'\"\n [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n color=\"primary\">\n </ion-progress-bar>\n\n <!-- Error col -->\n <ion-col (click)=\"showFullError = !showFullError\" *ngIf=\"refreshManager.lastError\" class=\"error-row\"\n size=\"12\">\n <div class=\"error-content\">\n <ion-icon class=\"error-icon\" name=\"warning-outline\"></ion-icon>\n <span [class.expanded]=\"showFullError\" class=\"error-text\">\n {{ showFullError ? refreshManager.lastError : truncatedError }}\n </span>\n <ion-icon [name]=\"showFullError ? 'chevron-up' : 'chevron-down'\" class=\"error-caret\"></ion-icon>\n </div>\n </ion-col>\n </ion-row>\n </ion-grid>\n\n <!-- Map section -->\n <ion-card class=\"map-card\">\n <ion-card-content class=\"map-content\">\n <ion-grid>\n <ion-row *ngIf=\"toggleMap && percentOfImages\">\n <!-- Map component -->\n <ion-col class=\"map-column\" size-lg=\"7\" size-md=\"12\">\n <div class=\"map-container\">\n <raain-map #raainMapRef\n (changedDate)=\"onDateChangeInMap($event)\"\n (changedSum)=\"onSumChangeInMap($event)\"\n (selectedMarker)=\"onGaugeSelectInMap($event)\"\n [coordinates]=\"coordinates\"\n [cumulativeDurationInMinutes]=\"cumulativeDurationInMinutes\"\n [currentHeight]=\"500\"\n [defaultDate]=\"dateShown\"\n [markers]=\"{\n borders,\n gauges: compareManager.gaugesInMap,\n gaugesInCompare: compareManager.gaugesInCompare,\n selectedGauges: compareManager.selectedGauges,\n pixels: compareManager.selectedPixels,\n pixelsSolution: compareManager.pixelsSolutions?.length ? compareManager.pixelsSolutions[0] : [],\n speeds: compareManager.speeds\n }\"\n [showCumulative]=\"toggleCumulative\"\n [showVisiblePixelMarkers]=\"showPixelMarkers\"\n [sumFn]=\"refreshManager.sumFn\"\n [sumValues]=\"refreshManager.sumValues\"\n [timeframeContainers]=\"timeframeContainers\"\n [timeframeDates]=\"timeframeDates\">\n </raain-map>\n </div>\n\n <div class=\"data-column\">\n <!-- Technical details (collapsible for cleaner UI) -->\n <details class=\"technical-details\">\n <summary>Image Details</summary>\n <div class=\"details-content\">\n <div class=\"detail-row\">\n <span class=\"detail-label\">Computed:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Version:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapVersion }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Launched by:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapLaunchedBy }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Time spent:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapTimeSpentInMs }}\n ms</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Date:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapDate?.toISOString() }}\n | {{ refreshManager.rainComputationMapDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Water in the map:</span>\n <span class=\"detail-value\">{{ sumDetails }}</span>\n <ion-toggle\n (ionChange)=\"onTogglePixelMarkers()\"\n [(ngModel)]=\"showPixelMarkers\"\n style=\"margin-left: 8px; transform: scale(0.7);\">\n </ion-toggle>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Source DBZ min:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMin }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Source DBZ max:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMax }}</span>\n </div>\n </div>\n </details>\n </div>\n </ion-col>\n\n <!-- Data panel -->\n <ion-col *ngIf=\"!!compareManager.compareVersion\" class=\"data-column\" size-lg=\"5\" size-md=\"12\">\n <div class=\"data-panel\">\n <!-- Compare stack component -->\n <div class=\"compare-stack\">\n <raain-compare-stack\n (selectedPoint)=\"onGaugeSelectInCompare($event)\"\n [compareManager]=\"compareManager\"\n [cumulative]=\"toggleCumulative\">\n </raain-compare-stack>\n </div>\n\n <!-- Technical details (collapsible for cleaner UI) -->\n <details class=\"technical-details\">\n <summary>Compare Details</summary>\n <div class=\"details-content\">\n <div [ngClass]=\"{'warning': refreshManager.rainComputationMapDoneDate?.getTime() > compareManager.currentQualityDoneDate?.getTime() + 60000}\"\n class=\"detail-row\">\n <span class=\"detail-label\">Computed:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Version:</span>\n <span class=\"detail-value\">{{ compareManager.compareVersion }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Launched by:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityLaunchedBy }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Time spent:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityTimeSpentInMs }}\n ms</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Gauges:</span>\n <span class=\"detail-value\">{{ compareManager.gaugesInCompare.length }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Points:</span>\n <span class=\"detail-value\">{{ compareManager.globalComparePoints.length }}</span>\n </div>\n </div>\n </details>\n </div>\n </ion-col>\n </ion-row>\n <ion-row>\n <!-- Bottom progress bar -->\n <ion-progress-bar *ngIf=\"refreshInProgress\"\n [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n color=\"primary\">\n </ion-progress-bar>\n </ion-row>\n </ion-grid>\n </ion-card-content>\n </ion-card>\n\n <!-- Gauge values section -->\n <ion-card *ngIf=\"gaugeSelectedPoints.length && gaugeSelectedPoints[0].values.length\" class=\"gauge-card\">\n <ion-card-header>\n <ion-card-title>\n <ion-icon name=\"analytics-outline\"></ion-icon>\n Selected Gauge Data\n </ion-card-title>\n </ion-card-header>\n <ion-card-content>\n <raain-date-focus\n [currentHeight]=\"300\"\n [focusDate]=\"periodBegin\"\n [focusRange]=\"DateRange.DAY\"\n [points]=\"gaugeSelectedPoints\"\n [withoutAll]=\"true\">\n </raain-date-focus>\n </ion-card-content>\n </ion-card>\n\n <!-- Quality Performance Modal -->\n <div (click)=\"closeQualityModal()\" *ngIf=\"showQualityModal\" class=\"quality-modal-overlay\">\n <div (click)=\"$event.stopPropagation()\" class=\"quality-modal-content\">\n <div class=\"quality-modal-header\">\n <h2>Model Quality Performance</h2>\n <ion-button (click)=\"closeQualityModal()\" fill=\"clear\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </div>\n <div class=\"quality-modal-body\">\n <div *ngIf=\"qualityIndicatorsLoading\" class=\"quality-loading\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>Loading indicators...</span>\n </div>\n <div *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length === 0\" class=\"quality-empty\">\n No quality indicators available for this year.\n </div>\n <table *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length > 0\" class=\"quality-table\">\n <thead>\n <tr>\n <th>Model</th>\n <th>Compare</th>\n <th>Gauges</th>\n <th>Period</th>\n <th>Avg Quality</th>\n <th>Updated</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let indicator of qualityIndicators\">\n <td>{{ indicator.computingVersion }}</td>\n <td>{{ indicator.qualityVersion }}</td>\n <td>{{ indicator.provider }}<br>{{ indicator.timeStepInMinutes }} min</td>\n <td>{{ formatDate(indicator.startDate) }}<br>{{ formatDate(indicator.endDate) }}</td>\n <td>{{ indicator.averageQuality | number:'1.2-2' }}</td>\n <td>{{ formatDate(indicator.lastUpdatedAt) }}</td>\n </tr>\n </tbody>\n </table>\n </div>\n </div>\n </div>\n\n</div>\n", styles: [".raain-details-container{max-width:var(--app-max-width);margin:0 auto;padding:0 0 24px}.raain-details-card{width:100%;margin-bottom:20px}.raain-details-card ion-card-header{border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.raain-details-card ion-card-header ion-card-title{display:flex;align-items:center}.raain-details-card ion-card-header ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary)}.node-info-card{background-color:var(--ion-color-light)}.node-info-card .node-header{display:flex;align-items:center}.node-info-card .node-header .node-status{margin-right:16px}.node-info-card .node-header .node-status ion-icon{font-size:24px}.node-info-card .node-header .node-titles{flex:1}.node-info-card .node-header .node-titles ion-card-title{margin:0;font-size:1.4rem;font-weight:600;color:var(--ion-color-dark)}.node-info-card .node-header .node-titles ion-card-subtitle{padding-left:0;margin:4px 0 0;font-size:.9rem;color:var(--ion-color-medium)}.count-map-card,.period-card{background-color:var(--ion-color-light)}.period-card ion-card-title{display:flex;align-items:center}.period-card ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary);font-size:20px}.period-row{display:flex;flex-wrap:wrap;align-items:center;gap:16px;position:relative}.now-button{min-width:100px}#all-dates{display:flex;align-items:center;margin-left:auto}#all-dates .toggle-label{margin-right:8px;white-space:nowrap}.refresh-button ion-icon{margin-right:4px;color:var(--ion-color-light)}.provider-selection{display:flex;align-items:center}.quality-info-button{--padding-start: 8px;--padding-end: 8px;font-size:.85rem;color:var(--ion-color-medium)}.quality-info-button ion-icon{font-size:18px;margin-left:4px}.selection-row{display:flex;flex-direction:row;align-items:center;gap:12px}.selection-row ion-select{flex:0 0 auto;min-width:100px;margin-bottom:0;padding:4px 8px;border:1px solid rgba(var(--ion-color-medium-rgb),.3);border-radius:var(--ion-border-radius);background-color:var(--ion-color-light)}.period-start,.period-duration{display:flex;align-items:center}.toggle-cumulative{display:flex;align-items:center;gap:8px;margin-left:auto}.toggle-cumulative .text-primary{color:var(--ion-color-primary);font-weight:600}.hidden-label{display:none}.datetime-input,#periodDuration,.duration-select{padding:8px 12px;border:1px solid rgba(var(--ion-color-medium-rgb),.3);border-radius:var(--ion-border-radius);background-color:var(--ion-color-light);font-family:var(--ion-font-family)}.datetime-input:focus,#periodDuration:focus,.duration-select:focus{outline:none;border-color:var(--ion-color-primary)}#periodDuration,.duration-select{min-width:150px}.gauge-card{background-color:var(--ion-color-light)}.gauge-card ion-card-title{display:flex;align-items:center}.gauge-card ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary);font-size:20px}.error-row{cursor:pointer;background-color:rgba(var(--ion-color-danger-rgb),.1);border-left:3px solid var(--ion-color-danger);margin-top:8px;border-radius:4px;transition:background-color .2s ease}.error-row:hover{background-color:rgba(var(--ion-color-danger-rgb),.15)}.error-row .error-content{display:flex;align-items:flex-start;padding:8px 12px;gap:8px}.error-row .error-icon{color:var(--ion-color-danger);font-size:18px;flex-shrink:0;margin-top:2px}.error-row .error-text{flex:1;color:var(--ion-color-danger-shade);font-size:.9rem;word-break:break-word}.error-row .error-text.expanded{white-space:pre-wrap}.error-row .error-caret{color:var(--ion-color-danger);font-size:16px;flex-shrink:0;margin-top:2px}raain-compare-stack{width:100%;display:block}@media (max-width: 768px){.period-row{flex-direction:row;justify-content:space-between;align-items:center}#all-dates{margin-left:auto;padding-left:8px}#all-dates .toggle-label{font-size:.9rem}.map-header{flex-direction:row;justify-content:space-between;align-items:center;gap:16px}}.quality-modal-overlay{position:fixed;inset:0;background-color:#00000080;display:flex;align-items:center;justify-content:center;z-index:9999}.quality-modal-content{background-color:var(--ion-background-color, #fff);border-radius:12px;width:95%;max-width:900px;max-height:80vh;overflow:auto;box-shadow:0 4px 24px #0003}.quality-modal-header{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.quality-modal-header h2{margin:0;font-size:1.25rem;font-weight:600;color:var(--ion-color-dark)}.quality-modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.quality-modal-body{padding:20px}.quality-table{width:100%;border-collapse:collapse;margin-top:16px;table-layout:fixed}.quality-table th,.quality-table td{width:16.66%;padding:12px 16px;text-align:center;vertical-align:top;border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.quality-table th{background-color:rgba(var(--ion-color-primary-rgb),.1);font-weight:600;color:var(--ion-color-dark)}.quality-table td{color:var(--ion-color-dark-tint);font-size:.7rem}.quality-table tbody tr:hover{background-color:rgba(var(--ion-color-primary-rgb),.05)}.quality-loading{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:40px 20px;color:var(--ion-color-medium)}.quality-loading ion-spinner{margin-bottom:12px}.quality-empty{text-align:center;padding:40px 20px;color:var(--ion-color-medium);font-style:italic}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i4.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: i4.IonCard, selector: "ion-card", inputs: ["button", "color", "disabled", "download", "href", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: i4.IonCardContent, selector: "ion-card-content", inputs: ["mode"] }, { kind: "component", type: i4.IonCardHeader, selector: "ion-card-header", inputs: ["color", "mode", "translucent"] }, { kind: "component", type: i4.IonCardTitle, selector: "ion-card-title", inputs: ["color", "mode"] }, { kind: "component", type: i4.IonCol, selector: "ion-col", inputs: ["offset", "offsetLg", "offsetMd", "offsetSm", "offsetXl", "offsetXs", "pull", "pullLg", "pullMd", "pullSm", "pullXl", "pullXs", "push", "pushLg", "pushMd", "pushSm", "pushXl", "pushXs", "size", "sizeLg", "sizeMd", "sizeSm", "sizeXl", "sizeXs"] }, { kind: "component", type: i4.IonGrid, selector: "ion-grid", inputs: ["fixed"] }, { kind: "component", type: i4.IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: i4.IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: i4.IonProgressBar, selector: "ion-progress-bar", inputs: ["buffer", "color", "mode", "reversed", "type", "value"] }, { kind: "component", type: i4.IonRow, selector: "ion-row" }, { kind: "component", type: i4.IonSelect, selector: "ion-select", inputs: ["cancelText", "compareWith", "disabled", "interface", "interfaceOptions", "mode", "multiple", "name", "okText", "placeholder", "selectedText", "value"] }, { kind: "component", type: i4.IonSelectOption, selector: "ion-select-option", inputs: ["disabled", "value"] }, { kind: "component", type: i4.IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "component", type: i4.IonToggle, selector: "ion-toggle", inputs: ["checked", "color", "disabled", "mode", "name", "value"] }, { kind: "directive", type: i4.BooleanValueAccessor, selector: "ion-checkbox,ion-toggle" }, { kind: "directive", type: i4.SelectValueAccessor, selector: "ion-range, ion-select, ion-radio-group, ion-segment, ion-datetime" }, { kind: "component", type: i5.RaainMapComponent, selector: "raain-map", inputs: ["coordinates", "markers", "timeframeContainers", "autoplay", "showMarkers", "showSpeedMarkers", "showVisiblePixelMarkers", "showCumulative", "cumulativeDurationInMinutes", "currentHeight", "timeframeDates", "defaultDate", "sumValues", "sumFn"], outputs: ["selectedMarker", "changedDate", "changedSum"] }, { kind: "component", type: i6.RaainCompareStackComponent, selector: "raain-compare-stack", inputs: ["compareManager", "currentHeight", "cumulative"], outputs: ["selectedPoint"] }, { kind: "component", type: i7.RaainDateFocusComponent, selector: "raain-date-focus", inputs: ["points", "focusDate", "focusRange", "withoutAll", "currentHeight"] }, { kind: "component", type: i8.RaainDateDynamicComponent, selector: "raain-date-dynamic", inputs: ["points", "focusDate", "focusRange", "withoutAll", "currentHeight", "fetchData"], outputs: ["changedDate"] }, { kind: "pipe", type: i2.DecimalPipe, name: "number" }, { kind: "pipe", type: i2.DatePipe, name: "date" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
567
+ RaainDetailsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: RaainDetailsComponent, selector: "raain-details", inputs: { toggleAdmin: "toggleAdmin", rainNode: "rainNode", compareManager: "compareManager", refreshManager: "refreshManager", profileService: "profileService", radarService: "radarService" }, viewQueries: [{ propertyName: "cumulativeToggleRef", first: true, predicate: ["cumulativeToggle"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n <!-- Period selection section -->\n <ion-card class=\"period-card\">\n <ion-card-content>\n <div class=\"period-controls\">\n <div class=\"period-row\">\n\n <ion-button (click)=\"toggleHistory = !toggleHistory; onEnableCountHistoryTab(rainNode)\"\n fill=\"outline\">\n <ion-icon name=\"calendar-clear-outline\" slot=\"start\"></ion-icon>\n <ion-icon [name]=\"toggleHistory ? 'chevron-down' : 'chevron-forward'\" slot=\"end\"></ion-icon>\n </ion-button>\n\n <div class=\"period-start ion-hide-md-down\">\n <input (change)=\"onPeriodBeginChange($event)\"\n [disabled]=\"toggleCumulative\"\n [value]=\"periodBeginAsString\"\n class=\"datetime-input\"\n type=\"datetime-local\">\n </div>\n\n <div class=\"period-duration ion-hide-md-down\">\n <ion-select (ionDismiss)=\"onPeriodDurationChange($event)\"\n [(ngModel)]=\"periodDurationAsString\"\n [disabled]=\"toggleCumulative\"\n class=\"duration-select\"\n id=\"periodDuration\"\n interface=\"popover\">\n <ion-select-option value=\"0.25\">15 minutes</ion-select-option>\n <ion-select-option value=\"0.5\">30 minutes</ion-select-option>\n <ion-select-option value=\"1\">1 hour</ion-select-option>\n <ion-select-option value=\"2\">2 hours</ion-select-option>\n <ion-select-option value=\"3\">3 hours</ion-select-option>\n <ion-select-option value=\"4\">4 hours</ion-select-option>\n <ion-select-option value=\"5\">5 hours</ion-select-option>\n <ion-select-option value=\"6\">6 hours</ion-select-option>\n <ion-select-option value=\"8\">8 hours</ion-select-option>\n <ion-select-option value=\"10\">10 hours</ion-select-option>\n <ion-select-option value=\"12\">12 hours</ion-select-option>\n <ion-select-option *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n </ion-select>\n </div>\n\n <div class=\"period-start ion-hide-md-down\">\n <input (change)=\"onPeriodEndChange($event)\"\n [disabled]=\"toggleCumulative\"\n [value]=\"periodEndAsString\"\n class=\"datetime-input\"\n type=\"datetime-local\">\n </div>\n\n <div class=\"toggle-cumulative\" (click)=\"onCumulativeToggleClick($event)\">\n <ion-label [class.text-primary]=\"toggleCumulative\">\n {{ toggleCumulative ? 'Cumulative' : 'Granular' }}\n </ion-label>\n <ion-toggle #cumulativeToggle [checked]=\"toggleCumulative\">\n </ion-toggle>\n </div>\n </div>\n\n <!-- Hidden label for change detection (uncomment to debug)\n <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\n -->\n </div>\n\n <!-- Historical map section -->\n <div *ngIf=\"toggleHistory\" class=\"period-controls\">\n <raain-date-dynamic (changedDate)=\"onDateChangeInCount($event)\"\n [currentHeight]=\"300\"\n [fetchData]=\"fetchDataWrapper\"\n [points]=\"countPoints\">\n </raain-date-dynamic>\n </div>\n </ion-card-content>\n </ion-card>\n\n <!-- Map performance -->\n <ion-grid class=\"map-performance\">\n <ion-row id=\"progressAndRefresh\">\n <ion-col class=\"provider-selection\" size=\"12\" size-md=\"6\">\n <ion-button (click)=\"openQualityModal()\" class=\"quality-info-button\" fill=\"clear\">\n {{ refreshManager.rainComputationMapVersion }}\n <ion-icon name=\"help-circle-outline\" slot=\"end\"></ion-icon>\n </ion-button>\n <div *ngIf=\"availableProviders.length > 0 || availableTimeSteps.length > 0\" class=\"selection-row\">\n <ion-select (ionChange)=\"onProviderChanged($event)\"\n [value]=\"selectedProvider\"\n interface=\"popover\"\n label=\"Provider\"\n placeholder=\"Select Provider\">\n <ion-select-option *ngFor=\"let provider of availableProviders\" [value]=\"provider\">\n {{ provider }}\n </ion-select-option>\n </ion-select>\n\n <ion-select (ionChange)=\"onTimeStepChanged($event)\"\n [value]=\"selectedTimeStep\"\n interface=\"popover\"\n label=\"Time Step\"\n placeholder=\"Select Time Step\">\n <ion-select-option *ngFor=\"let step of availableTimeSteps\" [value]=\"step\">\n {{ step }} min\n </ion-select-option>\n </ion-select>\n </div>\n </ion-col>\n <ion-col class=\"ion-text-right\" size=\"12\" size-md=\"6\">\n <ion-label class=\"ion-margin-end\">\n <span *ngIf=\"percentOfComputations\">\n {{ percentOfComputations }}% Images\n <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n In Progress: {{ countsPeriod.progress }}...\n </i>\n </span>\n <span *ngIf=\"!percentOfComputations\">\n No image available\n </span>\n\n </ion-label>\n\n <ion-button (click)=\"refreshMap()\" [disabled]=\"refreshInProgress\" class=\"refresh-button\">\n Refresh Map\n </ion-button>\n </ion-col>\n </ion-row>\n\n <!-- status update row -->\n <ion-row>\n <!-- Progress col -->\n <ion-progress-bar\n [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n [style.visibility]=\"refreshInProgress ? 'visible' : 'hidden'\"\n [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n color=\"primary\">\n </ion-progress-bar>\n\n <!-- Error col -->\n <ion-col (click)=\"showFullError = !showFullError\" *ngIf=\"refreshManager.lastError\" class=\"error-row\"\n size=\"12\">\n <div class=\"error-content\">\n <ion-icon class=\"error-icon\" name=\"warning-outline\"></ion-icon>\n <span [class.expanded]=\"showFullError\" class=\"error-text\">\n {{ showFullError ? refreshManager.lastError : truncatedError }}\n </span>\n <ion-icon [name]=\"showFullError ? 'chevron-up' : 'chevron-down'\" class=\"error-caret\"></ion-icon>\n </div>\n </ion-col>\n </ion-row>\n </ion-grid>\n\n <!-- Map section -->\n <ion-card class=\"map-card\">\n <ion-card-content class=\"map-content\">\n <ion-grid>\n <ion-row *ngIf=\"toggleMap && percentOfImages\">\n <!-- Map component -->\n <ion-col class=\"map-column\" size-lg=\"7\" size-md=\"12\">\n <div class=\"map-container\">\n <raain-map #raainMapRef\n (changedDate)=\"onDateChangeInMap($event)\"\n (changedSum)=\"onSumChangeInMap($event)\"\n (selectedMarker)=\"onGaugeSelectInMap($event)\"\n [coordinates]=\"coordinates\"\n [cumulativeDurationInMinutes]=\"cumulativeDurationInMinutes\"\n [currentHeight]=\"500\"\n [defaultDate]=\"dateShown\"\n [markers]=\"{\n borders,\n gauges: compareManager.gaugesInMap,\n gaugesInCompare: compareManager.gaugesInCompare,\n selectedGauges: compareManager.selectedGauges,\n pixels: compareManager.selectedPixels,\n pixelsSolution: compareManager.pixelsSolutions?.length ? compareManager.pixelsSolutions[0] : [],\n speeds: compareManager.speeds\n }\"\n [showCumulative]=\"toggleCumulative\"\n [showVisiblePixelMarkers]=\"showPixelMarkers\"\n [sumFn]=\"refreshManager.sumFn\"\n [sumValues]=\"refreshManager.sumValues\"\n [timeframeContainers]=\"timeframeContainers\"\n [timeframeDates]=\"timeframeDates\">\n </raain-map>\n </div>\n\n <div class=\"data-column\">\n <!-- Technical details (collapsible for cleaner UI) -->\n <details class=\"technical-details\">\n <summary>Image Details</summary>\n <div class=\"details-content\">\n <div class=\"detail-row\">\n <span class=\"detail-label\">Computed:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Version:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapVersion }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Launched by:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapLaunchedBy }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Time spent:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapTimeSpentInMs }}\n ms</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Date:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapDate?.toISOString() }}\n | {{ refreshManager.rainComputationMapDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Water in the map:</span>\n <span class=\"detail-value\">{{ sumDetails }}</span>\n <ion-toggle\n (ionChange)=\"onTogglePixelMarkers()\"\n [(ngModel)]=\"showPixelMarkers\"\n style=\"margin-left: 8px; transform: scale(0.7);\">\n </ion-toggle>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Source DBZ min:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMin }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Source DBZ max:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMax }}</span>\n </div>\n </div>\n </details>\n </div>\n </ion-col>\n\n <!-- Data panel -->\n <ion-col *ngIf=\"!!compareManager.compareVersion\" class=\"data-column\" size-lg=\"5\" size-md=\"12\">\n <div class=\"data-panel\">\n <!-- Compare stack component -->\n <div class=\"compare-stack\">\n <raain-compare-stack\n (selectedPoint)=\"onGaugeSelectInCompare($event)\"\n [compareManager]=\"compareManager\"\n [cumulative]=\"toggleCumulative\">\n </raain-compare-stack>\n </div>\n\n <!-- Technical details (collapsible for cleaner UI) -->\n <details class=\"technical-details\">\n <summary>Compare Details</summary>\n <div class=\"details-content\">\n <div [ngClass]=\"{'warning': refreshManager.rainComputationMapDoneDate?.getTime() > compareManager.currentQualityDoneDate?.getTime() + 60000}\"\n class=\"detail-row\">\n <span class=\"detail-label\">Computed:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Version:</span>\n <span class=\"detail-value\">{{ compareManager.compareVersion }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Launched by:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityLaunchedBy }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Time spent:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityTimeSpentInMs }}\n ms</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Gauges:</span>\n <span class=\"detail-value\">{{ compareManager.gaugesInCompare.length }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Points:</span>\n <span class=\"detail-value\">{{ compareManager.globalComparePoints.length }}</span>\n </div>\n </div>\n </details>\n </div>\n </ion-col>\n </ion-row>\n <ion-row>\n <!-- Bottom progress bar -->\n <ion-progress-bar *ngIf=\"refreshInProgress\"\n [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n color=\"primary\">\n </ion-progress-bar>\n </ion-row>\n </ion-grid>\n </ion-card-content>\n </ion-card>\n\n <!-- Gauge values section -->\n <ion-card *ngIf=\"gaugeSelectedPoints.length && gaugeSelectedPoints[0].values.length\" class=\"gauge-card\">\n <ion-card-header>\n <ion-card-title>\n <ion-icon name=\"analytics-outline\"></ion-icon>\n Selected Gauge Data\n </ion-card-title>\n </ion-card-header>\n <ion-card-content>\n <raain-date-focus\n [currentHeight]=\"300\"\n [focusDate]=\"periodBegin\"\n [focusRange]=\"DateRange.DAY\"\n [points]=\"gaugeSelectedPoints\"\n [withoutAll]=\"true\">\n </raain-date-focus>\n </ion-card-content>\n </ion-card>\n\n <!-- Quality Performance Modal -->\n <div (click)=\"closeQualityModal()\" *ngIf=\"showQualityModal\" class=\"quality-modal-overlay\">\n <div (click)=\"$event.stopPropagation()\" class=\"quality-modal-content\">\n <div class=\"quality-modal-header\">\n <h2>Model Quality Performance</h2>\n <ion-button (click)=\"closeQualityModal()\" fill=\"clear\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </div>\n <div class=\"quality-modal-body\">\n <div *ngIf=\"qualityIndicatorsLoading\" class=\"quality-loading\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>Loading indicators...</span>\n </div>\n <div *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length === 0\" class=\"quality-empty\">\n No quality indicators available for this year.\n </div>\n <table *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length > 0\" class=\"quality-table\">\n <thead>\n <tr>\n <th>Model</th>\n <th>Compare</th>\n <th>Gauges</th>\n <th>Period</th>\n <th>Avg Quality</th>\n <th>Updated</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let indicator of qualityIndicators\">\n <td>{{ indicator.computingVersion }}</td>\n <td>{{ indicator.qualityVersion }}</td>\n <td>{{ indicator.provider }}<br>{{ indicator.timeStepInMinutes }} min</td>\n <td>{{ formatDate(indicator.startDate) }}<br>{{ formatDate(indicator.endDate) }}</td>\n <td>{{ indicator.averageQuality | number:'1.2-2' }}</td>\n <td>{{ formatDate(indicator.lastUpdatedAt) }}</td>\n </tr>\n </tbody>\n </table>\n </div>\n </div>\n </div>\n\n <!-- Cumulative Selector Modal -->\n <cumulative-selector *ngIf=\"showCumulativeSelector\"\n [rainId]=\"rainNode?.id\"\n [currentPeriodBegin]=\"periodBegin\"\n [currentPeriodEnd]=\"periodEnd\"\n [provider]=\"selectedProvider\"\n [timeStepInMinutes]=\"selectedTimeStep\"\n [isAdmin]=\"toggleAdmin\"\n (periodSelected)=\"onCumulativePeriodSelected($event)\"\n (cancelled)=\"onCumulativeSelectorCancelled()\">\n </cumulative-selector>\n\n</div>\n", styles: [".raain-details-container{max-width:var(--app-max-width);margin:0 auto;padding:0 0 24px}.raain-details-card{width:100%;margin-bottom:20px}.raain-details-card ion-card-header{border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.raain-details-card ion-card-header ion-card-title{display:flex;align-items:center}.raain-details-card ion-card-header ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary)}.node-info-card{background-color:var(--ion-color-light)}.node-info-card .node-header{display:flex;align-items:center}.node-info-card .node-header .node-status{margin-right:16px}.node-info-card .node-header .node-status ion-icon{font-size:24px}.node-info-card .node-header .node-titles{flex:1}.node-info-card .node-header .node-titles ion-card-title{margin:0;font-size:1.4rem;font-weight:600;color:var(--ion-color-dark)}.node-info-card .node-header .node-titles ion-card-subtitle{padding-left:0;margin:4px 0 0;font-size:.9rem;color:var(--ion-color-medium)}.count-map-card,.period-card{background-color:var(--ion-color-light)}.period-card ion-card-title{display:flex;align-items:center}.period-card ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary);font-size:20px}.period-row{display:flex;flex-wrap:wrap;align-items:center;gap:16px;position:relative}.now-button{min-width:100px}#all-dates{display:flex;align-items:center;margin-left:auto}#all-dates .toggle-label{margin-right:8px;white-space:nowrap}.refresh-button ion-icon{margin-right:4px;color:var(--ion-color-light)}.provider-selection{display:flex;align-items:center}.quality-info-button{--padding-start: 8px;--padding-end: 8px;font-size:.85rem;color:var(--ion-color-medium)}.quality-info-button ion-icon{font-size:18px;margin-left:4px}.selection-row{display:flex;flex-direction:row;align-items:center;gap:12px}.selection-row ion-select{flex:0 0 auto;min-width:100px;margin-bottom:0;padding:4px 8px;border:1px solid rgba(var(--ion-color-medium-rgb),.3);border-radius:var(--ion-border-radius);background-color:var(--ion-color-light)}.period-start,.period-duration{display:flex;align-items:center}.toggle-cumulative{display:flex;align-items:center;gap:8px;margin-left:auto}.toggle-cumulative .text-primary{color:var(--ion-color-primary);font-weight:600}.hidden-label{display:none}.datetime-input,#periodDuration,.duration-select{padding:8px 12px;border:1px solid rgba(var(--ion-color-medium-rgb),.3);border-radius:var(--ion-border-radius);background-color:var(--ion-color-light);font-family:var(--ion-font-family)}.datetime-input:focus,#periodDuration:focus,.duration-select:focus{outline:none;border-color:var(--ion-color-primary)}#periodDuration,.duration-select{min-width:150px}.gauge-card{background-color:var(--ion-color-light)}.gauge-card ion-card-title{display:flex;align-items:center}.gauge-card ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary);font-size:20px}.error-row{cursor:pointer;background-color:rgba(var(--ion-color-danger-rgb),.1);border-left:3px solid var(--ion-color-danger);margin-top:8px;border-radius:4px;transition:background-color .2s ease}.error-row:hover{background-color:rgba(var(--ion-color-danger-rgb),.15)}.error-row .error-content{display:flex;align-items:flex-start;padding:8px 12px;gap:8px}.error-row .error-icon{color:var(--ion-color-danger);font-size:18px;flex-shrink:0;margin-top:2px}.error-row .error-text{flex:1;color:var(--ion-color-danger-shade);font-size:.9rem;word-break:break-word}.error-row .error-text.expanded{white-space:pre-wrap}.error-row .error-caret{color:var(--ion-color-danger);font-size:16px;flex-shrink:0;margin-top:2px}raain-compare-stack{width:100%;display:block}@media (max-width: 768px){.period-row{flex-direction:row;justify-content:space-between;align-items:center}#all-dates{margin-left:auto;padding-left:8px}#all-dates .toggle-label{font-size:.9rem}.map-header{flex-direction:row;justify-content:space-between;align-items:center;gap:16px}}.quality-modal-overlay{position:fixed;inset:0;background-color:#00000080;display:flex;align-items:center;justify-content:center;z-index:9999}.quality-modal-content{background-color:var(--ion-background-color, #fff);border-radius:12px;width:95%;max-width:900px;max-height:80vh;overflow:auto;box-shadow:0 4px 24px #0003}.quality-modal-header{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.quality-modal-header h2{margin:0;font-size:1.25rem;font-weight:600;color:var(--ion-color-dark)}.quality-modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.quality-modal-body{padding:20px}.quality-table{width:100%;border-collapse:collapse;margin-top:16px;table-layout:fixed}.quality-table th,.quality-table td{width:16.66%;padding:12px 16px;text-align:center;vertical-align:top;border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.quality-table th{background-color:rgba(var(--ion-color-primary-rgb),.1);font-weight:600;color:var(--ion-color-dark)}.quality-table td{color:var(--ion-color-dark-tint);font-size:.7rem}.quality-table tbody tr:hover{background-color:rgba(var(--ion-color-primary-rgb),.05)}.quality-loading{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:40px 20px;color:var(--ion-color-medium)}.quality-loading ion-spinner{margin-bottom:12px}.quality-empty{text-align:center;padding:40px 20px;color:var(--ion-color-medium);font-style:italic}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i4.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: i4.IonCard, selector: "ion-card", inputs: ["button", "color", "disabled", "download", "href", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: i4.IonCardContent, selector: "ion-card-content", inputs: ["mode"] }, { kind: "component", type: i4.IonCardHeader, selector: "ion-card-header", inputs: ["color", "mode", "translucent"] }, { kind: "component", type: i4.IonCardTitle, selector: "ion-card-title", inputs: ["color", "mode"] }, { kind: "component", type: i4.IonCol, selector: "ion-col", inputs: ["offset", "offsetLg", "offsetMd", "offsetSm", "offsetXl", "offsetXs", "pull", "pullLg", "pullMd", "pullSm", "pullXl", "pullXs", "push", "pushLg", "pushMd", "pushSm", "pushXl", "pushXs", "size", "sizeLg", "sizeMd", "sizeSm", "sizeXl", "sizeXs"] }, { kind: "component", type: i4.IonGrid, selector: "ion-grid", inputs: ["fixed"] }, { kind: "component", type: i4.IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: i4.IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: i4.IonProgressBar, selector: "ion-progress-bar", inputs: ["buffer", "color", "mode", "reversed", "type", "value"] }, { kind: "component", type: i4.IonRow, selector: "ion-row" }, { kind: "component", type: i4.IonSelect, selector: "ion-select", inputs: ["cancelText", "compareWith", "disabled", "interface", "interfaceOptions", "mode", "multiple", "name", "okText", "placeholder", "selectedText", "value"] }, { kind: "component", type: i4.IonSelectOption, selector: "ion-select-option", inputs: ["disabled", "value"] }, { kind: "component", type: i4.IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "component", type: i4.IonToggle, selector: "ion-toggle", inputs: ["checked", "color", "disabled", "mode", "name", "value"] }, { kind: "directive", type: i4.BooleanValueAccessor, selector: "ion-checkbox,ion-toggle" }, { kind: "directive", type: i4.SelectValueAccessor, selector: "ion-range, ion-select, ion-radio-group, ion-segment, ion-datetime" }, { kind: "component", type: i5.RaainMapComponent, selector: "raain-map", inputs: ["coordinates", "markers", "timeframeContainers", "autoplay", "showMarkers", "showSpeedMarkers", "showVisiblePixelMarkers", "showCumulative", "cumulativeDurationInMinutes", "currentHeight", "timeframeDates", "defaultDate", "sumValues", "sumFn"], outputs: ["selectedMarker", "changedDate", "changedSum"] }, { kind: "component", type: i6.RaainCompareStackComponent, selector: "raain-compare-stack", inputs: ["compareManager", "currentHeight", "cumulative"], outputs: ["selectedPoint"] }, { kind: "component", type: i7.RaainDateFocusComponent, selector: "raain-date-focus", inputs: ["points", "focusDate", "focusRange", "withoutAll", "currentHeight"] }, { kind: "component", type: i8.RaainDateDynamicComponent, selector: "raain-date-dynamic", inputs: ["points", "focusDate", "focusRange", "withoutAll", "currentHeight", "fetchData"], outputs: ["changedDate"] }, { kind: "component", type: i9.CumulativeSelectorComponent, selector: "cumulative-selector", inputs: ["rainId", "currentPeriodBegin", "currentPeriodEnd", "provider", "timeStepInMinutes", "isAdmin"], outputs: ["periodSelected", "cancelled"] }, { kind: "pipe", type: i2.DecimalPipe, name: "number" }, { kind: "pipe", type: i2.DatePipe, name: "date" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
505
568
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RaainDetailsComponent, decorators: [{
506
569
  type: Component,
507
- args: [{ selector: 'raain-details', changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n <!-- Period selection section -->\n <ion-card class=\"period-card\">\n <ion-card-content>\n <div class=\"period-controls\">\n <div class=\"period-row\">\n\n <ion-button (click)=\"toggleHistory = !toggleHistory; onEnableCountHistoryTab(rainNode)\"\n fill=\"outline\">\n <ion-icon name=\"calendar-clear-outline\" slot=\"start\"></ion-icon>\n <ion-icon [name]=\"toggleHistory ? 'chevron-down' : 'chevron-forward'\" slot=\"end\"></ion-icon>\n </ion-button>\n\n <div class=\"period-start ion-hide-md-down\">\n <input (change)=\"onPeriodBeginChange($event)\"\n [disabled]=\"toggleCumulative\"\n [value]=\"periodBeginAsString\"\n class=\"datetime-input\"\n type=\"datetime-local\">\n </div>\n\n <div class=\"period-duration ion-hide-md-down\">\n <ion-select (ionDismiss)=\"onPeriodDurationChange($event)\"\n [(ngModel)]=\"periodDurationAsString\"\n class=\"duration-select\"\n id=\"periodDuration\"\n interface=\"popover\">\n <ion-select-option value=\"0.25\">15 minutes</ion-select-option>\n <ion-select-option value=\"0.5\">30 minutes</ion-select-option>\n <ion-select-option value=\"1\">1 hour</ion-select-option>\n <ion-select-option value=\"2\">2 hours</ion-select-option>\n <ion-select-option value=\"3\">3 hours</ion-select-option>\n <ion-select-option value=\"4\">4 hours</ion-select-option>\n <ion-select-option value=\"5\">5 hours</ion-select-option>\n <ion-select-option value=\"6\">6 hours</ion-select-option>\n <ion-select-option value=\"8\">8 hours</ion-select-option>\n <ion-select-option value=\"10\">10 hours</ion-select-option>\n <ion-select-option value=\"12\">12 hours</ion-select-option>\n <ion-select-option *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n </ion-select>\n </div>\n\n <div class=\"period-start ion-hide-md-down\">\n <input (change)=\"onPeriodEndChange($event)\"\n [disabled]=\"!toggleCumulative\"\n [value]=\"periodEndAsString\"\n class=\"datetime-input\"\n type=\"datetime-local\">\n </div>\n\n <div class=\"toggle-cumulative\">\n <ion-label [class.text-primary]=\"toggleCumulative\">\n {{ toggleCumulative ? 'Cumulative' : 'Granular' }}\n </ion-label>\n <ion-toggle (ionChange)=\"toggleCumulativeChanged($event)\"\n [(ngModel)]=\"toggleCumulative\"\n [checked]=\"toggleCumulative\">\n </ion-toggle>\n </div>\n </div>\n\n <!-- Hidden label for change detection (uncomment to debug)\n <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\n -->\n </div>\n\n <!-- Historical map section -->\n <div *ngIf=\"toggleHistory\" class=\"period-controls\">\n <raain-date-dynamic (changedDate)=\"onDateChangeInCount($event)\"\n [currentHeight]=\"300\"\n [fetchData]=\"fetchDataWrapper\"\n [points]=\"countPoints\">\n </raain-date-dynamic>\n </div>\n </ion-card-content>\n </ion-card>\n\n <!-- Map performance -->\n <ion-grid class=\"map-performance\">\n <ion-row id=\"progressAndRefresh\">\n <ion-col class=\"provider-selection\" size=\"12\" size-md=\"6\">\n <ion-button (click)=\"openQualityModal()\" class=\"quality-info-button\" fill=\"clear\">\n {{ refreshManager.rainComputationMapVersion }}\n <ion-icon name=\"help-circle-outline\" slot=\"end\"></ion-icon>\n </ion-button>\n <div *ngIf=\"availableProviders.length > 0 || availableTimeSteps.length > 0\" class=\"selection-row\">\n <ion-select (ionChange)=\"onProviderChanged($event)\"\n [value]=\"selectedProvider\"\n interface=\"popover\"\n label=\"Provider\"\n placeholder=\"Select Provider\">\n <ion-select-option *ngFor=\"let provider of availableProviders\" [value]=\"provider\">\n {{ provider }}\n </ion-select-option>\n </ion-select>\n\n <ion-select (ionChange)=\"onTimeStepChanged($event)\"\n [value]=\"selectedTimeStep\"\n interface=\"popover\"\n label=\"Time Step\"\n placeholder=\"Select Time Step\">\n <ion-select-option *ngFor=\"let step of availableTimeSteps\" [value]=\"step\">\n {{ step }} min\n </ion-select-option>\n </ion-select>\n </div>\n </ion-col>\n <ion-col class=\"ion-text-right\" size=\"12\" size-md=\"6\">\n <ion-label class=\"ion-margin-end\">\n <span *ngIf=\"percentOfComputations\">\n {{ percentOfComputations }}% Images\n <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n In Progress: {{ countsPeriod.progress }}...\n </i>\n </span>\n <span *ngIf=\"!percentOfComputations\">\n No image available\n </span>\n\n </ion-label>\n\n <ion-button (click)=\"refreshMap()\" [disabled]=\"refreshInProgress\" class=\"refresh-button\">\n Refresh Map\n </ion-button>\n </ion-col>\n </ion-row>\n\n <!-- status update row -->\n <ion-row>\n <!-- Progress col -->\n <ion-progress-bar\n [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n [style.visibility]=\"refreshInProgress ? 'visible' : 'hidden'\"\n [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n color=\"primary\">\n </ion-progress-bar>\n\n <!-- Error col -->\n <ion-col (click)=\"showFullError = !showFullError\" *ngIf=\"refreshManager.lastError\" class=\"error-row\"\n size=\"12\">\n <div class=\"error-content\">\n <ion-icon class=\"error-icon\" name=\"warning-outline\"></ion-icon>\n <span [class.expanded]=\"showFullError\" class=\"error-text\">\n {{ showFullError ? refreshManager.lastError : truncatedError }}\n </span>\n <ion-icon [name]=\"showFullError ? 'chevron-up' : 'chevron-down'\" class=\"error-caret\"></ion-icon>\n </div>\n </ion-col>\n </ion-row>\n </ion-grid>\n\n <!-- Map section -->\n <ion-card class=\"map-card\">\n <ion-card-content class=\"map-content\">\n <ion-grid>\n <ion-row *ngIf=\"toggleMap && percentOfImages\">\n <!-- Map component -->\n <ion-col class=\"map-column\" size-lg=\"7\" size-md=\"12\">\n <div class=\"map-container\">\n <raain-map #raainMapRef\n (changedDate)=\"onDateChangeInMap($event)\"\n (changedSum)=\"onSumChangeInMap($event)\"\n (selectedMarker)=\"onGaugeSelectInMap($event)\"\n [coordinates]=\"coordinates\"\n [cumulativeDurationInMinutes]=\"cumulativeDurationInMinutes\"\n [currentHeight]=\"500\"\n [defaultDate]=\"dateShown\"\n [markers]=\"{\n borders,\n gauges: compareManager.gaugesInMap,\n gaugesInCompare: compareManager.gaugesInCompare,\n selectedGauges: compareManager.selectedGauges,\n pixels: compareManager.selectedPixels,\n pixelsSolution: compareManager.pixelsSolutions?.length ? compareManager.pixelsSolutions[0] : [],\n speeds: compareManager.speeds\n }\"\n [showCumulative]=\"toggleCumulative\"\n [showVisiblePixelMarkers]=\"showPixelMarkers\"\n [sumFn]=\"refreshManager.sumFn\"\n [sumValues]=\"refreshManager.sumValues\"\n [timeframeContainers]=\"timeframeContainers\"\n [timeframeDates]=\"timeframeDates\">\n </raain-map>\n </div>\n\n <div class=\"data-column\">\n <!-- Technical details (collapsible for cleaner UI) -->\n <details class=\"technical-details\">\n <summary>Image Details</summary>\n <div class=\"details-content\">\n <div class=\"detail-row\">\n <span class=\"detail-label\">Computed:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Version:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapVersion }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Launched by:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapLaunchedBy }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Time spent:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapTimeSpentInMs }}\n ms</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Date:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapDate?.toISOString() }}\n | {{ refreshManager.rainComputationMapDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Water in the map:</span>\n <span class=\"detail-value\">{{ sumDetails }}</span>\n <ion-toggle\n (ionChange)=\"onTogglePixelMarkers()\"\n [(ngModel)]=\"showPixelMarkers\"\n style=\"margin-left: 8px; transform: scale(0.7);\">\n </ion-toggle>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Source DBZ min:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMin }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Source DBZ max:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMax }}</span>\n </div>\n </div>\n </details>\n </div>\n </ion-col>\n\n <!-- Data panel -->\n <ion-col *ngIf=\"!!compareManager.compareVersion\" class=\"data-column\" size-lg=\"5\" size-md=\"12\">\n <div class=\"data-panel\">\n <!-- Compare stack component -->\n <div class=\"compare-stack\">\n <raain-compare-stack\n (selectedPoint)=\"onGaugeSelectInCompare($event)\"\n [compareManager]=\"compareManager\"\n [cumulative]=\"toggleCumulative\">\n </raain-compare-stack>\n </div>\n\n <!-- Technical details (collapsible for cleaner UI) -->\n <details class=\"technical-details\">\n <summary>Compare Details</summary>\n <div class=\"details-content\">\n <div [ngClass]=\"{'warning': refreshManager.rainComputationMapDoneDate?.getTime() > compareManager.currentQualityDoneDate?.getTime() + 60000}\"\n class=\"detail-row\">\n <span class=\"detail-label\">Computed:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Version:</span>\n <span class=\"detail-value\">{{ compareManager.compareVersion }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Launched by:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityLaunchedBy }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Time spent:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityTimeSpentInMs }}\n ms</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Gauges:</span>\n <span class=\"detail-value\">{{ compareManager.gaugesInCompare.length }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Points:</span>\n <span class=\"detail-value\">{{ compareManager.globalComparePoints.length }}</span>\n </div>\n </div>\n </details>\n </div>\n </ion-col>\n </ion-row>\n <ion-row>\n <!-- Bottom progress bar -->\n <ion-progress-bar *ngIf=\"refreshInProgress\"\n [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n color=\"primary\">\n </ion-progress-bar>\n </ion-row>\n </ion-grid>\n </ion-card-content>\n </ion-card>\n\n <!-- Gauge values section -->\n <ion-card *ngIf=\"gaugeSelectedPoints.length && gaugeSelectedPoints[0].values.length\" class=\"gauge-card\">\n <ion-card-header>\n <ion-card-title>\n <ion-icon name=\"analytics-outline\"></ion-icon>\n Selected Gauge Data\n </ion-card-title>\n </ion-card-header>\n <ion-card-content>\n <raain-date-focus\n [currentHeight]=\"300\"\n [focusDate]=\"periodBegin\"\n [focusRange]=\"DateRange.DAY\"\n [points]=\"gaugeSelectedPoints\"\n [withoutAll]=\"true\">\n </raain-date-focus>\n </ion-card-content>\n </ion-card>\n\n <!-- Quality Performance Modal -->\n <div (click)=\"closeQualityModal()\" *ngIf=\"showQualityModal\" class=\"quality-modal-overlay\">\n <div (click)=\"$event.stopPropagation()\" class=\"quality-modal-content\">\n <div class=\"quality-modal-header\">\n <h2>Model Quality Performance</h2>\n <ion-button (click)=\"closeQualityModal()\" fill=\"clear\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </div>\n <div class=\"quality-modal-body\">\n <div *ngIf=\"qualityIndicatorsLoading\" class=\"quality-loading\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>Loading indicators...</span>\n </div>\n <div *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length === 0\" class=\"quality-empty\">\n No quality indicators available for this year.\n </div>\n <table *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length > 0\" class=\"quality-table\">\n <thead>\n <tr>\n <th>Model</th>\n <th>Compare</th>\n <th>Gauges</th>\n <th>Period</th>\n <th>Avg Quality</th>\n <th>Updated</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let indicator of qualityIndicators\">\n <td>{{ indicator.computingVersion }}</td>\n <td>{{ indicator.qualityVersion }}</td>\n <td>{{ indicator.provider }}<br>{{ indicator.timeStepInMinutes }} min</td>\n <td>{{ formatDate(indicator.startDate) }}<br>{{ formatDate(indicator.endDate) }}</td>\n <td>{{ indicator.averageQuality | number:'1.2-2' }}</td>\n <td>{{ formatDate(indicator.lastUpdatedAt) }}</td>\n </tr>\n </tbody>\n </table>\n </div>\n </div>\n </div>\n\n</div>\n", styles: [".raain-details-container{max-width:var(--app-max-width);margin:0 auto;padding:0 0 24px}.raain-details-card{width:100%;margin-bottom:20px}.raain-details-card ion-card-header{border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.raain-details-card ion-card-header ion-card-title{display:flex;align-items:center}.raain-details-card ion-card-header ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary)}.node-info-card{background-color:var(--ion-color-light)}.node-info-card .node-header{display:flex;align-items:center}.node-info-card .node-header .node-status{margin-right:16px}.node-info-card .node-header .node-status ion-icon{font-size:24px}.node-info-card .node-header .node-titles{flex:1}.node-info-card .node-header .node-titles ion-card-title{margin:0;font-size:1.4rem;font-weight:600;color:var(--ion-color-dark)}.node-info-card .node-header .node-titles ion-card-subtitle{padding-left:0;margin:4px 0 0;font-size:.9rem;color:var(--ion-color-medium)}.count-map-card,.period-card{background-color:var(--ion-color-light)}.period-card ion-card-title{display:flex;align-items:center}.period-card ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary);font-size:20px}.period-row{display:flex;flex-wrap:wrap;align-items:center;gap:16px;position:relative}.now-button{min-width:100px}#all-dates{display:flex;align-items:center;margin-left:auto}#all-dates .toggle-label{margin-right:8px;white-space:nowrap}.refresh-button ion-icon{margin-right:4px;color:var(--ion-color-light)}.provider-selection{display:flex;align-items:center}.quality-info-button{--padding-start: 8px;--padding-end: 8px;font-size:.85rem;color:var(--ion-color-medium)}.quality-info-button ion-icon{font-size:18px;margin-left:4px}.selection-row{display:flex;flex-direction:row;align-items:center;gap:12px}.selection-row ion-select{flex:0 0 auto;min-width:100px;margin-bottom:0;padding:4px 8px;border:1px solid rgba(var(--ion-color-medium-rgb),.3);border-radius:var(--ion-border-radius);background-color:var(--ion-color-light)}.period-start,.period-duration{display:flex;align-items:center}.toggle-cumulative{display:flex;align-items:center;gap:8px;margin-left:auto}.toggle-cumulative .text-primary{color:var(--ion-color-primary);font-weight:600}.hidden-label{display:none}.datetime-input,#periodDuration,.duration-select{padding:8px 12px;border:1px solid rgba(var(--ion-color-medium-rgb),.3);border-radius:var(--ion-border-radius);background-color:var(--ion-color-light);font-family:var(--ion-font-family)}.datetime-input:focus,#periodDuration:focus,.duration-select:focus{outline:none;border-color:var(--ion-color-primary)}#periodDuration,.duration-select{min-width:150px}.gauge-card{background-color:var(--ion-color-light)}.gauge-card ion-card-title{display:flex;align-items:center}.gauge-card ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary);font-size:20px}.error-row{cursor:pointer;background-color:rgba(var(--ion-color-danger-rgb),.1);border-left:3px solid var(--ion-color-danger);margin-top:8px;border-radius:4px;transition:background-color .2s ease}.error-row:hover{background-color:rgba(var(--ion-color-danger-rgb),.15)}.error-row .error-content{display:flex;align-items:flex-start;padding:8px 12px;gap:8px}.error-row .error-icon{color:var(--ion-color-danger);font-size:18px;flex-shrink:0;margin-top:2px}.error-row .error-text{flex:1;color:var(--ion-color-danger-shade);font-size:.9rem;word-break:break-word}.error-row .error-text.expanded{white-space:pre-wrap}.error-row .error-caret{color:var(--ion-color-danger);font-size:16px;flex-shrink:0;margin-top:2px}raain-compare-stack{width:100%;display:block}@media (max-width: 768px){.period-row{flex-direction:row;justify-content:space-between;align-items:center}#all-dates{margin-left:auto;padding-left:8px}#all-dates .toggle-label{font-size:.9rem}.map-header{flex-direction:row;justify-content:space-between;align-items:center;gap:16px}}.quality-modal-overlay{position:fixed;inset:0;background-color:#00000080;display:flex;align-items:center;justify-content:center;z-index:9999}.quality-modal-content{background-color:var(--ion-background-color, #fff);border-radius:12px;width:95%;max-width:900px;max-height:80vh;overflow:auto;box-shadow:0 4px 24px #0003}.quality-modal-header{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.quality-modal-header h2{margin:0;font-size:1.25rem;font-weight:600;color:var(--ion-color-dark)}.quality-modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.quality-modal-body{padding:20px}.quality-table{width:100%;border-collapse:collapse;margin-top:16px;table-layout:fixed}.quality-table th,.quality-table td{width:16.66%;padding:12px 16px;text-align:center;vertical-align:top;border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.quality-table th{background-color:rgba(var(--ion-color-primary-rgb),.1);font-weight:600;color:var(--ion-color-dark)}.quality-table td{color:var(--ion-color-dark-tint);font-size:.7rem}.quality-table tbody tr:hover{background-color:rgba(var(--ion-color-primary-rgb),.05)}.quality-loading{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:40px 20px;color:var(--ion-color-medium)}.quality-loading ion-spinner{margin-bottom:12px}.quality-empty{text-align:center;padding:40px 20px;color:var(--ion-color-medium);font-style:italic}\n"] }]
570
+ args: [{ selector: 'raain-details', changeDetection: ChangeDetectionStrategy.OnPush, template: "<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n <!-- Period selection section -->\n <ion-card class=\"period-card\">\n <ion-card-content>\n <div class=\"period-controls\">\n <div class=\"period-row\">\n\n <ion-button (click)=\"toggleHistory = !toggleHistory; onEnableCountHistoryTab(rainNode)\"\n fill=\"outline\">\n <ion-icon name=\"calendar-clear-outline\" slot=\"start\"></ion-icon>\n <ion-icon [name]=\"toggleHistory ? 'chevron-down' : 'chevron-forward'\" slot=\"end\"></ion-icon>\n </ion-button>\n\n <div class=\"period-start ion-hide-md-down\">\n <input (change)=\"onPeriodBeginChange($event)\"\n [disabled]=\"toggleCumulative\"\n [value]=\"periodBeginAsString\"\n class=\"datetime-input\"\n type=\"datetime-local\">\n </div>\n\n <div class=\"period-duration ion-hide-md-down\">\n <ion-select (ionDismiss)=\"onPeriodDurationChange($event)\"\n [(ngModel)]=\"periodDurationAsString\"\n [disabled]=\"toggleCumulative\"\n class=\"duration-select\"\n id=\"periodDuration\"\n interface=\"popover\">\n <ion-select-option value=\"0.25\">15 minutes</ion-select-option>\n <ion-select-option value=\"0.5\">30 minutes</ion-select-option>\n <ion-select-option value=\"1\">1 hour</ion-select-option>\n <ion-select-option value=\"2\">2 hours</ion-select-option>\n <ion-select-option value=\"3\">3 hours</ion-select-option>\n <ion-select-option value=\"4\">4 hours</ion-select-option>\n <ion-select-option value=\"5\">5 hours</ion-select-option>\n <ion-select-option value=\"6\">6 hours</ion-select-option>\n <ion-select-option value=\"8\">8 hours</ion-select-option>\n <ion-select-option value=\"10\">10 hours</ion-select-option>\n <ion-select-option value=\"12\">12 hours</ion-select-option>\n <ion-select-option *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n </ion-select>\n </div>\n\n <div class=\"period-start ion-hide-md-down\">\n <input (change)=\"onPeriodEndChange($event)\"\n [disabled]=\"toggleCumulative\"\n [value]=\"periodEndAsString\"\n class=\"datetime-input\"\n type=\"datetime-local\">\n </div>\n\n <div class=\"toggle-cumulative\" (click)=\"onCumulativeToggleClick($event)\">\n <ion-label [class.text-primary]=\"toggleCumulative\">\n {{ toggleCumulative ? 'Cumulative' : 'Granular' }}\n </ion-label>\n <ion-toggle #cumulativeToggle [checked]=\"toggleCumulative\">\n </ion-toggle>\n </div>\n </div>\n\n <!-- Hidden label for change detection (uncomment to debug)\n <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\n -->\n </div>\n\n <!-- Historical map section -->\n <div *ngIf=\"toggleHistory\" class=\"period-controls\">\n <raain-date-dynamic (changedDate)=\"onDateChangeInCount($event)\"\n [currentHeight]=\"300\"\n [fetchData]=\"fetchDataWrapper\"\n [points]=\"countPoints\">\n </raain-date-dynamic>\n </div>\n </ion-card-content>\n </ion-card>\n\n <!-- Map performance -->\n <ion-grid class=\"map-performance\">\n <ion-row id=\"progressAndRefresh\">\n <ion-col class=\"provider-selection\" size=\"12\" size-md=\"6\">\n <ion-button (click)=\"openQualityModal()\" class=\"quality-info-button\" fill=\"clear\">\n {{ refreshManager.rainComputationMapVersion }}\n <ion-icon name=\"help-circle-outline\" slot=\"end\"></ion-icon>\n </ion-button>\n <div *ngIf=\"availableProviders.length > 0 || availableTimeSteps.length > 0\" class=\"selection-row\">\n <ion-select (ionChange)=\"onProviderChanged($event)\"\n [value]=\"selectedProvider\"\n interface=\"popover\"\n label=\"Provider\"\n placeholder=\"Select Provider\">\n <ion-select-option *ngFor=\"let provider of availableProviders\" [value]=\"provider\">\n {{ provider }}\n </ion-select-option>\n </ion-select>\n\n <ion-select (ionChange)=\"onTimeStepChanged($event)\"\n [value]=\"selectedTimeStep\"\n interface=\"popover\"\n label=\"Time Step\"\n placeholder=\"Select Time Step\">\n <ion-select-option *ngFor=\"let step of availableTimeSteps\" [value]=\"step\">\n {{ step }} min\n </ion-select-option>\n </ion-select>\n </div>\n </ion-col>\n <ion-col class=\"ion-text-right\" size=\"12\" size-md=\"6\">\n <ion-label class=\"ion-margin-end\">\n <span *ngIf=\"percentOfComputations\">\n {{ percentOfComputations }}% Images\n <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n In Progress: {{ countsPeriod.progress }}...\n </i>\n </span>\n <span *ngIf=\"!percentOfComputations\">\n No image available\n </span>\n\n </ion-label>\n\n <ion-button (click)=\"refreshMap()\" [disabled]=\"refreshInProgress\" class=\"refresh-button\">\n Refresh Map\n </ion-button>\n </ion-col>\n </ion-row>\n\n <!-- status update row -->\n <ion-row>\n <!-- Progress col -->\n <ion-progress-bar\n [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n [style.visibility]=\"refreshInProgress ? 'visible' : 'hidden'\"\n [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n color=\"primary\">\n </ion-progress-bar>\n\n <!-- Error col -->\n <ion-col (click)=\"showFullError = !showFullError\" *ngIf=\"refreshManager.lastError\" class=\"error-row\"\n size=\"12\">\n <div class=\"error-content\">\n <ion-icon class=\"error-icon\" name=\"warning-outline\"></ion-icon>\n <span [class.expanded]=\"showFullError\" class=\"error-text\">\n {{ showFullError ? refreshManager.lastError : truncatedError }}\n </span>\n <ion-icon [name]=\"showFullError ? 'chevron-up' : 'chevron-down'\" class=\"error-caret\"></ion-icon>\n </div>\n </ion-col>\n </ion-row>\n </ion-grid>\n\n <!-- Map section -->\n <ion-card class=\"map-card\">\n <ion-card-content class=\"map-content\">\n <ion-grid>\n <ion-row *ngIf=\"toggleMap && percentOfImages\">\n <!-- Map component -->\n <ion-col class=\"map-column\" size-lg=\"7\" size-md=\"12\">\n <div class=\"map-container\">\n <raain-map #raainMapRef\n (changedDate)=\"onDateChangeInMap($event)\"\n (changedSum)=\"onSumChangeInMap($event)\"\n (selectedMarker)=\"onGaugeSelectInMap($event)\"\n [coordinates]=\"coordinates\"\n [cumulativeDurationInMinutes]=\"cumulativeDurationInMinutes\"\n [currentHeight]=\"500\"\n [defaultDate]=\"dateShown\"\n [markers]=\"{\n borders,\n gauges: compareManager.gaugesInMap,\n gaugesInCompare: compareManager.gaugesInCompare,\n selectedGauges: compareManager.selectedGauges,\n pixels: compareManager.selectedPixels,\n pixelsSolution: compareManager.pixelsSolutions?.length ? compareManager.pixelsSolutions[0] : [],\n speeds: compareManager.speeds\n }\"\n [showCumulative]=\"toggleCumulative\"\n [showVisiblePixelMarkers]=\"showPixelMarkers\"\n [sumFn]=\"refreshManager.sumFn\"\n [sumValues]=\"refreshManager.sumValues\"\n [timeframeContainers]=\"timeframeContainers\"\n [timeframeDates]=\"timeframeDates\">\n </raain-map>\n </div>\n\n <div class=\"data-column\">\n <!-- Technical details (collapsible for cleaner UI) -->\n <details class=\"technical-details\">\n <summary>Image Details</summary>\n <div class=\"details-content\">\n <div class=\"detail-row\">\n <span class=\"detail-label\">Computed:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Version:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapVersion }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Launched by:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapLaunchedBy }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Time spent:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapTimeSpentInMs }}\n ms</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Date:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapDate?.toISOString() }}\n | {{ refreshManager.rainComputationMapDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Water in the map:</span>\n <span class=\"detail-value\">{{ sumDetails }}</span>\n <ion-toggle\n (ionChange)=\"onTogglePixelMarkers()\"\n [(ngModel)]=\"showPixelMarkers\"\n style=\"margin-left: 8px; transform: scale(0.7);\">\n </ion-toggle>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Source DBZ min:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMin }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Source DBZ max:</span>\n <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMax }}</span>\n </div>\n </div>\n </details>\n </div>\n </ion-col>\n\n <!-- Data panel -->\n <ion-col *ngIf=\"!!compareManager.compareVersion\" class=\"data-column\" size-lg=\"5\" size-md=\"12\">\n <div class=\"data-panel\">\n <!-- Compare stack component -->\n <div class=\"compare-stack\">\n <raain-compare-stack\n (selectedPoint)=\"onGaugeSelectInCompare($event)\"\n [compareManager]=\"compareManager\"\n [cumulative]=\"toggleCumulative\">\n </raain-compare-stack>\n </div>\n\n <!-- Technical details (collapsible for cleaner UI) -->\n <details class=\"technical-details\">\n <summary>Compare Details</summary>\n <div class=\"details-content\">\n <div [ngClass]=\"{'warning': refreshManager.rainComputationMapDoneDate?.getTime() > compareManager.currentQualityDoneDate?.getTime() + 60000}\"\n class=\"detail-row\">\n <span class=\"detail-label\">Computed:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Version:</span>\n <span class=\"detail-value\">{{ compareManager.compareVersion }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Launched by:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityLaunchedBy }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Time spent:</span>\n <span class=\"detail-value\">{{ compareManager.currentQualityTimeSpentInMs }}\n ms</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Gauges:</span>\n <span class=\"detail-value\">{{ compareManager.gaugesInCompare.length }}</span>\n </div>\n <div class=\"detail-row\">\n <span class=\"detail-label\">Points:</span>\n <span class=\"detail-value\">{{ compareManager.globalComparePoints.length }}</span>\n </div>\n </div>\n </details>\n </div>\n </ion-col>\n </ion-row>\n <ion-row>\n <!-- Bottom progress bar -->\n <ion-progress-bar *ngIf=\"refreshInProgress\"\n [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n color=\"primary\">\n </ion-progress-bar>\n </ion-row>\n </ion-grid>\n </ion-card-content>\n </ion-card>\n\n <!-- Gauge values section -->\n <ion-card *ngIf=\"gaugeSelectedPoints.length && gaugeSelectedPoints[0].values.length\" class=\"gauge-card\">\n <ion-card-header>\n <ion-card-title>\n <ion-icon name=\"analytics-outline\"></ion-icon>\n Selected Gauge Data\n </ion-card-title>\n </ion-card-header>\n <ion-card-content>\n <raain-date-focus\n [currentHeight]=\"300\"\n [focusDate]=\"periodBegin\"\n [focusRange]=\"DateRange.DAY\"\n [points]=\"gaugeSelectedPoints\"\n [withoutAll]=\"true\">\n </raain-date-focus>\n </ion-card-content>\n </ion-card>\n\n <!-- Quality Performance Modal -->\n <div (click)=\"closeQualityModal()\" *ngIf=\"showQualityModal\" class=\"quality-modal-overlay\">\n <div (click)=\"$event.stopPropagation()\" class=\"quality-modal-content\">\n <div class=\"quality-modal-header\">\n <h2>Model Quality Performance</h2>\n <ion-button (click)=\"closeQualityModal()\" fill=\"clear\">\n <ion-icon name=\"close-outline\"></ion-icon>\n </ion-button>\n </div>\n <div class=\"quality-modal-body\">\n <div *ngIf=\"qualityIndicatorsLoading\" class=\"quality-loading\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>Loading indicators...</span>\n </div>\n <div *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length === 0\" class=\"quality-empty\">\n No quality indicators available for this year.\n </div>\n <table *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length > 0\" class=\"quality-table\">\n <thead>\n <tr>\n <th>Model</th>\n <th>Compare</th>\n <th>Gauges</th>\n <th>Period</th>\n <th>Avg Quality</th>\n <th>Updated</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let indicator of qualityIndicators\">\n <td>{{ indicator.computingVersion }}</td>\n <td>{{ indicator.qualityVersion }}</td>\n <td>{{ indicator.provider }}<br>{{ indicator.timeStepInMinutes }} min</td>\n <td>{{ formatDate(indicator.startDate) }}<br>{{ formatDate(indicator.endDate) }}</td>\n <td>{{ indicator.averageQuality | number:'1.2-2' }}</td>\n <td>{{ formatDate(indicator.lastUpdatedAt) }}</td>\n </tr>\n </tbody>\n </table>\n </div>\n </div>\n </div>\n\n <!-- Cumulative Selector Modal -->\n <cumulative-selector *ngIf=\"showCumulativeSelector\"\n [rainId]=\"rainNode?.id\"\n [currentPeriodBegin]=\"periodBegin\"\n [currentPeriodEnd]=\"periodEnd\"\n [provider]=\"selectedProvider\"\n [timeStepInMinutes]=\"selectedTimeStep\"\n [isAdmin]=\"toggleAdmin\"\n (periodSelected)=\"onCumulativePeriodSelected($event)\"\n (cancelled)=\"onCumulativeSelectorCancelled()\">\n </cumulative-selector>\n\n</div>\n", styles: [".raain-details-container{max-width:var(--app-max-width);margin:0 auto;padding:0 0 24px}.raain-details-card{width:100%;margin-bottom:20px}.raain-details-card ion-card-header{border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.raain-details-card ion-card-header ion-card-title{display:flex;align-items:center}.raain-details-card ion-card-header ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary)}.node-info-card{background-color:var(--ion-color-light)}.node-info-card .node-header{display:flex;align-items:center}.node-info-card .node-header .node-status{margin-right:16px}.node-info-card .node-header .node-status ion-icon{font-size:24px}.node-info-card .node-header .node-titles{flex:1}.node-info-card .node-header .node-titles ion-card-title{margin:0;font-size:1.4rem;font-weight:600;color:var(--ion-color-dark)}.node-info-card .node-header .node-titles ion-card-subtitle{padding-left:0;margin:4px 0 0;font-size:.9rem;color:var(--ion-color-medium)}.count-map-card,.period-card{background-color:var(--ion-color-light)}.period-card ion-card-title{display:flex;align-items:center}.period-card ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary);font-size:20px}.period-row{display:flex;flex-wrap:wrap;align-items:center;gap:16px;position:relative}.now-button{min-width:100px}#all-dates{display:flex;align-items:center;margin-left:auto}#all-dates .toggle-label{margin-right:8px;white-space:nowrap}.refresh-button ion-icon{margin-right:4px;color:var(--ion-color-light)}.provider-selection{display:flex;align-items:center}.quality-info-button{--padding-start: 8px;--padding-end: 8px;font-size:.85rem;color:var(--ion-color-medium)}.quality-info-button ion-icon{font-size:18px;margin-left:4px}.selection-row{display:flex;flex-direction:row;align-items:center;gap:12px}.selection-row ion-select{flex:0 0 auto;min-width:100px;margin-bottom:0;padding:4px 8px;border:1px solid rgba(var(--ion-color-medium-rgb),.3);border-radius:var(--ion-border-radius);background-color:var(--ion-color-light)}.period-start,.period-duration{display:flex;align-items:center}.toggle-cumulative{display:flex;align-items:center;gap:8px;margin-left:auto}.toggle-cumulative .text-primary{color:var(--ion-color-primary);font-weight:600}.hidden-label{display:none}.datetime-input,#periodDuration,.duration-select{padding:8px 12px;border:1px solid rgba(var(--ion-color-medium-rgb),.3);border-radius:var(--ion-border-radius);background-color:var(--ion-color-light);font-family:var(--ion-font-family)}.datetime-input:focus,#periodDuration:focus,.duration-select:focus{outline:none;border-color:var(--ion-color-primary)}#periodDuration,.duration-select{min-width:150px}.gauge-card{background-color:var(--ion-color-light)}.gauge-card ion-card-title{display:flex;align-items:center}.gauge-card ion-card-title ion-icon{margin-right:8px;color:var(--ion-color-primary);font-size:20px}.error-row{cursor:pointer;background-color:rgba(var(--ion-color-danger-rgb),.1);border-left:3px solid var(--ion-color-danger);margin-top:8px;border-radius:4px;transition:background-color .2s ease}.error-row:hover{background-color:rgba(var(--ion-color-danger-rgb),.15)}.error-row .error-content{display:flex;align-items:flex-start;padding:8px 12px;gap:8px}.error-row .error-icon{color:var(--ion-color-danger);font-size:18px;flex-shrink:0;margin-top:2px}.error-row .error-text{flex:1;color:var(--ion-color-danger-shade);font-size:.9rem;word-break:break-word}.error-row .error-text.expanded{white-space:pre-wrap}.error-row .error-caret{color:var(--ion-color-danger);font-size:16px;flex-shrink:0;margin-top:2px}raain-compare-stack{width:100%;display:block}@media (max-width: 768px){.period-row{flex-direction:row;justify-content:space-between;align-items:center}#all-dates{margin-left:auto;padding-left:8px}#all-dates .toggle-label{font-size:.9rem}.map-header{flex-direction:row;justify-content:space-between;align-items:center;gap:16px}}.quality-modal-overlay{position:fixed;inset:0;background-color:#00000080;display:flex;align-items:center;justify-content:center;z-index:9999}.quality-modal-content{background-color:var(--ion-background-color, #fff);border-radius:12px;width:95%;max-width:900px;max-height:80vh;overflow:auto;box-shadow:0 4px 24px #0003}.quality-modal-header{display:flex;align-items:center;justify-content:space-between;padding:16px 20px;border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.quality-modal-header h2{margin:0;font-size:1.25rem;font-weight:600;color:var(--ion-color-dark)}.quality-modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.quality-modal-body{padding:20px}.quality-table{width:100%;border-collapse:collapse;margin-top:16px;table-layout:fixed}.quality-table th,.quality-table td{width:16.66%;padding:12px 16px;text-align:center;vertical-align:top;border-bottom:1px solid rgba(var(--ion-color-medium-rgb),.2)}.quality-table th{background-color:rgba(var(--ion-color-primary-rgb),.1);font-weight:600;color:var(--ion-color-dark)}.quality-table td{color:var(--ion-color-dark-tint);font-size:.7rem}.quality-table tbody tr:hover{background-color:rgba(var(--ion-color-primary-rgb),.05)}.quality-loading{display:flex;flex-direction:column;align-items:center;justify-content:center;padding:40px 20px;color:var(--ion-color-medium)}.quality-loading ion-spinner{margin-bottom:12px}.quality-empty{text-align:center;padding:40px 20px;color:var(--ion-color-medium);font-style:italic}\n"] }]
508
571
  }], ctorParameters: function () { return [{ type: i1.Storage }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { toggleAdmin: [{
509
572
  type: Input
510
573
  }], rainNode: [{
@@ -517,5 +580,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
517
580
  type: Input
518
581
  }], radarService: [{
519
582
  type: Input
583
+ }], cumulativeToggleRef: [{
584
+ type: ViewChild,
585
+ args: ['cumulativeToggle']
520
586
  }] } });
521
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"raain-details.component.js","sourceRoot":"","sources":["../../../src/core/shared/raain-details/raain-details.component.ts","../../../src/core/shared/raain-details/raain-details.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    Input,\n    OnChanges,\n    OnDestroy,\n    SimpleChanges,\n} from '@angular/core';\nimport {ProfileService} from '../profile.service';\nimport {CartesianMeasureValue, QualityIndicator, RainNode, TeamNode} from 'raain-model';\nimport {DateRange, MapLatLng, TimeframeContainers} from 'raain-ui';\nimport {CompareManager, FrameSet, mapDateRangeToString, RefreshManager} from '../tools';\nimport {XYType} from '../xytype';\nimport {RadarService} from '../radar.service';\nimport {Storage} from '../storage.service';\n\nlet TEST_DETECTION = 0;\n\n@Component({\n    selector: 'raain-details',\n    templateUrl: 'raain-details.component.html',\n    styleUrls: ['raain-details.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class RaainDetailsComponent implements OnDestroy, OnChanges {\n    private static readonly HOUR_MS = 60 * 60000;\n    private static readonly DAY_MS = 24 * 60 * 60 * 1000;\n\n    @Input()\n    public toggleAdmin: boolean;\n    @Input()\n    public rainNode: RainNode;\n    @Input()\n    public compareManager: CompareManager;\n    @Input()\n    public refreshManager: RefreshManager;\n    @Input()\n    public profileService: ProfileService;\n    @Input()\n    public radarService: RadarService;\n\n    public availableProviders: string[] = [];\n    public availableTimeSteps: number[] = [];\n    public selectedProvider: string;\n    public selectedTimeStep: number;\n\n    public periodBegin: Date;\n    public periodEnd: Date;\n\n    // public UI variables\n    public teamNode: TeamNode;\n    public periodBeginAsString: string;\n    public periodEndAsString: string;\n    public periodDurationAsString: string;\n    public dateShown: Date;\n    public coordinates: MapLatLng;\n    public borders: MapLatLng[];\n    public timeframeContainers: TimeframeContainers;\n    public currentTimeframeTarget: FrameSet;\n    public timeframeDates: Date[];\n    public countPoints: {\n        label: string;\n        style: string;\n        values: {date: Date; value: number}[];\n    }[];\n    public countsPeriod: {\n        percentImages?: XYType[];\n        queueRunning?: number;\n        progress?: number;\n    };\n\n    public gaugeSelectedPoints: {\n        label: string;\n        style: string;\n        values: {date: Date; value: number}[];\n    }[];\n    public toggleHistory: boolean;\n    public toggleMap: boolean;\n    public toggleCompare: boolean;\n    public toggleGaugeMeasures: boolean;\n    public toggleRemoveCompareDuplicate: boolean;\n    public toggleCumulative: boolean;\n    public refreshInProgress: boolean;\n    public sumDetails: string;\n    public showPixelMarkers: boolean = false;\n    public isAdmin: boolean;\n    public showFullError: boolean;\n    public showQualityModal: boolean;\n    public qualityIndicators: QualityIndicator[] = [];\n    public qualityIndicatorsLoading: boolean = false;\n\n    // Cached computed values (to avoid method calls in templates)\n    public percentOfComputations: number = 0;\n    public percentOfImages: number = 0;\n    public truncatedError: string = '';\n    public cumulativeDurationInMinutes: number = 10;\n    protected readonly DateRange = DateRange;\n\n    constructor(\n        protected storage: Storage,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    private static PeriodDisplay(date: Date | string) {\n        let d = new Date();\n        if (date) {\n            d = new Date(date);\n            const userTimezoneOffset = d.getTimezoneOffset() * 60000;\n            d = new Date(d.getTime() - userTimezoneOffset);\n        }\n        const exampleFormattedDate = '2017-06-01T08:30';\n        return d.toISOString().substring(0, exampleFormattedDate.length);\n    }\n\n    private static DateUTC(date: string) {\n        const hasISOFormat = date.match(/^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d{3}Z$/);\n        let d = new Date(date);\n        if (!hasISOFormat) {\n            const userTimezoneOffset = d.getTimezoneOffset() * 60000;\n            d = new Date(d.getTime() - userTimezoneOffset);\n        }\n        return d;\n    }\n\n    private static MapCountToDateValue(c: XYType): {date: Date; value: number} {\n        return {\n            date: RaainDetailsComponent.DateUTC(c.name),\n            value: Math.min(100, c.x),\n        };\n    }\n\n    async openQualityModal() {\n        this.showQualityModal = true;\n        this.qualityIndicatorsLoading = true;\n        this.qualityIndicators = [];\n        this.cdr.markForCheck();\n\n        if (this.rainNode?.id) {\n            const result = await this.profileService.getIndicators(this.rainNode.id);\n            this.qualityIndicators = result.indicators;\n        }\n\n        this.qualityIndicatorsLoading = false;\n        this.cdr.markForCheck();\n    }\n\n    closeQualityModal() {\n        this.showQualityModal = false;\n    }\n\n    formatDate(dateStr: string): string {\n        const date = new Date(dateStr);\n        return (\n            date.toLocaleDateString() +\n            ' ' +\n            date.toLocaleTimeString([], {hour: '2-digit', minute: '2-digit'})\n        );\n    }\n\n    public async fetchData(focusDate: Date, focusRange: DateRange) {\n        const values = [];\n        for (let i = 0; i < 10; i++) {\n            values.push({date: new Date(2020 + i, 0), value: 10});\n        }\n        const fakeData = [\n            {\n                label: '% Rainy',\n                style: 'bar',\n                values,\n            },\n            {\n                label: '% Images',\n                style: 'bar',\n                values,\n            },\n            //  {\n            //      label: '% Quality',\n            //      style: 'line',\n            //      values,\n            //  },\n        ];\n\n        const range = mapDateRangeToString(focusRange);\n        let data = fakeData;\n        if (!this.rainNode) {\n            return data;\n        }\n\n        if (focusRange === DateRange.CENTURY) {\n            // fake\n        } else if (focusRange === DateRange.HOUR) {\n            const hourCounts = await this.profileService.getCountsHour(this.rainNode.id, {\n                periodBegin: focusDate,\n            });\n\n            data = [\n                {\n                    label: 'Rainy Count',\n                    style: 'line',\n                    values: hourCounts.rainyCount.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n                {\n                    label: '% Images',\n                    style: 'bar',\n                    values: hourCounts.percentImages.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n                {\n                    label: 'Rainy Sum',\n                    style: 'line',\n                    values: hourCounts.rainySum.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n            ];\n        } else {\n            const counts = await this.profileService.getCounts(this.rainNode.id, {\n                range,\n                periodBegin: focusDate,\n            });\n\n            data = [\n                {\n                    label: '% Rainy',\n                    style: 'bar',\n                    values: counts.percentRainy.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n                {\n                    label: '% Images',\n                    style: 'bar',\n                    values: counts.percentImages.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n            ];\n        }\n\n        // console.log(`fetchData DONE ${range}`, data);\n        return data;\n    }\n\n    // Wrapper function that preserves the async nature of fetchData\n    public fetchDataWrapper = async (focusDate: Date, focusRange: DateRange) => {\n        return await this.fetchData(focusDate, focusRange);\n    };\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        this.change(changes).then((ignored) => {});\n    }\n\n    ngOnDestroy() {\n        this.cleanAll();\n    }\n\n    async onEnableCountHistoryTab(rain: RainNode) {\n        if (!this.toggleHistory) {\n            this.countPoints = [];\n        }\n    }\n\n    async onPeriodBeginChange(event: any) {\n        const newValue = event?.target?.value ?? this.periodBeginAsString;\n        this.periodBegin = new Date(newValue);\n        this.periodBeginAsString = RaainDetailsComponent.PeriodDisplay(this.periodBegin);\n        this.storage.set('raain-periodBegin', this.periodBegin);\n\n        await this.onPeriodDurationChange(event);\n    }\n\n    async onPeriodEndChange(event: any) {\n        const newValue = event?.target?.value ?? this.periodEndAsString;\n        this.periodEnd = new Date(newValue);\n        this.periodEndAsString = RaainDetailsComponent.PeriodDisplay(this.periodEnd);\n        this.storage.set('raain-periodEnd', this.periodEnd);\n\n        this.periodBegin = new Date(\n            this.periodEnd.getTime() - this.getDurationInHours() * RaainDetailsComponent.HOUR_MS\n        );\n        this.periodBeginAsString = RaainDetailsComponent.PeriodDisplay(this.periodBegin);\n        this.storage.set('raain-periodBegin', this.periodBegin);\n        this.updateRefreshManagerPeriod();\n    }\n\n    async onPeriodDurationChange(_event: object) {\n        const durationInHours = this.getDurationInHours();\n        this.periodEnd = new Date(\n            this.periodBegin.getTime() + durationInHours * RaainDetailsComponent.HOUR_MS\n        );\n        this.periodEndAsString = RaainDetailsComponent.PeriodDisplay(this.periodEnd);\n        this.storage.set('raain-periodEnd', this.periodEnd);\n        this.storage.set('raain-periodDurationInHours', durationInHours);\n        this.updateRefreshManagerPeriod();\n        this.updateCumulativeDurationInMinutes();\n    }\n\n    async onDateChangeInCount(dateChanged: Date) {\n        const dateString =\n            dateChanged.toISOString().substring(0, 11) +\n            dateChanged.toLocaleTimeString().substring(0, 5);\n        this.periodDurationAsString = '1';\n\n        if (this.toggleCumulative) {\n            // Cumulative: select periodEnd\n            this.periodEndAsString = dateString;\n            await this.onPeriodEndChange(null);\n        } else {\n            // Granular: select periodBegin\n            this.periodBeginAsString = dateString;\n            await this.onPeriodBeginChange(null);\n        }\n        await this.refreshManager.refresh(false, this.toggleAdmin);\n    }\n\n    async onDateChangeInMap(dateShown: Date) {\n        this.dateShown = dateShown;\n        await this.fetchAndUpdateMap();\n        await this.refreshManager.setReportPeriod(this.dateShown);\n    }\n\n    async onSumChangeInMap(sum: string) {\n        this.sumDetails = sum;\n    }\n\n    async onGaugeSelectInMap(mapLatLng: MapLatLng) {\n        const gaugesFiltered = this.compareManager.gaugesInMap.filter(\n            (g) => g.lat === mapLatLng.lat && g.lng === mapLatLng.lng\n        );\n        if (gaugesFiltered.length !== 1) {\n            return;\n        }\n        const gaugeSelected = gaugesFiltered[0];\n\n        await this.refreshGaugeValues({id: gaugeSelected.id, name: gaugeSelected.name});\n\n        await this.compareManager.selectGauge(gaugeSelected.id, 0);\n    }\n\n    async refreshGaugeValues(gaugeSelected: {id: string; name: string}) {\n        const gaugeValueShowBegin = new Date(\n            this.periodBegin.getTime() - RaainDetailsComponent.DAY_MS\n        );\n        const gaugeValueShowEnd = new Date(this.periodEnd.getTime() + RaainDetailsComponent.DAY_MS);\n        gaugeValueShowBegin.setHours(0, 0);\n        gaugeValueShowEnd.setHours(23, 59);\n\n        const gaugeMeasures = await this.profileService.getGaugeMeasures(\n            gaugeSelected.id,\n            gaugeValueShowBegin,\n            gaugeValueShowEnd\n        );\n        const gaugeValues: {date: Date; value: number}[] = gaugeMeasures.map((gm) => {\n            // eslint-disable-next-line @typescript-eslint/no-explicit-any\n            const cartesianMeasureValue = new CartesianMeasureValue(gm.values[0] as any);\n            return {\n                date: gm.date,\n                value: cartesianMeasureValue.getCartesianValues()[0].value,\n            };\n        });\n        this.gaugeSelectedPoints = [\n            {\n                label: gaugeSelected.name,\n                style: 'bar',\n                values: gaugeValues,\n            },\n        ];\n        this.cdr.markForCheck();\n    }\n\n    async onGaugeSelectInCompare(e: {point: XYType; compareIndex: number}) {\n        await this.refreshGaugeValues({id: e.point.id, name: e.point.name});\n\n        await this.compareManager.selectGauge(e.point.id, e.compareIndex);\n    }\n\n    async onToggleMap($event: Event) {\n        // if (this.toggleMap) {\n        //    await this.refreshMap();\n        // }\n    }\n\n    onTogglePixelMarkers() {\n        // Toggle is bound to showPixelMarkers, raain-map handles marker display\n    }\n\n    async toggleCumulativeChanged(_$event: Event) {\n        this.storage.set('raain-toggleCumulative', this.toggleCumulative);\n        this.dateShown = this.getDateBasedOnCumulativeMode(this.timeframeDates);\n        this.updateCumulativeDurationInMinutes();\n        if (this.toggleMap) {\n            this.updateRefreshManagerPeriod(); // Update cumulative flag before refresh\n            await this.refreshManager.refresh(false, this.toggleAdmin);\n        }\n    }\n\n    updateCumulativeDurationInMinutes() {\n        if (this.toggleCumulative) {\n            // Cumulative mode: use period duration\n            this.cumulativeDurationInMinutes = parseFloat(this.periodDurationAsString) * 60;\n        } else {\n            // Granular mode: use selectedTimeStep\n            this.cumulativeDurationInMinutes = this.selectedTimeStep || 10;\n        }\n    }\n\n    async onProviderChanged($event: Event) {\n        this.selectedProvider = ($event as CustomEvent)?.detail?.value;\n        this.storage.set('raain-selectedProvider', this.selectedProvider);\n        await this.applyRefreshManagerSettings();\n    }\n\n    async onTimeStepChanged($event: Event) {\n        this.selectedTimeStep = ($event as CustomEvent)?.detail?.value;\n        this.storage.set('raain-selectedTimeStep', this.selectedTimeStep);\n        this.updateCumulativeDurationInMinutes();\n        await this.applyRefreshManagerSettings();\n    }\n\n    async loadProviders() {\n        if (!this.rainNode?.id) {\n            return;\n        }\n\n        try {\n            const result = await this.profileService.getProviders(this.rainNode.id);\n            this.availableProviders = result.providers;\n            this.availableTimeSteps = result.timeStepInMinutes;\n\n            // Load saved selections or use defaults\n            this.selectedProvider =\n                this.storage.get('raain-selectedProvider') ||\n                (this.availableProviders.length > 0 ? this.availableProviders[0] : 'Raain');\n            this.selectedTimeStep =\n                this.storage.get('raain-selectedTimeStep') ||\n                (this.availableTimeSteps.length > 0 ? this.availableTimeSteps[0] : 10);\n        } catch (e) {\n            // Set fallback values\n            this.availableProviders = ['Raain'];\n            this.availableTimeSteps = [5, 10, 15, 30, 60];\n            this.selectedProvider = 'Raain';\n            this.selectedTimeStep = 10;\n        }\n\n        // Set provider and timeStep on refreshManager\n        if (this.refreshManager) {\n            this.refreshManager.provider = this.selectedProvider;\n            this.refreshManager.timeStepInMinutes = this.selectedTimeStep;\n        }\n        this.cdr.markForCheck();\n    }\n\n    onChangeDetectionTest(rainNode: RainNode) {\n        console.log(TEST_DETECTION++, 'onChangeDetectionTest');\n        return '';\n    }\n\n    updateTruncatedError() {\n        const error = this.refreshManager?.lastError || '';\n        if (error.length <= 80) {\n            this.truncatedError = error;\n            return;\n        }\n        this.truncatedError = error.substring(0, 80) + '...';\n    }\n\n    updateCachedValues() {\n        this.updatePercentOfImages();\n        this.updatePercentOfComputations();\n        this.updateTruncatedError();\n        this.updateCumulativeDurationInMinutes();\n    }\n\n    async refreshMap() {\n        this.gaugeSelectedPoints = [];\n        this.dateShown = this.getDateBasedOnCumulativeMode();\n        this.borders = [];\n\n        this.compareManager.cleanAll();\n        await this.compareManager.setGaugesInMap();\n        await this.refreshManager.refresh(false, this.toggleAdmin);\n        this.cdr.markForCheck();\n    }\n\n    async setPeriodOfNow() {\n        const last30mn = new Date();\n        last30mn.setMinutes(last30mn.getMinutes() - 30);\n        this.periodBeginAsString =\n            last30mn.toISOString().substring(0, 11) + last30mn.toLocaleTimeString().substring(0, 5);\n        this.periodDurationAsString = '1';\n        await this.onPeriodBeginChange(null);\n        await this.refreshManager.refresh(false, this.toggleAdmin);\n    }\n\n    updatePercentOfImages() {\n        if (!this.countsPeriod?.percentImages?.length) {\n            this.percentOfImages = 0;\n            return;\n        }\n\n        const duringPeriod = this.countsPeriod.percentImages.filter(\n            (a) =>\n                this.periodBegin.getTime() <= new Date(a.name).getTime() &&\n                new Date(a.name).getTime() <= this.periodEnd.getTime()\n        );\n        const summed = duringPeriod.reduce((a, b) => a + (b.x ?? 0), 0);\n        this.percentOfImages = Math.round(summed / duringPeriod.length);\n    }\n\n    updatePercentOfComputations() {\n        const timeline = this.refreshManager?.getTimelineFrameSet();\n        if (!timeline?.length) {\n            this.percentOfComputations = 0;\n            return;\n        }\n\n        const timelineWithComputation = timeline.filter(\n            (a) => !!a.rainComputationCumulativeId || !!a.rainComputationId\n        );\n        const ratio = timelineWithComputation.length / timeline.length;\n        this.percentOfComputations = Math.round(ratio * 100);\n    }\n\n    private getDateBasedOnCumulativeMode(fallbackDates?: Date[]): Date {\n        if (fallbackDates?.length > 0) {\n            const dateExists = fallbackDates.some((d) => d.getTime() === this.dateShown?.getTime());\n            if (!dateExists) {\n                return this.toggleCumulative\n                    ? fallbackDates[fallbackDates.length - 1]\n                    : fallbackDates[0];\n            }\n        }\n        return this.toggleCumulative ? this.periodEnd : this.periodBegin;\n    }\n\n    private getCumulativeHours(): number {\n        return this.toggleCumulative ? parseFloat(this.periodDurationAsString) : 0;\n    }\n\n    private getDurationInHours(): number {\n        return parseFloat(this.periodDurationAsString);\n    }\n\n    private updateRefreshManagerPeriod() {\n        this.refreshManager.cumulative = this.toggleCumulative;\n\n        // Align dates to 5-minute boundaries (floor) for consistency with raain-ground\n        const alignTo5minFloor = (date: Date): Date => {\n            const minutes = date.getMinutes();\n            const alignedMinutes = Math.floor(minutes / 5) * 5;\n            return new Date(\n                date.getFullYear(),\n                date.getMonth(),\n                date.getDate(),\n                date.getHours(),\n                alignedMinutes,\n                0,\n                0\n            );\n        };\n\n        this.refreshManager.period = {\n            begin: alignTo5minFloor(this.periodBegin),\n            end: alignTo5minFloor(this.periodEnd),\n        };\n    }\n\n    private async fetchAndUpdateMap() {\n        await this.refreshManager.fetch(\n            this.dateShown,\n            this.toggleGaugeMeasures,\n            this.getCumulativeHours()\n        );\n        this.currentTimeframeTarget = this.refreshManager.getTimelineSelectedFrameSet();\n        this.cdr.markForCheck();\n    }\n\n    private async applyRefreshManagerSettings() {\n        if (!this.refreshManager) {\n            return;\n        }\n        this.refreshManager.provider = this.selectedProvider;\n        this.refreshManager.timeStepInMinutes = this.selectedTimeStep;\n        await this.refreshManager.refresh(false, this.toggleAdmin);\n    }\n\n    private cleanAll() {\n        this.borders = [];\n        this.isAdmin = false;\n        this.timeframeContainers = new TimeframeContainers([]);\n        this.currentTimeframeTarget = null;\n        this.timeframeDates = [];\n        this.countPoints = [];\n        this.countsPeriod = {progress: 0, queueRunning: 0, percentImages: []};\n        this.gaugeSelectedPoints = [];\n\n        this.toggleHistory = false;\n        this.toggleMap = true;\n        this.toggleCompare = false;\n        this.toggleGaugeMeasures = false;\n        this.toggleRemoveCompareDuplicate = true;\n        this.toggleCumulative = this.storage.get('raain-toggleCumulative');\n\n        this.periodBegin = new Date(this.storage.get('raain-periodBegin'));\n        this.periodEnd = new Date(this.storage.get('raain-periodEnd'));\n        this.periodBeginAsString = RaainDetailsComponent.PeriodDisplay(this.periodBegin);\n        this.periodEndAsString = RaainDetailsComponent.PeriodDisplay(this.periodEnd);\n        const durationMs = this.periodEnd.getTime() - this.periodBegin.getTime();\n        this.periodDurationAsString = '' + durationMs / RaainDetailsComponent.HOUR_MS;\n\n        this.dateShown = this.getDateBasedOnCumulativeMode();\n        this.refreshInProgress = false;\n        this.showFullError = false;\n        this.showQualityModal = false;\n        this.qualityIndicators = [];\n        this.qualityIndicatorsLoading = false;\n\n        this.compareManager?.cleanAll();\n        this.refreshManager?.cleanAll();\n    }\n\n    private async init() {\n        this.cleanAll();\n        this.updateCachedValues();\n        await this.initRain();\n        this.cdr.markForCheck();\n    }\n\n    private async initRain() {\n        if (!this.rainNode) {\n            return;\n        }\n\n        this.isAdmin = this.profileService.isAdmin();\n\n        this.refreshManager.rainNode = this.rainNode;\n        this.compareManager.rainNode = this.rainNode;\n\n        // Load providers and set on refreshManager\n        await this.loadProviders();\n\n        this.refreshManager.setMethods(\n            this.onRefreshInProgress.bind(this),\n            this.onRefreshDone.bind(this),\n            this.onFetchDone.bind(this)\n        );\n\n        const center = this.rainNode.getCenter();\n        this.coordinates = new MapLatLng(center.lat, center.lng);\n        this.teamNode = await this.profileService.getTeam(\n            this.rainNode.getLink(TeamNode.TYPE).getId()\n        );\n\n        if (this.periodBegin && this.periodEnd) {\n            this.updateRefreshManagerPeriod();\n            await this.refreshManager.refresh(false, this.toggleAdmin);\n        }\n    }\n\n    private async onRefreshInProgress(\n        countPeriods: {\n            percentImages?: XYType[];\n            queueRunning?: number;\n            progress?: number;\n        },\n        timeframeDates: Date[]\n    ) {\n        this.refreshInProgress = true;\n        this.countsPeriod = countPeriods;\n        this.timeframeDates = timeframeDates;\n        this.updateCachedValues();\n        this.cdr.markForCheck();\n    }\n\n    private async onRefreshDone(timeframeDates: Date[]) {\n        this.timeframeDates = timeframeDates;\n        this.refreshInProgress = false;\n        this.updateCachedValues();\n        this.cdr.markForCheck();\n\n        if (this.toggleMap && timeframeDates.length > 0) {\n            this.dateShown = this.getDateBasedOnCumulativeMode(timeframeDates);\n            if (this.dateShown) {\n                await this.fetchAndUpdateMap();\n            }\n        }\n    }\n\n    private async onFetchDone(timeframeContainers?: TimeframeContainers) {\n        if (timeframeContainers) {\n            this.timeframeContainers = timeframeContainers;\n        }\n        this.cdr.markForCheck();\n    }\n\n    private async change(_changes: SimpleChanges) {\n        await this.init();\n    }\n}\n","<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n    <!-- Period selection section -->\n    <ion-card class=\"period-card\">\n        <ion-card-content>\n            <div class=\"period-controls\">\n                <div class=\"period-row\">\n\n                    <ion-button (click)=\"toggleHistory = !toggleHistory; onEnableCountHistoryTab(rainNode)\"\n                                fill=\"outline\">\n                        <ion-icon name=\"calendar-clear-outline\" slot=\"start\"></ion-icon>\n                        <ion-icon [name]=\"toggleHistory ? 'chevron-down' : 'chevron-forward'\" slot=\"end\"></ion-icon>\n                    </ion-button>\n\n                    <div class=\"period-start ion-hide-md-down\">\n                        <input (change)=\"onPeriodBeginChange($event)\"\n                               [disabled]=\"toggleCumulative\"\n                               [value]=\"periodBeginAsString\"\n                               class=\"datetime-input\"\n                               type=\"datetime-local\">\n                    </div>\n\n                    <div class=\"period-duration ion-hide-md-down\">\n                        <ion-select (ionDismiss)=\"onPeriodDurationChange($event)\"\n                                    [(ngModel)]=\"periodDurationAsString\"\n                                    class=\"duration-select\"\n                                    id=\"periodDuration\"\n                                    interface=\"popover\">\n                            <ion-select-option value=\"0.25\">15 minutes</ion-select-option>\n                            <ion-select-option value=\"0.5\">30 minutes</ion-select-option>\n                            <ion-select-option value=\"1\">1 hour</ion-select-option>\n                            <ion-select-option value=\"2\">2 hours</ion-select-option>\n                            <ion-select-option value=\"3\">3 hours</ion-select-option>\n                            <ion-select-option value=\"4\">4 hours</ion-select-option>\n                            <ion-select-option value=\"5\">5 hours</ion-select-option>\n                            <ion-select-option value=\"6\">6 hours</ion-select-option>\n                            <ion-select-option value=\"8\">8 hours</ion-select-option>\n                            <ion-select-option value=\"10\">10 hours</ion-select-option>\n                            <ion-select-option value=\"12\">12 hours</ion-select-option>\n                            <ion-select-option *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n                        </ion-select>\n                    </div>\n\n                    <div class=\"period-start ion-hide-md-down\">\n                        <input (change)=\"onPeriodEndChange($event)\"\n                               [disabled]=\"!toggleCumulative\"\n                               [value]=\"periodEndAsString\"\n                               class=\"datetime-input\"\n                               type=\"datetime-local\">\n                    </div>\n\n                    <div class=\"toggle-cumulative\">\n                        <ion-label [class.text-primary]=\"toggleCumulative\">\n                            {{ toggleCumulative ? 'Cumulative' : 'Granular' }}\n                        </ion-label>\n                        <ion-toggle (ionChange)=\"toggleCumulativeChanged($event)\"\n                                    [(ngModel)]=\"toggleCumulative\"\n                                    [checked]=\"toggleCumulative\">\n                        </ion-toggle>\n                    </div>\n                </div>\n\n                <!-- Hidden label for change detection (uncomment to debug)\n                <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\n                -->\n            </div>\n\n            <!-- Historical map section -->\n            <div *ngIf=\"toggleHistory\" class=\"period-controls\">\n                <raain-date-dynamic (changedDate)=\"onDateChangeInCount($event)\"\n                                    [currentHeight]=\"300\"\n                                    [fetchData]=\"fetchDataWrapper\"\n                                    [points]=\"countPoints\">\n                </raain-date-dynamic>\n            </div>\n        </ion-card-content>\n    </ion-card>\n\n    <!-- Map performance -->\n    <ion-grid class=\"map-performance\">\n        <ion-row id=\"progressAndRefresh\">\n            <ion-col class=\"provider-selection\" size=\"12\" size-md=\"6\">\n                <ion-button (click)=\"openQualityModal()\" class=\"quality-info-button\" fill=\"clear\">\n                    {{ refreshManager.rainComputationMapVersion }}\n                    <ion-icon name=\"help-circle-outline\" slot=\"end\"></ion-icon>\n                </ion-button>\n                <div *ngIf=\"availableProviders.length > 0 || availableTimeSteps.length > 0\" class=\"selection-row\">\n                    <ion-select (ionChange)=\"onProviderChanged($event)\"\n                                [value]=\"selectedProvider\"\n                                interface=\"popover\"\n                                label=\"Provider\"\n                                placeholder=\"Select Provider\">\n                        <ion-select-option *ngFor=\"let provider of availableProviders\" [value]=\"provider\">\n                            {{ provider }}\n                        </ion-select-option>\n                    </ion-select>\n\n                    <ion-select (ionChange)=\"onTimeStepChanged($event)\"\n                                [value]=\"selectedTimeStep\"\n                                interface=\"popover\"\n                                label=\"Time Step\"\n                                placeholder=\"Select Time Step\">\n                        <ion-select-option *ngFor=\"let step of availableTimeSteps\" [value]=\"step\">\n                            {{ step }} min\n                        </ion-select-option>\n                    </ion-select>\n                </div>\n            </ion-col>\n            <ion-col class=\"ion-text-right\" size=\"12\" size-md=\"6\">\n                <ion-label class=\"ion-margin-end\">\n                            <span *ngIf=\"percentOfComputations\">\n                                {{ percentOfComputations }}% Images\n                                <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n                                    In Progress: {{ countsPeriod.progress }}...\n                                </i>\n                            </span>\n                    <span *ngIf=\"!percentOfComputations\">\n                                No image available\n                            </span>\n\n                </ion-label>\n\n                <ion-button (click)=\"refreshMap()\" [disabled]=\"refreshInProgress\" class=\"refresh-button\">\n                    Refresh Map\n                </ion-button>\n            </ion-col>\n        </ion-row>\n\n        <!-- status update row -->\n        <ion-row>\n            <!-- Progress col -->\n            <ion-progress-bar\n                    [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n                    [style.visibility]=\"refreshInProgress ? 'visible' : 'hidden'\"\n                    [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n                    color=\"primary\">\n            </ion-progress-bar>\n\n            <!-- Error col -->\n            <ion-col (click)=\"showFullError = !showFullError\" *ngIf=\"refreshManager.lastError\" class=\"error-row\"\n                     size=\"12\">\n                <div class=\"error-content\">\n                    <ion-icon class=\"error-icon\" name=\"warning-outline\"></ion-icon>\n                    <span [class.expanded]=\"showFullError\" class=\"error-text\">\n                        {{ showFullError ? refreshManager.lastError : truncatedError }}\n                    </span>\n                    <ion-icon [name]=\"showFullError ? 'chevron-up' : 'chevron-down'\" class=\"error-caret\"></ion-icon>\n                </div>\n            </ion-col>\n        </ion-row>\n    </ion-grid>\n\n    <!-- Map section -->\n    <ion-card class=\"map-card\">\n        <ion-card-content class=\"map-content\">\n            <ion-grid>\n                <ion-row *ngIf=\"toggleMap && percentOfImages\">\n                    <!-- Map component -->\n                    <ion-col class=\"map-column\" size-lg=\"7\" size-md=\"12\">\n                        <div class=\"map-container\">\n                            <raain-map #raainMapRef\n                                       (changedDate)=\"onDateChangeInMap($event)\"\n                                       (changedSum)=\"onSumChangeInMap($event)\"\n                                       (selectedMarker)=\"onGaugeSelectInMap($event)\"\n                                       [coordinates]=\"coordinates\"\n                                       [cumulativeDurationInMinutes]=\"cumulativeDurationInMinutes\"\n                                       [currentHeight]=\"500\"\n                                       [defaultDate]=\"dateShown\"\n                                       [markers]=\"{\n                                        borders,\n                                        gauges: compareManager.gaugesInMap,\n                                        gaugesInCompare: compareManager.gaugesInCompare,\n                                        selectedGauges: compareManager.selectedGauges,\n                                        pixels: compareManager.selectedPixels,\n                                        pixelsSolution: compareManager.pixelsSolutions?.length ? compareManager.pixelsSolutions[0] : [],\n                                        speeds: compareManager.speeds\n                                    }\"\n                                       [showCumulative]=\"toggleCumulative\"\n                                       [showVisiblePixelMarkers]=\"showPixelMarkers\"\n                                       [sumFn]=\"refreshManager.sumFn\"\n                                       [sumValues]=\"refreshManager.sumValues\"\n                                       [timeframeContainers]=\"timeframeContainers\"\n                                       [timeframeDates]=\"timeframeDates\">\n                            </raain-map>\n                        </div>\n\n                        <div class=\"data-column\">\n                            <!-- Technical details (collapsible for cleaner UI) -->\n                            <details class=\"technical-details\">\n                                <summary>Image Details</summary>\n                                <div class=\"details-content\">\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Computed:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Version:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapVersion }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Launched by:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapLaunchedBy }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Time spent:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapTimeSpentInMs }}\n                                            ms</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Date:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapDate?.toISOString() }}\n                                            | {{ refreshManager.rainComputationMapDate | date:'yyyy-MM-dd HH:mm' }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Water in the map:</span>\n                                        <span class=\"detail-value\">{{ sumDetails }}</span>\n                                        <ion-toggle\n                                                (ionChange)=\"onTogglePixelMarkers()\"\n                                                [(ngModel)]=\"showPixelMarkers\"\n                                                style=\"margin-left: 8px; transform: scale(0.7);\">\n                                        </ion-toggle>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Source DBZ min:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMin }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Source DBZ max:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMax }}</span>\n                                    </div>\n                                </div>\n                            </details>\n                        </div>\n                    </ion-col>\n\n                    <!-- Data panel -->\n                    <ion-col *ngIf=\"!!compareManager.compareVersion\" class=\"data-column\" size-lg=\"5\" size-md=\"12\">\n                        <div class=\"data-panel\">\n                            <!-- Compare stack component -->\n                            <div class=\"compare-stack\">\n                                <raain-compare-stack\n                                        (selectedPoint)=\"onGaugeSelectInCompare($event)\"\n                                        [compareManager]=\"compareManager\"\n                                        [cumulative]=\"toggleCumulative\">\n                                </raain-compare-stack>\n                            </div>\n\n                            <!-- Technical details (collapsible for cleaner UI) -->\n                            <details class=\"technical-details\">\n                                <summary>Compare Details</summary>\n                                <div class=\"details-content\">\n                                    <div [ngClass]=\"{'warning': refreshManager.rainComputationMapDoneDate?.getTime() > compareManager.currentQualityDoneDate?.getTime() + 60000}\"\n                                         class=\"detail-row\">\n                                        <span class=\"detail-label\">Computed:</span>\n                                        <span class=\"detail-value\">{{ compareManager.currentQualityDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Version:</span>\n                                        <span class=\"detail-value\">{{ compareManager.compareVersion }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Launched by:</span>\n                                        <span class=\"detail-value\">{{ compareManager.currentQualityLaunchedBy }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Time spent:</span>\n                                        <span class=\"detail-value\">{{ compareManager.currentQualityTimeSpentInMs }}\n                                            ms</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Gauges:</span>\n                                        <span class=\"detail-value\">{{ compareManager.gaugesInCompare.length }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Points:</span>\n                                        <span class=\"detail-value\">{{ compareManager.globalComparePoints.length }}</span>\n                                    </div>\n                                </div>\n                            </details>\n                        </div>\n                    </ion-col>\n                </ion-row>\n                <ion-row>\n                    <!-- Bottom progress bar -->\n                    <ion-progress-bar *ngIf=\"refreshInProgress\"\n                                      [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n                                      [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n                                      color=\"primary\">\n                    </ion-progress-bar>\n                </ion-row>\n            </ion-grid>\n        </ion-card-content>\n    </ion-card>\n\n    <!-- Gauge values section -->\n    <ion-card *ngIf=\"gaugeSelectedPoints.length && gaugeSelectedPoints[0].values.length\" class=\"gauge-card\">\n        <ion-card-header>\n            <ion-card-title>\n                <ion-icon name=\"analytics-outline\"></ion-icon>\n                Selected Gauge Data\n            </ion-card-title>\n        </ion-card-header>\n        <ion-card-content>\n            <raain-date-focus\n                    [currentHeight]=\"300\"\n                    [focusDate]=\"periodBegin\"\n                    [focusRange]=\"DateRange.DAY\"\n                    [points]=\"gaugeSelectedPoints\"\n                    [withoutAll]=\"true\">\n            </raain-date-focus>\n        </ion-card-content>\n    </ion-card>\n\n    <!-- Quality Performance Modal -->\n    <div (click)=\"closeQualityModal()\" *ngIf=\"showQualityModal\" class=\"quality-modal-overlay\">\n        <div (click)=\"$event.stopPropagation()\" class=\"quality-modal-content\">\n            <div class=\"quality-modal-header\">\n                <h2>Model Quality Performance</h2>\n                <ion-button (click)=\"closeQualityModal()\" fill=\"clear\">\n                    <ion-icon name=\"close-outline\"></ion-icon>\n                </ion-button>\n            </div>\n            <div class=\"quality-modal-body\">\n                <div *ngIf=\"qualityIndicatorsLoading\" class=\"quality-loading\">\n                    <ion-spinner name=\"crescent\"></ion-spinner>\n                    <span>Loading indicators...</span>\n                </div>\n                <div *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length === 0\" class=\"quality-empty\">\n                    No quality indicators available for this year.\n                </div>\n                <table *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length > 0\" class=\"quality-table\">\n                    <thead>\n                    <tr>\n                        <th>Model</th>\n                        <th>Compare</th>\n                        <th>Gauges</th>\n                        <th>Period</th>\n                        <th>Avg Quality</th>\n                        <th>Updated</th>\n                    </tr>\n                    </thead>\n                    <tbody>\n                    <tr *ngFor=\"let indicator of qualityIndicators\">\n                        <td>{{ indicator.computingVersion }}</td>\n                        <td>{{ indicator.qualityVersion }}</td>\n                        <td>{{ indicator.provider }}<br>{{ indicator.timeStepInMinutes }} min</td>\n                        <td>{{ formatDate(indicator.startDate) }}<br>{{ formatDate(indicator.endDate) }}</td>\n                        <td>{{ indicator.averageQuality | number:'1.2-2' }}</td>\n                        <td>{{ formatDate(indicator.lastUpdatedAt) }}</td>\n                    </tr>\n                    </tbody>\n                </table>\n            </div>\n        </div>\n    </div>\n\n</div>\n"]}
587
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"raain-details.component.js","sourceRoot":"","sources":["../../../src/core/shared/raain-details/raain-details.component.ts","../../../src/core/shared/raain-details/raain-details.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    ElementRef,\n    Input,\n    OnChanges,\n    OnDestroy,\n    SimpleChanges,\n    ViewChild,\n} from '@angular/core';\nimport {IonToggle} from '@ionic/angular';\nimport {ProfileService} from '../profile.service';\nimport {CartesianMeasureValue, QualityIndicator, RainNode, TeamNode} from 'raain-model';\nimport {DateRange, MapLatLng, TimeframeContainers} from 'raain-ui';\nimport {CompareManager, FrameSet, mapDateRangeToString, RefreshManager} from '../tools';\nimport {XYType} from '../xytype';\nimport {RadarService} from '../radar.service';\nimport {Storage} from '../storage.service';\nimport {CumulativeSelection} from '../cumulative-selector/cumulative-selector.component';\n\nlet TEST_DETECTION = 0;\n\n@Component({\n    selector: 'raain-details',\n    templateUrl: 'raain-details.component.html',\n    styleUrls: ['raain-details.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class RaainDetailsComponent implements OnDestroy, OnChanges {\n    private static readonly HOUR_MS = 60 * 60000;\n    private static readonly DAY_MS = 24 * 60 * 60 * 1000;\n\n    @Input()\n    public toggleAdmin: boolean;\n    @Input()\n    public rainNode: RainNode;\n    @Input()\n    public compareManager: CompareManager;\n    @Input()\n    public refreshManager: RefreshManager;\n    @Input()\n    public profileService: ProfileService;\n    @Input()\n    public radarService: RadarService;\n\n    public availableProviders: string[] = [];\n    public availableTimeSteps: number[] = [];\n    public selectedProvider: string;\n    public selectedTimeStep: number;\n\n    public periodBegin: Date;\n    public periodEnd: Date;\n\n    // public UI variables\n    public teamNode: TeamNode;\n    public periodBeginAsString: string;\n    public periodEndAsString: string;\n    public periodDurationAsString: string;\n    public dateShown: Date;\n    public coordinates: MapLatLng;\n    public borders: MapLatLng[];\n    public timeframeContainers: TimeframeContainers;\n    public currentTimeframeTarget: FrameSet;\n    public timeframeDates: Date[];\n    public countPoints: {\n        label: string;\n        style: string;\n        values: {date: Date; value: number}[];\n    }[];\n    public countsPeriod: {\n        percentImages?: XYType[];\n        queueRunning?: number;\n        progress?: number;\n    };\n\n    public gaugeSelectedPoints: {\n        label: string;\n        style: string;\n        values: {date: Date; value: number}[];\n    }[];\n    public toggleHistory: boolean;\n    public toggleMap: boolean;\n    public toggleCompare: boolean;\n    public toggleGaugeMeasures: boolean;\n    public toggleRemoveCompareDuplicate: boolean;\n    public toggleCumulative: boolean;\n    public refreshInProgress: boolean;\n    public sumDetails: string;\n    public showPixelMarkers: boolean = false;\n    public isAdmin: boolean;\n    public showFullError: boolean;\n    public showQualityModal: boolean;\n    public qualityIndicators: QualityIndicator[] = [];\n    public qualityIndicatorsLoading: boolean = false;\n    public showCumulativeSelector: boolean = false;\n\n    @ViewChild('cumulativeToggle') cumulativeToggleRef: IonToggle;\n\n    // Cached computed values (to avoid method calls in templates)\n    public percentOfComputations: number = 0;\n    public percentOfImages: number = 0;\n    public truncatedError: string = '';\n    public cumulativeDurationInMinutes: number = 10;\n    protected readonly DateRange = DateRange;\n\n    constructor(\n        protected storage: Storage,\n        private readonly cdr: ChangeDetectorRef\n    ) {}\n\n    private static PeriodDisplay(date: Date | string) {\n        let d = new Date();\n        if (date) {\n            d = new Date(date);\n            const userTimezoneOffset = d.getTimezoneOffset() * 60000;\n            d = new Date(d.getTime() - userTimezoneOffset);\n        }\n        const exampleFormattedDate = '2017-06-01T08:30';\n        return d.toISOString().substring(0, exampleFormattedDate.length);\n    }\n\n    private static DateUTC(date: string) {\n        const hasISOFormat = date.match(/^\\d{4}-\\d{2}-\\d{2}T\\d{2}:\\d{2}:\\d{2}\\.\\d{3}Z$/);\n        let d = new Date(date);\n        if (!hasISOFormat) {\n            const userTimezoneOffset = d.getTimezoneOffset() * 60000;\n            d = new Date(d.getTime() - userTimezoneOffset);\n        }\n        return d;\n    }\n\n    private static MapCountToDateValue(c: XYType): {date: Date; value: number} {\n        return {\n            date: RaainDetailsComponent.DateUTC(c.name),\n            value: Math.min(100, c.x),\n        };\n    }\n\n    async openQualityModal() {\n        this.showQualityModal = true;\n        this.qualityIndicatorsLoading = true;\n        this.qualityIndicators = [];\n        this.cdr.markForCheck();\n\n        if (this.rainNode?.id) {\n            const result = await this.profileService.getIndicators(this.rainNode.id);\n            this.qualityIndicators = result.indicators;\n        }\n\n        this.qualityIndicatorsLoading = false;\n        this.cdr.markForCheck();\n    }\n\n    closeQualityModal() {\n        this.showQualityModal = false;\n    }\n\n    formatDate(dateStr: string): string {\n        const date = new Date(dateStr);\n        return (\n            date.toLocaleDateString() +\n            ' ' +\n            date.toLocaleTimeString([], {hour: '2-digit', minute: '2-digit'})\n        );\n    }\n\n    public async fetchData(focusDate: Date, focusRange: DateRange) {\n        const values = [];\n        for (let i = 0; i < 10; i++) {\n            values.push({date: new Date(2020 + i, 0), value: 10});\n        }\n        const fakeData = [\n            {\n                label: '% Rainy',\n                style: 'bar',\n                values,\n            },\n            {\n                label: '% Images',\n                style: 'bar',\n                values,\n            },\n            //  {\n            //      label: '% Quality',\n            //      style: 'line',\n            //      values,\n            //  },\n        ];\n\n        const range = mapDateRangeToString(focusRange);\n        let data = fakeData;\n        if (!this.rainNode) {\n            return data;\n        }\n\n        if (focusRange === DateRange.CENTURY) {\n            // fake\n        } else if (focusRange === DateRange.HOUR) {\n            const hourCounts = await this.profileService.getCountsHour(this.rainNode.id, {\n                periodBegin: focusDate,\n            });\n\n            data = [\n                {\n                    label: 'Rainy Count',\n                    style: 'line',\n                    values: hourCounts.rainyCount.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n                {\n                    label: '% Images',\n                    style: 'bar',\n                    values: hourCounts.percentImages.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n                {\n                    label: 'Rainy Sum',\n                    style: 'line',\n                    values: hourCounts.rainySum.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n            ];\n        } else {\n            const counts = await this.profileService.getCounts(this.rainNode.id, {\n                range,\n                periodBegin: focusDate,\n            });\n\n            data = [\n                {\n                    label: '% Rainy',\n                    style: 'bar',\n                    values: counts.percentRainy.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n                {\n                    label: '% Images',\n                    style: 'bar',\n                    values: counts.percentImages.map(RaainDetailsComponent.MapCountToDateValue),\n                },\n            ];\n        }\n\n        // console.log(`fetchData DONE ${range}`, data);\n        return data;\n    }\n\n    // Wrapper function that preserves the async nature of fetchData\n    public fetchDataWrapper = async (focusDate: Date, focusRange: DateRange) => {\n        return await this.fetchData(focusDate, focusRange);\n    };\n\n    public ngOnChanges(changes: SimpleChanges): void {\n        this.change(changes).then((ignored) => {});\n    }\n\n    ngOnDestroy() {\n        this.cleanAll();\n    }\n\n    async onEnableCountHistoryTab(rain: RainNode) {\n        if (!this.toggleHistory) {\n            this.countPoints = [];\n        }\n    }\n\n    async onPeriodBeginChange(event: any) {\n        const newValue = event?.target?.value ?? this.periodBeginAsString;\n        this.periodBegin = new Date(newValue);\n        this.periodBeginAsString = RaainDetailsComponent.PeriodDisplay(this.periodBegin);\n        this.storage.set('raain-periodBegin', this.periodBegin);\n\n        await this.onPeriodDurationChange(event);\n    }\n\n    async onPeriodEndChange(event: any) {\n        const newValue = event?.target?.value ?? this.periodEndAsString;\n        this.periodEnd = new Date(newValue);\n        this.periodEndAsString = RaainDetailsComponent.PeriodDisplay(this.periodEnd);\n        this.storage.set('raain-periodEnd', this.periodEnd);\n\n        this.periodBegin = new Date(\n            this.periodEnd.getTime() - this.getDurationInHours() * RaainDetailsComponent.HOUR_MS\n        );\n        this.periodBeginAsString = RaainDetailsComponent.PeriodDisplay(this.periodBegin);\n        this.storage.set('raain-periodBegin', this.periodBegin);\n        this.updateRefreshManagerPeriod();\n    }\n\n    async onPeriodDurationChange(_event: object) {\n        const durationInHours = this.getDurationInHours();\n        this.periodEnd = new Date(\n            this.periodBegin.getTime() + durationInHours * RaainDetailsComponent.HOUR_MS\n        );\n        this.periodEndAsString = RaainDetailsComponent.PeriodDisplay(this.periodEnd);\n        this.storage.set('raain-periodEnd', this.periodEnd);\n        this.storage.set('raain-periodDurationInHours', durationInHours);\n        this.updateRefreshManagerPeriod();\n        this.updateCumulativeDurationInMinutes();\n    }\n\n    async onDateChangeInCount(dateChanged: Date) {\n        const dateString =\n            dateChanged.toISOString().substring(0, 11) +\n            dateChanged.toLocaleTimeString().substring(0, 5);\n        this.periodDurationAsString = '1';\n\n        if (this.toggleCumulative) {\n            // Cumulative: select periodEnd\n            this.periodEndAsString = dateString;\n            await this.onPeriodEndChange(null);\n        } else {\n            // Granular: select periodBegin\n            this.periodBeginAsString = dateString;\n            await this.onPeriodBeginChange(null);\n        }\n        await this.refreshManager.refresh(false, this.toggleAdmin);\n    }\n\n    async onDateChangeInMap(dateShown: Date) {\n        this.dateShown = dateShown;\n        await this.fetchAndUpdateMap();\n        await this.refreshManager.setReportPeriod(this.dateShown);\n    }\n\n    async onSumChangeInMap(sum: string) {\n        this.sumDetails = sum;\n    }\n\n    async onGaugeSelectInMap(mapLatLng: MapLatLng) {\n        const gaugesFiltered = this.compareManager.gaugesInMap.filter(\n            (g) => g.lat === mapLatLng.lat && g.lng === mapLatLng.lng\n        );\n        if (gaugesFiltered.length !== 1) {\n            return;\n        }\n        const gaugeSelected = gaugesFiltered[0];\n\n        await this.refreshGaugeValues({id: gaugeSelected.id, name: gaugeSelected.name});\n\n        await this.compareManager.selectGauge(gaugeSelected.id, 0);\n    }\n\n    async refreshGaugeValues(gaugeSelected: {id: string; name: string}) {\n        const gaugeValueShowBegin = new Date(\n            this.periodBegin.getTime() - RaainDetailsComponent.DAY_MS\n        );\n        const gaugeValueShowEnd = new Date(this.periodEnd.getTime() + RaainDetailsComponent.DAY_MS);\n        gaugeValueShowBegin.setHours(0, 0);\n        gaugeValueShowEnd.setHours(23, 59);\n\n        const gaugeMeasures = await this.profileService.getGaugeMeasures(\n            gaugeSelected.id,\n            gaugeValueShowBegin,\n            gaugeValueShowEnd\n        );\n        const gaugeValues: {date: Date; value: number}[] = gaugeMeasures.map((gm) => {\n            // eslint-disable-next-line @typescript-eslint/no-explicit-any\n            const cartesianMeasureValue = new CartesianMeasureValue(gm.values[0] as any);\n            return {\n                date: gm.date,\n                value: cartesianMeasureValue.getCartesianValues()[0].value,\n            };\n        });\n        this.gaugeSelectedPoints = [\n            {\n                label: gaugeSelected.name,\n                style: 'bar',\n                values: gaugeValues,\n            },\n        ];\n        this.cdr.markForCheck();\n    }\n\n    async onGaugeSelectInCompare(e: {point: XYType; compareIndex: number}) {\n        await this.refreshGaugeValues({id: e.point.id, name: e.point.name});\n\n        await this.compareManager.selectGauge(e.point.id, e.compareIndex);\n    }\n\n    async onToggleMap($event: Event) {\n        // if (this.toggleMap) {\n        //    await this.refreshMap();\n        // }\n    }\n\n    onTogglePixelMarkers() {\n        // Toggle is bound to showPixelMarkers, raain-map handles marker display\n    }\n\n    async onCumulativeToggleClick($event: Event) {\n        console.log('[CumulativeToggle] click event', {\n            currentValue: this.toggleCumulative,\n            eventType: $event.type,\n            target: ($event.target as HTMLElement)?.tagName,\n        });\n\n        $event.preventDefault();\n        $event.stopPropagation();\n\n        if (!this.toggleCumulative) {\n            console.log('[CumulativeToggle] showing popup, toggle stays OFF');\n            // Currently OFF, user wants to enable - show selector popup\n            this.showCumulativeSelector = true;\n\n            // Force reset the toggle visual state after ion-toggle's internal handler\n            setTimeout(() => {\n                if (this.cumulativeToggleRef) {\n                    this.cumulativeToggleRef.checked = false;\n                    console.log('[CumulativeToggle] force reset toggle to OFF');\n                }\n            }, 0);\n\n            this.cdr.markForCheck();\n        } else {\n            console.log('[CumulativeToggle] turning OFF');\n            // Currently ON, user wants to disable - direct toggle off\n            this.toggleCumulative = false;\n            this.storage.set('raain-toggleCumulative', false);\n            this.dateShown = this.getDateBasedOnCumulativeMode(this.timeframeDates);\n            this.updateCumulativeDurationInMinutes();\n            if (this.toggleMap) {\n                this.updateRefreshManagerPeriod();\n                await this.refreshManager.refresh(false, this.toggleAdmin);\n            }\n            this.cdr.markForCheck();\n        }\n        console.log('[CumulativeToggle] after handler, toggleCumulative =', this.toggleCumulative);\n    }\n\n    async onCumulativePeriodSelected(selection: CumulativeSelection) {\n        this.showCumulativeSelector = false;\n\n        // Update period to match selection\n        this.periodBegin = selection.periodBegin;\n        this.periodEnd = selection.periodEnd;\n        this.periodBeginAsString = RaainDetailsComponent.PeriodDisplay(this.periodBegin);\n        this.periodEndAsString = RaainDetailsComponent.PeriodDisplay(this.periodEnd);\n\n        // Calculate duration in hours\n        const durationMs = this.periodEnd.getTime() - this.periodBegin.getTime();\n        const durationHours = durationMs / RaainDetailsComponent.HOUR_MS;\n        this.periodDurationAsString = String(durationHours);\n\n        // Store cumulative period\n        this.storage.set('raain-periodBegin', this.periodBegin);\n        this.storage.set('raain-periodEnd', this.periodEnd);\n        this.storage.set('raain-periodDurationInHours', durationHours);\n\n        // Enable cumulative mode\n        this.toggleCumulative = true;\n        this.storage.set('raain-toggleCumulative', true);\n\n        this.dateShown = this.getDateBasedOnCumulativeMode(this.timeframeDates);\n        this.updateCumulativeDurationInMinutes();\n\n        // Refresh map\n        if (this.toggleMap) {\n            this.updateRefreshManagerPeriod();\n            await this.refreshManager.refresh(false, this.toggleAdmin);\n        }\n\n        this.cdr.markForCheck();\n    }\n\n    onCumulativeSelectorCancelled() {\n        console.log('[CumulativeToggle] cancelled, toggleCumulative =', this.toggleCumulative);\n        this.showCumulativeSelector = false;\n        this.cdr.markForCheck();\n    }\n\n    updateCumulativeDurationInMinutes() {\n        if (this.toggleCumulative) {\n            // Cumulative mode: use period duration\n            this.cumulativeDurationInMinutes = parseFloat(this.periodDurationAsString) * 60;\n        } else {\n            // Granular mode: use selectedTimeStep\n            this.cumulativeDurationInMinutes = this.selectedTimeStep || 10;\n        }\n    }\n\n    async onProviderChanged($event: Event) {\n        this.selectedProvider = ($event as CustomEvent)?.detail?.value;\n        this.storage.set('raain-selectedProvider', this.selectedProvider);\n        await this.applyRefreshManagerSettings();\n    }\n\n    async onTimeStepChanged($event: Event) {\n        this.selectedTimeStep = ($event as CustomEvent)?.detail?.value;\n        this.storage.set('raain-selectedTimeStep', this.selectedTimeStep);\n        this.updateCumulativeDurationInMinutes();\n        await this.applyRefreshManagerSettings();\n    }\n\n    async loadProviders() {\n        if (!this.rainNode?.id) {\n            return;\n        }\n\n        try {\n            const result = await this.profileService.getProviders(this.rainNode.id);\n            this.availableProviders = result.providers;\n            this.availableTimeSteps = result.timeStepInMinutes;\n\n            // Load saved selections or use defaults\n            this.selectedProvider =\n                this.storage.get('raain-selectedProvider') ||\n                (this.availableProviders.length > 0 ? this.availableProviders[0] : 'Raain');\n            this.selectedTimeStep =\n                this.storage.get('raain-selectedTimeStep') ||\n                (this.availableTimeSteps.length > 0 ? this.availableTimeSteps[0] : 10);\n        } catch (e) {\n            // Set fallback values\n            this.availableProviders = ['Raain'];\n            this.availableTimeSteps = [5, 10, 15, 30, 60];\n            this.selectedProvider = 'Raain';\n            this.selectedTimeStep = 10;\n        }\n\n        // Set provider and timeStep on refreshManager\n        if (this.refreshManager) {\n            this.refreshManager.provider = this.selectedProvider;\n            this.refreshManager.timeStepInMinutes = this.selectedTimeStep;\n        }\n        this.cdr.markForCheck();\n    }\n\n    onChangeDetectionTest(rainNode: RainNode) {\n        console.log(TEST_DETECTION++, 'onChangeDetectionTest');\n        return '';\n    }\n\n    updateTruncatedError() {\n        const error = this.refreshManager?.lastError || '';\n        if (error.length <= 80) {\n            this.truncatedError = error;\n            return;\n        }\n        this.truncatedError = error.substring(0, 80) + '...';\n    }\n\n    updateCachedValues() {\n        this.updatePercentOfImages();\n        this.updatePercentOfComputations();\n        this.updateTruncatedError();\n        this.updateCumulativeDurationInMinutes();\n    }\n\n    async refreshMap() {\n        this.gaugeSelectedPoints = [];\n        this.dateShown = this.getDateBasedOnCumulativeMode();\n        this.borders = [];\n\n        this.compareManager.cleanAll();\n        await this.compareManager.setGaugesInMap();\n        await this.refreshManager.refresh(false, this.toggleAdmin);\n        this.cdr.markForCheck();\n    }\n\n    async setPeriodOfNow() {\n        const last30mn = new Date();\n        last30mn.setMinutes(last30mn.getMinutes() - 30);\n        this.periodBeginAsString =\n            last30mn.toISOString().substring(0, 11) + last30mn.toLocaleTimeString().substring(0, 5);\n        this.periodDurationAsString = '1';\n        await this.onPeriodBeginChange(null);\n        await this.refreshManager.refresh(false, this.toggleAdmin);\n    }\n\n    updatePercentOfImages() {\n        if (!this.countsPeriod?.percentImages?.length) {\n            this.percentOfImages = 0;\n            return;\n        }\n\n        const duringPeriod = this.countsPeriod.percentImages.filter(\n            (a) =>\n                this.periodBegin.getTime() <= new Date(a.name).getTime() &&\n                new Date(a.name).getTime() <= this.periodEnd.getTime()\n        );\n        const summed = duringPeriod.reduce((a, b) => a + (b.x ?? 0), 0);\n        this.percentOfImages = Math.round(summed / duringPeriod.length);\n    }\n\n    updatePercentOfComputations() {\n        const timeline = this.refreshManager?.getTimelineFrameSet();\n        if (!timeline?.length) {\n            this.percentOfComputations = 0;\n            return;\n        }\n\n        const timelineWithComputation = timeline.filter(\n            (a) => !!a.rainComputationCumulativeId || !!a.rainComputationId\n        );\n        const ratio = timelineWithComputation.length / timeline.length;\n        this.percentOfComputations = Math.round(ratio * 100);\n    }\n\n    private getDateBasedOnCumulativeMode(fallbackDates?: Date[]): Date {\n        if (fallbackDates?.length > 0) {\n            const dateExists = fallbackDates.some((d) => d.getTime() === this.dateShown?.getTime());\n            if (!dateExists) {\n                return this.toggleCumulative\n                    ? fallbackDates[fallbackDates.length - 1]\n                    : fallbackDates[0];\n            }\n        }\n        return this.toggleCumulative ? this.periodEnd : this.periodBegin;\n    }\n\n    private getCumulativeHours(): number {\n        return this.toggleCumulative ? parseFloat(this.periodDurationAsString) : 0;\n    }\n\n    private getDurationInHours(): number {\n        return parseFloat(this.periodDurationAsString);\n    }\n\n    private updateRefreshManagerPeriod() {\n        this.refreshManager.cumulative = this.toggleCumulative;\n\n        // Align dates to 5-minute boundaries (floor) for consistency with raain-ground\n        const alignTo5minFloor = (date: Date): Date => {\n            const minutes = date.getMinutes();\n            const alignedMinutes = Math.floor(minutes / 5) * 5;\n            return new Date(\n                date.getFullYear(),\n                date.getMonth(),\n                date.getDate(),\n                date.getHours(),\n                alignedMinutes,\n                0,\n                0\n            );\n        };\n\n        this.refreshManager.period = {\n            begin: alignTo5minFloor(this.periodBegin),\n            end: alignTo5minFloor(this.periodEnd),\n        };\n    }\n\n    private async fetchAndUpdateMap() {\n        await this.refreshManager.fetch(\n            this.dateShown,\n            this.toggleGaugeMeasures,\n            this.getCumulativeHours()\n        );\n        this.currentTimeframeTarget = this.refreshManager.getTimelineSelectedFrameSet();\n        this.cdr.markForCheck();\n    }\n\n    private async applyRefreshManagerSettings() {\n        if (!this.refreshManager) {\n            return;\n        }\n        this.refreshManager.provider = this.selectedProvider;\n        this.refreshManager.timeStepInMinutes = this.selectedTimeStep;\n        await this.refreshManager.refresh(false, this.toggleAdmin);\n    }\n\n    private cleanAll() {\n        this.borders = [];\n        this.isAdmin = false;\n        this.timeframeContainers = new TimeframeContainers([]);\n        this.currentTimeframeTarget = null;\n        this.timeframeDates = [];\n        this.countPoints = [];\n        this.countsPeriod = {progress: 0, queueRunning: 0, percentImages: []};\n        this.gaugeSelectedPoints = [];\n\n        this.toggleHistory = false;\n        this.toggleMap = true;\n        this.toggleCompare = false;\n        this.toggleGaugeMeasures = false;\n        this.toggleRemoveCompareDuplicate = true;\n        this.toggleCumulative = this.storage.get('raain-toggleCumulative');\n\n        this.periodBegin = new Date(this.storage.get('raain-periodBegin'));\n        this.periodEnd = new Date(this.storage.get('raain-periodEnd'));\n        this.periodBeginAsString = RaainDetailsComponent.PeriodDisplay(this.periodBegin);\n        this.periodEndAsString = RaainDetailsComponent.PeriodDisplay(this.periodEnd);\n        const durationMs = this.periodEnd.getTime() - this.periodBegin.getTime();\n        this.periodDurationAsString = '' + durationMs / RaainDetailsComponent.HOUR_MS;\n\n        this.dateShown = this.getDateBasedOnCumulativeMode();\n        this.refreshInProgress = false;\n        this.showFullError = false;\n        this.showQualityModal = false;\n        this.qualityIndicators = [];\n        this.qualityIndicatorsLoading = false;\n\n        this.compareManager?.cleanAll();\n        this.refreshManager?.cleanAll();\n    }\n\n    private async init() {\n        this.cleanAll();\n        this.updateCachedValues();\n        await this.initRain();\n        this.cdr.markForCheck();\n    }\n\n    private async initRain() {\n        if (!this.rainNode) {\n            return;\n        }\n\n        this.isAdmin = this.profileService.isAdmin();\n\n        this.refreshManager.rainNode = this.rainNode;\n        this.compareManager.rainNode = this.rainNode;\n\n        // Load providers and set on refreshManager\n        await this.loadProviders();\n\n        this.refreshManager.setMethods(\n            this.onRefreshInProgress.bind(this),\n            this.onRefreshDone.bind(this),\n            this.onFetchDone.bind(this)\n        );\n\n        const center = this.rainNode.getCenter();\n        this.coordinates = new MapLatLng(center.lat, center.lng);\n        this.teamNode = await this.profileService.getTeam(\n            this.rainNode.getLink(TeamNode.TYPE).getId()\n        );\n\n        // Load all gauges linked to the rainNode on map\n        await this.compareManager.setGaugesInMap();\n\n        if (this.periodBegin && this.periodEnd) {\n            this.updateRefreshManagerPeriod();\n            await this.refreshManager.refresh(false, this.toggleAdmin);\n        }\n    }\n\n    private async onRefreshInProgress(\n        countPeriods: {\n            percentImages?: XYType[];\n            queueRunning?: number;\n            progress?: number;\n        },\n        timeframeDates: Date[]\n    ) {\n        this.refreshInProgress = true;\n        this.countsPeriod = countPeriods;\n        this.timeframeDates = timeframeDates;\n        this.updateCachedValues();\n        this.cdr.markForCheck();\n    }\n\n    private async onRefreshDone(timeframeDates: Date[]) {\n        this.timeframeDates = timeframeDates;\n        this.refreshInProgress = false;\n        this.updateCachedValues();\n        this.cdr.markForCheck();\n\n        if (this.toggleMap && timeframeDates.length > 0) {\n            this.dateShown = this.getDateBasedOnCumulativeMode(timeframeDates);\n            if (this.dateShown) {\n                await this.fetchAndUpdateMap();\n            }\n        }\n    }\n\n    private async onFetchDone(timeframeContainers?: TimeframeContainers) {\n        if (timeframeContainers) {\n            this.timeframeContainers = timeframeContainers;\n        }\n        this.cdr.markForCheck();\n    }\n\n    private async change(_changes: SimpleChanges) {\n        await this.init();\n    }\n}\n","<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n    <!-- Period selection section -->\n    <ion-card class=\"period-card\">\n        <ion-card-content>\n            <div class=\"period-controls\">\n                <div class=\"period-row\">\n\n                    <ion-button (click)=\"toggleHistory = !toggleHistory; onEnableCountHistoryTab(rainNode)\"\n                                fill=\"outline\">\n                        <ion-icon name=\"calendar-clear-outline\" slot=\"start\"></ion-icon>\n                        <ion-icon [name]=\"toggleHistory ? 'chevron-down' : 'chevron-forward'\" slot=\"end\"></ion-icon>\n                    </ion-button>\n\n                    <div class=\"period-start ion-hide-md-down\">\n                        <input (change)=\"onPeriodBeginChange($event)\"\n                               [disabled]=\"toggleCumulative\"\n                               [value]=\"periodBeginAsString\"\n                               class=\"datetime-input\"\n                               type=\"datetime-local\">\n                    </div>\n\n                    <div class=\"period-duration ion-hide-md-down\">\n                        <ion-select (ionDismiss)=\"onPeriodDurationChange($event)\"\n                                    [(ngModel)]=\"periodDurationAsString\"\n                                    [disabled]=\"toggleCumulative\"\n                                    class=\"duration-select\"\n                                    id=\"periodDuration\"\n                                    interface=\"popover\">\n                            <ion-select-option value=\"0.25\">15 minutes</ion-select-option>\n                            <ion-select-option value=\"0.5\">30 minutes</ion-select-option>\n                            <ion-select-option value=\"1\">1 hour</ion-select-option>\n                            <ion-select-option value=\"2\">2 hours</ion-select-option>\n                            <ion-select-option value=\"3\">3 hours</ion-select-option>\n                            <ion-select-option value=\"4\">4 hours</ion-select-option>\n                            <ion-select-option value=\"5\">5 hours</ion-select-option>\n                            <ion-select-option value=\"6\">6 hours</ion-select-option>\n                            <ion-select-option value=\"8\">8 hours</ion-select-option>\n                            <ion-select-option value=\"10\">10 hours</ion-select-option>\n                            <ion-select-option value=\"12\">12 hours</ion-select-option>\n                            <ion-select-option *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n                        </ion-select>\n                    </div>\n\n                    <div class=\"period-start ion-hide-md-down\">\n                        <input (change)=\"onPeriodEndChange($event)\"\n                               [disabled]=\"toggleCumulative\"\n                               [value]=\"periodEndAsString\"\n                               class=\"datetime-input\"\n                               type=\"datetime-local\">\n                    </div>\n\n                    <div class=\"toggle-cumulative\" (click)=\"onCumulativeToggleClick($event)\">\n                        <ion-label [class.text-primary]=\"toggleCumulative\">\n                            {{ toggleCumulative ? 'Cumulative' : 'Granular' }}\n                        </ion-label>\n                        <ion-toggle #cumulativeToggle [checked]=\"toggleCumulative\">\n                        </ion-toggle>\n                    </div>\n                </div>\n\n                <!-- Hidden label for change detection (uncomment to debug)\n                <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\n                -->\n            </div>\n\n            <!-- Historical map section -->\n            <div *ngIf=\"toggleHistory\" class=\"period-controls\">\n                <raain-date-dynamic (changedDate)=\"onDateChangeInCount($event)\"\n                                    [currentHeight]=\"300\"\n                                    [fetchData]=\"fetchDataWrapper\"\n                                    [points]=\"countPoints\">\n                </raain-date-dynamic>\n            </div>\n        </ion-card-content>\n    </ion-card>\n\n    <!-- Map performance -->\n    <ion-grid class=\"map-performance\">\n        <ion-row id=\"progressAndRefresh\">\n            <ion-col class=\"provider-selection\" size=\"12\" size-md=\"6\">\n                <ion-button (click)=\"openQualityModal()\" class=\"quality-info-button\" fill=\"clear\">\n                    {{ refreshManager.rainComputationMapVersion }}\n                    <ion-icon name=\"help-circle-outline\" slot=\"end\"></ion-icon>\n                </ion-button>\n                <div *ngIf=\"availableProviders.length > 0 || availableTimeSteps.length > 0\" class=\"selection-row\">\n                    <ion-select (ionChange)=\"onProviderChanged($event)\"\n                                [value]=\"selectedProvider\"\n                                interface=\"popover\"\n                                label=\"Provider\"\n                                placeholder=\"Select Provider\">\n                        <ion-select-option *ngFor=\"let provider of availableProviders\" [value]=\"provider\">\n                            {{ provider }}\n                        </ion-select-option>\n                    </ion-select>\n\n                    <ion-select (ionChange)=\"onTimeStepChanged($event)\"\n                                [value]=\"selectedTimeStep\"\n                                interface=\"popover\"\n                                label=\"Time Step\"\n                                placeholder=\"Select Time Step\">\n                        <ion-select-option *ngFor=\"let step of availableTimeSteps\" [value]=\"step\">\n                            {{ step }} min\n                        </ion-select-option>\n                    </ion-select>\n                </div>\n            </ion-col>\n            <ion-col class=\"ion-text-right\" size=\"12\" size-md=\"6\">\n                <ion-label class=\"ion-margin-end\">\n                            <span *ngIf=\"percentOfComputations\">\n                                {{ percentOfComputations }}% Images\n                                <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n                                    In Progress: {{ countsPeriod.progress }}...\n                                </i>\n                            </span>\n                    <span *ngIf=\"!percentOfComputations\">\n                                No image available\n                            </span>\n\n                </ion-label>\n\n                <ion-button (click)=\"refreshMap()\" [disabled]=\"refreshInProgress\" class=\"refresh-button\">\n                    Refresh Map\n                </ion-button>\n            </ion-col>\n        </ion-row>\n\n        <!-- status update row -->\n        <ion-row>\n            <!-- Progress col -->\n            <ion-progress-bar\n                    [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n                    [style.visibility]=\"refreshInProgress ? 'visible' : 'hidden'\"\n                    [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n                    color=\"primary\">\n            </ion-progress-bar>\n\n            <!-- Error col -->\n            <ion-col (click)=\"showFullError = !showFullError\" *ngIf=\"refreshManager.lastError\" class=\"error-row\"\n                     size=\"12\">\n                <div class=\"error-content\">\n                    <ion-icon class=\"error-icon\" name=\"warning-outline\"></ion-icon>\n                    <span [class.expanded]=\"showFullError\" class=\"error-text\">\n                        {{ showFullError ? refreshManager.lastError : truncatedError }}\n                    </span>\n                    <ion-icon [name]=\"showFullError ? 'chevron-up' : 'chevron-down'\" class=\"error-caret\"></ion-icon>\n                </div>\n            </ion-col>\n        </ion-row>\n    </ion-grid>\n\n    <!-- Map section -->\n    <ion-card class=\"map-card\">\n        <ion-card-content class=\"map-content\">\n            <ion-grid>\n                <ion-row *ngIf=\"toggleMap && percentOfImages\">\n                    <!-- Map component -->\n                    <ion-col class=\"map-column\" size-lg=\"7\" size-md=\"12\">\n                        <div class=\"map-container\">\n                            <raain-map #raainMapRef\n                                       (changedDate)=\"onDateChangeInMap($event)\"\n                                       (changedSum)=\"onSumChangeInMap($event)\"\n                                       (selectedMarker)=\"onGaugeSelectInMap($event)\"\n                                       [coordinates]=\"coordinates\"\n                                       [cumulativeDurationInMinutes]=\"cumulativeDurationInMinutes\"\n                                       [currentHeight]=\"500\"\n                                       [defaultDate]=\"dateShown\"\n                                       [markers]=\"{\n                                        borders,\n                                        gauges: compareManager.gaugesInMap,\n                                        gaugesInCompare: compareManager.gaugesInCompare,\n                                        selectedGauges: compareManager.selectedGauges,\n                                        pixels: compareManager.selectedPixels,\n                                        pixelsSolution: compareManager.pixelsSolutions?.length ? compareManager.pixelsSolutions[0] : [],\n                                        speeds: compareManager.speeds\n                                    }\"\n                                       [showCumulative]=\"toggleCumulative\"\n                                       [showVisiblePixelMarkers]=\"showPixelMarkers\"\n                                       [sumFn]=\"refreshManager.sumFn\"\n                                       [sumValues]=\"refreshManager.sumValues\"\n                                       [timeframeContainers]=\"timeframeContainers\"\n                                       [timeframeDates]=\"timeframeDates\">\n                            </raain-map>\n                        </div>\n\n                        <div class=\"data-column\">\n                            <!-- Technical details (collapsible for cleaner UI) -->\n                            <details class=\"technical-details\">\n                                <summary>Image Details</summary>\n                                <div class=\"details-content\">\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Computed:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Version:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapVersion }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Launched by:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapLaunchedBy }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Time spent:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapTimeSpentInMs }}\n                                            ms</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Date:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapDate?.toISOString() }}\n                                            | {{ refreshManager.rainComputationMapDate | date:'yyyy-MM-dd HH:mm' }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Water in the map:</span>\n                                        <span class=\"detail-value\">{{ sumDetails }}</span>\n                                        <ion-toggle\n                                                (ionChange)=\"onTogglePixelMarkers()\"\n                                                [(ngModel)]=\"showPixelMarkers\"\n                                                style=\"margin-left: 8px; transform: scale(0.7);\">\n                                        </ion-toggle>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Source DBZ min:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMin }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Source DBZ max:</span>\n                                        <span class=\"detail-value\">{{ refreshManager.rainComputationMapOriginalMax }}</span>\n                                    </div>\n                                </div>\n                            </details>\n                        </div>\n                    </ion-col>\n\n                    <!-- Data panel -->\n                    <ion-col *ngIf=\"!!compareManager.compareVersion\" class=\"data-column\" size-lg=\"5\" size-md=\"12\">\n                        <div class=\"data-panel\">\n                            <!-- Compare stack component -->\n                            <div class=\"compare-stack\">\n                                <raain-compare-stack\n                                        (selectedPoint)=\"onGaugeSelectInCompare($event)\"\n                                        [compareManager]=\"compareManager\"\n                                        [cumulative]=\"toggleCumulative\">\n                                </raain-compare-stack>\n                            </div>\n\n                            <!-- Technical details (collapsible for cleaner UI) -->\n                            <details class=\"technical-details\">\n                                <summary>Compare Details</summary>\n                                <div class=\"details-content\">\n                                    <div [ngClass]=\"{'warning': refreshManager.rainComputationMapDoneDate?.getTime() > compareManager.currentQualityDoneDate?.getTime() + 60000}\"\n                                         class=\"detail-row\">\n                                        <span class=\"detail-label\">Computed:</span>\n                                        <span class=\"detail-value\">{{ compareManager.currentQualityDoneDate | date:'yyyy-MM-dd HH:mm' }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Version:</span>\n                                        <span class=\"detail-value\">{{ compareManager.compareVersion }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Launched by:</span>\n                                        <span class=\"detail-value\">{{ compareManager.currentQualityLaunchedBy }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Time spent:</span>\n                                        <span class=\"detail-value\">{{ compareManager.currentQualityTimeSpentInMs }}\n                                            ms</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Gauges:</span>\n                                        <span class=\"detail-value\">{{ compareManager.gaugesInCompare.length }}</span>\n                                    </div>\n                                    <div class=\"detail-row\">\n                                        <span class=\"detail-label\">Points:</span>\n                                        <span class=\"detail-value\">{{ compareManager.globalComparePoints.length }}</span>\n                                    </div>\n                                </div>\n                            </details>\n                        </div>\n                    </ion-col>\n                </ion-row>\n                <ion-row>\n                    <!-- Bottom progress bar -->\n                    <ion-progress-bar *ngIf=\"refreshInProgress\"\n                                      [buffer]=\"(countsPeriod.progress / ((timeframeDates?.length || 10) +3))+0.01\"\n                                      [value]=\"countsPeriod.progress / ((timeframeDates?.length || 10) +3)\"\n                                      color=\"primary\">\n                    </ion-progress-bar>\n                </ion-row>\n            </ion-grid>\n        </ion-card-content>\n    </ion-card>\n\n    <!-- Gauge values section -->\n    <ion-card *ngIf=\"gaugeSelectedPoints.length && gaugeSelectedPoints[0].values.length\" class=\"gauge-card\">\n        <ion-card-header>\n            <ion-card-title>\n                <ion-icon name=\"analytics-outline\"></ion-icon>\n                Selected Gauge Data\n            </ion-card-title>\n        </ion-card-header>\n        <ion-card-content>\n            <raain-date-focus\n                    [currentHeight]=\"300\"\n                    [focusDate]=\"periodBegin\"\n                    [focusRange]=\"DateRange.DAY\"\n                    [points]=\"gaugeSelectedPoints\"\n                    [withoutAll]=\"true\">\n            </raain-date-focus>\n        </ion-card-content>\n    </ion-card>\n\n    <!-- Quality Performance Modal -->\n    <div (click)=\"closeQualityModal()\" *ngIf=\"showQualityModal\" class=\"quality-modal-overlay\">\n        <div (click)=\"$event.stopPropagation()\" class=\"quality-modal-content\">\n            <div class=\"quality-modal-header\">\n                <h2>Model Quality Performance</h2>\n                <ion-button (click)=\"closeQualityModal()\" fill=\"clear\">\n                    <ion-icon name=\"close-outline\"></ion-icon>\n                </ion-button>\n            </div>\n            <div class=\"quality-modal-body\">\n                <div *ngIf=\"qualityIndicatorsLoading\" class=\"quality-loading\">\n                    <ion-spinner name=\"crescent\"></ion-spinner>\n                    <span>Loading indicators...</span>\n                </div>\n                <div *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length === 0\" class=\"quality-empty\">\n                    No quality indicators available for this year.\n                </div>\n                <table *ngIf=\"!qualityIndicatorsLoading && qualityIndicators.length > 0\" class=\"quality-table\">\n                    <thead>\n                    <tr>\n                        <th>Model</th>\n                        <th>Compare</th>\n                        <th>Gauges</th>\n                        <th>Period</th>\n                        <th>Avg Quality</th>\n                        <th>Updated</th>\n                    </tr>\n                    </thead>\n                    <tbody>\n                    <tr *ngFor=\"let indicator of qualityIndicators\">\n                        <td>{{ indicator.computingVersion }}</td>\n                        <td>{{ indicator.qualityVersion }}</td>\n                        <td>{{ indicator.provider }}<br>{{ indicator.timeStepInMinutes }} min</td>\n                        <td>{{ formatDate(indicator.startDate) }}<br>{{ formatDate(indicator.endDate) }}</td>\n                        <td>{{ indicator.averageQuality | number:'1.2-2' }}</td>\n                        <td>{{ formatDate(indicator.lastUpdatedAt) }}</td>\n                    </tr>\n                    </tbody>\n                </table>\n            </div>\n        </div>\n    </div>\n\n    <!-- Cumulative Selector Modal -->\n    <cumulative-selector *ngIf=\"showCumulativeSelector\"\n                         [rainId]=\"rainNode?.id\"\n                         [currentPeriodBegin]=\"periodBegin\"\n                         [currentPeriodEnd]=\"periodEnd\"\n                         [provider]=\"selectedProvider\"\n                         [timeStepInMinutes]=\"selectedTimeStep\"\n                         [isAdmin]=\"toggleAdmin\"\n                         (periodSelected)=\"onCumulativePeriodSelected($event)\"\n                         (cancelled)=\"onCumulativeSelectorCancelled()\">\n    </cumulative-selector>\n\n</div>\n"]}