@arthakosh/chat 0.0.1

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.
Files changed (32) hide show
  1. package/README.md +24 -0
  2. package/esm2022/arthakosh-chat.mjs +5 -0
  3. package/esm2022/lib/all-chats/all-chats.component.mjs +121 -0
  4. package/esm2022/lib/chat-launcher/chat-launcher.component.mjs +96 -0
  5. package/esm2022/lib/chat-widget-notification/chat-widget-notification.component.mjs +69 -0
  6. package/esm2022/lib/chat-widget.module.mjs +47 -0
  7. package/esm2022/lib/chat-widget.service.mjs +12 -0
  8. package/esm2022/lib/chat-window/chat-window.component.mjs +210 -0
  9. package/esm2022/lib/core/config/chat-config.mjs +2 -0
  10. package/esm2022/lib/core/config/chat.tokens.mjs +4 -0
  11. package/esm2022/lib/core/models/chat.models.mjs +2 -0
  12. package/esm2022/lib/core/models/notification.model.mjs +2 -0
  13. package/esm2022/lib/core/services/chat.service.mjs +117 -0
  14. package/esm2022/lib/core/services/socket.service.mjs +148 -0
  15. package/esm2022/public-api.mjs +10 -0
  16. package/fesm2022/arthakosh-chat.mjs +776 -0
  17. package/fesm2022/arthakosh-chat.mjs.map +1 -0
  18. package/index.d.ts +5 -0
  19. package/lib/all-chats/all-chats.component.d.ts +23 -0
  20. package/lib/chat-launcher/chat-launcher.component.d.ts +34 -0
  21. package/lib/chat-widget-notification/chat-widget-notification.component.d.ts +22 -0
  22. package/lib/chat-widget.module.d.ts +14 -0
  23. package/lib/chat-widget.service.d.ts +6 -0
  24. package/lib/chat-window/chat-window.component.d.ts +49 -0
  25. package/lib/core/config/chat-config.d.ts +4 -0
  26. package/lib/core/config/chat.tokens.d.ts +3 -0
  27. package/lib/core/models/chat.models.d.ts +22 -0
  28. package/lib/core/models/notification.model.d.ts +6 -0
  29. package/lib/core/services/chat.service.d.ts +34 -0
  30. package/lib/core/services/socket.service.d.ts +38 -0
  31. package/package.json +42 -0
  32. package/public-api.d.ts +6 -0
