raain-app 1.5.35 → 1.5.36

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/CHANGELOG.md CHANGED
@@ -7,6 +7,12 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
7
7
 
8
8
  ## [Unreleased]
9
9
 
10
+ ### Changed
11
+
12
+ - 24h period added for admin
13
+
14
+ ## [1.5.35] - 2025-08-07
15
+
10
16
  ### Fixed
11
17
 
12
18
  - RAD-136: RefreshManager build current Frameset
@@ -282,6 +282,7 @@ export class RaainDetailsComponent {
282
282
  }
283
283
  cleanAll() {
284
284
  this.borders = [];
285
+ this.isAdmin = false;
285
286
  this.timeframeContainers = new TimeframeContainers([]);
286
287
  this.currentTimeframeTarget = null;
287
288
  this.timeframeDates = [];
@@ -312,6 +313,7 @@ export class RaainDetailsComponent {
312
313
  if (!this.rainNode) {
313
314
  return;
314
315
  }
316
+ this.isAdmin = this.profileService.isAdmin();
315
317
  this.refreshManager.rainNode = this.rainNode;
316
318
  this.compareManager.rainNode = this.rainNode;
317
319
  this.refreshManager.setMethods(this.onRefreshInProgress.bind(this), this.onRefreshDone.bind(this), this.onFetchDone.bind(this));
@@ -340,10 +342,10 @@ export class RaainDetailsComponent {
340
342
  }
341
343
  }
342
344
  RaainDetailsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RaainDetailsComponent, deps: [{ token: i1.Storage }], target: i0.ɵɵFactoryTarget.Component });
