diff options
Diffstat (limited to 'client/src/app/menu')
-rw-r--r-- | client/src/app/menu/avatar-notification.component.html | 23 | ||||
-rw-r--r-- | client/src/app/menu/avatar-notification.component.scss | 82 | ||||
-rw-r--r-- | client/src/app/menu/avatar-notification.component.ts | 64 | ||||
-rw-r--r-- | client/src/app/menu/index.ts | 2 | ||||
-rw-r--r-- | client/src/app/menu/menu.component.html | 6 | ||||
-rw-r--r-- | client/src/app/menu/menu.component.scss | 7 |
6 files changed, 173 insertions, 11 deletions
diff --git a/client/src/app/menu/avatar-notification.component.html b/client/src/app/menu/avatar-notification.component.html new file mode 100644 index 000000000..2f0b7c669 --- /dev/null +++ b/client/src/app/menu/avatar-notification.component.html | |||
@@ -0,0 +1,23 @@ | |||
1 | <div | ||
2 | [ngbPopover]="popContent" autoClose="outside" placement="bottom-left" container="body" popoverClass="popover-notifications" | ||
3 | i18n-title title="View your notifications" class="notification-avatar" #popover="ngbPopover" | ||
4 | > | ||
5 | <div *ngIf="unreadNotifications > 0" class="unread-notifications">{{ unreadNotifications }}</div> | ||
6 | |||
7 | <img [src]="user.accountAvatarUrl" alt="Avatar" /> | ||
8 | </div> | ||
9 | |||
10 | <ng-template #popContent> | ||
11 | <div class="notifications-header"> | ||
12 | <div i18n>Notifications</div> | ||
13 | |||
14 | <a | ||
15 | i18n-title title="Update your notification preferences" class="glyphicon glyphicon-cog" | ||
16 | routerLink="/my-account/settings" fragment="notifications" | ||
17 | ></a> | ||
18 | </div> | ||
19 | |||
20 | <my-user-notifications [ignoreLoadingBar]="true" [infiniteScroll]="false"></my-user-notifications> | ||
21 | |||
22 | <a class="all-notifications" routerLink="/my-account/notifications" i18n>See all your notifications</a> | ||
23 | </ng-template> | ||
diff --git a/client/src/app/menu/avatar-notification.component.scss b/client/src/app/menu/avatar-notification.component.scss new file mode 100644 index 000000000..c86667469 --- /dev/null +++ b/client/src/app/menu/avatar-notification.component.scss | |||
@@ -0,0 +1,82 @@ | |||
1 | @import '_variables'; | ||
2 | @import '_mixins'; | ||
3 | |||
4 | /deep/ { | ||
5 | .popover-notifications.popover { | ||
6 | max-width: 400px; | ||
7 | |||
8 | .popover-body { | ||
9 | padding: 0; | ||
10 | font-size: 14px; | ||
11 | font-family: $main-fonts; | ||
12 | overflow-y: auto; | ||
13 | max-height: 500px; | ||
14 | box-shadow: 0 6px 14px rgba(0, 0, 0, 0.30); | ||
15 | |||
16 | .notifications-header { | ||
17 | display: flex; | ||
18 | justify-content: space-between; | ||
19 | |||
20 | background-color: rgba(0, 0, 0, 0.10); | ||
21 | align-items: center; | ||
22 | padding: 0 10px; | ||
23 | font-size: 16px; | ||
24 | height: 50px; | ||
25 | |||
26 | a { | ||
27 | @include disable-default-a-behaviour; | ||
28 | |||
29 | color: rgba(20, 20, 20, 0.5); | ||
30 | |||
31 | &:hover { | ||
32 | color: rgba(20, 20, 20, 0.8); | ||
33 | } | ||
34 | } | ||
35 | } | ||
36 | |||
37 | .all-notifications { | ||
38 | display: flex; | ||
39 | align-items: center; | ||
40 | justify-content: center; | ||
41 | font-weight: $font-semibold; | ||
42 | color: var(--mainForegroundColor); | ||
43 | height: 30px; | ||
44 | } | ||
45 | } | ||
46 | } | ||
47 | } | ||
48 | |||
49 | .notification-avatar { | ||
50 | cursor: pointer; | ||
51 | position: relative; | ||
52 | |||
53 | img, | ||
54 | .unread-notifications { | ||
55 | margin-left: 20px; | ||
56 | } | ||
57 | |||
58 | img { | ||
59 | @include avatar(34px); | ||
60 | |||
61 | margin-right: 10px; | ||
62 | } | ||
63 | |||
64 | .unread-notifications { | ||
65 | position: absolute; | ||
66 | top: -5px; | ||
67 | left: -5px; | ||
68 | |||
69 | display: flex; | ||
70 | align-items: center; | ||
71 | justify-content: center; | ||
72 | |||
73 | background-color: var(--mainColor); | ||
74 | color: var(--mainBackgroundColor); | ||
75 | font-size: 10px; | ||
76 | font-weight: $font-semibold; | ||
77 | |||
78 | border-radius: 15px; | ||
79 | width: 15px; | ||
80 | height: 15px; | ||
81 | } | ||
82 | } | ||
diff --git a/client/src/app/menu/avatar-notification.component.ts b/client/src/app/menu/avatar-notification.component.ts new file mode 100644 index 000000000..60e090726 --- /dev/null +++ b/client/src/app/menu/avatar-notification.component.ts | |||
@@ -0,0 +1,64 @@ | |||
1 | import { Component, Input, OnDestroy, OnInit, ViewChild } from '@angular/core' | ||
2 | import { User } from '../shared/users/user.model' | ||
3 | import { UserNotificationService } from '@app/shared/users/user-notification.service' | ||
4 | import { Subscription } from 'rxjs' | ||
5 | import { Notifier } from '@app/core' | ||
6 | import { NgbPopover } from '@ng-bootstrap/ng-bootstrap' | ||
7 | import { NavigationEnd, Router } from '@angular/router' | ||
8 | import { filter } from 'rxjs/operators' | ||
9 | |||
10 | @Component({ | ||
11 | selector: 'my-avatar-notification', | ||
12 | templateUrl: './avatar-notification.component.html', | ||
13 | styleUrls: [ './avatar-notification.component.scss' ] | ||
14 | }) | ||
15 | export class AvatarNotificationComponent implements OnInit, OnDestroy { | ||
16 | @ViewChild('popover') popover: NgbPopover | ||
17 | @Input() user: User | ||
18 | |||
19 | unreadNotifications = 0 | ||
20 | |||
21 | private notificationSub: Subscription | ||
22 | private routeSub: Subscription | ||
23 | |||
24 | constructor ( | ||
25 | private userNotificationService: UserNotificationService, | ||
26 | private notifier: Notifier, | ||
27 | private router: Router | ||
28 | ) {} | ||
29 | |||
30 | ngOnInit () { | ||
31 | this.userNotificationService.countUnreadNotifications() | ||
32 | .subscribe( | ||
33 | result => { | ||
34 | this.unreadNotifications = Math.min(result, 99) // Limit number to 99 | ||
35 | this.subscribeToNotifications() | ||
36 | }, | ||
37 | |||
38 | err => this.notifier.error(err.message) | ||
39 | ) | ||
40 | |||
41 | this.routeSub = this.router.events | ||
42 | .pipe(filter(event => event instanceof NavigationEnd)) | ||
43 | .subscribe(() => this.closePopover()) | ||
44 | } | ||
45 | |||
46 | ngOnDestroy () { | ||
47 | if (this.notificationSub) this.notificationSub.unsubscribe() | ||
48 | if (this.routeSub) this.routeSub.unsubscribe() | ||
49 | } | ||
50 | |||
51 | closePopover () { | ||
52 | this.popover.close() | ||
53 | } | ||
54 | |||
55 | private subscribeToNotifications () { | ||
56 | this.notificationSub = this.userNotificationService.getMyNotificationsSocket() | ||
57 | .subscribe(data => { | ||
58 | if (data.type === 'new') return this.unreadNotifications++ | ||
59 | if (data.type === 'read') return this.unreadNotifications-- | ||
60 | if (data.type === 'read-all') return this.unreadNotifications = 0 | ||
61 | }) | ||
62 | } | ||
63 | |||
64 | } | ||
diff --git a/client/src/app/menu/index.ts b/client/src/app/menu/index.ts index 421271c12..39dbde750 100644 --- a/client/src/app/menu/index.ts +++ b/client/src/app/menu/index.ts | |||
@@ -1 +1,3 @@ | |||
1 | export * from './language-chooser.component' | ||
2 | export * from './avatar-notification.component' | ||
1 | export * from './menu.component' | 3 | export * from './menu.component' |
diff --git a/client/src/app/menu/menu.component.html b/client/src/app/menu/menu.component.html index e04bdf3d6..aa5bfa9c9 100644 --- a/client/src/app/menu/menu.component.html +++ b/client/src/app/menu/menu.component.html | |||
@@ -2,9 +2,7 @@ | |||
2 | <menu> | 2 | <menu> |
3 | <div class="top-menu"> | 3 | <div class="top-menu"> |
4 | <div *ngIf="isLoggedIn" class="logged-in-block"> | 4 | <div *ngIf="isLoggedIn" class="logged-in-block"> |
5 | <a routerLink="/my-account/settings"> | 5 | <my-avatar-notification [user]="user"></my-avatar-notification> |
6 | <img [src]="user.accountAvatarUrl" alt="Avatar" /> | ||
7 | </a> | ||
8 | 6 | ||
9 | <div class="logged-in-info"> | 7 | <div class="logged-in-info"> |
10 | <a routerLink="/my-account/settings" class="logged-in-username">{{ user.account?.displayName }}</a> | 8 | <a routerLink="/my-account/settings" class="logged-in-username">{{ user.account?.displayName }}</a> |
@@ -97,4 +95,4 @@ | |||
97 | </menu> | 95 | </menu> |
98 | </div> | 96 | </div> |
99 | 97 | ||
100 | <my-language-chooser #languageChooserModal></my-language-chooser> \ No newline at end of file | 98 | <my-language-chooser #languageChooserModal></my-language-chooser> |
diff --git a/client/src/app/menu/menu.component.scss b/client/src/app/menu/menu.component.scss index b271ebfd2..a4aaadc7f 100644 --- a/client/src/app/menu/menu.component.scss +++ b/client/src/app/menu/menu.component.scss | |||
@@ -39,13 +39,6 @@ menu { | |||
39 | justify-content: center; | 39 | justify-content: center; |
40 | margin-bottom: 35px; | 40 | margin-bottom: 35px; |
41 | 41 | ||
42 | img { | ||
43 | @include avatar(34px); | ||
44 | |||
45 | margin-left: 20px; | ||
46 | margin-right: 10px; | ||
47 | } | ||
48 | |||
49 | .logged-in-info { | 42 | .logged-in-info { |
50 | flex-grow: 1; | 43 | flex-grow: 1; |
51 | 44 | ||