package/README.md ADDED
@@ -0,0 +1,24 @@
1
+ # ChatWidget
2
+
3
+ This library was generated with [Angular CLI](https://github.com/angular/angular-cli) version 17.3.0.
4
+
5
+ ## Code scaffolding
6
+
7
+ Run `ng generate component component-name --project chat-widget` to generate a new component. You can also use `ng generate directive|pipe|service|class|guard|interface|enum|module --project chat-widget`.
8
+ > Note: Don't forget to add `--project chat-widget` or else it will be added to the default project in your `angular.json` file.
9
+
10
+ ## Build
11
+
12
+ Run `ng build chat-widget` to build the project. The build artifacts will be stored in the `dist/` directory.
13
+
14
+ ## Publishing
15
+
16
+ After building your library with `ng build chat-widget`, go to the dist folder `cd dist/chat-widget` and run `npm publish`.
17
+
18
+ ## Running unit tests
19
+
20
+ Run `ng test chat-widget` to execute the unit tests via [Karma](https://karma-runner.github.io).
21
+
22
+ ## Further help
23
+
24
+ To get more help on the Angular CLI use `ng help` or go check out the [Angular CLI Overview and Command Reference](https://angular.io/cli) page.
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ export * from './public-api';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXJ0aGFrb3NoLWNoYXQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9wcm9qZWN0cy9jaGF0LXdpZGdldC9zcmMvYXJ0aGFrb3NoLWNoYXQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7O0dBRUc7QUFFSCxjQUFjLGNBQWMsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogR2VuZXJhdGVkIGJ1bmRsZSBpbmRleC4gRG8gbm90IGVkaXQuXG4gKi9cblxuZXhwb3J0ICogZnJvbSAnLi9wdWJsaWMtYXBpJztcbiJdfQ==
@@ -0,0 +1,121 @@
1
+ import { Component, EventEmitter, inject, input, Output, signal } from '@angular/core';
2
+ import { ChatService } from '../core/services/chat.service';
3
+ import { ChatWindowComponent } from "../chat-window/chat-window.component";
4
+ import { CommonModule } from '@angular/common';
5
+ import { FormsModule } from '@angular/forms';
6
+ import { AvatarModule } from 'primeng/avatar';
7
+ import { ButtonModule } from 'primeng/button';
8
+ import { ChipModule } from 'primeng/chip';
9
+ import { DropdownModule } from 'primeng/dropdown';
10
+ import { InputTextareaModule } from 'primeng/inputtextarea';
11
+ import { OverlayPanelModule } from 'primeng/overlaypanel';
12
+ import { ToastModule } from 'primeng/toast';
13
+ import { TooltipModule } from 'primeng/tooltip';
14
+ import { animate, style, transition, trigger } from '@angular/animations';
15
+ import { SocketService } from '../core/services/socket.service';
16
+ import * as i0 from "@angular/core";
17
+ import * as i1 from "@angular/common";
18
+ import * as i2 from "primeng/avatar";
19
+ export class AllChatsComponent {
20
+ constructor() {
21
+ this.user = input(null);
22
+ this.roomId = input(null);
23
+ this.chatUsers = signal([]);
24
+ this.newRoomLoaded = new EventEmitter();
25
+ this.rooms = [];
26
+ this.activeRoomId = signal('');
27
+ this.activeRoomHeader = signal('');
28
+ this.activeRoomMetadata = signal('');
29
+ this.chatService = inject(ChatService);
30
+ this.socketService = inject(SocketService);
31
+ this.showMessages = false;
32
+ }
33
+ ngOnInit() {
34
+ if (!this.user) {
35
+ console.error('user is required');
36
+ return;
37
+ }
38
+ this.chatService.currentUser.set({
39
+ roomId: this.activeRoomId,
40
+ userId: this.user()?.id,
41
+ name: this.user()?.name
42
+ });
43
+ this.socketService.connect(this.user().id);
44
+ this.chatService.getRooms(this.user()?.id, this.roomId()).subscribe(res => {
45
+ this.rooms = res;
46
+ if (this.rooms.length != 0) {
47
+ this.selectRoom(this.rooms[0]);
48
+ this.activeRoomId.set(this.rooms[0].ChatRoomId);
49
+ this.activeRoomHeader.set(this.rooms[0].ChatRoomName);
50
+ this.activeRoomMetadata.set(this.rooms[0].Metadata);
51
+ }
52
+ else {
53
+ this.chatService.addUserToRoom(this.roomId(), this.user()?.id, this.user()?.name, this.chatService.currentUser().userId).subscribe((res) => {
54
+ this.rooms = [];
55
+ this.getRooms();
56
+ });
57
+ }
58
+ });
59
+ }
60
+ getRooms() {
61
+ this.chatService.getRooms(this.user()?.id, this.roomId()).subscribe(res => {
62
+ this.rooms = res;
63
+ if (this.rooms.length != 0) {
64
+ this.selectRoom(this.rooms[0]);
65
+ this.activeRoomId.set(this.rooms[0].ChatRoomId);
66
+ this.activeRoomHeader.set(this.rooms[0].ChatRoomName);
67
+ this.activeRoomMetadata.set(this.rooms[0].Metadata);
68
+ }
69
+ });
70
+ }
71
+ selectRoom(room) {
72
+ this.showMessages = false;
73
+ this.activeRoomId.set(room.ChatRoomId);
74
+ this.activeRoomHeader.set(room.ChatRoomName);
75
+ this.activeRoomMetadata.set(room.Metadata);
76
+ setTimeout(() => {
77
+ this.showMessages = true;
78
+ }, 800);
79
+ }
80
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AllChatsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
81
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: AllChatsComponent, isStandalone: true, selector: "app-all-chats", inputs: { user: { classPropertyName: "user", publicName: "user", isSignal: true, isRequired: false, transformFunction: null }, roomId: { classPropertyName: "roomId", publicName: "roomId", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { newRoomLoaded: "newRoomLoaded" }, ngImport: i0, template: "<div class=\"chat-page\">\r\n <!-- LEFT SIDEBAR : CHAT ROOMS -->\r\n <aside class=\"chat-sidebar glass\">\r\n <div class=\"sidebar-header\">\r\n <h3>Chat Rooms</h3>\r\n </div>\r\n <ul class=\"room-list\">\r\n <li (click)=\"selectRoom(room)\" *ngFor=\"let room of rooms;\" [class.active]=\"activeRoomId() === room.ChatRoomId\"\r\n class=\"room-item\">\r\n <div class=\"room-avatar\">\r\n <p-avatar [label]=\"room.ChatRoomName[0]\"></p-avatar>\r\n </div>\r\n <div class=\"room-info\">\r\n <span class=\"room-name\">{{ room.ChatRoomName }}</span>\r\n <span class=\"room-name\">{{ room.Metadata }}</span>\r\n </div>\r\n </li>\r\n </ul>\r\n </aside>\r\n <!-- RIGHT PANEL : CHAT WINDOW -->\r\n @if (showMessages) {\r\n <section *ngIf=\"showMessages\" @roomSwitch class=\"chat-main glass\">\r\n <app-chat-window [chatRoomId]=\"activeRoomId()\" [header]=\"activeRoomHeader()\" [metadata]=\"activeRoomMetadata()\" [showActions]=\"false\"\r\n mode=\"fullscreen\" theme=\"light\">\r\n </app-chat-window>\r\n </section>\r\n } @else {\r\n <section class=\"chat-main glass loading-container\">\r\n <div class=\"spinner\"></div>\r\n <p>Loading messages\u2026</p>\r\n </section>\r\n }\r\n\r\n</div>", styles: [":root{--glass-bg: rgba(255, 255, 255, .75);--glass-border: rgba(255, 255, 255, .25);--primary: #5b6dff;--text-dark: #1c1e21;--text-muted: #65676b}*{box-sizing:border-box;font-family:Segoe UI,system-ui,-apple-system,BlinkMacSystemFont}.chat-page{display:flex;height:100vh;padding:16px;gap:16px;background:linear-gradient(135deg,#e9efff,#f6f7fb)}.glass{background:var(--glass-bg);backdrop-filter:blur(14px);-webkit-backdrop-filter:blur(14px);border-radius:18px;border:1px solid var(--glass-border);box-shadow:0 20px 40px #00000014,inset 0 1px #fff6}.chat-sidebar{width:300px;display:flex;flex-direction:column;overflow:hidden}.sidebar-header{padding:18px 20px;font-size:18px;font-weight:700;color:var(--text-dark);border-bottom:1px solid rgba(0,0,0,.06)}.room-list{list-style:none;overflow-y:auto;padding:6px}.room-item{display:flex;align-items:center;gap:14px;padding:12px 14px;border-radius:14px;cursor:pointer;transition:all .25s ease;margin-bottom:6px}.room-item:hover{background:#5b6dff14}.room-item.active{background:#5b6dff26;box-shadow:inset 0 0 0 1px #5b6dff40}.room-avatar p-avatar{width:44px;height:44px;border-radius:50%;background:linear-gradient(135deg,#5b6dff,#7a89ff);color:#fff;font-weight:700;display:flex;align-items:center;justify-content:center}.room-info{display:flex;flex-direction:column;overflow:hidden}.room-name{font-size:14px;font-weight:600;color:var(--text-dark);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.room-info span:last-child{font-size:12px;color:var(--text-muted)}.chat-main{flex:1;display:flex;flex-direction:column;overflow:hidden;position:relative}.chat-main:after{content:\"\";position:absolute;inset:0;border-radius:18px;pointer-events:none;box-shadow:inset 0 0 0 1px #fff3}.room-item:hover{transform:translateY(-1px)}.loading-container{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%}.spinner{width:40px;height:40px;border:4px solid rgba(0,0,0,.1);border-top-color:#4f46e5;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "ngmodule", type: ToastModule }, { kind: "ngmodule", type: ButtonModule }, { kind: "ngmodule", type: AvatarModule }, { kind: "component", type: i2.Avatar, selector: "p-avatar", inputs: ["label", "icon", "image", "size", "shape", "style", "styleClass", "ariaLabel", "ariaLabelledBy"], outputs: ["onImageError"] }, { kind: "ngmodule", type: TooltipModule }, { kind: "ngmodule", type: InputTextareaModule }, { kind: "ngmodule", type: ChipModule }, { kind: "ngmodule", type: DropdownModule }, { kind: "ngmodule", type: OverlayPanelModule }, { kind: "component", type: ChatWindowComponent, selector: "app-chat-window", inputs: ["chatRoomId", "header", "metadata", "chatUsers", "showActions", "mode", "theme", "refreshRoom"], outputs: ["onArchive"] }], animations: [
82
+ trigger('roomSwitch', [
83
+ transition(':enter', [
84
+ style({ opacity: 0, transform: 'translateX(24px)' }),
85
+ animate('300ms cubic-bezier(.4,0,.2,1)', style({ opacity: 1, transform: 'translateX(0)' }))
86
+ ]),
87
+ transition(':leave', [
88
+ animate('200ms ease-in', style({ opacity: 0, transform: 'translateX(-24px)' }))
89
+ ])
90
+ ])
91
+ ] }); }
92
+ }
93
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: AllChatsComponent, decorators: [{
94
+ type: Component,
95
+ args: [{ selector: 'app-all-chats', standalone: true, imports: [
96
+ CommonModule,
97
+ FormsModule,
98
+ ToastModule,
99
+ ButtonModule,
100
+ AvatarModule,
101
+ TooltipModule,
102
+ InputTextareaModule,
103
+ ChipModule,
104
+ DropdownModule,
105
+ OverlayPanelModule,
106
+ ChatWindowComponent
107
+ ], animations: [
108
+ trigger('roomSwitch', [
109
+ transition(':enter', [
110
+ style({ opacity: 0, transform: 'translateX(24px)' }),
111
+ animate('300ms cubic-bezier(.4,0,.2,1)', style({ opacity: 1, transform: 'translateX(0)' }))
112
+ ]),
113
+ transition(':leave', [
114
+ animate('200ms ease-in', style({ opacity: 0, transform: 'translateX(-24px)' }))
115
+ ])
116
+ ])
117
+ ], template: "<div class=\"chat-page\">\r\n <!-- LEFT SIDEBAR : CHAT ROOMS -->\r\n <aside class=\"chat-sidebar glass\">\r\n <div class=\"sidebar-header\">\r\n <h3>Chat Rooms</h3>\r\n </div>\r\n <ul class=\"room-list\">\r\n <li (click)=\"selectRoom(room)\" *ngFor=\"let room of rooms;\" [class.active]=\"activeRoomId() === room.ChatRoomId\"\r\n class=\"room-item\">\r\n <div class=\"room-avatar\">\r\n <p-avatar [label]=\"room.ChatRoomName[0]\"></p-avatar>\r\n </div>\r\n <div class=\"room-info\">\r\n <span class=\"room-name\">{{ room.ChatRoomName }}</span>\r\n <span class=\"room-name\">{{ room.Metadata }}</span>\r\n </div>\r\n </li>\r\n </ul>\r\n </aside>\r\n <!-- RIGHT PANEL : CHAT WINDOW -->\r\n @if (showMessages) {\r\n <section *ngIf=\"showMessages\" @roomSwitch class=\"chat-main glass\">\r\n <app-chat-window [chatRoomId]=\"activeRoomId()\" [header]=\"activeRoomHeader()\" [metadata]=\"activeRoomMetadata()\" [showActions]=\"false\"\r\n mode=\"fullscreen\" theme=\"light\">\r\n </app-chat-window>\r\n </section>\r\n } @else {\r\n <section class=\"chat-main glass loading-container\">\r\n <div class=\"spinner\"></div>\r\n <p>Loading messages\u2026</p>\r\n </section>\r\n }\r\n\r\n</div>", styles: [":root{--glass-bg: rgba(255, 255, 255, .75);--glass-border: rgba(255, 255, 255, .25);--primary: #5b6dff;--text-dark: #1c1e21;--text-muted: #65676b}*{box-sizing:border-box;font-family:Segoe UI,system-ui,-apple-system,BlinkMacSystemFont}.chat-page{display:flex;height:100vh;padding:16px;gap:16px;background:linear-gradient(135deg,#e9efff,#f6f7fb)}.glass{background:var(--glass-bg);backdrop-filter:blur(14px);-webkit-backdrop-filter:blur(14px);border-radius:18px;border:1px solid var(--glass-border);box-shadow:0 20px 40px #00000014,inset 0 1px #fff6}.chat-sidebar{width:300px;display:flex;flex-direction:column;overflow:hidden}.sidebar-header{padding:18px 20px;font-size:18px;font-weight:700;color:var(--text-dark);border-bottom:1px solid rgba(0,0,0,.06)}.room-list{list-style:none;overflow-y:auto;padding:6px}.room-item{display:flex;align-items:center;gap:14px;padding:12px 14px;border-radius:14px;cursor:pointer;transition:all .25s ease;margin-bottom:6px}.room-item:hover{background:#5b6dff14}.room-item.active{background:#5b6dff26;box-shadow:inset 0 0 0 1px #5b6dff40}.room-avatar p-avatar{width:44px;height:44px;border-radius:50%;background:linear-gradient(135deg,#5b6dff,#7a89ff);color:#fff;font-weight:700;display:flex;align-items:center;justify-content:center}.room-info{display:flex;flex-direction:column;overflow:hidden}.room-name{font-size:14px;font-weight:600;color:var(--text-dark);white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.room-info span:last-child{font-size:12px;color:var(--text-muted)}.chat-main{flex:1;display:flex;flex-direction:column;overflow:hidden;position:relative}.chat-main:after{content:\"\";position:absolute;inset:0;border-radius:18px;pointer-events:none;box-shadow:inset 0 0 0 1px #fff3}.room-item:hover{transform:translateY(-1px)}.loading-container{display:flex;flex-direction:column;align-items:center;justify-content:center;height:100%}.spinner{width:40px;height:40px;border:4px solid rgba(0,0,0,.1);border-top-color:#4f46e5;border-radius:50%;animation:spin 1s linear infinite}@keyframes spin{to{transform:rotate(360deg)}}\n"] }]
118
+ }], ctorParameters: () => [], propDecorators: { newRoomLoaded: [{
119
+ type: Output
120
+ }] } });
121
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"all-chats.component.js","sourceRoot":"","sources":["../../../../../projects/chat-widget/src/lib/all-chats/all-chats.component.ts","../../../../../projects/chat-widget/src/lib/all-chats/all-chats.component.html"],"names":[],"mappings":"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","sourcesContent":["import {Component, EventEmitter, inject, input, Output, signal} from '@angular/core';\r\nimport {ChatService} from '../core/services/chat.service';\r\nimport {ChatWindowComponent} from \"../chat-window/chat-window.component\";\r\nimport {CommonModule} from '@angular/common';\r\nimport {FormsModule} from '@angular/forms';\r\nimport {AvatarModule} from 'primeng/avatar';\r\nimport {ButtonModule} from 'primeng/button';\r\nimport {ChipModule} from 'primeng/chip';\r\nimport {DropdownModule} from 'primeng/dropdown';\r\nimport {InputTextareaModule} from 'primeng/inputtextarea';\r\nimport {OverlayPanelModule} from 'primeng/overlaypanel';\r\nimport {ToastModule} from 'primeng/toast';\r\nimport {TooltipModule} from 'primeng/tooltip';\r\nimport {animate, style, transition, trigger} from '@angular/animations';\r\nimport {SocketService} from '../core/services/socket.service';\r\n\r\n@Component({\r\n    selector: 'app-all-chats',\r\n    standalone: true,\r\n    imports: [\r\n        CommonModule,\r\n        FormsModule,\r\n        ToastModule,\r\n        ButtonModule,\r\n        AvatarModule,\r\n        TooltipModule,\r\n        InputTextareaModule,\r\n        ChipModule,\r\n        DropdownModule,\r\n        OverlayPanelModule,\r\n        ChatWindowComponent\r\n    ],\r\n    templateUrl: './all-chats.component.html',\r\n    styleUrl: './all-chats.component.scss',\r\n    animations: [\r\n        trigger('roomSwitch', [\r\n            transition(':enter', [\r\n                style({opacity: 0, transform: 'translateX(24px)'}),\r\n                animate(\r\n                    '300ms cubic-bezier(.4,0,.2,1)',\r\n                    style({opacity: 1, transform: 'translateX(0)'})\r\n                )\r\n            ]),\r\n            transition(':leave', [\r\n                animate(\r\n                    '200ms ease-in',\r\n                    style({opacity: 0, transform: 'translateX(-24px)'})\r\n                )\r\n            ])\r\n        ])\r\n    ]\r\n})\r\nexport class AllChatsComponent {\r\n\r\n    user = input<any>(null);\r\n    roomId = input<any>(null);\r\n    chatUsers = signal<any[]>([]);\r\n\r\n    @Output() newRoomLoaded = new EventEmitter();\r\n\r\n    rooms: any[] = [];\r\n    activeRoomId = signal<string>('');\r\n    activeRoomHeader = signal<string>('');\r\n    activeRoomMetadata = signal<string>('');\r\n\r\n    chatService = inject(ChatService);\r\n    socketService = inject(SocketService);\r\n    showMessages: boolean = false\r\n\r\n    constructor() {\r\n    }\r\n\r\n    ngOnInit() {\r\n        if (!this.user) {\r\n            console.error('user is required');\r\n            return;\r\n        }\r\n\r\n        this.chatService.currentUser.set({\r\n            roomId: this.activeRoomId,\r\n            userId: this.user()?.id,\r\n            name: this.user()?.name\r\n        });\r\n\r\n        this.socketService.connect(this.user().id);\r\n\r\n        this.chatService.getRooms(this.user()?.id, this.roomId()).subscribe(res => {\r\n            this.rooms = res;\r\n            if (this.rooms.length != 0) {\r\n                this.selectRoom(this.rooms[0]);\r\n                this.activeRoomId.set(this.rooms[0].ChatRoomId);\r\n                this.activeRoomHeader.set(this.rooms[0].ChatRoomName);\r\n                this.activeRoomMetadata.set(this.rooms[0].Metadata);\r\n            } else {\r\n                this.chatService.addUserToRoom(\r\n                    this.roomId(),\r\n                    this.user()?.id,\r\n                    this.user()?.name,\r\n                    this.chatService.currentUser().userId).subscribe((res) => {\r\n                    this.rooms = [];\r\n                    this.getRooms();\r\n                });\r\n            }\r\n        });\r\n    }\r\n\r\n    getRooms() {\r\n        this.chatService.getRooms(this.user()?.id, this.roomId()).subscribe(res => {\r\n            this.rooms = res;\r\n            if (this.rooms.length != 0) {\r\n                this.selectRoom(this.rooms[0]);\r\n                this.activeRoomId.set(this.rooms[0].ChatRoomId);\r\n                this.activeRoomHeader.set(this.rooms[0].ChatRoomName);\r\n                this.activeRoomMetadata.set(this.rooms[0].Metadata);\r\n            }\r\n        });\r\n    }\r\n\r\n    selectRoom(room: any) {\r\n        this.showMessages = false;\r\n        this.activeRoomId.set(room.ChatRoomId);\r\n        this.activeRoomHeader.set(room.ChatRoomName);\r\n        this.activeRoomMetadata.set(room.Metadata);\r\n        setTimeout(() => {\r\n            this.showMessages = true;\r\n        }, 800);\r\n    }\r\n}\r\n","<div class=\"chat-page\">\r\n    <!-- LEFT SIDEBAR : CHAT ROOMS -->\r\n    <aside class=\"chat-sidebar glass\">\r\n        <div class=\"sidebar-header\">\r\n            <h3>Chat Rooms</h3>\r\n        </div>\r\n        <ul class=\"room-list\">\r\n            <li (click)=\"selectRoom(room)\" *ngFor=\"let room of rooms;\" [class.active]=\"activeRoomId() === room.ChatRoomId\"\r\n                class=\"room-item\">\r\n                <div class=\"room-avatar\">\r\n                    <p-avatar [label]=\"room.ChatRoomName[0]\"></p-avatar>\r\n                </div>\r\n                <div class=\"room-info\">\r\n                    <span class=\"room-name\">{{ room.ChatRoomName }}</span>\r\n                    <span class=\"room-name\">{{ room.Metadata }}</span>\r\n                </div>\r\n            </li>\r\n        </ul>\r\n    </aside>\r\n    <!-- RIGHT PANEL : CHAT WINDOW -->\r\n    @if (showMessages) {\r\n        <section *ngIf=\"showMessages\" @roomSwitch class=\"chat-main glass\">\r\n            <app-chat-window [chatRoomId]=\"activeRoomId()\" [header]=\"activeRoomHeader()\" [metadata]=\"activeRoomMetadata()\" [showActions]=\"false\"\r\n                             mode=\"fullscreen\" theme=\"light\">\r\n            </app-chat-window>\r\n        </section>\r\n    } @else {\r\n        <section class=\"chat-main glass loading-container\">\r\n            <div class=\"spinner\"></div>\r\n            <p>Loading messages…</p>\r\n        </section>\r\n    }\r\n\r\n</div>"]}
@@ -0,0 +1,96 @@
1
+ import { Component, effect, inject, input, signal, ViewEncapsulation } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ChatWindowComponent } from '../chat-window/chat-window.component';
4
+ import { ChatService } from '../core/services/chat.service';
5
+ import { DropdownModule } from 'primeng/dropdown';
6
+ import { ButtonModule } from 'primeng/button';
7
+ import { ToastModule } from 'primeng/toast';
8
+ import { MessageService } from 'primeng/api';
9
+ import { FormsModule } from '@angular/forms';
10
+ import { TabViewModule } from 'primeng/tabview';
11
+ import { SocketService } from '../core/services/socket.service';
12
+ import { TooltipModule } from "primeng/tooltip";
13
+ import * as i0 from "@angular/core";
14
+ import * as i1 from "@angular/common";
15
+ import * as i2 from "primeng/api";
16
+ import * as i3 from "primeng/button";
17
+ import * as i4 from "primeng/tabview";
18
+ import * as i5 from "primeng/tooltip";
19
+ export class ChatLauncherComponent {
20
+ constructor() {
21
+ this.roomId = input('');
22
+ this.roomHeader = input('');
23
+ this.metadata = input('');
24
+ this.user = input(null);
25
+ this.userList = input([]);
26
+ this.chatUsers = input([]);
27
+ this.chatService = inject(ChatService);
28
+ this.socketService = inject(SocketService);
29
+ this.messageService = inject(MessageService);
30
+ this.openWindows = signal([]);
31
+ this.lastOpenedRoom = signal(null);
32
+ this.activeIndex = signal(0);
33
+ this.isVisible = signal(true);
34
+ effect(() => {
35
+ const roomId = this.roomId();
36
+ const header = this.roomHeader();
37
+ const metadata = this.metadata();
38
+ const user = this.user();
39
+ this.chatService.currentUser.set({
40
+ roomId: this.roomId(),
41
+ userId: user?.userId,
42
+ name: user?.name
43
+ });
44
+ this.chatService.userList.set(this.userList());
45
+ if (!roomId || !user?.userId)
46
+ return;
47
+ if (this.lastOpenedRoom() === roomId)
48
+ return;
49
+ this.lastOpenedRoom.set(roomId);
50
+ this.openNewRoom(roomId, header, metadata, user.userId, user.name, this.chatUsers());
51
+ }, { allowSignalWrites: true });
52
+ }
53
+ ngOnInit() {
54
+ console.log('ChatLauncher initialized');
55
+ }
56
+ openNewRoom(roomId, roomHeader, metadata, userId, username, chatUsers) {
57
+ this.isVisible.set(true);
58
+ this.chatService.createRoom(roomId, roomHeader, metadata, userId, username, chatUsers).subscribe(room => {
59
+ this.addWindow(room.roomKey, room.name, room.metadata);
60
+ });
61
+ }
62
+ addWindow(roomId, header, metadata) {
63
+ const windows = this.openWindows();
64
+ const existingIndex = windows.findIndex(w => w.roomId === roomId);
65
+ // already open → focus
66
+ if (existingIndex !== -1) {
67
+ this.activeIndex.set(existingIndex);
68
+ return;
69
+ }
70
+ const newIndex = windows.length;
71
+ this.openWindows.update(ws => [...ws, { roomId, header, metadata }]);
72
+ // ✅ force real change: -1 → 0
73
+ setTimeout(() => {
74
+ this.activeIndex.set(newIndex);
75
+ }, 500);
76
+ }
77
+ closeWindow(roomId) {
78
+ this.openWindows.update(ws => ws.filter(w => w.roomId !== roomId));
79
+ }
80
+ setActive(i) {
81
+ this.activeIndex.set(i);
82
+ }
83
+ closeAll() {
84
+ this.openWindows.set([]);
85
+ // this.lastOpenedRoom.set(null);
86
+ // this.activeIndex.set(0);
87
+ this.isVisible.set(false);
88
+ }
89
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatLauncherComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
90
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: ChatLauncherComponent, isStandalone: true, selector: "app-chat-launcher", inputs: { roomId: { classPropertyName: "roomId", publicName: "roomId", isSignal: true, isRequired: false, transformFunction: null }, roomHeader: { classPropertyName: "roomHeader", publicName: "roomHeader", isSignal: true, isRequired: false, transformFunction: null }, metadata: { classPropertyName: "metadata", publicName: "metadata", isSignal: true, isRequired: false, transformFunction: null }, user: { classPropertyName: "user", publicName: "user", isSignal: true, isRequired: false, transformFunction: null }, userList: { classPropertyName: "userList", publicName: "userList", isSignal: true, isRequired: false, transformFunction: null }, chatUsers: { classPropertyName: "chatUsers", publicName: "chatUsers", isSignal: true, isRequired: false, transformFunction: null } }, providers: [MessageService], ngImport: i0, template: "@if (isVisible()) {\r\n <div class=\"chat-float-wrapper\">\r\n <button (click)=\"closeAll()\" icon=\"pi pi-times\" pButton pTooltip=\"Close\"\r\n style=\"font-size: 0.5rem\" class=\"closeButton\"></button>\r\n <p-tabView (activeIndexChange)=\"setActive($event)\" [activeIndex]=\"activeIndex()\" [scrollable]=\"true\"\r\n class=\"chat-tabview\">\r\n <p-tabPanel *ngFor=\"let w of openWindows(); let i = index\" [header]=\"w.header\">\r\n <ng-template pTemplate=\"content\">\r\n <app-chat-window (onArchive)=\"closeWindow($event)\"\r\n [chatRoomId]=\"w.roomId\" [header]=\"w.header\"\r\n [metadata]=\"w.metadata\" mode=\"floating\" theme=\"light\">\r\n </app-chat-window>\r\n </ng-template>\r\n </p-tabPanel>\r\n </p-tabView>\r\n </div>\r\n}\r\n", styles: ["@charset \"UTF-8\";.chat-float-wrapper{position:fixed!important;bottom:16px!important;right:16px!important;z-index:99999;pointer-events:auto}:host{position:fixed;right:16px;bottom:16px;z-index:2147483647;font-family:Arial,sans-serif}:host,:host *{box-sizing:border-box}:host{isolation:isolate}.chat-tabview{width:100%;height:100%}:host ::ng-deep .p-tabview{height:100%;display:flex;flex-direction:column}:host ::ng-deep .p-tabview-nav{flex-shrink:0}:host ::ng-deep .p-tabview-panels{flex:1;overflow:hidden}:host ::ng-deep .p-tabview-panel{height:100%}:host ::ng-deep app-chat-window{display:block;height:100%}.chat-tabview ::ng-deep .p-tabview-nav{background:transparent;border:none;display:flex;gap:8px;padding:6px 4px}.chat-tabview ::ng-deep .p-tabview-nav li{border:none}.chat-tabview ::ng-deep .p-tabview-nav li .p-tabview-nav-link{border:none;border-radius:18px;padding:6px 14px;font-size:13px;font-weight:500;color:#555;box-shadow:0 1px 4px #00000014;transition:all .2s ease;background:#d3d3d3}.chat-tabview ::ng-deep .p-tabview-nav li .p-tabview-nav-link:hover{color:#1a73e8}.chat-tabview ::ng-deep .p-tabview-nav li.p-highlight .p-tabview-nav-link{background:#1a73e8;color:#fff;box-shadow:0 4px 10px #1a73e859;border-bottom:outset;border-color:#1a73e8}.chat-tabview ::ng-deep .p-tabview-ink-bar{display:none}.chat-tabview ::ng-deep .p-tabview-panels{padding:0;border:none;background:transparent}.closeButton{position:absolute;right:0;top:25px;border-radius:50%;border:solid 2px}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "component", type: ChatWindowComponent, selector: "app-chat-window", inputs: ["chatRoomId", "header", "metadata", "chatUsers", "showActions", "mode", "theme", "refreshRoom"], outputs: ["onArchive"] }, { kind: "ngmodule", type: DropdownModule }, { kind: "directive", type: i2.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "ngmodule", type: ButtonModule }, { kind: "directive", type: i3.ButtonDirective, selector: "[pButton]", inputs: ["iconPos", "loadingIcon", "label", "icon", "loading", "severity", "raised", "rounded", "text", "outlined", "size", "plain"] }, { kind: "ngmodule", type: ToastModule }, { kind: "ngmodule", type: FormsModule }, { kind: "ngmodule", type: TabViewModule }, { kind: "component", type: i4.TabView, selector: "p-tabView", inputs: ["style", "styleClass", "controlClose", "scrollable", "activeIndex", "selectOnFocus", "nextButtonAriaLabel", "prevButtonAriaLabel", "autoHideButtons", "tabindex"], outputs: ["onChange", "onClose", "activeIndexChange"] }, { kind: "component", type: i4.TabPanel, selector: "p-tabPanel", inputs: ["closable", "headerStyle", "headerStyleClass", "cache", "tooltip", "tooltipPosition", "tooltipPositionStyle", "tooltipStyleClass", "selected", "disabled", "header", "leftIcon", "rightIcon"] }, { kind: "ngmodule", type: TooltipModule }, { kind: "directive", type: i5.Tooltip, selector: "[pTooltip]", inputs: ["tooltipPosition", "tooltipEvent", "appendTo", "positionStyle", "tooltipStyleClass", "tooltipZIndex", "escape", "showDelay", "hideDelay", "life", "positionTop", "positionLeft", "autoHide", "fitContent", "hideOnEscape", "pTooltip", "tooltipDisabled", "tooltipOptions"] }] }); }
91
+ }
92
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatLauncherComponent, decorators: [{
93
+ type: Component,
94
+ args: [{ selector: 'app-chat-launcher', standalone: true, imports: [CommonModule, ChatWindowComponent, DropdownModule, ButtonModule, ToastModule, FormsModule, TabViewModule, TooltipModule], providers: [MessageService], encapsulation: ViewEncapsulation.Emulated, template: "@if (isVisible()) {\r\n <div class=\"chat-float-wrapper\">\r\n <button (click)=\"closeAll()\" icon=\"pi pi-times\" pButton pTooltip=\"Close\"\r\n style=\"font-size: 0.5rem\" class=\"closeButton\"></button>\r\n <p-tabView (activeIndexChange)=\"setActive($event)\" [activeIndex]=\"activeIndex()\" [scrollable]=\"true\"\r\n class=\"chat-tabview\">\r\n <p-tabPanel *ngFor=\"let w of openWindows(); let i = index\" [header]=\"w.header\">\r\n <ng-template pTemplate=\"content\">\r\n <app-chat-window (onArchive)=\"closeWindow($event)\"\r\n [chatRoomId]=\"w.roomId\" [header]=\"w.header\"\r\n [metadata]=\"w.metadata\" mode=\"floating\" theme=\"light\">\r\n </app-chat-window>\r\n </ng-template>\r\n </p-tabPanel>\r\n </p-tabView>\r\n </div>\r\n}\r\n", styles: ["@charset \"UTF-8\";.chat-float-wrapper{position:fixed!important;bottom:16px!important;right:16px!important;z-index:99999;pointer-events:auto}:host{position:fixed;right:16px;bottom:16px;z-index:2147483647;font-family:Arial,sans-serif}:host,:host *{box-sizing:border-box}:host{isolation:isolate}.chat-tabview{width:100%;height:100%}:host ::ng-deep .p-tabview{height:100%;display:flex;flex-direction:column}:host ::ng-deep .p-tabview-nav{flex-shrink:0}:host ::ng-deep .p-tabview-panels{flex:1;overflow:hidden}:host ::ng-deep .p-tabview-panel{height:100%}:host ::ng-deep app-chat-window{display:block;height:100%}.chat-tabview ::ng-deep .p-tabview-nav{background:transparent;border:none;display:flex;gap:8px;padding:6px 4px}.chat-tabview ::ng-deep .p-tabview-nav li{border:none}.chat-tabview ::ng-deep .p-tabview-nav li .p-tabview-nav-link{border:none;border-radius:18px;padding:6px 14px;font-size:13px;font-weight:500;color:#555;box-shadow:0 1px 4px #00000014;transition:all .2s ease;background:#d3d3d3}.chat-tabview ::ng-deep .p-tabview-nav li .p-tabview-nav-link:hover{color:#1a73e8}.chat-tabview ::ng-deep .p-tabview-nav li.p-highlight .p-tabview-nav-link{background:#1a73e8;color:#fff;box-shadow:0 4px 10px #1a73e859;border-bottom:outset;border-color:#1a73e8}.chat-tabview ::ng-deep .p-tabview-ink-bar{display:none}.chat-tabview ::ng-deep .p-tabview-panels{padding:0;border:none;background:transparent}.closeButton{position:absolute;right:0;top:25px;border-radius:50%;border:solid 2px}\n"] }]
95
+ }], ctorParameters: () => [] });
96
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-launcher.component.js","sourceRoot":"","sources":["../../../../../projects/chat-widget/src/lib/chat-launcher/chat-launcher.component.ts","../../../../../projects/chat-widget/src/lib/chat-launcher/chat-launcher.component.html"],"names":[],"mappings":"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","sourcesContent":["import {Component, effect, inject, input, signal, ViewEncapsulation} from '@angular/core';\r\nimport {CommonModule} from '@angular/common';\r\nimport {ChatWindowComponent} from '../chat-window/chat-window.component';\r\nimport {ChatService} from '../core/services/chat.service';\r\nimport {DropdownModule} from 'primeng/dropdown';\r\nimport {ButtonModule} from 'primeng/button';\r\nimport {ToastModule} from 'primeng/toast';\r\nimport {MessageService} from 'primeng/api';\r\nimport {FormsModule} from '@angular/forms';\r\nimport {TabViewModule} from 'primeng/tabview';\r\nimport {SocketService} from '../core/services/socket.service';\r\nimport {TooltipModule} from \"primeng/tooltip\";\r\n\r\ninterface OpenChatWindow {\r\n    roomId: string;\r\n    header: string;\r\n    metadata: string;\r\n}\r\n\r\n@Component({\r\n    selector: 'app-chat-launcher',\r\n    standalone: true,\r\n    imports: [CommonModule, ChatWindowComponent, DropdownModule, ButtonModule, ToastModule, FormsModule, TabViewModule, TooltipModule],\r\n    templateUrl: './chat-launcher.component.html',\r\n    styleUrls: ['./chat-launcher.component.scss'],\r\n    providers: [MessageService],\r\n    encapsulation: ViewEncapsulation.Emulated,\r\n})\r\nexport class ChatLauncherComponent {\r\n\r\n    roomId = input<string>('');\r\n    roomHeader = input<string>('');\r\n    metadata = input<string>('');\r\n    user = input<any>(null);\r\n    userList = input<any[]>([]);\r\n    chatUsers = input<any[]>([]);\r\n\r\n    chatService = inject(ChatService);\r\n    socketService = inject(SocketService);\r\n    messageService = inject(MessageService);\r\n    openWindows = signal<OpenChatWindow[]>([]);\r\n    lastOpenedRoom = signal<string | null>(null);\r\n    activeIndex = signal(0);\r\n    isVisible = signal(true);\r\n\r\n    constructor() {\r\n        effect(\r\n            () => {\r\n                const roomId = this.roomId();\r\n                const header = this.roomHeader();\r\n                const metadata = this.metadata();\r\n                const user = this.user();\r\n\r\n                this.chatService.currentUser.set({\r\n                    roomId: this.roomId(),\r\n                    userId: user?.userId,\r\n                    name: user?.name\r\n                });\r\n\r\n                this.chatService.userList.set(this.userList());\r\n\r\n                if (!roomId || !user?.userId) return;\r\n                if (this.lastOpenedRoom() === roomId) return;\r\n\r\n                this.lastOpenedRoom.set(roomId);\r\n                this.openNewRoom(roomId, header, metadata, user.userId, user.name, this.chatUsers());\r\n            },\r\n            {allowSignalWrites: true}\r\n        );\r\n    }\r\n\r\n    ngOnInit() {\r\n        console.log('ChatLauncher initialized');\r\n    }\r\n\r\n    openNewRoom(roomId: string, roomHeader: string, metadata: string, userId: string, username: string, chatUsers: any) {\r\n        this.isVisible.set(true);\r\n        this.chatService.createRoom(roomId, roomHeader, metadata, userId, username, chatUsers).subscribe(room => {\r\n            this.addWindow(room.roomKey, room.name, room.metadata);\r\n        });\r\n    }\r\n\r\n    addWindow(roomId: string, header: string, metadata: string) {\r\n        const windows = this.openWindows();\r\n        const existingIndex = windows.findIndex(w => w.roomId === roomId);\r\n\r\n        // already open → focus\r\n        if (existingIndex !== -1) {\r\n            this.activeIndex.set(existingIndex);\r\n            return;\r\n        }\r\n\r\n        const newIndex = windows.length;\r\n\r\n        this.openWindows.update(ws => [...ws, {roomId, header, metadata}]);\r\n\r\n        // ✅ force real change: -1 → 0\r\n        setTimeout(() => {\r\n            this.activeIndex.set(newIndex);\r\n        }, 500);\r\n    }\r\n\r\n\r\n    closeWindow(roomId: any) {\r\n        this.openWindows.update(ws => ws.filter(w => w.roomId !== roomId));\r\n    }\r\n\r\n    setActive(i: number) {\r\n        this.activeIndex.set(i);\r\n    }\r\n\r\n    closeAll() {\r\n        this.openWindows.set([]);\r\n        // this.lastOpenedRoom.set(null);\r\n        // this.activeIndex.set(0);\r\n        this.isVisible.set(false);\r\n    }\r\n\r\n}\r\n","@if (isVisible()) {\r\n    <div class=\"chat-float-wrapper\">\r\n        <button (click)=\"closeAll()\" icon=\"pi pi-times\" pButton pTooltip=\"Close\"\r\n                style=\"font-size: 0.5rem\" class=\"closeButton\"></button>\r\n        <p-tabView (activeIndexChange)=\"setActive($event)\" [activeIndex]=\"activeIndex()\" [scrollable]=\"true\"\r\n                   class=\"chat-tabview\">\r\n            <p-tabPanel *ngFor=\"let w of openWindows(); let i = index\" [header]=\"w.header\">\r\n                <ng-template pTemplate=\"content\">\r\n                    <app-chat-window (onArchive)=\"closeWindow($event)\"\r\n                                     [chatRoomId]=\"w.roomId\" [header]=\"w.header\"\r\n                                     [metadata]=\"w.metadata\" mode=\"floating\" theme=\"light\">\r\n                    </app-chat-window>\r\n                </ng-template>\r\n            </p-tabPanel>\r\n        </p-tabView>\r\n    </div>\r\n}\r\n"]}
@@ -0,0 +1,69 @@
1
+ import { Component, computed, EventEmitter, inject, input, Output } from '@angular/core';
2
+ import { ChatService } from "../core/services/chat.service";
3
+ import { CommonModule } from "@angular/common";
4
+ import { FormsModule } from "@angular/forms";
5
+ import { SocketService } from "../core/services/socket.service";
6
+ import { ChatLauncherComponent } from "../chat-launcher/chat-launcher.component";
7
+ import * as i0 from "@angular/core";
8
+ export class ChatWidgetNotificationComponent {
9
+ constructor() {
10
+ this.open = false;
11
+ this.chatService = inject(ChatService);
12
+ this.socketService = inject(SocketService);
13
+ this.userId = input(null);
14
+ this.userName = input(null);
15
+ this.chatLauncher = inject(ChatLauncherComponent);
16
+ this.chatRoomDetails = new EventEmitter();
17
+ this.totalCount = computed(() => this.socketService.notifications().reduce((a, b) => a + b.unreadCount, 0));
18
+ }
19
+ ngOnInit() {
20
+ this.socketService.connect(this.userId());
21
+ this.chatService.getAllUserNotifications(this.userId()).subscribe(notifications => {
22
+ const data = notifications.map(n => {
23
+ return {
24
+ entityId: n.room_id,
25
+ eventType: n.type,
26
+ payload: { chatRoomId: n.room_id, chatRoomName: n.chat_room_name },
27
+ unreadCount: 0,
28
+ title: n.description
29
+ };
30
+ });
31
+ this.socketService.notifications.set(data);
32
+ this.socketService.notifyArrival();
33
+ });
34
+ }
35
+ toggle() {
36
+ this.open = !this.open;
37
+ }
38
+ openRoom(n) {
39
+ this.chatLauncher.isVisible.set(true);
40
+ this.chatService.markAsRead(n.entityId, n.eventType, this.userId()).subscribe(notification => {
41
+ console.log('markAsRead', notification);
42
+ setTimeout(() => {
43
+ this.chatService.getRoomDetails(n.payload.chatRoomId).subscribe(room => {
44
+ const users = room[0].Users
45
+ ? JSON.parse(room[0].Users).map((u) => ({
46
+ user_id: u.user_id,
47
+ full_name: u.full_name
48
+ }))
49
+ : [];
50
+ this.chatRoomDetails.emit({ roomId: room[0].Id, roomName: room[0].Name, metadata: room[0].Metadata, users });
51
+ this.socketService.clearRoom(n);
52
+ });
53
+ }, 800);
54
+ });
55
+ this.open = false;
56
+ }
57
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetNotificationComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
58
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.3.12", type: ChatWidgetNotificationComponent, isStandalone: true, selector: "chat-widget-notification", inputs: { userId: { classPropertyName: "userId", publicName: "userId", isSignal: true, isRequired: false, transformFunction: null }, userName: { classPropertyName: "userName", publicName: "userName", isSignal: true, isRequired: false, transformFunction: null } }, outputs: { chatRoomDetails: "chatRoomDetails" }, providers: [ChatLauncherComponent], ngImport: i0, template: "<div (click)=\"toggle()\" class=\"notif-icon\">\r\n <i class=\"pi pi-bell\" [class.pulse]=\"socketService.hasNewNotification()\"></i>\r\n\r\n @if (totalCount() > 0) {\r\n <span class=\"dot\"></span>\r\n <span class=\"badge\">{{ totalCount() }}</span>\r\n }\r\n</div>\r\n\r\n@if (open) {\r\n <div class=\"notif-panel\">\r\n <div class=\"notif-list\">\r\n @for (n of socketService.notifications(); track n.entityId + '_' + n.eventType) {\r\n <div class=\"notif-item\" (click)=\"openRoom(n)\">\r\n <div class=\"notif-left\">\r\n <i class=\"pi pi-comments\"></i>\r\n </div>\r\n\r\n <div class=\"notif-body\">\r\n <div class=\"notif-title\">{{ n?.payload?.chatRoomName }}</div>\r\n <div class=\"notif-text\">\r\n @if (n.eventType === 'NEW_MESSAGE') {\r\n New Message Received\r\n }\r\n\r\n @if (n.eventType === 'USER_ADDED') {\r\n You have been added to the chat\r\n }\r\n </div>\r\n </div>\r\n\r\n<!-- <span class=\"count\">{{ n.count }}</span>-->\r\n </div>\r\n }\r\n </div>\r\n </div>\r\n}\r\n", styles: [".notif-icon{position:relative;cursor:pointer;display:flex;align-items:center;justify-content:center}.notif-icon i{font-size:1.4rem;color:#444}.pulse{animation:pulse 1.2s ease-out}@keyframes pulse{0%{transform:scale(1)}40%{transform:scale(1.2)}to{transform:scale(1)}}.dot{position:absolute;top:2px;right:4px;width:8px;height:8px;background:#e53935;border-radius:50%}.badge{position:absolute;top:-6px;right:-10px;background:#e53935;color:#fff;font-size:.65rem;min-width:18px;height:18px;padding:0 5px;border-radius:10px;display:flex;align-items:center;justify-content:center;font-weight:600}.notif-panel{position:absolute;top:42px;right:0;width:320px;background:#fff;border-radius:8px;box-shadow:0 12px 28px #00000026;z-index:9999;overflow:hidden;animation:fadeIn .15s ease-out}@keyframes fadeIn{0%{opacity:0;transform:translateY(-4px)}to{opacity:1;transform:translateY(0)}}.notif-header{padding:12px 16px;font-weight:600;font-size:.9rem;border-bottom:1px solid #eee}.notif-list{max-height:360px;overflow-y:auto}.notif-item{display:flex;align-items:center;padding:12px 14px;cursor:pointer;transition:background .15s ease}.notif-item:hover{background:#f5f7fa}.notif-left{width:36px;height:36px;border-radius:50%;background:#e3f2fd;color:#1976d2;display:flex;align-items:center;justify-content:center;margin-right:12px}.notif-body{flex:1;min-width:0;display:flex;gap:5px;flex-direction:column}.notif-title{font-size:.85rem;font-weight:600;color:#222;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.notif-text{font-size:.75rem;color:#666;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.count{background:#1976d2;color:#fff;font-size:.65rem;min-width:20px;height:20px;border-radius:10px;display:flex;align-items:center;justify-content:center;font-weight:600}.notif-item{background:#f0f6ff}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: FormsModule }] }); }
59
+ }
60
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetNotificationComponent, decorators: [{
61
+ type: Component,
62
+ args: [{ selector: 'chat-widget-notification', standalone: true, imports: [
63
+ CommonModule,
64
+ FormsModule
65
+ ], providers: [ChatLauncherComponent], template: "<div (click)=\"toggle()\" class=\"notif-icon\">\r\n <i class=\"pi pi-bell\" [class.pulse]=\"socketService.hasNewNotification()\"></i>\r\n\r\n @if (totalCount() > 0) {\r\n <span class=\"dot\"></span>\r\n <span class=\"badge\">{{ totalCount() }}</span>\r\n }\r\n</div>\r\n\r\n@if (open) {\r\n <div class=\"notif-panel\">\r\n <div class=\"notif-list\">\r\n @for (n of socketService.notifications(); track n.entityId + '_' + n.eventType) {\r\n <div class=\"notif-item\" (click)=\"openRoom(n)\">\r\n <div class=\"notif-left\">\r\n <i class=\"pi pi-comments\"></i>\r\n </div>\r\n\r\n <div class=\"notif-body\">\r\n <div class=\"notif-title\">{{ n?.payload?.chatRoomName }}</div>\r\n <div class=\"notif-text\">\r\n @if (n.eventType === 'NEW_MESSAGE') {\r\n New Message Received\r\n }\r\n\r\n @if (n.eventType === 'USER_ADDED') {\r\n You have been added to the chat\r\n }\r\n </div>\r\n </div>\r\n\r\n<!-- <span class=\"count\">{{ n.count }}</span>-->\r\n </div>\r\n }\r\n </div>\r\n </div>\r\n}\r\n", styles: [".notif-icon{position:relative;cursor:pointer;display:flex;align-items:center;justify-content:center}.notif-icon i{font-size:1.4rem;color:#444}.pulse{animation:pulse 1.2s ease-out}@keyframes pulse{0%{transform:scale(1)}40%{transform:scale(1.2)}to{transform:scale(1)}}.dot{position:absolute;top:2px;right:4px;width:8px;height:8px;background:#e53935;border-radius:50%}.badge{position:absolute;top:-6px;right:-10px;background:#e53935;color:#fff;font-size:.65rem;min-width:18px;height:18px;padding:0 5px;border-radius:10px;display:flex;align-items:center;justify-content:center;font-weight:600}.notif-panel{position:absolute;top:42px;right:0;width:320px;background:#fff;border-radius:8px;box-shadow:0 12px 28px #00000026;z-index:9999;overflow:hidden;animation:fadeIn .15s ease-out}@keyframes fadeIn{0%{opacity:0;transform:translateY(-4px)}to{opacity:1;transform:translateY(0)}}.notif-header{padding:12px 16px;font-weight:600;font-size:.9rem;border-bottom:1px solid #eee}.notif-list{max-height:360px;overflow-y:auto}.notif-item{display:flex;align-items:center;padding:12px 14px;cursor:pointer;transition:background .15s ease}.notif-item:hover{background:#f5f7fa}.notif-left{width:36px;height:36px;border-radius:50%;background:#e3f2fd;color:#1976d2;display:flex;align-items:center;justify-content:center;margin-right:12px}.notif-body{flex:1;min-width:0;display:flex;gap:5px;flex-direction:column}.notif-title{font-size:.85rem;font-weight:600;color:#222;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.notif-text{font-size:.75rem;color:#666;white-space:nowrap;overflow:hidden;text-overflow:ellipsis}.count{background:#1976d2;color:#fff;font-size:.65rem;min-width:20px;height:20px;border-radius:10px;display:flex;align-items:center;justify-content:center;font-weight:600}.notif-item{background:#f0f6ff}\n"] }]
66
+ }], ctorParameters: () => [], propDecorators: { chatRoomDetails: [{
67
+ type: Output
68
+ }] } });
69
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"chat-widget-notification.component.js","sourceRoot":"","sources":["../../../../../projects/chat-widget/src/lib/chat-widget-notification/chat-widget-notification.component.ts","../../../../../projects/chat-widget/src/lib/chat-widget-notification/chat-widget-notification.component.html"],"names":[],"mappings":"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","sourcesContent":["import {Component, computed, EventEmitter, inject, input, Output} from '@angular/core';\r\nimport {ChatService} from \"../core/services/chat.service\";\r\nimport {CommonModule} from \"@angular/common\";\r\nimport {FormsModule} from \"@angular/forms\";\r\nimport {ChatNotification} from \"../core/models/notification.model\";\r\nimport {SocketService} from \"../core/services/socket.service\";\r\nimport {ChatLauncherComponent} from \"../chat-launcher/chat-launcher.component\";\r\n\r\n@Component({\r\n    selector: 'chat-widget-notification',\r\n    standalone: true,\r\n    imports: [\r\n        CommonModule,\r\n        FormsModule\r\n    ],\r\n    templateUrl: './chat-widget-notification.component.html',\r\n    styleUrl: './chat-widget-notification.component.css',\r\n    providers: [ChatLauncherComponent]\r\n})\r\nexport class ChatWidgetNotificationComponent {\r\n    open = false;\r\n\r\n    chatService = inject(ChatService);\r\n    socketService = inject(SocketService);\r\n    userId = input<any>(null);\r\n    userName = input<any>(null);\r\n    chatLauncher = inject(ChatLauncherComponent);\r\n    @Output() chatRoomDetails = new EventEmitter();\r\n\r\n    totalCount = computed(() =>\r\n        this.socketService.notifications().reduce((a: any, b: { unreadCount: any; }) => a + b.unreadCount, 0)\r\n    );\r\n\r\n    constructor() {}\r\n\r\n    ngOnInit() {\r\n        this.socketService.connect(this.userId());\r\n        this.chatService.getAllUserNotifications(this.userId()).subscribe(notifications => {\r\n            const data = notifications.map(n => {\r\n                return {\r\n                    entityId: n.room_id,\r\n                    eventType: n.type,\r\n                    payload: {chatRoomId: n.room_id, chatRoomName: n.chat_room_name},\r\n                    unreadCount: 0,\r\n                    title: n.description\r\n                }\r\n            });\r\n            this.socketService.notifications.set(data);\r\n            this.socketService.notifyArrival();\r\n        });\r\n    }\r\n\r\n    toggle() {\r\n        this.open = !this.open;\r\n    }\r\n\r\n    openRoom(n: ChatNotification) {\r\n        this.chatLauncher.isVisible.set(true);\r\n        this.chatService.markAsRead(n.entityId, n.eventType, this.userId()).subscribe(notification => {\r\n            console.log('markAsRead', notification);\r\n            setTimeout(() => {\r\n                this.chatService.getRoomDetails(n.payload.chatRoomId).subscribe(room => {\r\n                    const users = room[0].Users\r\n                        ? JSON.parse(room[0].Users).map((u: any) => ({\r\n                            user_id: u.user_id,\r\n                            full_name: u.full_name\r\n                        }))\r\n                        : [];\r\n                    this.chatRoomDetails.emit({roomId: room[0].Id, roomName: room[0].Name, metadata: room[0].Metadata, users});\r\n                    this.socketService.clearRoom(n);\r\n                })\r\n            }, 800)\r\n        });\r\n        this.open = false;\r\n    }\r\n}\r\n","<div (click)=\"toggle()\" class=\"notif-icon\">\r\n    <i class=\"pi pi-bell\" [class.pulse]=\"socketService.hasNewNotification()\"></i>\r\n\r\n    @if (totalCount() > 0) {\r\n        <span class=\"dot\"></span>\r\n        <span class=\"badge\">{{ totalCount() }}</span>\r\n    }\r\n</div>\r\n\r\n@if (open) {\r\n    <div class=\"notif-panel\">\r\n        <div class=\"notif-list\">\r\n            @for (n of socketService.notifications(); track n.entityId + '_' + n.eventType) {\r\n                <div class=\"notif-item\" (click)=\"openRoom(n)\">\r\n                    <div class=\"notif-left\">\r\n                        <i class=\"pi pi-comments\"></i>\r\n                    </div>\r\n\r\n                    <div class=\"notif-body\">\r\n                        <div class=\"notif-title\">{{ n?.payload?.chatRoomName }}</div>\r\n                        <div class=\"notif-text\">\r\n                            @if (n.eventType === 'NEW_MESSAGE') {\r\n                                New Message Received\r\n                            }\r\n\r\n                            @if (n.eventType === 'USER_ADDED') {\r\n                                You have been added to the chat\r\n                            }\r\n                        </div>\r\n                    </div>\r\n\r\n<!--                    <span class=\"count\">{{ n.count }}</span>-->\r\n                </div>\r\n            }\r\n        </div>\r\n    </div>\r\n}\r\n"]}
@@ -0,0 +1,47 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { ChatLauncherComponent } from './chat-launcher/chat-launcher.component';
4
+ import { ChatWindowComponent } from './chat-window/chat-window.component';
5
+ import { AllChatsComponent } from './all-chats/all-chats.component';
6
+ import { CHAT_CONFIG } from './core/config/chat.tokens';
7
+ import { ChatWidgetNotificationComponent } from "./chat-widget-notification/chat-widget-notification.component";
8
+ import * as i0 from "@angular/core";
9
+ export class ChatWidgetModule {
10
+ static forRoot(config) {
11
+ return {
12
+ ngModule: ChatWidgetModule,
13
+ providers: [
14
+ {
15
+ provide: CHAT_CONFIG,
16
+ useValue: config
17
+ }
18
+ ]
19
+ };
20
+ }
21
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
22
+ static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetModule, imports: [CommonModule,
23
+ ChatLauncherComponent,
24
+ ChatWindowComponent,
25
+ AllChatsComponent,
26
+ ChatWidgetNotificationComponent], exports: [ChatLauncherComponent] }); }
27
+ static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetModule, imports: [CommonModule,
28
+ ChatLauncherComponent,
29
+ ChatWindowComponent,
30
+ AllChatsComponent,
31
+ ChatWidgetNotificationComponent] }); }
32
+ }
33
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetModule, decorators: [{
34
+ type: NgModule,
35
+ args: [{
36
+ declarations: [],
37
+ imports: [
38
+ CommonModule,
39
+ ChatLauncherComponent,
40
+ ChatWindowComponent,
41
+ AllChatsComponent,
42
+ ChatWidgetNotificationComponent
43
+ ],
44
+ exports: [ChatLauncherComponent]
45
+ }]
46
+ }] });
47
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC13aWRnZXQubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvY2hhdC13aWRnZXQvc3JjL2xpYi9jaGF0LXdpZGdldC5tb2R1bGUudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFzQixRQUFRLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFDNUQsT0FBTyxFQUFDLFlBQVksRUFBQyxNQUFNLGlCQUFpQixDQUFDO0FBQzdDLE9BQU8sRUFBQyxxQkFBcUIsRUFBQyxNQUFNLHlDQUF5QyxDQUFDO0FBQzlFLE9BQU8sRUFBQyxtQkFBbUIsRUFBQyxNQUFNLHFDQUFxQyxDQUFDO0FBQ3hFLE9BQU8sRUFBQyxpQkFBaUIsRUFBQyxNQUFNLGlDQUFpQyxDQUFDO0FBQ2xFLE9BQU8sRUFBQyxXQUFXLEVBQUMsTUFBTSwyQkFBMkIsQ0FBQztBQUV0RCxPQUFPLEVBQUMsK0JBQStCLEVBQUMsTUFBTSwrREFBK0QsQ0FBQzs7QUFhOUcsTUFBTSxPQUFPLGdCQUFnQjtJQUN6QixNQUFNLENBQUMsT0FBTyxDQUFDLE1BQWtCO1FBQzdCLE9BQU87WUFDSCxRQUFRLEVBQUUsZ0JBQWdCO1lBQzFCLFNBQVMsRUFBRTtnQkFDUDtvQkFDSSxPQUFPLEVBQUUsV0FBVztvQkFDcEIsUUFBUSxFQUFFLE1BQU07aUJBQ25CO2FBQ0o7U0FDSixDQUFDO0lBQ04sQ0FBQzsrR0FYUSxnQkFBZ0I7Z0hBQWhCLGdCQUFnQixZQVJyQixZQUFZO1lBQ1oscUJBQXFCO1lBQ3JCLG1CQUFtQjtZQUNuQixpQkFBaUI7WUFDakIsK0JBQStCLGFBRXpCLHFCQUFxQjtnSEFFdEIsZ0JBQWdCLFlBUnJCLFlBQVk7WUFDWixxQkFBcUI7WUFDckIsbUJBQW1CO1lBQ25CLGlCQUFpQjtZQUNqQiwrQkFBK0I7OzRGQUkxQixnQkFBZ0I7a0JBWDVCLFFBQVE7bUJBQUM7b0JBQ04sWUFBWSxFQUFFLEVBQUU7b0JBQ2hCLE9BQU8sRUFBRTt3QkFDTCxZQUFZO3dCQUNaLHFCQUFxQjt3QkFDckIsbUJBQW1CO3dCQUNuQixpQkFBaUI7d0JBQ2pCLCtCQUErQjtxQkFDbEM7b0JBQ0QsT0FBTyxFQUFFLENBQUMscUJBQXFCLENBQUM7aUJBQ25DIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtNb2R1bGVXaXRoUHJvdmlkZXJzLCBOZ01vZHVsZX0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XHJcbmltcG9ydCB7Q29tbW9uTW9kdWxlfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xyXG5pbXBvcnQge0NoYXRMYXVuY2hlckNvbXBvbmVudH0gZnJvbSAnLi9jaGF0LWxhdW5jaGVyL2NoYXQtbGF1bmNoZXIuY29tcG9uZW50JztcclxuaW1wb3J0IHtDaGF0V2luZG93Q29tcG9uZW50fSBmcm9tICcuL2NoYXQtd2luZG93L2NoYXQtd2luZG93LmNvbXBvbmVudCc7XHJcbmltcG9ydCB7QWxsQ2hhdHNDb21wb25lbnR9IGZyb20gJy4vYWxsLWNoYXRzL2FsbC1jaGF0cy5jb21wb25lbnQnO1xyXG5pbXBvcnQge0NIQVRfQ09ORklHfSBmcm9tICcuL2NvcmUvY29uZmlnL2NoYXQudG9rZW5zJztcclxuaW1wb3J0IHR5cGUge0NoYXRDb25maWd9IGZyb20gJy4vY29yZS9jb25maWcvY2hhdC1jb25maWcnO1xyXG5pbXBvcnQge0NoYXRXaWRnZXROb3RpZmljYXRpb25Db21wb25lbnR9IGZyb20gXCIuL2NoYXQtd2lkZ2V0LW5vdGlmaWNhdGlvbi9jaGF0LXdpZGdldC1ub3RpZmljYXRpb24uY29tcG9uZW50XCI7XHJcblxyXG5ATmdNb2R1bGUoe1xyXG4gICAgZGVjbGFyYXRpb25zOiBbXSxcclxuICAgIGltcG9ydHM6IFtcclxuICAgICAgICBDb21tb25Nb2R1bGUsXHJcbiAgICAgICAgQ2hhdExhdW5jaGVyQ29tcG9uZW50LFxyXG4gICAgICAgIENoYXRXaW5kb3dDb21wb25lbnQsXHJcbiAgICAgICAgQWxsQ2hhdHNDb21wb25lbnQsXHJcbiAgICAgICAgQ2hhdFdpZGdldE5vdGlmaWNhdGlvbkNvbXBvbmVudFxyXG4gICAgXSxcclxuICAgIGV4cG9ydHM6IFtDaGF0TGF1bmNoZXJDb21wb25lbnRdXHJcbn0pXHJcbmV4cG9ydCBjbGFzcyBDaGF0V2lkZ2V0TW9kdWxlIHtcclxuICAgIHN0YXRpYyBmb3JSb290KGNvbmZpZzogQ2hhdENvbmZpZyk6IE1vZHVsZVdpdGhQcm92aWRlcnM8Q2hhdFdpZGdldE1vZHVsZT4ge1xyXG4gICAgICAgIHJldHVybiB7XHJcbiAgICAgICAgICAgIG5nTW9kdWxlOiBDaGF0V2lkZ2V0TW9kdWxlLFxyXG4gICAgICAgICAgICBwcm92aWRlcnM6IFtcclxuICAgICAgICAgICAgICAgIHtcclxuICAgICAgICAgICAgICAgICAgICBwcm92aWRlOiBDSEFUX0NPTkZJRyxcclxuICAgICAgICAgICAgICAgICAgICB1c2VWYWx1ZTogY29uZmlnXHJcbiAgICAgICAgICAgICAgICB9XHJcbiAgICAgICAgICAgIF1cclxuICAgICAgICB9O1xyXG4gICAgfVxyXG59XHJcbiJdfQ==
@@ -0,0 +1,12 @@
1
+ import { Injectable } from "@angular/core";
2
+ import * as i0 from "@angular/core";
3
+ export class ChatWidgetService {
4
+ constructor() { }
5
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
6
+ static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetService, providedIn: 'root' }); }
7
+ }
8
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ChatWidgetService, decorators: [{
9
+ type: Injectable,
10
+ args: [{ providedIn: 'root' }]
11
+ }], ctorParameters: () => [] });
12
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hhdC13aWRnZXQuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL2NoYXQtd2lkZ2V0L3NyYy9saWIvY2hhdC13aWRnZXQuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQVMsVUFBVSxFQUFDLE1BQU0sZUFBZSxDQUFDOztBQU1qRCxNQUFNLE9BQU8saUJBQWlCO0lBRTFCLGdCQUFlLENBQUM7K0dBRlAsaUJBQWlCO21IQUFqQixpQkFBaUIsY0FESixNQUFNOzs0RkFDbkIsaUJBQWlCO2tCQUQ3QixVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7aW5qZWN0LCBJbmplY3RhYmxlfSBmcm9tIFwiQGFuZ3VsYXIvY29yZVwiO1xyXG5pbXBvcnQge0NoYXRTZXJ2aWNlfSBmcm9tIFwiLi9jb3JlL3NlcnZpY2VzL2NoYXQuc2VydmljZVwiO1xyXG5pbXBvcnQge1NvY2tldFNlcnZpY2V9IGZyb20gXCIuL2NvcmUvc2VydmljZXMvc29ja2V0LnNlcnZpY2VcIjtcclxuaW1wb3J0IHtDaGF0TWVzc2FnZX0gZnJvbSBcIi4vY29yZS9tb2RlbHMvY2hhdC5tb2RlbHNcIjtcclxuXHJcbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogJ3Jvb3QnIH0pXHJcbmV4cG9ydCBjbGFzcyBDaGF0V2lkZ2V0U2VydmljZSB7XHJcblxyXG4gICAgY29uc3RydWN0b3IoKSB7fVxyXG59XHJcbiJdfQ==