@muraai/mnl-chat 0.0.1-alpha-82b7eb9 → 0.0.1-alpha-51e7ded

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 (56) hide show
  1. package/{esm2020 → esm2022}/lib/chat.module.mjs +19 -19
  2. package/esm2022/lib/components/ng-chat/ng-chat.component.mjs +588 -0
  3. package/esm2022/lib/components/ng-chat-friends-list/ng-chat-friends-list.component.mjs +128 -0
  4. package/esm2022/lib/components/ng-chat-options/ng-chat-options.component.mjs +29 -0
  5. package/esm2022/lib/components/ng-chat-window/ng-chat-window.component.mjs +247 -0
  6. package/esm2022/lib/core/message-counter.mjs +25 -0
  7. package/{esm2020 → esm2022}/lib/firstLetter.pipe.mjs +5 -5
  8. package/esm2022/lib/material.module.mjs +60 -0
  9. package/{esm2020 → esm2022}/lib/pipes/emojify.pipe.mjs +4 -4
  10. package/esm2022/lib/pipes/group-message-display-name.pipe.mjs +24 -0
  11. package/esm2022/lib/pipes/linkfy.pipe.mjs +34 -0
  12. package/{esm2020 → esm2022}/lib/pipes/sanitize.pipe.mjs +5 -5
  13. package/{esm2020 → esm2022}/lib/services/chat.service.mjs +5 -5
  14. package/{fesm2020 → fesm2022}/muraai-mnl-chat.mjs +85 -85
  15. package/fesm2022/muraai-mnl-chat.mjs.map +1 -0
  16. package/lib/components/ng-chat/ng-chat.component.d.ts +1 -1
  17. package/lib/components/ng-chat-friends-list/ng-chat-friends-list.component.d.ts +1 -1
  18. package/lib/components/ng-chat-options/ng-chat-options.component.d.ts +1 -1
  19. package/lib/components/ng-chat-window/ng-chat-window.component.d.ts +1 -1
  20. package/lib/material.module.d.ts +2 -2
  21. package/package.json +7 -13
  22. package/esm2020/lib/components/ng-chat/ng-chat.component.mjs +0 -588
  23. package/esm2020/lib/components/ng-chat-friends-list/ng-chat-friends-list.component.mjs +0 -128
  24. package/esm2020/lib/components/ng-chat-options/ng-chat-options.component.mjs +0 -29
  25. package/esm2020/lib/components/ng-chat-window/ng-chat-window.component.mjs +0 -247
  26. package/esm2020/lib/core/message-counter.mjs +0 -25
  27. package/esm2020/lib/material.module.mjs +0 -60
  28. package/esm2020/lib/pipes/group-message-display-name.pipe.mjs +0 -24
  29. package/esm2020/lib/pipes/linkfy.pipe.mjs +0 -34
  30. package/fesm2015/muraai-mnl-chat.mjs +0 -1436
  31. package/fesm2015/muraai-mnl-chat.mjs.map +0 -1
  32. package/fesm2020/muraai-mnl-chat.mjs.map +0 -1
  33. /package/{esm2020 → esm2022}/lib/core/chat-adapter.mjs +0 -0
  34. /package/{esm2020 → esm2022}/lib/core/chat-controller.mjs +0 -0
  35. /package/{esm2020 → esm2022}/lib/core/chat-group-adapter.mjs +0 -0
  36. /package/{esm2020 → esm2022}/lib/core/chat-option.mjs +0 -0
  37. /package/{esm2020 → esm2022}/lib/core/chat-participant-status-descriptor.mjs +0 -0
  38. /package/{esm2020 → esm2022}/lib/core/chat-participant-status.enum.mjs +0 -0
  39. /package/{esm2020 → esm2022}/lib/core/chat-participant-type.enum.mjs +0 -0
  40. /package/{esm2020 → esm2022}/lib/core/chat-participant.mjs +0 -0
  41. /package/{esm2020 → esm2022}/lib/core/default-file-upload-adapter.mjs +0 -0
  42. /package/{esm2020 → esm2022}/lib/core/file-upload-adapter.mjs +0 -0
  43. /package/{esm2020 → esm2022}/lib/core/group.mjs +0 -0
  44. /package/{esm2020 → esm2022}/lib/core/guid.mjs +0 -0
  45. /package/{esm2020 → esm2022}/lib/core/localization.mjs +0 -0
  46. /package/{esm2020 → esm2022}/lib/core/message-type.enum.mjs +0 -0
  47. /package/{esm2020 → esm2022}/lib/core/message.mjs +0 -0
  48. /package/{esm2020 → esm2022}/lib/core/paged-history-chat-adapter.mjs +0 -0
  49. /package/{esm2020 → esm2022}/lib/core/participant-metadata.mjs +0 -0
  50. /package/{esm2020 → esm2022}/lib/core/participant-response.mjs +0 -0
  51. /package/{esm2020 → esm2022}/lib/core/scroll-direction.enum.mjs +0 -0
  52. /package/{esm2020 → esm2022}/lib/core/theme.enum.mjs +0 -0
  53. /package/{esm2020 → esm2022}/lib/core/user.mjs +0 -0
  54. /package/{esm2020 → esm2022}/lib/core/window.mjs +0 -0
  55. /package/{esm2020 → esm2022}/muraai-mnl-chat.mjs +0 -0
  56. /package/{esm2020 → esm2022}/public-api.mjs +0 -0
