@c8y/ngx-components 1021.22.0 → 1021.25.0
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/alarms/alarms-date-filter.component.d.ts +22 -10
- package/alarms/alarms-date-filter.component.d.ts.map +1 -1
- package/alarms/alarms-type-filter.component.d.ts +21 -14
- package/alarms/alarms-type-filter.component.d.ts.map +1 -1
- package/alarms/alarms-view.service.d.ts +7 -0
- package/alarms/alarms-view.service.d.ts.map +1 -1
- package/alarms/alarms.model.d.ts +1 -6
- package/alarms/alarms.model.d.ts.map +1 -1
- package/alarms/alarms.module.d.ts +1 -1
- package/alarms/index.d.ts +1 -0
- package/alarms/index.d.ts.map +1 -1
- package/context-dashboard/context-dashboard.service.d.ts.map +1 -1
- package/core/router/scoped-context-route.service.d.ts.map +1 -1
- package/datapoints-export-selector/datapoints-export-selector-modal/datapoints-export-selector-file-exporter/datapoints-export-selector-file-exporter.component.d.ts +4 -2
- package/datapoints-export-selector/datapoints-export-selector-modal/datapoints-export-selector-file-exporter/datapoints-export-selector-file-exporter.component.d.ts.map +1 -1
- package/datapoints-export-selector/datapoints-export-selector.component.d.ts +3 -1
- package/datapoints-export-selector/datapoints-export-selector.component.d.ts.map +1 -1
- package/datapoints-export-selector/datapoints-export-selector.model.d.ts +17 -0
- package/datapoints-export-selector/datapoints-export-selector.model.d.ts.map +1 -1
- package/esm2022/alarms/alarm-info.component.mjs +3 -3
- package/esm2022/alarms/alarm-severity-to-label.pipe.mjs +2 -2
- package/esm2022/alarms/alarms-date-filter.component.mjs +92 -42
- package/esm2022/alarms/alarms-type-filter.component.mjs +102 -72
- package/esm2022/alarms/alarms-view.service.mjs +16 -2
- package/esm2022/alarms/alarms.component.mjs +3 -3
- package/esm2022/alarms/alarms.model.mjs +1 -1
- package/esm2022/alarms/alarms.module.mjs +3 -3
- package/esm2022/alarms/index.mjs +2 -1
- package/esm2022/context-dashboard/context-dashboard.service.mjs +18 -5
- package/esm2022/core/router/scoped-context-route.service.mjs +9 -4
- package/esm2022/core/tabs/tabs-outlet.component.mjs +3 -3
- package/esm2022/datapoints-export-selector/datapoints-export-selector-modal/datapoints-export-selector-file-exporter/datapoints-export-selector-file-exporter.component.mjs +29 -6
- package/esm2022/datapoints-export-selector/datapoints-export-selector.component.mjs +11 -5
- package/esm2022/datapoints-export-selector/datapoints-export-selector.model.mjs +18 -1
- package/esm2022/interval-picker/interval-picker.component.mjs +1 -1
- package/esm2022/interval-picker/interval-picker.model.mjs +1 -1
- package/esm2022/widgets/definitions/alarms/alarm-list/index.mjs +2 -1
- package/esm2022/widgets/implementations/alarms/alarm-list-widget-config/alarm-list-widget-config.component.mjs +73 -4
- package/esm2022/widgets/implementations/alarms/alarm-list-widget-view/alarm-list-widget.component.mjs +29 -5
- package/esm2022/widgets/implementations/alarms/alarm-list-widget.model.mjs +7 -1
- package/esm2022/widgets/implementations/alarms/alarm-widget.service.mjs +11 -1
- package/fesm2022/c8y-ngx-components-alarms.mjs +199 -107
- package/fesm2022/c8y-ngx-components-alarms.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-context-dashboard.mjs +17 -4
- package/fesm2022/c8y-ngx-components-context-dashboard.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-datapoints-export-selector.mjs +52 -7
- package/fesm2022/c8y-ngx-components-datapoints-export-selector.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-interval-picker.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-widgets-definitions-alarms-alarm-list.mjs +1 -0
- package/fesm2022/c8y-ngx-components-widgets-definitions-alarms-alarm-list.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components-widgets-implementations-alarms.mjs +123 -14
- package/fesm2022/c8y-ngx-components-widgets-implementations-alarms.mjs.map +1 -1
- package/fesm2022/c8y-ngx-components.mjs +10 -5
- package/fesm2022/c8y-ngx-components.mjs.map +1 -1
- package/interval-picker/interval-picker.component.d.ts +2 -2
- package/interval-picker/interval-picker.component.d.ts.map +1 -1
- package/interval-picker/interval-picker.model.d.ts +5 -0
- package/interval-picker/interval-picker.model.d.ts.map +1 -1
- package/locales/de.po +2 -2
- package/locales/locales.pot +18 -6
- package/package.json +1 -1
- package/widgets/definitions/alarms/alarm-list/index.d.ts.map +1 -1
- package/widgets/implementations/alarms/alarm-list-widget-config/alarm-list-widget-config.component.d.ts +12 -1
- package/widgets/implementations/alarms/alarm-list-widget-config/alarm-list-widget-config.component.d.ts.map +1 -1
- package/widgets/implementations/alarms/alarm-list-widget-view/alarm-list-widget.component.d.ts +12 -1
- package/widgets/implementations/alarms/alarm-list-widget-view/alarm-list-widget.component.d.ts.map +1 -1
- package/widgets/implementations/alarms/alarm-list-widget.model.d.ts +11 -1
- package/widgets/implementations/alarms/alarm-list-widget.model.d.ts.map +1 -1
- package/widgets/implementations/alarms/alarm-widget.service.d.ts.map +1 -1
|
@@ -1,85 +1,80 @@
|
|
|
1
1
|
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
2
|
-
import { FormBuilder, FormControl } from '@angular/forms';
|
|
3
2
|
import { ActivatedRoute, Router } from '@angular/router';
|
|
3
|
+
import { ColorService } from '@c8y/ngx-components';
|
|
4
4
|
import { AlarmEventSelectorService } from '@c8y/ngx-components/alarm-event-selector';
|
|
5
|
-
import { Subject } from 'rxjs';
|
|
6
|
-
import { map, takeUntil } from 'rxjs/operators';
|
|
5
|
+
import { map, Subject, takeUntil } from 'rxjs';
|
|
7
6
|
import * as i0 from "@angular/core";
|
|
8
|
-
import * as i1 from "@
|
|
9
|
-
import * as i2 from "@
|
|
10
|
-
import * as i3 from "@
|
|
7
|
+
import * as i1 from "@c8y/ngx-components/alarm-event-selector";
|
|
8
|
+
import * as i2 from "@angular/router";
|
|
9
|
+
import * as i3 from "@c8y/ngx-components";
|
|
11
10
|
import * as i4 from "@angular/cdk/a11y";
|
|
12
11
|
import * as i5 from "ngx-bootstrap/dropdown";
|
|
13
|
-
import * as i6 from "@
|
|
14
|
-
import * as i7 from "@angular/
|
|
15
|
-
import * as i8 from "
|
|
12
|
+
import * as i6 from "@angular/common";
|
|
13
|
+
import * as i7 from "@angular/forms";
|
|
14
|
+
import * as i8 from "ngx-bootstrap/tooltip";
|
|
15
|
+
import * as i9 from "@ngx-translate/core";
|
|
16
16
|
export class AlarmsTypeFilterComponent {
|
|
17
|
-
constructor(
|
|
18
|
-
this.formBuilder = formBuilder;
|
|
17
|
+
constructor(alarmEventSelectorService, activatedRoute, router, colorService) {
|
|
19
18
|
this.alarmEventSelectorService = alarmEventSelectorService;
|
|
20
19
|
this.activatedRoute = activatedRoute;
|
|
21
20
|
this.router = router;
|
|
21
|
+
this.colorService = colorService;
|
|
22
|
+
this.possibleFilters = [];
|
|
22
23
|
this.activeFilters = [];
|
|
23
24
|
this.onFilterChanged = new EventEmitter();
|
|
24
|
-
this.
|
|
25
|
+
this.customAlarmTypes = [];
|
|
26
|
+
this.customAlarmTypeInput = '';
|
|
25
27
|
this.queryParamName = 'typeFilter';
|
|
26
|
-
this.
|
|
27
|
-
[this.formControlName]: new FormControl([])
|
|
28
|
-
});
|
|
28
|
+
this.STORAGE_ACCESS_KEY = 'customAlarmTypes';
|
|
29
29
|
this.destroy$ = new Subject();
|
|
30
|
-
this.
|
|
30
|
+
this.currentQueryParam = '';
|
|
31
31
|
}
|
|
32
32
|
ngOnInit() {
|
|
33
|
-
this.
|
|
34
|
-
.pipe(map(params => {
|
|
35
|
-
const alarms = this.formGroup.get(this.formControlName).value;
|
|
36
|
-
return this.getActiveAlarmFiltersFromQueryParameter(alarms, params[this.queryParamName]);
|
|
37
|
-
}), takeUntil(this.destroy$))
|
|
38
|
-
.subscribe((alarmFilters) => this.formGroup.get(this.formControlName).patchValue(alarmFilters));
|
|
39
|
-
this.formGroup
|
|
40
|
-
.get(this.formControlName)
|
|
41
|
-
.valueChanges.pipe(takeUntil(this.destroy$))
|
|
42
|
-
.subscribe((alarmFilters) => this.applyFilterChange(alarmFilters));
|
|
33
|
+
this.setQueryParameterObservable();
|
|
43
34
|
}
|
|
44
|
-
ngOnChanges(changes) {
|
|
35
|
+
async ngOnChanges(changes) {
|
|
45
36
|
if (changes.alarms && changes.alarms.currentValue && this.activeFilters.length === 0) {
|
|
46
|
-
this.
|
|
37
|
+
await this.setPossibleFilters();
|
|
38
|
+
this.applyFilterChange();
|
|
47
39
|
}
|
|
48
40
|
}
|
|
41
|
+
setQueryParameterObservable() {
|
|
42
|
+
this.activatedRoute.queryParams
|
|
43
|
+
.pipe(map(params => {
|
|
44
|
+
const alarms = this.possibleFilters;
|
|
45
|
+
const possibleFilters = this.setActiveAlarmFiltersFromQueryParameter(alarms, params[this.queryParamName]);
|
|
46
|
+
return possibleFilters;
|
|
47
|
+
}), takeUntil(this.destroy$))
|
|
48
|
+
.subscribe((possibleFilters) => {
|
|
49
|
+
this.possibleFilters = possibleFilters;
|
|
50
|
+
this.applyFilterChange();
|
|
51
|
+
});
|
|
52
|
+
}
|
|
49
53
|
ngOnDestroy() {
|
|
50
54
|
this.destroy$.next();
|
|
51
55
|
this.destroy$.complete();
|
|
52
56
|
}
|
|
57
|
+
toggleAlarmType(alarmType) {
|
|
58
|
+
alarmType.__active = !alarmType.__active;
|
|
59
|
+
}
|
|
53
60
|
deselect(type) {
|
|
54
|
-
const
|
|
55
|
-
const alarmFilter = alarms.find(alarm => alarm.filters.type === type.filters.type);
|
|
61
|
+
const alarmFilter = this.possibleFilters.find(alarm => alarm.filters.type === type.filters.type);
|
|
56
62
|
alarmFilter.__active = false;
|
|
57
|
-
this.
|
|
63
|
+
this.applyFilterChange();
|
|
58
64
|
}
|
|
59
65
|
deselectAll() {
|
|
60
|
-
|
|
61
|
-
const allFilters = alarms.map(alarm => {
|
|
66
|
+
this.possibleFilters = this.possibleFilters.map(alarm => {
|
|
62
67
|
return {
|
|
63
68
|
...alarm,
|
|
64
69
|
__active: false
|
|
65
70
|
};
|
|
66
71
|
});
|
|
67
|
-
this.
|
|
72
|
+
this.applyFilterChange();
|
|
68
73
|
}
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
return alarmFilters;
|
|
72
|
-
}
|
|
73
|
-
const types = filterTypesQuery.split(',');
|
|
74
|
-
return alarmFilters.map((alarm) => ({
|
|
75
|
-
...alarm,
|
|
76
|
-
__active: types.includes(alarm.filters.type)
|
|
77
|
-
}));
|
|
78
|
-
}
|
|
79
|
-
applyFilterChange(alarmFilters) {
|
|
80
|
-
const actives = alarmFilters.filter((alarmFilter) => alarmFilter.__active);
|
|
74
|
+
applyFilterChange() {
|
|
75
|
+
const actives = this.possibleFilters.filter((alarmFilter) => alarmFilter.__active);
|
|
81
76
|
const newQueryParam = this.getQueryParams(actives);
|
|
82
|
-
const hasChanged = newQueryParam !== this.
|
|
77
|
+
const hasChanged = newQueryParam !== this.currentQueryParam;
|
|
83
78
|
if (hasChanged) {
|
|
84
79
|
this.activeFilters = actives;
|
|
85
80
|
this.onFilterChanged.emit(this.activeFilters);
|
|
@@ -89,43 +84,78 @@ export class AlarmsTypeFilterComponent {
|
|
|
89
84
|
},
|
|
90
85
|
queryParamsHandling: 'merge'
|
|
91
86
|
});
|
|
92
|
-
this.
|
|
87
|
+
this.currentQueryParam = newQueryParam;
|
|
93
88
|
}
|
|
94
89
|
}
|
|
95
|
-
|
|
96
|
-
|
|
90
|
+
resetFilters() {
|
|
91
|
+
this.possibleFilters.forEach(possibleFilter => {
|
|
92
|
+
possibleFilter.__active = this.activeFilters.some((activeFilter) => activeFilter === possibleFilter);
|
|
93
|
+
});
|
|
97
94
|
}
|
|
98
|
-
|
|
95
|
+
removeCustomAlarm(alarmDetails) {
|
|
96
|
+
this.possibleFilters = this.possibleFilters.filter(filter => filter !== alarmDetails);
|
|
97
|
+
this.storeCustomAlarmTypes();
|
|
98
|
+
}
|
|
99
|
+
confirmWithEnter(event) {
|
|
100
|
+
if (event.key === 'Enter') {
|
|
101
|
+
this.addCustomAlarmType();
|
|
102
|
+
}
|
|
103
|
+
}
|
|
104
|
+
async addCustomAlarmType() {
|
|
105
|
+
if (!this.customAlarmTypeInput) {
|
|
106
|
+
return;
|
|
107
|
+
}
|
|
108
|
+
this.possibleFilters.unshift({
|
|
109
|
+
label: this.customAlarmTypeInput,
|
|
110
|
+
color: await this.colorService.generateColor(this.customAlarmTypeInput),
|
|
111
|
+
filters: {
|
|
112
|
+
type: this.customAlarmTypeInput
|
|
113
|
+
},
|
|
114
|
+
timelineType: 'ALARM',
|
|
115
|
+
__active: true,
|
|
116
|
+
__target: null
|
|
117
|
+
});
|
|
118
|
+
this.customAlarmTypeInput = '';
|
|
119
|
+
this.storeCustomAlarmTypes();
|
|
120
|
+
}
|
|
121
|
+
storeCustomAlarmTypes() {
|
|
122
|
+
const customTypes = this.possibleFilters.filter((filter) => !filter.__target);
|
|
123
|
+
window.localStorage.setItem(this.STORAGE_ACCESS_KEY, JSON.stringify(customTypes));
|
|
124
|
+
}
|
|
125
|
+
getCustomAlarmTypeFromStorage() {
|
|
126
|
+
const types = window.localStorage.getItem(this.STORAGE_ACCESS_KEY);
|
|
127
|
+
return types ? JSON.parse(types) : [];
|
|
128
|
+
}
|
|
129
|
+
async setPossibleFilters() {
|
|
99
130
|
const queryParameters = this.activatedRoute.snapshot.queryParamMap.get(this.queryParamName);
|
|
100
131
|
const alarmTypesFromCurrentlyShownAlarms = await this.alarmEventSelectorService.getUniqueAlarmsOnly(this.alarms.data);
|
|
101
|
-
const
|
|
102
|
-
const selectableAlarmTypes = this.
|
|
103
|
-
this.
|
|
132
|
+
const customAlarmTypesFromLocalStorage = this.getCustomAlarmTypeFromStorage();
|
|
133
|
+
const selectableAlarmTypes = this.setActiveAlarmFiltersFromQueryParameter([...customAlarmTypesFromLocalStorage, ...alarmTypesFromCurrentlyShownAlarms], queryParameters);
|
|
134
|
+
this.possibleFilters = selectableAlarmTypes;
|
|
104
135
|
}
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
const missingFilters = await Promise.all(types
|
|
111
|
-
.filter(type => !alarmFilters.find(alarmFilter => alarmFilter.filters.type === type))
|
|
112
|
-
.map(type => {
|
|
113
|
-
return this.alarmEventSelectorService.createItem('ALARM', {
|
|
114
|
-
type,
|
|
115
|
-
source: null
|
|
116
|
-
});
|
|
136
|
+
setActiveAlarmFiltersFromQueryParameter(alarmFilters, filterTypesQuery = '') {
|
|
137
|
+
const types = (filterTypesQuery ?? '').split(',');
|
|
138
|
+
return alarmFilters.map((alarm) => ({
|
|
139
|
+
...alarm,
|
|
140
|
+
__active: types.includes(alarm.filters.type)
|
|
117
141
|
}));
|
|
118
|
-
return [...alarmFilters, ...missingFilters];
|
|
119
142
|
}
|
|
120
|
-
|
|
121
|
-
|
|
143
|
+
getQueryParams(activeFilters) {
|
|
144
|
+
return activeFilters.map(filter => filter.filters.type).join(',');
|
|
145
|
+
}
|
|
146
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: AlarmsTypeFilterComponent, deps: [{ token: i1.AlarmEventSelectorService }, { token: i2.ActivatedRoute }, { token: i2.Router }, { token: i3.ColorService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
147
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.9", type: AlarmsTypeFilterComponent, selector: "c8y-alarms-type-filter", inputs: { alarms: "alarms", possibleFilters: "possibleFilters", activeFilters: "activeFilters" }, outputs: { onFilterChanged: "onFilterChanged" }, usesOnChanges: true, ngImport: i0, template: "<div class=\"d-flex a-i-center\">\n <div\n class=\"dropdown\"\n title=\"{{ 'Filter by Alarm types' | translate }}\"\n dropdown\n #filtersDropdown=\"bs-dropdown\"\n [cdkTrapFocus]=\"filtersDropdown.isOpen\"\n (onHidden)=\"resetFilters()\"\n [insideClick]=\"true\"\n >\n <div class=\"input-group fit-w\">\n <div class=\"form-control d-flex a-i-center inner-scroll\">\n <ng-container *ngIf=\"activeFilters.length > 0; else allTypes\">\n <span\n class=\"tag tag--info chip\"\n *ngFor=\"let filter of activeFilters\"\n >\n <button\n class=\"btn btn-xs btn-clean text-10\"\n title=\"{{ 'Remove' | translate }}\"\n type=\"button\"\n (click)=\"$event.stopPropagation(); deselect(filter)\"\n >\n <i c8yIcon=\"times\"></i>\n </button>\n <span\n class=\"circle-icon-wrapper circle-icon-wrapper--small\"\n [ngStyle]=\"{ 'background-color': filter.color }\"\n >\n <i\n class=\"stroked-icon\"\n c8yIcon=\"bell\"\n ></i>\n </span>\n {{ filter.filters.type }}\n </span>\n </ng-container>\n <ng-template #allTypes>\n <span class=\"text-nowrap\">\n {{ 'All alarm types' | translate }}\n </span>\n </ng-template>\n </div>\n <div class=\"input-group-btn input-group-btn--last text-center\">\n <button\n class=\"btn-default btn\"\n [title]=\"'Clear filters' | translate\"\n *ngIf=\"activeFilters.length\"\n (click)=\"deselectAll()\"\n >\n <i c8yIcon=\"times\"></i>\n </button>\n <button\n class=\"btn-default btn btn--caret\"\n [title]=\"'Alarm types' | translate\"\n data-cy=\"c8y-alarm-type-filter\"\n dropdownToggle\n >\n <i class=\"caret\"></i>\n </button>\n </div>\n </div>\n <div\n class=\"dropdown-menu dropdown-menu-action-bar\"\n style=\"max-width: unset; min-width: 250px\"\n *dropdownMenu\n >\n <div class=\"p-16 bg-level-2\">\n <div>\n <p>\n <i\n class=\"text-info m-r-4\"\n [c8yIcon]=\"'info-circle'\"\n ></i>\n <strong tanslate>The list below may not be complete.</strong>\n </p>\n <p tanslate>\n Recent alarms are displayed below. Past alarms might not be shown. Optionally you can\n add a custom alarm.\n </p>\n </div>\n </div>\n <c8y-list-group>\n <div class=\"input-group p-t-16 p-b-16 p-r-32 p-l-32 separator-bottom\">\n <input\n class=\"form-control\"\n placeholder=\"Custom alarm type\"\n type=\"text\"\n [(ngModel)]=\"customAlarmTypeInput\"\n (keydown)=\"confirmWithEnter($event)\"\n />\n <div class=\"input-group-btn\">\n <button\n class=\"btn-dot text-primary\"\n [attr.aria-label]=\"'Add custom alarm' | translate\"\n tooltip=\"Add\"\n placement=\"top\"\n [delay]=\"500\"\n (click)=\"addCustomAlarmType()\"\n >\n <i c8yIcon=\"plus-circle\"></i>\n </button>\n </div>\n </div>\n\n <c8y-li\n class=\"c8y-list__item__collapse--container-small cdk-drag\"\n style=\"cursor: pointer\"\n *ngFor=\"let alarmType of possibleFilters\"\n (click)=\"toggleAlarmType(alarmType)\"\n >\n <c8y-li-checkbox\n class=\"a-s-center m-t-4 p-r-0 p-l-0\"\n [selected]=\"alarmType.__active\"\n (click)=\"$event.stopPropagation()\"\n (change)=\"toggleAlarmType(alarmType); $event.stopPropagation()\"\n ></c8y-li-checkbox>\n <div class=\"d-flex a-i-center p-l-4\">\n <div class=\"c8y-list__item__colorpicker p-t-0 p-b-0 p-l-0\">\n <div class=\"c8y-colorpicker c8y-colorpicker--alarm\">\n <span\n class=\"circle-icon-wrapper\"\n [ngStyle]=\"{ 'background-color': alarmType.color }\"\n >\n <i\n class=\"stroked-icon\"\n [c8yIcon]=\"'bell'\"\n ></i>\n </span>\n </div>\n </div>\n <span class=\"text-truncate text-12 flex-grow\">\n {{ alarmType.label }}\n </span>\n <button\n class=\"btn-dot btn-dot--danger\"\n [attr.aria-label]=\"'Remove' | translate\"\n tooltip=\"Remove\"\n placement=\"top\"\n *ngIf=\"alarmType.__target === null\"\n [delay]=\"500\"\n (click)=\"removeCustomAlarm(alarmType); $event.stopPropagation()\"\n >\n <i c8yIcon=\"minus-circle\"></i>\n </button>\n </div>\n </c8y-li>\n <c8y-li *ngIf=\"possibleFilters.length === 0\">\n <c8y-ui-empty-state\n class=\"p-t-8\"\n icon=\"c8y-alarm\"\n [title]=\"'No alarm found' | translate\"\n [subtitle]=\"\n 'There is no alarm to filter. You can still add a custom alarm.' | translate\n \"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </c8y-li>\n <div class=\"sticky-bottom p-16\">\n <button\n class=\"btn btn-block btn-primary\"\n [disabled]=\"possibleFilters.length === 0\"\n (click)=\"applyFilterChange(); $event.stopPropagation(); filtersDropdown.hide()\"\n translate\n >\n Apply\n </button>\n </div>\n </c8y-list-group>\n </div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i4.CdkTrapFocus, selector: "[cdkTrapFocus]", inputs: ["cdkTrapFocus", "cdkTrapFocusAutoCapture"], exportAs: ["cdkTrapFocus"] }, { kind: "directive", type: i5.BsDropdownMenuDirective, selector: "[bsDropdownMenu],[dropdownMenu]", exportAs: ["bs-dropdown-menu"] }, { kind: "directive", type: i5.BsDropdownToggleDirective, selector: "[bsDropdownToggle],[dropdownToggle]", exportAs: ["bs-dropdown-toggle"] }, { kind: "directive", type: i5.BsDropdownDirective, selector: "[bsDropdown], [dropdown]", inputs: ["placement", "triggers", "container", "dropup", "autoClose", "isAnimated", "insideClick", "isDisabled", "isOpen"], outputs: ["isOpenChange", "onShown", "onHidden"], exportAs: ["bs-dropdown"] }, { kind: "component", type: i3.EmptyStateComponent, selector: "c8y-ui-empty-state", inputs: ["icon", "title", "subtitle", "horizontal"] }, { kind: "directive", type: i3.IconDirective, selector: "[c8yIcon]", inputs: ["c8yIcon"] }, { kind: "directive", type: i3.C8yTranslateDirective, selector: "[translate],[ngx-translate]" }, { kind: "directive", type: i6.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i6.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i7.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i7.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i7.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i3.ListGroupComponent, selector: "c8y-list-group" }, { kind: "component", type: i3.ListItemComponent, selector: "c8y-list-item, c8y-li", inputs: ["active", "highlighted", "emptyActions", "dense", "collapsed", "selectable"], outputs: ["collapsedChange"] }, { kind: "component", type: i3.ListItemCheckboxComponent, selector: "c8y-list-item-checkbox, c8y-li-checkbox", inputs: ["selected", "indeterminate", "disabled", "displayAsSwitch"], outputs: ["onSelect"] }, { kind: "directive", type: i8.TooltipDirective, selector: "[tooltip], [tooltipHtml]", inputs: ["adaptivePosition", "tooltip", "placement", "triggers", "container", "containerClass", "boundariesElement", "isOpen", "isDisabled", "delay", "tooltipHtml", "tooltipPlacement", "tooltipIsOpen", "tooltipEnable", "tooltipAppendToBody", "tooltipAnimation", "tooltipClass", "tooltipContext", "tooltipPopupDelay", "tooltipFadeDuration", "tooltipTrigger"], outputs: ["tooltipChange", "onShown", "onHidden", "tooltipStateChanged"], exportAs: ["bs-tooltip"] }, { kind: "directive", type: i9.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { kind: "pipe", type: i9.TranslatePipe, name: "translate" }] }); }
|
|
122
148
|
}
|
|
123
149
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImport: i0, type: AlarmsTypeFilterComponent, decorators: [{
|
|
124
150
|
type: Component,
|
|
125
|
-
args: [{ selector: 'c8y-alarms-type-filter', template: "<
|
|
126
|
-
}], ctorParameters: () => [{ type: i1.
|
|
151
|
+
args: [{ selector: 'c8y-alarms-type-filter', template: "<div class=\"d-flex a-i-center\">\n <div\n class=\"dropdown\"\n title=\"{{ 'Filter by Alarm types' | translate }}\"\n dropdown\n #filtersDropdown=\"bs-dropdown\"\n [cdkTrapFocus]=\"filtersDropdown.isOpen\"\n (onHidden)=\"resetFilters()\"\n [insideClick]=\"true\"\n >\n <div class=\"input-group fit-w\">\n <div class=\"form-control d-flex a-i-center inner-scroll\">\n <ng-container *ngIf=\"activeFilters.length > 0; else allTypes\">\n <span\n class=\"tag tag--info chip\"\n *ngFor=\"let filter of activeFilters\"\n >\n <button\n class=\"btn btn-xs btn-clean text-10\"\n title=\"{{ 'Remove' | translate }}\"\n type=\"button\"\n (click)=\"$event.stopPropagation(); deselect(filter)\"\n >\n <i c8yIcon=\"times\"></i>\n </button>\n <span\n class=\"circle-icon-wrapper circle-icon-wrapper--small\"\n [ngStyle]=\"{ 'background-color': filter.color }\"\n >\n <i\n class=\"stroked-icon\"\n c8yIcon=\"bell\"\n ></i>\n </span>\n {{ filter.filters.type }}\n </span>\n </ng-container>\n <ng-template #allTypes>\n <span class=\"text-nowrap\">\n {{ 'All alarm types' | translate }}\n </span>\n </ng-template>\n </div>\n <div class=\"input-group-btn input-group-btn--last text-center\">\n <button\n class=\"btn-default btn\"\n [title]=\"'Clear filters' | translate\"\n *ngIf=\"activeFilters.length\"\n (click)=\"deselectAll()\"\n >\n <i c8yIcon=\"times\"></i>\n </button>\n <button\n class=\"btn-default btn btn--caret\"\n [title]=\"'Alarm types' | translate\"\n data-cy=\"c8y-alarm-type-filter\"\n dropdownToggle\n >\n <i class=\"caret\"></i>\n </button>\n </div>\n </div>\n <div\n class=\"dropdown-menu dropdown-menu-action-bar\"\n style=\"max-width: unset; min-width: 250px\"\n *dropdownMenu\n >\n <div class=\"p-16 bg-level-2\">\n <div>\n <p>\n <i\n class=\"text-info m-r-4\"\n [c8yIcon]=\"'info-circle'\"\n ></i>\n <strong tanslate>The list below may not be complete.</strong>\n </p>\n <p tanslate>\n Recent alarms are displayed below. Past alarms might not be shown. Optionally you can\n add a custom alarm.\n </p>\n </div>\n </div>\n <c8y-list-group>\n <div class=\"input-group p-t-16 p-b-16 p-r-32 p-l-32 separator-bottom\">\n <input\n class=\"form-control\"\n placeholder=\"Custom alarm type\"\n type=\"text\"\n [(ngModel)]=\"customAlarmTypeInput\"\n (keydown)=\"confirmWithEnter($event)\"\n />\n <div class=\"input-group-btn\">\n <button\n class=\"btn-dot text-primary\"\n [attr.aria-label]=\"'Add custom alarm' | translate\"\n tooltip=\"Add\"\n placement=\"top\"\n [delay]=\"500\"\n (click)=\"addCustomAlarmType()\"\n >\n <i c8yIcon=\"plus-circle\"></i>\n </button>\n </div>\n </div>\n\n <c8y-li\n class=\"c8y-list__item__collapse--container-small cdk-drag\"\n style=\"cursor: pointer\"\n *ngFor=\"let alarmType of possibleFilters\"\n (click)=\"toggleAlarmType(alarmType)\"\n >\n <c8y-li-checkbox\n class=\"a-s-center m-t-4 p-r-0 p-l-0\"\n [selected]=\"alarmType.__active\"\n (click)=\"$event.stopPropagation()\"\n (change)=\"toggleAlarmType(alarmType); $event.stopPropagation()\"\n ></c8y-li-checkbox>\n <div class=\"d-flex a-i-center p-l-4\">\n <div class=\"c8y-list__item__colorpicker p-t-0 p-b-0 p-l-0\">\n <div class=\"c8y-colorpicker c8y-colorpicker--alarm\">\n <span\n class=\"circle-icon-wrapper\"\n [ngStyle]=\"{ 'background-color': alarmType.color }\"\n >\n <i\n class=\"stroked-icon\"\n [c8yIcon]=\"'bell'\"\n ></i>\n </span>\n </div>\n </div>\n <span class=\"text-truncate text-12 flex-grow\">\n {{ alarmType.label }}\n </span>\n <button\n class=\"btn-dot btn-dot--danger\"\n [attr.aria-label]=\"'Remove' | translate\"\n tooltip=\"Remove\"\n placement=\"top\"\n *ngIf=\"alarmType.__target === null\"\n [delay]=\"500\"\n (click)=\"removeCustomAlarm(alarmType); $event.stopPropagation()\"\n >\n <i c8yIcon=\"minus-circle\"></i>\n </button>\n </div>\n </c8y-li>\n <c8y-li *ngIf=\"possibleFilters.length === 0\">\n <c8y-ui-empty-state\n class=\"p-t-8\"\n icon=\"c8y-alarm\"\n [title]=\"'No alarm found' | translate\"\n [subtitle]=\"\n 'There is no alarm to filter. You can still add a custom alarm.' | translate\n \"\n [horizontal]=\"true\"\n ></c8y-ui-empty-state>\n </c8y-li>\n <div class=\"sticky-bottom p-16\">\n <button\n class=\"btn btn-block btn-primary\"\n [disabled]=\"possibleFilters.length === 0\"\n (click)=\"applyFilterChange(); $event.stopPropagation(); filtersDropdown.hide()\"\n translate\n >\n Apply\n </button>\n </div>\n </c8y-list-group>\n </div>\n </div>\n</div>\n" }]
|
|
152
|
+
}], ctorParameters: () => [{ type: i1.AlarmEventSelectorService }, { type: i2.ActivatedRoute }, { type: i2.Router }, { type: i3.ColorService }], propDecorators: { alarms: [{
|
|
153
|
+
type: Input
|
|
154
|
+
}], possibleFilters: [{
|
|
155
|
+
type: Input
|
|
156
|
+
}], activeFilters: [{
|
|
127
157
|
type: Input
|
|
128
158
|
}], onFilterChanged: [{
|
|
129
159
|
type: Output
|
|
130
160
|
}] } });
|
|
131
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alarms-type-filter.component.js","sourceRoot":"","sources":["../../../alarms/alarms-type-filter.component.ts","../../../alarms/alarms-type-filter.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges\n} from '@angular/core';\nimport { FormBuilder, FormControl } from '@angular/forms';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { IAlarm, IResultList } from '@c8y/client';\nimport { AlarmDetails, AlarmEventSelectorService } from '@c8y/ngx-components/alarm-event-selector';\nimport { Subject } from 'rxjs';\nimport { map, takeUntil } from 'rxjs/operators';\n\n@Component({\n  selector: 'c8y-alarms-type-filter',\n  templateUrl: './alarms-type-filter.component.html'\n})\nexport class AlarmsTypeFilterComponent implements OnInit, OnDestroy, OnChanges {\n  @Input()\n  alarms: IResultList<IAlarm>;\n\n  activeFilters: AlarmDetails[] = [];\n\n  @Output()\n  onFilterChanged: EventEmitter<AlarmDetails[]> = new EventEmitter();\n\n  readonly formControlName = 'alarms';\n  readonly queryParamName = 'typeFilter';\n\n  formGroup = this.formBuilder.group({\n    [this.formControlName]: new FormControl([])\n  });\n\n  private destroy$ = new Subject<void>();\n  private queryParam = '';\n\n  constructor(\n    private formBuilder: FormBuilder,\n    private alarmEventSelectorService: AlarmEventSelectorService,\n    private activatedRoute: ActivatedRoute,\n    private router: Router\n  ) {}\n\n  ngOnInit(): void {\n    this.activatedRoute.queryParams\n      .pipe(\n        map(params => {\n          const alarms: AlarmDetails[] = this.formGroup.get(this.formControlName).value;\n          return this.getActiveAlarmFiltersFromQueryParameter(alarms, params[this.queryParamName]);\n        }),\n        takeUntil(this.destroy$)\n      )\n      .subscribe((alarmFilters: AlarmDetails[]) =>\n        this.formGroup.get(this.formControlName).patchValue(alarmFilters)\n      );\n\n    this.formGroup\n      .get(this.formControlName)\n      .valueChanges.pipe(takeUntil(this.destroy$))\n      .subscribe((alarmFilters: AlarmDetails[]) => this.applyFilterChange(alarmFilters));\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.alarms && changes.alarms.currentValue && this.activeFilters.length === 0) {\n      this.setAlarms();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  deselect(type: AlarmDetails) {\n    const alarms: AlarmDetails[] = this.formGroup.get(this.formControlName).value;\n    const alarmFilter = alarms.find(alarm => alarm.filters.type === type.filters.type);\n    alarmFilter.__active = false;\n    this.formGroup.get(this.formControlName).patchValue(alarms);\n  }\n\n  deselectAll() {\n    const alarms: AlarmDetails[] = this.formGroup.get(this.formControlName).value;\n    const allFilters = alarms.map(alarm => {\n      return {\n        ...alarm,\n        __active: false\n      };\n    });\n    this.formGroup.get(this.formControlName).patchValue(allFilters);\n  }\n\n  private getActiveAlarmFiltersFromQueryParameter(\n    alarmFilters: AlarmDetails[],\n    filterTypesQuery: string\n  ): AlarmDetails[] {\n    if (!filterTypesQuery) {\n      return alarmFilters;\n    }\n    const types = filterTypesQuery.split(',');\n    return alarmFilters.map((alarm: AlarmDetails) => ({\n      ...alarm,\n      __active: types.includes(alarm.filters.type)\n    }));\n  }\n\n  private applyFilterChange(alarmFilters: AlarmDetails[]) {\n    const actives = alarmFilters.filter((alarmFilter: AlarmDetails) => alarmFilter.__active);\n    const newQueryParam = this.getQueryParams(actives);\n    const hasChanged = newQueryParam !== this.queryParam;\n    if (hasChanged) {\n      this.activeFilters = actives;\n      this.onFilterChanged.emit(this.activeFilters);\n      this.router.navigate([], {\n        queryParams: {\n          typeFilter: newQueryParam || null\n        },\n        queryParamsHandling: 'merge'\n      });\n      this.queryParam = newQueryParam;\n    }\n  }\n\n  private getQueryParams(activeFilters: AlarmDetails[]): string {\n    return activeFilters.map(filter => filter.filters.type).join(',');\n  }\n\n  private async setAlarms() {\n    const queryParameters = this.activatedRoute.snapshot.queryParamMap.get(this.queryParamName);\n    const alarmTypesFromCurrentlyShownAlarms =\n      await this.alarmEventSelectorService.getUniqueAlarmsOnly(this.alarms.data);\n    const alarmTypesWithMissingAndCustom = await this.addMissingAlarmTypesFromQueryParams(\n      alarmTypesFromCurrentlyShownAlarms,\n      queryParameters\n    );\n    const selectableAlarmTypes = this.getActiveAlarmFiltersFromQueryParameter(\n      alarmTypesWithMissingAndCustom,\n      queryParameters\n    );\n\n    this.formGroup.get(this.formControlName).patchValue(selectableAlarmTypes);\n  }\n\n  private async addMissingAlarmTypesFromQueryParams(\n    alarmFilters: AlarmDetails[],\n    queryParameters: string\n  ): Promise<AlarmDetails[]> {\n    if (!queryParameters) {\n      return alarmFilters;\n    }\n    const types = queryParameters.split(',');\n    const missingFilters = await Promise.all(\n      types\n        .filter(type => !alarmFilters.find(alarmFilter => alarmFilter.filters.type === type))\n        .map(type => {\n          return this.alarmEventSelectorService.createItem('ALARM', {\n            type,\n            source: null\n          } as IAlarm);\n        })\n    );\n    return [...alarmFilters, ...missingFilters];\n  }\n}\n","<form\n  class=\"d-flex a-i-center\"\n  [formGroup]=\"formGroup\"\n>\n  <div\n    class=\"dropdown\"\n    title=\"{{ 'Filter by Alarm types' | translate }}\"\n    dropdown\n    #filtersDropdown=\"bs-dropdown\"\n    [cdkTrapFocus]=\"filtersDropdown.isOpen\"\n    [insideClick]=\"true\"\n  >\n    <div class=\"input-group fit-w\">\n      <div class=\"form-control d-flex a-i-center inner-scroll\">\n        <ng-container *ngIf=\"activeFilters.length > 0; else allTypes\">\n          <span\n            class=\"tag tag--info chip\"\n            *ngFor=\"let filter of activeFilters\"\n          >\n            <button\n              class=\"btn btn-xs btn-clean text-10\"\n              title=\"{{ 'Remove' | translate }}\"\n              type=\"button\"\n              (click)=\"$event.stopPropagation(); deselect(filter)\"\n            >\n              <i c8yIcon=\"times\"></i>\n            </button>\n            <span\n              class=\"circle-icon-wrapper circle-icon-wrapper--small\"\n              [ngStyle]=\"{ 'background-color': filter.color }\"\n            >\n              <i\n                class=\"stroked-icon\"\n                c8yIcon=\"bell\"\n              ></i>\n            </span>\n            {{ filter.filters.type }}\n          </span>\n        </ng-container>\n        <ng-template #allTypes>\n          <span class=\"text-nowrap\">\n            {{ 'All alarm types' | translate }}\n          </span>\n        </ng-template>\n      </div>\n      <div class=\"input-group-btn input-group-btn--last text-center\">\n        <button\n          class=\"btn-default btn\"\n          [title]=\"'Clear filters' | translate\"\n          *ngIf=\"activeFilters.length\"\n          (click)=\"deselectAll()\"\n        >\n          <i c8yIcon=\"times\"></i>\n        </button>\n        <button\n          class=\"btn-default btn btn--caret\"\n          [title]=\"'Alarm types' | translate\"\n          data-cy=\"c8y-alarm-type-filter\"\n          dropdownToggle\n        >\n          <i class=\"caret\"></i>\n        </button>\n      </div>\n    </div>\n    <div\n      class=\"dropdown-menu dropdown-menu-action-bar\"\n      style=\"max-width:unset; min-width: 250px\"\n      *dropdownMenu\n    >\n      <c8y-alarm-event-selection-list\n        [title]=\"'Alarm type filter' | translate\"\n        [attr.name]=\"formControlName\"\n        [config]=\"{\n          title: 'Alarm type filter' | translate,\n          allowChangingContext: false\n        }\"\n        [inline]=\"true\"\n        [hideSource]=\"true\"\n        [addButtonLabel]=\"'Manage alarm types' | translate\"\n        [canEdit]=\"false\"\n        [canDragAndDrop]=\"false\"\n        [canRemove]=\"false\"\n        [timelineType]=\"'ALARM'\"\n        [activeToggleAsSwitch]=\"false\"\n        [formControlName]=\"formControlName\"\n      >\n        <c8y-ui-empty-state\n          class=\"p-t-8\"\n          icon=\"c8y-alarm\"\n          [title]=\"'No alarm found' | translate\"\n          [subtitle]=\"'There is no alarm to filter. You can still add a custom alarm.' | translate\"\n          [horizontal]=\"true\"\n        ></c8y-ui-empty-state>\n      </c8y-alarm-event-selection-list>\n    </div>\n  </div>\n</form>\n"]}
|
|
161
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alarms-type-filter.component.js","sourceRoot":"","sources":["../../../alarms/alarms-type-filter.component.ts","../../../alarms/alarms-type-filter.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges\n} from '@angular/core';\nimport { ActivatedRoute, Router } from '@angular/router';\nimport { IAlarm, IResultList } from '@c8y/client';\nimport { ColorService } from '@c8y/ngx-components';\nimport { AlarmDetails, AlarmEventSelectorService } from '@c8y/ngx-components/alarm-event-selector';\nimport { map, Subject, takeUntil } from 'rxjs';\n\n@Component({\n  selector: 'c8y-alarms-type-filter',\n  templateUrl: './alarms-type-filter.component.html'\n})\nexport class AlarmsTypeFilterComponent implements OnInit, OnDestroy, OnChanges {\n  @Input()\n  alarms: IResultList<IAlarm>;\n\n  @Input()\n  possibleFilters: AlarmDetails[] = [];\n\n  @Input()\n  activeFilters: AlarmDetails[] = [];\n\n  @Output()\n  onFilterChanged: EventEmitter<AlarmDetails[]> = new EventEmitter();\n\n  customAlarmTypes: AlarmDetails[] = [];\n  customAlarmTypeInput = '';\n\n  readonly queryParamName = 'typeFilter';\n  readonly STORAGE_ACCESS_KEY = 'customAlarmTypes';\n\n  private destroy$ = new Subject<void>();\n  private currentQueryParam = '';\n\n  constructor(\n    private alarmEventSelectorService: AlarmEventSelectorService,\n    private activatedRoute: ActivatedRoute,\n    private router: Router,\n    private colorService: ColorService\n  ) {}\n\n  ngOnInit(): void {\n    this.setQueryParameterObservable();\n  }\n\n  async ngOnChanges(changes: SimpleChanges): Promise<void> {\n    if (changes.alarms && changes.alarms.currentValue && this.activeFilters.length === 0) {\n      await this.setPossibleFilters();\n      this.applyFilterChange();\n    }\n  }\n\n  setQueryParameterObservable() {\n    this.activatedRoute.queryParams\n      .pipe(\n        map(params => {\n          const alarms: AlarmDetails[] = this.possibleFilters;\n          const possibleFilters = this.setActiveAlarmFiltersFromQueryParameter(\n            alarms,\n            params[this.queryParamName]\n          );\n          return possibleFilters;\n        }),\n        takeUntil(this.destroy$)\n      )\n      .subscribe((possibleFilters: AlarmDetails[]) => {\n        this.possibleFilters = possibleFilters;\n        this.applyFilterChange();\n      });\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  toggleAlarmType(alarmType: AlarmDetails) {\n    alarmType.__active = !alarmType.__active;\n  }\n\n  deselect(type: AlarmDetails) {\n    const alarmFilter = this.possibleFilters.find(\n      alarm => alarm.filters.type === type.filters.type\n    );\n    alarmFilter.__active = false;\n    this.applyFilterChange();\n  }\n\n  deselectAll() {\n    this.possibleFilters = this.possibleFilters.map(alarm => {\n      return {\n        ...alarm,\n        __active: false\n      };\n    });\n    this.applyFilterChange();\n  }\n\n  applyFilterChange() {\n    const actives = this.possibleFilters.filter(\n      (alarmFilter: AlarmDetails) => alarmFilter.__active\n    );\n    const newQueryParam = this.getQueryParams(actives);\n    const hasChanged = newQueryParam !== this.currentQueryParam;\n    if (hasChanged) {\n      this.activeFilters = actives;\n      this.onFilterChanged.emit(this.activeFilters);\n      this.router.navigate([], {\n        queryParams: {\n          typeFilter: newQueryParam || null\n        },\n        queryParamsHandling: 'merge'\n      });\n      this.currentQueryParam = newQueryParam;\n    }\n  }\n\n  resetFilters() {\n    this.possibleFilters.forEach(possibleFilter => {\n      possibleFilter.__active = this.activeFilters.some(\n        (activeFilter: AlarmDetails) => activeFilter === possibleFilter\n      );\n    });\n  }\n\n  removeCustomAlarm(alarmDetails: AlarmDetails) {\n    this.possibleFilters = this.possibleFilters.filter(filter => filter !== alarmDetails);\n    this.storeCustomAlarmTypes();\n  }\n\n  confirmWithEnter(event: KeyboardEvent) {\n    if (event.key === 'Enter') {\n      this.addCustomAlarmType();\n    }\n  }\n\n  async addCustomAlarmType() {\n    if (!this.customAlarmTypeInput) {\n      return;\n    }\n    this.possibleFilters.unshift({\n      label: this.customAlarmTypeInput,\n      color: await this.colorService.generateColor(this.customAlarmTypeInput),\n      filters: {\n        type: this.customAlarmTypeInput\n      },\n      timelineType: 'ALARM',\n      __active: true,\n      __target: null\n    });\n    this.customAlarmTypeInput = '';\n    this.storeCustomAlarmTypes();\n  }\n\n  private storeCustomAlarmTypes() {\n    const customTypes = this.possibleFilters.filter((filter: AlarmDetails) => !filter.__target);\n    window.localStorage.setItem(this.STORAGE_ACCESS_KEY, JSON.stringify(customTypes));\n  }\n\n  private getCustomAlarmTypeFromStorage() {\n    const types = window.localStorage.getItem(this.STORAGE_ACCESS_KEY);\n    return types ? JSON.parse(types) : [];\n  }\n\n  private async setPossibleFilters() {\n    const queryParameters = this.activatedRoute.snapshot.queryParamMap.get(this.queryParamName);\n    const alarmTypesFromCurrentlyShownAlarms =\n      await this.alarmEventSelectorService.getUniqueAlarmsOnly(this.alarms.data);\n    const customAlarmTypesFromLocalStorage = this.getCustomAlarmTypeFromStorage();\n    const selectableAlarmTypes = this.setActiveAlarmFiltersFromQueryParameter(\n      [...customAlarmTypesFromLocalStorage, ...alarmTypesFromCurrentlyShownAlarms],\n      queryParameters\n    );\n    this.possibleFilters = selectableAlarmTypes;\n  }\n\n  private setActiveAlarmFiltersFromQueryParameter(\n    alarmFilters: AlarmDetails[],\n    filterTypesQuery = ''\n  ): AlarmDetails[] {\n    const types = (filterTypesQuery ?? '').split(',');\n    return alarmFilters.map((alarm: AlarmDetails) => ({\n      ...alarm,\n      __active: types.includes(alarm.filters.type)\n    }));\n  }\n\n  private getQueryParams(activeFilters: AlarmDetails[]): string {\n    return activeFilters.map(filter => filter.filters.type).join(',');\n  }\n}\n","<div class=\"d-flex a-i-center\">\n  <div\n    class=\"dropdown\"\n    title=\"{{ 'Filter by Alarm types' | translate }}\"\n    dropdown\n    #filtersDropdown=\"bs-dropdown\"\n    [cdkTrapFocus]=\"filtersDropdown.isOpen\"\n    (onHidden)=\"resetFilters()\"\n    [insideClick]=\"true\"\n  >\n    <div class=\"input-group fit-w\">\n      <div class=\"form-control d-flex a-i-center inner-scroll\">\n        <ng-container *ngIf=\"activeFilters.length > 0; else allTypes\">\n          <span\n            class=\"tag tag--info chip\"\n            *ngFor=\"let filter of activeFilters\"\n          >\n            <button\n              class=\"btn btn-xs btn-clean text-10\"\n              title=\"{{ 'Remove' | translate }}\"\n              type=\"button\"\n              (click)=\"$event.stopPropagation(); deselect(filter)\"\n            >\n              <i c8yIcon=\"times\"></i>\n            </button>\n            <span\n              class=\"circle-icon-wrapper circle-icon-wrapper--small\"\n              [ngStyle]=\"{ 'background-color': filter.color }\"\n            >\n              <i\n                class=\"stroked-icon\"\n                c8yIcon=\"bell\"\n              ></i>\n            </span>\n            {{ filter.filters.type }}\n          </span>\n        </ng-container>\n        <ng-template #allTypes>\n          <span class=\"text-nowrap\">\n            {{ 'All alarm types' | translate }}\n          </span>\n        </ng-template>\n      </div>\n      <div class=\"input-group-btn input-group-btn--last text-center\">\n        <button\n          class=\"btn-default btn\"\n          [title]=\"'Clear filters' | translate\"\n          *ngIf=\"activeFilters.length\"\n          (click)=\"deselectAll()\"\n        >\n          <i c8yIcon=\"times\"></i>\n        </button>\n        <button\n          class=\"btn-default btn btn--caret\"\n          [title]=\"'Alarm types' | translate\"\n          data-cy=\"c8y-alarm-type-filter\"\n          dropdownToggle\n        >\n          <i class=\"caret\"></i>\n        </button>\n      </div>\n    </div>\n    <div\n      class=\"dropdown-menu dropdown-menu-action-bar\"\n      style=\"max-width: unset; min-width: 250px\"\n      *dropdownMenu\n    >\n      <div class=\"p-16 bg-level-2\">\n        <div>\n          <p>\n            <i\n              class=\"text-info m-r-4\"\n              [c8yIcon]=\"'info-circle'\"\n            ></i>\n            <strong tanslate>The list below may not be complete.</strong>\n          </p>\n          <p tanslate>\n            Recent alarms are displayed below. Past alarms might not be shown. Optionally you can\n            add a custom alarm.\n          </p>\n        </div>\n      </div>\n      <c8y-list-group>\n        <div class=\"input-group p-t-16 p-b-16 p-r-32 p-l-32 separator-bottom\">\n          <input\n            class=\"form-control\"\n            placeholder=\"Custom alarm type\"\n            type=\"text\"\n            [(ngModel)]=\"customAlarmTypeInput\"\n            (keydown)=\"confirmWithEnter($event)\"\n          />\n          <div class=\"input-group-btn\">\n            <button\n              class=\"btn-dot text-primary\"\n              [attr.aria-label]=\"'Add custom alarm' | translate\"\n              tooltip=\"Add\"\n              placement=\"top\"\n              [delay]=\"500\"\n              (click)=\"addCustomAlarmType()\"\n            >\n              <i c8yIcon=\"plus-circle\"></i>\n            </button>\n          </div>\n        </div>\n\n        <c8y-li\n          class=\"c8y-list__item__collapse--container-small cdk-drag\"\n          style=\"cursor: pointer\"\n          *ngFor=\"let alarmType of possibleFilters\"\n          (click)=\"toggleAlarmType(alarmType)\"\n        >\n          <c8y-li-checkbox\n            class=\"a-s-center m-t-4 p-r-0 p-l-0\"\n            [selected]=\"alarmType.__active\"\n            (click)=\"$event.stopPropagation()\"\n            (change)=\"toggleAlarmType(alarmType); $event.stopPropagation()\"\n          ></c8y-li-checkbox>\n          <div class=\"d-flex a-i-center p-l-4\">\n            <div class=\"c8y-list__item__colorpicker p-t-0 p-b-0 p-l-0\">\n              <div class=\"c8y-colorpicker c8y-colorpicker--alarm\">\n                <span\n                  class=\"circle-icon-wrapper\"\n                  [ngStyle]=\"{ 'background-color': alarmType.color }\"\n                >\n                  <i\n                    class=\"stroked-icon\"\n                    [c8yIcon]=\"'bell'\"\n                  ></i>\n                </span>\n              </div>\n            </div>\n            <span class=\"text-truncate text-12 flex-grow\">\n              {{ alarmType.label }}\n            </span>\n            <button\n              class=\"btn-dot btn-dot--danger\"\n              [attr.aria-label]=\"'Remove' | translate\"\n              tooltip=\"Remove\"\n              placement=\"top\"\n              *ngIf=\"alarmType.__target === null\"\n              [delay]=\"500\"\n              (click)=\"removeCustomAlarm(alarmType); $event.stopPropagation()\"\n            >\n              <i c8yIcon=\"minus-circle\"></i>\n            </button>\n          </div>\n        </c8y-li>\n        <c8y-li *ngIf=\"possibleFilters.length === 0\">\n          <c8y-ui-empty-state\n            class=\"p-t-8\"\n            icon=\"c8y-alarm\"\n            [title]=\"'No alarm found' | translate\"\n            [subtitle]=\"\n              'There is no alarm to filter. You can still add a custom alarm.' | translate\n            \"\n            [horizontal]=\"true\"\n          ></c8y-ui-empty-state>\n        </c8y-li>\n        <div class=\"sticky-bottom p-16\">\n          <button\n            class=\"btn btn-block btn-primary\"\n            [disabled]=\"possibleFilters.length === 0\"\n            (click)=\"applyFilterChange(); $event.stopPropagation(); filtersDropdown.hide()\"\n            translate\n          >\n            Apply\n          </button>\n        </div>\n      </c8y-list-group>\n    </div>\n  </div>\n</div>\n"]}
|
|
@@ -2,7 +2,7 @@ import { Injectable } from '@angular/core';
|
|
|
2
2
|
import { ALARM_STATUS_LABELS, AlarmService, AlarmStatus, Severity } from '@c8y/client';
|
|
3
3
|
import { OptionsService, ViewContext, gettext } from '@c8y/ngx-components';
|
|
4
4
|
import { BehaviorSubject, Subject } from 'rxjs';
|
|
5
|
-
import { ALARMS_PATH } from './alarms.model';
|
|
5
|
+
import { ALARMS_PATH, INTERVALS_EXTENDED } from './alarms.model';
|
|
6
6
|
import * as i0 from "@angular/core";
|
|
7
7
|
import * as i1 from "@c8y/client";
|
|
8
8
|
import * as i2 from "@c8y/ngx-components";
|
|
@@ -200,6 +200,20 @@ export class AlarmsViewService {
|
|
|
200
200
|
getRouterNavigationArray(contextData, alarm) {
|
|
201
201
|
return this.getRouterLink(contextData, alarm).split('/').filter(Boolean);
|
|
202
202
|
}
|
|
203
|
+
/**
|
|
204
|
+
* Returns the correct from and to dates based on the selected interval
|
|
205
|
+
* @param intervalId the selected interval. E.g. 'none', 'hours', 'custom' ...
|
|
206
|
+
* @returns The calculated date context based on the selected interval.
|
|
207
|
+
*/
|
|
208
|
+
getDateTimeContextByInterval(intervalId) {
|
|
209
|
+
const interval = INTERVALS_EXTENDED.find(({ id }) => id === intervalId);
|
|
210
|
+
if (interval.id === 'none') {
|
|
211
|
+
return [new Date(0), new Date()];
|
|
212
|
+
}
|
|
213
|
+
const dateTo = new Date();
|
|
214
|
+
const dateFrom = new Date(dateTo.valueOf() - interval.timespanInMs);
|
|
215
|
+
return [dateFrom, dateTo];
|
|
216
|
+
}
|
|
203
217
|
/**
|
|
204
218
|
* Creates a value for query parameter for filtering alarms by severity based on array of selected severities.
|
|
205
219
|
*
|
|
@@ -242,4 +256,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.9", ngImpor
|
|
|
242
256
|
providedIn: 'root'
|
|
243
257
|
}]
|
|
244
258
|
}], ctorParameters: () => [{ type: i1.AlarmService }, { type: i2.OptionsService }] });
|
|
245
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alarms-view.service.js","sourceRoot":"","sources":["../../../alarms/alarms-view.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport {\n  ALARM_STATUS_LABELS,\n  AlarmQueryFilter,\n  AlarmService,\n  AlarmStatus,\n  AlarmStatusType,\n  IAlarm,\n  IResultList,\n  Severity,\n  SeverityFilter,\n  SeverityType\n} from '@c8y/client';\nimport {\n  ApplicationOptions,\n  ContextData,\n  DateTimeContext,\n  OptionsService,\n  ViewContext,\n  gettext\n} from '@c8y/ngx-components';\nimport { BehaviorSubject, Observable, Subject } from 'rxjs';\nimport { ALARMS_PATH } from './alarms.model';\n\n/**\n * Service for managing and retrieving alarms data within the alarms view.\n *\n * The `AlarmsViewService` provides functionality to interact with alarms,\n * including filtering, counting, and translation-related operations in an alarms view.\n *\n * This service relies on the `AlarmService` for fetching alarm data and the `OptionsService`\n * for configuring alarms view options.\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class AlarmsViewService {\n  readonly ALARM_REFRESH_TYPE_KEY: keyof ApplicationOptions = 'alarmsRefreshType';\n  readonly DEFAULT_INTERVAL_VALUE = 30_000;\n  readonly DEFAULT_REFRESH_OPTION_VALUE = 'interval';\n  readonly DEFAULT_INTERVAL_VALUES = [5_000, 10_000, 15_000, 30_000, 60_000];\n  readonly REALTIME_UPDATE_ALARMS_MESSAGE = gettext('The list was updated, click to refresh.');\n\n  isIntervalEnabled$: Observable<boolean>;\n  reloadAlarmsList$ = new Subject<void>();\n\n  private _isIntervalEnabled: BehaviorSubject<boolean>;\n\n  constructor(\n    private alarmService: AlarmService,\n    private optionsService: OptionsService\n  ) {\n    if (this.isIntervalRefresh()) {\n      this._isIntervalEnabled = new BehaviorSubject<boolean>(true);\n      this.isIntervalEnabled$ = this._isIntervalEnabled.asObservable();\n    }\n  }\n\n  /**\n   * Emits a subject to initialize the alarms reloading.\n   */\n  updateAlarmList(): void {\n    this.reloadAlarmsList$.next();\n  }\n\n  /**\n   * Retrieves a list of alarms filtered by specified severities and other optional query filters.\n   *\n   * @param severities an array of severities to filter the alarms.\n   * @param showCleared flag indicating whether to show cleared alarms. Defaults to false.\n   * @param selectedDates an array of two dates to filter alarms by creation and last update dates.\n   * @param filter additional query filters for retrieving alarms.\n   *\n   * @returns A promise that resolves to a list of alarms satisfying the specified filters.\n   */\n  retrieveFilteredAlarms(\n    severities: SeverityType[],\n    showCleared = false,\n    selectedDates?: [Date, Date],\n    filter?: AlarmQueryFilter\n  ): Promise<IResultList<IAlarm>> {\n    const severitiesQuery = this.getSeverityQueryParameter(severities);\n    const statusesQuery = this.getStatusQueryParameter(showCleared);\n\n    const _filter: AlarmQueryFilter = {\n      pageSize: 50,\n      withTotalPages: true,\n      ...(severitiesQuery && { severity: severitiesQuery }),\n      ...(statusesQuery && { status: statusesQuery }),\n      ...(selectedDates && {\n        lastUpdatedFrom: selectedDates[0].toISOString(),\n        createdTo: selectedDates[1].toISOString()\n      }),\n      ...filter\n    };\n    return this.alarmService.list(_filter);\n  }\n\n  retrieveAlarmsByDate(dates: DateTimeContext) {\n    return this.alarmService.list({\n      lastUpdatedFrom: dates[0].toISOString(),\n      createdTo: dates[1].toISOString(),\n      pageSize: 50,\n      withTotalPages: true\n    });\n  }\n\n  /**\n   * Updates the state to enable or disable intervals.\n   * @param value - A boolean value to indicate whether to enable intervals.\n   */\n  updateIntervalState(value: boolean): void {\n    this._isIntervalEnabled?.next(value);\n  }\n\n  /**\n   * Fetches the count of alarms filtered by severity and clearance status.\n   *\n   * @param severity - The severity level to filter by (e.g., CRITICAL, MAJOR, etc.).\n   * @param showCleared - Whether or not to include cleared alarms in the count.\n   * @param filter - Additional filter criteria for alarms.\n   *\n   * @returns A promise that resolves to the number of alarms that match the filter criteria.\n   *\n   */\n  async getAlarmsCountBySeverity(\n    severity: SeverityType,\n    showCleared: boolean,\n    filter?: AlarmQueryFilter\n  ): Promise<number> {\n    const statusesQuery = this.getStatusQueryParameter(showCleared);\n    const _filter: AlarmQueryFilter = {\n      ...(severity && { severity: severity }),\n      ...(statusesQuery && { status: statusesQuery }),\n      ...filter\n    };\n    const { data } = await this.alarmService.count(_filter);\n\n    return data;\n  }\n\n  /**\n   * Retrieves the current alarms refresh type from the OptionsService\n   * and determines whether it is set to \"interval\".\n   *\n   * @returns `true` if the alarms refresh type is \"interval,\" otherwise `false`.\n   */\n  isIntervalRefresh(): boolean {\n    const value = this.optionsService.get(this.ALARM_REFRESH_TYPE_KEY, 'interval');\n    return value === 'interval';\n  }\n\n  /**\n   * Updates the list of selected severities based on the new severity filter.\n   *\n   * @param severityUpdates - The object representing the updates to each severity.\n   *\n   * @returns An array representing the updated selected severities.\n   */\n  updateSelectedSeverities(severityUpdates: SeverityFilter): SeverityType[] {\n    return (Object.keys(severityUpdates) as SeverityType[])\n      .filter(key => severityUpdates[key])\n      .map(key => key.toUpperCase() as SeverityType);\n  }\n\n  /**\n   * Clears all active alarms of the selected severities.\n   *\n   * This method clears all active alarms for the given list of severities by making bulk update calls. If no severities are selected, it defaults to using all available severities.\n   * It works by sending a series of update requests for each severity and returns a Promise that resolves with an object indicating if all alarms were resolved immediately.\n   *\n   * @param selectedSeverities An array of severities to be cleared. If not provided, all severities will be cleared.\n   * @param sourceId - Identifier for the source associated with the alarms to be cleared.\n   *\n   * @returns A Promise that resolves with an object with a flag `resolvedImmediately`. The flag is true if all alarms for all selected severities were cleared successfully; otherwise false.\n   *\n   * **Example**\n   * ```typescript\n   * const severitiesToClear: SeverityType[] = [Severity.MAJOR, Severity.MINOR];\n   *\n   * clearAllActiveAlarms(severitiesToClear).then(({ resolvedImmediately }) => {\n   *   if (resolvedImmediately) {\n   *     console.log('All selected alarms were cleared successfully.');\n   *   } else {\n   *     console.log('Some alarms could not be cleared.');\n   *   }\n   * });\n   * ```\n   *\n   * **Note**\n   * - The method uses the `alarmService.updateBulk` for each severity to clear the active alarms.\n   * - It may fetch the `sourceId` based on the view (if applicable) and include it as a query parameter in the update calls.\n   * - The method returns immediately but the returned Promise needs to have a `then` or `catch` method call to handle the result or error respectively.\n   * - Uses `Promise.all` to wait for all update requests to complete before resolving the final result.\n   */\n  async clearAllActiveAlarms(\n    selectedSeverities: SeverityType[],\n    sourceId: string | number\n  ): Promise<{ resolvedImmediately: boolean }> {\n    const severitiesToUpdate = selectedSeverities || Severity;\n\n    const promises = Object.values(severitiesToUpdate).map((severity: SeverityType) => {\n      const commonParams = { resolved: false, severity };\n\n      const parameters = sourceId\n        ? {\n            ...commonParams,\n            source: sourceId,\n            withSourceAssets: true,\n            withSourceDevices: true\n          }\n        : commonParams;\n\n      return this.alarmService.updateBulk({ status: AlarmStatus.CLEARED }, parameters);\n    });\n\n    const responses = await Promise.all(promises);\n    return {\n      resolvedImmediately: responses.every(res => res)\n    };\n  }\n\n  /**\n   * Returns the correct link based on the provided context data.\n   * @param contextData The context the navigation was triggered from.\n   * @param alarm The alarm to navigate to.\n   * @returns A link to be used as an url navigation.\n   */\n  getRouterLink(contextData?: ContextData, alarm?: IAlarm): string {\n    let detailUrl = `/${ALARMS_PATH}`;\n    if (alarm) {\n      detailUrl = `/${ALARMS_PATH}/${alarm.id}`;\n    }\n    if (!contextData) {\n      return detailUrl;\n    }\n\n    switch (contextData.context) {\n      case ViewContext.Device:\n        return `/device/${contextData.contextData.id}${detailUrl}`;\n      case ViewContext.Group:\n        return `/group/${contextData.contextData.id}${detailUrl}`;\n      default:\n        return detailUrl;\n    }\n  }\n\n  /**\n   * Returns the correct array navigation.\n   * @param contextData The context the navigation was triggered from.\n   * @param alarm The alarm to navigate to.\n   * @returns A link to be used as a router.navigation.\n   */\n  getRouterNavigationArray(contextData?: ContextData, alarm?: IAlarm): string[] {\n    return this.getRouterLink(contextData, alarm).split('/').filter(Boolean);\n  }\n\n  /**\n   * Creates a value for query parameter for filtering alarms by severity based on array of selected severities.\n   *\n   * @param severities - An array of alarm severity types to include in the filter.\n   * If the array is empty or undefined, no severity filter will be applied.\n   *\n   * @returns A comma-separated string of selected alarm severities,\n   * or null if no severities are provided.\n   */\n  private getSeverityQueryParameter(severities: SeverityType[]): string | null {\n    if (!severities || severities.length === 0) {\n      return;\n    }\n    if (severities.length === Object.keys(Severity).length) {\n      return;\n    }\n\n    return severities.join(',');\n  }\n\n  /**\n   * Creates a value for query parameter for filtering alarms by statuses based on showCleared option.\n   *\n   * @param showCleared - A flag indicating whether to include cleared statuses.\n   * If true, all statuses, including 'CLEARED', will be included; if false, 'CLEARED' will be excluded.\n   *\n   * @returns A comma-separated string of alarm statuses.\n   */\n  private getStatusQueryParameter(showCleared: boolean): string {\n    const statuses = Object.keys(ALARM_STATUS_LABELS) as AlarmStatusType[];\n    const filteredStatuses = showCleared\n      ? statuses\n      : statuses.filter(status => status !== 'CLEARED');\n    return filteredStatuses.join(',');\n  }\n}\n"]}
|
|
259
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alarms-view.service.js","sourceRoot":"","sources":["../../../alarms/alarms-view.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport {\n  ALARM_STATUS_LABELS,\n  AlarmQueryFilter,\n  AlarmService,\n  AlarmStatus,\n  AlarmStatusType,\n  IAlarm,\n  IResultList,\n  Severity,\n  SeverityFilter,\n  SeverityType\n} from '@c8y/client';\nimport {\n  ApplicationOptions,\n  ContextData,\n  DateTimeContext,\n  OptionsService,\n  ViewContext,\n  gettext\n} from '@c8y/ngx-components';\nimport { BehaviorSubject, Observable, Subject } from 'rxjs';\nimport { ALARMS_PATH, INTERVALS_EXTENDED } from './alarms.model';\nimport { AlarmFilterInterval } from '@c8y/ngx-components/interval-picker';\n\n/**\n * Service for managing and retrieving alarms data within the alarms view.\n *\n * The `AlarmsViewService` provides functionality to interact with alarms,\n * including filtering, counting, and translation-related operations in an alarms view.\n *\n * This service relies on the `AlarmService` for fetching alarm data and the `OptionsService`\n * for configuring alarms view options.\n */\n@Injectable({\n  providedIn: 'root'\n})\nexport class AlarmsViewService {\n  readonly ALARM_REFRESH_TYPE_KEY: keyof ApplicationOptions = 'alarmsRefreshType';\n  readonly DEFAULT_INTERVAL_VALUE = 30_000;\n  readonly DEFAULT_REFRESH_OPTION_VALUE = 'interval';\n  readonly DEFAULT_INTERVAL_VALUES = [5_000, 10_000, 15_000, 30_000, 60_000];\n  readonly REALTIME_UPDATE_ALARMS_MESSAGE = gettext('The list was updated, click to refresh.');\n\n  isIntervalEnabled$: Observable<boolean>;\n  reloadAlarmsList$ = new Subject<void>();\n\n  private _isIntervalEnabled: BehaviorSubject<boolean>;\n\n  constructor(\n    private alarmService: AlarmService,\n    private optionsService: OptionsService\n  ) {\n    if (this.isIntervalRefresh()) {\n      this._isIntervalEnabled = new BehaviorSubject<boolean>(true);\n      this.isIntervalEnabled$ = this._isIntervalEnabled.asObservable();\n    }\n  }\n\n  /**\n   * Emits a subject to initialize the alarms reloading.\n   */\n  updateAlarmList(): void {\n    this.reloadAlarmsList$.next();\n  }\n\n  /**\n   * Retrieves a list of alarms filtered by specified severities and other optional query filters.\n   *\n   * @param severities an array of severities to filter the alarms.\n   * @param showCleared flag indicating whether to show cleared alarms. Defaults to false.\n   * @param selectedDates an array of two dates to filter alarms by creation and last update dates.\n   * @param filter additional query filters for retrieving alarms.\n   *\n   * @returns A promise that resolves to a list of alarms satisfying the specified filters.\n   */\n  retrieveFilteredAlarms(\n    severities: SeverityType[],\n    showCleared = false,\n    selectedDates?: [Date, Date],\n    filter?: AlarmQueryFilter\n  ): Promise<IResultList<IAlarm>> {\n    const severitiesQuery = this.getSeverityQueryParameter(severities);\n    const statusesQuery = this.getStatusQueryParameter(showCleared);\n\n    const _filter: AlarmQueryFilter = {\n      pageSize: 50,\n      withTotalPages: true,\n      ...(severitiesQuery && { severity: severitiesQuery }),\n      ...(statusesQuery && { status: statusesQuery }),\n      ...(selectedDates && {\n        lastUpdatedFrom: selectedDates[0].toISOString(),\n        createdTo: selectedDates[1].toISOString()\n      }),\n      ...filter\n    };\n    return this.alarmService.list(_filter);\n  }\n\n  retrieveAlarmsByDate(dates: DateTimeContext) {\n    return this.alarmService.list({\n      lastUpdatedFrom: dates[0].toISOString(),\n      createdTo: dates[1].toISOString(),\n      pageSize: 50,\n      withTotalPages: true\n    });\n  }\n\n  /**\n   * Updates the state to enable or disable intervals.\n   * @param value - A boolean value to indicate whether to enable intervals.\n   */\n  updateIntervalState(value: boolean): void {\n    this._isIntervalEnabled?.next(value);\n  }\n\n  /**\n   * Fetches the count of alarms filtered by severity and clearance status.\n   *\n   * @param severity - The severity level to filter by (e.g., CRITICAL, MAJOR, etc.).\n   * @param showCleared - Whether or not to include cleared alarms in the count.\n   * @param filter - Additional filter criteria for alarms.\n   *\n   * @returns A promise that resolves to the number of alarms that match the filter criteria.\n   *\n   */\n  async getAlarmsCountBySeverity(\n    severity: SeverityType,\n    showCleared: boolean,\n    filter?: AlarmQueryFilter\n  ): Promise<number> {\n    const statusesQuery = this.getStatusQueryParameter(showCleared);\n    const _filter: AlarmQueryFilter = {\n      ...(severity && { severity: severity }),\n      ...(statusesQuery && { status: statusesQuery }),\n      ...filter\n    };\n    const { data } = await this.alarmService.count(_filter);\n\n    return data;\n  }\n\n  /**\n   * Retrieves the current alarms refresh type from the OptionsService\n   * and determines whether it is set to \"interval\".\n   *\n   * @returns `true` if the alarms refresh type is \"interval,\" otherwise `false`.\n   */\n  isIntervalRefresh(): boolean {\n    const value = this.optionsService.get(this.ALARM_REFRESH_TYPE_KEY, 'interval');\n    return value === 'interval';\n  }\n\n  /**\n   * Updates the list of selected severities based on the new severity filter.\n   *\n   * @param severityUpdates - The object representing the updates to each severity.\n   *\n   * @returns An array representing the updated selected severities.\n   */\n  updateSelectedSeverities(severityUpdates: SeverityFilter): SeverityType[] {\n    return (Object.keys(severityUpdates) as SeverityType[])\n      .filter(key => severityUpdates[key])\n      .map(key => key.toUpperCase() as SeverityType);\n  }\n\n  /**\n   * Clears all active alarms of the selected severities.\n   *\n   * This method clears all active alarms for the given list of severities by making bulk update calls. If no severities are selected, it defaults to using all available severities.\n   * It works by sending a series of update requests for each severity and returns a Promise that resolves with an object indicating if all alarms were resolved immediately.\n   *\n   * @param selectedSeverities An array of severities to be cleared. If not provided, all severities will be cleared.\n   * @param sourceId - Identifier for the source associated with the alarms to be cleared.\n   *\n   * @returns A Promise that resolves with an object with a flag `resolvedImmediately`. The flag is true if all alarms for all selected severities were cleared successfully; otherwise false.\n   *\n   * **Example**\n   * ```typescript\n   * const severitiesToClear: SeverityType[] = [Severity.MAJOR, Severity.MINOR];\n   *\n   * clearAllActiveAlarms(severitiesToClear).then(({ resolvedImmediately }) => {\n   *   if (resolvedImmediately) {\n   *     console.log('All selected alarms were cleared successfully.');\n   *   } else {\n   *     console.log('Some alarms could not be cleared.');\n   *   }\n   * });\n   * ```\n   *\n   * **Note**\n   * - The method uses the `alarmService.updateBulk` for each severity to clear the active alarms.\n   * - It may fetch the `sourceId` based on the view (if applicable) and include it as a query parameter in the update calls.\n   * - The method returns immediately but the returned Promise needs to have a `then` or `catch` method call to handle the result or error respectively.\n   * - Uses `Promise.all` to wait for all update requests to complete before resolving the final result.\n   */\n  async clearAllActiveAlarms(\n    selectedSeverities: SeverityType[],\n    sourceId: string | number\n  ): Promise<{ resolvedImmediately: boolean }> {\n    const severitiesToUpdate = selectedSeverities || Severity;\n\n    const promises = Object.values(severitiesToUpdate).map((severity: SeverityType) => {\n      const commonParams = { resolved: false, severity };\n\n      const parameters = sourceId\n        ? {\n            ...commonParams,\n            source: sourceId,\n            withSourceAssets: true,\n            withSourceDevices: true\n          }\n        : commonParams;\n\n      return this.alarmService.updateBulk({ status: AlarmStatus.CLEARED }, parameters);\n    });\n\n    const responses = await Promise.all(promises);\n    return {\n      resolvedImmediately: responses.every(res => res)\n    };\n  }\n\n  /**\n   * Returns the correct link based on the provided context data.\n   * @param contextData The context the navigation was triggered from.\n   * @param alarm The alarm to navigate to.\n   * @returns A link to be used as an url navigation.\n   */\n  getRouterLink(contextData?: ContextData, alarm?: IAlarm): string {\n    let detailUrl = `/${ALARMS_PATH}`;\n    if (alarm) {\n      detailUrl = `/${ALARMS_PATH}/${alarm.id}`;\n    }\n    if (!contextData) {\n      return detailUrl;\n    }\n\n    switch (contextData.context) {\n      case ViewContext.Device:\n        return `/device/${contextData.contextData.id}${detailUrl}`;\n      case ViewContext.Group:\n        return `/group/${contextData.contextData.id}${detailUrl}`;\n      default:\n        return detailUrl;\n    }\n  }\n\n  /**\n   * Returns the correct array navigation.\n   * @param contextData The context the navigation was triggered from.\n   * @param alarm The alarm to navigate to.\n   * @returns A link to be used as a router.navigation.\n   */\n  getRouterNavigationArray(contextData?: ContextData, alarm?: IAlarm): string[] {\n    return this.getRouterLink(contextData, alarm).split('/').filter(Boolean);\n  }\n\n  /**\n   * Returns the correct from and to dates based on the selected interval\n   * @param intervalId the selected interval. E.g. 'none', 'hours', 'custom' ...\n   * @returns The calculated date context based on the selected interval.\n   */\n  getDateTimeContextByInterval(intervalId: AlarmFilterInterval['id']): DateTimeContext {\n    const interval = INTERVALS_EXTENDED.find(({ id }) => id === intervalId);\n    if (interval.id === 'none') {\n      return [new Date(0), new Date()];\n    }\n\n    const dateTo = new Date();\n    const dateFrom = new Date(dateTo.valueOf() - interval.timespanInMs);\n    return [dateFrom, dateTo];\n  }\n\n  /**\n   * Creates a value for query parameter for filtering alarms by severity based on array of selected severities.\n   *\n   * @param severities - An array of alarm severity types to include in the filter.\n   * If the array is empty or undefined, no severity filter will be applied.\n   *\n   * @returns A comma-separated string of selected alarm severities,\n   * or null if no severities are provided.\n   */\n  private getSeverityQueryParameter(severities: SeverityType[]): string | null {\n    if (!severities || severities.length === 0) {\n      return;\n    }\n    if (severities.length === Object.keys(Severity).length) {\n      return;\n    }\n\n    return severities.join(',');\n  }\n\n  /**\n   * Creates a value for query parameter for filtering alarms by statuses based on showCleared option.\n   *\n   * @param showCleared - A flag indicating whether to include cleared statuses.\n   * If true, all statuses, including 'CLEARED', will be included; if false, 'CLEARED' will be excluded.\n   *\n   * @returns A comma-separated string of alarm statuses.\n   */\n  private getStatusQueryParameter(showCleared: boolean): string {\n    const statuses = Object.keys(ALARM_STATUS_LABELS) as AlarmStatusType[];\n    const filteredStatuses = showCleared\n      ? statuses\n      : statuses.filter(status => status !== 'CLEARED');\n    return filteredStatuses.join(',');\n  }\n}\n"]}
|