@c8y/ngx-components
Version:
Angular modules for Cumulocity IoT applications
177 lines • 48.5 kB
JavaScript
import { Component, ElementRef, EventEmitter, Input, Output, ViewChild } from '@angular/core';
import { ActivatedRoute, NavigationEnd, Router } from '@angular/router';
import { ContextRouteService, DynamicComponentAlert, DynamicComponentAlertAggregator, gettext } from '@c8y/ngx-components';
import { BehaviorSubject, Observable, Subject, fromEvent, pipe } from 'rxjs';
import { debounceTime, distinctUntilChanged, filter, map, shareReplay, switchMap, takeUntil } from 'rxjs/operators';
import { AlarmsViewService } from './alarms-view.service';
import * as i0 from "@angular/core";
import * as i1 from "@angular/router";
import * as i2 from "./alarms-view.service";
import * as i3 from "@c8y/ngx-components";
import * as i4 from "@angular/common";
import * as i5 from "ngx-bootstrap/tooltip";
import * as i6 from "@ngx-translate/core";
import * as i7 from "./alarms-icon.component";
import * as i8 from "./alarm-list-custom-indicator/alarm-list-indicator.pipe";
export class AlarmsListComponent {
constructor(activatedRoute, alarmsViewService, contextRouteService, router) {
this.activatedRoute = activatedRoute;
this.alarmsViewService = alarmsViewService;
this.contextRouteService = contextRouteService;
this.router = router;
this.alarmBadgeTooltip = gettext('Number of occurrences`number of occurrences of alarm`. First occurrence {{ alarmFirstOccurrenceTime }} (device time).');
this.alarmLastOccurrenceLabel = gettext('Last occurrence of this alarm (device time).');
this.hasPermissions = true;
/**
* Input property for the currently applied type filters.
*/
this.typeFilters = [];
/**
* Input property for receiving load more mode.
*/
this.loadMoreMode = 'hidden';
/**
* Defines options, how the alarm list should be navigated if a user
* clicks on an alarm.
*/
this.navigationOptions = {
allowNavigationToAlarmsView: true,
alwaysNavigateToAllAlarms: false,
includeClearedQueryParams: false,
queryParamsHandling: 'merge'
};
/**
* Controls the visibility of the loading bar
* When set to `false`, the alarm list is displayed. When set to `true`, the opacity of alarms list is changed and a loading bar is shown.
*/
this.isInitialLoading = false;
/**
* Controls the visibility and functionality of some components
* When set to `true`, means the list is displayed in a split view layout:
* the list on the first column and the selected record detail on the second column (the cockpit
* alarms view for example)
* When set to false, the list is displayed as a standalone component, opening the detail will
* redirect to the alarms
*/
this.splitView = false;
/**
* Emits an instance of a selected alarm when one is chosen from the list.
*/
this.onSelectedAlarm = new EventEmitter();
/**
* Emits a boolean value indicating the scrolling state: true when the user starts scrolling, and false when the user reaches the top of the list.
*/
this.onScrollingStateChange = new EventEmitter();
/**
* Current alarm or last alarm marked as active by the routerLinkActive directive.
*/
this.activeAlarm$ = new BehaviorSubject(null);
this.activeChildParam$ = new Observable();
this.isScrolling = false;
/**
* Determines whether the c8y-loading component should be displayed.
* The loading component is shown when no alarms are displayed in the view or when the request is initial,
* as we don't want to see empty space on alarm list during loading.
*/
this.isEmptyListLoading = true;
this.alertAggregator = new DynamicComponentAlertAggregator();
this.mapAlarmLink = pipe(map((alarms) => alarms.map((alarm) => {
alarm.link = this.getRouterLink(alarm);
return alarm;
})));
this.destroy$ = new Subject();
this.HIDE_INTERVAL_COUNTDOWN_SCROLL = 50;
this.verifyIfFiltersMatchingAlarm();
}
/**
* Handles the change of the active route.
*
* @param isActive - A boolean indicating whether the route is active or not.
* @param scrollAnchor - The ListItemComponent used as a scroll anchor.
* @param alarm - The IAlarm object representing the active alarm.
*/
activeRouteChanged(isActive, scrollAnchor, alarm) {
if (isActive) {
scrollAnchor.element.nativeElement.scrollIntoView({
behavior: 'smooth',
block: 'nearest'
});
this.activeAlarm$.next(alarm);
}
}
ngOnChanges(changes) {
if (this.alarms && changes.alarms) {
this.activeAlarm$.next(null);
this.isEmptyListLoading = !this.alarms?.data?.length;
}
if (changes.hasPermissions?.currentValue === false) {
this.alertAggregator.addAlerts(new DynamicComponentAlert({
type: 'system',
text: gettext("You don't have permission to view alarms.")
}));
}
}
ngAfterViewInit() {
if (this.alarmsViewService.isIntervalRefresh()) {
const scrollElement = this.innerScrollWrapper.nativeElement;
fromEvent(scrollElement, 'scroll')
.pipe(takeUntil(this.destroy$), debounceTime(300))
.subscribe((event) => {
const target = event.target;
this.isScrolling = this.shouldCountdownIntervalBeHidden(target);
this.onScrollingStateChange.emit(this.isScrolling);
});
}
}
ngOnDestroy() {
this.destroy$.next();
this.destroy$.complete();
}
onAlarmOpen(alarm) {
this.onSelectedAlarm.emit(alarm);
}
getRouterLink(alarm) {
if (this.navigationOptions.alwaysNavigateToAllAlarms) {
return this.alarmsViewService.getRouterLink(null, alarm);
}
const contextData = this.contextRouteService.getContextData(this.activatedRoute);
return this.alarmsViewService.getRouterLink(contextData, alarm);
}
shouldCountdownIntervalBeHidden(target) {
const scrollTopPixels = target.scrollTop;
return scrollTopPixels > this.HIDE_INTERVAL_COUNTDOWN_SCROLL;
}
verifyIfFiltersMatchingAlarm() {
this.activeChildParam$ = this.router.events.pipe(filter(e => e instanceof NavigationEnd && this.activatedRoute.children.length > 0), switchMap(() => this.activatedRoute.children[0].params), map(params => params.id), distinctUntilChanged(), shareReplay(), takeUntil(this.destroy$));
// done to get the first navigation
this.activeChildParam$.subscribe();
}
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AlarmsListComponent, deps: [{ token: i1.ActivatedRoute }, { token: i2.AlarmsViewService }, { token: i3.ContextRouteService }, { token: i1.Router }], target: i0.ɵɵFactoryTarget.Component }); }
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: AlarmsListComponent, selector: "c8y-alarms-list", inputs: { alarms: "alarms", hasPermissions: "hasPermissions", typeFilters: "typeFilters", loadMoreMode: "loadMoreMode", navigationOptions: "navigationOptions", isInitialLoading: "isInitialLoading", splitView: "splitView" }, outputs: { onSelectedAlarm: "onSelectedAlarm", onScrollingStateChange: "onScrollingStateChange" }, viewQueries: [{ propertyName: "innerScrollWrapper", first: true, predicate: ["scrollWrapper"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"inner-scroll\"\n [ngClass]=\"{ 'split-view__list bg-level-1': splitView, 'bg-component': !splitView }\"\n data-cy=\"c8y-alarms-list\"\n #scrollWrapper\n>\n <div\n class=\"flex-wrap flex-no-shrink sticky-top m-b-16\"\n [ngClass]=\"{\n 'separator-bottom card-header p-b-0': splitView,\n 'd-flex fit-w separator-top-bottom widget-bar p-l-16 p-r-16':\n !splitView && navigationOptions.allowNavigationToAlarmsView\n }\"\n >\n <div\n class=\"h4 card-title\"\n *ngIf=\"splitView\"\n >\n {{ 'Alarms list' | translate }}\n </div>\n <div\n [ngClass]=\"{ 'fit-w d-flex a-i-center gap-16': !splitView, 'fit-h-20 m-l-auto': splitView }\"\n >\n <ng-content></ng-content>\n </div>\n <!-- Loading -->\n <div\n class=\"fit-w overflow-hidden\"\n [ngClass]=\"{ 'p-t-16': splitView }\"\n >\n <div\n class=\"loading-bar\"\n style=\"z-index: 101\"\n [ngClass]=\"{ active: isInitialLoading && !isEmptyListLoading }\"\n ></div>\n </div>\n\n <div\n class=\"alert alert-warning\"\n role=\"alert\"\n translate\n *ngIf=\"\n !isEmptyListLoading &&\n (activeChildParam$ | async) &&\n (activeAlarm$ | async)?.id !== (activeChildParam$ | async)\n \"\n >\n The selected alarm is not currently in the list, change your filter.\n </div>\n </div>\n <c8y-list-group\n class=\"p-r-16 interactive\"\n [ngStyle]=\"{ opacity: isInitialLoading && !isEmptyListLoading ? 0.2 : 1 }\"\n style=\"transition: opacity 0.15s linear\"\n data-cy=\"c8y-alarm-list--group\"\n >\n <c8y-li-timeline\n class=\"pointer\"\n role=\"button\"\n data-cy=\"c8y-alarm-list--timeline-repeat\"\n *c8yFor=\"let alarm of alarms; let i = index; pipe: mapAlarmLink; loadMore: loadMoreMode\"\n [routerLink]=\"navigationOptions.allowNavigationToAlarmsView ? alarm.link : null\"\n routerLinkActive=\"active\"\n [queryParamsHandling]=\"navigationOptions.queryParamsHandling\"\n (isActiveChange)=\"activeRouteChanged($event, liScrollAnchor, alarm)\"\n (click)=\"onAlarmOpen(alarm)\"\n [queryParams]=\"\n navigationOptions.includeClearedQueryParams\n ? { showCleared: alarm.status === 'CLEARED' }\n : {}\n \"\n >\n <span\n [attr.aria-label]=\"alarmLastOccurrenceLabel | translate\"\n [tooltip]=\"alarmLastOccurrenceLabel | translate\"\n placement=\"right\"\n container=\"body\"\n [delay]=\"500\"\n >\n {{ alarm.time | c8yDate: 'mediumDate' }}\n {{ alarm.time | c8yDate: 'mediumTime' }}\n </span>\n <c8y-li\n style=\"scroll-margin-top: 56px\"\n #liScrollAnchor\n >\n <c8y-li-icon class=\"a-s-start\">\n <div class=\"alarm-icons\">\n <c8y-alarms-icon [typeFilters]=\"typeFilters\" [alarm]=\"alarm\"></c8y-alarms-icon>\n </div>\n <button\n class=\"btn-clean text-center\"\n [attr.aria-label]=\"\n alarmBadgeTooltip\n | translate\n : { alarmFirstOccurrenceTime: alarm.firstOccurrenceTime | c8yDate: 'medium' }\n \"\n [tooltip]=\"\n alarmBadgeTooltip\n | translate\n : { alarmFirstOccurrenceTime: alarm.firstOccurrenceTime | c8yDate: 'medium' }\n \"\n placement=\"right\"\n container=\"body\"\n type=\"button\"\n *ngIf=\"alarm.firstOccurrenceTime\"\n (click)=\"$event.stopPropagation()\"\n [delay]=\"500\"\n >\n <span\n class=\"badge badge-info\"\n *ngIf=\"alarm.count > 1\"\n >\n {{ alarm.count }}\n </span>\n </button>\n </c8y-li-icon>\n <c8y-li-body class=\"a-s-stretch\">\n <div class=\"d-flex a-i-start fit-h\">\n <div class=\"min-width-0 flex-grow\">\n <p class=\"text-truncate-wrap p-b-4\">\n {{ alarm.text | translate }}\n </p>\n <div class=\"d-flex\">\n <p\n class=\"small text-muted text-truncate flex-grow\"\n [title]=\"alarm.source.name\"\n >\n <i [c8yIcon]=\"'exchange'\"></i>\n {{ alarm.source.name }}\n </p>\n <div class=\"d-flex\">\n <div\n [title]=\"item.title | translate\"\n *ngFor=\"let item of alarm | alarmListIndicator | async\"\n >\n <i\n [class]=\"item.class\"\n [c8yIcon]=\"item.icon\"\n ></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n </c8y-li-body>\n </c8y-li>\n </c8y-li-timeline>\n <c8y-loading *ngIf=\"isInitialLoading && isEmptyListLoading\"></c8y-loading>\n <div\n class=\"p-relative p-l-24\"\n *ngIf=\"isEmptyListLoading && !isInitialLoading\"\n >\n <c8y-ui-empty-state\n [icon]=\"'c8y-alert-idle'\"\n [title]=\"'No alarms to display.' | translate\"\n data-cy=\"alarm-list--empty-state\"\n *ngIf=\"hasPermissions; else alertsA\"\n >\n <p c8y-guide-docs>\n <small\n translate\n ngNonBindable\n >\n Find out more in the\n <a\n c8y-guide-href=\"/docs/device-management-application/monitoring-and-controlling-devices/#working-with-alarms\"\n >\n user documentation\n </a>\n .\n </small>\n </p>\n </c8y-ui-empty-state>\n </div>\n </c8y-list-group>\n</div>\n\n<ng-template #alertsA>\n <c8y-dynamic-component-alerts [alerts]=\"alertAggregator\"></c8y-dynamic-component-alerts>\n</ng-template>\n", dependencies: [{ 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: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i3.ForOfDirective, selector: "[c8yFor]", inputs: ["c8yForOf", "c8yForLoadMore", "c8yForPipe", "c8yForNotFound", "c8yForMaxIterations", "c8yForLoadingTemplate", "c8yForLoadNextLabel", "c8yForLoadingLabel", "c8yForRealtime", "c8yForRealtimeOptions", "c8yForComparator", "c8yForEnableVirtualScroll", "c8yForVirtualScrollElementSize", "c8yForVirtualScrollStrategy", "c8yForVirtualScrollContainerHeight"], outputs: ["c8yForCount", "c8yForChange", "c8yForLoadMoreComponent"] }, { kind: "component", type: i3.LoadingComponent, selector: "c8y-loading", inputs: ["layout", "progress", "message"] }, { kind: "directive", type: i3.GuideHrefDirective, selector: "[c8y-guide-href]", inputs: ["c8y-guide-href"] }, { kind: "component", type: i3.GuideDocsComponent, selector: "[c8y-guide-docs]" }, { kind: "component", type: i3.DynamicComponentAlertsComponent, selector: "c8y-dynamic-component-alerts", inputs: ["alerts"] }, { 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.ListItemIconComponent, selector: "c8y-list-item-icon, c8y-li-icon", inputs: ["icon", "status"] }, { kind: "component", type: i3.ListItemBodyComponent, selector: "c8y-list-item-body, c8y-li-body", inputs: ["body"] }, { kind: "component", type: i3.ListItemTimelineComponent, selector: "c8y-list-item-timeline, c8y-li-timeline" }, { kind: "directive", type: i5.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: i6.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { kind: "directive", type: i1.RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: i1.RouterLinkActive, selector: "[routerLinkActive]", inputs: ["routerLinkActiveOptions", "ariaCurrentWhenActive", "routerLinkActive"], outputs: ["isActiveChange"], exportAs: ["routerLinkActive"] }, { kind: "component", type: i7.AlarmsIconComponent, selector: "c8y-alarms-icon", inputs: ["alarm", "typeFilters"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i3.DatePipe, name: "c8yDate" }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }, { kind: "pipe", type: i8.AlarmListIndicatorPipe, name: "alarmListIndicator" }] }); }
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AlarmsListComponent, decorators: [{
type: Component,
args: [{ selector: 'c8y-alarms-list', template: "<div\n class=\"inner-scroll\"\n [ngClass]=\"{ 'split-view__list bg-level-1': splitView, 'bg-component': !splitView }\"\n data-cy=\"c8y-alarms-list\"\n #scrollWrapper\n>\n <div\n class=\"flex-wrap flex-no-shrink sticky-top m-b-16\"\n [ngClass]=\"{\n 'separator-bottom card-header p-b-0': splitView,\n 'd-flex fit-w separator-top-bottom widget-bar p-l-16 p-r-16':\n !splitView && navigationOptions.allowNavigationToAlarmsView\n }\"\n >\n <div\n class=\"h4 card-title\"\n *ngIf=\"splitView\"\n >\n {{ 'Alarms list' | translate }}\n </div>\n <div\n [ngClass]=\"{ 'fit-w d-flex a-i-center gap-16': !splitView, 'fit-h-20 m-l-auto': splitView }\"\n >\n <ng-content></ng-content>\n </div>\n <!-- Loading -->\n <div\n class=\"fit-w overflow-hidden\"\n [ngClass]=\"{ 'p-t-16': splitView }\"\n >\n <div\n class=\"loading-bar\"\n style=\"z-index: 101\"\n [ngClass]=\"{ active: isInitialLoading && !isEmptyListLoading }\"\n ></div>\n </div>\n\n <div\n class=\"alert alert-warning\"\n role=\"alert\"\n translate\n *ngIf=\"\n !isEmptyListLoading &&\n (activeChildParam$ | async) &&\n (activeAlarm$ | async)?.id !== (activeChildParam$ | async)\n \"\n >\n The selected alarm is not currently in the list, change your filter.\n </div>\n </div>\n <c8y-list-group\n class=\"p-r-16 interactive\"\n [ngStyle]=\"{ opacity: isInitialLoading && !isEmptyListLoading ? 0.2 : 1 }\"\n style=\"transition: opacity 0.15s linear\"\n data-cy=\"c8y-alarm-list--group\"\n >\n <c8y-li-timeline\n class=\"pointer\"\n role=\"button\"\n data-cy=\"c8y-alarm-list--timeline-repeat\"\n *c8yFor=\"let alarm of alarms; let i = index; pipe: mapAlarmLink; loadMore: loadMoreMode\"\n [routerLink]=\"navigationOptions.allowNavigationToAlarmsView ? alarm.link : null\"\n routerLinkActive=\"active\"\n [queryParamsHandling]=\"navigationOptions.queryParamsHandling\"\n (isActiveChange)=\"activeRouteChanged($event, liScrollAnchor, alarm)\"\n (click)=\"onAlarmOpen(alarm)\"\n [queryParams]=\"\n navigationOptions.includeClearedQueryParams\n ? { showCleared: alarm.status === 'CLEARED' }\n : {}\n \"\n >\n <span\n [attr.aria-label]=\"alarmLastOccurrenceLabel | translate\"\n [tooltip]=\"alarmLastOccurrenceLabel | translate\"\n placement=\"right\"\n container=\"body\"\n [delay]=\"500\"\n >\n {{ alarm.time | c8yDate: 'mediumDate' }}\n {{ alarm.time | c8yDate: 'mediumTime' }}\n </span>\n <c8y-li\n style=\"scroll-margin-top: 56px\"\n #liScrollAnchor\n >\n <c8y-li-icon class=\"a-s-start\">\n <div class=\"alarm-icons\">\n <c8y-alarms-icon [typeFilters]=\"typeFilters\" [alarm]=\"alarm\"></c8y-alarms-icon>\n </div>\n <button\n class=\"btn-clean text-center\"\n [attr.aria-label]=\"\n alarmBadgeTooltip\n | translate\n : { alarmFirstOccurrenceTime: alarm.firstOccurrenceTime | c8yDate: 'medium' }\n \"\n [tooltip]=\"\n alarmBadgeTooltip\n | translate\n : { alarmFirstOccurrenceTime: alarm.firstOccurrenceTime | c8yDate: 'medium' }\n \"\n placement=\"right\"\n container=\"body\"\n type=\"button\"\n *ngIf=\"alarm.firstOccurrenceTime\"\n (click)=\"$event.stopPropagation()\"\n [delay]=\"500\"\n >\n <span\n class=\"badge badge-info\"\n *ngIf=\"alarm.count > 1\"\n >\n {{ alarm.count }}\n </span>\n </button>\n </c8y-li-icon>\n <c8y-li-body class=\"a-s-stretch\">\n <div class=\"d-flex a-i-start fit-h\">\n <div class=\"min-width-0 flex-grow\">\n <p class=\"text-truncate-wrap p-b-4\">\n {{ alarm.text | translate }}\n </p>\n <div class=\"d-flex\">\n <p\n class=\"small text-muted text-truncate flex-grow\"\n [title]=\"alarm.source.name\"\n >\n <i [c8yIcon]=\"'exchange'\"></i>\n {{ alarm.source.name }}\n </p>\n <div class=\"d-flex\">\n <div\n [title]=\"item.title | translate\"\n *ngFor=\"let item of alarm | alarmListIndicator | async\"\n >\n <i\n [class]=\"item.class\"\n [c8yIcon]=\"item.icon\"\n ></i>\n </div>\n </div>\n </div>\n </div>\n </div>\n </c8y-li-body>\n </c8y-li>\n </c8y-li-timeline>\n <c8y-loading *ngIf=\"isInitialLoading && isEmptyListLoading\"></c8y-loading>\n <div\n class=\"p-relative p-l-24\"\n *ngIf=\"isEmptyListLoading && !isInitialLoading\"\n >\n <c8y-ui-empty-state\n [icon]=\"'c8y-alert-idle'\"\n [title]=\"'No alarms to display.' | translate\"\n data-cy=\"alarm-list--empty-state\"\n *ngIf=\"hasPermissions; else alertsA\"\n >\n <p c8y-guide-docs>\n <small\n translate\n ngNonBindable\n >\n Find out more in the\n <a\n c8y-guide-href=\"/docs/device-management-application/monitoring-and-controlling-devices/#working-with-alarms\"\n >\n user documentation\n </a>\n .\n </small>\n </p>\n </c8y-ui-empty-state>\n </div>\n </c8y-list-group>\n</div>\n\n<ng-template #alertsA>\n <c8y-dynamic-component-alerts [alerts]=\"alertAggregator\"></c8y-dynamic-component-alerts>\n</ng-template>\n" }]
}], ctorParameters: () => [{ type: i1.ActivatedRoute }, { type: i2.AlarmsViewService }, { type: i3.ContextRouteService }, { type: i1.Router }], propDecorators: { alarms: [{
type: Input
}], hasPermissions: [{
type: Input
}], typeFilters: [{
type: Input
}], loadMoreMode: [{
type: Input
}], navigationOptions: [{
type: Input
}], isInitialLoading: [{
type: Input
}], splitView: [{
type: Input
}], onSelectedAlarm: [{
type: Output
}], onScrollingStateChange: [{
type: Output
}], innerScrollWrapper: [{
type: ViewChild,
args: ['scrollWrapper']
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"alarms-list.component.js","sourceRoot":"","sources":["../../../alarms/alarms-list.component.ts","../../../alarms/alarms-list.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  Output,\n  SimpleChanges,\n  ViewChild\n} from '@angular/core';\nimport { ActivatedRoute, NavigationEnd, Router } from '@angular/router';\nimport { IAlarm, IResultList } from '@c8y/client';\nimport {\n  ContextRouteService,\n  DynamicComponentAlert,\n  DynamicComponentAlertAggregator,\n  ListItemComponent,\n  LoadMoreMode,\n  gettext\n} from '@c8y/ngx-components';\nimport { BehaviorSubject, Observable, Subject, fromEvent, pipe } from 'rxjs';\nimport {\n  debounceTime,\n  distinctUntilChanged,\n  filter,\n  map,\n  shareReplay,\n  switchMap,\n  takeUntil\n} from 'rxjs/operators';\nimport { AlarmsViewService } from './alarms-view.service';\nimport { AlarmNavigationOptions } from './alarms.model';\nimport { AlarmDetails } from '@c8y/ngx-components/alarm-event-selector';\n\n@Component({\n  selector: 'c8y-alarms-list',\n  templateUrl: './alarms-list.component.html'\n})\nexport class AlarmsListComponent implements OnChanges, AfterViewInit, OnDestroy {\n  readonly alarmBadgeTooltip = gettext(\n    'Number of occurrences`number of occurrences of alarm`. First occurrence {{ alarmFirstOccurrenceTime }} (device time).'\n  );\n  readonly alarmLastOccurrenceLabel = gettext('Last occurrence of this alarm (device time).');\n\n  /**\n   * Input property for receiving a list of alarms.\n   */\n  @Input()\n  alarms: IResultList<IAlarm & { link?: string }>;\n\n  @Input() hasPermissions = true;\n\n  /**\n   * Input property for the currently applied type filters.\n   */\n  @Input()\n  typeFilters: AlarmDetails[] = [];\n\n  /**\n   * Input property for receiving load more mode.\n   */\n  @Input()\n  loadMoreMode: LoadMoreMode = 'hidden';\n\n  /**\n   * Defines options, how the alarm list should be navigated if a user\n   * clicks on an alarm.\n   */\n  @Input()\n  navigationOptions: AlarmNavigationOptions = {\n    allowNavigationToAlarmsView: true,\n    alwaysNavigateToAllAlarms: false,\n    includeClearedQueryParams: false,\n    queryParamsHandling: 'merge'\n  };\n\n  /**\n   * Controls the visibility of the loading bar\n   * When set to `false`, the alarm list is displayed. When set to `true`, the opacity of alarms list is changed and a loading bar is shown.\n   */\n  @Input()\n  isInitialLoading = false;\n\n  /**\n   * Controls the visibility and functionality of some components\n   * When set to `true`, means the list is displayed in a split view layout:\n   * the list on the first column and the selected record detail on the second column (the cockpit\n   * alarms view for example)\n   * When set to false, the list is displayed as a standalone component, opening the detail will\n   * redirect to the alarms\n   */\n  @Input()\n  splitView = false;\n\n  /**\n   * Emits an instance of a selected alarm when one is chosen from the list.\n   */\n  @Output()\n  onSelectedAlarm = new EventEmitter<IAlarm>();\n\n  /**\n   * Emits a boolean value indicating the scrolling state: true when the user starts scrolling, and false when the user reaches the top of the list.\n   */\n  @Output()\n  onScrollingStateChange = new EventEmitter<boolean>();\n\n  /**\n   * Current alarm or last alarm marked as active by the routerLinkActive directive.\n   */\n  activeAlarm$ = new BehaviorSubject<IAlarm>(null);\n  activeChildParam$ = new Observable<string>();\n\n  @ViewChild('scrollWrapper')\n  innerScrollWrapper: ElementRef;\n\n  isScrolling = false;\n  /**\n   * Determines whether the c8y-loading component should be displayed.\n   * The loading component is shown when no alarms are displayed in the view or when the request is initial,\n   * as we don't want to see empty space on alarm list during loading.\n   */\n  isEmptyListLoading = true;\n\n  alertAggregator = new DynamicComponentAlertAggregator();\n\n  mapAlarmLink = pipe(\n    map((alarms: IAlarm[]) =>\n      alarms.map((alarm: IAlarm) => {\n        alarm.link = this.getRouterLink(alarm);\n        return alarm;\n      })\n    )\n  );\n\n  private destroy$: Subject<void> = new Subject<void>();\n  private HIDE_INTERVAL_COUNTDOWN_SCROLL = 50;\n\n  constructor(\n    private activatedRoute: ActivatedRoute,\n    private alarmsViewService: AlarmsViewService,\n    private contextRouteService: ContextRouteService,\n    private router: Router\n  ) {\n    this.verifyIfFiltersMatchingAlarm();\n  }\n\n  /**\n   * Handles the change of the active route.\n   *\n   * @param isActive - A boolean indicating whether the route is active or not.\n   * @param scrollAnchor - The ListItemComponent used as a scroll anchor.\n   * @param alarm - The IAlarm object representing the active alarm.\n   */\n  activeRouteChanged(isActive: boolean, scrollAnchor: ListItemComponent, alarm: IAlarm): void {\n    if (isActive) {\n      scrollAnchor.element.nativeElement.scrollIntoView({\n        behavior: 'smooth',\n        block: 'nearest'\n      });\n      this.activeAlarm$.next(alarm);\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (this.alarms && changes.alarms) {\n      this.activeAlarm$.next(null);\n      this.isEmptyListLoading = !this.alarms?.data?.length;\n    }\n\n    if (changes.hasPermissions?.currentValue === false) {\n      this.alertAggregator.addAlerts(\n        new DynamicComponentAlert({\n          type: 'system',\n          text: gettext(\"You don't have permission to view alarms.\")\n        })\n      );\n    }\n  }\n\n  ngAfterViewInit(): void {\n    if (this.alarmsViewService.isIntervalRefresh()) {\n      const scrollElement = this.innerScrollWrapper.nativeElement;\n      fromEvent(scrollElement, 'scroll')\n        .pipe(takeUntil(this.destroy$), debounceTime(300))\n        .subscribe((event: Event) => {\n          const target = event.target as HTMLElement;\n          this.isScrolling = this.shouldCountdownIntervalBeHidden(target);\n          this.onScrollingStateChange.emit(this.isScrolling);\n        });\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n  }\n\n  onAlarmOpen(alarm: IAlarm): void {\n    this.onSelectedAlarm.emit(alarm);\n  }\n\n  getRouterLink(alarm: IAlarm): string {\n    if (this.navigationOptions.alwaysNavigateToAllAlarms) {\n      return this.alarmsViewService.getRouterLink(null, alarm);\n    }\n    const contextData = this.contextRouteService.getContextData(this.activatedRoute);\n    return this.alarmsViewService.getRouterLink(contextData, alarm);\n  }\n\n  private shouldCountdownIntervalBeHidden(target: HTMLElement): boolean {\n    const scrollTopPixels = target.scrollTop;\n    return scrollTopPixels > this.HIDE_INTERVAL_COUNTDOWN_SCROLL;\n  }\n\n  private verifyIfFiltersMatchingAlarm() {\n    this.activeChildParam$ = this.router.events.pipe(\n      filter(e => e instanceof NavigationEnd && this.activatedRoute.children.length > 0),\n      switchMap(() => this.activatedRoute.children[0].params),\n      map(params => params.id),\n      distinctUntilChanged(),\n      shareReplay(),\n      takeUntil(this.destroy$)\n    );\n\n    // done to get the first navigation\n    this.activeChildParam$.subscribe();\n  }\n}\n","<div\n  class=\"inner-scroll\"\n  [ngClass]=\"{ 'split-view__list bg-level-1': splitView, 'bg-component': !splitView }\"\n  data-cy=\"c8y-alarms-list\"\n  #scrollWrapper\n>\n  <div\n    class=\"flex-wrap flex-no-shrink sticky-top m-b-16\"\n    [ngClass]=\"{\n      'separator-bottom card-header p-b-0': splitView,\n      'd-flex fit-w separator-top-bottom widget-bar p-l-16 p-r-16':\n        !splitView && navigationOptions.allowNavigationToAlarmsView\n    }\"\n  >\n    <div\n      class=\"h4 card-title\"\n      *ngIf=\"splitView\"\n    >\n      {{ 'Alarms list' | translate }}\n    </div>\n    <div\n      [ngClass]=\"{ 'fit-w d-flex a-i-center gap-16': !splitView, 'fit-h-20 m-l-auto': splitView }\"\n    >\n      <ng-content></ng-content>\n    </div>\n    <!--  Loading -->\n    <div\n      class=\"fit-w overflow-hidden\"\n      [ngClass]=\"{ 'p-t-16': splitView }\"\n    >\n      <div\n        class=\"loading-bar\"\n        style=\"z-index: 101\"\n        [ngClass]=\"{ active: isInitialLoading && !isEmptyListLoading }\"\n      ></div>\n    </div>\n\n    <div\n      class=\"alert alert-warning\"\n      role=\"alert\"\n      translate\n      *ngIf=\"\n        !isEmptyListLoading &&\n        (activeChildParam$ | async) &&\n        (activeAlarm$ | async)?.id !== (activeChildParam$ | async)\n      \"\n    >\n      The selected alarm is not currently in the list, change your filter.\n    </div>\n  </div>\n  <c8y-list-group\n    class=\"p-r-16 interactive\"\n    [ngStyle]=\"{ opacity: isInitialLoading && !isEmptyListLoading ? 0.2 : 1 }\"\n    style=\"transition: opacity 0.15s linear\"\n    data-cy=\"c8y-alarm-list--group\"\n  >\n    <c8y-li-timeline\n      class=\"pointer\"\n      role=\"button\"\n      data-cy=\"c8y-alarm-list--timeline-repeat\"\n      *c8yFor=\"let alarm of alarms; let i = index; pipe: mapAlarmLink; loadMore: loadMoreMode\"\n      [routerLink]=\"navigationOptions.allowNavigationToAlarmsView ? alarm.link : null\"\n      routerLinkActive=\"active\"\n      [queryParamsHandling]=\"navigationOptions.queryParamsHandling\"\n      (isActiveChange)=\"activeRouteChanged($event, liScrollAnchor, alarm)\"\n      (click)=\"onAlarmOpen(alarm)\"\n      [queryParams]=\"\n        navigationOptions.includeClearedQueryParams\n          ? { showCleared: alarm.status === 'CLEARED' }\n          : {}\n      \"\n    >\n      <span\n        [attr.aria-label]=\"alarmLastOccurrenceLabel | translate\"\n        [tooltip]=\"alarmLastOccurrenceLabel | translate\"\n        placement=\"right\"\n        container=\"body\"\n        [delay]=\"500\"\n      >\n        {{ alarm.time | c8yDate: 'mediumDate' }}\n        {{ alarm.time | c8yDate: 'mediumTime' }}\n      </span>\n      <c8y-li\n        style=\"scroll-margin-top: 56px\"\n        #liScrollAnchor\n      >\n        <c8y-li-icon class=\"a-s-start\">\n          <div class=\"alarm-icons\">\n            <c8y-alarms-icon [typeFilters]=\"typeFilters\" [alarm]=\"alarm\"></c8y-alarms-icon>\n          </div>\n          <button\n            class=\"btn-clean text-center\"\n            [attr.aria-label]=\"\n              alarmBadgeTooltip\n                | translate\n                  : { alarmFirstOccurrenceTime: alarm.firstOccurrenceTime | c8yDate: 'medium' }\n            \"\n            [tooltip]=\"\n              alarmBadgeTooltip\n                | translate\n                  : { alarmFirstOccurrenceTime: alarm.firstOccurrenceTime | c8yDate: 'medium' }\n            \"\n            placement=\"right\"\n            container=\"body\"\n            type=\"button\"\n            *ngIf=\"alarm.firstOccurrenceTime\"\n            (click)=\"$event.stopPropagation()\"\n            [delay]=\"500\"\n          >\n            <span\n              class=\"badge badge-info\"\n              *ngIf=\"alarm.count > 1\"\n            >\n              {{ alarm.count }}\n            </span>\n          </button>\n        </c8y-li-icon>\n        <c8y-li-body class=\"a-s-stretch\">\n          <div class=\"d-flex a-i-start fit-h\">\n            <div class=\"min-width-0 flex-grow\">\n              <p class=\"text-truncate-wrap p-b-4\">\n                {{ alarm.text | translate }}\n              </p>\n              <div class=\"d-flex\">\n                <p\n                  class=\"small text-muted text-truncate flex-grow\"\n                  [title]=\"alarm.source.name\"\n                >\n                  <i [c8yIcon]=\"'exchange'\"></i>\n                  {{ alarm.source.name }}\n                </p>\n                <div class=\"d-flex\">\n                  <div\n                    [title]=\"item.title | translate\"\n                    *ngFor=\"let item of alarm | alarmListIndicator | async\"\n                  >\n                    <i\n                      [class]=\"item.class\"\n                      [c8yIcon]=\"item.icon\"\n                    ></i>\n                  </div>\n                </div>\n              </div>\n            </div>\n          </div>\n        </c8y-li-body>\n      </c8y-li>\n    </c8y-li-timeline>\n    <c8y-loading *ngIf=\"isInitialLoading && isEmptyListLoading\"></c8y-loading>\n    <div\n      class=\"p-relative p-l-24\"\n      *ngIf=\"isEmptyListLoading && !isInitialLoading\"\n    >\n      <c8y-ui-empty-state\n        [icon]=\"'c8y-alert-idle'\"\n        [title]=\"'No alarms to display.' | translate\"\n        data-cy=\"alarm-list--empty-state\"\n        *ngIf=\"hasPermissions; else alertsA\"\n      >\n        <p c8y-guide-docs>\n          <small\n            translate\n            ngNonBindable\n          >\n            Find out more in the\n            <a\n              c8y-guide-href=\"/docs/device-management-application/monitoring-and-controlling-devices/#working-with-alarms\"\n            >\n              user documentation\n            </a>\n            .\n          </small>\n        </p>\n      </c8y-ui-empty-state>\n    </div>\n  </c8y-list-group>\n</div>\n\n<ng-template #alertsA>\n  <c8y-dynamic-component-alerts [alerts]=\"alertAggregator\"></c8y-dynamic-component-alerts>\n</ng-template>\n"]}