@@ -0,0 +1,588 @@
1
+ /* eslint-disable */
2
+ import { Component, Input, ViewChildren, HostListener, Output, EventEmitter, ViewEncapsulation } from '@angular/core';
3
+ import { MessageType } from "../../core/message-type.enum";
4
+ import { Window } from "../../core/window";
5
+ import { ChatParticipantStatus } from "../../core/chat-participant-status.enum";
6
+ import { ScrollDirection } from "../../core/scroll-direction.enum";
7
+ import { PagedHistoryChatAdapter } from '../../core/paged-history-chat-adapter';
8
+ import { DefaultFileUploadAdapter } from '../../core/default-file-upload-adapter';
9
+ import { Theme } from '../../core/theme.enum';
10
+ import { Group } from "../../core/group";
11
+ import { ChatParticipantType } from "../../core/chat-participant-type.enum";
12
+ import { map } from 'rxjs/operators';
13
+ import * as i0 from "@angular/core";
14
+ import * as i1 from "@angular/common/http";
15
+ import * as i2 from "@angular/common";
16
+ import * as i3 from "../ng-chat-friends-list/ng-chat-friends-list.component";
17
+ import * as i4 from "../ng-chat-window/ng-chat-window.component";
18
+ import * as i5 from "../../pipes/sanitize.pipe";
19
+ export class NgChat {
20
+ constructor(_httpClient) {
21
+ this._httpClient = _httpClient;
22
+ // Exposes enums for the ng-template
23
+ this.ChatParticipantType = ChatParticipantType;
24
+ this.ChatParticipantStatus = ChatParticipantStatus;
25
+ this.MessageType = MessageType;
26
+ this._isDisabled = false;
27
+ this.isCollapsed = false;
28
+ this.maximizeWindowOnNewMessage = true;
29
+ this.pollFriendsList = false;
30
+ this.pollingInterval = 5000;
31
+ this.historyEnabled = true;
32
+ this.emojisEnabled = true;
33
+ this.linkfyEnabled = true;
34
+ this.audioEnabled = true;
35
+ this.searchEnabled = true;
36
+ this.audioSource = '../../../assets/notification.wav';
37
+ this.persistWindowsState = true;
38
+ this.title = "Quick chat";
39
+ this.messagePlaceholder = "Type a message";
40
+ this.searchPlaceholder = "Search";
41
+ this.browserNotificationsEnabled = true;
42
+ this.browserNotificationIconSource = '../../../assets/notification.png';
43
+ this.browserNotificationTitle = "New message from";
44
+ this.historyPageSize = 10;
45
+ this.hideFriendsList = false;
46
+ this.hideFriendsListOnUnsupportedViewport = true;
47
+ this.theme = Theme.Light;
48
+ this.messageDatePipeFormat = "short";
49
+ this.showMessageDate = true;
50
+ this.isViewportOnMobileEnabled = false;
51
+ this.onParticipantClicked = new EventEmitter();
52
+ this.onParticipantChatOpened = new EventEmitter();
53
+ this.onParticipantChatClosed = new EventEmitter();
54
+ this.onMessagesSeen = new EventEmitter();
55
+ this.onCloseWindowClicked = new EventEmitter();
56
+ this.browserNotificationsBootstrapped = false;
57
+ this.hasPagedHistory = false;
58
+ // Don't want to add this as a setting to simplify usage. Previous placeholder and title settings available to be used, or use full Localization object.
59
+ this.statusDescription = {
60
+ online: 'Online',
61
+ busy: 'Busy',
62
+ away: 'Away',
63
+ offline: 'Offline'
64
+ };
65
+ this.participantsInteractedWith = [];
66
+ // Defines the size of each opened window to calculate how many windows can be opened on the viewport at the same time.
67
+ this.windowSizeFactor = 320;
68
+ // Total width size of the friends list section
69
+ this.friendsListWidth = 262;
70
+ // Set to true if there is no space to display at least one chat window and 'hideFriendsListOnUnsupportedViewport' is true
71
+ this.unsupportedViewport = false;
72
+ this.windows = [];
73
+ this.isBootstrapped = false;
74
+ }
75
+ get isDisabled() {
76
+ return this._isDisabled;
77
+ }
78
+ set isDisabled(value) {
79
+ this._isDisabled = value;
80
+ if (value) {
81
+ window.clearInterval(this.pollingIntervalWindowInstance);
82
+ }
83
+ else {
84
+ this.activateFriendListFetch();
85
+ }
86
+ }
87
+ get localStorageKey() {
88
+ return `ng-chat-users-${this.userId}`; // Appending the user id so the state is unique per user in a computer.
89
+ }
90
+ ;
91
+ ngOnInit() {
92
+ this.bootstrapChat();
93
+ }
94
+ onResize(event) {
95
+ this.viewPortTotalArea = event.target.innerWidth;
96
+ this.NormalizeWindows();
97
+ }
98
+ // Checks if there are more opened windows than the view port can display
99
+ NormalizeWindows() {
100
+ const maxSupportedOpenedWindows = Math.floor((this.viewPortTotalArea - (!this.hideFriendsList ? this.friendsListWidth : 0)) / this.windowSizeFactor);
101
+ const difference = this.windows.length - maxSupportedOpenedWindows;
102
+ if (difference >= 0) {
103
+ this.windows.splice(this.windows.length - difference);
104
+ }
105
+ this.updateWindowsState(this.windows);
106
+ // Viewport should have space for at least one chat window but should show in mobile if option is enabled.
107
+ this.unsupportedViewport = this.isViewportOnMobileEnabled ? false : this.hideFriendsListOnUnsupportedViewport && maxSupportedOpenedWindows < 1;
108
+ }
109
+ // Initializes the chat plugin and the messaging adapter
110
+ bootstrapChat() {
111
+ let initializationException = null;
112
+ if (this.adapter != null && this.userId != null) {
113
+ try {
114
+ this.viewPortTotalArea = window.innerWidth;
115
+ this.initializeTheme();
116
+ this.initializeDefaultText();
117
+ this.initializeBrowserNotifications();
118
+ // Binding event listeners
119
+ this.adapter.messageReceivedHandler = (participant, msg) => this.onMessageReceived(participant, msg);
120
+ this.adapter.friendsListChangedHandler = (participantsResponse) => this.onFriendsListChanged(participantsResponse);
121
+ this.activateFriendListFetch();
122
+ this.bufferAudioFile();
123
+ this.hasPagedHistory = this.adapter instanceof PagedHistoryChatAdapter;
124
+ if (this.fileUploadUrl && this.fileUploadUrl !== "") {
125
+ this.fileUploadAdapter = new DefaultFileUploadAdapter(this.fileUploadUrl, this._httpClient);
126
+ }
127
+ this.NormalizeWindows();
128
+ this.isBootstrapped = true;
129
+ }
130
+ catch (ex) {
131
+ initializationException = ex;
132
+ }
133
+ }
134
+ if (!this.isBootstrapped) {
135
+ console.error("ng-chat component couldn't be bootstrapped.");
136
+ if (this.userId == null) {
137
+ console.error("ng-chat can't be initialized without an user id. Please make sure you've provided an userId as a parameter of the ng-chat component.");
138
+ }
139
+ if (this.adapter == null) {
140
+ console.error("ng-chat can't be bootstrapped without a ChatAdapter. Please make sure you've provided a ChatAdapter implementation as a parameter of the ng-chat component.");
141
+ }
142
+ if (initializationException) {
143
+ console.error(`An exception has occurred while initializing ng-chat. Details: ${initializationException.message}`);
144
+ console.error(initializationException);
145
+ }
146
+ }
147
+ }
148
+ activateFriendListFetch() {
149
+ if (this.adapter) {
150
+ // Loading current users list
151
+ if (this.pollFriendsList) {
152
+ // Setting a long poll interval to update the friends list
153
+ this.fetchFriendsList(true);
154
+ this.pollingIntervalWindowInstance = window.setInterval(() => this.fetchFriendsList(false), this.pollingInterval);
155
+ }
156
+ else {
157
+ // Since polling was disabled, a friends list update mechanism will have to be implemented in the ChatAdapter.
158
+ this.fetchFriendsList(true);
159
+ }
160
+ }
161
+ }
162
+ // Initializes browser notifications
163
+ async initializeBrowserNotifications() {
164
+ if (this.browserNotificationsEnabled && ("Notification" in window)) {
165
+ if (await Notification.requestPermission() === "granted") {
166
+ this.browserNotificationsBootstrapped = true;
167
+ }
168
+ }
169
+ }
170
+ // Initializes default text
171
+ initializeDefaultText() {
172
+ if (!this.localization) {
173
+ this.localization = {
174
+ messagePlaceholder: this.messagePlaceholder,
175
+ searchPlaceholder: this.searchPlaceholder,
176
+ title: this.title,
177
+ statusDescription: this.statusDescription,
178
+ browserNotificationTitle: this.browserNotificationTitle,
179
+ loadMessageHistoryPlaceholder: "Load older messages"
180
+ };
181
+ }
182
+ }
183
+ initializeTheme() {
184
+ if (this.customTheme) {
185
+ this.theme = Theme.Custom;
186
+ }
187
+ else if (this.theme != Theme.Light && this.theme != Theme.Dark) {
188
+ // TODO: Use es2017 in future with Object.values(Theme).includes(this.theme) to do this check
189
+ throw new Error(`Invalid theme configuration for ng-chat. "${this.theme}" is not a valid theme value.`);
190
+ }
191
+ }
192
+ // Sends a request to load the friends list
193
+ fetchFriendsList(isBootstrapping) {
194
+ this.adapter.listFriends()
195
+ .pipe(map((participantsResponse) => {
196
+ this.participantsResponse = participantsResponse;
197
+ this.participants = participantsResponse.map((response) => {
198
+ return response.participant;
199
+ });
200
+ })).subscribe(() => {
201
+ if (isBootstrapping) {
202
+ this.restoreWindowsState();
203
+ }
204
+ });
205
+ }
206
+ fetchMessageHistory(window) {
207
+ // Not ideal but will keep this until we decide if we are shipping pagination with the default adapter
208
+ if (this.adapter instanceof PagedHistoryChatAdapter) {
209
+ window.isLoadingHistory = true;
210
+ this.adapter.getMessageHistoryByPage(window.participant.id, this.historyPageSize, ++window.historyPage)
211
+ .pipe(map((result) => {
212
+ result.forEach((message) => this.assertMessageType(message));
213
+ window.messages = result.concat(window.messages);
214
+ window.isLoadingHistory = false;
215
+ const direction = (window.historyPage == 1) ? ScrollDirection.Bottom : ScrollDirection.Top;
216
+ window.hasMoreMessages = result.length == this.historyPageSize;
217
+ setTimeout(() => this.onFetchMessageHistoryLoaded(result, window, direction, true));
218
+ })).subscribe();
219
+ }
220
+ else {
221
+ this.adapter.getMessageHistory(window.participant.id)
222
+ .pipe(map((result) => {
223
+ result.forEach((message) => this.assertMessageType(message));
224
+ window.messages = result.concat(window.messages);
225
+ window.isLoadingHistory = false;
226
+ setTimeout(() => this.onFetchMessageHistoryLoaded(result, window, ScrollDirection.Bottom));
227
+ })).subscribe();
228
+ }
229
+ }
230
+ onFetchMessageHistoryLoaded(messages, window, direction, forceMarkMessagesAsSeen = false) {
231
+ this.scrollChatWindow(window, direction);
232
+ if (window.hasFocus || forceMarkMessagesAsSeen) {
233
+ const unseenMessages = messages.filter(m => !m.dateSeen);
234
+ this.markMessagesAsRead(unseenMessages);
235
+ }
236
+ }
237
+ // Updates the friends list via the event handler
238
+ onFriendsListChanged(participantsResponse) {
239
+ if (participantsResponse) {
240
+ this.participantsResponse = participantsResponse;
241
+ this.participants = participantsResponse.map((response) => {
242
+ return response.participant;
243
+ });
244
+ this.participantsInteractedWith = [];
245
+ }
246
+ }
247
+ // Handles received messages by the adapter
248
+ onMessageReceived(participant, message) {
249
+ if (participant && message) {
250
+ const chatWindow = this.openChatWindow(participant);
251
+ this.assertMessageType(message);
252
+ if (!chatWindow[1] || !this.historyEnabled) {
253
+ chatWindow[0].messages.push(message);
254
+ this.scrollChatWindow(chatWindow[0], ScrollDirection.Bottom);
255
+ if (chatWindow[0].hasFocus) {
256
+ this.markMessagesAsRead([message]);
257
+ }
258
+ }
259
+ this.emitMessageSound(chatWindow[0]);
260
+ // Do not push browser notifications with message content for privacy purposes if the 'maximizeWindowOnNewMessage' setting is off and this is a new chat window.
261
+ if (this.maximizeWindowOnNewMessage || (!chatWindow[1] && !chatWindow[0].isCollapsed)) {
262
+ // Some messages are not pushed because they are loaded by fetching the history hence why we supply the message here
263
+ this.emitBrowserNotification(chatWindow[0], message);
264
+ }
265
+ }
266
+ }
267
+ onParticipantClickedFromFriendsList(participant) {
268
+ this.openChatWindow(participant, true, true);
269
+ }
270
+ cancelOptionPrompt() {
271
+ if (this.currentActiveOption) {
272
+ this.currentActiveOption.isActive = false;
273
+ this.currentActiveOption = null;
274
+ }
275
+ }
276
+ onOptionPromptCanceled() {
277
+ this.cancelOptionPrompt();
278
+ }
279
+ onOptionPromptConfirmed(event) {
280
+ // For now this is fine as there is only one option available. Introduce option types and type checking if a new option is added.
281
+ this.confirmNewGroup(event);
282
+ // Canceling current state
283
+ this.cancelOptionPrompt();
284
+ }
285
+ confirmNewGroup(users) {
286
+ const newGroup = new Group(users);
287
+ this.openChatWindow(newGroup);
288
+ if (this.groupAdapter) {
289
+ this.groupAdapter.groupCreated(newGroup);
290
+ }
291
+ }
292
+ // Opens a new chat whindow. Takes care of available viewport
293
+ // Works for opening a chat window for an user or for a group
294
+ // Returns => [Window: Window object reference, boolean: Indicates if this window is a new chat window]
295
+ openChatWindow(participant, focusOnNewWindow = false, invokedByUserClick = false) {
296
+ // Is this window opened?
297
+ const openedWindow = this.windows.find(x => x.participant.id == participant.id);
298
+ if (!openedWindow) {
299
+ if (invokedByUserClick) {
300
+ this.onParticipantClicked.emit(participant);
301
+ }
302
+ // Refer to issue #58 on Github
303
+ const collapseWindow = invokedByUserClick ? false : !this.maximizeWindowOnNewMessage;
304
+ const newChatWindow = new Window(participant, this.historyEnabled, collapseWindow);
305
+ // Loads the chat history via an RxJs Observable
306
+ if (this.historyEnabled) {
307
+ this.fetchMessageHistory(newChatWindow);
308
+ }
309
+ this.windows.unshift(newChatWindow);
310
+ // Is there enough space left in the view port ? but should be displayed in mobile if option is enabled
311
+ if (!this.isViewportOnMobileEnabled) {
312
+ if (this.windows.length * this.windowSizeFactor >= this.viewPortTotalArea - (!this.hideFriendsList ? this.friendsListWidth : 0)) {
313
+ this.windows.pop();
314
+ }
315
+ }
316
+ this.updateWindowsState(this.windows);
317
+ if (focusOnNewWindow && !collapseWindow) {
318
+ this.focusOnWindow(newChatWindow);
319
+ }
320
+ this.participantsInteractedWith.push(participant);
321
+ this.onParticipantChatOpened.emit(participant);
322
+ return [newChatWindow, true];
323
+ }
324
+ else {
325
+ // Returns the existing chat window
326
+ return [openedWindow, false];
327
+ }
328
+ }
329
+ // Focus on the input element of the supplied window
330
+ focusOnWindow(window, callback = () => { }) {
331
+ const windowIndex = this.windows.indexOf(window);
332
+ if (windowIndex >= 0) {
333
+ setTimeout(() => {
334
+ if (this.chatWindows) {
335
+ const chatWindowToFocus = this.chatWindows.toArray()[windowIndex];
336
+ chatWindowToFocus.chatWindowInput.nativeElement.focus();
337
+ }
338
+ callback();
339
+ });
340
+ }
341
+ }
342
+ assertMessageType(message) {
343
+ // Always fallback to "Text" messages to avoid rendenring issues
344
+ if (!message.type) {
345
+ message.type = MessageType.Text;
346
+ }
347
+ }
348
+ // Marks all messages provided as read with the current time.
349
+ markMessagesAsRead(messages) {
350
+ const currentDate = new Date();
351
+ messages.forEach((msg) => {
352
+ msg.dateSeen = currentDate;
353
+ });
354
+ this.onMessagesSeen.emit(messages);
355
+ }
356
+ // Buffers audio file (For component's bootstrapping)
357
+ bufferAudioFile() {
358
+ if (this.audioSource && this.audioSource.length > 0) {
359
+ this.audioFile = new Audio();
360
+ this.audioFile.src = this.audioSource;
361
+ this.audioFile.load();
362
+ }
363
+ }
364
+ // Emits a message notification audio if enabled after every message received
365
+ emitMessageSound(window) {
366
+ if (this.audioEnabled && !window.hasFocus && this.audioFile) {
367
+ this.audioFile.play();
368
+ }
369
+ }
370
+ // Emits a browser notification
371
+ emitBrowserNotification(window, message) {
372
+ // if (this.browserNotificationsBootstrapped && !window.hasFocus && message) {
373
+ if (this.browserNotificationsBootstrapped && message) {
374
+ const notification = new Notification(`${this.localization.browserNotificationTitle} ${window.participant.displayName}`, {
375
+ 'body': message.message,
376
+ 'icon': this.browserNotificationIconSource
377
+ });
378
+ setTimeout(() => {
379
+ notification.close();
380
+ }, message.message.length <= 50 ? 5000 : 7000); // More time to read longer messages
381
+ }
382
+ }
383
+ // Saves current windows state into local storage if persistence is enabled
384
+ updateWindowsState(windows) {
385
+ if (this.persistWindowsState) {
386
+ const participantIds = windows.map((w) => {
387
+ return w.participant.id;
388
+ });
389
+ localStorage.setItem(this.localStorageKey, JSON.stringify(participantIds));
390
+ }
391
+ }
392
+ restoreWindowsState() {
393
+ try {
394
+ if (this.persistWindowsState) {
395
+ const stringfiedParticipantIds = localStorage.getItem(this.localStorageKey);
396
+ if (stringfiedParticipantIds && stringfiedParticipantIds.length > 0) {
397
+ const participantIds = JSON.parse(stringfiedParticipantIds);
398
+ const participantsToRestore = this.participants.filter(u => participantIds.indexOf(u.id) >= 0);
399
+ participantsToRestore.forEach((participant) => {
400
+ this.openChatWindow(participant);
401
+ });
402
+ }
403
+ }
404
+ }
405
+ catch (ex) {
406
+ console.error(`An error occurred while restoring ng-chat windows state. Details: ${ex}`);
407
+ }
408
+ }
409
+ // Gets closest open window if any. Most recent opened has priority (Right)
410
+ getClosestWindow(window) {
411
+ const index = this.windows.indexOf(window);
412
+ if (index > 0) {
413
+ return this.windows[index - 1];
414
+ }
415
+ else if (index == 0 && this.windows.length > 1) {
416
+ return this.windows[index + 1];
417
+ }
418
+ }
419
+ closeWindow(window) {
420
+ const index = this.windows.indexOf(window);
421
+ this.windows.splice(index, 1);
422
+ this.updateWindowsState(this.windows);
423
+ this.onParticipantChatClosed.emit(window.participant);
424
+ }
425
+ getChatWindowComponentInstance(targetWindow) {
426
+ const windowIndex = this.windows.indexOf(targetWindow);
427
+ if (this.chatWindows) {
428
+ let targetWindow = this.chatWindows.toArray()[windowIndex];
429
+ return targetWindow;
430
+ }
431
+ return null;
432
+ }
433
+ // Scrolls a chat window message flow to the bottom
434
+ scrollChatWindow(window, direction) {
435
+ const chatWindow = this.getChatWindowComponentInstance(window);
436
+ if (chatWindow) {
437
+ chatWindow.scrollChatWindow(window, direction);
438
+ }
439
+ }
440
+ onWindowMessagesSeen(messagesSeen) {
441
+ this.markMessagesAsRead(messagesSeen);
442
+ }
443
+ onWindowChatClosed(payload) {
444
+ const { closedWindow, closedViaEscapeKey } = payload;
445
+ if (closedViaEscapeKey) {
446
+ let closestWindow = this.getClosestWindow(closedWindow);
447
+ if (closestWindow) {
448
+ this.focusOnWindow(closestWindow, () => { this.closeWindow(closedWindow); });
449
+ }
450
+ else {
451
+ this.closeWindow(closedWindow);
452
+ }
453
+ }
454
+ else {
455
+ this.closeWindow(closedWindow);
456
+ }
457
+ }
458
+ onWindowTabTriggered(payload) {
459
+ const { triggeringWindow, shiftKeyPressed } = payload;
460
+ const currentWindowIndex = this.windows.indexOf(triggeringWindow);
461
+ let windowToFocus = this.windows[currentWindowIndex + (shiftKeyPressed ? 1 : -1)]; // Goes back on shift + tab
462
+ if (!windowToFocus) {
463
+ // Edge windows, go to start or end
464
+ windowToFocus = this.windows[currentWindowIndex > 0 ? 0 : this.chatWindows.length - 1];
465
+ }
466
+ this.focusOnWindow(windowToFocus);
467
+ }
468
+ onWindowMessageSent(messageSent) {
469
+ this.adapter.sendMessage(messageSent);
470
+ }
471
+ onWindowOptionTriggered(option) {
472
+ this.currentActiveOption = option;
473
+ }
474
+ triggerOpenChatWindow(user) {
475
+ if (user) {
476
+ this.openChatWindow(user);
477
+ }
478
+ }
479
+ triggerCloseChatWindow(userId) {
480
+ const openedWindow = this.windows.find(x => x.participant.id == userId);
481
+ if (openedWindow) {
482
+ this.closeWindow(openedWindow);
483
+ }
484
+ }
485
+ triggerToggleChatWindowVisibility(userId) {
486
+ const openedWindow = this.windows.find(x => x.participant.id == userId);
487
+ if (openedWindow) {
488
+ const chatWindow = this.getChatWindowComponentInstance(openedWindow);
489
+ if (chatWindow) {
490
+ chatWindow.onChatWindowClicked(openedWindow);
491
+ }
492
+ }
493
+ }
494
+ // closeFriendList(){
495
+ // this.hideFriendsList = !this.hideFriendsList
496
+ // this.hideFriendsListChange.emit(this.hideFriendsList)
497
+ // }
498
+ onCloseWindowClick(click) {
499
+ this.onCloseWindowClicked.emit(click);
500
+ }
501
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.1.0", ngImport: i0, type: NgChat, deps: [{ token: i1.HttpClient }], target: i0.ɵɵFactoryTarget.Component }); }
502
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.1.0", type: NgChat, selector: "mnl-chat", inputs: { isDisabled: "isDisabled", adapter: "adapter", groupAdapter: "groupAdapter", userId: "userId", isCollapsed: "isCollapsed", maximizeWindowOnNewMessage: "maximizeWindowOnNewMessage", pollFriendsList: "pollFriendsList", pollingInterval: "pollingInterval", historyEnabled: "historyEnabled", emojisEnabled: "emojisEnabled", linkfyEnabled: "linkfyEnabled", audioEnabled: "audioEnabled", searchEnabled: "searchEnabled", audioSource: "audioSource", persistWindowsState: "persistWindowsState", title: "title", messagePlaceholder: "messagePlaceholder", searchPlaceholder: "searchPlaceholder", browserNotificationsEnabled: "browserNotificationsEnabled", browserNotificationIconSource: "browserNotificationIconSource", browserNotificationTitle: "browserNotificationTitle", historyPageSize: "historyPageSize", localization: "localization", hideFriendsList: "hideFriendsList", hideFriendsListOnUnsupportedViewport: "hideFriendsListOnUnsupportedViewport", fileUploadUrl: "fileUploadUrl", theme: "theme", customTheme: "customTheme", messageDatePipeFormat: "messageDatePipeFormat", showMessageDate: "showMessageDate", isViewportOnMobileEnabled: "isViewportOnMobileEnabled", fileUploadAdapter: "fileUploadAdapter" }, outputs: { onParticipantClicked: "onParticipantClicked", onParticipantChatOpened: "onParticipantChatOpened", onParticipantChatClosed: "onParticipantChatClosed", onMessagesSeen: "onMessagesSeen", onCloseWindowClicked: "onCloseWindowClicked" }, host: { listeners: { "window:resize": "onResize($event)" } }, viewQueries: [{ propertyName: "chatWindows", predicate: ["chatWindow"], descendants: true }], ngImport: i0, template: "<link *ngIf=\"customTheme\" rel=\"stylesheet\" [href]='customTheme | sanitize'>\n\n<div id=\"ng-chat\" *ngIf=\"!isDisabled && isBootstrapped && !unsupportedViewport\" [ngClass]=\"theme\">\n <ng-chat-friends-list\n [localization]=\"localization\"\n [shouldDisplay]=\"!hideFriendsList\"\n [userId]=\"userId\"\n [isCollapsed]=\"isCollapsed\"\n [searchEnabled]=\"searchEnabled\"\n [participants]=\"participants\"\n [participantsResponse]=\"participantsResponse\"\n [participantsInteractedWith]=\"participantsInteractedWith\"\n [windows]=\"windows\"\n [currentActiveOption]=\"currentActiveOption\"\n (onParticipantClicked)=\"onParticipantClickedFromFriendsList($event)\"\n (onOptionPromptCanceled)=\"onOptionPromptCanceled()\"\n (onOptionPromptConfirmed)=\"onOptionPromptConfirmed($event)\"\n (onCloseWindowClick)=\"onCloseWindowClick($event)\"\n >\n </ng-chat-friends-list>\n\n <div *ngFor=\"let window of windows; let i = index\" [ngClass]=\"{'ng-chat-window': true, 'primary-outline-color': true, 'ng-chat-window-collapsed': window.isCollapsed}\" [ngStyle]=\"{'right': (!hideFriendsList ? friendsListWidth : 0) + 20 + windowSizeFactor * i + 'px'}\">\n <ng-chat-window\n #chatWindow\n [fileUploadAdapter]=\"fileUploadAdapter\"\n [localization]=\"localization\"\n [userId]=\"userId\"\n [window]=\"window\"\n [showOptions]=\"groupAdapter\"\n [emojisEnabled]=\"emojisEnabled\"\n [linkfyEnabled]=\"linkfyEnabled\"\n [showMessageDate]=\"showMessageDate\"\n [messageDatePipeFormat]=\"messageDatePipeFormat\"\n [hasPagedHistory]=\"hasPagedHistory\"\n (onMessagesSeen)=\"onWindowMessagesSeen($event)\"\n (onMessageSent)=\"onWindowMessageSent($event)\"\n (onTabTriggered)=\"onWindowTabTriggered($event)\"\n (onChatWindowClosed)=\"onWindowChatClosed($event)\"\n (onOptionTriggered)=\"onWindowOptionTriggered($event)\"\n (onLoadHistoryTriggered)=\"fetchMessageHistory($event)\"\n >\n </ng-chat-window>\n </div>\n</div>\n", styles: [".user-icon{box-sizing:border-box;background-color:#fff;border:2px solid;width:32px;height:20px;border-radius:64px 64px 0 0/64px;margin-top:14px;margin-left:-1px;display:inline-block;vertical-align:middle;position:relative;font-style:normal;color:#ddd;text-align:left;text-indent:-9999px}.user-icon:before{border:2px solid;background-color:#fff;width:12px;height:12px;top:-19px;border-radius:50%;position:absolute;left:50%;-webkit-transform:translateX(-50%);-ms-transform:translateX(-50%);transform:translate(-50%)}.user-icon:before,.user-icon:after{content:\"\";pointer-events:none}.upload-icon{position:absolute;margin-left:3px;margin-top:12px;width:13px;height:4px;border:solid 1px currentColor;border-top:none;border-radius:1px}.upload-icon:before{content:\"\";position:absolute;top:-8px;left:6px;width:1px;height:9px;background-color:currentColor}.upload-icon:after{content:\"\";position:absolute;top:-8px;left:4px;width:4px;height:4px;border-top:solid 1px currentColor;border-right:solid 1px currentColor;-webkit-transform:rotate(-45deg);transform:rotate(-45deg)}.paperclip-icon{position:absolute;margin-left:9px;margin-top:2px;width:6px;height:12px;border-radius:4px 4px 0 0;border-left:solid 1px currentColor;border-right:solid 1px currentColor;border-top:solid 1px currentColor;-webkit-transform:rotate(45deg);transform:rotate(45deg)}.paperclip-icon:before{content:\"\";position:absolute;top:11px;left:-1px;width:4px;height:6px;border-radius:0 0 3px 3px;border-left:solid 1px currentColor;border-right:solid 1px currentColor;border-bottom:solid 1px currentColor}.paperclip-icon:after{content:\"\";position:absolute;left:1px;top:1px;width:2px;height:10px;border-radius:4px 4px 0 0;border-left:solid 1px currentColor;border-right:solid 1px currentColor;border-top:solid 1px currentColor}.check-icon{color:#000;position:absolute;margin-left:3px;margin-top:4px;width:14px;height:8px;border-bottom:solid 1px currentColor;border-left:solid 1px currentColor;-webkit-transform:rotate(-45deg);transform:rotate(-45deg)}.remove-icon{color:#000;position:absolute;margin-left:3px;margin-top:10px}.remove-icon:before{content:\"\";position:absolute;width:15px;height:1px;background-color:currentColor;-webkit-transform:rotate(45deg);transform:rotate(45deg)}.remove-icon:after{content:\"\";position:absolute;width:15px;height:1px;background-color:currentColor;-webkit-transform:rotate(-45deg);transform:rotate(-45deg)}\n", ".loader,.loader:before,.loader:after{background:#e3e3e3;-webkit-animation:load1 1s infinite ease-in-out;animation:load1 1s infinite ease-in-out;width:1em;height:4em}.loader{color:#e3e3e3;text-indent:-9999em;margin:4px auto 0;position:relative;font-size:4px;-webkit-transform:translateZ(0);-ms-transform:translateZ(0);transform:translateZ(0);-webkit-animation-delay:-.16s;animation-delay:-.16s}.loader:before,.loader:after{position:absolute;top:0;content:\"\"}.loader:before{left:-1.5em;-webkit-animation-delay:-.32s;animation-delay:-.32s}.loader:after{left:1.5em}@-webkit-keyframes load1{0%,80%,to{box-shadow:0 0;height:4em}40%{box-shadow:0 -2em;height:5em}}@keyframes load1{0%,80%,to{box-shadow:0 0;height:4em}40%{box-shadow:0 -2em;height:5em}}\n", "#ng-chat{position:fixed;z-index:999;right:0;bottom:0;box-sizing:initial;font-size:11pt;text-align:left}#ng-chat input{outline:none}#ng-chat .shadowed{box-shadow:0 4px 8px #00000040}.ng-chat-loading-wrapper{height:30px;text-align:center;font-size:.9em}.ng-chat-close{text-decoration:none!important;color:red!important;float:right!important;font-weight:900!important;font-size:15px!important}.ng-chat-title,.ng-chat-title:hover{position:relative;z-index:2;height:30px;line-height:30px;font-size:.9em;padding:0 10px;display:block;text-decoration:none;color:#fff!important;font-weight:600!important;cursor:pointer;background-color:#1b1b70!important}.ng-chat-title .ng-chat-title-visibility-toggle-area{display:inline-block;width:85%}.ng-chat-title .ng-chat-title-visibility-toggle-area>strong{font-weight:600;display:block;overflow:hidden;height:30px;text-overflow:ellipsis;white-space:nowrap;max-width:85%;float:left}.ng-chat-title .ng-chat-title-visibility-toggle-area .ng-chat-participant-status{float:left;margin-left:5px}.ng-chat-participant-status{display:inline-block;border-radius:25px;width:8px;height:8px;margin-top:10px}.ng-chat-participant-status.online{background-color:#92a400}.ng-chat-participant-status.busy{background-color:#f91c1e}.ng-chat-participant-status.away{background-color:#f7d21b}.ng-chat-participant-status.offline{background-color:#bababa}.ng-chat-unread-messages-count{margin-left:5px;padding:0 5px;border-radius:25px;font-size:.9em;line-height:30px}.ng-chat-options-container{float:right;margin-right:5px}\n", "#ng-chat.light-theme .primary-text,#ng-chat.light-theme{color:#5c5c5c;font-family:Arial,Helvetica,sans-serif}#ng-chat.light-theme .primary-background{background-color:#fff}#ng-chat.light-theme .secondary-background{background-color:#fafafa}#ng-chat.light-theme .primary-outline-color{border-color:#a3a3a3}#ng-chat.light-theme .friends-search-bar{background-color:#fff}#ng-chat.light-theme .unread-messages-counter-container,#ng-chat.light-theme .ng-chat-people-action,#ng-chat.light-theme .ng-chat-people-action>i{color:#5c5c5c;background-color:#e3e3e3}#ng-chat.light-theme .load-history-action{background-color:#e3e3e3}#ng-chat.light-theme .chat-window-input{background-color:#fff}#ng-chat.light-theme .sent-chat-message-container,#ng-chat.light-theme .file-message-container{background-color:#e3e3e3;border-color:#e3e3e3}#ng-chat.light-theme .received-chat-message-container,#ng-chat.light-theme .file-message-container.received{background-color:#fff;border-color:#e3e3e3}body{font-family:Roboto,Helvetica Neue,sans-serif!important}*{box-shadow:none!important}\n", "#ng-chat.dark-theme .primary-text,#ng-chat.dark-theme{color:#fff;font-family:Arial,Helvetica,sans-serif}#ng-chat.dark-theme .primary-background{background-color:#565656}#ng-chat.dark-theme .secondary-background{background-color:#444}#ng-chat.dark-theme .primary-outline-color{border-color:#353535}#ng-chat.dark-theme .friends-search-bar{background-color:#444;border:1px solid #444;color:#fff}#ng-chat.dark-theme .unread-messages-counter-container,#ng-chat.dark-theme .ng-chat-people-action,#ng-chat.dark-theme .ng-chat-people-action>i{background-color:#fff;color:#444}#ng-chat.dark-theme .load-history-action{background-color:#444}#ng-chat.dark-theme .chat-window-input{background-color:#444;color:#fff}#ng-chat.dark-theme .sent-chat-message-container,#ng-chat.dark-theme .file-message-container{border-color:#444;background-color:#444}#ng-chat.dark-theme .received-chat-message-container,#ng-chat.dark-theme .file-message-container.received{background-color:#565656;border-color:#444}#ng-chat.dark-theme .ng-chat-footer{background-color:#444}#ng-chat.dark-theme .ng-chat-message a{color:#fff}body{font-family:Roboto,Helvetica Neue,sans-serif!important}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "component", type: i3.NgChatFriendsListComponent, selector: "ng-chat-friends-list", inputs: ["participants", "participantsResponse", "participantsInteractedWith", "windows", "userId", "localization", "shouldDisplay", "isCollapsed", "searchEnabled", "currentActiveOption", "hideFriendsList"], outputs: ["onParticipantClicked", "onOptionPromptCanceled", "onOptionPromptConfirmed", "onCloseWindowClick"] }, { kind: "component", type: i4.NgChatWindowComponent, selector: "ng-chat-window", inputs: ["fileUploadAdapter", "window", "userId", "localization", "showOptions", "emojisEnabled", "linkfyEnabled", "showMessageDate", "messageDatePipeFormat", "hasPagedHistory"], outputs: ["onChatWindowClosed", "onMessagesSeen", "onMessageSent", "onTabTriggered", "onOptionTriggered", "onLoadHistoryTriggered"] }, { kind: "pipe", type: i5.SanitizePipe, name: "sanitize" }], encapsulation: i0.ViewEncapsulation.None }); }
503
+ }
504
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.1.0", ngImport: i0, type: NgChat, decorators: [{
505
+ type: Component,
506
+ args: [{ selector: 'mnl-chat', encapsulation: ViewEncapsulation.None, template: "<link *ngIf=\"customTheme\" rel=\"stylesheet\" [href]='customTheme | sanitize'>\n\n<div id=\"ng-chat\" *ngIf=\"!isDisabled && isBootstrapped && !unsupportedViewport\" [ngClass]=\"theme\">\n <ng-chat-friends-list\n [localization]=\"localization\"\n [shouldDisplay]=\"!hideFriendsList\"\n [userId]=\"userId\"\n [isCollapsed]=\"isCollapsed\"\n [searchEnabled]=\"searchEnabled\"\n [participants]=\"participants\"\n [participantsResponse]=\"participantsResponse\"\n [participantsInteractedWith]=\"participantsInteractedWith\"\n [windows]=\"windows\"\n [currentActiveOption]=\"currentActiveOption\"\n (onParticipantClicked)=\"onParticipantClickedFromFriendsList($event)\"\n (onOptionPromptCanceled)=\"onOptionPromptCanceled()\"\n (onOptionPromptConfirmed)=\"onOptionPromptConfirmed($event)\"\n (onCloseWindowClick)=\"onCloseWindowClick($event)\"\n >\n </ng-chat-friends-list>\n\n <div *ngFor=\"let window of windows; let i = index\" [ngClass]=\"{'ng-chat-window': true, 'primary-outline-color': true, 'ng-chat-window-collapsed': window.isCollapsed}\" [ngStyle]=\"{'right': (!hideFriendsList ? friendsListWidth : 0) + 20 + windowSizeFactor * i + 'px'}\">\n <ng-chat-window\n #chatWindow\n [fileUploadAdapter]=\"fileUploadAdapter\"\n [localization]=\"localization\"\n [userId]=\"userId\"\n [window]=\"window\"\n [showOptions]=\"groupAdapter\"\n [emojisEnabled]=\"emojisEnabled\"\n [linkfyEnabled]=\"linkfyEnabled\"\n [showMessageDate]=\"showMessageDate\"\n [messageDatePipeFormat]=\"messageDatePipeFormat\"\n [hasPagedHistory]=\"hasPagedHistory\"\n (onMessagesSeen)=\"onWindowMessagesSeen($event)\"\n (onMessageSent)=\"onWindowMessageSent($event)\"\n (onTabTriggered)=\"onWindowTabTriggered($event)\"\n (onChatWindowClosed)=\"onWindowChatClosed($event)\"\n (onOptionTriggered)=\"onWindowOptionTriggered($event)\"\n (onLoadHistoryTriggered)=\"fetchMessageHistory($event)\"\n >\n </ng-chat-window>\n </div>\n</div>\n", styles: [".user-icon{box-sizing:border-box;background-color:#fff;border:2px solid;width:32px;height:20px;border-radius:64px 64px 0 0/64px;margin-top:14px;margin-left:-1px;display:inline-block;vertical-align:middle;position:relative;font-style:normal;color:#ddd;text-align:left;text-indent:-9999px}.user-icon:before{border:2px solid;background-color:#fff;width:12px;height:12px;top:-19px;border-radius:50%;position:absolute;left:50%;-webkit-transform:translateX(-50%);-ms-transform:translateX(-50%);transform:translate(-50%)}.user-icon:before,.user-icon:after{content:\"\";pointer-events:none}.upload-icon{position:absolute;margin-left:3px;margin-top:12px;width:13px;height:4px;border:solid 1px currentColor;border-top:none;border-radius:1px}.upload-icon:before{content:\"\";position:absolute;top:-8px;left:6px;width:1px;height:9px;background-color:currentColor}.upload-icon:after{content:\"\";position:absolute;top:-8px;left:4px;width:4px;height:4px;border-top:solid 1px currentColor;border-right:solid 1px currentColor;-webkit-transform:rotate(-45deg);transform:rotate(-45deg)}.paperclip-icon{position:absolute;margin-left:9px;margin-top:2px;width:6px;height:12px;border-radius:4px 4px 0 0;border-left:solid 1px currentColor;border-right:solid 1px currentColor;border-top:solid 1px currentColor;-webkit-transform:rotate(45deg);transform:rotate(45deg)}.paperclip-icon:before{content:\"\";position:absolute;top:11px;left:-1px;width:4px;height:6px;border-radius:0 0 3px 3px;border-left:solid 1px currentColor;border-right:solid 1px currentColor;border-bottom:solid 1px currentColor}.paperclip-icon:after{content:\"\";position:absolute;left:1px;top:1px;width:2px;height:10px;border-radius:4px 4px 0 0;border-left:solid 1px currentColor;border-right:solid 1px currentColor;border-top:solid 1px currentColor}.check-icon{color:#000;position:absolute;margin-left:3px;margin-top:4px;width:14px;height:8px;border-bottom:solid 1px currentColor;border-left:solid 1px currentColor;-webkit-transform:rotate(-45deg);transform:rotate(-45deg)}.remove-icon{color:#000;position:absolute;margin-left:3px;margin-top:10px}.remove-icon:before{content:\"\";position:absolute;width:15px;height:1px;background-color:currentColor;-webkit-transform:rotate(45deg);transform:rotate(45deg)}.remove-icon:after{content:\"\";position:absolute;width:15px;height:1px;background-color:currentColor;-webkit-transform:rotate(-45deg);transform:rotate(-45deg)}\n", ".loader,.loader:before,.loader:after{background:#e3e3e3;-webkit-animation:load1 1s infinite ease-in-out;animation:load1 1s infinite ease-in-out;width:1em;height:4em}.loader{color:#e3e3e3;text-indent:-9999em;margin:4px auto 0;position:relative;font-size:4px;-webkit-transform:translateZ(0);-ms-transform:translateZ(0);transform:translateZ(0);-webkit-animation-delay:-.16s;animation-delay:-.16s}.loader:before,.loader:after{position:absolute;top:0;content:\"\"}.loader:before{left:-1.5em;-webkit-animation-delay:-.32s;animation-delay:-.32s}.loader:after{left:1.5em}@-webkit-keyframes load1{0%,80%,to{box-shadow:0 0;height:4em}40%{box-shadow:0 -2em;height:5em}}@keyframes load1{0%,80%,to{box-shadow:0 0;height:4em}40%{box-shadow:0 -2em;height:5em}}\n", "#ng-chat{position:fixed;z-index:999;right:0;bottom:0;box-sizing:initial;font-size:11pt;text-align:left}#ng-chat input{outline:none}#ng-chat .shadowed{box-shadow:0 4px 8px #00000040}.ng-chat-loading-wrapper{height:30px;text-align:center;font-size:.9em}.ng-chat-close{text-decoration:none!important;color:red!important;float:right!important;font-weight:900!important;font-size:15px!important}.ng-chat-title,.ng-chat-title:hover{position:relative;z-index:2;height:30px;line-height:30px;font-size:.9em;padding:0 10px;display:block;text-decoration:none;color:#fff!important;font-weight:600!important;cursor:pointer;background-color:#1b1b70!important}.ng-chat-title .ng-chat-title-visibility-toggle-area{display:inline-block;width:85%}.ng-chat-title .ng-chat-title-visibility-toggle-area>strong{font-weight:600;display:block;overflow:hidden;height:30px;text-overflow:ellipsis;white-space:nowrap;max-width:85%;float:left}.ng-chat-title .ng-chat-title-visibility-toggle-area .ng-chat-participant-status{float:left;margin-left:5px}.ng-chat-participant-status{display:inline-block;border-radius:25px;width:8px;height:8px;margin-top:10px}.ng-chat-participant-status.online{background-color:#92a400}.ng-chat-participant-status.busy{background-color:#f91c1e}.ng-chat-participant-status.away{background-color:#f7d21b}.ng-chat-participant-status.offline{background-color:#bababa}.ng-chat-unread-messages-count{margin-left:5px;padding:0 5px;border-radius:25px;font-size:.9em;line-height:30px}.ng-chat-options-container{float:right;margin-right:5px}\n", "#ng-chat.light-theme .primary-text,#ng-chat.light-theme{color:#5c5c5c;font-family:Arial,Helvetica,sans-serif}#ng-chat.light-theme .primary-background{background-color:#fff}#ng-chat.light-theme .secondary-background{background-color:#fafafa}#ng-chat.light-theme .primary-outline-color{border-color:#a3a3a3}#ng-chat.light-theme .friends-search-bar{background-color:#fff}#ng-chat.light-theme .unread-messages-counter-container,#ng-chat.light-theme .ng-chat-people-action,#ng-chat.light-theme .ng-chat-people-action>i{color:#5c5c5c;background-color:#e3e3e3}#ng-chat.light-theme .load-history-action{background-color:#e3e3e3}#ng-chat.light-theme .chat-window-input{background-color:#fff}#ng-chat.light-theme .sent-chat-message-container,#ng-chat.light-theme .file-message-container{background-color:#e3e3e3;border-color:#e3e3e3}#ng-chat.light-theme .received-chat-message-container,#ng-chat.light-theme .file-message-container.received{background-color:#fff;border-color:#e3e3e3}body{font-family:Roboto,Helvetica Neue,sans-serif!important}*{box-shadow:none!important}\n", "#ng-chat.dark-theme .primary-text,#ng-chat.dark-theme{color:#fff;font-family:Arial,Helvetica,sans-serif}#ng-chat.dark-theme .primary-background{background-color:#565656}#ng-chat.dark-theme .secondary-background{background-color:#444}#ng-chat.dark-theme .primary-outline-color{border-color:#353535}#ng-chat.dark-theme .friends-search-bar{background-color:#444;border:1px solid #444;color:#fff}#ng-chat.dark-theme .unread-messages-counter-container,#ng-chat.dark-theme .ng-chat-people-action,#ng-chat.dark-theme .ng-chat-people-action>i{background-color:#fff;color:#444}#ng-chat.dark-theme .load-history-action{background-color:#444}#ng-chat.dark-theme .chat-window-input{background-color:#444;color:#fff}#ng-chat.dark-theme .sent-chat-message-container,#ng-chat.dark-theme .file-message-container{border-color:#444;background-color:#444}#ng-chat.dark-theme .received-chat-message-container,#ng-chat.dark-theme .file-message-container.received{background-color:#565656;border-color:#444}#ng-chat.dark-theme .ng-chat-footer{background-color:#444}#ng-chat.dark-theme .ng-chat-message a{color:#fff}body{font-family:Roboto,Helvetica Neue,sans-serif!important}\n"] }]
507
+ }], ctorParameters: () => [{ type: i1.HttpClient }], propDecorators: { isDisabled: [{
508
+ type: Input
509
+ }], adapter: [{
510
+ type: Input
511
+ }], groupAdapter: [{
512
+ type: Input
513
+ }], userId: [{
514
+ type: Input
515
+ }], isCollapsed: [{
516
+ type: Input
517
+ }], maximizeWindowOnNewMessage: [{
518
+ type: Input
519
+ }], pollFriendsList: [{
520
+ type: Input
521
+ }], pollingInterval: [{
522
+ type: Input
523
+ }], historyEnabled: [{
524
+ type: Input
525
+ }], emojisEnabled: [{
526
+ type: Input
527
+ }], linkfyEnabled: [{
528
+ type: Input
529
+ }], audioEnabled: [{
530
+ type: Input
531
+ }], searchEnabled: [{
532
+ type: Input
533
+ }], audioSource: [{
534
+ type: Input
535
+ }], persistWindowsState: [{
536
+ type: Input
537
+ }], title: [{
538
+ type: Input
539
+ }], messagePlaceholder: [{
540
+ type: Input
541
+ }], searchPlaceholder: [{
542
+ type: Input
543
+ }], browserNotificationsEnabled: [{
544
+ type: Input
545
+ }], browserNotificationIconSource: [{
546
+ type: Input
547
+ }], browserNotificationTitle: [{
548
+ type: Input
549
+ }], historyPageSize: [{
550
+ type: Input
551
+ }], localization: [{
552
+ type: Input
553
+ }], hideFriendsList: [{
554
+ type: Input
555
+ }], hideFriendsListOnUnsupportedViewport: [{
556
+ type: Input
557
+ }], fileUploadUrl: [{
558
+ type: Input
559
+ }], theme: [{
560
+ type: Input
561
+ }], customTheme: [{
562
+ type: Input
563
+ }], messageDatePipeFormat: [{
564
+ type: Input
565
+ }], showMessageDate: [{
566
+ type: Input
567
+ }], isViewportOnMobileEnabled: [{
568
+ type: Input
569
+ }], fileUploadAdapter: [{
570
+ type: Input
571
+ }], onParticipantClicked: [{
572
+ type: Output
573
+ }], onParticipantChatOpened: [{
574
+ type: Output
575
+ }], onParticipantChatClosed: [{
576
+ type: Output
577
+ }], onMessagesSeen: [{
578
+ type: Output
579
+ }], onCloseWindowClicked: [{
580
+ type: Output
581
+ }], chatWindows: [{
582
+ type: ViewChildren,
583
+ args: ['chatWindow']
584
+ }], onResize: [{
585
+ type: HostListener,
586
+ args: ['window:resize', ['$event']]
587
+ }] } });
588
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-chat.component.js","sourceRoot":"","sources":["../../../../../../../projects/chat/src/lib/components/ng-chat/ng-chat.component.ts","../../../../../../../projects/chat/src/lib/components/ng-chat/ng-chat.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable */\nimport { Component, Input, OnInit, ViewChildren, QueryList, HostListener, Output, EventEmitter, ViewEncapsulation } from '@angular/core';\nimport { HttpClient } from '@angular/common/http';\nimport { ChatAdapter } from '../../core/chat-adapter';\nimport { IChatGroupAdapter } from '../../core/chat-group-adapter';\nimport { User } from \"../../core/user\";\nimport { ParticipantResponse } from \"../../core/participant-response\";\nimport { Message } from \"../../core/message\";\nimport { MessageType } from \"../../core/message-type.enum\";\nimport { Window } from \"../../core/window\";\nimport { ChatParticipantStatus } from \"../../core/chat-participant-status.enum\";\nimport { ScrollDirection } from \"../../core/scroll-direction.enum\";\nimport { Localization, StatusDescription } from '../../core/localization';\nimport { IChatController } from '../../core/chat-controller';\nimport { PagedHistoryChatAdapter } from '../../core/paged-history-chat-adapter';\nimport { IFileUploadAdapter } from '../../core/file-upload-adapter';\nimport { DefaultFileUploadAdapter } from '../../core/default-file-upload-adapter';\nimport { Theme } from '../../core/theme.enum';\nimport { IChatOption } from '../../core/chat-option';\nimport { Group } from \"../../core/group\";\nimport { ChatParticipantType } from \"../../core/chat-participant-type.enum\";\nimport { IChatParticipant } from \"../../core/chat-participant\";\nimport { map } from 'rxjs/operators';\nimport { NgChatWindowComponent } from '../ng-chat-window/ng-chat-window.component';\n\n\n@Component({\n    selector: 'mnl-chat',\n    templateUrl: 'ng-chat.component.html',\n    styleUrls: [\n        '../../../assets/icons.css',\n        '../../../assets/loading-spinner.css',\n        '../../../assets/ng-chat.component.default.css',\n        '../../../assets/themes/ng-chat.theme.default.scss',\n        '../../../assets/themes/ng-chat.theme.dark.scss'\n    ],\n    encapsulation: ViewEncapsulation.None\n})\n\nexport class NgChat implements OnInit, IChatController {\n    constructor(private _httpClient: HttpClient) { }\n\n    // Exposes enums for the ng-template\n    public ChatParticipantType = ChatParticipantType;\n    public ChatParticipantStatus = ChatParticipantStatus;\n    public MessageType = MessageType;\n\n    private _isDisabled: boolean = false;\n\n    get isDisabled(): boolean {\n        return this._isDisabled;\n    }\n\n    @Input()\n    set isDisabled(value: boolean) {\n        this._isDisabled = value;\n\n        if (value)\n        {\n            window.clearInterval(this.pollingIntervalWindowInstance)\n        }\n        else\n        {\n            this.activateFriendListFetch();\n        }\n    }\n\n    @Input()\n    public adapter: ChatAdapter;\n\n    @Input()\n    public groupAdapter: IChatGroupAdapter;\n\n    @Input()\n    public userId: any;\n\n    @Input()\n    public isCollapsed: boolean = false;\n\n    @Input()\n    public maximizeWindowOnNewMessage: boolean = true;\n\n    @Input()\n    public pollFriendsList: boolean = false;\n\n    @Input()\n    public pollingInterval: number = 5000;\n\n    @Input()\n    public historyEnabled: boolean = true;\n\n    @Input()\n    public emojisEnabled: boolean = true;\n\n    @Input()\n    public linkfyEnabled: boolean = true;\n\n    @Input()\n    public audioEnabled: boolean = true;\n\n    @Input()\n    public searchEnabled: boolean = true;\n\n    @Input() // TODO: This might need a better content strategy\n    public audioSource: string = '../../../assets/notification.wav';\n\n    @Input()\n    public persistWindowsState: boolean = true;\n\n    @Input()\n    public title: string = \"Quick chat\";\n\n    @Input()\n    public messagePlaceholder: string = \"Type a message\";\n\n    @Input()\n    public searchPlaceholder: string = \"Search\";\n\n    @Input()\n    public browserNotificationsEnabled: boolean = true;\n\n    @Input() // TODO: This might need a better content strategy\n    public browserNotificationIconSource: string = '../../../assets/notification.png';\n\n    @Input()\n    public browserNotificationTitle: string = \"New message from\";\n\n    @Input()\n    public historyPageSize: number = 10;\n\n    @Input()\n    public localization: Localization;\n\n    @Input()\n    public hideFriendsList: boolean = false;\n\n    @Input()\n    public hideFriendsListOnUnsupportedViewport: boolean = true;\n\n    @Input()\n    public fileUploadUrl: string;\n\n    @Input()\n    public theme: Theme = Theme.Light;\n\n    @Input()\n    public customTheme: string;\n\n    @Input()\n    public messageDatePipeFormat: string = \"short\";\n\n    @Input()\n    public showMessageDate: boolean = true;\n\n    @Input()\n    public isViewportOnMobileEnabled: boolean = false;\n\n    @Input()\n    public fileUploadAdapter: IFileUploadAdapter;\n\n    @Output()\n    public onParticipantClicked: EventEmitter<IChatParticipant> = new EventEmitter<IChatParticipant>();\n\n    @Output()\n    public onParticipantChatOpened: EventEmitter<IChatParticipant> = new EventEmitter<IChatParticipant>();\n\n    @Output()\n    public onParticipantChatClosed: EventEmitter<IChatParticipant> = new EventEmitter<IChatParticipant>();\n\n    @Output()\n    public onMessagesSeen: EventEmitter<Message[]> = new EventEmitter<Message[]>();\n\n    @Output()\n    public onCloseWindowClicked:EventEmitter<boolean> = new EventEmitter<boolean>()\n\n    private browserNotificationsBootstrapped: boolean = false;\n\n    public hasPagedHistory: boolean = false;\n\n    // Don't want to add this as a setting to simplify usage. Previous placeholder and title settings available to be used, or use full Localization object.\n    private statusDescription: StatusDescription = {\n        online: 'Online',\n        busy: 'Busy',\n        away: 'Away',\n        offline: 'Offline'\n    };\n\n    private audioFile: HTMLAudioElement;\n\n    public participants: IChatParticipant[];\n\n    public participantsResponse: ParticipantResponse[];\n\n    public participantsInteractedWith: IChatParticipant[] = [];\n\n    public currentActiveOption: IChatOption | null;\n\n    private pollingIntervalWindowInstance: number;\n\n    private get localStorageKey(): string\n    {\n        return `ng-chat-users-${this.userId}`; // Appending the user id so the state is unique per user in a computer.\n    };\n\n    // Defines the size of each opened window to calculate how many windows can be opened on the viewport at the same time.\n    public windowSizeFactor: number = 320;\n\n    // Total width size of the friends list section\n    public friendsListWidth: number = 262;\n\n    // Available area to render the plugin\n    private viewPortTotalArea: number;\n\n    // Set to true if there is no space to display at least one chat window and 'hideFriendsListOnUnsupportedViewport' is true\n    public unsupportedViewport: boolean = false;\n\n    windows: Window[] = [];\n    isBootstrapped: boolean = false;\n\n    @ViewChildren('chatWindow') chatWindows: QueryList<NgChatWindowComponent>;\n\n    ngOnInit() {\n        this.bootstrapChat();\n    }\n\n    @HostListener('window:resize', ['$event'])\n    onResize(event: any){\n       this.viewPortTotalArea = event.target.innerWidth;\n\n       this.NormalizeWindows();\n    }\n\n    // Checks if there are more opened windows than the view port can display\n    private NormalizeWindows(): void\n    {\n        const maxSupportedOpenedWindows = Math.floor((this.viewPortTotalArea - (!this.hideFriendsList ? this.friendsListWidth : 0)) / this.windowSizeFactor);\n        const difference = this.windows.length - maxSupportedOpenedWindows;\n\n        if (difference >= 0)\n        {\n            this.windows.splice(this.windows.length - difference);\n        }\n\n        this.updateWindowsState(this.windows);\n\n        // Viewport should have space for at least one chat window but should show in mobile if option is enabled.\n        this.unsupportedViewport = this.isViewportOnMobileEnabled? false : this.hideFriendsListOnUnsupportedViewport && maxSupportedOpenedWindows < 1;\n    }\n\n    // Initializes the chat plugin and the messaging adapter\n    private bootstrapChat(): void\n    {\n        let initializationException = null;\n\n        if (this.adapter != null && this.userId != null)\n        {\n            try\n            {\n                this.viewPortTotalArea = window.innerWidth;\n\n                this.initializeTheme();\n                this.initializeDefaultText();\n                this.initializeBrowserNotifications();\n\n                // Binding event listeners\n                this.adapter.messageReceivedHandler = (participant, msg) => this.onMessageReceived(participant, msg);\n                this.adapter.friendsListChangedHandler = (participantsResponse) => this.onFriendsListChanged(participantsResponse);\n\n                this.activateFriendListFetch();\n\n                this.bufferAudioFile();\n\n                this.hasPagedHistory = this.adapter instanceof PagedHistoryChatAdapter;\n\n                if (this.fileUploadUrl && this.fileUploadUrl !== \"\")\n                {\n                    this.fileUploadAdapter = new DefaultFileUploadAdapter(this.fileUploadUrl, this._httpClient);\n                }\n\n                this.NormalizeWindows();\n\n                this.isBootstrapped = true;\n            }\n            catch(ex)\n            {\n                initializationException = ex;\n            }\n        }\n\n        if (!this.isBootstrapped){\n            console.error(\"ng-chat component couldn't be bootstrapped.\");\n\n            if (this.userId == null){\n                console.error(\"ng-chat can't be initialized without an user id. Please make sure you've provided an userId as a parameter of the ng-chat component.\");\n            }\n            if (this.adapter == null){\n                console.error(\"ng-chat can't be bootstrapped without a ChatAdapter. Please make sure you've provided a ChatAdapter implementation as a parameter of the ng-chat component.\");\n            }\n            if (initializationException)\n            {\n                console.error(`An exception has occurred while initializing ng-chat. Details: ${initializationException.message}`);\n                console.error(initializationException);\n            }\n        }\n    }\n\n    private activateFriendListFetch(): void {\n        if (this.adapter)\n        {\n            // Loading current users list\n            if (this.pollFriendsList){\n                // Setting a long poll interval to update the friends list\n                this.fetchFriendsList(true);\n                this.pollingIntervalWindowInstance = window.setInterval(() => this.fetchFriendsList(false), this.pollingInterval);\n            }\n            else\n            {\n                // Since polling was disabled, a friends list update mechanism will have to be implemented in the ChatAdapter.\n                this.fetchFriendsList(true);\n            }\n        }\n    }\n\n    // Initializes browser notifications\n    private async initializeBrowserNotifications()\n    {\n        if (this.browserNotificationsEnabled && (\"Notification\" in window))\n        {\n            if (await Notification.requestPermission() === \"granted\")\n            {\n                this.browserNotificationsBootstrapped = true;\n            }\n        }\n    }\n\n    // Initializes default text\n    private initializeDefaultText() : void\n    {\n        if (!this.localization)\n        {\n            this.localization = {\n                messagePlaceholder: this.messagePlaceholder,\n                searchPlaceholder: this.searchPlaceholder,\n                title: this.title,\n                statusDescription: this.statusDescription,\n                browserNotificationTitle: this.browserNotificationTitle,\n                loadMessageHistoryPlaceholder: \"Load older messages\"\n            };\n        }\n    }\n\n    private initializeTheme(): void\n    {\n        if (this.customTheme)\n        {\n            this.theme = Theme.Custom;\n        }\n        else if (this.theme != Theme.Light && this.theme != Theme.Dark)\n        {\n            // TODO: Use es2017 in future with Object.values(Theme).includes(this.theme) to do this check\n            throw new Error(`Invalid theme configuration for ng-chat. \"${this.theme}\" is not a valid theme value.`);\n        }\n    }\n\n    // Sends a request to load the friends list\n    private fetchFriendsList(isBootstrapping: boolean): void\n    {\n        this.adapter.listFriends()\n        .pipe(\n            map((participantsResponse: ParticipantResponse[]) => {\n                this.participantsResponse = participantsResponse;\n\n                this.participants = participantsResponse.map((response: ParticipantResponse) => {\n                    return response.participant;\n                });\n            })\n        ).subscribe(() => {\n            if (isBootstrapping)\n            {\n                this.restoreWindowsState();\n            }\n        });\n    }\n\n    fetchMessageHistory(window: Window) {\n        // Not ideal but will keep this until we decide if we are shipping pagination with the default adapter\n        if (this.adapter instanceof PagedHistoryChatAdapter)\n        {\n            window.isLoadingHistory = true;\n\n            this.adapter.getMessageHistoryByPage(window.participant.id, this.historyPageSize, ++window.historyPage)\n            .pipe(\n                map((result: Message[]) => {\n                    result.forEach((message) => this.assertMessageType(message));\n\n                    window.messages = result.concat(window.messages);\n                    window.isLoadingHistory = false;\n\n                    const direction: ScrollDirection = (window.historyPage == 1) ? ScrollDirection.Bottom : ScrollDirection.Top;\n                    window.hasMoreMessages = result.length == this.historyPageSize;\n\n                    setTimeout(() => this.onFetchMessageHistoryLoaded(result, window, direction, true));\n                })\n            ).subscribe();\n        }\n        else\n        {\n            this.adapter.getMessageHistory(window.participant.id)\n            .pipe(\n                map((result: Message[]) => {\n                    result.forEach((message) => this.assertMessageType(message));\n\n                    window.messages = result.concat(window.messages);\n                    window.isLoadingHistory = false;\n\n                    setTimeout(() => this.onFetchMessageHistoryLoaded(result, window, ScrollDirection.Bottom));\n                })\n            ).subscribe();\n        }\n    }\n\n    private onFetchMessageHistoryLoaded(messages: Message[], window: Window, direction: ScrollDirection, forceMarkMessagesAsSeen: boolean = false): void\n    {\n        this.scrollChatWindow(window, direction)\n\n        if (window.hasFocus || forceMarkMessagesAsSeen)\n        {\n            const unseenMessages = messages.filter(m => !m.dateSeen);\n\n            this.markMessagesAsRead(unseenMessages);\n        }\n    }\n\n    // Updates the friends list via the event handler\n    private onFriendsListChanged(participantsResponse: ParticipantResponse[]): void\n    {\n        if (participantsResponse)\n        {\n            this.participantsResponse = participantsResponse;\n\n            this.participants = participantsResponse.map((response: ParticipantResponse) => {\n                return response.participant;\n            });\n\n            this.participantsInteractedWith = [];\n        }\n    }\n\n    // Handles received messages by the adapter\n    private onMessageReceived(participant: IChatParticipant, message: Message)\n    {\n        if (participant && message)\n        {\n            const chatWindow = this.openChatWindow(participant);\n\n            this.assertMessageType(message);\n\n            if (!chatWindow[1] || !this.historyEnabled){\n                chatWindow[0].messages.push(message);\n\n                this.scrollChatWindow(chatWindow[0], ScrollDirection.Bottom);\n\n                if (chatWindow[0].hasFocus)\n                {\n                    this.markMessagesAsRead([message]);\n                }\n            }\n\n            this.emitMessageSound(chatWindow[0]);\n\n            // Do not push browser notifications with message content for privacy purposes if the 'maximizeWindowOnNewMessage' setting is off and this is a new chat window.\n            if (this.maximizeWindowOnNewMessage || (!chatWindow[1] && !chatWindow[0].isCollapsed))\n            {\n                // Some messages are not pushed because they are loaded by fetching the history hence why we supply the message here\n                this.emitBrowserNotification(chatWindow[0], message);\n            }\n        }\n    }\n\n    onParticipantClickedFromFriendsList(participant: IChatParticipant): void {\n        this.openChatWindow(participant, true, true);\n    }\n\n    private cancelOptionPrompt(): void {\n        if (this.currentActiveOption)\n        {\n            this.currentActiveOption.isActive = false;\n            this.currentActiveOption = null;\n        }\n    }\n\n    onOptionPromptCanceled(): void {\n        this.cancelOptionPrompt();\n    }\n\n    onOptionPromptConfirmed(event: any): void {\n        // For now this is fine as there is only one option available. Introduce option types and type checking if a new option is added.\n        this.confirmNewGroup(event);\n\n        // Canceling current state\n        this.cancelOptionPrompt();\n    }\n\n    private confirmNewGroup(users: User[]): void {\n        const newGroup = new Group(users);\n\n        this.openChatWindow(newGroup);\n\n        if (this.groupAdapter)\n        {\n            this.groupAdapter.groupCreated(newGroup);\n        }\n    }\n\n    // Opens a new chat whindow. Takes care of available viewport\n    // Works for opening a chat window for an user or for a group\n    // Returns => [Window: Window object reference, boolean: Indicates if this window is a new chat window]\n    private openChatWindow(participant: IChatParticipant, focusOnNewWindow: boolean = false, invokedByUserClick: boolean = false): [Window, boolean]\n    {\n        // Is this window opened?\n        const openedWindow = this.windows.find(x => x.participant.id == participant.id);\n\n        if (!openedWindow)\n        {\n            if (invokedByUserClick)\n            {\n                this.onParticipantClicked.emit(participant);\n            }\n\n            // Refer to issue #58 on Github\n            const collapseWindow = invokedByUserClick ? false : !this.maximizeWindowOnNewMessage;\n\n            const newChatWindow: Window = new Window(participant, this.historyEnabled, collapseWindow);\n\n            // Loads the chat history via an RxJs Observable\n            if (this.historyEnabled)\n            {\n                this.fetchMessageHistory(newChatWindow);\n            }\n\n            this.windows.unshift(newChatWindow);\n\n            // Is there enough space left in the view port ? but should be displayed in mobile if option is enabled\n            if (!this.isViewportOnMobileEnabled) {\n                if (this.windows.length * this.windowSizeFactor >= this.viewPortTotalArea - (!this.hideFriendsList ? this.friendsListWidth : 0)) {\n                    this.windows.pop();\n                }\n            }\n\n            this.updateWindowsState(this.windows);\n\n            if (focusOnNewWindow && !collapseWindow)\n            {\n                this.focusOnWindow(newChatWindow);\n            }\n\n            this.participantsInteractedWith.push(participant);\n            this.onParticipantChatOpened.emit(participant);\n\n            return [newChatWindow, true];\n        }\n        else\n        {\n            // Returns the existing chat window\n            return [openedWindow, false];\n        }\n    }\n\n    // Focus on the input element of the supplied window\n    private focusOnWindow(window: Window, callback: Function = () => {}) : void\n    {\n        const windowIndex = this.windows.indexOf(window);\n        if (windowIndex >= 0)\n        {\n            setTimeout(() => {\n                if (this.chatWindows)\n                {\n                    const chatWindowToFocus = this.chatWindows.toArray()[windowIndex];\n\n                    chatWindowToFocus.chatWindowInput.nativeElement.focus();\n                }\n\n                callback();\n            });\n        }\n    }\n\n    private assertMessageType(message: Message): void {\n        // Always fallback to \"Text\" messages to avoid rendenring issues\n        if (!message.type)\n        {\n            message.type = MessageType.Text;\n        }\n    }\n\n    // Marks all messages provided as read with the current time.\n    markMessagesAsRead(messages: Message[]): void\n    {\n        const currentDate = new Date();\n\n        messages.forEach((msg)=>{\n            msg.dateSeen = currentDate;\n        });\n\n        this.onMessagesSeen.emit(messages);\n    }\n\n    // Buffers audio file (For component's bootstrapping)\n    private bufferAudioFile(): void {\n        if (this.audioSource && this.audioSource.length > 0)\n        {\n            this.audioFile = new Audio();\n            this.audioFile.src = this.audioSource;\n            this.audioFile.load();\n        }\n    }\n\n    // Emits a message notification audio if enabled after every message received\n    private emitMessageSound(window: Window): void\n    {\n        if (this.audioEnabled && !window.hasFocus && this.audioFile) {\n            this.audioFile.play();\n        }\n    }\n\n    // Emits a browser notification\n    private emitBrowserNotification(window: Window, message: Message): void\n    {\n        // if (this.browserNotificationsBootstrapped && !window.hasFocus && message) {\n        if (this.browserNotificationsBootstrapped && message) {\n            const notification = new Notification(`${this.localization.browserNotificationTitle} ${window.participant.displayName}`, {\n                'body': message.message,\n                'icon': this.browserNotificationIconSource\n            });\n\n            setTimeout(() => {\n                notification.close();\n            }, message.message.length <= 50 ? 5000 : 7000); // More time to read longer messages\n        }\n    }\n\n    // Saves current windows state into local storage if persistence is enabled\n    private updateWindowsState(windows: Window[]): void\n    {\n        if (this.persistWindowsState)\n        {\n            const participantIds = windows.map((w) => {\n                return w.participant.id;\n            });\n\n            localStorage.setItem(this.localStorageKey, JSON.stringify(participantIds));\n        }\n    }\n\n    private restoreWindowsState(): void\n    {\n        try\n        {\n            if (this.persistWindowsState)\n            {\n                const stringfiedParticipantIds = localStorage.getItem(this.localStorageKey);\n\n                if (stringfiedParticipantIds && stringfiedParticipantIds.length > 0)\n                {\n                    const participantIds = <number[]>JSON.parse(stringfiedParticipantIds);\n\n                    const participantsToRestore = this.participants.filter(u => participantIds.indexOf(u.id) >= 0);\n\n                    participantsToRestore.forEach((participant) => {\n                        this.openChatWindow(participant);\n                    });\n                }\n            }\n        }\n        catch (ex)\n        {\n            console.error(`An error occurred while restoring ng-chat windows state. Details: ${ex}`);\n        }\n    }\n\n    // Gets closest open window if any. Most recent opened has priority (Right)\n    private getClosestWindow(window: Window): Window | undefined\n    {\n        const index = this.windows.indexOf(window);\n\n        if (index > 0)\n        {\n            return this.windows[index - 1];\n        }\n        else if (index == 0 && this.windows.length > 1)\n        {\n            return this.windows[index + 1];\n        }\n    }\n\n    private closeWindow(window: Window): void\n    {\n        const index = this.windows.indexOf(window);\n\n        this.windows.splice(index, 1);\n\n        this.updateWindowsState(this.windows);\n\n        this.onParticipantChatClosed.emit(window.participant);\n    }\n\n    private getChatWindowComponentInstance(targetWindow: Window): NgChatWindowComponent | null {\n        const windowIndex = this.windows.indexOf(targetWindow);\n\n        if (this.chatWindows){\n            let targetWindow = this.chatWindows.toArray()[windowIndex];\n\n            return targetWindow;\n        }\n\n        return null;\n    }\n\n    // Scrolls a chat window message flow to the bottom\n    private scrollChatWindow(window: Window, direction: ScrollDirection): void\n    {\n        const chatWindow = this.getChatWindowComponentInstance(window);\n\n        if (chatWindow){\n            chatWindow.scrollChatWindow(window, direction);\n        }\n    }\n\n    onWindowMessagesSeen(messagesSeen: Message[]): void {\n        this.markMessagesAsRead(messagesSeen);\n    }\n\n    onWindowChatClosed(payload: { closedWindow: Window, closedViaEscapeKey: boolean }): void {\n        const { closedWindow, closedViaEscapeKey } = payload;\n\n        if (closedViaEscapeKey) {\n            let closestWindow = this.getClosestWindow(closedWindow);\n\n            if (closestWindow)\n            {\n                this.focusOnWindow(closestWindow, () => { this.closeWindow(closedWindow); });\n            }\n            else\n            {\n                this.closeWindow(closedWindow);\n            }\n        }\n        else {\n            this.closeWindow(closedWindow);\n        }\n    }\n\n    onWindowTabTriggered(payload: { triggeringWindow: Window, shiftKeyPressed: boolean }): void {\n        const { triggeringWindow, shiftKeyPressed } = payload;\n\n        const currentWindowIndex = this.windows.indexOf(triggeringWindow);\n        let windowToFocus = this.windows[currentWindowIndex + (shiftKeyPressed ? 1 : -1)]; // Goes back on shift + tab\n\n        if (!windowToFocus)\n        {\n            // Edge windows, go to start or end\n            windowToFocus = this.windows[currentWindowIndex > 0 ? 0 : this.chatWindows.length - 1];\n        }\n\n        this.focusOnWindow(windowToFocus);\n    }\n\n    onWindowMessageSent(messageSent: Message): void {\n        this.adapter.sendMessage(messageSent);\n    }\n\n    onWindowOptionTriggered(option: IChatOption): void {\n        this.currentActiveOption = option;\n    }\n\n    triggerOpenChatWindow(user: User): void {\n        if (user)\n        {\n            this.openChatWindow(user);\n        }\n    }\n\n  public  triggerCloseChatWindow(userId: any): void {\n        const openedWindow = this.windows.find(x => x.participant.id == userId);\n\n        if (openedWindow)\n        {\n            this.closeWindow(openedWindow);\n        }\n    }\n\n    triggerToggleChatWindowVisibility(userId: any): void {\n        const openedWindow = this.windows.find(x => x.participant.id == userId);\n\n        if (openedWindow)\n        {\n            const chatWindow = this.getChatWindowComponentInstance(openedWindow);\n\n            if (chatWindow){\n                chatWindow.onChatWindowClicked(openedWindow);\n            }\n        }\n    }\n\n    // closeFriendList(){\n    //     this.hideFriendsList = !this.hideFriendsList\n    //     this.hideFriendsListChange.emit(this.hideFriendsList)\n    // }\n    onCloseWindowClick(click:boolean){\n\n        this.onCloseWindowClicked.emit(click)\n\n\n\n    }\n}\n/* eslint-enable */","<link *ngIf=\"customTheme\" rel=\"stylesheet\" [href]='customTheme | sanitize'>\n\n<div id=\"ng-chat\" *ngIf=\"!isDisabled && isBootstrapped && !unsupportedViewport\" [ngClass]=\"theme\">\n    <ng-chat-friends-list\n        [localization]=\"localization\"\n        [shouldDisplay]=\"!hideFriendsList\"\n        [userId]=\"userId\"\n        [isCollapsed]=\"isCollapsed\"\n        [searchEnabled]=\"searchEnabled\"\n        [participants]=\"participants\"\n        [participantsResponse]=\"participantsResponse\"\n        [participantsInteractedWith]=\"participantsInteractedWith\"\n        [windows]=\"windows\"\n        [currentActiveOption]=\"currentActiveOption\"\n        (onParticipantClicked)=\"onParticipantClickedFromFriendsList($event)\"\n        (onOptionPromptCanceled)=\"onOptionPromptCanceled()\"\n        (onOptionPromptConfirmed)=\"onOptionPromptConfirmed($event)\"\n        (onCloseWindowClick)=\"onCloseWindowClick($event)\"\n    >\n    </ng-chat-friends-list>\n\n    <div *ngFor=\"let window of windows; let i = index\" [ngClass]=\"{'ng-chat-window': true, 'primary-outline-color': true, 'ng-chat-window-collapsed': window.isCollapsed}\" [ngStyle]=\"{'right': (!hideFriendsList ? friendsListWidth : 0) + 20 + windowSizeFactor * i + 'px'}\">\n        <ng-chat-window\n            #chatWindow\n            [fileUploadAdapter]=\"fileUploadAdapter\"\n            [localization]=\"localization\"\n            [userId]=\"userId\"\n            [window]=\"window\"\n            [showOptions]=\"groupAdapter\"\n            [emojisEnabled]=\"emojisEnabled\"\n            [linkfyEnabled]=\"linkfyEnabled\"\n            [showMessageDate]=\"showMessageDate\"\n            [messageDatePipeFormat]=\"messageDatePipeFormat\"\n            [hasPagedHistory]=\"hasPagedHistory\"\n            (onMessagesSeen)=\"onWindowMessagesSeen($event)\"\n            (onMessageSent)=\"onWindowMessageSent($event)\"\n            (onTabTriggered)=\"onWindowTabTriggered($event)\"\n            (onChatWindowClosed)=\"onWindowChatClosed($event)\"\n            (onOptionTriggered)=\"onWindowOptionTriggered($event)\"\n            (onLoadHistoryTriggered)=\"fetchMessageHistory($event)\"\n        >\n        </ng-chat-window>\n    </div>\n</div>\n"]}