raain-app 1.6.28 → 1.6.30

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
+ ## [1.6.30] - 2026-02-24
11
+
12
+ ### Changed
13
+
14
+ - [RAD-162] Refactored cumulative-selector, raain-compare-stack, raain-details and raain-map components
15
+
10
16
  ## [1.6.28] - 2026-02-02
11
17
 
12
18
  ### Fixed
@@ -38,6 +38,8 @@ export class CumulativeSelectorComponent {
38
38
  const response = (await this.profileService.getCumulativePeriods(this.rainId, {
39
39
  provider: this.provider,
40
40
  forced: this.isAdmin,
41
+ periodBegin: this.currentPeriodBegin.toISOString(),
42
+ periodEnd: this.currentPeriodEnd.toISOString(),
41
43
  }));
42
44
  // Filter for existing custom cumulatives (window > 0)
43
45
  this.availablePeriods = response.periods.filter((p) => p.windowInMinutes > 0);
@@ -53,32 +55,22 @@ export class CumulativeSelectorComponent {
53
55
  this.cdr.markForCheck();
54
56
  }
55
57
  calculateCoverage() {
56
- if (!this.baseCumulatives) {
58
+ if (!this.baseCumulatives || !this.baseCumulatives.count) {
57
59
  this.coveragePercent = 0;
58
60
  this.canCreateNew = false;
59
61
  return;
60
62
  }
61
- const baseBegin = new Date(this.baseCumulatives.periodBegin);
62
- const baseEnd = new Date(this.baseCumulatives.periodEnd);
63
- // Check if current period is within base coverage
64
- const currentInRange = this.currentPeriodBegin >= baseBegin && this.currentPeriodEnd <= baseEnd;
65
- if (!currentInRange) {
66
- this.coveragePercent = 0;
67
- this.canCreateNew = false;
68
- return;
69
- }
70
- // Calculate expected number of base cumulatives needed
71
- const expectedCount = Math.ceil(this.currentWindowMinutes / this.timeStepInMinutes);
72
- // Check if enough base cumulatives exist
73
- // We assume coverage is complete if count >= expected (simplified check)
63
+ // Calculate expected number of base cumulatives (always 5-min granularity)
64
+ const baseTimeStep = 5;
65
+ const expectedCount = Math.ceil(this.currentWindowMinutes / baseTimeStep);
74
66
  if (this.baseCumulatives.count >= expectedCount) {
75
67
  this.coveragePercent = 100;
76
- this.canCreateNew = true;
77
68
  }
78
69
  else {
79
70
  this.coveragePercent = Math.round((this.baseCumulatives.count / expectedCount) * 100);
80
- this.canCreateNew = this.coveragePercent >= 100;
81
71
  }
72
+ // Allow creation as long as some base data exists
73
+ this.canCreateNew = this.coveragePercent > 0;
82
74
  }
83
75
  selectPeriod(period) {
84
76
  this.periodSelected.emit({
@@ -88,7 +80,7 @@ export class CumulativeSelectorComponent {
88
80
  });
89
81
  }
90
82
  async toggleExpand(period) {
91
- const key = `${period.windowInMinutes}-${period.provider}`;
83
+ const key = `${period.windowInMinutes}`;
92
84
  if (this.expandedPeriods.has(key)) {
93
85
  this.expandedPeriods.delete(key);
94
86
  this.cdr.markForCheck();
@@ -124,13 +116,13 @@ export class CumulativeSelectorComponent {
124
116
  });
125
117
  }
126
118
  isExpanded(period) {
127
- return this.expandedPeriods.has(`${period.windowInMinutes}-${period.provider}`);
119
+ return this.expandedPeriods.has(`${period.windowInMinutes}`);
128
120
  }
129
121
  isLoadingExpanded(period) {
130
- return this.loadingExpanded.has(`${period.windowInMinutes}-${period.provider}`);
122
+ return this.loadingExpanded.has(`${period.windowInMinutes}`);
131
123
  }
132
124
  getExpandedItems(period) {
133
- return this.expandedPeriods.get(`${period.windowInMinutes}-${period.provider}`) || [];
125
+ return this.expandedPeriods.get(`${period.windowInMinutes}`) || [];
134
126
  }
135
127
  getItemPeriodBegin(item) {
136
128
  const endDate = new Date(item.date);
@@ -183,7 +175,9 @@ export class CumulativeSelectorComponent {
183
175
  try {
184
176
  const progress = await this.profileService.getRainProgress(this.rainId);
185
177
  if (progress === 0) {
186
- // Queue is empty, check if cumulative exists (admin sees all cumulatives)
178
+ // Trigger background fetch from raain-ground via timeframeCumulative
179
+ await this.profileService.getRainTimeframe(this.rainId, this.currentPeriodBegin, this.currentPeriodEnd, this.isAdmin, this.provider, this.timeStepInMinutes, this.currentWindowMinutes);
180
+ // Check if cumulative now exists (admin sees all cumulatives)
187
181
  const response = (await this.profileService.getCumulativePeriods(this.rainId, {
188
182
  provider: this.provider,
189
183
  windowInMinutes: this.currentWindowMinutes,
@@ -226,10 +220,10 @@ export class CumulativeSelectorComponent {
226
220
  }
227
221
  }
228
222
  CumulativeSelectorComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CumulativeSelectorComponent, deps: [{ token: i1.ProfileService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
229
- CumulativeSelectorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CumulativeSelectorComponent, selector: "cumulative-selector", inputs: { rainId: "rainId", currentPeriodBegin: "currentPeriodBegin", currentPeriodEnd: "currentPeriodEnd", provider: "provider", timeStepInMinutes: "timeStepInMinutes", isAdmin: "isAdmin" }, outputs: { periodSelected: "periodSelected", cancelled: "cancelled" }, ngImport: i0, template: "<div class=\"cumulative-selector-overlay\">\n <div class=\"cumulative-selector-modal\">\n <div class=\"modal-header\">\n <h2>Select Cumulative Period</h2>\n <ion-button fill=\"clear\" (click)=\"cancel()\">\n <ion-icon name=\"close\"></ion-icon>\n </ion-button>\n </div>\n\n <div class=\"modal-content\">\n <!-- Loading state -->\n <div *ngIf=\"loading\" class=\"loading-state\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <p>Loading available periods...</p>\n </div>\n\n <!-- Error message -->\n <div *ngIf=\"errorMessage\" class=\"error-message\">\n <ion-icon name=\"warning-outline\"></ion-icon>\n <span>{{ errorMessage }}</span>\n </div>\n\n <!-- Available periods list -->\n <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n <h3>Available Cumulative Periods</h3>\n <ion-list>\n <ng-container *ngFor=\"let period of availablePeriods\">\n <!-- Period header (click to expand) -->\n <ion-item button (click)=\"toggleExpand(period)\" [disabled]=\"creating\">\n <ion-icon [name]=\"isExpanded(period) ? 'chevron-down' : 'chevron-forward'\" slot=\"start\"></ion-icon>\n <ion-label>\n <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n <p>{{ formatPeriod(period) }}</p>\n <p class=\"count-info\">{{ period.count }} cumulative(s) - click to expand</p>\n </ion-label>\n </ion-item>\n\n <!-- Expanded: individual cumulatives -->\n <div *ngIf=\"isExpanded(period)\" class=\"expanded-items\">\n <ion-spinner *ngIf=\"isLoadingExpanded(period)\" name=\"crescent\"></ion-spinner>\n <ion-list *ngIf=\"!isLoadingExpanded(period)\" class=\"nested-list\">\n <ion-item *ngFor=\"let item of getExpandedItems(period)\"\n button\n (click)=\"selectIndividualCumulative(item)\"\n class=\"nested-item\"\n [disabled]=\"creating\">\n <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n <ion-label>\n <h3>{{ getItemPeriodBegin(item) | date:'HH:mm' }} - {{ item.date | date:'HH:mm' }}</h3>\n </ion-label>\n <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n </ion-item>\n <ion-item *ngIf=\"getExpandedItems(period).length === 0\" class=\"nested-item\">\n <ion-label>\n <p>No individual cumulatives found</p>\n </ion-label>\n </ion-item>\n </ion-list>\n </div>\n </ng-container>\n </ion-list>\n </div>\n\n <!-- No periods available -->\n <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n <ion-icon name=\"information-circle-outline\"></ion-icon>\n <p>No cumulative periods available yet.</p>\n </div>\n\n <!-- Create new section (admin only) -->\n <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n <h3>Create New Cumulative</h3>\n <div class=\"create-info\">\n <p>\n <strong>Period:</strong>\n {{ currentPeriodBegin?.toLocaleString() }} \u2192 {{ currentPeriodEnd?.toLocaleString() }}\n </p>\n <p>\n <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n </p>\n <p *ngIf=\"baseCumulatives\" class=\"coverage-info\"\n [class.coverage-ok]=\"coveragePercent >= 100\"\n [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n [class.coverage-error]=\"coveragePercent === 0\">\n <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n Base data coverage: {{ coveragePercent }}%\n </p>\n <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n <ion-icon name=\"alert-circle\"></ion-icon>\n No base cumulatives available\n </p>\n </div>\n\n <!-- Creation progress -->\n <div *ngIf=\"creating\" class=\"creation-progress\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>{{ creationProgress }}</span>\n </div>\n\n <ion-button [disabled]=\"!canCreateNew || creating\"\n expand=\"block\"\n (click)=\"createNewPeriod()\">\n <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n </ion-button>\n\n <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n Create is disabled because base 5-min cumulatives don't fully cover the selected period.\n </p>\n </div>\n </div>\n\n <div class=\"modal-footer\">\n <ion-button fill=\"outline\" (click)=\"cancel()\" [disabled]=\"creating\">\n Cancel\n </ion-button>\n </div>\n </div>\n</div>\n", styles: [".cumulative-selector-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);display:flex;align-items:center;justify-content:center;z-index:1000}.cumulative-selector-modal{background:var(--ion-background-color, #fff);border-radius:12px;max-width:500px;width:90%;max-height:80vh;display:flex;flex-direction:column;box-shadow:0 4px 20px #0000004d}.modal-header{display:flex;justify-content:space-between;align-items:center;padding:16px 20px;border-bottom:1px solid var(--ion-border-color, #ddd)}.modal-header h2{margin:0;font-size:1.25rem;font-weight:600}.modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.modal-content{flex:1;overflow-y:auto;padding:16px 20px}.loading-state{display:flex;flex-direction:column;align-items:center;padding:40px 0}.loading-state ion-spinner{margin-bottom:16px}.loading-state p{color:var(--ion-color-medium)}.error-message{display:flex;align-items:center;gap:8px;padding:12px;background:var(--ion-color-danger-tint);color:var(--ion-color-danger);border-radius:8px;margin-bottom:16px}.error-message ion-icon{font-size:1.25rem}.periods-section{margin-bottom:24px}.periods-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.periods-section ion-list{border-radius:8px;overflow:hidden}.periods-section ion-item{--padding-start: 12px;--padding-end: 12px}.periods-section ion-item ion-label h2{font-weight:500}.periods-section ion-item ion-label p{font-size:.85rem;color:var(--ion-color-medium)}.periods-section ion-item ion-label .count-info{font-size:.75rem;color:var(--ion-color-primary)}.periods-section .expanded-items{background:var(--ion-color-light);padding:8px 0 8px 24px;border-left:3px solid var(--ion-color-primary);margin-left:16px}.periods-section .expanded-items ion-spinner{display:block;margin:16px auto}.periods-section .expanded-items .nested-list{background:transparent;padding:0}.periods-section .expanded-items .nested-list .nested-item{--background: transparent;--padding-start: 8px;--padding-top: 4px;--padding-bottom: 4px;--min-height: 32px}.periods-section .expanded-items .nested-list .nested-item ion-label{margin:0}.periods-section .expanded-items .nested-list .nested-item ion-label h3{font-size:.85rem;font-weight:500;margin:0}.periods-section .expanded-items .nested-list .nested-item ion-icon[slot=start]{color:var(--ion-color-primary);font-size:.9rem;margin-right:8px}.periods-section .expanded-items .nested-list .nested-item ion-icon[slot=end]{font-size:.8rem}.no-periods{display:flex;flex-direction:column;align-items:center;padding:24px;text-align:center;color:var(--ion-color-medium)}.no-periods ion-icon{font-size:2rem;margin-bottom:8px}.create-section{border-top:1px solid var(--ion-border-color, #ddd);padding-top:16px}.create-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.create-section .create-info{background:var(--ion-color-light);padding:12px;border-radius:8px;margin-bottom:16px}.create-section .create-info p{margin:4px 0;font-size:.9rem}.create-section .coverage-info{display:flex;align-items:center;gap:6px}.create-section .coverage-info ion-icon{font-size:1.1rem}.create-section .coverage-ok{color:var(--ion-color-success)}.create-section .coverage-warn{color:var(--ion-color-warning)}.create-section .coverage-error{color:var(--ion-color-danger)}.create-section .creation-progress{display:flex;align-items:center;gap:12px;padding:16px;background:var(--ion-color-primary-tint);border-radius:8px;margin-bottom:16px}.create-section .creation-progress ion-spinner{--color: var(--ion-color-primary)}.create-section .creation-progress span{color:var(--ion-color-primary);font-size:.9rem}.create-section .create-hint{font-size:.8rem;color:var(--ion-color-medium);text-align:center;margin-top:8px}.modal-footer{padding:16px 20px;border-top:1px solid var(--ion-border-color, #ddd);display:flex;justify-content:flex-end}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: i3.IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: i3.IonItem, selector: "ion-item", inputs: ["button", "color", "counter", "counterFormatter", "detail", "detailIcon", "disabled", "download", "fill", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "shape", "target", "type"] }, { kind: "component", type: i3.IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: i3.IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: i3.IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "pipe", type: i2.DatePipe, name: "date" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
223
+ CumulativeSelectorComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CumulativeSelectorComponent, selector: "cumulative-selector", inputs: { rainId: "rainId", currentPeriodBegin: "currentPeriodBegin", currentPeriodEnd: "currentPeriodEnd", provider: "provider", timeStepInMinutes: "timeStepInMinutes", isAdmin: "isAdmin" }, outputs: { periodSelected: "periodSelected", cancelled: "cancelled" }, ngImport: i0, template: "<div class=\"cumulative-selector-overlay\">\n <div class=\"cumulative-selector-modal\">\n <div class=\"modal-header\">\n <h2>Select Cumulative Period</h2>\n <ion-button (click)=\"cancel()\" fill=\"clear\">\n <ion-icon name=\"close\"></ion-icon>\n </ion-button>\n </div>\n\n <div class=\"modal-content\">\n <!-- Loading state -->\n <div *ngIf=\"loading\" class=\"loading-state\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <p>Loading available periods...</p>\n </div>\n\n <!-- Error message -->\n <div *ngIf=\"errorMessage\" class=\"error-message\">\n <ion-icon name=\"warning-outline\"></ion-icon>\n <span>{{ errorMessage }}</span>\n </div>\n\n <!-- Available periods list -->\n <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n <h3>Available Cumulative Periods</h3>\n <ion-list>\n <ng-container *ngFor=\"let period of availablePeriods\">\n <!-- Period header (click to expand) -->\n <ion-item (click)=\"toggleExpand(period)\" [disabled]=\"creating\" button>\n <ion-icon [name]=\"isExpanded(period) ? 'chevron-down' : 'chevron-forward'\"\n slot=\"start\"></ion-icon>\n <ion-label>\n <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n <p>{{ formatPeriod(period) }}</p>\n <p class=\"count-info\">{{ period.count }} cumulative(s) - click to expand</p>\n </ion-label>\n </ion-item>\n\n <!-- Expanded: individual cumulatives -->\n <div *ngIf=\"isExpanded(period)\" class=\"expanded-items\">\n <ion-spinner *ngIf=\"isLoadingExpanded(period)\" name=\"crescent\"></ion-spinner>\n <ion-list *ngIf=\"!isLoadingExpanded(period)\" class=\"nested-list\">\n <ion-item (click)=\"selectIndividualCumulative(item)\"\n *ngFor=\"let item of getExpandedItems(period)\"\n [disabled]=\"creating\"\n button\n class=\"nested-item\">\n <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n <ion-label>\n <h3>{{ getItemPeriodBegin(item) | date:'HH:mm' }}\n - {{ item.date | date:'HH:mm' }}</h3>\n </ion-label>\n <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n </ion-item>\n <ion-item *ngIf=\"getExpandedItems(period).length === 0\" class=\"nested-item\">\n <ion-label>\n <p>No individual cumulatives found</p>\n </ion-label>\n </ion-item>\n </ion-list>\n </div>\n </ng-container>\n </ion-list>\n </div>\n\n <!-- No periods available -->\n <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n <ion-icon name=\"information-circle-outline\"></ion-icon>\n <p>No cumulative periods available yet.</p>\n </div>\n\n <!-- Create new section (admin only) -->\n <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n <h3>Create New Cumulative</h3>\n <div class=\"create-info\">\n <p>\n <strong>Period:</strong>\n {{ currentPeriodBegin?.toLocaleString() }} \u2192 {{ currentPeriodEnd?.toLocaleString() }}\n </p>\n <p>\n <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n </p>\n <p *ngIf=\"baseCumulatives\" [class.coverage-error]=\"coveragePercent === 0\"\n [class.coverage-ok]=\"coveragePercent >= 100\"\n [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n class=\"coverage-info\">\n <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n Base data coverage: {{ coveragePercent }}%\n </p>\n <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n <ion-icon name=\"alert-circle\"></ion-icon>\n No base cumulatives available\n </p>\n </div>\n\n <!-- Creation progress -->\n <div *ngIf=\"creating\" class=\"creation-progress\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>{{ creationProgress }}</span>\n </div>\n\n <ion-button (click)=\"createNewPeriod()\" [disabled]=\"!canCreateNew || creating\"\n expand=\"block\"\n id=\"createNewCumulative\">\n <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n </ion-button>\n\n <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n Create is disabled because no base 5-min cumulatives exist for the selected period.\n </p>\n </div>\n </div>\n\n <div class=\"modal-footer\">\n <ion-button (click)=\"cancel()\" [disabled]=\"creating\" fill=\"outline\">\n Cancel\n </ion-button>\n </div>\n </div>\n</div>\n", styles: [".cumulative-selector-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);display:flex;align-items:center;justify-content:center;z-index:1000}.cumulative-selector-modal{background:var(--ion-background-color, #fff);border-radius:12px;max-width:500px;width:90%;max-height:80vh;display:flex;flex-direction:column;box-shadow:0 4px 20px #0000004d}.modal-header{display:flex;justify-content:space-between;align-items:center;padding:16px 20px;border-bottom:1px solid var(--ion-border-color, #ddd)}.modal-header h2{margin:0;font-size:1.25rem;font-weight:600}.modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.modal-content{flex:1;overflow-y:auto;padding:16px 20px}.loading-state{display:flex;flex-direction:column;align-items:center;padding:40px 0}.loading-state ion-spinner{margin-bottom:16px}.loading-state p{color:var(--ion-color-medium)}.error-message{display:flex;align-items:center;gap:8px;padding:12px;background:var(--ion-color-danger-tint);color:var(--ion-color-danger);border-radius:8px;margin-bottom:16px}.error-message ion-icon{font-size:1.25rem}.periods-section{margin-bottom:24px}.periods-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.periods-section ion-list{border-radius:8px;overflow:hidden}.periods-section ion-item{--padding-start: 12px;--padding-end: 12px}.periods-section ion-item ion-label h2{font-weight:500}.periods-section ion-item ion-label p{font-size:.85rem;color:var(--ion-color-medium)}.periods-section ion-item ion-label .count-info{font-size:.75rem;color:var(--ion-color-primary)}.periods-section .expanded-items{background:var(--ion-color-light);padding:8px 0 8px 24px;border-left:3px solid var(--ion-color-primary);margin-left:16px}.periods-section .expanded-items ion-spinner{display:block;margin:16px auto}.periods-section .expanded-items .nested-list{background:transparent;padding:0}.periods-section .expanded-items .nested-list .nested-item{--background: transparent;--padding-start: 8px;--padding-top: 4px;--padding-bottom: 4px;--min-height: 32px}.periods-section .expanded-items .nested-list .nested-item ion-label{margin:0}.periods-section .expanded-items .nested-list .nested-item ion-label h3{font-size:.85rem;font-weight:500;margin:0}.periods-section .expanded-items .nested-list .nested-item ion-icon[slot=start]{color:var(--ion-color-primary);font-size:.9rem;margin-right:8px}.periods-section .expanded-items .nested-list .nested-item ion-icon[slot=end]{font-size:.8rem}.no-periods{display:flex;flex-direction:column;align-items:center;padding:24px;text-align:center;color:var(--ion-color-medium)}.no-periods ion-icon{font-size:2rem;margin-bottom:8px}.create-section{border-top:1px solid var(--ion-border-color, #ddd);padding-top:16px}.create-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.create-section .create-info{background:var(--ion-color-light);padding:12px;border-radius:8px;margin-bottom:16px}.create-section .create-info p{margin:4px 0;font-size:.9rem}.create-section .coverage-info{display:flex;align-items:center;gap:6px}.create-section .coverage-info ion-icon{font-size:1.1rem}.create-section .coverage-ok{color:var(--ion-color-success)}.create-section .coverage-warn{color:var(--ion-color-warning)}.create-section .coverage-error{color:var(--ion-color-danger)}.create-section .creation-progress{display:flex;align-items:center;gap:12px;padding:16px;background:var(--ion-color-primary-tint);border-radius:8px;margin-bottom:16px}.create-section .creation-progress ion-spinner{--color: var(--ion-color-primary)}.create-section .creation-progress span{color:var(--ion-color-primary);font-size:.9rem}.create-section .create-hint{font-size:.8rem;color:var(--ion-color-medium);text-align:center;margin-top:8px}.modal-footer{padding:16px 20px;border-top:1px solid var(--ion-border-color, #ddd);display:flex;justify-content:flex-end}\n"], dependencies: [{ kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: i3.IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: i3.IonItem, selector: "ion-item", inputs: ["button", "color", "counter", "counterFormatter", "detail", "detailIcon", "disabled", "download", "fill", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "shape", "target", "type"] }, { kind: "component", type: i3.IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: i3.IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: i3.IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }, { kind: "pipe", type: i2.DatePipe, name: "date" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
230
224
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CumulativeSelectorComponent, decorators: [{
231
225
  type: Component,
232
- args: [{ selector: 'cumulative-selector', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"cumulative-selector-overlay\">\n <div class=\"cumulative-selector-modal\">\n <div class=\"modal-header\">\n <h2>Select Cumulative Period</h2>\n <ion-button fill=\"clear\" (click)=\"cancel()\">\n <ion-icon name=\"close\"></ion-icon>\n </ion-button>\n </div>\n\n <div class=\"modal-content\">\n <!-- Loading state -->\n <div *ngIf=\"loading\" class=\"loading-state\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <p>Loading available periods...</p>\n </div>\n\n <!-- Error message -->\n <div *ngIf=\"errorMessage\" class=\"error-message\">\n <ion-icon name=\"warning-outline\"></ion-icon>\n <span>{{ errorMessage }}</span>\n </div>\n\n <!-- Available periods list -->\n <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n <h3>Available Cumulative Periods</h3>\n <ion-list>\n <ng-container *ngFor=\"let period of availablePeriods\">\n <!-- Period header (click to expand) -->\n <ion-item button (click)=\"toggleExpand(period)\" [disabled]=\"creating\">\n <ion-icon [name]=\"isExpanded(period) ? 'chevron-down' : 'chevron-forward'\" slot=\"start\"></ion-icon>\n <ion-label>\n <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n <p>{{ formatPeriod(period) }}</p>\n <p class=\"count-info\">{{ period.count }} cumulative(s) - click to expand</p>\n </ion-label>\n </ion-item>\n\n <!-- Expanded: individual cumulatives -->\n <div *ngIf=\"isExpanded(period)\" class=\"expanded-items\">\n <ion-spinner *ngIf=\"isLoadingExpanded(period)\" name=\"crescent\"></ion-spinner>\n <ion-list *ngIf=\"!isLoadingExpanded(period)\" class=\"nested-list\">\n <ion-item *ngFor=\"let item of getExpandedItems(period)\"\n button\n (click)=\"selectIndividualCumulative(item)\"\n class=\"nested-item\"\n [disabled]=\"creating\">\n <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n <ion-label>\n <h3>{{ getItemPeriodBegin(item) | date:'HH:mm' }} - {{ item.date | date:'HH:mm' }}</h3>\n </ion-label>\n <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n </ion-item>\n <ion-item *ngIf=\"getExpandedItems(period).length === 0\" class=\"nested-item\">\n <ion-label>\n <p>No individual cumulatives found</p>\n </ion-label>\n </ion-item>\n </ion-list>\n </div>\n </ng-container>\n </ion-list>\n </div>\n\n <!-- No periods available -->\n <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n <ion-icon name=\"information-circle-outline\"></ion-icon>\n <p>No cumulative periods available yet.</p>\n </div>\n\n <!-- Create new section (admin only) -->\n <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n <h3>Create New Cumulative</h3>\n <div class=\"create-info\">\n <p>\n <strong>Period:</strong>\n {{ currentPeriodBegin?.toLocaleString() }} \u2192 {{ currentPeriodEnd?.toLocaleString() }}\n </p>\n <p>\n <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n </p>\n <p *ngIf=\"baseCumulatives\" class=\"coverage-info\"\n [class.coverage-ok]=\"coveragePercent >= 100\"\n [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n [class.coverage-error]=\"coveragePercent === 0\">\n <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n Base data coverage: {{ coveragePercent }}%\n </p>\n <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n <ion-icon name=\"alert-circle\"></ion-icon>\n No base cumulatives available\n </p>\n </div>\n\n <!-- Creation progress -->\n <div *ngIf=\"creating\" class=\"creation-progress\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>{{ creationProgress }}</span>\n </div>\n\n <ion-button [disabled]=\"!canCreateNew || creating\"\n expand=\"block\"\n (click)=\"createNewPeriod()\">\n <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n </ion-button>\n\n <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n Create is disabled because base 5-min cumulatives don't fully cover the selected period.\n </p>\n </div>\n </div>\n\n <div class=\"modal-footer\">\n <ion-button fill=\"outline\" (click)=\"cancel()\" [disabled]=\"creating\">\n Cancel\n </ion-button>\n </div>\n </div>\n</div>\n", styles: [".cumulative-selector-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);display:flex;align-items:center;justify-content:center;z-index:1000}.cumulative-selector-modal{background:var(--ion-background-color, #fff);border-radius:12px;max-width:500px;width:90%;max-height:80vh;display:flex;flex-direction:column;box-shadow:0 4px 20px #0000004d}.modal-header{display:flex;justify-content:space-between;align-items:center;padding:16px 20px;border-bottom:1px solid var(--ion-border-color, #ddd)}.modal-header h2{margin:0;font-size:1.25rem;font-weight:600}.modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.modal-content{flex:1;overflow-y:auto;padding:16px 20px}.loading-state{display:flex;flex-direction:column;align-items:center;padding:40px 0}.loading-state ion-spinner{margin-bottom:16px}.loading-state p{color:var(--ion-color-medium)}.error-message{display:flex;align-items:center;gap:8px;padding:12px;background:var(--ion-color-danger-tint);color:var(--ion-color-danger);border-radius:8px;margin-bottom:16px}.error-message ion-icon{font-size:1.25rem}.periods-section{margin-bottom:24px}.periods-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.periods-section ion-list{border-radius:8px;overflow:hidden}.periods-section ion-item{--padding-start: 12px;--padding-end: 12px}.periods-section ion-item ion-label h2{font-weight:500}.periods-section ion-item ion-label p{font-size:.85rem;color:var(--ion-color-medium)}.periods-section ion-item ion-label .count-info{font-size:.75rem;color:var(--ion-color-primary)}.periods-section .expanded-items{background:var(--ion-color-light);padding:8px 0 8px 24px;border-left:3px solid var(--ion-color-primary);margin-left:16px}.periods-section .expanded-items ion-spinner{display:block;margin:16px auto}.periods-section .expanded-items .nested-list{background:transparent;padding:0}.periods-section .expanded-items .nested-list .nested-item{--background: transparent;--padding-start: 8px;--padding-top: 4px;--padding-bottom: 4px;--min-height: 32px}.periods-section .expanded-items .nested-list .nested-item ion-label{margin:0}.periods-section .expanded-items .nested-list .nested-item ion-label h3{font-size:.85rem;font-weight:500;margin:0}.periods-section .expanded-items .nested-list .nested-item ion-icon[slot=start]{color:var(--ion-color-primary);font-size:.9rem;margin-right:8px}.periods-section .expanded-items .nested-list .nested-item ion-icon[slot=end]{font-size:.8rem}.no-periods{display:flex;flex-direction:column;align-items:center;padding:24px;text-align:center;color:var(--ion-color-medium)}.no-periods ion-icon{font-size:2rem;margin-bottom:8px}.create-section{border-top:1px solid var(--ion-border-color, #ddd);padding-top:16px}.create-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.create-section .create-info{background:var(--ion-color-light);padding:12px;border-radius:8px;margin-bottom:16px}.create-section .create-info p{margin:4px 0;font-size:.9rem}.create-section .coverage-info{display:flex;align-items:center;gap:6px}.create-section .coverage-info ion-icon{font-size:1.1rem}.create-section .coverage-ok{color:var(--ion-color-success)}.create-section .coverage-warn{color:var(--ion-color-warning)}.create-section .coverage-error{color:var(--ion-color-danger)}.create-section .creation-progress{display:flex;align-items:center;gap:12px;padding:16px;background:var(--ion-color-primary-tint);border-radius:8px;margin-bottom:16px}.create-section .creation-progress ion-spinner{--color: var(--ion-color-primary)}.create-section .creation-progress span{color:var(--ion-color-primary);font-size:.9rem}.create-section .create-hint{font-size:.8rem;color:var(--ion-color-medium);text-align:center;margin-top:8px}.modal-footer{padding:16px 20px;border-top:1px solid var(--ion-border-color, #ddd);display:flex;justify-content:flex-end}\n"] }]
226
+ args: [{ selector: 'cumulative-selector', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"cumulative-selector-overlay\">\n <div class=\"cumulative-selector-modal\">\n <div class=\"modal-header\">\n <h2>Select Cumulative Period</h2>\n <ion-button (click)=\"cancel()\" fill=\"clear\">\n <ion-icon name=\"close\"></ion-icon>\n </ion-button>\n </div>\n\n <div class=\"modal-content\">\n <!-- Loading state -->\n <div *ngIf=\"loading\" class=\"loading-state\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <p>Loading available periods...</p>\n </div>\n\n <!-- Error message -->\n <div *ngIf=\"errorMessage\" class=\"error-message\">\n <ion-icon name=\"warning-outline\"></ion-icon>\n <span>{{ errorMessage }}</span>\n </div>\n\n <!-- Available periods list -->\n <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n <h3>Available Cumulative Periods</h3>\n <ion-list>\n <ng-container *ngFor=\"let period of availablePeriods\">\n <!-- Period header (click to expand) -->\n <ion-item (click)=\"toggleExpand(period)\" [disabled]=\"creating\" button>\n <ion-icon [name]=\"isExpanded(period) ? 'chevron-down' : 'chevron-forward'\"\n slot=\"start\"></ion-icon>\n <ion-label>\n <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n <p>{{ formatPeriod(period) }}</p>\n <p class=\"count-info\">{{ period.count }} cumulative(s) - click to expand</p>\n </ion-label>\n </ion-item>\n\n <!-- Expanded: individual cumulatives -->\n <div *ngIf=\"isExpanded(period)\" class=\"expanded-items\">\n <ion-spinner *ngIf=\"isLoadingExpanded(period)\" name=\"crescent\"></ion-spinner>\n <ion-list *ngIf=\"!isLoadingExpanded(period)\" class=\"nested-list\">\n <ion-item (click)=\"selectIndividualCumulative(item)\"\n *ngFor=\"let item of getExpandedItems(period)\"\n [disabled]=\"creating\"\n button\n class=\"nested-item\">\n <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n <ion-label>\n <h3>{{ getItemPeriodBegin(item) | date:'HH:mm' }}\n - {{ item.date | date:'HH:mm' }}</h3>\n </ion-label>\n <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n </ion-item>\n <ion-item *ngIf=\"getExpandedItems(period).length === 0\" class=\"nested-item\">\n <ion-label>\n <p>No individual cumulatives found</p>\n </ion-label>\n </ion-item>\n </ion-list>\n </div>\n </ng-container>\n </ion-list>\n </div>\n\n <!-- No periods available -->\n <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n <ion-icon name=\"information-circle-outline\"></ion-icon>\n <p>No cumulative periods available yet.</p>\n </div>\n\n <!-- Create new section (admin only) -->\n <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n <h3>Create New Cumulative</h3>\n <div class=\"create-info\">\n <p>\n <strong>Period:</strong>\n {{ currentPeriodBegin?.toLocaleString() }} \u2192 {{ currentPeriodEnd?.toLocaleString() }}\n </p>\n <p>\n <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n </p>\n <p *ngIf=\"baseCumulatives\" [class.coverage-error]=\"coveragePercent === 0\"\n [class.coverage-ok]=\"coveragePercent >= 100\"\n [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n class=\"coverage-info\">\n <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n Base data coverage: {{ coveragePercent }}%\n </p>\n <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n <ion-icon name=\"alert-circle\"></ion-icon>\n No base cumulatives available\n </p>\n </div>\n\n <!-- Creation progress -->\n <div *ngIf=\"creating\" class=\"creation-progress\">\n <ion-spinner name=\"crescent\"></ion-spinner>\n <span>{{ creationProgress }}</span>\n </div>\n\n <ion-button (click)=\"createNewPeriod()\" [disabled]=\"!canCreateNew || creating\"\n expand=\"block\"\n id=\"createNewCumulative\">\n <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n </ion-button>\n\n <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n Create is disabled because no base 5-min cumulatives exist for the selected period.\n </p>\n </div>\n </div>\n\n <div class=\"modal-footer\">\n <ion-button (click)=\"cancel()\" [disabled]=\"creating\" fill=\"outline\">\n Cancel\n </ion-button>\n </div>\n </div>\n</div>\n", styles: [".cumulative-selector-overlay{position:fixed;top:0;left:0;width:100%;height:100%;background:rgba(0,0,0,.5);display:flex;align-items:center;justify-content:center;z-index:1000}.cumulative-selector-modal{background:var(--ion-background-color, #fff);border-radius:12px;max-width:500px;width:90%;max-height:80vh;display:flex;flex-direction:column;box-shadow:0 4px 20px #0000004d}.modal-header{display:flex;justify-content:space-between;align-items:center;padding:16px 20px;border-bottom:1px solid var(--ion-border-color, #ddd)}.modal-header h2{margin:0;font-size:1.25rem;font-weight:600}.modal-header ion-button{--padding-start: 8px;--padding-end: 8px}.modal-content{flex:1;overflow-y:auto;padding:16px 20px}.loading-state{display:flex;flex-direction:column;align-items:center;padding:40px 0}.loading-state ion-spinner{margin-bottom:16px}.loading-state p{color:var(--ion-color-medium)}.error-message{display:flex;align-items:center;gap:8px;padding:12px;background:var(--ion-color-danger-tint);color:var(--ion-color-danger);border-radius:8px;margin-bottom:16px}.error-message ion-icon{font-size:1.25rem}.periods-section{margin-bottom:24px}.periods-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.periods-section ion-list{border-radius:8px;overflow:hidden}.periods-section ion-item{--padding-start: 12px;--padding-end: 12px}.periods-section ion-item ion-label h2{font-weight:500}.periods-section ion-item ion-label p{font-size:.85rem;color:var(--ion-color-medium)}.periods-section ion-item ion-label .count-info{font-size:.75rem;color:var(--ion-color-primary)}.periods-section .expanded-items{background:var(--ion-color-light);padding:8px 0 8px 24px;border-left:3px solid var(--ion-color-primary);margin-left:16px}.periods-section .expanded-items ion-spinner{display:block;margin:16px auto}.periods-section .expanded-items .nested-list{background:transparent;padding:0}.periods-section .expanded-items .nested-list .nested-item{--background: transparent;--padding-start: 8px;--padding-top: 4px;--padding-bottom: 4px;--min-height: 32px}.periods-section .expanded-items .nested-list .nested-item ion-label{margin:0}.periods-section .expanded-items .nested-list .nested-item ion-label h3{font-size:.85rem;font-weight:500;margin:0}.periods-section .expanded-items .nested-list .nested-item ion-icon[slot=start]{color:var(--ion-color-primary);font-size:.9rem;margin-right:8px}.periods-section .expanded-items .nested-list .nested-item ion-icon[slot=end]{font-size:.8rem}.no-periods{display:flex;flex-direction:column;align-items:center;padding:24px;text-align:center;color:var(--ion-color-medium)}.no-periods ion-icon{font-size:2rem;margin-bottom:8px}.create-section{border-top:1px solid var(--ion-border-color, #ddd);padding-top:16px}.create-section h3{font-size:1rem;font-weight:600;margin-bottom:12px;color:var(--ion-color-dark)}.create-section .create-info{background:var(--ion-color-light);padding:12px;border-radius:8px;margin-bottom:16px}.create-section .create-info p{margin:4px 0;font-size:.9rem}.create-section .coverage-info{display:flex;align-items:center;gap:6px}.create-section .coverage-info ion-icon{font-size:1.1rem}.create-section .coverage-ok{color:var(--ion-color-success)}.create-section .coverage-warn{color:var(--ion-color-warning)}.create-section .coverage-error{color:var(--ion-color-danger)}.create-section .creation-progress{display:flex;align-items:center;gap:12px;padding:16px;background:var(--ion-color-primary-tint);border-radius:8px;margin-bottom:16px}.create-section .creation-progress ion-spinner{--color: var(--ion-color-primary)}.create-section .creation-progress span{color:var(--ion-color-primary);font-size:.9rem}.create-section .create-hint{font-size:.8rem;color:var(--ion-color-medium);text-align:center;margin-top:8px}.modal-footer{padding:16px 20px;border-top:1px solid var(--ion-border-color, #ddd);display:flex;justify-content:flex-end}\n"] }]
233
227
  }], ctorParameters: function () { return [{ type: i1.ProfileService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { rainId: [{
234
228
  type: Input
235
229
  }], currentPeriodBegin: [{
@@ -247,4 +241,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImpo
247
241
  }], cancelled: [{
248
242
  type: Output
249
243
  }] } });
250
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cumulative-selector.component.js","sourceRoot":"","sources":["../../../src/core/shared/cumulative-selector/cumulative-selector.component.ts","../../../src/core/shared/cumulative-selector/cumulative-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    Input,\n    OnInit,\n    Output,\n} from '@angular/core';\nimport {ProfileService} from '../profile.service';\nimport {\n    CumulativePeriod,\n    IndividualCumulative,\n    RaainApiRainsCumulativesDetailedResponse,\n    RaainApiRainsCumulativesListResponse,\n} from 'raain-model';\n\nexport interface CumulativeSelection {\n    periodBegin: Date;\n    periodEnd: Date;\n    windowInMinutes: number;\n    cumulativeId?: string;\n    cumulativeDate?: Date;\n}\n\n@Component({\n    selector: 'cumulative-selector',\n    templateUrl: './cumulative-selector.component.html',\n    styleUrls: ['./cumulative-selector.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CumulativeSelectorComponent implements OnInit {\n    @Input() rainId: string;\n    @Input() currentPeriodBegin: Date;\n    @Input() currentPeriodEnd: Date;\n    @Input() provider: string;\n    @Input() timeStepInMinutes: number = 5;\n    @Input() isAdmin: boolean = false;\n\n    @Output() periodSelected = new EventEmitter<CumulativeSelection>();\n    @Output() cancelled = new EventEmitter<void>();\n\n    availablePeriods: CumulativePeriod[] = [];\n    baseCumulatives: CumulativePeriod = null;\n    loading = true;\n    creating = false;\n    creationProgress = '';\n    errorMessage = '';\n\n    coveragePercent = 0;\n    canCreateNew = false;\n    currentWindowMinutes = 0;\n\n    expandedPeriods: Map<string, IndividualCumulative[]> = new Map();\n    loadingExpanded: Set<string> = new Set();\n\n    private readonly POLL_TIMEOUT_MS = 900000; // 900 seconds\n    private readonly POLL_INTERVAL_MS = 3000;\n\n    constructor(\n        private profileService: ProfileService,\n        private cdr: ChangeDetectorRef\n    ) {}\n\n    async ngOnInit() {\n        this.currentWindowMinutes = Math.round(\n            (this.currentPeriodEnd.getTime() - this.currentPeriodBegin.getTime()) / 60000\n        );\n        await this.loadAvailablePeriods();\n    }\n\n    async loadAvailablePeriods() {\n        this.loading = true;\n        this.errorMessage = '';\n        this.cdr.markForCheck();\n\n        try {\n            // Fetch all cumulative periods (admin sees all, non-admin sees only customer-launched)\n            const response = (await this.profileService.getCumulativePeriods(this.rainId, {\n                provider: this.provider,\n                forced: this.isAdmin,\n            })) as RaainApiRainsCumulativesListResponse;\n\n            // Filter for existing custom cumulatives (window > 0)\n            this.availablePeriods = response.periods.filter((p) => p.windowInMinutes > 0);\n\n            // Get base cumulatives (window = 0) to check coverage\n            this.baseCumulatives = response.periods.find((p) => p.windowInMinutes === 0);\n\n            this.calculateCoverage();\n        } catch (e) {\n            this.errorMessage = 'Failed to load cumulative periods';\n            console.error('Error loading cumulative periods:', e);\n        }\n\n        this.loading = false;\n        this.cdr.markForCheck();\n    }\n\n    calculateCoverage() {\n        if (!this.baseCumulatives) {\n            this.coveragePercent = 0;\n            this.canCreateNew = false;\n            return;\n        }\n\n        const baseBegin = new Date(this.baseCumulatives.periodBegin);\n        const baseEnd = new Date(this.baseCumulatives.periodEnd);\n\n        // Check if current period is within base coverage\n        const currentInRange =\n            this.currentPeriodBegin >= baseBegin && this.currentPeriodEnd <= baseEnd;\n\n        if (!currentInRange) {\n            this.coveragePercent = 0;\n            this.canCreateNew = false;\n            return;\n        }\n\n        // Calculate expected number of base cumulatives needed\n        const expectedCount = Math.ceil(this.currentWindowMinutes / this.timeStepInMinutes);\n\n        // Check if enough base cumulatives exist\n        // We assume coverage is complete if count >= expected (simplified check)\n        if (this.baseCumulatives.count >= expectedCount) {\n            this.coveragePercent = 100;\n            this.canCreateNew = true;\n        } else {\n            this.coveragePercent = Math.round((this.baseCumulatives.count / expectedCount) * 100);\n            this.canCreateNew = this.coveragePercent >= 100;\n        }\n    }\n\n    selectPeriod(period: CumulativePeriod) {\n        this.periodSelected.emit({\n            periodBegin: new Date(period.periodBegin),\n            periodEnd: new Date(period.periodEnd),\n            windowInMinutes: period.windowInMinutes,\n        });\n    }\n\n    async toggleExpand(period: CumulativePeriod) {\n        const key = `${period.windowInMinutes}-${period.provider}`;\n        if (this.expandedPeriods.has(key)) {\n            this.expandedPeriods.delete(key);\n            this.cdr.markForCheck();\n            return;\n        }\n\n        this.loadingExpanded.add(key);\n        this.cdr.markForCheck();\n\n        try {\n            const response = (await this.profileService.getCumulativePeriods(this.rainId, {\n                provider: this.provider,\n                windowInMinutes: period.windowInMinutes,\n                forced: this.isAdmin,\n                detailed: true,\n            })) as RaainApiRainsCumulativesDetailedResponse;\n\n            this.expandedPeriods.set(key, response.cumulatives || []);\n        } catch (e) {\n            console.error('Error loading individual cumulatives:', e);\n            this.expandedPeriods.set(key, []);\n        }\n\n        this.loadingExpanded.delete(key);\n        this.cdr.markForCheck();\n    }\n\n    selectIndividualCumulative(item: IndividualCumulative) {\n        const endDate = new Date(item.date);\n        const beginDate = new Date(endDate.getTime() - item.windowInMinutes * 60000);\n\n        this.periodSelected.emit({\n            periodBegin: beginDate,\n            periodEnd: endDate,\n            windowInMinutes: item.windowInMinutes,\n            cumulativeId: item.id,\n            cumulativeDate: endDate,\n        });\n    }\n\n    isExpanded(period: CumulativePeriod): boolean {\n        return this.expandedPeriods.has(`${period.windowInMinutes}-${period.provider}`);\n    }\n\n    isLoadingExpanded(period: CumulativePeriod): boolean {\n        return this.loadingExpanded.has(`${period.windowInMinutes}-${period.provider}`);\n    }\n\n    getExpandedItems(period: CumulativePeriod): IndividualCumulative[] {\n        return this.expandedPeriods.get(`${period.windowInMinutes}-${period.provider}`) || [];\n    }\n\n    getItemPeriodBegin(item: IndividualCumulative): Date {\n        const endDate = new Date(item.date);\n        return new Date(endDate.getTime() - item.windowInMinutes * 60000);\n    }\n\n    async createNewPeriod() {\n        if (!this.canCreateNew || this.creating) {\n            return;\n        }\n\n        this.creating = true;\n        this.creationProgress = 'Starting cumulative computation...';\n        this.errorMessage = '';\n        this.cdr.markForCheck();\n\n        try {\n            // Trigger cumulative creation\n            const result = await this.profileService.createCumulativePeriod(this.rainId, {\n                periodBegin: this.currentPeriodBegin,\n                periodEnd: this.currentPeriodEnd,\n                provider: this.provider,\n                timeStepInMinutes: this.timeStepInMinutes,\n            });\n\n            if (!result) {\n                throw new Error('Failed to trigger cumulative computation');\n            }\n\n            this.creationProgress = `Jobs queued. Polling for completion...`;\n            this.cdr.markForCheck();\n\n            // Poll for completion\n            const success = await this.pollForCompletion();\n\n            if (success) {\n                this.periodSelected.emit({\n                    periodBegin: this.currentPeriodBegin,\n                    periodEnd: this.currentPeriodEnd,\n                    windowInMinutes: this.currentWindowMinutes,\n                });\n            } else {\n                this.errorMessage = 'Timeout waiting for cumulative computation';\n            }\n        } catch (e) {\n            this.errorMessage = `Error: ${e.message || 'Unknown error'}`;\n            console.error('Error creating cumulative:', e);\n        }\n\n        this.creating = false;\n        this.cdr.markForCheck();\n    }\n\n    private async pollForCompletion(): Promise<boolean> {\n        const startTime = Date.now();\n\n        while (Date.now() - startTime < this.POLL_TIMEOUT_MS) {\n            try {\n                const progress = await this.profileService.getRainProgress(this.rainId);\n\n                if (progress === 0) {\n                    // Queue is empty, check if cumulative exists (admin sees all cumulatives)\n                    const response = (await this.profileService.getCumulativePeriods(this.rainId, {\n                        provider: this.provider,\n                        windowInMinutes: this.currentWindowMinutes,\n                        forced: this.isAdmin,\n                    })) as RaainApiRainsCumulativesListResponse;\n\n                    const found = response.periods.find(\n                        (p) => p.windowInMinutes === this.currentWindowMinutes\n                    );\n\n                    if (found) {\n                        return true;\n                    }\n                }\n\n                const elapsed = Math.round((Date.now() - startTime) / 1000);\n                this.creationProgress = `Computing... (${elapsed}s, queue: ${progress})`;\n                this.cdr.markForCheck();\n\n                await this.sleep(this.POLL_INTERVAL_MS);\n            } catch (e) {\n                console.warn('Poll error:', e);\n                await this.sleep(this.POLL_INTERVAL_MS);\n            }\n        }\n\n        return false;\n    }\n\n    private sleep(ms: number): Promise<void> {\n        return new Promise((resolve) => setTimeout(resolve, ms));\n    }\n\n    cancel() {\n        this.cancelled.emit();\n    }\n\n    formatPeriod(period: CumulativePeriod): string {\n        const begin = new Date(period.periodBegin);\n        const end = new Date(period.periodEnd);\n        return `${begin.toLocaleString()} → ${end.toLocaleString()}`;\n    }\n\n    formatWindow(minutes: number): string {\n        if (minutes < 60) {\n            return `${minutes} min`;\n        }\n        const hours = minutes / 60;\n        return hours === 1 ? '1 hour' : `${hours} hours`;\n    }\n}\n","<div class=\"cumulative-selector-overlay\">\n    <div class=\"cumulative-selector-modal\">\n        <div class=\"modal-header\">\n            <h2>Select Cumulative Period</h2>\n            <ion-button fill=\"clear\" (click)=\"cancel()\">\n                <ion-icon name=\"close\"></ion-icon>\n            </ion-button>\n        </div>\n\n        <div class=\"modal-content\">\n            <!-- Loading state -->\n            <div *ngIf=\"loading\" class=\"loading-state\">\n                <ion-spinner name=\"crescent\"></ion-spinner>\n                <p>Loading available periods...</p>\n            </div>\n\n            <!-- Error message -->\n            <div *ngIf=\"errorMessage\" class=\"error-message\">\n                <ion-icon name=\"warning-outline\"></ion-icon>\n                <span>{{ errorMessage }}</span>\n            </div>\n\n            <!-- Available periods list -->\n            <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n                <h3>Available Cumulative Periods</h3>\n                <ion-list>\n                    <ng-container *ngFor=\"let period of availablePeriods\">\n                        <!-- Period header (click to expand) -->\n                        <ion-item button (click)=\"toggleExpand(period)\" [disabled]=\"creating\">\n                            <ion-icon [name]=\"isExpanded(period) ? 'chevron-down' : 'chevron-forward'\" slot=\"start\"></ion-icon>\n                            <ion-label>\n                                <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n                                <p>{{ formatPeriod(period) }}</p>\n                                <p class=\"count-info\">{{ period.count }} cumulative(s) - click to expand</p>\n                            </ion-label>\n                        </ion-item>\n\n                        <!-- Expanded: individual cumulatives -->\n                        <div *ngIf=\"isExpanded(period)\" class=\"expanded-items\">\n                            <ion-spinner *ngIf=\"isLoadingExpanded(period)\" name=\"crescent\"></ion-spinner>\n                            <ion-list *ngIf=\"!isLoadingExpanded(period)\" class=\"nested-list\">\n                                <ion-item *ngFor=\"let item of getExpandedItems(period)\"\n                                          button\n                                          (click)=\"selectIndividualCumulative(item)\"\n                                          class=\"nested-item\"\n                                          [disabled]=\"creating\">\n                                    <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n                                    <ion-label>\n                                        <h3>{{ getItemPeriodBegin(item) | date:'HH:mm' }} - {{ item.date | date:'HH:mm' }}</h3>\n                                    </ion-label>\n                                    <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n                                </ion-item>\n                                <ion-item *ngIf=\"getExpandedItems(period).length === 0\" class=\"nested-item\">\n                                    <ion-label>\n                                        <p>No individual cumulatives found</p>\n                                    </ion-label>\n                                </ion-item>\n                            </ion-list>\n                        </div>\n                    </ng-container>\n                </ion-list>\n            </div>\n\n            <!-- No periods available -->\n            <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n                <ion-icon name=\"information-circle-outline\"></ion-icon>\n                <p>No cumulative periods available yet.</p>\n            </div>\n\n            <!-- Create new section (admin only) -->\n            <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n                <h3>Create New Cumulative</h3>\n                <div class=\"create-info\">\n                    <p>\n                        <strong>Period:</strong>\n                        {{ currentPeriodBegin?.toLocaleString() }} → {{ currentPeriodEnd?.toLocaleString() }}\n                    </p>\n                    <p>\n                        <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n                    </p>\n                    <p *ngIf=\"baseCumulatives\" class=\"coverage-info\"\n                       [class.coverage-ok]=\"coveragePercent >= 100\"\n                       [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n                       [class.coverage-error]=\"coveragePercent === 0\">\n                        <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n                        Base data coverage: {{ coveragePercent }}%\n                    </p>\n                    <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n                        <ion-icon name=\"alert-circle\"></ion-icon>\n                        No base cumulatives available\n                    </p>\n                </div>\n\n                <!-- Creation progress -->\n                <div *ngIf=\"creating\" class=\"creation-progress\">\n                    <ion-spinner name=\"crescent\"></ion-spinner>\n                    <span>{{ creationProgress }}</span>\n                </div>\n\n                <ion-button [disabled]=\"!canCreateNew || creating\"\n                            expand=\"block\"\n                            (click)=\"createNewPeriod()\">\n                    <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n                    Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n                </ion-button>\n\n                <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n                    Create is disabled because base 5-min cumulatives don't fully cover the selected period.\n                </p>\n            </div>\n        </div>\n\n        <div class=\"modal-footer\">\n            <ion-button fill=\"outline\" (click)=\"cancel()\" [disabled]=\"creating\">\n                Cancel\n            </ion-button>\n        </div>\n    </div>\n</div>\n"]}
244
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cumulative-selector.component.js","sourceRoot":"","sources":["../../../src/core/shared/cumulative-selector/cumulative-selector.component.ts","../../../src/core/shared/cumulative-selector/cumulative-selector.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n    ChangeDetectionStrategy,\n    ChangeDetectorRef,\n    Component,\n    EventEmitter,\n    Input,\n    OnInit,\n    Output,\n} from '@angular/core';\nimport {ProfileService} from '../profile.service';\nimport {\n    CumulativePeriod,\n    IndividualCumulative,\n    RaainApiRainsCumulativesDetailedResponse,\n    RaainApiRainsCumulativesListResponse,\n} from 'raain-model';\n\nexport interface CumulativeSelection {\n    periodBegin: Date;\n    periodEnd: Date;\n    windowInMinutes: number;\n    cumulativeId?: string;\n    cumulativeDate?: Date;\n}\n\n@Component({\n    selector: 'cumulative-selector',\n    templateUrl: './cumulative-selector.component.html',\n    styleUrls: ['./cumulative-selector.component.scss'],\n    changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CumulativeSelectorComponent implements OnInit {\n    @Input() rainId: string;\n    @Input() currentPeriodBegin: Date;\n    @Input() currentPeriodEnd: Date;\n    @Input() provider: string;\n    @Input() timeStepInMinutes: number = 5;\n    @Input() isAdmin: boolean = false;\n\n    @Output() periodSelected = new EventEmitter<CumulativeSelection>();\n    @Output() cancelled = new EventEmitter<void>();\n\n    availablePeriods: CumulativePeriod[] = [];\n    baseCumulatives: CumulativePeriod = null;\n    loading = true;\n    creating = false;\n    creationProgress = '';\n    errorMessage = '';\n\n    coveragePercent = 0;\n    canCreateNew = false;\n    currentWindowMinutes = 0;\n\n    expandedPeriods: Map<string, IndividualCumulative[]> = new Map();\n    loadingExpanded: Set<string> = new Set();\n\n    private readonly POLL_TIMEOUT_MS = 900000; // 900 seconds\n    private readonly POLL_INTERVAL_MS = 3000;\n\n    constructor(\n        private profileService: ProfileService,\n        private cdr: ChangeDetectorRef\n    ) {}\n\n    async ngOnInit() {\n        this.currentWindowMinutes = Math.round(\n            (this.currentPeriodEnd.getTime() - this.currentPeriodBegin.getTime()) / 60000\n        );\n        await this.loadAvailablePeriods();\n    }\n\n    async loadAvailablePeriods() {\n        this.loading = true;\n        this.errorMessage = '';\n        this.cdr.markForCheck();\n\n        try {\n            // Fetch all cumulative periods (admin sees all, non-admin sees only customer-launched)\n            const response = (await this.profileService.getCumulativePeriods(this.rainId, {\n                provider: this.provider,\n                forced: this.isAdmin,\n                periodBegin: this.currentPeriodBegin.toISOString(),\n                periodEnd: this.currentPeriodEnd.toISOString(),\n            })) as RaainApiRainsCumulativesListResponse;\n\n            // Filter for existing custom cumulatives (window > 0)\n            this.availablePeriods = response.periods.filter((p) => p.windowInMinutes > 0);\n\n            // Get base cumulatives (window = 0) to check coverage\n            this.baseCumulatives = response.periods.find((p) => p.windowInMinutes === 0);\n\n            this.calculateCoverage();\n        } catch (e) {\n            this.errorMessage = 'Failed to load cumulative periods';\n            console.error('Error loading cumulative periods:', e);\n        }\n\n        this.loading = false;\n        this.cdr.markForCheck();\n    }\n\n    calculateCoverage() {\n        if (!this.baseCumulatives || !this.baseCumulatives.count) {\n            this.coveragePercent = 0;\n            this.canCreateNew = false;\n            return;\n        }\n\n        // Calculate expected number of base cumulatives (always 5-min granularity)\n        const baseTimeStep = 5;\n        const expectedCount = Math.ceil(this.currentWindowMinutes / baseTimeStep);\n\n        if (this.baseCumulatives.count >= expectedCount) {\n            this.coveragePercent = 100;\n        } else {\n            this.coveragePercent = Math.round((this.baseCumulatives.count / expectedCount) * 100);\n        }\n\n        // Allow creation as long as some base data exists\n        this.canCreateNew = this.coveragePercent > 0;\n    }\n\n    selectPeriod(period: CumulativePeriod) {\n        this.periodSelected.emit({\n            periodBegin: new Date(period.periodBegin),\n            periodEnd: new Date(period.periodEnd),\n            windowInMinutes: period.windowInMinutes,\n        });\n    }\n\n    async toggleExpand(period: CumulativePeriod) {\n        const key = `${period.windowInMinutes}`;\n        if (this.expandedPeriods.has(key)) {\n            this.expandedPeriods.delete(key);\n            this.cdr.markForCheck();\n            return;\n        }\n\n        this.loadingExpanded.add(key);\n        this.cdr.markForCheck();\n\n        try {\n            const response = (await this.profileService.getCumulativePeriods(this.rainId, {\n                provider: this.provider,\n                windowInMinutes: period.windowInMinutes,\n                forced: this.isAdmin,\n                detailed: true,\n            })) as RaainApiRainsCumulativesDetailedResponse;\n\n            this.expandedPeriods.set(key, response.cumulatives || []);\n        } catch (e) {\n            console.error('Error loading individual cumulatives:', e);\n            this.expandedPeriods.set(key, []);\n        }\n\n        this.loadingExpanded.delete(key);\n        this.cdr.markForCheck();\n    }\n\n    selectIndividualCumulative(item: IndividualCumulative) {\n        const endDate = new Date(item.date);\n        const beginDate = new Date(endDate.getTime() - item.windowInMinutes * 60000);\n\n        this.periodSelected.emit({\n            periodBegin: beginDate,\n            periodEnd: endDate,\n            windowInMinutes: item.windowInMinutes,\n            cumulativeId: item.id,\n            cumulativeDate: endDate,\n        });\n    }\n\n    isExpanded(period: CumulativePeriod): boolean {\n        return this.expandedPeriods.has(`${period.windowInMinutes}`);\n    }\n\n    isLoadingExpanded(period: CumulativePeriod): boolean {\n        return this.loadingExpanded.has(`${period.windowInMinutes}`);\n    }\n\n    getExpandedItems(period: CumulativePeriod): IndividualCumulative[] {\n        return this.expandedPeriods.get(`${period.windowInMinutes}`) || [];\n    }\n\n    getItemPeriodBegin(item: IndividualCumulative): Date {\n        const endDate = new Date(item.date);\n        return new Date(endDate.getTime() - item.windowInMinutes * 60000);\n    }\n\n    async createNewPeriod() {\n        if (!this.canCreateNew || this.creating) {\n            return;\n        }\n\n        this.creating = true;\n        this.creationProgress = 'Starting cumulative computation...';\n        this.errorMessage = '';\n        this.cdr.markForCheck();\n\n        try {\n            // Trigger cumulative creation\n            const result = await this.profileService.createCumulativePeriod(this.rainId, {\n                periodBegin: this.currentPeriodBegin,\n                periodEnd: this.currentPeriodEnd,\n                provider: this.provider,\n                timeStepInMinutes: this.timeStepInMinutes,\n            });\n\n            if (!result) {\n                throw new Error('Failed to trigger cumulative computation');\n            }\n\n            this.creationProgress = `Jobs queued. Polling for completion...`;\n            this.cdr.markForCheck();\n\n            // Poll for completion\n            const success = await this.pollForCompletion();\n\n            if (success) {\n                this.periodSelected.emit({\n                    periodBegin: this.currentPeriodBegin,\n                    periodEnd: this.currentPeriodEnd,\n                    windowInMinutes: this.currentWindowMinutes,\n                });\n            } else {\n                this.errorMessage = 'Timeout waiting for cumulative computation';\n            }\n        } catch (e) {\n            this.errorMessage = `Error: ${e.message || 'Unknown error'}`;\n            console.error('Error creating cumulative:', e);\n        }\n\n        this.creating = false;\n        this.cdr.markForCheck();\n    }\n\n    private async pollForCompletion(): Promise<boolean> {\n        const startTime = Date.now();\n\n        while (Date.now() - startTime < this.POLL_TIMEOUT_MS) {\n            try {\n                const progress = await this.profileService.getRainProgress(this.rainId);\n\n                if (progress === 0) {\n                    // Trigger background fetch from raain-ground via timeframeCumulative\n                    await this.profileService.getRainTimeframe(\n                        this.rainId,\n                        this.currentPeriodBegin,\n                        this.currentPeriodEnd,\n                        this.isAdmin,\n                        this.provider,\n                        this.timeStepInMinutes,\n                        this.currentWindowMinutes\n                    );\n\n                    // Check if cumulative now exists (admin sees all cumulatives)\n                    const response = (await this.profileService.getCumulativePeriods(this.rainId, {\n                        provider: this.provider,\n                        windowInMinutes: this.currentWindowMinutes,\n                        forced: this.isAdmin,\n                    })) as RaainApiRainsCumulativesListResponse;\n\n                    const found = response.periods.find(\n                        (p) => p.windowInMinutes === this.currentWindowMinutes\n                    );\n\n                    if (found) {\n                        return true;\n                    }\n                }\n\n                const elapsed = Math.round((Date.now() - startTime) / 1000);\n                this.creationProgress = `Computing... (${elapsed}s, queue: ${progress})`;\n                this.cdr.markForCheck();\n\n                await this.sleep(this.POLL_INTERVAL_MS);\n            } catch (e) {\n                console.warn('Poll error:', e);\n                await this.sleep(this.POLL_INTERVAL_MS);\n            }\n        }\n\n        return false;\n    }\n\n    private sleep(ms: number): Promise<void> {\n        return new Promise((resolve) => setTimeout(resolve, ms));\n    }\n\n    cancel() {\n        this.cancelled.emit();\n    }\n\n    formatPeriod(period: CumulativePeriod): string {\n        const begin = new Date(period.periodBegin);\n        const end = new Date(period.periodEnd);\n        return `${begin.toLocaleString()} → ${end.toLocaleString()}`;\n    }\n\n    formatWindow(minutes: number): string {\n        if (minutes < 60) {\n            return `${minutes} min`;\n        }\n        const hours = minutes / 60;\n        return hours === 1 ? '1 hour' : `${hours} hours`;\n    }\n}\n","<div class=\"cumulative-selector-overlay\">\n    <div class=\"cumulative-selector-modal\">\n        <div class=\"modal-header\">\n            <h2>Select Cumulative Period</h2>\n            <ion-button (click)=\"cancel()\" fill=\"clear\">\n                <ion-icon name=\"close\"></ion-icon>\n            </ion-button>\n        </div>\n\n        <div class=\"modal-content\">\n            <!-- Loading state -->\n            <div *ngIf=\"loading\" class=\"loading-state\">\n                <ion-spinner name=\"crescent\"></ion-spinner>\n                <p>Loading available periods...</p>\n            </div>\n\n            <!-- Error message -->\n            <div *ngIf=\"errorMessage\" class=\"error-message\">\n                <ion-icon name=\"warning-outline\"></ion-icon>\n                <span>{{ errorMessage }}</span>\n            </div>\n\n            <!-- Available periods list -->\n            <div *ngIf=\"!loading && availablePeriods.length > 0\" class=\"periods-section\">\n                <h3>Available Cumulative Periods</h3>\n                <ion-list>\n                    <ng-container *ngFor=\"let period of availablePeriods\">\n                        <!-- Period header (click to expand) -->\n                        <ion-item (click)=\"toggleExpand(period)\" [disabled]=\"creating\" button>\n                            <ion-icon [name]=\"isExpanded(period) ? 'chevron-down' : 'chevron-forward'\"\n                                      slot=\"start\"></ion-icon>\n                            <ion-label>\n                                <h2>{{ formatWindow(period.windowInMinutes) }}</h2>\n                                <p>{{ formatPeriod(period) }}</p>\n                                <p class=\"count-info\">{{ period.count }} cumulative(s) - click to expand</p>\n                            </ion-label>\n                        </ion-item>\n\n                        <!-- Expanded: individual cumulatives -->\n                        <div *ngIf=\"isExpanded(period)\" class=\"expanded-items\">\n                            <ion-spinner *ngIf=\"isLoadingExpanded(period)\" name=\"crescent\"></ion-spinner>\n                            <ion-list *ngIf=\"!isLoadingExpanded(period)\" class=\"nested-list\">\n                                <ion-item (click)=\"selectIndividualCumulative(item)\"\n                                          *ngFor=\"let item of getExpandedItems(period)\"\n                                          [disabled]=\"creating\"\n                                          button\n                                          class=\"nested-item\">\n                                    <ion-icon name=\"time-outline\" slot=\"start\"></ion-icon>\n                                    <ion-label>\n                                        <h3>{{ getItemPeriodBegin(item) | date:'HH:mm' }}\n                                            - {{ item.date | date:'HH:mm' }}</h3>\n                                    </ion-label>\n                                    <ion-icon name=\"chevron-forward\" slot=\"end\"></ion-icon>\n                                </ion-item>\n                                <ion-item *ngIf=\"getExpandedItems(period).length === 0\" class=\"nested-item\">\n                                    <ion-label>\n                                        <p>No individual cumulatives found</p>\n                                    </ion-label>\n                                </ion-item>\n                            </ion-list>\n                        </div>\n                    </ng-container>\n                </ion-list>\n            </div>\n\n            <!-- No periods available -->\n            <div *ngIf=\"!loading && availablePeriods.length === 0\" class=\"no-periods\">\n                <ion-icon name=\"information-circle-outline\"></ion-icon>\n                <p>No cumulative periods available yet.</p>\n            </div>\n\n            <!-- Create new section (admin only) -->\n            <div *ngIf=\"!loading && isAdmin\" class=\"create-section\">\n                <h3>Create New Cumulative</h3>\n                <div class=\"create-info\">\n                    <p>\n                        <strong>Period:</strong>\n                        {{ currentPeriodBegin?.toLocaleString() }} → {{ currentPeriodEnd?.toLocaleString() }}\n                    </p>\n                    <p>\n                        <strong>Window:</strong> {{ formatWindow(currentWindowMinutes) }}\n                    </p>\n                    <p *ngIf=\"baseCumulatives\" [class.coverage-error]=\"coveragePercent === 0\"\n                       [class.coverage-ok]=\"coveragePercent >= 100\"\n                       [class.coverage-warn]=\"coveragePercent > 0 && coveragePercent < 100\"\n                       class=\"coverage-info\">\n                        <ion-icon [name]=\"coveragePercent >= 100 ? 'checkmark-circle' : 'alert-circle'\"></ion-icon>\n                        Base data coverage: {{ coveragePercent }}%\n                    </p>\n                    <p *ngIf=\"!baseCumulatives\" class=\"coverage-error\">\n                        <ion-icon name=\"alert-circle\"></ion-icon>\n                        No base cumulatives available\n                    </p>\n                </div>\n\n                <!-- Creation progress -->\n                <div *ngIf=\"creating\" class=\"creation-progress\">\n                    <ion-spinner name=\"crescent\"></ion-spinner>\n                    <span>{{ creationProgress }}</span>\n                </div>\n\n                <ion-button (click)=\"createNewPeriod()\" [disabled]=\"!canCreateNew || creating\"\n                            expand=\"block\"\n                            id=\"createNewCumulative\">\n                    <ion-icon name=\"add-circle-outline\" slot=\"start\"></ion-icon>\n                    Create {{ formatWindow(currentWindowMinutes) }} Cumulative\n                </ion-button>\n\n                <p *ngIf=\"!canCreateNew && !creating\" class=\"create-hint\">\n                    Create is disabled because no base 5-min cumulatives exist for the selected period.\n                </p>\n            </div>\n        </div>\n\n        <div class=\"modal-footer\">\n            <ion-button (click)=\"cancel()\" [disabled]=\"creating\" fill=\"outline\">\n                Cancel\n            </ion-button>\n        </div>\n    </div>\n</div>\n"]}