343
- 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", toggleCumulative: "toggleCumulative" }, usesOnChanges: true, ngImport: i0, template: "<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n <!-- Header section with node info\n <ion-card class=\"node-info-card\">\n <ion-card-content>\n <div class=\"node-header\">\n <div class=\"node-status\">\n <ion-icon *ngIf=\"(rainNode | hasGoodQuality)\" color=\"success\" name=\"checkmark-circle\"></ion-icon>\n <ion-icon *ngIf=\"!(rainNode | hasGoodQuality)\" color=\"warning\" name=\"alert-circle\"></ion-icon>\n </div>\n <div class=\"node-titles\">\n <ion-card-title>{{ rainNode.name }}</ion-card-title>\n <ion-card-subtitle>by {{ teamNode?.name }}</ion-card-subtitle>\n </div>\n <div class=\"node-controls\">\n <ion-label class=\"toggle-label\">History</ion-label>\n <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n [(ngModel)]=\"toggleHistory\"\n [checked]=\"toggleHistory\">\n </ion-toggle>\n </div>\n </div>\n </ion-card-content>\n </ion-card>\n-->\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 <ion-button (click)=\"setPeriodOfNow()\" class=\"now-button\" fill=\"outline\">\n <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n Now\n </ion-button>\n\n <div class=\"period-start ion-hide-md-down\">\n <ion-label>Start:</ion-label>\n <input (change)=\"onPeriodBeginChange($event)\"\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-label>Duration:</ion-label>\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>\n </div>\n\n <div id=\"all-dates\" slot=\"end\">\n <ion-label class=\"toggle-label\">All dates</ion-label>\n <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n [(ngModel)]=\"toggleHistory\"\n [checked]=\"toggleHistory\">\n </ion-toggle>\n </div>\n </div>\n\n <!-- Hidden label for change detection -->\n <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\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\n <!-- Map section -->\n <ion-card class=\"map-card\">\n <ion-card-content class=\"map-content\">\n <ion-grid>\n <ion-row id=\"progressAndRefresh\">\n <ion-label>\n <span *ngIf=\"getPercentOfComputations()\" class=\"map-stats\">\n {{ getPercentOfComputations() }}% Images\n <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n In Progress: {{ countsPeriod.progress }}...\n </i>\n </span>\n <span *ngIf=\"!getPercentOfComputations()\" class=\"no-data\">\n No image available\n </span>\n </ion-label>\n\n <ion-button (click)=\"refreshMap()\" *ngIf=\"!refreshInProgress\" class=\"refresh-button\">\n <ion-icon name=\"refresh-outline\" slot=\"start\"></ion-icon>\n Refresh Map\n </ion-button>\n\n <!-- Progress indicator -->\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-row *ngIf=\"toggleMap && getPercentOfImages()\">\n <!-- Map component -->\n <ion-col class=\"map-column\" size-lg=\"7\">\n <div class=\"map-container\">\n <raain-map\n (changedDate)=\"onDateChangeInMap($event)\"\n (changedSum)=\"onSumChangeInMap($event)\"\n (selectedMarker)=\"onGaugeSelectInMap($event)\"\n [coordinates]=\"coordinates\"\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 }\"\n [showCumulative]=\"toggleCumulative\"\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 </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 class=\"data-column\" size-lg=\"5\">\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</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)}.period-start,.period-duration{display:flex;align-items:center}.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}.map-card{background-color:var(--ion-color-light)}.map-card .map-header{display:flex;justify-content:space-between;align-items:center;flex-direction:row;gap:16px}.map-card .map-stats,.map-card .no-data{font-size:.8rem;font-weight:400;color:var(--ion-color-medium);margin-left:8px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.map-card .progress-indicator{color:var(--ion-color-primary)}.map-card .map-toggle{display:flex;align-items:center}.map-card .map-content{padding:0}.map-card .map-content ion-grid,.map-card .map-content ion-row{padding:0;margin:0}.map-card .map-content #progressAndRefresh{display:flex;justify-content:flex-end;align-items:center;padding:8px 16px}.map-card .map-content #progressAndRefresh ion-label{margin-right:16px}.map-card .map-content .map-column{padding:0}.map-card .map-content .map-column .map-container{border-radius:var(--ion-border-radius);overflow:visible}.map-card .map-content .data-column{padding:16px}.map-card .map-content .data-column .data-panel{height:100%;display:flex;flex-direction:column}.map-card .map-content .data-column .panel-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:16px}.map-card .map-content .data-column .panel-header h3{margin:0;font-size:1.2rem;font-weight:600}.map-card .map-content .data-column .computation-info,.map-card .map-content .data-column .technical-details{background-color:rgba(var(--ion-color-light-rgb),.5);border-radius:var(--ion-border-radius);padding:12px;margin-bottom:16px}.map-card .map-content .data-column .info-row,.map-card .map-content .data-column .detail-row{display:flex;margin-bottom:4px;font-size:.9rem}.map-card .map-content .data-column .info-row:last-child,.map-card .map-content .data-column .detail-row:last-child{margin-bottom:0}.map-card .map-content .data-column .compare-stack{flex:1;margin-bottom:16px}.map-card .map-content .data-column .technical-details{background-color:rgba(var(--ion-color-light-rgb),.5);border-radius:var(--ion-border-radius);padding:12px}.map-card .map-content .data-column .technical-details summary{font-weight:500;color:var(--ion-color-medium);cursor:pointer}.map-card .map-content .data-column .technical-details summary:hover{color:var(--ion-color-primary)}.map-card .map-content .data-column .technical-details .details-content{margin-top:8px;font-size:.85rem}.map-card .map-content .data-column .technical-details .details-content .detail-row{display:flex;flex-wrap:wrap;margin-bottom:4px}.map-card .map-content .data-column .technical-details .details-content .detail-row:last-child{margin-bottom:0}.map-card .map-content .data-column .technical-details .details-content .detail-row.warning{color:var(--ion-color-warning)}.map-card .map-content .data-column .technical-details .details-content .detail-row .detail-label{font-weight:500;color:var(--ion-color-dark);width:140px}.map-card .map-content .data-column .technical-details .details-content .detail-row .detail-value{color:var(--ion-color-medium)}.map-card .map-content .data-column .technical-details .details-content .detail-row .warning-text{color:var(--ion-color-danger);margin-left:8px}.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}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}}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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", "form", "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.IonToggle, selector: "ion-toggle", inputs: ["checked", "color", "disabled", "enableOnOffLabels", "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", "showCumulative", "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.DatePipe, name: "date" }] });
345
+ 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", toggleCumulative: "toggleCumulative" }, usesOnChanges: true, ngImport: i0, template: "<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n <!-- Header section with node info\n <ion-card class=\"node-info-card\">\n <ion-card-content>\n <div class=\"node-header\">\n <div class=\"node-status\">\n <ion-icon *ngIf=\"(rainNode | hasGoodQuality)\" color=\"success\" name=\"checkmark-circle\"></ion-icon>\n <ion-icon *ngIf=\"!(rainNode | hasGoodQuality)\" color=\"warning\" name=\"alert-circle\"></ion-icon>\n </div>\n <div class=\"node-titles\">\n <ion-card-title>{{ rainNode.name }}</ion-card-title>\n <ion-card-subtitle>by {{ teamNode?.name }}</ion-card-subtitle>\n </div>\n <div class=\"node-controls\">\n <ion-label class=\"toggle-label\">History</ion-label>\n <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n [(ngModel)]=\"toggleHistory\"\n [checked]=\"toggleHistory\">\n </ion-toggle>\n </div>\n </div>\n </ion-card-content>\n </ion-card>\n-->\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 <ion-button (click)=\"setPeriodOfNow()\" class=\"now-button\" fill=\"outline\">\n <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n Now\n </ion-button>\n\n <div class=\"period-start ion-hide-md-down\">\n <ion-label>Start:</ion-label>\n <input (change)=\"onPeriodBeginChange($event)\"\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-label>Duration:</ion-label>\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 *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n </ion-select>\n </div>\n\n <div id=\"all-dates\" slot=\"end\">\n <ion-label class=\"toggle-label\">All dates</ion-label>\n <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n [(ngModel)]=\"toggleHistory\"\n [checked]=\"toggleHistory\">\n </ion-toggle>\n </div>\n </div>\n\n <!-- Hidden label for change detection -->\n <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\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\n <!-- Map section -->\n <ion-card class=\"map-card\">\n <ion-card-content class=\"map-content\">\n <ion-grid>\n <ion-row id=\"progressAndRefresh\">\n <ion-label>\n <span *ngIf=\"getPercentOfComputations()\" class=\"map-stats\">\n {{ getPercentOfComputations() }}% Images\n <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n In Progress: {{ countsPeriod.progress }}...\n </i>\n </span>\n <span *ngIf=\"!getPercentOfComputations()\" class=\"no-data\">\n No image available\n </span>\n </ion-label>\n\n <ion-button (click)=\"refreshMap()\" *ngIf=\"!refreshInProgress\" class=\"refresh-button\">\n <ion-icon name=\"refresh-outline\" slot=\"start\"></ion-icon>\n Refresh Map\n </ion-button>\n\n <!-- Progress indicator -->\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-row *ngIf=\"toggleMap && getPercentOfImages()\">\n <!-- Map component -->\n <ion-col class=\"map-column\" size-lg=\"7\">\n <div class=\"map-container\">\n <raain-map\n (changedDate)=\"onDateChangeInMap($event)\"\n (changedSum)=\"onSumChangeInMap($event)\"\n (selectedMarker)=\"onGaugeSelectInMap($event)\"\n [coordinates]=\"coordinates\"\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 }\"\n [showCumulative]=\"toggleCumulative\"\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 </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 class=\"data-column\" size-lg=\"5\">\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</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)}.period-start,.period-duration{display:flex;align-items:center}.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}.map-card{background-color:var(--ion-color-light)}.map-card .map-header{display:flex;justify-content:space-between;align-items:center;flex-direction:row;gap:16px}.map-card .map-stats,.map-card .no-data{font-size:.8rem;font-weight:400;color:var(--ion-color-medium);margin-left:8px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.map-card .progress-indicator{color:var(--ion-color-primary)}.map-card .map-toggle{display:flex;align-items:center}.map-card .map-content{padding:0}.map-card .map-content ion-grid,.map-card .map-content ion-row{padding:0;margin:0}.map-card .map-content #progressAndRefresh{display:flex;justify-content:flex-end;align-items:center;padding:8px 16px}.map-card .map-content #progressAndRefresh ion-label{margin-right:16px}.map-card .map-content .map-column{padding:0}.map-card .map-content .map-column .map-container{border-radius:var(--ion-border-radius);overflow:visible}.map-card .map-content .data-column{padding:16px}.map-card .map-content .data-column .data-panel{height:100%;display:flex;flex-direction:column}.map-card .map-content .data-column .panel-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:16px}.map-card .map-content .data-column .panel-header h3{margin:0;font-size:1.2rem;font-weight:600}.map-card .map-content .data-column .computation-info,.map-card .map-content .data-column .technical-details{background-color:rgba(var(--ion-color-light-rgb),.5);border-radius:var(--ion-border-radius);padding:12px;margin-bottom:16px}.map-card .map-content .data-column .info-row,.map-card .map-content .data-column .detail-row{display:flex;margin-bottom:4px;font-size:.9rem}.map-card .map-content .data-column .info-row:last-child,.map-card .map-content .data-column .detail-row:last-child{margin-bottom:0}.map-card .map-content .data-column .compare-stack{flex:1;margin-bottom:16px}.map-card .map-content .data-column .technical-details{background-color:rgba(var(--ion-color-light-rgb),.5);border-radius:var(--ion-border-radius);padding:12px}.map-card .map-content .data-column .technical-details summary{font-weight:500;color:var(--ion-color-medium);cursor:pointer}.map-card .map-content .data-column .technical-details summary:hover{color:var(--ion-color-primary)}.map-card .map-content .data-column .technical-details .details-content{margin-top:8px;font-size:.85rem}.map-card .map-content .data-column .technical-details .details-content .detail-row{display:flex;flex-wrap:wrap;margin-bottom:4px}.map-card .map-content .data-column .technical-details .details-content .detail-row:last-child{margin-bottom:0}.map-card .map-content .data-column .technical-details .details-content .detail-row.warning{color:var(--ion-color-warning)}.map-card .map-content .data-column .technical-details .details-content .detail-row .detail-label{font-weight:500;color:var(--ion-color-dark);width:140px}.map-card .map-content .data-column .technical-details .details-content .detail-row .detail-value{color:var(--ion-color-medium)}.map-card .map-content .data-column .technical-details .details-content .detail-row .warning-text{color:var(--ion-color-danger);margin-left:8px}.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}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}}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { 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", "form", "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.IonToggle, selector: "ion-toggle", inputs: ["checked", "color", "disabled", "enableOnOffLabels", "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", "showCumulative", "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.DatePipe, name: "date" }] });
344
346
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: RaainDetailsComponent, decorators: [{
345
347
  type: Component,
346
- args: [{ selector: 'raain-details', template: "<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n <!-- Header section with node info\n <ion-card class=\"node-info-card\">\n <ion-card-content>\n <div class=\"node-header\">\n <div class=\"node-status\">\n <ion-icon *ngIf=\"(rainNode | hasGoodQuality)\" color=\"success\" name=\"checkmark-circle\"></ion-icon>\n <ion-icon *ngIf=\"!(rainNode | hasGoodQuality)\" color=\"warning\" name=\"alert-circle\"></ion-icon>\n </div>\n <div class=\"node-titles\">\n <ion-card-title>{{ rainNode.name }}</ion-card-title>\n <ion-card-subtitle>by {{ teamNode?.name }}</ion-card-subtitle>\n </div>\n <div class=\"node-controls\">\n <ion-label class=\"toggle-label\">History</ion-label>\n <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n [(ngModel)]=\"toggleHistory\"\n [checked]=\"toggleHistory\">\n </ion-toggle>\n </div>\n </div>\n </ion-card-content>\n </ion-card>\n-->\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 <ion-button (click)=\"setPeriodOfNow()\" class=\"now-button\" fill=\"outline\">\n <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n Now\n </ion-button>\n\n <div class=\"period-start ion-hide-md-down\">\n <ion-label>Start:</ion-label>\n <input (change)=\"onPeriodBeginChange($event)\"\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-label>Duration:</ion-label>\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>\n </div>\n\n <div id=\"all-dates\" slot=\"end\">\n <ion-label class=\"toggle-label\">All dates</ion-label>\n <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n [(ngModel)]=\"toggleHistory\"\n [checked]=\"toggleHistory\">\n </ion-toggle>\n </div>\n </div>\n\n <!-- Hidden label for change detection -->\n <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\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\n <!-- Map section -->\n <ion-card class=\"map-card\">\n <ion-card-content class=\"map-content\">\n <ion-grid>\n <ion-row id=\"progressAndRefresh\">\n <ion-label>\n <span *ngIf=\"getPercentOfComputations()\" class=\"map-stats\">\n {{ getPercentOfComputations() }}% Images\n <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n In Progress: {{ countsPeriod.progress }}...\n </i>\n </span>\n <span *ngIf=\"!getPercentOfComputations()\" class=\"no-data\">\n No image available\n </span>\n </ion-label>\n\n <ion-button (click)=\"refreshMap()\" *ngIf=\"!refreshInProgress\" class=\"refresh-button\">\n <ion-icon name=\"refresh-outline\" slot=\"start\"></ion-icon>\n Refresh Map\n </ion-button>\n\n <!-- Progress indicator -->\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-row *ngIf=\"toggleMap && getPercentOfImages()\">\n <!-- Map component -->\n <ion-col class=\"map-column\" size-lg=\"7\">\n <div class=\"map-container\">\n <raain-map\n (changedDate)=\"onDateChangeInMap($event)\"\n (changedSum)=\"onSumChangeInMap($event)\"\n (selectedMarker)=\"onGaugeSelectInMap($event)\"\n [coordinates]=\"coordinates\"\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 }\"\n [showCumulative]=\"toggleCumulative\"\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 </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 class=\"data-column\" size-lg=\"5\">\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</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)}.period-start,.period-duration{display:flex;align-items:center}.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}.map-card{background-color:var(--ion-color-light)}.map-card .map-header{display:flex;justify-content:space-between;align-items:center;flex-direction:row;gap:16px}.map-card .map-stats,.map-card .no-data{font-size:.8rem;font-weight:400;color:var(--ion-color-medium);margin-left:8px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.map-card .progress-indicator{color:var(--ion-color-primary)}.map-card .map-toggle{display:flex;align-items:center}.map-card .map-content{padding:0}.map-card .map-content ion-grid,.map-card .map-content ion-row{padding:0;margin:0}.map-card .map-content #progressAndRefresh{display:flex;justify-content:flex-end;align-items:center;padding:8px 16px}.map-card .map-content #progressAndRefresh ion-label{margin-right:16px}.map-card .map-content .map-column{padding:0}.map-card .map-content .map-column .map-container{border-radius:var(--ion-border-radius);overflow:visible}.map-card .map-content .data-column{padding:16px}.map-card .map-content .data-column .data-panel{height:100%;display:flex;flex-direction:column}.map-card .map-content .data-column .panel-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:16px}.map-card .map-content .data-column .panel-header h3{margin:0;font-size:1.2rem;font-weight:600}.map-card .map-content .data-column .computation-info,.map-card .map-content .data-column .technical-details{background-color:rgba(var(--ion-color-light-rgb),.5);border-radius:var(--ion-border-radius);padding:12px;margin-bottom:16px}.map-card .map-content .data-column .info-row,.map-card .map-content .data-column .detail-row{display:flex;margin-bottom:4px;font-size:.9rem}.map-card .map-content .data-column .info-row:last-child,.map-card .map-content .data-column .detail-row:last-child{margin-bottom:0}.map-card .map-content .data-column .compare-stack{flex:1;margin-bottom:16px}.map-card .map-content .data-column .technical-details{background-color:rgba(var(--ion-color-light-rgb),.5);border-radius:var(--ion-border-radius);padding:12px}.map-card .map-content .data-column .technical-details summary{font-weight:500;color:var(--ion-color-medium);cursor:pointer}.map-card .map-content .data-column .technical-details summary:hover{color:var(--ion-color-primary)}.map-card .map-content .data-column .technical-details .details-content{margin-top:8px;font-size:.85rem}.map-card .map-content .data-column .technical-details .details-content .detail-row{display:flex;flex-wrap:wrap;margin-bottom:4px}.map-card .map-content .data-column .technical-details .details-content .detail-row:last-child{margin-bottom:0}.map-card .map-content .data-column .technical-details .details-content .detail-row.warning{color:var(--ion-color-warning)}.map-card .map-content .data-column .technical-details .details-content .detail-row .detail-label{font-weight:500;color:var(--ion-color-dark);width:140px}.map-card .map-content .data-column .technical-details .details-content .detail-row .detail-value{color:var(--ion-color-medium)}.map-card .map-content .data-column .technical-details .details-content .detail-row .warning-text{color:var(--ion-color-danger);margin-left:8px}.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}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}}\n"] }]
348
+ args: [{ selector: 'raain-details', template: "<!-- Main content container -->\n<div *ngIf=\"rainNode\" class=\"raain-details-container\">\n\n <!-- Header section with node info\n <ion-card class=\"node-info-card\">\n <ion-card-content>\n <div class=\"node-header\">\n <div class=\"node-status\">\n <ion-icon *ngIf=\"(rainNode | hasGoodQuality)\" color=\"success\" name=\"checkmark-circle\"></ion-icon>\n <ion-icon *ngIf=\"!(rainNode | hasGoodQuality)\" color=\"warning\" name=\"alert-circle\"></ion-icon>\n </div>\n <div class=\"node-titles\">\n <ion-card-title>{{ rainNode.name }}</ion-card-title>\n <ion-card-subtitle>by {{ teamNode?.name }}</ion-card-subtitle>\n </div>\n <div class=\"node-controls\">\n <ion-label class=\"toggle-label\">History</ion-label>\n <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n [(ngModel)]=\"toggleHistory\"\n [checked]=\"toggleHistory\">\n </ion-toggle>\n </div>\n </div>\n </ion-card-content>\n </ion-card>\n-->\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 <ion-button (click)=\"setPeriodOfNow()\" class=\"now-button\" fill=\"outline\">\n <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n Now\n </ion-button>\n\n <div class=\"period-start ion-hide-md-down\">\n <ion-label>Start:</ion-label>\n <input (change)=\"onPeriodBeginChange($event)\"\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-label>Duration:</ion-label>\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 *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n </ion-select>\n </div>\n\n <div id=\"all-dates\" slot=\"end\">\n <ion-label class=\"toggle-label\">All dates</ion-label>\n <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n [(ngModel)]=\"toggleHistory\"\n [checked]=\"toggleHistory\">\n </ion-toggle>\n </div>\n </div>\n\n <!-- Hidden label for change detection -->\n <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\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\n <!-- Map section -->\n <ion-card class=\"map-card\">\n <ion-card-content class=\"map-content\">\n <ion-grid>\n <ion-row id=\"progressAndRefresh\">\n <ion-label>\n <span *ngIf=\"getPercentOfComputations()\" class=\"map-stats\">\n {{ getPercentOfComputations() }}% Images\n <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n In Progress: {{ countsPeriod.progress }}...\n </i>\n </span>\n <span *ngIf=\"!getPercentOfComputations()\" class=\"no-data\">\n No image available\n </span>\n </ion-label>\n\n <ion-button (click)=\"refreshMap()\" *ngIf=\"!refreshInProgress\" class=\"refresh-button\">\n <ion-icon name=\"refresh-outline\" slot=\"start\"></ion-icon>\n Refresh Map\n </ion-button>\n\n <!-- Progress indicator -->\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-row *ngIf=\"toggleMap && getPercentOfImages()\">\n <!-- Map component -->\n <ion-col class=\"map-column\" size-lg=\"7\">\n <div class=\"map-container\">\n <raain-map\n (changedDate)=\"onDateChangeInMap($event)\"\n (changedSum)=\"onSumChangeInMap($event)\"\n (selectedMarker)=\"onGaugeSelectInMap($event)\"\n [coordinates]=\"coordinates\"\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 }\"\n [showCumulative]=\"toggleCumulative\"\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 </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 class=\"data-column\" size-lg=\"5\">\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</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)}.period-start,.period-duration{display:flex;align-items:center}.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}.map-card{background-color:var(--ion-color-light)}.map-card .map-header{display:flex;justify-content:space-between;align-items:center;flex-direction:row;gap:16px}.map-card .map-stats,.map-card .no-data{font-size:.8rem;font-weight:400;color:var(--ion-color-medium);margin-left:8px;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.map-card .progress-indicator{color:var(--ion-color-primary)}.map-card .map-toggle{display:flex;align-items:center}.map-card .map-content{padding:0}.map-card .map-content ion-grid,.map-card .map-content ion-row{padding:0;margin:0}.map-card .map-content #progressAndRefresh{display:flex;justify-content:flex-end;align-items:center;padding:8px 16px}.map-card .map-content #progressAndRefresh ion-label{margin-right:16px}.map-card .map-content .map-column{padding:0}.map-card .map-content .map-column .map-container{border-radius:var(--ion-border-radius);overflow:visible}.map-card .map-content .data-column{padding:16px}.map-card .map-content .data-column .data-panel{height:100%;display:flex;flex-direction:column}.map-card .map-content .data-column .panel-header{display:flex;justify-content:space-between;align-items:center;margin-bottom:16px}.map-card .map-content .data-column .panel-header h3{margin:0;font-size:1.2rem;font-weight:600}.map-card .map-content .data-column .computation-info,.map-card .map-content .data-column .technical-details{background-color:rgba(var(--ion-color-light-rgb),.5);border-radius:var(--ion-border-radius);padding:12px;margin-bottom:16px}.map-card .map-content .data-column .info-row,.map-card .map-content .data-column .detail-row{display:flex;margin-bottom:4px;font-size:.9rem}.map-card .map-content .data-column .info-row:last-child,.map-card .map-content .data-column .detail-row:last-child{margin-bottom:0}.map-card .map-content .data-column .compare-stack{flex:1;margin-bottom:16px}.map-card .map-content .data-column .technical-details{background-color:rgba(var(--ion-color-light-rgb),.5);border-radius:var(--ion-border-radius);padding:12px}.map-card .map-content .data-column .technical-details summary{font-weight:500;color:var(--ion-color-medium);cursor:pointer}.map-card .map-content .data-column .technical-details summary:hover{color:var(--ion-color-primary)}.map-card .map-content .data-column .technical-details .details-content{margin-top:8px;font-size:.85rem}.map-card .map-content .data-column .technical-details .details-content .detail-row{display:flex;flex-wrap:wrap;margin-bottom:4px}.map-card .map-content .data-column .technical-details .details-content .detail-row:last-child{margin-bottom:0}.map-card .map-content .data-column .technical-details .details-content .detail-row.warning{color:var(--ion-color-warning)}.map-card .map-content .data-column .technical-details .details-content .detail-row .detail-label{font-weight:500;color:var(--ion-color-dark);width:140px}.map-card .map-content .data-column .technical-details .details-content .detail-row .detail-value{color:var(--ion-color-medium)}.map-card .map-content .data-column .technical-details .details-content .detail-row .warning-text{color:var(--ion-color-danger);margin-left:8px}.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}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}}\n"] }]
347
349
  }], ctorParameters: function () { return [{ type: i1.Storage }]; }, propDecorators: { toggleAdmin: [{
348
350
  type: Input
349
351
  }], rainNode: [{
@@ -359,4 +361,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
359
361
  }], toggleCumulative: [{
360
362
  type: Input
361
363
  }] } });
