diff --git a/application/src/main/data/json/system/widget_bundles/cards.json b/application/src/main/data/json/system/widget_bundles/cards.json
index 84f8cd9785..81cb2fdbb1 100644
--- a/application/src/main/data/json/system/widget_bundles/cards.json
+++ b/application/src/main/data/json/system/widget_bundles/cards.json
@@ -23,6 +23,7 @@
"cards.html_card",
"cards.html_value_card",
"cards.markdown_card",
- "cards.simple_card"
+ "cards.simple_card",
+ "unread_notifications"
]
}
\ No newline at end of file
diff --git a/application/src/main/data/json/system/widget_types/unread_notifications.json b/application/src/main/data/json/system/widget_types/unread_notifications.json
new file mode 100644
index 0000000000..7e3e6b0477
--- /dev/null
+++ b/application/src/main/data/json/system/widget_types/unread_notifications.json
@@ -0,0 +1,23 @@
+{
+ "fqn": "unread_notifications",
+ "name": "Unread notifications",
+ "deprecated": false,
+ "image": "tb-image:dW5yZWFkX25vdGlmaWNhdGlvbl9zeXN0ZW1fd2lkZ2V0X2ltYWdlLnBuZw==:IlVucmVhZCBub3RpZmljYXRpb24iIHN5c3RlbSB3aWRnZXQgaW1hZ2U=;",
+ "description": null,
+ "descriptor": {
+ "type": "static",
+ "sizeX": 5.5,
+ "sizeY": 5,
+ "resources": [],
+ "templateHtml": "\n",
+ "templateCss": "",
+ "controllerScript": "self.onInit = function() {\n self.ctx.$scope.unreadNotificationWidget.onInit();\n}\n\nself.typeParameters = function() {\n return {\n previewWidth: '400px',\n previewHeight: '300px',\n embedTitlePanel: true\n };\n}\n\nself.onDestroy = function() {\n}\n",
+ "settingsSchema": "",
+ "dataKeySettingsSchema": "",
+ "settingsDirective": "tb-unread-notification-widget-settings",
+ "hasBasicMode": true,
+ "basicModeDirective": "tb-unread-notification-basic-config",
+ "defaultConfig": "{\"datasources\":[{\"type\":\"static\",\"name\":\"function\",\"dataKeys\":[{\"name\":\"f(x)\",\"type\":\"function\",\"label\":\"Random\",\"color\":\"#2196f3\",\"settings\":{},\"_hash\":0.15479322438769105,\"funcBody\":\"var value = prevValue + Math.random() * 100 - 50;\\nvar multiplier = Math.pow(10, 2 || 0);\\nvar value = Math.round(value * multiplier) / multiplier;\\nif (value < -1000) {\\n\\tvalue = -1000;\\n} else if (value > 1000) {\\n\\tvalue = 1000;\\n}\\nreturn value;\"}]}],\"timewindow\":{\"realtime\":{\"timewindowMs\":60000}},\"showTitle\":true,\"backgroundColor\":\"rgb(255, 255, 255)\",\"color\":\"rgba(0, 0, 0, 0.87)\",\"padding\":\"0\",\"settings\":{\"cardHtml\":\"
HTML code here
\",\"cardCss\":\".card {\\n font-weight: bold;\\n font-size: 32px;\\n color: #999;\\n width: 100%;\\n height: 100%;\\n display: flex;\\n align-items: center;\\n justify-content: center;\\n}\",\"maxNotificationDisplay\":6,\"showCounter\":true,\"counterValueFont\":{\"family\":\"Roboto\",\"size\":14,\"sizeUnit\":\"px\",\"style\":\"normal\",\"weight\":\"600\",\"lineHeight\":\"\"},\"counterValueColor\":\"#fff\",\"background\":{\"type\":\"color\",\"color\":\"#fff\",\"overlay\":{\"enabled\":false,\"color\":\"rgba(255,255,255,0.72)\",\"blur\":3}},\"enableViewAll\":true,\"enableFilter\":true,\"enableMarkAsRead\":true},\"title\":\"Unread notification\",\"dropShadow\":true,\"configMode\":\"basic\",\"titleFont\":{\"size\":16,\"sizeUnit\":\"px\",\"family\":\"Roboto\",\"weight\":\"500\",\"style\":\"normal\",\"lineHeight\":\"24px\"},\"titleColor\":\"#000000\",\"showTitleIcon\":true,\"iconSize\":\"22px\",\"titleIcon\":\"notifications\",\"iconColor\":\"#000000\",\"actions\":{},\"enableFullscreen\":false,\"borderRadius\":\"4px\",\"titleTooltip\":\"\",\"widgetStyle\":{},\"widgetCss\":\"\",\"titleStyle\":{\"fontSize\":\"16px\",\"fontWeight\":400},\"pageSize\":1024,\"noDataDisplayMessage\":\"\"}"
+ },
+ "tags": null
+}
diff --git a/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.html b/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.html
index 851eff2375..d96739a5cf 100644
--- a/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.html
+++ b/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.html
@@ -44,7 +44,11 @@
-
+
+
+
notification.no-notifications-yet
diff --git a/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.scss b/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.scss
new file mode 100644
index 0000000000..577c3ccb72
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.scss
@@ -0,0 +1,20 @@
+/**
+ * Copyright © 2016-2024 The Thingsboard Authors
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+@import '../../../../../scss/constants';
+
+.tb-no-notification-svg-color {
+ color: $tb-primary-color;
+}
diff --git a/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.ts b/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.ts
index 6020d6586a..32fcf2dc4a 100644
--- a/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.ts
+++ b/ui-ngx/src/app/modules/home/components/notification/show-notification-popover.component.ts
@@ -29,7 +29,7 @@ import { NotificationSubscriber } from '@shared/models/telemetry/telemetry.model
@Component({
selector: 'tb-show-notification-popover',
templateUrl: './show-notification-popover.component.html',
- styleUrls: []
+ styleUrls: ['show-notification-popover.component.scss']
})
export class ShowNotificationPopoverComponent extends PageComponent implements OnDestroy, OnInit {
diff --git a/ui-ngx/src/app/modules/home/components/widget/config/basic/basic-widget-config.module.ts b/ui-ngx/src/app/modules/home/components/widget/config/basic/basic-widget-config.module.ts
index ac8cb25d60..06af5402ca 100644
--- a/ui-ngx/src/app/modules/home/components/widget/config/basic/basic-widget-config.module.ts
+++ b/ui-ngx/src/app/modules/home/components/widget/config/basic/basic-widget-config.module.ts
@@ -139,6 +139,9 @@ import {
import {
LabelValueCardBasicConfigComponent
} from '@home/components/widget/config/basic/cards/label-value-card-basic-config.component';
+import {
+ UnreadNotificationBasicConfigComponent
+} from '@home/components/widget/config/basic/cards/unread-notification-basic-config.component';
@NgModule({
declarations: [
@@ -185,7 +188,8 @@ import {
DigitalSimpleGaugeBasicConfigComponent,
MobileAppQrCodeBasicConfigComponent,
LabelCardBasicConfigComponent,
- LabelValueCardBasicConfigComponent
+ LabelValueCardBasicConfigComponent,
+ UnreadNotificationBasicConfigComponent
],
imports: [
CommonModule,
@@ -234,7 +238,8 @@ import {
DigitalSimpleGaugeBasicConfigComponent,
MobileAppQrCodeBasicConfigComponent,
LabelCardBasicConfigComponent,
- LabelValueCardBasicConfigComponent
+ LabelValueCardBasicConfigComponent,
+ UnreadNotificationBasicConfigComponent
]
})
export class BasicWidgetConfigModule {
@@ -277,5 +282,6 @@ export const basicWidgetConfigComponentsMap: {[key: string]: Type
+
+
+
+
+
+
+
+
+
+
diff --git a/ui-ngx/src/app/modules/home/components/widget/config/basic/cards/unread-notification-basic-config.component.ts b/ui-ngx/src/app/modules/home/components/widget/config/basic/cards/unread-notification-basic-config.component.ts
new file mode 100644
index 0000000000..78fc5ce791
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/config/basic/cards/unread-notification-basic-config.component.ts
@@ -0,0 +1,177 @@
+///
+/// Copyright © 2016-2024 The Thingsboard Authors
+///
+/// Licensed under the Apache License, Version 2.0 (the "License");
+/// you may not use this file except in compliance with the License.
+/// You may obtain a copy of the License at
+///
+/// http://www.apache.org/licenses/LICENSE-2.0
+///
+/// Unless required by applicable law or agreed to in writing, software
+/// distributed under the License is distributed on an "AS IS" BASIS,
+/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+/// See the License for the specific language governing permissions and
+/// limitations under the License.
+///
+
+import { Component } from '@angular/core';
+import { UntypedFormBuilder, UntypedFormGroup, Validators } from '@angular/forms';
+import { Store } from '@ngrx/store';
+import { AppState } from '@core/core.state';
+import { BasicWidgetConfigComponent } from '@home/components/widget/config/widget-config.component.models';
+import { WidgetConfigComponentData } from '@home/models/widget-component.models';
+import { WidgetConfig, } from '@shared/models/widget.models';
+import { WidgetConfigComponent } from '@home/components/widget/widget-config.component';
+import { isUndefined } from '@core/utils';
+import { cssSizeToStrSize, resolveCssSize } from '@shared/models/widget-settings.models';
+import {
+ unreadNotificationDefaultSettings,
+ UnreadNotificationWidgetSettings
+} from '@home/components/widget/lib/cards/unread-notification-widget.models';
+
+@Component({
+ selector: 'tb-unread-notification-basic-config',
+ templateUrl: './unread-notification-basic-config.component.html',
+ styleUrls: ['../basic-config.scss']
+})
+export class UnreadNotificationBasicConfigComponent extends BasicWidgetConfigComponent {
+
+ unreadNotificationWidgetConfigForm: UntypedFormGroup;
+
+ constructor(protected store: Store,
+ protected widgetConfigComponent: WidgetConfigComponent,
+ private fb: UntypedFormBuilder) {
+ super(store, widgetConfigComponent);
+ }
+
+ protected configForm(): UntypedFormGroup {
+ return this.unreadNotificationWidgetConfigForm;
+ }
+
+ protected onConfigSet(configData: WidgetConfigComponentData) {
+ const iconSize = resolveCssSize(configData.config.iconSize);
+ const settings: UnreadNotificationWidgetSettings = {...unreadNotificationDefaultSettings, ...(configData.config.settings || {})};
+ this.unreadNotificationWidgetConfigForm = this.fb.group({
+
+ showTitle: [configData.config.showTitle, []],
+ title: [configData.config.title, []],
+ titleFont: [configData.config.titleFont, []],
+ titleColor: [configData.config.titleColor, []],
+
+ showIcon: [configData.config.showTitleIcon, []],
+ iconSize: [iconSize[0], [Validators.min(0)]],
+ iconSizeUnit: [iconSize[1], []],
+ icon: [configData.config.titleIcon, []],
+ iconColor: [configData.config.iconColor, []],
+
+ maxNotificationDisplay: [settings.maxNotificationDisplay, [Validators.required, Validators.min(1)]],
+ showCounter: [settings.showCounter, []],
+ counterValueFont: [settings.counterValueFont, []],
+ counterValueColor: [settings.counterValueColor, []],
+ counterColor: [settings.counterColor, []],
+
+ background: [settings.background, []],
+ padding: [settings.padding, []],
+
+ cardButtons: [this.getCardButtons(configData.config), []],
+ borderRadius: [configData.config.borderRadius, []],
+ actions: [configData.config.actions || {}, []]
+ });
+ }
+ protected validatorTriggers(): string[] {
+ return ['showCounter', 'showTitle', 'showIcon'];
+ }
+
+ protected updateValidators(emitEvent: boolean, trigger?: string) {
+ const showCounter: boolean = this.unreadNotificationWidgetConfigForm.get('showCounter').value;
+ const showTitle: boolean = this.unreadNotificationWidgetConfigForm.get('showTitle').value;
+ const showIcon: boolean = this.unreadNotificationWidgetConfigForm.get('showIcon').value;
+
+ if (showTitle) {
+ this.unreadNotificationWidgetConfigForm.get('title').enable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('titleFont').enable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('titleColor').enable({emitEvent});
+ } else {
+ this.unreadNotificationWidgetConfigForm.get('title').disable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('titleFont').disable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('titleColor').disable({emitEvent});
+ }
+
+ if (showIcon) {
+ this.unreadNotificationWidgetConfigForm.get('iconSize').enable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('iconSizeUnit').enable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('icon').enable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('iconColor').enable({emitEvent});
+ } else {
+ this.unreadNotificationWidgetConfigForm.get('iconSize').disable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('iconSizeUnit').disable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('icon').disable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('iconColor').disable({emitEvent});
+ }
+
+ if (showCounter) {
+ this.unreadNotificationWidgetConfigForm.get('counterValueFont').enable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('counterValueColor').enable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('counterColor').enable({emitEvent});
+ } else {
+ this.unreadNotificationWidgetConfigForm.get('counterValueFont').disable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('counterValueColor').disable({emitEvent});
+ this.unreadNotificationWidgetConfigForm.get('counterColor').disable({emitEvent});
+ }
+ }
+
+ protected prepareOutputConfig(config: any): WidgetConfigComponentData {
+
+ this.widgetConfig.config.showTitle = config.showTitle;
+ this.widgetConfig.config.title = config.title;
+ this.widgetConfig.config.titleFont = config.titleFont;
+ this.widgetConfig.config.titleColor = config.titleColor;
+
+ this.widgetConfig.config.showTitleIcon = config.showIcon;
+ this.widgetConfig.config.iconSize = cssSizeToStrSize(config.iconSize, config.iconSizeUnit);
+ this.widgetConfig.config.titleIcon = config.icon;
+ this.widgetConfig.config.iconColor = config.iconColor;
+
+ this.widgetConfig.config.settings = this.widgetConfig.config.settings || {};
+
+ this.widgetConfig.config.settings.maxNotificationDisplay = config.maxNotificationDisplay;
+ this.widgetConfig.config.settings.showCounter = config.showCounter;
+ this.widgetConfig.config.settings.counterValueFont = config.counterValueFont;
+ this.widgetConfig.config.settings.counterValueColor = config.counterValueColor;
+ this.widgetConfig.config.settings.counterColor = config.counterColor;
+
+ this.widgetConfig.config.settings.background = config.background;
+ this.widgetConfig.config.settings.padding = config.padding;
+
+ this.widgetConfig.config.actions = config.actions;
+ this.setCardButtons(config.cardButtons, this.widgetConfig.config);
+ this.widgetConfig.config.borderRadius = config.borderRadius;
+ return this.widgetConfig;
+ }
+
+ private getCardButtons(config: WidgetConfig): string[] {
+ const buttons: string[] = [];
+ if (isUndefined(config.settings?.enableViewAll) || config.settings?.enableViewAll) {
+ buttons.push('viewAll');
+ }
+ if (isUndefined(config.settings?.enableFilter) || config.settings?.enableFilter) {
+ buttons.push('filter');
+ }
+ if (isUndefined(config.settings?.enableMarkAsRead) || config.settings?.enableMarkAsRead) {
+ buttons.push('markAsRead');
+ }
+ if (isUndefined(config.enableFullscreen) || config.enableFullscreen) {
+ buttons.push('fullscreen');
+ }
+ return buttons;
+ }
+
+ private setCardButtons(buttons: string[], config: WidgetConfig) {
+ config.settings.enableViewAll = buttons.includes('viewAll');
+ config.settings.enableFilter = buttons.includes('filter');
+ config.settings.enableMarkAsRead = buttons.includes('markAsRead');
+
+ config.enableFullscreen = buttons.includes('fullscreen');
+ }
+
+}
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.html b/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.html
new file mode 100644
index 0000000000..00ec196552
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.html
@@ -0,0 +1,77 @@
+
+
+
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.scss b/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.scss
new file mode 100644
index 0000000000..77a1e610d3
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.scss
@@ -0,0 +1,25 @@
+/**
+ * Copyright © 2016-2024 The Thingsboard Authors
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+:host {
+ display: flex;
+ width: 100%;
+ max-width: 100%;
+
+ .mdc-button {
+ max-width: 100%;
+ }
+}
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.ts b/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.ts
new file mode 100644
index 0000000000..ba52310c4b
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/cards/notification-type-filter-panel.component.ts
@@ -0,0 +1,140 @@
+///
+/// Copyright © 2016-2024 The Thingsboard Authors
+///
+/// Licensed under the Apache License, Version 2.0 (the "License");
+/// you may not use this file except in compliance with the License.
+/// You may obtain a copy of the License at
+///
+/// http://www.apache.org/licenses/LICENSE-2.0
+///
+/// Unless required by applicable law or agreed to in writing, software
+/// distributed under the License is distributed on an "AS IS" BASIS,
+/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+/// See the License for the specific language governing permissions and
+/// limitations under the License.
+///
+
+import { Component, ElementRef, Inject, InjectionToken, OnInit, ViewChild } from '@angular/core';
+import { NotificationTemplateTypeTranslateMap, NotificationType } from '@shared/models/notification.models';
+import { MatChipInputEvent } from '@angular/material/chips';
+import { COMMA, ENTER, SEMICOLON } from '@angular/cdk/keycodes';
+import { MatAutocompleteSelectedEvent } from '@angular/material/autocomplete';
+import { Observable } from 'rxjs';
+import { FormControl } from '@angular/forms';
+import { debounceTime, map } from 'rxjs/operators';
+import { OverlayRef } from '@angular/cdk/overlay';
+
+export const NOTIFICATION_TYPE_FILTER_PANEL_DATA = new InjectionToken('NotificationTypeFilterPanelData');
+
+export interface NotificationTypeFilterPanelData {
+ notificationTypes: Array;
+ notificationTypesUpdated: (notificationTypes: Array) => void;
+}
+
+@Component({
+ selector: 'tb-notification-type-filter-panel',
+ templateUrl: './notification-type-filter-panel.component.html',
+ styleUrls: ['notification-type-filter-panel.component.scss']
+})
+export class NotificationTypeFilterPanelComponent implements OnInit{
+
+ @ViewChild('searchInput') searchInputField: ElementRef;
+
+ searchText = '';
+ searchControlName = new FormControl('');
+
+ filteredNotificationTypesList: Observable>;
+ selectedNotificationTypes: Array = [];
+ notificationTypesTranslateMap = NotificationTemplateTypeTranslateMap;
+
+ separatorKeysCodes: number[] = [ENTER, COMMA, SEMICOLON];
+
+ private notificationType = NotificationType;
+ private notificationTypes = Object.keys(NotificationType) as Array;
+
+ private dirty = false;
+
+ @ViewChild('notificationTypeInput') notificationTypeInput: ElementRef;
+
+ constructor(@Inject(NOTIFICATION_TYPE_FILTER_PANEL_DATA) public data: NotificationTypeFilterPanelData,
+ private overlayRef: OverlayRef) {
+ this.selectedNotificationTypes = this.data.notificationTypes;
+ this.dirty = true;
+ }
+
+ ngOnInit() {
+ this.filteredNotificationTypesList = this.searchControlName.valueChanges.pipe(
+ debounceTime(150),
+ map(value => {
+ this.searchText = value;
+ return this.notificationTypes.filter(type => !this.selectedNotificationTypes.includes(type))
+ .filter(type => value ? type.toUpperCase().startsWith(value.toUpperCase()) : true);
+ })
+ );
+ }
+
+ public update() {
+ this.data.notificationTypesUpdated(this.selectedNotificationTypes);
+ if (this.overlayRef) {
+ this.overlayRef.dispose();
+ }
+ }
+
+ cancel() {
+ if (this.overlayRef) {
+ this.overlayRef.dispose();
+ }
+ }
+
+ public reset() {
+ this.selectedNotificationTypes.length = 0;
+ this.searchControlName.updateValueAndValidity({emitEvent: true});
+ }
+
+ remove(type: NotificationType) {
+ const index = this.selectedNotificationTypes.indexOf(type);
+ if (index >= 0) {
+ this.selectedNotificationTypes.splice(index, 1);
+ this.searchControlName.updateValueAndValidity({emitEvent: true});
+ }
+ }
+
+ onFocus() {
+ if (this.dirty) {
+ this.searchControlName.updateValueAndValidity({emitEvent: true});
+ this.dirty = false;
+ }
+ }
+
+ private add(type: NotificationType): void {
+ this.selectedNotificationTypes.push(type);
+ }
+
+ chipAdd(event: MatChipInputEvent): void {
+ const value = (event.value || '').trim();
+ if (value && this.notificationType[value]) {
+ this.add(this.notificationType[value]);
+ this.clear('');
+ }
+ }
+
+ selected(event: MatAutocompleteSelectedEvent): void {
+ if (this.notificationType[event.option.value]) {
+ this.add(this.notificationType[event.option.value]);
+ }
+ this.clear('');
+ }
+
+ clear(value: string = '') {
+ this.notificationTypeInput.nativeElement.value = value;
+ this.searchControlName.patchValue(value, {emitEvent: true});
+ setTimeout(() => {
+ this.notificationTypeInput.nativeElement.blur();
+ this.notificationTypeInput.nativeElement.focus();
+ }, 0);
+ }
+
+ displayTypeFn(type?: string): string | undefined {
+ return type ? type : undefined;
+ }
+}
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.html b/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.html
new file mode 100644
index 0000000000..3537c8bb1d
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.html
@@ -0,0 +1,54 @@
+
+
+
+
+
+ {{ count$ | async }}
+
+
+
+
+
+
+
+
+
+
+
+
+ notification.no-notifications-yet
+
+
+
+
+
notification.loading-notifications
+
+
+
+
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.scss b/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.scss
new file mode 100644
index 0000000000..5fb3ad8950
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.scss
@@ -0,0 +1,65 @@
+/**
+ * Copyright © 2016-2024 The Thingsboard Authors
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+
+@import "../../../../../../../scss/constants";
+
+.tb-no-notification-svg-color {
+ color: $tb-primary-color;
+}
+
+.tb-unread-notification-panel {
+ width: 100%;
+ height: 100%;
+ position: relative;
+ display: flex;
+ flex-direction: column;
+ gap: 8px;
+ padding: 20px 24px 24px 24px;
+ > div:not(.tb-unread-notification-overlay) {
+ z-index: 1;
+ }
+ div.tb-widget-title {
+ padding: 0;
+ }
+ .tb-unread-notification-overlay {
+ position: absolute;
+ top: 12px;
+ left: 12px;
+ bottom: 12px;
+ right: 12px;
+ }
+ .tb-unread-notification-content {
+ height: 100%;
+ min-height: 0;
+ display: flex;
+ flex-direction: column;
+ align-items: center;
+ .tb-no-notification-text {
+ text-align: center;
+ margin-bottom: 12px;
+ color: rgba(0, 0, 0, 0.38);
+ }
+ }
+ .notification-counter {
+ display: flex;
+ align-items: center;
+ justify-content: center;
+ width: 24px;
+ height: 22px;
+ background-color: green;
+ border-radius: 7px;
+ }
+}
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.ts b/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.ts
new file mode 100644
index 0000000000..fcad768e66
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.component.ts
@@ -0,0 +1,284 @@
+///
+/// Copyright © 2016-2024 The Thingsboard Authors
+///
+/// Licensed under the Apache License, Version 2.0 (the "License");
+/// you may not use this file except in compliance with the License.
+/// You may obtain a copy of the License at
+///
+/// http://www.apache.org/licenses/LICENSE-2.0
+///
+/// Unless required by applicable law or agreed to in writing, software
+/// distributed under the License is distributed on an "AS IS" BASIS,
+/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+/// See the License for the specific language governing permissions and
+/// limitations under the License.
+///
+
+import {
+ ChangeDetectorRef,
+ Component,
+ Injector,
+ Input,
+ NgZone,
+ OnDestroy,
+ OnInit,
+ StaticProvider,
+ TemplateRef,
+ ViewContainerRef,
+ ViewEncapsulation
+} from '@angular/core';
+import { WidgetAction, WidgetContext } from '@home/models/widget-component.models';
+import { isDefined } from '@core/utils';
+import { backgroundStyle, ComponentStyle, overlayStyle, textStyle } from '@shared/models/widget-settings.models';
+import { ResizeObserver } from '@juggle/resize-observer';
+import { BehaviorSubject, fromEvent, Observable, ReplaySubject, Subscription } from 'rxjs';
+import { ImagePipe } from '@shared/pipe/image.pipe';
+import { DomSanitizer } from '@angular/platform-browser';
+import {
+ unreadNotificationDefaultSettings,
+ UnreadNotificationWidgetSettings
+} from '@home/components/widget/lib/cards/unread-notification-widget.models';
+import { Notification, NotificationRequest, NotificationType } from '@shared/models/notification.models';
+import { NotificationSubscriber } from '@shared/models/telemetry/telemetry.models';
+import { NotificationWebsocketService } from '@core/ws/notification-websocket.service';
+import { distinctUntilChanged, map, share, skip, take, tap } from 'rxjs/operators';
+import { Router } from '@angular/router';
+import { Overlay, OverlayConfig, OverlayRef } from '@angular/cdk/overlay';
+import { DEFAULT_OVERLAY_POSITIONS } from '@shared/models/overlay.models';
+import { ComponentPortal } from '@angular/cdk/portal';
+import {
+ NOTIFICATION_TYPE_FILTER_PANEL_DATA,
+ NotificationTypeFilterPanelComponent
+} from '@home/components/widget/lib/cards/notification-type-filter-panel.component';
+import { selectUserDetails } from '@core/auth/auth.selectors';
+import { select, Store } from '@ngrx/store';
+import { AppState } from '@core/core.state';
+
+@Component({
+ selector: 'tb-unread-notification-widget',
+ templateUrl: './unread-notification-widget.component.html',
+ styleUrls: ['unread-notification-widget.component.scss'],
+ encapsulation: ViewEncapsulation.None
+})
+export class UnreadNotificationWidgetComponent implements OnInit, OnDestroy {
+
+ settings: UnreadNotificationWidgetSettings;
+
+ @Input()
+ ctx: WidgetContext;
+
+ @Input()
+ widgetTitlePanel: TemplateRef;
+
+ showCounter = true;
+ counterValueStyle: ComponentStyle;
+ counterBackground: string;
+
+ notifications: Notification[];
+ loadNotification = false;
+
+ backgroundStyle$: Observable;
+ overlayStyle: ComponentStyle = {};
+ padding: string;
+
+ private counterValue: BehaviorSubject = new BehaviorSubject(0);
+
+ count$ = this.counterValue.asObservable().pipe(
+ distinctUntilChanged(),
+ map((value) => value >= 100 ? '99+' : value),
+ tap(() => Promise.resolve().then(() => this.cd.markForCheck())),
+ share({
+ connector: () => new ReplaySubject(1)
+ })
+ );
+
+
+ private notificationTypes: Array = [];
+
+ private notificationSubscriber: NotificationSubscriber;
+ private notificationCountSubscriber: Subscription;
+ private notification: Subscription;
+
+ private contentResize$: ResizeObserver;
+
+ private defaultDashboardFullscreen = false;
+
+ private viewAllAction: WidgetAction = {
+ name: 'widgets.notification.button-view-all',
+ show: !this.defaultDashboardFullscreen,
+ icon: 'open_in_new',
+ onAction: ($event) => {
+ this.viewAll($event);
+ }
+ };
+
+ private filterAction: WidgetAction = {
+ name: 'widgets.notification.button-filter',
+ show: true,
+ icon: 'filter_list',
+ onAction: ($event) => {
+ this.editNotificationTypeFilter($event);
+ }
+ };
+
+ private markAsReadAction: WidgetAction = {
+ name: 'widgets.notification.button-mark-read',
+ show: true,
+ icon: 'done_all',
+ onAction: ($event) => {
+ this.markAsAllRead($event);
+ }
+ };
+
+ constructor(private store: Store,
+ private imagePipe: ImagePipe,
+ private notificationWsService: NotificationWebsocketService,
+ private sanitizer: DomSanitizer,
+ private router: Router,
+ private zone: NgZone,
+ private overlay: Overlay,
+ private viewContainerRef: ViewContainerRef,
+ private cd: ChangeDetectorRef) {
+ }
+
+ ngOnInit(): void {
+ this.ctx.$scope.unreadNotificationWidget = this;
+ this.settings = {...unreadNotificationDefaultSettings, ...this.ctx.settings};
+
+ this.showCounter = this.settings.showCounter;
+ this.counterValueStyle = textStyle(this.settings.counterValueFont);
+ this.counterValueStyle.color = this.settings.counterValueColor;
+ this.counterBackground = this.settings.counterColor;
+
+ this.ctx.widgetActions = [this.viewAllAction, this.filterAction, this.markAsReadAction];
+
+ this.viewAllAction.show = isDefined(this.settings.enableViewAll) ? this.settings.enableViewAll : true;
+ this.store.pipe(select(selectUserDetails), take(1)).subscribe(
+ user => this.viewAllAction.show = !user.additionalInfo?.defaultDashboardFullscreen
+ );
+ this.filterAction.show = isDefined(this.settings.enableFilter) ? this.settings.enableFilter : true;
+ this.markAsReadAction.show = isDefined(this.settings.enableMarkAsRead) ? this.settings.enableMarkAsRead : true;
+
+ this.initSubscription();
+
+ this.backgroundStyle$ = backgroundStyle(this.settings.background, this.imagePipe, this.sanitizer);
+ this.overlayStyle = overlayStyle(this.settings.background.overlay);
+ this.padding = this.settings.background.overlay.enabled ? undefined : this.settings.padding;
+ }
+
+
+ ngOnDestroy() {
+ if (this.contentResize$) {
+ this.contentResize$.disconnect();
+ }
+ this.unsubscribeSubscription();
+ }
+
+ private initSubscription() {
+ this.notificationSubscriber = NotificationSubscriber.createNotificationsSubscription(
+ this.notificationWsService, this.zone, this.settings.maxNotificationDisplay, this.notificationTypes);
+ this.notification = this.notificationSubscriber.notifications$.subscribe(value => {
+ if (Array.isArray(value)) {
+ this.loadNotification = true;
+ this.notifications = value;
+ this.cd.markForCheck();
+ }
+ });
+ this.notificationCountSubscriber = this.notificationSubscriber.notificationCount$.pipe(
+ skip(1),
+ ).subscribe(value => this.counterValue.next(value));
+ this.notificationSubscriber.subscribe();
+ }
+
+ private unsubscribeSubscription() {
+ this.notificationSubscriber.unsubscribe();
+ this.notificationCountSubscriber.unsubscribe();
+ this.notification.unsubscribe();
+ }
+
+ public onInit() {
+ const borderRadius = this.ctx.$widgetElement.css('borderRadius');
+ this.overlayStyle = {...this.overlayStyle, ...{borderRadius}};
+ this.cd.detectChanges();
+ }
+
+ markAsRead(id: string) {
+ const cmd = NotificationSubscriber.createMarkAsReadCommand(this.notificationWsService, [id]);
+ cmd.subscribe();
+ }
+
+ markAsAllRead($event: Event) {
+ if ($event) {
+ $event.stopPropagation();
+ }
+ const cmd = NotificationSubscriber.createMarkAllAsReadCommand(this.notificationWsService);
+ cmd.subscribe();
+ }
+
+ viewAll($event: Event) {
+ if ($event) {
+ $event.stopPropagation();
+ }
+ this.router.navigateByUrl(this.router.parseUrl('/notification/inbox')).then(() => {});
+ }
+
+ trackById(index: number, item: NotificationRequest): string {
+ return item.id.id;
+ }
+
+ private editNotificationTypeFilter($event: Event) {
+ if ($event) {
+ $event.stopPropagation();
+ }
+ const target = $event.target || $event.srcElement || $event.currentTarget;
+ const config = new OverlayConfig({
+ panelClass: 'tb-panel-container',
+ backdropClass: 'cdk-overlay-transparent-backdrop',
+ hasBackdrop: true,
+ height: 'fit-content',
+ maxHeight: '75vh',
+ width: '100%',
+ maxWidth: 700
+ });
+ config.positionStrategy = this.overlay.position()
+ .flexibleConnectedTo(target as HTMLElement)
+ .withPositions(DEFAULT_OVERLAY_POSITIONS);
+
+ const overlayRef = this.overlay.create(config);
+ overlayRef.backdropClick().subscribe(() => {
+ overlayRef.dispose();
+ });
+
+ const providers: StaticProvider[] = [
+ {
+ provide: NOTIFICATION_TYPE_FILTER_PANEL_DATA,
+ useValue: {
+ notificationTypes: this.notificationTypes,
+ notificationTypesUpdated: (notificationTypes: Array) => {
+ this.notificationTypes = notificationTypes;
+ this.unsubscribeSubscription();
+ this.initSubscription();
+ }
+ }
+ },
+ {
+ provide: OverlayRef,
+ useValue: overlayRef
+ }
+ ];
+
+ const injector = Injector.create({parent: this.viewContainerRef.injector, providers});
+ const componentRef = overlayRef.attach(new ComponentPortal(NotificationTypeFilterPanelComponent,
+ this.viewContainerRef, injector));
+
+ const resizeWindows$ = fromEvent(window, 'resize').subscribe(() => {
+ overlayRef.updatePosition();
+ });
+ componentRef.onDestroy(() => {
+ resizeWindows$.unsubscribe();
+ });
+
+ this.ctx.detectChanges();
+ }
+
+}
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.models.ts b/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.models.ts
new file mode 100644
index 0000000000..008c1d9a9d
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/cards/unread-notification-widget.models.ts
@@ -0,0 +1,59 @@
+///
+/// Copyright © 2016-2024 The Thingsboard Authors
+///
+/// Licensed under the Apache License, Version 2.0 (the "License");
+/// you may not use this file except in compliance with the License.
+/// You may obtain a copy of the License at
+///
+/// http://www.apache.org/licenses/LICENSE-2.0
+///
+/// Unless required by applicable law or agreed to in writing, software
+/// distributed under the License is distributed on an "AS IS" BASIS,
+/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+/// See the License for the specific language governing permissions and
+/// limitations under the License.
+///
+
+import { BackgroundSettings, BackgroundType, Font } from '@shared/models/widget-settings.models';
+
+export interface UnreadNotificationWidgetSettings {
+ maxNotificationDisplay: number;
+ showCounter: boolean;
+ counterValueFont: Font;
+ counterValueColor: string;
+ counterColor: string;
+
+ enableViewAll: boolean;
+ enableFilter: boolean;
+ enableMarkAsRead: boolean;
+ background: BackgroundSettings;
+ padding: string;
+}
+
+export const unreadNotificationDefaultSettings: UnreadNotificationWidgetSettings = {
+ maxNotificationDisplay: 6,
+ showCounter: true,
+ counterValueFont: {
+ family: 'Roboto',
+ size: 14,
+ sizeUnit: 'px',
+ style: 'normal',
+ weight: '600',
+ lineHeight: ''
+ },
+ counterValueColor: '#fff',
+ counterColor: '#305680',
+ enableViewAll: true,
+ enableFilter: true,
+ enableMarkAsRead: true,
+ background: {
+ type: BackgroundType.color,
+ color: '#fff',
+ overlay: {
+ enabled: false,
+ color: 'rgba(255,255,255,0.72)',
+ blur: 3
+ }
+ },
+ padding: '12px'
+};
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/settings/cards/unread-notification-widget-settings.component.html b/ui-ngx/src/app/modules/home/components/widget/lib/settings/cards/unread-notification-widget-settings.component.html
new file mode 100644
index 0000000000..5f452f3508
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/settings/cards/unread-notification-widget-settings.component.html
@@ -0,0 +1,79 @@
+
+
+
+
+
+
+
+
+
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/settings/cards/unread-notification-widget-settings.component.ts b/ui-ngx/src/app/modules/home/components/widget/lib/settings/cards/unread-notification-widget-settings.component.ts
new file mode 100644
index 0000000000..1bfe38f73c
--- /dev/null
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/settings/cards/unread-notification-widget-settings.component.ts
@@ -0,0 +1,81 @@
+///
+/// Copyright © 2016-2024 The Thingsboard Authors
+///
+/// Licensed under the Apache License, Version 2.0 (the "License");
+/// you may not use this file except in compliance with the License.
+/// You may obtain a copy of the License at
+///
+/// http://www.apache.org/licenses/LICENSE-2.0
+///
+/// Unless required by applicable law or agreed to in writing, software
+/// distributed under the License is distributed on an "AS IS" BASIS,
+/// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+/// See the License for the specific language governing permissions and
+/// limitations under the License.
+///
+
+import { Component } from '@angular/core';
+import { WidgetSettings, WidgetSettingsComponent } from '@shared/models/widget.models';
+import { UntypedFormBuilder, UntypedFormGroup, Validators } from '@angular/forms';
+import { Store } from '@ngrx/store';
+import { AppState } from '@core/core.state';
+import { unreadNotificationDefaultSettings } from '@home/components/widget/lib/cards/unread-notification-widget.models';
+
+@Component({
+ selector: 'tb-unread-notification-widget-settings',
+ templateUrl: './unread-notification-widget-settings.component.html',
+ styleUrls: ['./../widget-settings.scss']
+})
+export class UnreadNotificationWidgetSettingsComponent extends WidgetSettingsComponent {
+
+ unreadNotificationWidgetSettingsForm: UntypedFormGroup;
+
+ constructor(protected store: Store,
+ private fb: UntypedFormBuilder) {
+ super(store);
+ }
+
+ protected settingsForm(): UntypedFormGroup {
+ return this.unreadNotificationWidgetSettingsForm;
+ }
+
+ protected defaultSettings(): WidgetSettings {
+ return {...unreadNotificationDefaultSettings};
+ }
+
+ protected onSettingsSet(settings: WidgetSettings) {
+ this.unreadNotificationWidgetSettingsForm = this.fb.group({
+ maxNotificationDisplay: [settings?.maxNotificationDisplay, [Validators.required, Validators.min(1)]],
+ showCounter: [settings?.showCounter, []],
+ counterValueFont: [settings?.counterValueFont, []],
+ counterValueColor: [settings?.counterValueColor, []],
+ counterColor: [settings?.counterColor, []],
+
+ enableViewAll: [settings?.enableViewAll, []],
+ enableFilter: [settings?.enableFilter, []],
+ enableMarkAsRead: [settings?.enableMarkAsRead, []],
+
+ background: [settings?.background, []],
+ padding: [settings.padding, []]
+ });
+ }
+
+ protected validatorTriggers(): string[] {
+ return ['showCounter'];
+ }
+
+ protected updateValidators(emitEvent: boolean) {
+ const showCounter: boolean = this.unreadNotificationWidgetSettingsForm.get('showCounter').value;
+
+ if (showCounter) {
+ this.unreadNotificationWidgetSettingsForm.get('counterValueFont').enable({emitEvent});
+ this.unreadNotificationWidgetSettingsForm.get('counterValueColor').enable({emitEvent});
+ this.unreadNotificationWidgetSettingsForm.get('counterColor').enable({emitEvent});
+ } else {
+ this.unreadNotificationWidgetSettingsForm.get('counterValueFont').disable({emitEvent});
+ this.unreadNotificationWidgetSettingsForm.get('counterValueColor').disable({emitEvent});
+ this.unreadNotificationWidgetSettingsForm.get('counterColor').disable({emitEvent});
+ }
+ }
+
+}
diff --git a/ui-ngx/src/app/modules/home/components/widget/lib/settings/widget-settings.module.ts b/ui-ngx/src/app/modules/home/components/widget/lib/settings/widget-settings.module.ts
index e4df67e2f8..778e84a614 100644
--- a/ui-ngx/src/app/modules/home/components/widget/lib/settings/widget-settings.module.ts
+++ b/ui-ngx/src/app/modules/home/components/widget/lib/settings/widget-settings.module.ts
@@ -362,6 +362,9 @@ import {
import {
LabelValueCardWidgetSettingsComponent
} from '@home/components/widget/lib/settings/cards/label-value-card-widget-settings.component';
+import {
+ UnreadNotificationWidgetSettingsComponent
+} from '@home/components/widget/lib/settings/cards/unread-notification-widget-settings.component';
@NgModule({
declarations: [
@@ -490,7 +493,8 @@ import {
PolarAreaChartWidgetSettingsComponent,
RadarChartWidgetSettingsComponent,
LabelCardWidgetSettingsComponent,
- LabelValueCardWidgetSettingsComponent
+ LabelValueCardWidgetSettingsComponent,
+ UnreadNotificationWidgetSettingsComponent,
],
imports: [
CommonModule,
@@ -624,7 +628,8 @@ import {
PolarAreaChartWidgetSettingsComponent,
RadarChartWidgetSettingsComponent,
LabelCardWidgetSettingsComponent,
- LabelValueCardWidgetSettingsComponent
+ LabelValueCardWidgetSettingsComponent,
+ UnreadNotificationWidgetSettingsComponent
]
})
export class WidgetSettingsModule {
@@ -725,5 +730,6 @@ export const widgetSettingsComponentsMap: {[key: string]: Type
-
+
+ [ngStyle]="{borderColor: notificationColor(), backgroundColor: notificationBackgroundColor()}">
{{ notification.additionalConfig.icon.icon }}
diff --git a/ui-ngx/src/app/shared/components/notification/notification.component.ts b/ui-ngx/src/app/shared/components/notification/notification.component.ts
index 0d784a4138..43935bbea4 100644
--- a/ui-ngx/src/app/shared/components/notification/notification.component.ts
+++ b/ui-ngx/src/app/shared/components/notification/notification.component.ts
@@ -145,6 +145,13 @@ export class NotificationComponent implements OnInit {
return 'transparent';
}
+ notificationBackgroundColor(): string {
+ if (this.notification.type === NotificationType.ALARM && !this.notification.info.cleared) {
+ return '#fff';
+ }
+ return 'transparent';
+ }
+
notificationIconColor(): object {
if (this.notification.type === NotificationType.ALARM) {
return {color: AlarmSeverityNotificationColors.get(this.notification.info.alarmSeverity)};
diff --git a/ui-ngx/src/app/shared/models/telemetry/telemetry.models.ts b/ui-ngx/src/app/shared/models/telemetry/telemetry.models.ts
index e1c6af620b..8cf07e1128 100644
--- a/ui-ngx/src/app/shared/models/telemetry/telemetry.models.ts
+++ b/ui-ngx/src/app/shared/models/telemetry/telemetry.models.ts
@@ -38,7 +38,7 @@ import { entityFields } from '@shared/models/entity.models';
import { isDefinedAndNotNull, isUndefined } from '@core/utils';
import { CmdWrapper, WsService, WsSubscriber } from '@shared/models/websocket/websocket.models';
import { TelemetryWebsocketService } from '@core/ws/telemetry-websocket.service';
-import { Notification } from '@shared/models/notification.models';
+import { Notification, NotificationType } from '@shared/models/notification.models';
import { WebsocketService } from '@core/ws/websocket.service';
export const NOT_SUPPORTED = 'Not supported!';
@@ -304,11 +304,14 @@ export class UnreadCountSubCmd implements WebsocketCmd {
export class UnreadSubCmd implements WebsocketCmd {
limit: number;
+ types: Array;
cmdId: number;
type = WsCmdType.NOTIFICATIONS;
- constructor(limit = 10) {
+ constructor(limit = 10,
+ types: Array = []) {
this.limit = limit;
+ this.types = types;
}
}
@@ -911,6 +914,8 @@ export class NotificationSubscriber extends WsSubscriber {
public messageLimit = 10;
+ public notificationType = [];
+
public notificationCount$ = this.notificationCountSubject.asObservable().pipe(map(msg => msg.totalUnreadCount));
public notifications$ = this.notificationsSubject.asObservable().pipe(map(msg => msg.notifications ));
@@ -923,8 +928,8 @@ export class NotificationSubscriber extends WsSubscriber {
}
public static createNotificationsSubscription(websocketService: WebsocketService,
- zone: NgZone, limit = 10): NotificationSubscriber {
- const subscriptionCommand = new UnreadSubCmd(limit);
+ zone: NgZone, limit = 10, types: Array = []): NotificationSubscriber {
+ const subscriptionCommand = new UnreadSubCmd(limit, types);
const subscriber = new NotificationSubscriber(websocketService, zone);
subscriber.messageLimit = limit;
subscriber.subscriptionCommands.push(subscriptionCommand);
diff --git a/ui-ngx/src/assets/locale/locale.constant-en_US.json b/ui-ngx/src/assets/locale/locale.constant-en_US.json
index b1b17ca3aa..b161e1e3bc 100644
--- a/ui-ngx/src/assets/locale/locale.constant-en_US.json
+++ b/ui-ngx/src/assets/locale/locale.constant-en_US.json
@@ -7001,6 +7001,22 @@
"bar-background": "Bar background",
"progress-bar-card-style": "Progress bar card style"
},
+ "notification": {
+ "max-notification-display": "Maximum notifications to display",
+ "counter": "Counter",
+ "icon": "Icon",
+ "counter-value": "Value",
+ "counter-color": "Color",
+ "notification-button": "Notification buttons",
+ "button-view-all": "View all",
+ "button-filter": "Filter",
+ "type-filter": "Type filter",
+ "button-mark-read": "Mark as read",
+ "notification-types": "Notification types",
+ "notification-type": "Notification type",
+ "search-type": "Search type",
+ "any-type": "Any type"
+ },
"alarm-count": {
"alarm-count-card-style": "Alarm count card style"
},
diff --git a/ui-ngx/src/assets/notification-bell.svg b/ui-ngx/src/assets/notification-bell.svg
index f5bb251293..25b9cc3086 100644
--- a/ui-ngx/src/assets/notification-bell.svg
+++ b/ui-ngx/src/assets/notification-bell.svg
@@ -1 +1,13 @@
-
+