@muraai/mnl-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.
- package/README.md +24 -0
- package/esm2020/lib/chat.module.mjs +87 -0
- package/esm2020/lib/components/ng-chat/ng-chat.component.mjs +588 -0
- package/esm2020/lib/components/ng-chat-friends-list/ng-chat-friends-list.component.mjs +128 -0
- package/esm2020/lib/components/ng-chat-options/ng-chat-options.component.mjs +29 -0
- package/esm2020/lib/components/ng-chat-window/ng-chat-window.component.mjs +247 -0
- package/esm2020/lib/core/chat-adapter.mjs +18 -0
- package/esm2020/lib/core/chat-controller.mjs +2 -0
- package/esm2020/lib/core/chat-group-adapter.mjs +2 -0
- package/esm2020/lib/core/chat-option.mjs +2 -0
- package/esm2020/lib/core/chat-participant-status-descriptor.mjs +6 -0
- package/esm2020/lib/core/chat-participant-status.enum.mjs +8 -0
- package/esm2020/lib/core/chat-participant-type.enum.mjs +6 -0
- package/esm2020/lib/core/chat-participant.mjs +2 -0
- package/esm2020/lib/core/default-file-upload-adapter.mjs +43 -0
- package/esm2020/lib/core/file-upload-adapter.mjs +2 -0
- package/esm2020/lib/core/group.mjs +14 -0
- package/esm2020/lib/core/guid.mjs +11 -0
- package/esm2020/lib/core/localization.mjs +2 -0
- package/esm2020/lib/core/message-counter.mjs +25 -0
- package/esm2020/lib/core/message-type.enum.mjs +7 -0
- package/esm2020/lib/core/message.mjs +7 -0
- package/esm2020/lib/core/paged-history-chat-adapter.mjs +9 -0
- package/esm2020/lib/core/participant-metadata.mjs +6 -0
- package/esm2020/lib/core/participant-response.mjs +3 -0
- package/esm2020/lib/core/scroll-direction.enum.mjs +6 -0
- package/esm2020/lib/core/theme.enum.mjs +7 -0
- package/esm2020/lib/core/user.mjs +7 -0
- package/esm2020/lib/core/window.mjs +20 -0
- package/esm2020/lib/firstLetter.pipe.mjs +20 -0
- package/esm2020/lib/material.module.mjs +50 -0
- package/esm2020/lib/pipes/emojify.pipe.mjs +41 -0
- package/esm2020/lib/pipes/group-message-display-name.pipe.mjs +24 -0
- package/esm2020/lib/pipes/linkfy.pipe.mjs +34 -0
- package/esm2020/lib/pipes/sanitize.pipe.mjs +21 -0
- package/esm2020/lib/services/chat.service.mjs +14 -0
- package/esm2020/muraai-mnl-chat.mjs +5 -0
- package/esm2020/public-api.mjs +25 -0
- package/fesm2015/muraai-mnl-chat.mjs +1426 -0
- package/fesm2015/muraai-mnl-chat.mjs.map +1 -0
- package/fesm2020/muraai-mnl-chat.mjs +1424 -0
- package/fesm2020/muraai-mnl-chat.mjs.map +1 -0
- package/index.d.ts +5 -0
- package/lib/chat.module.d.ts +24 -0
- package/lib/components/ng-chat/ng-chat.component.d.ts +130 -0
- package/lib/components/ng-chat-friends-list/ng-chat-friends-list.component.d.ts +45 -0
- package/lib/components/ng-chat-options/ng-chat-options.component.d.ts +12 -0
- package/lib/components/ng-chat-window/ng-chat-window.component.d.ts +65 -0
- package/lib/core/chat-adapter.d.ts +15 -0
- package/lib/core/chat-controller.d.ts +6 -0
- package/lib/core/chat-group-adapter.d.ts +4 -0
- package/lib/core/chat-option.d.ts +9 -0
- package/lib/core/chat-participant-status-descriptor.d.ts +3 -0
- package/lib/core/chat-participant-status.enum.d.ts +6 -0
- package/lib/core/chat-participant-type.enum.d.ts +4 -0
- package/lib/core/chat-participant.d.ts +9 -0
- package/lib/core/default-file-upload-adapter.d.ts +14 -0
- package/lib/core/file-upload-adapter.d.ts +5 -0
- package/lib/core/group.d.ts +13 -0
- package/lib/core/guid.d.ts +3 -0
- package/lib/core/localization.d.ts +15 -0
- package/lib/core/message-counter.d.ts +10 -0
- package/lib/core/message-type.enum.d.ts +5 -0
- package/lib/core/message.d.ts +10 -0
- package/lib/core/paged-history-chat-adapter.d.ts +10 -0
- package/lib/core/participant-metadata.d.ts +3 -0
- package/lib/core/participant-response.d.ts +6 -0
- package/lib/core/scroll-direction.enum.d.ts +4 -0
- package/lib/core/theme.enum.d.ts +5 -0
- package/lib/core/user.d.ts +10 -0
- package/lib/core/window.d.ts +13 -0
- package/lib/firstLetter.pipe.d.ts +7 -0
- package/lib/material.module.d.ts +14 -0
- package/lib/pipes/emojify.pipe.d.ts +7 -0
- package/lib/pipes/group-message-display-name.pipe.d.ts +9 -0
- package/lib/pipes/linkfy.pipe.d.ts +7 -0
- package/lib/pipes/sanitize.pipe.d.ts +10 -0
- package/lib/services/chat.service.d.ts +6 -0
- package/package.json +31 -0
- package/public-api.d.ts +21 -0
- package/src/assets/icons.css +140 -0
- package/src/assets/loading-spinner.css +59 -0
- package/src/assets/ng-chat.component.default.css +106 -0
- package/src/assets/notification.png +0 -0
- package/src/assets/notification.wav +0 -0
- package/src/assets/themes/ng-chat.theme.dark.scss +71 -0
- package/src/assets/themes/ng-chat.theme.default.scss +63 -0
- package/src/assets/user.png +0 -0
|
@@ -0,0 +1,128 @@
|
|
|
1
|
+
/* eslint-disable */
|
|
2
|
+
import { Component, Input, Output, EventEmitter, ViewEncapsulation } from '@angular/core';
|
|
3
|
+
import { ChatParticipantStatus } from "../../core/chat-participant-status.enum";
|
|
4
|
+
import { MessageCounter } from "../../core/message-counter";
|
|
5
|
+
import { chatParticipantStatusDescriptor } from '../../core/chat-participant-status-descriptor';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/forms";
|
|
8
|
+
import * as i2 from "@angular/common";
|
|
9
|
+
import * as i3 from "../../pipes/sanitize.pipe";
|
|
10
|
+
import * as i4 from "../../firstLetter.pipe";
|
|
11
|
+
export class NgChatFriendsListComponent {
|
|
12
|
+
constructor() {
|
|
13
|
+
this.participantsInteractedWith = [];
|
|
14
|
+
this.onParticipantClicked = new EventEmitter();
|
|
15
|
+
this.onOptionPromptCanceled = new EventEmitter();
|
|
16
|
+
this.onOptionPromptConfirmed = new EventEmitter();
|
|
17
|
+
this.hideFriendsList = false;
|
|
18
|
+
this.onCloseWindowClick = new EventEmitter();
|
|
19
|
+
this.selectedUsersFromFriendsList = [];
|
|
20
|
+
this.searchInput = '';
|
|
21
|
+
// Exposes enums and functions for the ng-template
|
|
22
|
+
this.ChatParticipantStatus = ChatParticipantStatus;
|
|
23
|
+
this.chatParticipantStatusDescriptor = chatParticipantStatusDescriptor;
|
|
24
|
+
this.cleanUpUserSelection = () => this.selectedUsersFromFriendsList = [];
|
|
25
|
+
}
|
|
26
|
+
ngOnChanges(changes) {
|
|
27
|
+
if (this.currentActiveOption) {
|
|
28
|
+
const currentOptionTriggeredBy = this.currentActiveOption && this.currentActiveOption.chattingTo.participant.id;
|
|
29
|
+
const isActivatedUserInSelectedList = (this.selectedUsersFromFriendsList.filter(item => item.id == currentOptionTriggeredBy)).length > 0;
|
|
30
|
+
if (!isActivatedUserInSelectedList) {
|
|
31
|
+
this.selectedUsersFromFriendsList = this.selectedUsersFromFriendsList.concat(this.currentActiveOption.chattingTo.participant);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
// let input = user.displayName;
|
|
35
|
+
// let parts = input.split(" ");
|
|
36
|
+
// let firstLetter = parts[0];
|
|
37
|
+
// let secondLetters = parts[1].slice(0, 2);
|
|
38
|
+
}
|
|
39
|
+
get filteredParticipants() {
|
|
40
|
+
if (this.searchInput.length > 0) {
|
|
41
|
+
// Searches in the friend list by the inputted search string
|
|
42
|
+
return this.participants.filter(x => x.displayName.toUpperCase().includes(this.searchInput.toUpperCase()));
|
|
43
|
+
}
|
|
44
|
+
console.log("this.participants", this.participants);
|
|
45
|
+
return this.participants;
|
|
46
|
+
}
|
|
47
|
+
isUserSelectedFromFriendsList(user) {
|
|
48
|
+
return (this.selectedUsersFromFriendsList.filter(item => item.id == user.id)).length > 0;
|
|
49
|
+
}
|
|
50
|
+
unreadMessagesTotalByParticipant(participant) {
|
|
51
|
+
let openedWindow = this.windows.find(x => x.participant.id == participant.id);
|
|
52
|
+
if (openedWindow) {
|
|
53
|
+
return MessageCounter.unreadMessagesTotal(openedWindow, this.userId);
|
|
54
|
+
}
|
|
55
|
+
else {
|
|
56
|
+
let totalUnreadMessages = this.participantsResponse
|
|
57
|
+
.filter(x => x.participant.id == participant.id && !this.participantsInteractedWith.find(u => u.id == participant.id) && x.metadata && x.metadata.totalUnreadMessages > 0)
|
|
58
|
+
.map((participantResponse) => {
|
|
59
|
+
return participantResponse.metadata.totalUnreadMessages;
|
|
60
|
+
})[0];
|
|
61
|
+
return MessageCounter.formatUnreadMessagesTotal(totalUnreadMessages);
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
// Toggle friends list visibility
|
|
65
|
+
onChatTitleClicked() {
|
|
66
|
+
this.isCollapsed = !this.isCollapsed;
|
|
67
|
+
}
|
|
68
|
+
onFriendsListCheckboxChange(selectedUser, isChecked) {
|
|
69
|
+
if (isChecked) {
|
|
70
|
+
this.selectedUsersFromFriendsList.push(selectedUser);
|
|
71
|
+
}
|
|
72
|
+
else {
|
|
73
|
+
this.selectedUsersFromFriendsList.splice(this.selectedUsersFromFriendsList.indexOf(selectedUser), 1);
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
onUserClick(clickedUser) {
|
|
77
|
+
this.onParticipantClicked.emit(clickedUser);
|
|
78
|
+
}
|
|
79
|
+
onFriendsListActionCancelClicked() {
|
|
80
|
+
this.onOptionPromptCanceled.emit();
|
|
81
|
+
this.cleanUpUserSelection();
|
|
82
|
+
}
|
|
83
|
+
onFriendsListActionConfirmClicked() {
|
|
84
|
+
this.onOptionPromptConfirmed.emit(this.selectedUsersFromFriendsList);
|
|
85
|
+
this.cleanUpUserSelection();
|
|
86
|
+
}
|
|
87
|
+
onCloseWindow() {
|
|
88
|
+
this.onCloseWindowClick.emit(true);
|
|
89
|
+
console.log('hoi latest friendlist', this.shouldDisplay);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
NgChatFriendsListComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NgChatFriendsListComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
93
|
+
NgChatFriendsListComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: NgChatFriendsListComponent, selector: "ng-chat-friends-list", inputs: { participants: "participants", participantsResponse: "participantsResponse", participantsInteractedWith: "participantsInteractedWith", windows: "windows", userId: "userId", localization: "localization", shouldDisplay: "shouldDisplay", isCollapsed: "isCollapsed", searchEnabled: "searchEnabled", currentActiveOption: "currentActiveOption", hideFriendsList: "hideFriendsList" }, outputs: { onParticipantClicked: "onParticipantClicked", onOptionPromptCanceled: "onOptionPromptCanceled", onOptionPromptConfirmed: "onOptionPromptConfirmed", onCloseWindowClick: "onCloseWindowClick" }, usesOnChanges: true, ngImport: i0, template: "<div *ngIf=\"shouldDisplay\" id=\"ng-chat-people\" [ngClass]=\"{'primary-outline-color': true, 'primary-background': true, 'ng-chat-people-collapsed': isCollapsed}\">\n\t<a href=\"javascript:void(0);\" class=\"ng-chat-title secondary-background shadowed\" (click)=\"onChatTitleClicked()\">\n\t\t<span>\n\t\t\t{{localization.title}}\n\t\t</span>\n\t<a href=\"javascript:void(0);\" class=\" ng-chat-close \" (click)=\"onCloseWindow()\" >✕</a>\n\t</a>\n\t<div *ngIf=\"currentActiveOption\" class=\"ng-chat-people-actions\" (click)=\"onFriendsListActionCancelClicked()\">\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-people-action\">\n\t\t\t<i class=\"remove-icon\"></i>\n\t\t</a>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-people-action\" (click)=\"onFriendsListActionConfirmClicked()\">\n\t\t\t<i class=\"check-icon\"></i>\n\t\t</a>\n\t</div>\n\t<input *ngIf=\"searchEnabled\" id=\"ng-chat-search_friend\" class=\"friends-search-bar\" type=\"search\" [placeholder]=\"localization.searchPlaceholder\" [(ngModel)]=\"searchInput\" />\n\t<ul id=\"ng-chat-users\" *ngIf=\"!isCollapsed\" [ngClass]=\"{'offset-search': searchEnabled}\">\n\t\t<li *ngFor=\"let user of filteredParticipants\">\n\t\t\t<input \n\t\t\t\t*ngIf=\"currentActiveOption && currentActiveOption.validateContext(user)\" \n\t\t\t\ttype=\"checkbox\" \n\t\t\t\tclass=\"ng-chat-users-checkbox\" \n\t\t\t\t(change)=\"onFriendsListCheckboxChange(user, $event.target.checked)\" \n\t\t\t\t[checked]=\"isUserSelectedFromFriendsList(user)\"/>\n\t\t\t<div [ngClass]=\"{'ng-chat-friends-list-selectable-offset': currentActiveOption, 'ng-chat-friends-list-container': true}\" (click)=\"onUserClick(user)\">\n\t\t\t\t<!-- <div *ngIf=\"!user.avatar\" class=\"icon-wrapper\">\n\t\t\t\t\t<i class=\"user-icon\"></i>\n\t\t\t\t</div> -->\n\t\t\t\t<div class=\"user\" *ngIf=\"!user.avatar\">\n\t\t\t\t\t{{user.displayName|firstLetterPipe}}\n\n\t\t\t\t</div>\n\n\t\t\t\t<img *ngIf=\"user.avatar\" alt=\"\" class=\"avatar\" height=\"30\" width=\"30\" [src]=\"user.avatar | sanitize\"/>\n\t\t\t\t<div class=\"\"></div>\n\t\t\t\t<strong title=\"{{user.displayName}}\">{{user.displayName}}</strong>\n\t\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': user.status === ChatParticipantStatus.Online, 'busy': user.status === ChatParticipantStatus.Busy, 'away': user.status === ChatParticipantStatus.Away, 'offline': user.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(user.status, localization)}}\"></span>\n\t\t\t\t<span *ngIf=\"unreadMessagesTotalByParticipant(user).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotalByParticipant(user)}}</span>\n\t\t\t</div>\n\t\t</li>\n\t</ul>\n</div>", styles: ["#ng-chat-people{position:relative;width:240px;height:360px;border-width:1px;border-style:solid;margin-right:20px;box-shadow:0 4px 8px #00000040;border-bottom:0;border-radius:10px!important;box-sizing:border-box!important;overflow:hidden!important;margin-bottom:10px!important;border:1px solid grey!important}#ng-chat-people.ng-chat-people-collapsed{height:30px}#ng-chat-search_friend{display:block;padding:7px 10px;margin:10px auto 0;width:calc(100% - 20px);font-size:.9em;border-radius:20px!important;border:.5px solid grey!important}#ng-chat-users{padding:0 10px;list-style:none;margin:0;overflow:auto;position:absolute;top:42px;bottom:0;width:100%;box-sizing:border-box}#ng-chat-users.offset-search{top:84px}#ng-chat-users .ng-chat-users-checkbox{float:left;margin-right:5px;margin-top:8px}#ng-chat-users li{clear:both;margin-bottom:10px;overflow:hidden;cursor:pointer;max-height:30px}#ng-chat-users li>.ng-chat-friends-list-selectable-offset{margin-left:22px}#ng-chat-users li .ng-chat-friends-list-container{display:inline-block;width:100%}#ng-chat-users li>.ng-chat-friends-list-selectable-offset.ng-chat-friends-list-container{display:block;width:auto}#ng-chat-users li .ng-chat-friends-list-container>img.avatar,#ng-chat-users li .ng-chat-friends-list-container>.user{float:left;display:flex;justify-content:center;margin-right:5px}#ng-chat-users li .ng-chat-friends-list-container>.user{overflow:hidden;width:30px;height:30px}#ng-chat-users li .ng-chat-friends-list-container>.user>i{color:#fff;transform:scale(.7)}#ng-chat-users li .ng-chat-friends-list-container>strong{float:left;line-height:30px;font-size:.8em;max-width:57%;max-height:30px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}#ng-chat-users li .ng-chat-friends-list-container>.ng-chat-participant-status{float:right}.ng-chat-people-actions{position:absolute;top:4px;right:5px;margin:0;padding:0;z-index:2}.ng-chat-people-actions>a.ng-chat-people-action{display:inline-block;width:21px;height:21px;margin-right:8px;text-decoration:none;border:none;border-radius:25px;padding:1px}@media only screen and (max-width: 581px){#ng-chat-people{width:300px;height:360px;margin-right:0}}.user{display:flex;align-items:center;justify-content:center;width:24px;height:24px;border-radius:50%;background-color:#00b04c;color:#fff;font-size:12px;font-family:inherit}\n"], dependencies: [{ kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { 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: "pipe", type: i3.SanitizePipe, name: "sanitize" }, { kind: "pipe", type: i4.FirstLetterPipe, name: "firstLetterPipe" }], encapsulation: i0.ViewEncapsulation.None });
|
|
94
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NgChatFriendsListComponent, decorators: [{
|
|
95
|
+
type: Component,
|
|
96
|
+
args: [{ selector: 'ng-chat-friends-list', encapsulation: ViewEncapsulation.None, template: "<div *ngIf=\"shouldDisplay\" id=\"ng-chat-people\" [ngClass]=\"{'primary-outline-color': true, 'primary-background': true, 'ng-chat-people-collapsed': isCollapsed}\">\n\t<a href=\"javascript:void(0);\" class=\"ng-chat-title secondary-background shadowed\" (click)=\"onChatTitleClicked()\">\n\t\t<span>\n\t\t\t{{localization.title}}\n\t\t</span>\n\t<a href=\"javascript:void(0);\" class=\" ng-chat-close \" (click)=\"onCloseWindow()\" >✕</a>\n\t</a>\n\t<div *ngIf=\"currentActiveOption\" class=\"ng-chat-people-actions\" (click)=\"onFriendsListActionCancelClicked()\">\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-people-action\">\n\t\t\t<i class=\"remove-icon\"></i>\n\t\t</a>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-people-action\" (click)=\"onFriendsListActionConfirmClicked()\">\n\t\t\t<i class=\"check-icon\"></i>\n\t\t</a>\n\t</div>\n\t<input *ngIf=\"searchEnabled\" id=\"ng-chat-search_friend\" class=\"friends-search-bar\" type=\"search\" [placeholder]=\"localization.searchPlaceholder\" [(ngModel)]=\"searchInput\" />\n\t<ul id=\"ng-chat-users\" *ngIf=\"!isCollapsed\" [ngClass]=\"{'offset-search': searchEnabled}\">\n\t\t<li *ngFor=\"let user of filteredParticipants\">\n\t\t\t<input \n\t\t\t\t*ngIf=\"currentActiveOption && currentActiveOption.validateContext(user)\" \n\t\t\t\ttype=\"checkbox\" \n\t\t\t\tclass=\"ng-chat-users-checkbox\" \n\t\t\t\t(change)=\"onFriendsListCheckboxChange(user, $event.target.checked)\" \n\t\t\t\t[checked]=\"isUserSelectedFromFriendsList(user)\"/>\n\t\t\t<div [ngClass]=\"{'ng-chat-friends-list-selectable-offset': currentActiveOption, 'ng-chat-friends-list-container': true}\" (click)=\"onUserClick(user)\">\n\t\t\t\t<!-- <div *ngIf=\"!user.avatar\" class=\"icon-wrapper\">\n\t\t\t\t\t<i class=\"user-icon\"></i>\n\t\t\t\t</div> -->\n\t\t\t\t<div class=\"user\" *ngIf=\"!user.avatar\">\n\t\t\t\t\t{{user.displayName|firstLetterPipe}}\n\n\t\t\t\t</div>\n\n\t\t\t\t<img *ngIf=\"user.avatar\" alt=\"\" class=\"avatar\" height=\"30\" width=\"30\" [src]=\"user.avatar | sanitize\"/>\n\t\t\t\t<div class=\"\"></div>\n\t\t\t\t<strong title=\"{{user.displayName}}\">{{user.displayName}}</strong>\n\t\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': user.status === ChatParticipantStatus.Online, 'busy': user.status === ChatParticipantStatus.Busy, 'away': user.status === ChatParticipantStatus.Away, 'offline': user.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(user.status, localization)}}\"></span>\n\t\t\t\t<span *ngIf=\"unreadMessagesTotalByParticipant(user).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotalByParticipant(user)}}</span>\n\t\t\t</div>\n\t\t</li>\n\t</ul>\n</div>", styles: ["#ng-chat-people{position:relative;width:240px;height:360px;border-width:1px;border-style:solid;margin-right:20px;box-shadow:0 4px 8px #00000040;border-bottom:0;border-radius:10px!important;box-sizing:border-box!important;overflow:hidden!important;margin-bottom:10px!important;border:1px solid grey!important}#ng-chat-people.ng-chat-people-collapsed{height:30px}#ng-chat-search_friend{display:block;padding:7px 10px;margin:10px auto 0;width:calc(100% - 20px);font-size:.9em;border-radius:20px!important;border:.5px solid grey!important}#ng-chat-users{padding:0 10px;list-style:none;margin:0;overflow:auto;position:absolute;top:42px;bottom:0;width:100%;box-sizing:border-box}#ng-chat-users.offset-search{top:84px}#ng-chat-users .ng-chat-users-checkbox{float:left;margin-right:5px;margin-top:8px}#ng-chat-users li{clear:both;margin-bottom:10px;overflow:hidden;cursor:pointer;max-height:30px}#ng-chat-users li>.ng-chat-friends-list-selectable-offset{margin-left:22px}#ng-chat-users li .ng-chat-friends-list-container{display:inline-block;width:100%}#ng-chat-users li>.ng-chat-friends-list-selectable-offset.ng-chat-friends-list-container{display:block;width:auto}#ng-chat-users li .ng-chat-friends-list-container>img.avatar,#ng-chat-users li .ng-chat-friends-list-container>.user{float:left;display:flex;justify-content:center;margin-right:5px}#ng-chat-users li .ng-chat-friends-list-container>.user{overflow:hidden;width:30px;height:30px}#ng-chat-users li .ng-chat-friends-list-container>.user>i{color:#fff;transform:scale(.7)}#ng-chat-users li .ng-chat-friends-list-container>strong{float:left;line-height:30px;font-size:.8em;max-width:57%;max-height:30px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}#ng-chat-users li .ng-chat-friends-list-container>.ng-chat-participant-status{float:right}.ng-chat-people-actions{position:absolute;top:4px;right:5px;margin:0;padding:0;z-index:2}.ng-chat-people-actions>a.ng-chat-people-action{display:inline-block;width:21px;height:21px;margin-right:8px;text-decoration:none;border:none;border-radius:25px;padding:1px}@media only screen and (max-width: 581px){#ng-chat-people{width:300px;height:360px;margin-right:0}}.user{display:flex;align-items:center;justify-content:center;width:24px;height:24px;border-radius:50%;background-color:#00b04c;color:#fff;font-size:12px;font-family:inherit}\n"] }]
|
|
97
|
+
}], ctorParameters: function () { return []; }, propDecorators: { participants: [{
|
|
98
|
+
type: Input
|
|
99
|
+
}], participantsResponse: [{
|
|
100
|
+
type: Input
|
|
101
|
+
}], participantsInteractedWith: [{
|
|
102
|
+
type: Input
|
|
103
|
+
}], windows: [{
|
|
104
|
+
type: Input
|
|
105
|
+
}], userId: [{
|
|
106
|
+
type: Input
|
|
107
|
+
}], localization: [{
|
|
108
|
+
type: Input
|
|
109
|
+
}], shouldDisplay: [{
|
|
110
|
+
type: Input
|
|
111
|
+
}], isCollapsed: [{
|
|
112
|
+
type: Input
|
|
113
|
+
}], searchEnabled: [{
|
|
114
|
+
type: Input
|
|
115
|
+
}], currentActiveOption: [{
|
|
116
|
+
type: Input
|
|
117
|
+
}], onParticipantClicked: [{
|
|
118
|
+
type: Output
|
|
119
|
+
}], onOptionPromptCanceled: [{
|
|
120
|
+
type: Output
|
|
121
|
+
}], onOptionPromptConfirmed: [{
|
|
122
|
+
type: Output
|
|
123
|
+
}], hideFriendsList: [{
|
|
124
|
+
type: Input
|
|
125
|
+
}], onCloseWindowClick: [{
|
|
126
|
+
type: Output
|
|
127
|
+
}] } });
|
|
128
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-chat-friends-list.component.js","sourceRoot":"","sources":["../../../../../../../projects/chat/src/lib/components/ng-chat-friends-list/ng-chat-friends-list.component.ts","../../../../../../../projects/chat/src/lib/components/ng-chat-friends-list/ng-chat-friends-list.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable */\nimport { Component, Input, Output, EventEmitter, ViewEncapsulation, OnChanges, SimpleChanges } from '@angular/core';\nimport { Localization } from '../../core/localization';\nimport { IChatOption } from '../../core/chat-option';\nimport { ChatParticipantStatus } from \"../../core/chat-participant-status.enum\";\nimport { IChatParticipant } from \"../../core/chat-participant\";\nimport { User } from \"../../core/user\";\nimport { Window } from \"../../core/window\";\nimport { ParticipantResponse } from \"../../core/participant-response\";\nimport { MessageCounter } from \"../../core/message-counter\";\nimport { chatParticipantStatusDescriptor } from '../../core/chat-participant-status-descriptor';\n\n@Component({\n    selector: 'ng-chat-friends-list',\n    templateUrl: './ng-chat-friends-list.component.html',\n    styleUrls: ['./ng-chat-friends-list.component.css'],\n    encapsulation: ViewEncapsulation.None\n})\nexport class NgChatFriendsListComponent implements OnChanges {\n    constructor() {\n       \n     }\n\n    @Input()\n    public participants: IChatParticipant[];\n\n    @Input()\n    public participantsResponse: ParticipantResponse[];\n\n    @Input()\n    public participantsInteractedWith: IChatParticipant[] = [];\n\n    @Input()\n    public windows: Window[];\n\n    @Input()\n    public userId: any;\n\n    @Input()\n    public localization: Localization;\n\n    @Input()\n    public shouldDisplay: boolean;\n\n    @Input()\n    public isCollapsed: boolean;\n\n    @Input()\n    public searchEnabled: boolean;\n\n    @Input()\n    public currentActiveOption: IChatOption | null;\n\n    @Output()\n    public onParticipantClicked: EventEmitter<IChatParticipant> = new EventEmitter();\n\n    @Output()\n    public onOptionPromptCanceled: EventEmitter<any> = new EventEmitter();\n\n    @Output()\n    public onOptionPromptConfirmed: EventEmitter<any> = new EventEmitter();\n    @Input()\n    public hideFriendsList: boolean =false ;\n\n    @Output()\n    public onCloseWindowClick:EventEmitter<boolean> = new EventEmitter<boolean>()\n    \n   \n\n    public selectedUsersFromFriendsList: User[] = [];\n\n    public searchInput: string = '';\n\n    // Exposes enums and functions for the ng-template\n    public ChatParticipantStatus = ChatParticipantStatus;\n    public chatParticipantStatusDescriptor = chatParticipantStatusDescriptor;\n\n    ngOnChanges(changes: SimpleChanges) {\n        if (this.currentActiveOption) {\n            const currentOptionTriggeredBy = this.currentActiveOption && this.currentActiveOption.chattingTo.participant.id;\n            const isActivatedUserInSelectedList = (this.selectedUsersFromFriendsList.filter(item => item.id == currentOptionTriggeredBy)).length > 0;\n\n            if (!isActivatedUserInSelectedList) {\n                this.selectedUsersFromFriendsList = this.selectedUsersFromFriendsList.concat(this.currentActiveOption.chattingTo.participant as User);\n            }\n        }\n        // let input = user.displayName;\n        // let parts = input.split(\" \");\n        //  let firstLetter = parts[0];\n        //  let secondLetters = parts[1].slice(0, 2);\n    }\n\n    get filteredParticipants(): IChatParticipant[]\n    {\n        if (this.searchInput.length > 0){\n            // Searches in the friend list by the inputted search string\n            return this.participants.filter(x => x.displayName.toUpperCase().includes(this.searchInput.toUpperCase()));\n        }\nconsole.log(\"this.participants\",this.participants);\n\n        return this.participants;\n    }\n\n    isUserSelectedFromFriendsList(user: User) : boolean\n    {\n        return (this.selectedUsersFromFriendsList.filter(item => item.id == user.id)).length > 0\n    }\n\n    unreadMessagesTotalByParticipant(participant: IChatParticipant): string\n    {\n        let openedWindow = this.windows.find(x => x.participant.id == participant.id);\n\n        if (openedWindow){\n            return MessageCounter.unreadMessagesTotal(openedWindow, this.userId);\n        }\n        else\n        {\n            let totalUnreadMessages = this.participantsResponse\n                .filter(x => x.participant.id == participant.id && !this.participantsInteractedWith.find(u => u.id == participant.id) && x.metadata && x.metadata.totalUnreadMessages > 0)\n                .map((participantResponse) => {\n                    return participantResponse.metadata.totalUnreadMessages\n                })[0];\n\n            return MessageCounter.formatUnreadMessagesTotal(totalUnreadMessages);\n        }\n    }\n\n    cleanUpUserSelection = () => this.selectedUsersFromFriendsList = [];\n\n    // Toggle friends list visibility\n    onChatTitleClicked(): void\n    {\n        this.isCollapsed = !this.isCollapsed;\n    }\n\n    onFriendsListCheckboxChange(selectedUser: User, isChecked: boolean): void\n    {\n        if(isChecked) {\n            this.selectedUsersFromFriendsList.push(selectedUser);\n        }\n        else\n        {\n            this.selectedUsersFromFriendsList.splice(this.selectedUsersFromFriendsList.indexOf(selectedUser), 1);\n        }\n    }\n\n    onUserClick(clickedUser: User): void\n    {\n        this.onParticipantClicked.emit(clickedUser);\n    }\n\n    onFriendsListActionCancelClicked(): void\n    {\n        this.onOptionPromptCanceled.emit();\n        this.cleanUpUserSelection();\n    }\n\n    onFriendsListActionConfirmClicked() : void\n    {\n        this.onOptionPromptConfirmed.emit(this.selectedUsersFromFriendsList);\n        this.cleanUpUserSelection();\n    }\n    onCloseWindow(){\n          \n       \n        this.onCloseWindowClick.emit(true)\n\n        console.log('hoi latest friendlist',this.shouldDisplay)\n\n        \n    }\n\n}\n/* eslint-enable */","<div *ngIf=\"shouldDisplay\" id=\"ng-chat-people\" [ngClass]=\"{'primary-outline-color': true, 'primary-background': true, 'ng-chat-people-collapsed': isCollapsed}\">\n\t<a href=\"javascript:void(0);\" class=\"ng-chat-title secondary-background shadowed\" (click)=\"onChatTitleClicked()\">\n\t\t<span>\n\t\t\t{{localization.title}}\n\t\t</span>\n\t<a  href=\"javascript:void(0);\" class=\" ng-chat-close \" (click)=\"onCloseWindow()\" >&#x2715;</a>\n\t</a>\n\t<div *ngIf=\"currentActiveOption\" class=\"ng-chat-people-actions\" (click)=\"onFriendsListActionCancelClicked()\">\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-people-action\">\n\t\t\t<i class=\"remove-icon\"></i>\n\t\t</a>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-people-action\" (click)=\"onFriendsListActionConfirmClicked()\">\n\t\t\t<i class=\"check-icon\"></i>\n\t\t</a>\n\t</div>\n\t<input *ngIf=\"searchEnabled\" id=\"ng-chat-search_friend\" class=\"friends-search-bar\" type=\"search\" [placeholder]=\"localization.searchPlaceholder\" [(ngModel)]=\"searchInput\" />\n\t<ul id=\"ng-chat-users\" *ngIf=\"!isCollapsed\" [ngClass]=\"{'offset-search': searchEnabled}\">\n\t\t<li *ngFor=\"let user of filteredParticipants\">\n\t\t\t<input \n\t\t\t\t*ngIf=\"currentActiveOption && currentActiveOption.validateContext(user)\" \n\t\t\t\ttype=\"checkbox\" \n\t\t\t\tclass=\"ng-chat-users-checkbox\" \n\t\t\t\t(change)=\"onFriendsListCheckboxChange(user, $event.target.checked)\" \n\t\t\t\t[checked]=\"isUserSelectedFromFriendsList(user)\"/>\n\t\t\t<div [ngClass]=\"{'ng-chat-friends-list-selectable-offset': currentActiveOption, 'ng-chat-friends-list-container': true}\" (click)=\"onUserClick(user)\">\n\t\t\t\t<!-- <div *ngIf=\"!user.avatar\" class=\"icon-wrapper\">\n\t\t\t\t\t<i class=\"user-icon\"></i>\n\t\t\t\t</div> -->\n\t\t\t\t<div class=\"user\" *ngIf=\"!user.avatar\">\n\t\t\t\t\t{{user.displayName|firstLetterPipe}}\n\n\t\t\t\t</div>\n\n\t\t\t\t<img *ngIf=\"user.avatar\" alt=\"\" class=\"avatar\" height=\"30\" width=\"30\"  [src]=\"user.avatar | sanitize\"/>\n\t\t\t\t<div class=\"\"></div>\n\t\t\t\t<strong title=\"{{user.displayName}}\">{{user.displayName}}</strong>\n\t\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': user.status === ChatParticipantStatus.Online, 'busy': user.status === ChatParticipantStatus.Busy, 'away': user.status === ChatParticipantStatus.Away, 'offline': user.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(user.status, localization)}}\"></span>\n\t\t\t\t<span *ngIf=\"unreadMessagesTotalByParticipant(user).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotalByParticipant(user)}}</span>\n\t\t\t</div>\n\t\t</li>\n\t</ul>\n</div>"]}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
/* eslint-disable */
|
|
2
|
+
import { Component, Input, Output, EventEmitter } from '@angular/core';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
export class NgChatOptionsComponent {
|
|
6
|
+
constructor() {
|
|
7
|
+
this.activeOptionTrackerChange = new EventEmitter();
|
|
8
|
+
}
|
|
9
|
+
onOptionClicked(option) {
|
|
10
|
+
option.isActive = true;
|
|
11
|
+
if (option.action) {
|
|
12
|
+
option.action(option.chattingTo);
|
|
13
|
+
}
|
|
14
|
+
this.activeOptionTrackerChange.emit(option);
|
|
15
|
+
}
|
|
16
|
+
}
|
|
17
|
+
NgChatOptionsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NgChatOptionsComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
18
|
+
NgChatOptionsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: NgChatOptionsComponent, selector: "ng-chat-options", inputs: { options: "options", activeOptionTracker: "activeOptionTracker" }, outputs: { activeOptionTrackerChange: "activeOptionTrackerChange" }, ngImport: i0, template: "<div *ngIf=\"options && options.length > 0\" class=\"ng-chat-options\">\n\t\t<button class=\"ng-chat-options-activator\">\n\t\t\t<span class=\"primary-text\">...</span>\n\t\t</button>\n\t<div class=\"ng-chat-options-content primary-background shadowed\">\n\t\t<a *ngFor=\"let option of options; let i = index\" [ngClass]=\"'primary-text'\" (click)=\"onOptionClicked(option)\">\n\t\t\t{{option.displayLabel}}\n\t\t</a>\n\t</div> \n</div>\n", styles: [".ng-chat-options-activator{background-color:unset;color:#fff;line-height:28px;border:none;position:relative}.ng-chat-options-activator>span{position:relative;top:-5px;left:0}.ng-chat-options{position:relative;display:inline-block}.ng-chat-options:hover .ng-chat-options-content{display:block}.ng-chat-options:hover .ng-chat-options-activator{background-color:#ddd}.ng-chat-options-content{display:none;position:absolute;min-width:160px;z-index:1}.ng-chat-options-content a:hover{background-color:#ddd}.ng-chat-options-content a{padding:6px 16px;text-decoration:none;display:block}@media only screen and (max-width: 581px){.ng-chat-options-content{right:0}}\n"], dependencies: [{ kind: "directive", type: i1.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }] });
|
|
19
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NgChatOptionsComponent, decorators: [{
|
|
20
|
+
type: Component,
|
|
21
|
+
args: [{ selector: 'ng-chat-options', template: "<div *ngIf=\"options && options.length > 0\" class=\"ng-chat-options\">\n\t\t<button class=\"ng-chat-options-activator\">\n\t\t\t<span class=\"primary-text\">...</span>\n\t\t</button>\n\t<div class=\"ng-chat-options-content primary-background shadowed\">\n\t\t<a *ngFor=\"let option of options; let i = index\" [ngClass]=\"'primary-text'\" (click)=\"onOptionClicked(option)\">\n\t\t\t{{option.displayLabel}}\n\t\t</a>\n\t</div> \n</div>\n", styles: [".ng-chat-options-activator{background-color:unset;color:#fff;line-height:28px;border:none;position:relative}.ng-chat-options-activator>span{position:relative;top:-5px;left:0}.ng-chat-options{position:relative;display:inline-block}.ng-chat-options:hover .ng-chat-options-content{display:block}.ng-chat-options:hover .ng-chat-options-activator{background-color:#ddd}.ng-chat-options-content{display:none;position:absolute;min-width:160px;z-index:1}.ng-chat-options-content a:hover{background-color:#ddd}.ng-chat-options-content a{padding:6px 16px;text-decoration:none;display:block}@media only screen and (max-width: 581px){.ng-chat-options-content{right:0}}\n"] }]
|
|
22
|
+
}], ctorParameters: function () { return []; }, propDecorators: { options: [{
|
|
23
|
+
type: Input
|
|
24
|
+
}], activeOptionTracker: [{
|
|
25
|
+
type: Input
|
|
26
|
+
}], activeOptionTrackerChange: [{
|
|
27
|
+
type: Output
|
|
28
|
+
}] } });
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibmctY2hhdC1vcHRpb25zLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2NoYXQvc3JjL2xpYi9jb21wb25lbnRzL25nLWNoYXQtb3B0aW9ucy9uZy1jaGF0LW9wdGlvbnMuY29tcG9uZW50LnRzIiwiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvY2hhdC9zcmMvbGliL2NvbXBvbmVudHMvbmctY2hhdC1vcHRpb25zL25nLWNoYXQtb3B0aW9ucy5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxvQkFBb0I7QUFDcEIsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxFQUFFLFlBQVksRUFBRSxNQUFNLGVBQWUsQ0FBQzs7O0FBUXZFLE1BQU0sT0FBTyxzQkFBc0I7SUFFbEM7UUFTTyw4QkFBeUIsR0FBOEIsSUFBSSxZQUFZLEVBQWUsQ0FBQztJQVQ5RSxDQUFDO0lBV2pCLGVBQWUsQ0FBQyxNQUFtQjtRQUVsQyxNQUFNLENBQUMsUUFBUSxHQUFHLElBQUksQ0FBQztRQUV2QixJQUFJLE1BQU0sQ0FBQyxNQUFNLEVBQUU7WUFDbEIsTUFBTSxDQUFDLE1BQU0sQ0FBQyxNQUFNLENBQUMsVUFBVSxDQUFDLENBQUM7U0FDakM7UUFFRCxJQUFJLENBQUMseUJBQXlCLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQzdDLENBQUM7O21IQXRCVyxzQkFBc0I7dUdBQXRCLHNCQUFzQix3TUNUbkMsNmJBVUE7MkZERGEsc0JBQXNCO2tCQUxsQyxTQUFTOytCQUNJLGlCQUFpQjswRUFTdkIsT0FBTztzQkFEYixLQUFLO2dCQUlDLG1CQUFtQjtzQkFEekIsS0FBSztnQkFJQyx5QkFBeUI7c0JBRC9CLE1BQU0iLCJzb3VyY2VzQ29udGVudCI6WyIvKiBlc2xpbnQtZGlzYWJsZSAqL1xuaW1wb3J0IHsgQ29tcG9uZW50LCBJbnB1dCwgT3V0cHV0LCBFdmVudEVtaXR0ZXIgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IElDaGF0T3B0aW9uIH0gZnJvbSAnLi4vLi4vY29yZS9jaGF0LW9wdGlvbic7XG5cbkBDb21wb25lbnQoe1xuICAgIHNlbGVjdG9yOiAnbmctY2hhdC1vcHRpb25zJyxcbiAgICB0ZW1wbGF0ZVVybDogJy4vbmctY2hhdC1vcHRpb25zLmNvbXBvbmVudC5odG1sJyxcbiAgICBzdHlsZVVybHM6IFsnLi9uZy1jaGF0LW9wdGlvbnMuY29tcG9uZW50LmNzcyddXG59KVxuZXhwb3J0IGNsYXNzIE5nQ2hhdE9wdGlvbnNDb21wb25lbnQge1xuXG5cdGNvbnN0cnVjdG9yKCkgeyB9XG5cblx0QElucHV0KClcblx0cHVibGljIG9wdGlvbnM6IElDaGF0T3B0aW9uW107XG5cblx0QElucHV0KClcblx0cHVibGljIGFjdGl2ZU9wdGlvblRyYWNrZXI6IElDaGF0T3B0aW9uO1xuXG5cdEBPdXRwdXQoKVxuXHRwdWJsaWMgYWN0aXZlT3B0aW9uVHJhY2tlckNoYW5nZTogRXZlbnRFbWl0dGVyPElDaGF0T3B0aW9uPiA9IG5ldyBFdmVudEVtaXR0ZXI8SUNoYXRPcHRpb24+KCk7XG5cblx0b25PcHRpb25DbGlja2VkKG9wdGlvbjogSUNoYXRPcHRpb24pOiB2b2lkXG5cdHtcblx0XHRvcHRpb24uaXNBY3RpdmUgPSB0cnVlO1xuXG5cdFx0aWYgKG9wdGlvbi5hY3Rpb24pIHtcblx0XHRcdG9wdGlvbi5hY3Rpb24ob3B0aW9uLmNoYXR0aW5nVG8pO1xuXHRcdH1cblxuXHRcdHRoaXMuYWN0aXZlT3B0aW9uVHJhY2tlckNoYW5nZS5lbWl0KG9wdGlvbik7XG5cdH1cbn1cbi8qIGVzbGludC1lbmFibGUgKi8iLCI8ZGl2ICpuZ0lmPVwib3B0aW9ucyAmJiBvcHRpb25zLmxlbmd0aCA+IDBcIiBjbGFzcz1cIm5nLWNoYXQtb3B0aW9uc1wiPlxuXHRcdDxidXR0b24gY2xhc3M9XCJuZy1jaGF0LW9wdGlvbnMtYWN0aXZhdG9yXCI+XG5cdFx0XHQ8c3BhbiBjbGFzcz1cInByaW1hcnktdGV4dFwiPi4uLjwvc3Bhbj5cblx0XHQ8L2J1dHRvbj5cblx0PGRpdiBjbGFzcz1cIm5nLWNoYXQtb3B0aW9ucy1jb250ZW50IHByaW1hcnktYmFja2dyb3VuZCBzaGFkb3dlZFwiPlxuXHRcdDxhICpuZ0Zvcj1cImxldCBvcHRpb24gb2Ygb3B0aW9uczsgbGV0IGkgPSBpbmRleFwiIFtuZ0NsYXNzXT1cIidwcmltYXJ5LXRleHQnXCIgKGNsaWNrKT1cIm9uT3B0aW9uQ2xpY2tlZChvcHRpb24pXCI+XG5cdFx0XHR7e29wdGlvbi5kaXNwbGF5TGFiZWx9fVxuXHRcdDwvYT5cblx0PC9kaXY+ICAgICAgXG48L2Rpdj5cbiJdfQ==
|
|
@@ -0,0 +1,247 @@
|
|
|
1
|
+
/* eslint-disable */
|
|
2
|
+
import { Component, Input, Output, EventEmitter, ViewEncapsulation, ViewChild } from '@angular/core';
|
|
3
|
+
import { Message } from "../../core/message";
|
|
4
|
+
import { MessageType } from "../../core/message-type.enum";
|
|
5
|
+
import { ChatParticipantStatus } from "../../core/chat-participant-status.enum";
|
|
6
|
+
import { ScrollDirection } from "../../core/scroll-direction.enum";
|
|
7
|
+
import { ChatParticipantType } from "../../core/chat-participant-type.enum";
|
|
8
|
+
import { MessageCounter } from "../../core/message-counter";
|
|
9
|
+
import { chatParticipantStatusDescriptor } from '../../core/chat-participant-status-descriptor';
|
|
10
|
+
import * as i0 from "@angular/core";
|
|
11
|
+
import * as i1 from "@angular/forms";
|
|
12
|
+
import * as i2 from "@angular/common";
|
|
13
|
+
import * as i3 from "../../pipes/emojify.pipe";
|
|
14
|
+
import * as i4 from "../../pipes/linkfy.pipe";
|
|
15
|
+
import * as i5 from "../../pipes/group-message-display-name.pipe";
|
|
16
|
+
export class NgChatWindowComponent {
|
|
17
|
+
constructor() {
|
|
18
|
+
this.emojisEnabled = true;
|
|
19
|
+
this.linkfyEnabled = true;
|
|
20
|
+
this.showMessageDate = true;
|
|
21
|
+
this.messageDatePipeFormat = "short";
|
|
22
|
+
this.hasPagedHistory = true;
|
|
23
|
+
this.onChatWindowClosed = new EventEmitter();
|
|
24
|
+
this.onMessagesSeen = new EventEmitter();
|
|
25
|
+
this.onMessageSent = new EventEmitter();
|
|
26
|
+
this.onTabTriggered = new EventEmitter();
|
|
27
|
+
this.onOptionTriggered = new EventEmitter();
|
|
28
|
+
this.onLoadHistoryTriggered = new EventEmitter();
|
|
29
|
+
// File upload state
|
|
30
|
+
this.fileUploadersInUse = []; // Id bucket of uploaders in use
|
|
31
|
+
// Exposes enums and functions for the ng-template
|
|
32
|
+
this.ChatParticipantType = ChatParticipantType;
|
|
33
|
+
this.ChatParticipantStatus = ChatParticipantStatus;
|
|
34
|
+
this.MessageType = MessageType;
|
|
35
|
+
this.chatParticipantStatusDescriptor = chatParticipantStatusDescriptor;
|
|
36
|
+
}
|
|
37
|
+
defaultWindowOptions(currentWindow) {
|
|
38
|
+
if (this.showOptions && currentWindow.participant.participantType == ChatParticipantType.User) {
|
|
39
|
+
return [{
|
|
40
|
+
isActive: false,
|
|
41
|
+
chattingTo: currentWindow,
|
|
42
|
+
validateContext: (participant) => {
|
|
43
|
+
return participant.participantType == ChatParticipantType.User;
|
|
44
|
+
},
|
|
45
|
+
displayLabel: 'Add People' // TODO: Localize this
|
|
46
|
+
}];
|
|
47
|
+
}
|
|
48
|
+
return [];
|
|
49
|
+
}
|
|
50
|
+
// Asserts if a user avatar is visible in a chat cluster
|
|
51
|
+
isAvatarVisible(window, message, index) {
|
|
52
|
+
if (message.fromId != this.userId) {
|
|
53
|
+
if (index == 0) {
|
|
54
|
+
return true; // First message, good to show the thumbnail
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
// Check if the previous message belongs to the same user, if it belongs there is no need to show the avatar again to form the message cluster
|
|
58
|
+
if (window.messages[index - 1].fromId != message.fromId) {
|
|
59
|
+
return true;
|
|
60
|
+
}
|
|
61
|
+
}
|
|
62
|
+
}
|
|
63
|
+
return false;
|
|
64
|
+
}
|
|
65
|
+
getChatWindowAvatar(participant, message) {
|
|
66
|
+
if (participant.participantType == ChatParticipantType.User) {
|
|
67
|
+
return participant.avatar;
|
|
68
|
+
}
|
|
69
|
+
else if (participant.participantType == ChatParticipantType.Group) {
|
|
70
|
+
let group = participant;
|
|
71
|
+
let userIndex = group.chattingTo.findIndex(x => x.id == message.fromId);
|
|
72
|
+
return group.chattingTo[userIndex >= 0 ? userIndex : 0].avatar;
|
|
73
|
+
}
|
|
74
|
+
return null;
|
|
75
|
+
}
|
|
76
|
+
isUploadingFile(window) {
|
|
77
|
+
const fileUploadInstanceId = this.getUniqueFileUploadInstanceId(window);
|
|
78
|
+
return this.fileUploadersInUse.indexOf(fileUploadInstanceId) > -1;
|
|
79
|
+
}
|
|
80
|
+
// Generates a unique file uploader id for each participant
|
|
81
|
+
getUniqueFileUploadInstanceId(window) {
|
|
82
|
+
if (window && window.participant) {
|
|
83
|
+
return `ng-chat-file-upload-${window.participant.id}`;
|
|
84
|
+
}
|
|
85
|
+
return 'ng-chat-file-upload';
|
|
86
|
+
}
|
|
87
|
+
unreadMessagesTotal(window) {
|
|
88
|
+
return MessageCounter.unreadMessagesTotal(window, this.userId);
|
|
89
|
+
}
|
|
90
|
+
// Scrolls a chat window message flow to the bottom
|
|
91
|
+
scrollChatWindow(window, direction) {
|
|
92
|
+
if (!window.isCollapsed) {
|
|
93
|
+
setTimeout(() => {
|
|
94
|
+
if (this.chatMessages) {
|
|
95
|
+
let element = this.chatMessages.nativeElement;
|
|
96
|
+
let position = (direction === ScrollDirection.Top) ? 0 : element.scrollHeight;
|
|
97
|
+
element.scrollTop = position;
|
|
98
|
+
}
|
|
99
|
+
});
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
activeOptionTrackerChange(option) {
|
|
103
|
+
this.onOptionTriggered.emit(option);
|
|
104
|
+
}
|
|
105
|
+
// Triggers native file upload for file selection from the user
|
|
106
|
+
triggerNativeFileUpload(window) {
|
|
107
|
+
if (window) {
|
|
108
|
+
if (this.nativeFileInput)
|
|
109
|
+
this.nativeFileInput.nativeElement.click();
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
// Toggles a window focus on the focus/blur of a 'newMessage' input
|
|
113
|
+
toggleWindowFocus(window) {
|
|
114
|
+
window.hasFocus = !window.hasFocus;
|
|
115
|
+
if (window.hasFocus) {
|
|
116
|
+
const unreadMessages = window.messages
|
|
117
|
+
.filter(message => message.dateSeen == null
|
|
118
|
+
&& (message.toId == this.userId || window.participant.participantType === ChatParticipantType.Group));
|
|
119
|
+
if (unreadMessages && unreadMessages.length > 0) {
|
|
120
|
+
this.onMessagesSeen.emit(unreadMessages);
|
|
121
|
+
}
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
markMessagesAsRead(messages) {
|
|
125
|
+
this.onMessagesSeen.emit(messages);
|
|
126
|
+
}
|
|
127
|
+
fetchMessageHistory(window) {
|
|
128
|
+
this.onLoadHistoryTriggered.emit(window);
|
|
129
|
+
}
|
|
130
|
+
// Closes a chat window via the close 'X' button
|
|
131
|
+
onCloseChatWindow() {
|
|
132
|
+
this.onChatWindowClosed.emit({ closedWindow: this.window, closedViaEscapeKey: false });
|
|
133
|
+
}
|
|
134
|
+
/* Monitors pressed keys on a chat window
|
|
135
|
+
- Dispatches a message when the ENTER key is pressed
|
|
136
|
+
- Tabs between windows on TAB or SHIFT + TAB
|
|
137
|
+
- Closes the current focused window on ESC
|
|
138
|
+
*/
|
|
139
|
+
onChatInputTyped(event, window) {
|
|
140
|
+
switch (event.keyCode) {
|
|
141
|
+
case 13:
|
|
142
|
+
if (window.newMessage && window.newMessage.trim() != "") {
|
|
143
|
+
let message = new Message();
|
|
144
|
+
message.fromId = this.userId;
|
|
145
|
+
message.toId = window.participant.id;
|
|
146
|
+
message.message = window.newMessage;
|
|
147
|
+
message.dateSent = new Date();
|
|
148
|
+
window.messages.push(message);
|
|
149
|
+
this.onMessageSent.emit(message);
|
|
150
|
+
window.newMessage = ""; // Resets the new message input
|
|
151
|
+
this.scrollChatWindow(window, ScrollDirection.Bottom);
|
|
152
|
+
}
|
|
153
|
+
break;
|
|
154
|
+
case 9:
|
|
155
|
+
event.preventDefault();
|
|
156
|
+
this.onTabTriggered.emit({ triggeringWindow: window, shiftKeyPressed: event.shiftKey });
|
|
157
|
+
break;
|
|
158
|
+
case 27:
|
|
159
|
+
this.onChatWindowClosed.emit({ closedWindow: window, closedViaEscapeKey: true });
|
|
160
|
+
break;
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
// Toggles a chat window visibility between maximized/minimized
|
|
164
|
+
onChatWindowClicked(window) {
|
|
165
|
+
window.isCollapsed = !window.isCollapsed;
|
|
166
|
+
this.scrollChatWindow(window, ScrollDirection.Bottom);
|
|
167
|
+
}
|
|
168
|
+
clearInUseFileUploader(fileUploadInstanceId) {
|
|
169
|
+
const uploaderInstanceIdIndex = this.fileUploadersInUse.indexOf(fileUploadInstanceId);
|
|
170
|
+
if (uploaderInstanceIdIndex > -1) {
|
|
171
|
+
this.fileUploadersInUse.splice(uploaderInstanceIdIndex, 1);
|
|
172
|
+
}
|
|
173
|
+
}
|
|
174
|
+
// Handles file selection and uploads the selected file using the file upload adapter
|
|
175
|
+
onFileChosen(window) {
|
|
176
|
+
const fileUploadInstanceId = this.getUniqueFileUploadInstanceId(window);
|
|
177
|
+
const uploadElementRef = this.nativeFileInput;
|
|
178
|
+
if (uploadElementRef) {
|
|
179
|
+
const file = uploadElementRef.nativeElement.files[0];
|
|
180
|
+
this.fileUploadersInUse.push(fileUploadInstanceId);
|
|
181
|
+
this.fileUploadAdapter.uploadFile(file, window.participant.id)
|
|
182
|
+
.subscribe(fileMessage => {
|
|
183
|
+
this.clearInUseFileUploader(fileUploadInstanceId);
|
|
184
|
+
fileMessage.fromId = this.userId;
|
|
185
|
+
// Push file message to current user window
|
|
186
|
+
window.messages.push(fileMessage);
|
|
187
|
+
this.onMessageSent.emit(fileMessage);
|
|
188
|
+
this.scrollChatWindow(window, ScrollDirection.Bottom);
|
|
189
|
+
// Resets the file upload element
|
|
190
|
+
uploadElementRef.nativeElement.value = '';
|
|
191
|
+
}, (error) => {
|
|
192
|
+
this.clearInUseFileUploader(fileUploadInstanceId);
|
|
193
|
+
// Resets the file upload element
|
|
194
|
+
uploadElementRef.nativeElement.value = '';
|
|
195
|
+
// TODO: Invoke a file upload adapter error here
|
|
196
|
+
});
|
|
197
|
+
}
|
|
198
|
+
}
|
|
199
|
+
}
|
|
200
|
+
NgChatWindowComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NgChatWindowComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
|
|
201
|
+
NgChatWindowComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "14.3.0", type: NgChatWindowComponent, selector: "ng-chat-window", inputs: { fileUploadAdapter: "fileUploadAdapter", window: "window", userId: "userId", localization: "localization", showOptions: "showOptions", emojisEnabled: "emojisEnabled", linkfyEnabled: "linkfyEnabled", showMessageDate: "showMessageDate", messageDatePipeFormat: "messageDatePipeFormat", hasPagedHistory: "hasPagedHistory" }, outputs: { onChatWindowClosed: "onChatWindowClosed", onMessagesSeen: "onMessagesSeen", onMessageSent: "onMessageSent", onTabTriggered: "onTabTriggered", onOptionTriggered: "onOptionTriggered", onLoadHistoryTriggered: "onLoadHistoryTriggered" }, viewQueries: [{ propertyName: "chatMessages", first: true, predicate: ["chatMessages"], descendants: true }, { propertyName: "nativeFileInput", first: true, predicate: ["nativeFileInput"], descendants: true }, { propertyName: "chatWindowInput", first: true, predicate: ["chatWindowInput"], descendants: true }], ngImport: i0, template: "<ng-container *ngIf=\"window && window.isCollapsed\">\n\t<div class=\"ng-chat-title secondary-background\">\n\t\t<div class=\"ng-chat-title-visibility-toggle-area\" (click)=\"onChatWindowClicked(window)\">\n\t\t\t<strong title=\"{{window.participant.displayName}}\">\n\t\t\t\t{{window.participant.displayName}}\n\t\t\t</strong>\n\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': window.participant.status === ChatParticipantStatus.Online, 'busy': window.participant.status === ChatParticipantStatus.Busy, 'away': window.participant.status === ChatParticipantStatus.Away, 'offline': window.participant.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(window.participant.status, localization)}}\"></span>\n\t\t\t<span *ngIf=\"unreadMessagesTotal(window).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotal(window)}}</span>\n\t\t</div>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-close primary-text\" (click)=\"onCloseChatWindow()\">✕</a>\n\t</div>\n</ng-container>\n<ng-container *ngIf=\"window && !window.isCollapsed\"> \n\t<div class=\"ng-chat-title secondary-background\">\n\t\t<div class=\"ng-chat-title-visibility-toggle-area\" (click)=\"onChatWindowClicked(window)\">\n\t\t\t<strong title=\"{{window.participant.displayName}}\">\n\t\t\t\t{{window.participant.displayName}}\n\t\t\t</strong>\n\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': window.participant.status === ChatParticipantStatus.Online, 'busy': window.participant.status === ChatParticipantStatus.Busy, 'away': window.participant.status === ChatParticipantStatus.Away, 'offline': window.participant.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(window.participant.status, localization)}}\"></span>\n\t\t\t<span *ngIf=\"unreadMessagesTotal(window).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotal(window)}}</span>\n\t\t</div>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-close primary-text\" (click)=\"onCloseChatWindow()\">✕</a>\n\t\t<!-- <ng-chat-options [ngClass]=\"'ng-chat-options-container'\" [options]=\"defaultWindowOptions(window)\" (activeOptionTrackerChange)=\"activeOptionTrackerChange($event)\"></ng-chat-options> -->\n\t</div>\n\t\t<div #chatMessages class=\"ng-chat-messages primary-background\">\n\t\t\t<div *ngIf=\"window.isLoadingHistory\" class=\"ng-chat-loading-wrapper\">\n\t\t\t\t<div class=\"loader\">Loading history...</div>\n\t\t\t</div>\n\t\t\t<div *ngIf=\"hasPagedHistory && window.hasMoreMessages && !window.isLoadingHistory\" class=\"ng-chat-load-history\">\n\t\t\t\t<a class=\"load-history-action\" (click)=\"fetchMessageHistory(window)\">{{localization.loadMessageHistoryPlaceholder}}</a>\n\t\t\t</div>\n\n\t\t\t<div *ngFor=\"let message of window.messages; let i = index\" [ngClass]=\"{'ng-chat-message': true, 'ng-chat-message-received': message.fromId !== userId}\">\n\t\t\t\t<ng-container *ngIf=\"isAvatarVisible(window, message, i)\">\n\t\t\t\t\t<div *ngIf=\"!getChatWindowAvatar(window.participant, message)\" class=\"icon-wrapper\">\n\t\t\t\t\t\t<i class=\"user-icon\"></i>\n\t\t\t\t\t</div>\n\t\t\t\t\t<!-- <img *ngIf=\"getChatWindowAvatar(window.participant, message)\" alt=\"\" class=\"avatar\" height=\"30\" width=\"30\" [src]=\"getChatWindowAvatar(window.participant, message) | sanitize\" /> -->\n\t\t\t\t\t<span *ngIf=\"window.participant.participantType === ChatParticipantType.Group\" class=\"ng-chat-participant-name\">{{window.participant | groupMessageDisplayName:message}}</span>\n\t\t\t\t</ng-container>\n\t\t\t\t<ng-container [ngSwitch]=\"message.type\">\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.Text\" [ngClass]=\"{'sent-chat-message-container': message.fromId === userId, 'received-chat-message-container': message.fromId !== userId}\">\n\t\t\t \t\t\t<span [innerHtml]=\"message.message | emojify:emojisEnabled | linkfy:linkfyEnabled\"></span>\n\t\t\t\t\t\t<span *ngIf=\"showMessageDate && message.dateSent\" class=\"message-sent-date\">{{message.dateSent | date:'dd'}}/{{message.dateSent | date:'MM'}}/{{message.dateSent | date:'y'}} {{message.dateSent | date:'h:mm a '}}</span>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.Image\" [ngClass]=\"{'sent-chat-message-container': message.fromId === userId, 'received-chat-message-container': message.fromId !== userId}\">\n\t\n\t\t\t \t\t\t<img src=\"{{message.message}}\" class=\"image-message\" />\n\t\n\t\t\t\t\t\t<span *ngIf=\"showMessageDate && message.dateSent\" class=\"message-sent-date\">{{message.dateSent | date:'dd'}}/{{message.dateSent | date:'MM'}}/{{message.dateSent | date:'y'}} {{message.dateSent | date:'h:mm a '}}</span>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.File\" [ngClass]=\"{'file-message-container': true, 'received': message.fromId !== userId}\">\n\t\t\t\t\t\t<div class=\"file-message-icon-container\">\n\t\t\t\t\t\t\t<i class=\"paperclip-icon\"></i>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<!-- <a class=\"file-details\" [attr.href]=\"message.downloadUrl\" target=\"_blank\" rel=\"noopener noreferrer\" (click)=\"this.markMessagesAsRead([message])\" download>\n\t\t\t\t\t\t\t<span class=\"file-message-title\" [attr.title]=\"message.message\">{{message.message}}</span>\n\t\t\t\t\t\t\t<span *ngIf=\"message.fileSizeInBytes\" class=\"file-message-size\">{{message.fileSizeInBytes}} Bytes</span>\n\t\t\t\t\t\t</a> -->\n\t\t\t\t\t</div>\n\t\t\t\t</ng-container>\n\t\t\t</div>\n\t</div>\n\n\t<div class=\"ng-chat-footer primary-outline-color primary-background\">\n\t\t<input #chatWindowInput\n\t\t\ttype=\"text\"\n\t\t\t[ngModel]=\"window.newMessage | emojify:emojisEnabled\"\n\t\t\t(ngModelChange)=\"window.newMessage=$event\"\n\t\t\t[placeholder]=\"localization.messagePlaceholder\"\n\t\t\t[ngClass]=\"{'chat-window-input': true, 'has-side-action': fileUploadAdapter}\"\n\t\t\t(keydown)=\"onChatInputTyped($event, window)\"\n\t\t\t(blur)=\"toggleWindowFocus(window)\"\n\t\t\t(focus)=\"toggleWindowFocus(window)\"/>\n\n\t\t<!-- File Upload -->\n\t\t<ng-container *ngIf=\"fileUploadAdapter\">\n\t\t\t<a *ngIf=\"!isUploadingFile(window)\" class=\"btn-add-file\" (click)=\"triggerNativeFileUpload(window)\">\n\t\t\t\t<i class=\"upload-icon\"></i>\n\t\t\t</a>\n\t\t\t<input\n\t\t\t\ttype=\"file\"\n\t\t\t\t#nativeFileInput\n\t\t\t\tstyle=\"display: none;\"\n\t\t\t\t[attr.id]=\"getUniqueFileUploadInstanceId(window)\"\n\t\t\t\t(change)=\"onFileChosen(window)\" />\n\t\t\t<div *ngIf=\"isUploadingFile(window)\" class=\"loader\"></div>\n\t\t</ng-container>\n\t</div>\n</ng-container>", styles: [".ng-chat-window{right:260px;height:360px;z-index:999;bottom:0;width:300px;position:fixed;border-width:1px;border-style:solid;border-bottom:0;box-shadow:0 4px 8px #00000040;margin-bottom:10px!important;border:1px solid grey!important;border-radius:10px!important;box-sizing:border-box!important;overflow:hidden!important;background-color:#fff;::-webkit-scrollbar {width: 8px !important;} ::-webkit-scrollbar-thumb {background: #1b1b70 !important; border-radius: 5px !important;}}.ng-chat-window-collapsed{height:30px!important}.ng-chat-window .ng-chat-footer{box-sizing:border-box!important;padding-bottom:10px!important;display:block;height:10%;width:95%!important;margin:auto!important;background-color:inherit!important}.ng-chat-window .ng-chat-footer>input{font-size:.8em;box-sizing:border-box!important;padding:15px!important;height:100%;width:100%;border-radius:20px!important;border:1px solid grey!important}.ng-chat-window .ng-chat-footer>input.has-side-action{width:calc(100% - 30px)}.ng-chat-window .ng-chat-footer .btn-add-file{position:absolute;right:5px;bottom:7px;height:20px;width:20px;cursor:pointer}.ng-chat-window .ng-chat-footer .loader{position:absolute;right:14px;bottom:8px}.ng-chat-window .ng-chat-load-history{height:30px;text-align:center;font-size:.8em}.ng-chat-window .ng-chat-load-history>a{border-radius:15px;cursor:pointer;padding:5px 10px}.ng-chat-window .ng-chat-messages{padding:10px;width:100%;height:calc(90% - 30px);box-sizing:border-box;position:relative;overflow:auto}.ng-chat-window .ng-chat-messages .ng-chat-message{clear:both}.ng-chat-window .ng-chat-messages .ng-chat-message>img.avatar,.ng-chat-window .ng-chat-messages .ng-chat-message>.icon-wrapper{position:absolute;left:10px;border-radius:25px}.ng-chat-window .ng-chat-messages .ng-chat-message .ng-chat-participant-name{display:inline-block;margin-left:40px;padding-bottom:5px;font-weight:700;font-size:.8em;text-overflow:ellipsis;max-width:180px}.ng-chat-window .ng-chat-messages .ng-chat-message>.icon-wrapper{background-color:#bababa;overflow:hidden;width:30px;height:30px;padding:0}.ng-chat-window .ng-chat-messages .ng-chat-message>.icon-wrapper>i{color:#fff;transform:scale(.7)}.ng-chat-window .ng-chat-messages .ng-chat-message .message-sent-date{font-size:.5em!important;display:block;text-align:right;margin-top:5px;color:#a1a0a0!important;font-style:italic!important}.ng-chat-window .ng-chat-messages .ng-chat-message>div{float:right;width:182px;border-radius:15px 15px 0!important;padding:10px;margin-top:0;margin-bottom:5px;font-size:.9em;word-wrap:break-word;background-color:#93d4f0!important}.ng-chat-window .ng-chat-message-received>div.received-chat-message-container{float:left;margin-left:40px;padding-top:7px;padding-bottom:7px;margin-top:0;margin-bottom:5px;background-color:#daf2fc!important;border-radius:15px 15px 15px 0!important}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container{float:right;width:202px;border-style:solid;border-width:3px;border-radius:5px;overflow:hidden;margin-bottom:5px;display:block;text-decoration:none;font-size:.9em;padding:0;box-sizing:border-box}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container.received{float:left;margin-left:40px;width:208px}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-message-icon-container{width:20px;height:35px;padding:10px 5px;float:left}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-message-icon-container i{margin-top:8px}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details{float:left;padding:10px;width:calc(100% - 60px);color:currentColor;text-decoration:none}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details:hover{text-decoration:underline}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details span{display:block;width:100%;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details .file-message-title{font-weight:700}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details .file-message-size{font-size:.8em;margin-top:5px}.ng-chat-window .image-message{width:100%;height:auto}@media only screen and (max-width: 581px){.ng-chat-window{position:initial}}\n"], dependencies: [{ kind: "directive", type: i1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { 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.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i2.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "pipe", type: i2.DatePipe, name: "date" }, { kind: "pipe", type: i3.EmojifyPipe, name: "emojify" }, { kind: "pipe", type: i4.LinkfyPipe, name: "linkfy" }, { kind: "pipe", type: i5.GroupMessageDisplayNamePipe, name: "groupMessageDisplayName" }], encapsulation: i0.ViewEncapsulation.None });
|
|
202
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: NgChatWindowComponent, decorators: [{
|
|
203
|
+
type: Component,
|
|
204
|
+
args: [{ selector: 'ng-chat-window', encapsulation: ViewEncapsulation.None, template: "<ng-container *ngIf=\"window && window.isCollapsed\">\n\t<div class=\"ng-chat-title secondary-background\">\n\t\t<div class=\"ng-chat-title-visibility-toggle-area\" (click)=\"onChatWindowClicked(window)\">\n\t\t\t<strong title=\"{{window.participant.displayName}}\">\n\t\t\t\t{{window.participant.displayName}}\n\t\t\t</strong>\n\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': window.participant.status === ChatParticipantStatus.Online, 'busy': window.participant.status === ChatParticipantStatus.Busy, 'away': window.participant.status === ChatParticipantStatus.Away, 'offline': window.participant.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(window.participant.status, localization)}}\"></span>\n\t\t\t<span *ngIf=\"unreadMessagesTotal(window).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotal(window)}}</span>\n\t\t</div>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-close primary-text\" (click)=\"onCloseChatWindow()\">✕</a>\n\t</div>\n</ng-container>\n<ng-container *ngIf=\"window && !window.isCollapsed\"> \n\t<div class=\"ng-chat-title secondary-background\">\n\t\t<div class=\"ng-chat-title-visibility-toggle-area\" (click)=\"onChatWindowClicked(window)\">\n\t\t\t<strong title=\"{{window.participant.displayName}}\">\n\t\t\t\t{{window.participant.displayName}}\n\t\t\t</strong>\n\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': window.participant.status === ChatParticipantStatus.Online, 'busy': window.participant.status === ChatParticipantStatus.Busy, 'away': window.participant.status === ChatParticipantStatus.Away, 'offline': window.participant.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(window.participant.status, localization)}}\"></span>\n\t\t\t<span *ngIf=\"unreadMessagesTotal(window).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotal(window)}}</span>\n\t\t</div>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-close primary-text\" (click)=\"onCloseChatWindow()\">✕</a>\n\t\t<!-- <ng-chat-options [ngClass]=\"'ng-chat-options-container'\" [options]=\"defaultWindowOptions(window)\" (activeOptionTrackerChange)=\"activeOptionTrackerChange($event)\"></ng-chat-options> -->\n\t</div>\n\t\t<div #chatMessages class=\"ng-chat-messages primary-background\">\n\t\t\t<div *ngIf=\"window.isLoadingHistory\" class=\"ng-chat-loading-wrapper\">\n\t\t\t\t<div class=\"loader\">Loading history...</div>\n\t\t\t</div>\n\t\t\t<div *ngIf=\"hasPagedHistory && window.hasMoreMessages && !window.isLoadingHistory\" class=\"ng-chat-load-history\">\n\t\t\t\t<a class=\"load-history-action\" (click)=\"fetchMessageHistory(window)\">{{localization.loadMessageHistoryPlaceholder}}</a>\n\t\t\t</div>\n\n\t\t\t<div *ngFor=\"let message of window.messages; let i = index\" [ngClass]=\"{'ng-chat-message': true, 'ng-chat-message-received': message.fromId !== userId}\">\n\t\t\t\t<ng-container *ngIf=\"isAvatarVisible(window, message, i)\">\n\t\t\t\t\t<div *ngIf=\"!getChatWindowAvatar(window.participant, message)\" class=\"icon-wrapper\">\n\t\t\t\t\t\t<i class=\"user-icon\"></i>\n\t\t\t\t\t</div>\n\t\t\t\t\t<!-- <img *ngIf=\"getChatWindowAvatar(window.participant, message)\" alt=\"\" class=\"avatar\" height=\"30\" width=\"30\" [src]=\"getChatWindowAvatar(window.participant, message) | sanitize\" /> -->\n\t\t\t\t\t<span *ngIf=\"window.participant.participantType === ChatParticipantType.Group\" class=\"ng-chat-participant-name\">{{window.participant | groupMessageDisplayName:message}}</span>\n\t\t\t\t</ng-container>\n\t\t\t\t<ng-container [ngSwitch]=\"message.type\">\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.Text\" [ngClass]=\"{'sent-chat-message-container': message.fromId === userId, 'received-chat-message-container': message.fromId !== userId}\">\n\t\t\t \t\t\t<span [innerHtml]=\"message.message | emojify:emojisEnabled | linkfy:linkfyEnabled\"></span>\n\t\t\t\t\t\t<span *ngIf=\"showMessageDate && message.dateSent\" class=\"message-sent-date\">{{message.dateSent | date:'dd'}}/{{message.dateSent | date:'MM'}}/{{message.dateSent | date:'y'}} {{message.dateSent | date:'h:mm a '}}</span>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.Image\" [ngClass]=\"{'sent-chat-message-container': message.fromId === userId, 'received-chat-message-container': message.fromId !== userId}\">\n\t\n\t\t\t \t\t\t<img src=\"{{message.message}}\" class=\"image-message\" />\n\t\n\t\t\t\t\t\t<span *ngIf=\"showMessageDate && message.dateSent\" class=\"message-sent-date\">{{message.dateSent | date:'dd'}}/{{message.dateSent | date:'MM'}}/{{message.dateSent | date:'y'}} {{message.dateSent | date:'h:mm a '}}</span>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.File\" [ngClass]=\"{'file-message-container': true, 'received': message.fromId !== userId}\">\n\t\t\t\t\t\t<div class=\"file-message-icon-container\">\n\t\t\t\t\t\t\t<i class=\"paperclip-icon\"></i>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<!-- <a class=\"file-details\" [attr.href]=\"message.downloadUrl\" target=\"_blank\" rel=\"noopener noreferrer\" (click)=\"this.markMessagesAsRead([message])\" download>\n\t\t\t\t\t\t\t<span class=\"file-message-title\" [attr.title]=\"message.message\">{{message.message}}</span>\n\t\t\t\t\t\t\t<span *ngIf=\"message.fileSizeInBytes\" class=\"file-message-size\">{{message.fileSizeInBytes}} Bytes</span>\n\t\t\t\t\t\t</a> -->\n\t\t\t\t\t</div>\n\t\t\t\t</ng-container>\n\t\t\t</div>\n\t</div>\n\n\t<div class=\"ng-chat-footer primary-outline-color primary-background\">\n\t\t<input #chatWindowInput\n\t\t\ttype=\"text\"\n\t\t\t[ngModel]=\"window.newMessage | emojify:emojisEnabled\"\n\t\t\t(ngModelChange)=\"window.newMessage=$event\"\n\t\t\t[placeholder]=\"localization.messagePlaceholder\"\n\t\t\t[ngClass]=\"{'chat-window-input': true, 'has-side-action': fileUploadAdapter}\"\n\t\t\t(keydown)=\"onChatInputTyped($event, window)\"\n\t\t\t(blur)=\"toggleWindowFocus(window)\"\n\t\t\t(focus)=\"toggleWindowFocus(window)\"/>\n\n\t\t<!-- File Upload -->\n\t\t<ng-container *ngIf=\"fileUploadAdapter\">\n\t\t\t<a *ngIf=\"!isUploadingFile(window)\" class=\"btn-add-file\" (click)=\"triggerNativeFileUpload(window)\">\n\t\t\t\t<i class=\"upload-icon\"></i>\n\t\t\t</a>\n\t\t\t<input\n\t\t\t\ttype=\"file\"\n\t\t\t\t#nativeFileInput\n\t\t\t\tstyle=\"display: none;\"\n\t\t\t\t[attr.id]=\"getUniqueFileUploadInstanceId(window)\"\n\t\t\t\t(change)=\"onFileChosen(window)\" />\n\t\t\t<div *ngIf=\"isUploadingFile(window)\" class=\"loader\"></div>\n\t\t</ng-container>\n\t</div>\n</ng-container>", styles: [".ng-chat-window{right:260px;height:360px;z-index:999;bottom:0;width:300px;position:fixed;border-width:1px;border-style:solid;border-bottom:0;box-shadow:0 4px 8px #00000040;margin-bottom:10px!important;border:1px solid grey!important;border-radius:10px!important;box-sizing:border-box!important;overflow:hidden!important;background-color:#fff;::-webkit-scrollbar {width: 8px !important;} ::-webkit-scrollbar-thumb {background: #1b1b70 !important; border-radius: 5px !important;}}.ng-chat-window-collapsed{height:30px!important}.ng-chat-window .ng-chat-footer{box-sizing:border-box!important;padding-bottom:10px!important;display:block;height:10%;width:95%!important;margin:auto!important;background-color:inherit!important}.ng-chat-window .ng-chat-footer>input{font-size:.8em;box-sizing:border-box!important;padding:15px!important;height:100%;width:100%;border-radius:20px!important;border:1px solid grey!important}.ng-chat-window .ng-chat-footer>input.has-side-action{width:calc(100% - 30px)}.ng-chat-window .ng-chat-footer .btn-add-file{position:absolute;right:5px;bottom:7px;height:20px;width:20px;cursor:pointer}.ng-chat-window .ng-chat-footer .loader{position:absolute;right:14px;bottom:8px}.ng-chat-window .ng-chat-load-history{height:30px;text-align:center;font-size:.8em}.ng-chat-window .ng-chat-load-history>a{border-radius:15px;cursor:pointer;padding:5px 10px}.ng-chat-window .ng-chat-messages{padding:10px;width:100%;height:calc(90% - 30px);box-sizing:border-box;position:relative;overflow:auto}.ng-chat-window .ng-chat-messages .ng-chat-message{clear:both}.ng-chat-window .ng-chat-messages .ng-chat-message>img.avatar,.ng-chat-window .ng-chat-messages .ng-chat-message>.icon-wrapper{position:absolute;left:10px;border-radius:25px}.ng-chat-window .ng-chat-messages .ng-chat-message .ng-chat-participant-name{display:inline-block;margin-left:40px;padding-bottom:5px;font-weight:700;font-size:.8em;text-overflow:ellipsis;max-width:180px}.ng-chat-window .ng-chat-messages .ng-chat-message>.icon-wrapper{background-color:#bababa;overflow:hidden;width:30px;height:30px;padding:0}.ng-chat-window .ng-chat-messages .ng-chat-message>.icon-wrapper>i{color:#fff;transform:scale(.7)}.ng-chat-window .ng-chat-messages .ng-chat-message .message-sent-date{font-size:.5em!important;display:block;text-align:right;margin-top:5px;color:#a1a0a0!important;font-style:italic!important}.ng-chat-window .ng-chat-messages .ng-chat-message>div{float:right;width:182px;border-radius:15px 15px 0!important;padding:10px;margin-top:0;margin-bottom:5px;font-size:.9em;word-wrap:break-word;background-color:#93d4f0!important}.ng-chat-window .ng-chat-message-received>div.received-chat-message-container{float:left;margin-left:40px;padding-top:7px;padding-bottom:7px;margin-top:0;margin-bottom:5px;background-color:#daf2fc!important;border-radius:15px 15px 15px 0!important}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container{float:right;width:202px;border-style:solid;border-width:3px;border-radius:5px;overflow:hidden;margin-bottom:5px;display:block;text-decoration:none;font-size:.9em;padding:0;box-sizing:border-box}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container.received{float:left;margin-left:40px;width:208px}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-message-icon-container{width:20px;height:35px;padding:10px 5px;float:left}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-message-icon-container i{margin-top:8px}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details{float:left;padding:10px;width:calc(100% - 60px);color:currentColor;text-decoration:none}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details:hover{text-decoration:underline}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details span{display:block;width:100%;text-overflow:ellipsis;overflow:hidden;white-space:nowrap}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details .file-message-title{font-weight:700}.ng-chat-window .ng-chat-messages .ng-chat-message .file-message-container>.file-details .file-message-size{font-size:.8em;margin-top:5px}.ng-chat-window .image-message{width:100%;height:auto}@media only screen and (max-width: 581px){.ng-chat-window{position:initial}}\n"] }]
|
|
205
|
+
}], ctorParameters: function () { return []; }, propDecorators: { fileUploadAdapter: [{
|
|
206
|
+
type: Input
|
|
207
|
+
}], window: [{
|
|
208
|
+
type: Input
|
|
209
|
+
}], userId: [{
|
|
210
|
+
type: Input
|
|
211
|
+
}], localization: [{
|
|
212
|
+
type: Input
|
|
213
|
+
}], showOptions: [{
|
|
214
|
+
type: Input
|
|
215
|
+
}], emojisEnabled: [{
|
|
216
|
+
type: Input
|
|
217
|
+
}], linkfyEnabled: [{
|
|
218
|
+
type: Input
|
|
219
|
+
}], showMessageDate: [{
|
|
220
|
+
type: Input
|
|
221
|
+
}], messageDatePipeFormat: [{
|
|
222
|
+
type: Input
|
|
223
|
+
}], hasPagedHistory: [{
|
|
224
|
+
type: Input
|
|
225
|
+
}], onChatWindowClosed: [{
|
|
226
|
+
type: Output
|
|
227
|
+
}], onMessagesSeen: [{
|
|
228
|
+
type: Output
|
|
229
|
+
}], onMessageSent: [{
|
|
230
|
+
type: Output
|
|
231
|
+
}], onTabTriggered: [{
|
|
232
|
+
type: Output
|
|
233
|
+
}], onOptionTriggered: [{
|
|
234
|
+
type: Output
|
|
235
|
+
}], onLoadHistoryTriggered: [{
|
|
236
|
+
type: Output
|
|
237
|
+
}], chatMessages: [{
|
|
238
|
+
type: ViewChild,
|
|
239
|
+
args: ['chatMessages']
|
|
240
|
+
}], nativeFileInput: [{
|
|
241
|
+
type: ViewChild,
|
|
242
|
+
args: ['nativeFileInput']
|
|
243
|
+
}], chatWindowInput: [{
|
|
244
|
+
type: ViewChild,
|
|
245
|
+
args: ['chatWindowInput']
|
|
246
|
+
}] } });
|
|
247
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-chat-window.component.js","sourceRoot":"","sources":["../../../../../../../projects/chat/src/lib/components/ng-chat-window/ng-chat-window.component.ts","../../../../../../../projects/chat/src/lib/components/ng-chat-window/ng-chat-window.component.html"],"names":[],"mappings":"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","sourcesContent":["/* eslint-disable */\nimport { Component, Input, Output, EventEmitter, ViewEncapsulation, ViewChild, ElementRef } from '@angular/core';\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 } from '../../core/localization';\nimport { IFileUploadAdapter } from '../../core/file-upload-adapter';\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 { MessageCounter } from \"../../core/message-counter\";\nimport { chatParticipantStatusDescriptor } from '../../core/chat-participant-status-descriptor';\n\n@Component({\n    selector: 'ng-chat-window',\n    templateUrl: './ng-chat-window.component.html',\n    styleUrls: ['./ng-chat-window.component.css'],\n    encapsulation: ViewEncapsulation.None\n})\nexport class NgChatWindowComponent {\n    constructor() { }\n\n    @Input()\n    public fileUploadAdapter: IFileUploadAdapter;\n\n    @Input()\n    public window: Window;\n\n    @Input()\n    public userId: any;\n\n    @Input()\n    public localization: Localization;\n\n    @Input()\n    public showOptions: boolean;\n\n    @Input()\n    public emojisEnabled: boolean = true;\n\n    @Input()\n    public linkfyEnabled: boolean = true;\n\n    @Input()\n    public showMessageDate: boolean = true;\n\n    @Input()\n    public messageDatePipeFormat: string = \"short\";\n\n    @Input()\n    public hasPagedHistory: boolean = true;\n\n    @Output()\n    public onChatWindowClosed: EventEmitter<{ closedWindow: Window, closedViaEscapeKey: boolean}> = new EventEmitter();\n\n    @Output()\n    public onMessagesSeen: EventEmitter<Message[]> = new EventEmitter();\n\n    @Output()\n    public onMessageSent: EventEmitter<Message> = new EventEmitter();\n\n    @Output()\n    public onTabTriggered: EventEmitter<{ triggeringWindow: Window, shiftKeyPressed: boolean }> = new EventEmitter();\n\n    @Output()\n    public onOptionTriggered: EventEmitter<IChatOption> = new EventEmitter();\n\n    @Output()\n    public onLoadHistoryTriggered: EventEmitter<Window> = new EventEmitter();\n\n    @ViewChild('chatMessages') chatMessages: any;\n    @ViewChild('nativeFileInput') nativeFileInput: ElementRef;\n    @ViewChild('chatWindowInput') chatWindowInput: any;\n\n    // File upload state\n    public fileUploadersInUse: string[] = []; // Id bucket of uploaders in use\n\n    // Exposes enums and functions for the ng-template\n    public ChatParticipantType = ChatParticipantType;\n    public ChatParticipantStatus = ChatParticipantStatus;\n    public MessageType = MessageType;\n    public chatParticipantStatusDescriptor = chatParticipantStatusDescriptor;\n\n    defaultWindowOptions(currentWindow: Window): IChatOption[]\n    {\n        if (this.showOptions && currentWindow.participant.participantType == ChatParticipantType.User)\n        {\n            return [{\n                isActive: false,\n                chattingTo: currentWindow,\n                validateContext: (participant: IChatParticipant) => {\n                    return participant.participantType == ChatParticipantType.User;\n                },\n                displayLabel: 'Add People' // TODO: Localize this\n            }];\n        }\n\n        return [];\n    }\n\n    // Asserts if a user avatar is visible in a chat cluster\n    isAvatarVisible(window: Window, message: Message, index: number): boolean\n    {\n        if (message.fromId != this.userId){\n            if (index == 0){\n                return true; // First message, good to show the thumbnail\n            }\n            else{\n                // Check if the previous message belongs to the same user, if it belongs there is no need to show the avatar again to form the message cluster\n                if (window.messages[index - 1].fromId != message.fromId){\n                    return true;\n                }\n            }\n        }\n\n        return false;\n    }\n\n    getChatWindowAvatar(participant: IChatParticipant, message: Message): string | null\n    {\n        if (participant.participantType == ChatParticipantType.User)\n        {\n            return participant.avatar;\n        }\n        else if (participant.participantType == ChatParticipantType.Group)\n        {\n            let group = participant as Group;\n            let userIndex = group.chattingTo.findIndex(x => x.id == message.fromId);\n\n            return group.chattingTo[userIndex >= 0 ? userIndex : 0].avatar;\n        }\n\n        return null;\n    }\n\n    isUploadingFile(window: Window): boolean\n    {\n        const fileUploadInstanceId = this.getUniqueFileUploadInstanceId(window);\n\n        return this.fileUploadersInUse.indexOf(fileUploadInstanceId) > -1;\n    }\n\n    // Generates a unique file uploader id for each participant\n    getUniqueFileUploadInstanceId(window: Window): string\n    {\n        if (window && window.participant)\n        {\n            return `ng-chat-file-upload-${window.participant.id}`;\n        }\n        \n        return 'ng-chat-file-upload';\n    }\n\n    unreadMessagesTotal(window: Window): string\n    {\n        return MessageCounter.unreadMessagesTotal(window, this.userId);\n    }\n\n    // Scrolls a chat window message flow to the bottom\n    scrollChatWindow(window: Window, direction: ScrollDirection): void\n    {\n        if (!window.isCollapsed){\n            setTimeout(() => {\n                if (this.chatMessages){\n                    let element = this.chatMessages.nativeElement;\n                    let position = ( direction === ScrollDirection.Top ) ? 0 : element.scrollHeight;\n                    element.scrollTop = position;\n                }\n            });\n        }\n    }\n\n    activeOptionTrackerChange(option: IChatOption): void {\n        this.onOptionTriggered.emit(option);\n    }\n\n    // Triggers native file upload for file selection from the user\n    triggerNativeFileUpload(window: Window): void\n    {\n        if (window)\n        {\n            if (this.nativeFileInput) this.nativeFileInput.nativeElement.click();\n        }\n    }\n\n    // Toggles a window focus on the focus/blur of a 'newMessage' input\n    toggleWindowFocus(window: Window): void\n    {\n        window.hasFocus = !window.hasFocus;\n        if(window.hasFocus) {\n            const unreadMessages = window.messages\n                .filter(message => message.dateSeen == null\n                    && (message.toId == this.userId || window.participant.participantType === ChatParticipantType.Group));\n            \n            if (unreadMessages && unreadMessages.length > 0)\n            {\n                this.onMessagesSeen.emit(unreadMessages);\n            }\n        }\n    }\n\n    markMessagesAsRead(messages: Message[]): void\n    {\n        this.onMessagesSeen.emit(messages);\n    }\n\n    fetchMessageHistory(window: Window): void {\n        this.onLoadHistoryTriggered.emit(window);\n    }\n\n    // Closes a chat window via the close 'X' button\n    onCloseChatWindow(): void\n    {\n        this.onChatWindowClosed.emit({ closedWindow: this.window, closedViaEscapeKey: false });\n    }\n\n    /*  Monitors pressed keys on a chat window\n        - Dispatches a message when the ENTER key is pressed\n        - Tabs between windows on TAB or SHIFT + TAB\n        - Closes the current focused window on ESC\n    */\n   onChatInputTyped(event: any, window: Window): void\n   {\n       switch (event.keyCode)\n       {\n           case 13:\n               if (window.newMessage && window.newMessage.trim() != \"\")\n               {\n                   let message = new Message();\n            \n                   message.fromId = this.userId;\n                   message.toId = window.participant.id;\n                   message.message = window.newMessage;\n                   message.dateSent = new Date();\n       \n                   window.messages.push(message);\n       \n                   this.onMessageSent.emit(message);\n       \n                   window.newMessage = \"\"; // Resets the new message input\n       \n                   this.scrollChatWindow(window, ScrollDirection.Bottom);\n               }\n               break;\n           case 9:\n               event.preventDefault();\n\n               this.onTabTriggered.emit({ triggeringWindow: window, shiftKeyPressed: event.shiftKey });\n\n               break;\n           case 27:\n               this.onChatWindowClosed.emit({ closedWindow: window, closedViaEscapeKey: true });\n\n               break;\n       }\n   }\n\n    // Toggles a chat window visibility between maximized/minimized\n    onChatWindowClicked(window: Window): void\n    {\n        window.isCollapsed = !window.isCollapsed;\n        this.scrollChatWindow(window, ScrollDirection.Bottom);\n    }\n\n    private clearInUseFileUploader(fileUploadInstanceId: string): void\n    {\n        const uploaderInstanceIdIndex = this.fileUploadersInUse.indexOf(fileUploadInstanceId);\n\n        if (uploaderInstanceIdIndex > -1) {\n            this.fileUploadersInUse.splice(uploaderInstanceIdIndex, 1);\n        }\n    }\n\n    // Handles file selection and uploads the selected file using the file upload adapter\n    onFileChosen(window: Window): void {\n        const fileUploadInstanceId = this.getUniqueFileUploadInstanceId(window);\n        const uploadElementRef = this.nativeFileInput;\n\n        if (uploadElementRef)\n        {\n            const file: File = uploadElementRef.nativeElement.files[0];\n\n            this.fileUploadersInUse.push(fileUploadInstanceId);\n\n            this.fileUploadAdapter.uploadFile(file, window.participant.id)\n                .subscribe(fileMessage => {\n                    this.clearInUseFileUploader(fileUploadInstanceId);\n\n                    fileMessage.fromId = this.userId;\n\n                    // Push file message to current user window\n                    window.messages.push(fileMessage);\n        \n                    this.onMessageSent.emit(fileMessage);\n        \n                    this.scrollChatWindow(window, ScrollDirection.Bottom);\n\n                    // Resets the file upload element\n                    uploadElementRef.nativeElement.value = '';\n                }, (error) => {\n                    this.clearInUseFileUploader(fileUploadInstanceId);\n\n                    // Resets the file upload element\n                    uploadElementRef.nativeElement.value = '';\n\n                    // TODO: Invoke a file upload adapter error here\n                });\n        }\n    }\n}\n\n/* eslint-enable */","<ng-container *ngIf=\"window && window.isCollapsed\">\n\t<div class=\"ng-chat-title secondary-background\">\n\t\t<div class=\"ng-chat-title-visibility-toggle-area\" (click)=\"onChatWindowClicked(window)\">\n\t\t\t<strong title=\"{{window.participant.displayName}}\">\n\t\t\t\t{{window.participant.displayName}}\n\t\t\t</strong>\n\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': window.participant.status === ChatParticipantStatus.Online, 'busy': window.participant.status === ChatParticipantStatus.Busy, 'away': window.participant.status === ChatParticipantStatus.Away, 'offline': window.participant.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(window.participant.status, localization)}}\"></span>\n\t\t\t<span *ngIf=\"unreadMessagesTotal(window).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotal(window)}}</span>\n\t\t</div>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-close primary-text\" (click)=\"onCloseChatWindow()\">&#x2715;</a>\n\t</div>\n</ng-container>\n<ng-container *ngIf=\"window && !window.isCollapsed\">  \n\t<div class=\"ng-chat-title secondary-background\">\n\t\t<div class=\"ng-chat-title-visibility-toggle-area\" (click)=\"onChatWindowClicked(window)\">\n\t\t\t<strong title=\"{{window.participant.displayName}}\">\n\t\t\t\t{{window.participant.displayName}}\n\t\t\t</strong>\n\t\t\t<span [ngClass]=\"{'ng-chat-participant-status': true, 'online': window.participant.status === ChatParticipantStatus.Online, 'busy': window.participant.status === ChatParticipantStatus.Busy, 'away': window.participant.status === ChatParticipantStatus.Away, 'offline': window.participant.status === ChatParticipantStatus.Offline}\" title=\"{{chatParticipantStatusDescriptor(window.participant.status, localization)}}\"></span>\n\t\t\t<span *ngIf=\"unreadMessagesTotal(window).length > 0\" class=\"ng-chat-unread-messages-count unread-messages-counter-container primary-text\">{{unreadMessagesTotal(window)}}</span>\n\t\t</div>\n\t\t<a href=\"javascript:void(0);\" class=\"ng-chat-close primary-text\" (click)=\"onCloseChatWindow()\">&#x2715;</a>\n\t\t<!-- <ng-chat-options [ngClass]=\"'ng-chat-options-container'\" [options]=\"defaultWindowOptions(window)\" (activeOptionTrackerChange)=\"activeOptionTrackerChange($event)\"></ng-chat-options> -->\n\t</div>\n\t\t<div #chatMessages class=\"ng-chat-messages primary-background\">\n\t\t\t<div *ngIf=\"window.isLoadingHistory\" class=\"ng-chat-loading-wrapper\">\n\t\t\t\t<div class=\"loader\">Loading history...</div>\n\t\t\t</div>\n\t\t\t<div *ngIf=\"hasPagedHistory && window.hasMoreMessages && !window.isLoadingHistory\" class=\"ng-chat-load-history\">\n\t\t\t\t<a class=\"load-history-action\" (click)=\"fetchMessageHistory(window)\">{{localization.loadMessageHistoryPlaceholder}}</a>\n\t\t\t</div>\n\n\t\t\t<div *ngFor=\"let message of window.messages; let i = index\" [ngClass]=\"{'ng-chat-message': true, 'ng-chat-message-received': message.fromId !== userId}\">\n\t\t\t\t<ng-container *ngIf=\"isAvatarVisible(window, message, i)\">\n\t\t\t\t\t<div *ngIf=\"!getChatWindowAvatar(window.participant, message)\" class=\"icon-wrapper\">\n\t\t\t\t\t\t<i class=\"user-icon\"></i>\n\t\t\t\t\t</div>\n\t\t\t\t\t<!-- <img *ngIf=\"getChatWindowAvatar(window.participant, message)\" alt=\"\" class=\"avatar\" height=\"30\" width=\"30\" [src]=\"getChatWindowAvatar(window.participant, message) | sanitize\" /> -->\n\t\t\t\t\t<span *ngIf=\"window.participant.participantType === ChatParticipantType.Group\" class=\"ng-chat-participant-name\">{{window.participant | groupMessageDisplayName:message}}</span>\n\t\t\t\t</ng-container>\n\t\t\t\t<ng-container [ngSwitch]=\"message.type\">\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.Text\" [ngClass]=\"{'sent-chat-message-container': message.fromId === userId, 'received-chat-message-container': message.fromId !== userId}\">\n\t\t\t  \t\t\t<span [innerHtml]=\"message.message | emojify:emojisEnabled | linkfy:linkfyEnabled\"></span>\n\t\t\t\t\t\t<span *ngIf=\"showMessageDate && message.dateSent\" class=\"message-sent-date\">{{message.dateSent | date:'dd'}}/{{message.dateSent | date:'MM'}}/{{message.dateSent | date:'y'}} {{message.dateSent | date:'h:mm a '}}</span>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.Image\" [ngClass]=\"{'sent-chat-message-container': message.fromId === userId, 'received-chat-message-container': message.fromId !== userId}\">\n\t\n\t\t\t  \t\t\t<img src=\"{{message.message}}\" class=\"image-message\" />\n\t\n\t\t\t\t\t\t<span *ngIf=\"showMessageDate && message.dateSent\" class=\"message-sent-date\">{{message.dateSent | date:'dd'}}/{{message.dateSent | date:'MM'}}/{{message.dateSent | date:'y'}} {{message.dateSent | date:'h:mm a '}}</span>\n\t\t\t\t\t</div>\n\t\t\t\t\t<div *ngSwitchCase=\"MessageType.File\" [ngClass]=\"{'file-message-container': true, 'received': message.fromId !== userId}\">\n\t\t\t\t\t\t<div class=\"file-message-icon-container\">\n\t\t\t\t\t\t\t<i class=\"paperclip-icon\"></i>\n\t\t\t\t\t\t</div>\n\t\t\t\t\t\t<!-- <a class=\"file-details\" [attr.href]=\"message.downloadUrl\" target=\"_blank\" rel=\"noopener noreferrer\" (click)=\"this.markMessagesAsRead([message])\" download>\n\t\t\t\t\t\t\t<span class=\"file-message-title\" [attr.title]=\"message.message\">{{message.message}}</span>\n\t\t\t\t\t\t\t<span *ngIf=\"message.fileSizeInBytes\"  class=\"file-message-size\">{{message.fileSizeInBytes}} Bytes</span>\n\t\t\t\t\t\t</a> -->\n\t\t\t\t\t</div>\n\t\t\t\t</ng-container>\n\t\t\t</div>\n\t</div>\n\n\t<div class=\"ng-chat-footer primary-outline-color primary-background\">\n\t\t<input #chatWindowInput\n\t\t\ttype=\"text\"\n\t\t\t[ngModel]=\"window.newMessage | emojify:emojisEnabled\"\n\t\t\t(ngModelChange)=\"window.newMessage=$event\"\n\t\t\t[placeholder]=\"localization.messagePlaceholder\"\n\t\t\t[ngClass]=\"{'chat-window-input': true, 'has-side-action': fileUploadAdapter}\"\n\t\t\t(keydown)=\"onChatInputTyped($event, window)\"\n\t\t\t(blur)=\"toggleWindowFocus(window)\"\n\t\t\t(focus)=\"toggleWindowFocus(window)\"/>\n\n\t\t<!-- File Upload -->\n\t\t<ng-container *ngIf=\"fileUploadAdapter\">\n\t\t\t<a *ngIf=\"!isUploadingFile(window)\" class=\"btn-add-file\" (click)=\"triggerNativeFileUpload(window)\">\n\t\t\t\t<i class=\"upload-icon\"></i>\n\t\t\t</a>\n\t\t\t<input\n\t\t\t\ttype=\"file\"\n\t\t\t\t#nativeFileInput\n\t\t\t\tstyle=\"display: none;\"\n\t\t\t\t[attr.id]=\"getUniqueFileUploadInstanceId(window)\"\n\t\t\t\t(change)=\"onFileChosen(window)\" />\n\t\t\t<div *ngIf=\"isUploadingFile(window)\" class=\"loader\"></div>\n\t\t</ng-container>\n\t</div>\n</ng-container>"]}
|