362
- //# 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 {Component, Input, OnChanges, OnDestroy, SimpleChanges} from '@angular/core';\nimport {ProfileService} from '../profile.service';\nimport {CartesianMeasureValue, 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\n@Component({\n    selector: 'raain-details',\n    templateUrl: 'raain-details.component.html',\n    styleUrls: ['raain-details.component.scss'],\n})\nexport class RaainDetailsComponent implements OnDestroy, OnChanges {\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    @Input()\n    public toggleCumulative: boolean;\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 refreshInProgress: boolean;\n    public sumDetails: string;\n    protected readonly DateRange = DateRange;\n\n    constructor(protected storage: Storage) {}\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    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((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n                {\n                    label: '% Images',\n                    style: 'bar',\n                    values: hourCounts.percentImages.map((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n                {\n                    label: 'Rainy Sum',\n                    style: 'line',\n                    values: hourCounts.rainySum.map((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n            ];\n        } else {\n            console.log('fetchData TODO:', focusDate.toISOString(), range);\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((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n                {\n                    label: '% Images',\n                    style: 'bar',\n                    values: counts.percentImages.map((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n                //  {\n                //      label: '% Quality',\n                //      style: 'line',\n                //      values: counts.percentQ.map((c) => {\n                //          return {\n                //              date: RaainDetailsComponent.DateUTC(c.name),\n                //              value: Math.min(100, c.x),\n                //          };\n                //      }),\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 = newValue;\n        this.storage.set('raain-periodBegin', this.periodBegin);\n\n        await this.onPeriodDurationChange(event);\n    }\n\n    async onPeriodDurationChange(_event: Event) {\n        const durationInHour = parseFloat(this.periodDurationAsString);\n        this.periodEnd = new Date(this.periodBegin.getTime() + durationInHour * (60 * 60000));\n        this.periodEndAsString = this.periodEnd.toISOString();\n        this.storage.set('raain-periodEnd', this.periodEnd);\n        this.refreshManager.period = {begin: this.periodBegin, end: this.periodEnd};\n    }\n\n    async onDateChangeInCount(dateChanged: Date) {\n        console.log('onDateChangeInCount', dateChanged);\n        this.periodBeginAsString =\n            dateChanged.toISOString().substring(0, 11) +\n            dateChanged.toLocaleTimeString().substring(0, 5);\n        this.periodDurationAsString = '1';\n        await this.onPeriodBeginChange(null);\n        this.refreshManager.refresh(false, true);\n    }\n\n    async onDateChangeInMap(dateShown: Date) {\n        // console.log('onDateChangeInMap', dateShown);\n        this.dateShown = dateShown;\n        const cumulativeHours: number = !this.toggleCumulative\n            ? 0\n            : parseFloat(this.periodDurationAsString);\n        await this.refreshManager.fetch(this.dateShown, this.toggleGaugeMeasures, cumulativeHours);\n        this.currentTimeframeTarget = this.refreshManager.getTimelineSelectedFrameSet();\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 dayBegin = new Date(this.periodBegin.toISOString().substring(0, 10));\n        const dayEnd = new Date(dayBegin.getTime() + 24 * 60 * 60 * 1000 - 1);\n        const gaugeMeasures = await this.profileService.getGaugeMeasures(\n            gaugeSelected.id,\n            dayBegin,\n            dayEnd\n        );\n        const gaugeValues: {date: Date; value: number}[] = gaugeMeasures.map((gm) => {\n            const cartesianMeasureValue = new CartesianMeasureValue(gm.values[0] as any);\n            const value = cartesianMeasureValue.getCartesianValues()[0].value;\n            return {date: gm.date, value};\n        });\n        this.gaugeSelectedPoints = [\n            {\n                label: gaugeSelected.name,\n                style: 'bar',\n                values: gaugeValues,\n            },\n        ];\n        console.log('this.gaugeSelectedPoints:', this.gaugeSelectedPoints);\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    onChangeDetectionTest(rainNode: RainNode) {\n        // TODO console.log(TEST_DETECTION++, 'onChangeDetectionTest');\n        return '';\n    }\n\n    async refreshMap() {\n        this.gaugeSelectedPoints = [];\n        // this.qualityConf.trustfulGaugeIds = this.qualityConf.trustfulGaugeIds?.length > 0 ?\n        //  this.qualityConf.trustfulGaugeIds : this.gaugesToFilter.filter(g => g.isIn).map(g => g.id);\n        this.borders = [\n            //  new MapLatLng(rainNode.latitude - 250 * 0.008, rainNode.longitude),\n            //  new MapLatLng(rainNode.latitude, rainNode.longitude - 250 * 0.013),\n            //  new MapLatLng(rainNode.latitude, rainNode.longitude + 250 * 0.013),\n            //  new MapLatLng(rainNode.latitude + 250 * 0.008, rainNode.longitude),\n        ];\n\n        this.compareManager.cleanAll();\n        await this.compareManager.setGaugesInMap();\n        this.refreshManager.refresh(false, this.toggleAdmin);\n    }\n\n    async launchComputation() {\n        await this.profileService.launchRainComputation(this.rainNode.id, this.dateShown);\n    }\n\n    async setPeriodOfNow() {\n        // should look like: \"2024-08-26T12:13\"\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        this.refreshManager.refresh(false, true);\n    }\n\n    getPercentOfImages() {\n        if (!this.countsPeriod.percentImages?.length) {\n            return 0;\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        return Math.round(summed / duringPeriod.length);\n    }\n\n    getPercentOfComputations() {\n        const timeline = this.refreshManager.getTimelineFrameSet();\n        if (!timeline?.length) {\n            return 0;\n        }\n\n        const timelineWithComputation = timeline.filter((a) => !!a.rainComputationId);\n        const ratio = timelineWithComputation.length / timeline.length;\n        return Math.round(ratio * 100);\n    }\n\n    private cleanAll() {\n        this.borders = [];\n\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\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        this.periodDurationAsString =\n            '' + (this.periodEnd.getTime() - this.periodBegin.getTime()) / (60 * 60000);\n\n        this.dateShown = this.periodBegin;\n\n        this.refreshInProgress = false;\n\n        this.compareManager?.cleanAll();\n        this.refreshManager?.cleanAll();\n    }\n\n    private async init() {\n        this.cleanAll();\n        await this.initRain();\n    }\n\n    private async initRain() {\n        if (!this.rainNode) {\n            return;\n        }\n\n        this.refreshManager.rainNode = this.rainNode;\n        this.compareManager.rainNode = this.rainNode;\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(this.rainNode.getLink('team').getId());\n\n        if (this.periodBegin && this.periodEnd) {\n            this.refreshManager.period = {begin: this.periodBegin, end: this.periodEnd};\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    }\n\n    private async onRefreshDone(timeframeDates: Date[]) {\n        this.timeframeDates = timeframeDates;\n        this.refreshInProgress = false;\n    }\n\n    private async onFetchDone(timeframeContainers: TimeframeContainers) {\n        this.timeframeContainers = timeframeContainers;\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    <!-- Header section with node info\n    <ion-card class=\"node-info-card\">\n        <ion-card-content>\n            <div class=\"node-header\">\n                <div class=\"node-status\">\n                    <ion-icon *ngIf=\"(rainNode | hasGoodQuality)\" color=\"success\" name=\"checkmark-circle\"></ion-icon>\n                    <ion-icon *ngIf=\"!(rainNode | hasGoodQuality)\" color=\"warning\" name=\"alert-circle\"></ion-icon>\n                </div>\n                <div class=\"node-titles\">\n                    <ion-card-title>{{ rainNode.name }}</ion-card-title>\n                    <ion-card-subtitle>by {{ teamNode?.name }}</ion-card-subtitle>\n                </div>\n                <div class=\"node-controls\">\n                    <ion-label class=\"toggle-label\">History</ion-label>\n                    <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n                                [(ngModel)]=\"toggleHistory\"\n                                [checked]=\"toggleHistory\">\n                    </ion-toggle>\n                </div>\n            </div>\n        </ion-card-content>\n    </ion-card>\n-->\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                    <ion-button (click)=\"setPeriodOfNow()\" class=\"now-button\" fill=\"outline\">\n                        <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n                        Now\n                    </ion-button>\n\n                    <div class=\"period-start ion-hide-md-down\">\n                        <ion-label>Start:</ion-label>\n                        <input (change)=\"onPeriodBeginChange($event)\"\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-label>Duration:</ion-label>\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>\n                    </div>\n\n                    <div id=\"all-dates\" slot=\"end\">\n                        <ion-label class=\"toggle-label\">All dates</ion-label>\n                        <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n                                    [(ngModel)]=\"toggleHistory\"\n                                    [checked]=\"toggleHistory\">\n                        </ion-toggle>\n                    </div>\n                </div>\n\n                <!-- Hidden label for change detection -->\n                <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\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\n    <!-- Map section -->\n    <ion-card class=\"map-card\">\n        <ion-card-content class=\"map-content\">\n            <ion-grid>\n                <ion-row id=\"progressAndRefresh\">\n                    <ion-label>\n                        <span *ngIf=\"getPercentOfComputations()\" class=\"map-stats\">\n                            {{ getPercentOfComputations() }}% Images\n                            <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n                                In Progress: {{ countsPeriod.progress }}...\n                            </i>\n                        </span>\n                        <span *ngIf=\"!getPercentOfComputations()\" class=\"no-data\">\n                            No image available\n                        </span>\n                    </ion-label>\n\n                    <ion-button (click)=\"refreshMap()\" *ngIf=\"!refreshInProgress\" class=\"refresh-button\">\n                        <ion-icon name=\"refresh-outline\" slot=\"start\"></ion-icon>\n                        Refresh Map\n                    </ion-button>\n\n                    <!-- Progress indicator -->\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-row *ngIf=\"toggleMap && getPercentOfImages()\">\n                    <!-- Map component -->\n                    <ion-col class=\"map-column\" size-lg=\"7\">\n                        <div class=\"map-container\">\n                            <raain-map\n                                    (changedDate)=\"onDateChangeInMap($event)\"\n                                    (changedSum)=\"onSumChangeInMap($event)\"\n                                    (selectedMarker)=\"onGaugeSelectInMap($event)\"\n                                    [coordinates]=\"coordinates\"\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                                }\"\n                                    [showCumulative]=\"toggleCumulative\"\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                                    </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 class=\"data-column\" size-lg=\"5\">\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</div>\n"]}
364
+ //# 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 {Component, Input, OnChanges, OnDestroy, SimpleChanges} from '@angular/core';\nimport {ProfileService} from '../profile.service';\nimport {CartesianMeasureValue, 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\n@Component({\n    selector: 'raain-details',\n    templateUrl: 'raain-details.component.html',\n    styleUrls: ['raain-details.component.scss'],\n})\nexport class RaainDetailsComponent implements OnDestroy, OnChanges {\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    @Input()\n    public toggleCumulative: boolean;\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 refreshInProgress: boolean;\n    public sumDetails: string;\n    public isAdmin: boolean;\n    protected readonly DateRange = DateRange;\n\n    constructor(protected storage: Storage) {}\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    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((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n                {\n                    label: '% Images',\n                    style: 'bar',\n                    values: hourCounts.percentImages.map((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n                {\n                    label: 'Rainy Sum',\n                    style: 'line',\n                    values: hourCounts.rainySum.map((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n            ];\n        } else {\n            console.log('fetchData TODO:', focusDate.toISOString(), range);\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((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n                {\n                    label: '% Images',\n                    style: 'bar',\n                    values: counts.percentImages.map((c) => {\n                        return {\n                            date: RaainDetailsComponent.DateUTC(c.name),\n                            value: Math.min(100, c.x),\n                        };\n                    }),\n                },\n                //  {\n                //      label: '% Quality',\n                //      style: 'line',\n                //      values: counts.percentQ.map((c) => {\n                //          return {\n                //              date: RaainDetailsComponent.DateUTC(c.name),\n                //              value: Math.min(100, c.x),\n                //          };\n                //      }),\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 = newValue;\n        this.storage.set('raain-periodBegin', this.periodBegin);\n\n        await this.onPeriodDurationChange(event);\n    }\n\n    async onPeriodDurationChange(_event: Event) {\n        const durationInHour = parseFloat(this.periodDurationAsString);\n        this.periodEnd = new Date(this.periodBegin.getTime() + durationInHour * (60 * 60000));\n        this.periodEndAsString = this.periodEnd.toISOString();\n        this.storage.set('raain-periodEnd', this.periodEnd);\n        this.refreshManager.period = {begin: this.periodBegin, end: this.periodEnd};\n    }\n\n    async onDateChangeInCount(dateChanged: Date) {\n        console.log('onDateChangeInCount', dateChanged);\n        this.periodBeginAsString =\n            dateChanged.toISOString().substring(0, 11) +\n            dateChanged.toLocaleTimeString().substring(0, 5);\n        this.periodDurationAsString = '1';\n        await this.onPeriodBeginChange(null);\n        this.refreshManager.refresh(false, true);\n    }\n\n    async onDateChangeInMap(dateShown: Date) {\n        // console.log('onDateChangeInMap', dateShown);\n        this.dateShown = dateShown;\n        const cumulativeHours: number = !this.toggleCumulative\n            ? 0\n            : parseFloat(this.periodDurationAsString);\n        await this.refreshManager.fetch(this.dateShown, this.toggleGaugeMeasures, cumulativeHours);\n        this.currentTimeframeTarget = this.refreshManager.getTimelineSelectedFrameSet();\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 dayBegin = new Date(this.periodBegin.toISOString().substring(0, 10));\n        const dayEnd = new Date(dayBegin.getTime() + 24 * 60 * 60 * 1000 - 1);\n        const gaugeMeasures = await this.profileService.getGaugeMeasures(\n            gaugeSelected.id,\n            dayBegin,\n            dayEnd\n        );\n        const gaugeValues: {date: Date; value: number}[] = gaugeMeasures.map((gm) => {\n            const cartesianMeasureValue = new CartesianMeasureValue(gm.values[0] as any);\n            const value = cartesianMeasureValue.getCartesianValues()[0].value;\n            return {date: gm.date, value};\n        });\n        this.gaugeSelectedPoints = [\n            {\n                label: gaugeSelected.name,\n                style: 'bar',\n                values: gaugeValues,\n            },\n        ];\n        console.log('this.gaugeSelectedPoints:', this.gaugeSelectedPoints);\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    onChangeDetectionTest(rainNode: RainNode) {\n        // TODO console.log(TEST_DETECTION++, 'onChangeDetectionTest');\n        return '';\n    }\n\n    async refreshMap() {\n        this.gaugeSelectedPoints = [];\n        // this.qualityConf.trustfulGaugeIds = this.qualityConf.trustfulGaugeIds?.length > 0 ?\n        //  this.qualityConf.trustfulGaugeIds : this.gaugesToFilter.filter(g => g.isIn).map(g => g.id);\n        this.borders = [\n            //  new MapLatLng(rainNode.latitude - 250 * 0.008, rainNode.longitude),\n            //  new MapLatLng(rainNode.latitude, rainNode.longitude - 250 * 0.013),\n            //  new MapLatLng(rainNode.latitude, rainNode.longitude + 250 * 0.013),\n            //  new MapLatLng(rainNode.latitude + 250 * 0.008, rainNode.longitude),\n        ];\n\n        this.compareManager.cleanAll();\n        await this.compareManager.setGaugesInMap();\n        this.refreshManager.refresh(false, this.toggleAdmin);\n    }\n\n    async launchComputation() {\n        await this.profileService.launchRainComputation(this.rainNode.id, this.dateShown);\n    }\n\n    async setPeriodOfNow() {\n        // should look like: \"2024-08-26T12:13\"\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        this.refreshManager.refresh(false, true);\n    }\n\n    getPercentOfImages() {\n        if (!this.countsPeriod.percentImages?.length) {\n            return 0;\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        return Math.round(summed / duringPeriod.length);\n    }\n\n    getPercentOfComputations() {\n        const timeline = this.refreshManager.getTimelineFrameSet();\n        if (!timeline?.length) {\n            return 0;\n        }\n\n        const timelineWithComputation = timeline.filter((a) => !!a.rainComputationId);\n        const ratio = timelineWithComputation.length / timeline.length;\n        return Math.round(ratio * 100);\n    }\n\n    private cleanAll() {\n        this.borders = [];\n\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\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        this.periodDurationAsString =\n            '' + (this.periodEnd.getTime() - this.periodBegin.getTime()) / (60 * 60000);\n\n        this.dateShown = this.periodBegin;\n\n        this.refreshInProgress = false;\n\n        this.compareManager?.cleanAll();\n        this.refreshManager?.cleanAll();\n    }\n\n    private async init() {\n        this.cleanAll();\n        await this.initRain();\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        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(this.rainNode.getLink('team').getId());\n\n        if (this.periodBegin && this.periodEnd) {\n            this.refreshManager.period = {begin: this.periodBegin, end: this.periodEnd};\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    }\n\n    private async onRefreshDone(timeframeDates: Date[]) {\n        this.timeframeDates = timeframeDates;\n        this.refreshInProgress = false;\n    }\n\n    private async onFetchDone(timeframeContainers: TimeframeContainers) {\n        this.timeframeContainers = timeframeContainers;\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    <!-- Header section with node info\n    <ion-card class=\"node-info-card\">\n        <ion-card-content>\n            <div class=\"node-header\">\n                <div class=\"node-status\">\n                    <ion-icon *ngIf=\"(rainNode | hasGoodQuality)\" color=\"success\" name=\"checkmark-circle\"></ion-icon>\n                    <ion-icon *ngIf=\"!(rainNode | hasGoodQuality)\" color=\"warning\" name=\"alert-circle\"></ion-icon>\n                </div>\n                <div class=\"node-titles\">\n                    <ion-card-title>{{ rainNode.name }}</ion-card-title>\n                    <ion-card-subtitle>by {{ teamNode?.name }}</ion-card-subtitle>\n                </div>\n                <div class=\"node-controls\">\n                    <ion-label class=\"toggle-label\">History</ion-label>\n                    <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n                                [(ngModel)]=\"toggleHistory\"\n                                [checked]=\"toggleHistory\">\n                    </ion-toggle>\n                </div>\n            </div>\n        </ion-card-content>\n    </ion-card>\n-->\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                    <ion-button (click)=\"setPeriodOfNow()\" class=\"now-button\" fill=\"outline\">\n                        <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n                        Now\n                    </ion-button>\n\n                    <div class=\"period-start ion-hide-md-down\">\n                        <ion-label>Start:</ion-label>\n                        <input (change)=\"onPeriodBeginChange($event)\"\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-label>Duration:</ion-label>\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 *ngIf=\"isAdmin\" value=\"24\">24 hours</ion-select-option>\n                        </ion-select>\n                    </div>\n\n                    <div id=\"all-dates\" slot=\"end\">\n                        <ion-label class=\"toggle-label\">All dates</ion-label>\n                        <ion-toggle (ionChange)=\"onEnableCountHistoryTab(rainNode)\"\n                                    [(ngModel)]=\"toggleHistory\"\n                                    [checked]=\"toggleHistory\">\n                        </ion-toggle>\n                    </div>\n                </div>\n\n                <!-- Hidden label for change detection -->\n                <div class=\"hidden-label\">{{ onChangeDetectionTest(rainNode) }}</div>\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\n    <!-- Map section -->\n    <ion-card class=\"map-card\">\n        <ion-card-content class=\"map-content\">\n            <ion-grid>\n                <ion-row id=\"progressAndRefresh\">\n                    <ion-label>\n                        <span *ngIf=\"getPercentOfComputations()\" class=\"map-stats\">\n                            {{ getPercentOfComputations() }}% Images\n                            <i *ngIf=\"countsPeriod.progress\" class=\"progress-indicator\">\n                                In Progress: {{ countsPeriod.progress }}...\n                            </i>\n                        </span>\n                        <span *ngIf=\"!getPercentOfComputations()\" class=\"no-data\">\n                            No image available\n                        </span>\n                    </ion-label>\n\n                    <ion-button (click)=\"refreshMap()\" *ngIf=\"!refreshInProgress\" class=\"refresh-button\">\n                        <ion-icon name=\"refresh-outline\" slot=\"start\"></ion-icon>\n                        Refresh Map\n                    </ion-button>\n\n                    <!-- Progress indicator -->\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-row *ngIf=\"toggleMap && getPercentOfImages()\">\n                    <!-- Map component -->\n                    <ion-col class=\"map-column\" size-lg=\"7\">\n                        <div class=\"map-container\">\n                            <raain-map\n                                    (changedDate)=\"onDateChangeInMap($event)\"\n                                    (changedSum)=\"onSumChangeInMap($event)\"\n                                    (selectedMarker)=\"onGaugeSelectInMap($event)\"\n                                    [coordinates]=\"coordinates\"\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                                }\"\n                                    [showCumulative]=\"toggleCumulative\"\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                                    </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 class=\"data-column\" size-lg=\"5\">\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</div>\n"]}