stream-chat-angular 4.53.0 → 4.54.0
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/assets/i18n/en.d.ts +1 -0
- package/assets/version.d.ts +1 -1
- package/bundles/stream-chat-angular.umd.js +187 -54
- package/bundles/stream-chat-angular.umd.js.map +1 -1
- package/esm2015/assets/i18n/en.js +2 -1
- package/esm2015/assets/version.js +2 -2
- package/esm2015/lib/channel.service.js +38 -1
- package/esm2015/lib/message-list/message-list.component.js +2 -2
- package/esm2015/lib/message-reactions/message-reactions.component.js +78 -8
- package/fesm2015/stream-chat-angular.js +134 -30
- package/fesm2015/stream-chat-angular.js.map +1 -1
- package/lib/channel.service.d.ts +13 -3
- package/lib/message-reactions/message-reactions.component.d.ts +15 -3
- package/package.json +1 -1
- package/src/assets/i18n/en.ts +1 -0
- package/src/assets/styles/v2/css/index.css +1 -1
- package/src/assets/styles/v2/css/index.layout.css +1 -1
- package/src/assets/styles/v2/scss/AttachmentList/AttachmentList-layout.scss +1 -0
- package/src/assets/styles/v2/scss/AttachmentList/AttachmentList-theme.scss +3 -3
- package/src/assets/styles/v2/scss/AttachmentPreviewList/AttachmentPreviewList-layout.scss +4 -0
- package/src/assets/styles/v2/scss/BaseImage/BaseImage-layout.scss +21 -0
- package/src/assets/styles/v2/scss/BaseImage/BaseImage-theme.scss +35 -0
- package/src/assets/styles/v2/scss/BaseImage/index.scss +2 -0
- package/src/assets/styles/v2/scss/Message/Message-layout.scss +59 -9
- package/src/assets/styles/v2/scss/Message/Message-theme.scss +32 -1
- package/src/assets/styles/v2/scss/MessageReactions/MessageReactions-layout.scss +68 -0
- package/src/assets/styles/v2/scss/MessageReactions/MessageReactions-theme.scss +17 -0
- package/src/assets/styles/v2/scss/_global-theme-variables.scss +3 -0
- package/src/assets/styles/v2/scss/_icons.scss +3 -0
- package/src/assets/styles/v2/scss/_utils.scss +12 -0
- package/src/assets/styles/v2/scss/index.layout.scss +2 -0
- package/src/assets/styles/v2/scss/index.scss +1 -0
- package/src/assets/styles/v2/scss/vendor/react-image-gallery.scss +19 -0
- package/src/assets/version.ts +1 -1
|
@@ -1,19 +1,25 @@
|
|
|
1
|
+
import { __awaiter } from "tslib";
|
|
1
2
|
import { Component, EventEmitter, Input, Output, ViewChild, } from '@angular/core';
|
|
2
3
|
import { NgxPopperjsTriggers, NgxPopperjsPlacements } from 'ngx-popperjs';
|
|
3
4
|
import * as i0 from "@angular/core";
|
|
4
5
|
import * as i1 from "../channel.service";
|
|
5
6
|
import * as i2 from "../message-reactions.service";
|
|
6
|
-
import * as i3 from "
|
|
7
|
-
import * as i4 from "../
|
|
8
|
-
import * as i5 from "
|
|
7
|
+
import * as i3 from "../custom-templates.service";
|
|
8
|
+
import * as i4 from "../theme.service";
|
|
9
|
+
import * as i5 from "../avatar-placeholder/avatar-placeholder.component";
|
|
10
|
+
import * as i6 from "../modal/modal.component";
|
|
11
|
+
import * as i7 from "../loading-indicator/loading-indicator.component";
|
|
12
|
+
import * as i8 from "@angular/common";
|
|
9
13
|
/**
|
|
10
14
|
* The `MessageReactions` component displays the reactions of a message, the current user can add and remove reactions. You can read more about [message reactions](https://getstream.io/chat/docs/javascript/send_reaction/?language=javascript) in the platform documentation.
|
|
11
15
|
*/
|
|
12
16
|
export class MessageReactionsComponent {
|
|
13
|
-
constructor(cdRef, channelService, messageReactionsService) {
|
|
17
|
+
constructor(cdRef, channelService, messageReactionsService, customTemplatesService, themeService) {
|
|
14
18
|
this.cdRef = cdRef;
|
|
15
19
|
this.channelService = channelService;
|
|
16
20
|
this.messageReactionsService = messageReactionsService;
|
|
21
|
+
this.customTemplatesService = customTemplatesService;
|
|
22
|
+
this.themeService = themeService;
|
|
17
23
|
/**
|
|
18
24
|
* The number of reactions grouped by [reaction types](https://github.com/GetStream/stream-chat-angular/tree/master/projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts)
|
|
19
25
|
*/
|
|
@@ -36,6 +42,11 @@ export class MessageReactionsComponent {
|
|
|
36
42
|
this.isSelectorOpenChange = new EventEmitter();
|
|
37
43
|
this.popperTriggerHover = NgxPopperjsTriggers.hover;
|
|
38
44
|
this.popperPlacementAuto = NgxPopperjsPlacements.AUTO;
|
|
45
|
+
this.isLoading = true;
|
|
46
|
+
this.reactions = [];
|
|
47
|
+
this.isOpenChange = (isOpen) => {
|
|
48
|
+
this.selectedReactionType = isOpen ? this.selectedReactionType : undefined;
|
|
49
|
+
};
|
|
39
50
|
this.eventHandler = (event) => {
|
|
40
51
|
var _a;
|
|
41
52
|
if (!((_a = this.selectorContainer) === null || _a === void 0 ? void 0 : _a.nativeElement.contains(event.target))) {
|
|
@@ -74,6 +85,14 @@ export class MessageReactionsComponent {
|
|
|
74
85
|
getEmojiByReaction(reactionType) {
|
|
75
86
|
return this.messageReactionsService.reactions[reactionType];
|
|
76
87
|
}
|
|
88
|
+
reactionSelected(event, reactionType) {
|
|
89
|
+
event.stopPropagation();
|
|
90
|
+
if (this.themeService.themeVersion === '1') {
|
|
91
|
+
return;
|
|
92
|
+
}
|
|
93
|
+
this.selectedReactionType = reactionType;
|
|
94
|
+
void this.fetchAllReactions();
|
|
95
|
+
}
|
|
77
96
|
getUsersByReaction(reactionType) {
|
|
78
97
|
return this.latestReactions
|
|
79
98
|
.filter((r) => r.type === reactionType)
|
|
@@ -81,6 +100,36 @@ export class MessageReactionsComponent {
|
|
|
81
100
|
.filter((i) => !!i)
|
|
82
101
|
.join(', ');
|
|
83
102
|
}
|
|
103
|
+
getAllUsersByReaction(reactionType) {
|
|
104
|
+
if (!reactionType) {
|
|
105
|
+
return [];
|
|
106
|
+
}
|
|
107
|
+
const users = this.reactions
|
|
108
|
+
.filter((r) => r.type === reactionType)
|
|
109
|
+
.map((r) => r.user)
|
|
110
|
+
.filter((i) => !!i);
|
|
111
|
+
users.sort((u1, u2) => {
|
|
112
|
+
var _a, _b;
|
|
113
|
+
const name1 = (_a = u1.name) === null || _a === void 0 ? void 0 : _a.toLowerCase();
|
|
114
|
+
const name2 = (_b = u2.name) === null || _b === void 0 ? void 0 : _b.toLowerCase();
|
|
115
|
+
if (!name1) {
|
|
116
|
+
return 1;
|
|
117
|
+
}
|
|
118
|
+
if (!name2) {
|
|
119
|
+
return -1;
|
|
120
|
+
}
|
|
121
|
+
if (name1 === name2) {
|
|
122
|
+
return 0;
|
|
123
|
+
}
|
|
124
|
+
if (name1 < name2) {
|
|
125
|
+
return -1;
|
|
126
|
+
}
|
|
127
|
+
else {
|
|
128
|
+
return 1;
|
|
129
|
+
}
|
|
130
|
+
});
|
|
131
|
+
return users;
|
|
132
|
+
}
|
|
84
133
|
showTooltip(event, reactionType) {
|
|
85
134
|
this.currentTooltipTarget = event.target;
|
|
86
135
|
this.tooltipText = this.getUsersByReaction(reactionType);
|
|
@@ -93,6 +142,9 @@ export class MessageReactionsComponent {
|
|
|
93
142
|
trackByMessageReaction(index, item) {
|
|
94
143
|
return item;
|
|
95
144
|
}
|
|
145
|
+
trackByUserId(index, item) {
|
|
146
|
+
return item.id;
|
|
147
|
+
}
|
|
96
148
|
react(type) {
|
|
97
149
|
this.ownReactions.find((r) => r.type === type)
|
|
98
150
|
? void this.channelService.removeReaction(this.messageId, type)
|
|
@@ -123,9 +175,27 @@ export class MessageReactionsComponent {
|
|
|
123
175
|
arrow: arrowPosition,
|
|
124
176
|
};
|
|
125
177
|
}
|
|
178
|
+
fetchAllReactions() {
|
|
179
|
+
return __awaiter(this, void 0, void 0, function* () {
|
|
180
|
+
if (!this.messageId) {
|
|
181
|
+
return;
|
|
182
|
+
}
|
|
183
|
+
this.isLoading = true;
|
|
184
|
+
try {
|
|
185
|
+
this.reactions = yield this.channelService.getMessageReactions(this.messageId);
|
|
186
|
+
}
|
|
187
|
+
catch (error) {
|
|
188
|
+
this.selectedReactionType = undefined;
|
|
189
|
+
}
|
|
190
|
+
finally {
|
|
191
|
+
this.isLoading = false;
|
|
192
|
+
this.cdRef.detectChanges();
|
|
193
|
+
}
|
|
194
|
+
});
|
|
195
|
+
}
|
|
126
196
|
}
|
|
127
|
-
MessageReactionsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageReactionsComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.ChannelService }, { token: i2.MessageReactionsService }], target: i0.ɵɵFactoryTarget.Component });
|
|
128
|
-
MessageReactionsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageReactionsComponent, selector: "stream-message-reactions", inputs: { messageId: "messageId", messageReactionCounts: "messageReactionCounts", isSelectorOpen: "isSelectorOpen", latestReactions: "latestReactions", ownReactions: "ownReactions" }, outputs: { isSelectorOpenChange: "isSelectorOpenChange" }, viewQueries: [{ propertyName: "selectorContainer", first: true, predicate: ["selectorContainer"], descendants: true }, { propertyName: "selectorTooltip", first: true, predicate: ["selectorTooltip"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n *ngIf=\"existingReactions.length > 0\"\n class=\"str-chat__reaction-list str-chat__message-reactions-container\"\n [class.str-chat__reaction-list--reverse]=\"true\"\n [class.str-chat__reaction-list-hidden]=\"isSelectorOpen\"\n data-testid=\"reaction-list\"\n>\n <ul class=\"str-chat__message-reactions\">\n <li\n class=\"str-chat__message-reaction\"\n *ngFor=\"\n let reactionType of existingReactions;\n trackBy: trackByMessageReaction\n \"\n [class.str-chat__message-reaction-own]=\"isOwnReaction(reactionType)\"\n data-testclass=\"emoji\"\n
|
|
197
|
+
MessageReactionsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageReactionsComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.ChannelService }, { token: i2.MessageReactionsService }, { token: i3.CustomTemplatesService }, { token: i4.ThemeService }], target: i0.ɵɵFactoryTarget.Component });
|
|
198
|
+
MessageReactionsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.5", type: MessageReactionsComponent, selector: "stream-message-reactions", inputs: { messageId: "messageId", messageReactionCounts: "messageReactionCounts", isSelectorOpen: "isSelectorOpen", latestReactions: "latestReactions", ownReactions: "ownReactions" }, outputs: { isSelectorOpenChange: "isSelectorOpenChange" }, viewQueries: [{ propertyName: "selectorContainer", first: true, predicate: ["selectorContainer"], descendants: true }, { propertyName: "selectorTooltip", first: true, predicate: ["selectorTooltip"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n *ngIf=\"existingReactions.length > 0\"\n class=\"str-chat__reaction-list str-chat__message-reactions-container\"\n [class.str-chat__reaction-list--reverse]=\"true\"\n [class.str-chat__reaction-list-hidden]=\"isSelectorOpen\"\n data-testid=\"reaction-list\"\n>\n <ul class=\"str-chat__message-reactions\">\n <li\n class=\"str-chat__message-reaction\"\n *ngFor=\"\n let reactionType of existingReactions;\n trackBy: trackByMessageReaction\n \"\n [class.str-chat__message-reaction-own]=\"isOwnReaction(reactionType)\"\n data-testclass=\"emoji\"\n (click)=\"reactionSelected($event, reactionType)\"\n (keyup.enter)=\"reactionSelected($event, reactionType)\"\n >\n <span class=\"emoji str-chat__message-reaction-emoji\">\n {{ getEmojiByReaction(reactionType) }} \n </span>\n <span\n data-testclass=\"reaction-list-reaction-count\"\n class=\"str-chat__message-reaction-count\"\n >\n {{ messageReactionCounts[reactionType] }}\n </span>\n </li>\n <li>\n <span\n data-testid=\"reactions-count\"\n class=\"str-chat__reaction-list--counter\"\n >{{ reactionsCount }}</span\n >\n </li>\n </ul>\n</div>\n\n<div\n #selectorContainer\n class=\"str-chat__reaction-selector str-chat__message-reaction-selector\"\n *ngIf=\"isSelectorOpen\"\n data-testid=\"reaction-selector\"\n>\n <div\n *ngIf=\"tooltipText\"\n data-testid=\"tooltip\"\n #selectorTooltip\n class=\"str-chat__reaction-selector-tooltip\"\n [ngStyle]=\"{\n left: tooltipPositions?.tooltip + 'px',\n visibility: tooltipPositions ? 'visible' : 'hidden'\n }\"\n >\n <div\n class=\"arrow\"\n [ngStyle]=\"{ left: tooltipPositions?.arrow + 'px' }\"\n ></div>\n <span class=\"latest-user-username\">\n {{ tooltipText }}\n </span>\n </div>\n <ul\n class=\"str-chat__message-reactions-list str-chat__message-reactions-options\"\n >\n <li\n class=\"\n str-chat__message-reactions-option\n str-chat__message-reactions-list-item\n str-chat__emoji\n \"\n *ngFor=\"\n let reactionType of reactionOptions;\n trackBy: trackByMessageReaction\n \"\n [class.str-chat__message-reactions-option-selected]=\"\n isOwnReaction(reactionType)\n \"\n data-testclass=\"emoji-option\"\n (click)=\"react(reactionType)\"\n (keyup.enter)=\"react(reactionType)\"\n >\n <div\n *ngIf=\"getLatestUserByReaction(reactionType) as user\"\n class=\"latest-user str-chat__message-reactions-last-user\"\n (click)=\"hideTooltip()\"\n (keyup.enter)=\"hideTooltip()\"\n (mouseenter)=\"showTooltip($event, reactionType)\"\n (mouseleave)=\"hideTooltip()\"\n attr.data-testid=\"{{ reactionType }}-last-user\"\n >\n <stream-avatar-placeholder\n attr.data-testid=\"{{ reactionType }}-avatar\"\n [imageUrl]=\"user.image\"\n [name]=\"user.name || user.id\"\n [size]=\"20\"\n location=\"reaction\"\n ></stream-avatar-placeholder>\n </div>\n <span\n class=\"\n emoji\n str-chat__emoji-selector-emoji-angular\n str-chat__message-reaction-emoji\n \"\n >\n {{ getEmojiByReaction(reactionType) }}\n </span>\n <span\n *ngIf=\"\n messageReactionCounts[reactionType] &&\n messageReactionCounts[reactionType]! > 0\n \"\n class=\"str-chat__message-reactions-list-item__count\"\n attr.data-testid=\"{{ reactionType }}-reaction-count\"\n >\n {{ messageReactionCounts[reactionType] }}\n </span>\n </li>\n </ul>\n</div>\n\n<ng-container *ngIf=\"selectedReactionType\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.modalTemplate$ | async) || defaultModal;\n context: {\n isOpen: !!selectedReactionType,\n messageId: messageId,\n reactionType: selectedReactionType,\n isOpenChangeHandler: isOpenChange\n }\n \"\n ></ng-container>\n</ng-container>\n\n<ng-template\n #defaultModal\n let-isOpen=\"isOpen\"\n let-messageId=\"messageId\"\n let-reactionType=\"reactionType\"\n let-isOpenChangeHandler=\"isOpenChangeHandler\"\n>\n <stream-modal\n class=\"str-chat__message-reactions-details-modal\"\n [isOpen]=\"isOpen\"\n (isOpenChange)=\"isOpenChangeHandler($event)\"\n >\n <div class=\"str-chat__message-reactions-details\">\n <div class=\"str-chat__message-reactions-details-reaction-types\">\n <div\n class=\"str-chat__message-reactions-details-reaction-type\"\n *ngFor=\"\n let reactionType of existingReactions;\n trackBy: trackByMessageReaction\n \"\n attr.data-testid=\"reaction-details-selector-{{ reactionType }}\"\n [class.str-chat__message-reactions-details-reaction-type--selected]=\"\n reactionType === selectedReactionType\n \"\n (click)=\"selectedReactionType = reactionType; allUsers.scrollTop = 0\"\n (keyup.enter)=\"\n selectedReactionType = reactionType; allUsers.scrollTop = 0\n \"\n >\n <span class=\"emoji str-chat__message-reaction-emoji\">\n {{ getEmojiByReaction(reactionType) }} \n </span>\n <span class=\"str-chat__message-reaction-count\">\n {{ messageReactionCounts[reactionType] }}\n </span>\n </div>\n </div>\n <div\n class=\"\n emoji\n str-chat__message-reaction-emoji str-chat__message-reaction-emoji-big\n \"\n >\n {{ getEmojiByReaction(selectedReactionType!) }}\n </div>\n <div\n #allUsers\n data-testid=\"all-reacting-users\"\n class=\"str-chat__message-reactions-details-reacting-users\"\n >\n <stream-loading-indicator\n *ngIf=\"isLoading; else reactions\"\n ></stream-loading-indicator>\n <ng-template #reactions>\n <div\n class=\"str-chat__message-reactions-details-reacting-user\"\n *ngFor=\"\n let user of getAllUsersByReaction(selectedReactionType);\n trackBy: trackByUserId\n \"\n >\n <stream-avatar-placeholder\n data-testclass=\"avatar\"\n class=\"str-chat__avatar str-chat__avatar--circle\"\n style=\"max-height: 100%; overflow-y: hidden\"\n [size]=\"30\"\n [imageUrl]=\"user.image\"\n [name]=\"user.name\"\n type=\"user\"\n location=\"reaction\"\n [user]=\"user\"\n ></stream-avatar-placeholder>\n <span\n data-testclass=\"reaction-user-username\"\n class=\"str-chat__user-item--name\"\n >{{ user.name }}</span\n >\n </div>\n </ng-template>\n </div>\n </div>\n </stream-modal>\n</ng-template>\n", styles: [".emoji {position: relative; display: inline-block; }"], components: [{ type: i5.AvatarPlaceholderComponent, selector: "stream-avatar-placeholder", inputs: ["name", "imageUrl", "size", "location", "channel", "user", "type", "initialsType", "showOnlineIndicator"] }, { type: i6.ModalComponent, selector: "stream-modal", inputs: ["isOpen", "content"], outputs: ["isOpenChange"] }, { type: i7.LoadingIndicatorComponent, selector: "stream-loading-indicator", inputs: ["size", "color"] }], directives: [{ type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i8.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i8.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }], pipes: { "async": i8.AsyncPipe } });
|
|
129
199
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImport: i0, type: MessageReactionsComponent, decorators: [{
|
|
130
200
|
type: Component,
|
|
131
201
|
args: [{
|
|
@@ -133,7 +203,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
133
203
|
templateUrl: './message-reactions.component.html',
|
|
134
204
|
styles: ['.emoji {position: relative; display: inline-block; }'],
|
|
135
205
|
}]
|
|
136
|
-
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.ChannelService }, { type: i2.MessageReactionsService }]; }, propDecorators: { messageId: [{
|
|
206
|
+
}], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }, { type: i1.ChannelService }, { type: i2.MessageReactionsService }, { type: i3.CustomTemplatesService }, { type: i4.ThemeService }]; }, propDecorators: { messageId: [{
|
|
137
207
|
type: Input
|
|
138
208
|
}], messageReactionCounts: [{
|
|
139
209
|
type: Input
|
|
@@ -152,4 +222,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.5", ngImpor
|
|
|
152
222
|
type: ViewChild,
|
|
153
223
|
args: ['selectorTooltip']
|
|
154
224
|
}] } });
|
|
155
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-reactions.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { ReactionResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { MessageReactionType, DefaultStreamChatGenerics } from '../types';\nimport { NgxPopperjsTriggers, NgxPopperjsPlacements } from 'ngx-popperjs';\nimport { MessageReactionsService } from '../message-reactions.service';\n\n/**\n * The `MessageReactions` component displays the reactions of a message, the current user can add and remove reactions. You can read more about [message reactions](https://getstream.io/chat/docs/javascript/send_reaction/?language=javascript) in the platform documentation.\n */\n@Component({\n  selector: 'stream-message-reactions',\n  templateUrl: './message-reactions.component.html',\n  styles: ['.emoji {position: relative; display: inline-block; }'],\n})\nexport class MessageReactionsComponent implements AfterViewChecked, OnChanges {\n  /**\n   * The id of the message the reactions belong to\n   */\n  @Input() messageId: string | undefined;\n  /**\n   * The number of reactions grouped by [reaction types](https://github.com/GetStream/stream-chat-angular/tree/master/projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts)\n   */\n  @Input() messageReactionCounts: { [key in MessageReactionType]?: number } =\n    {};\n  /**\n   * Indicates if the selector should be opened or closed. Adding a UI element to open and close the selector is the parent's component responsibility.\n   */\n  @Input() isSelectorOpen: boolean = false;\n  /**\n   * List of reactions of a [message](../types/stream-message.mdx), used to display the users of a reaction type.\n   */\n  @Input() latestReactions: ReactionResponse<DefaultStreamChatGenerics>[] = [];\n  /**\n   * List of the user's own reactions of a [message](../types/stream-message.mdx), used to display the users of a reaction type.\n   */\n  @Input() ownReactions: ReactionResponse<DefaultStreamChatGenerics>[] = [];\n  /**\n   * Indicates if the selector should be opened or closed. Adding a UI element to open and close the selector is the parent's component responsibility.\n   */\n  @Output() readonly isSelectorOpenChange = new EventEmitter<boolean>();\n  tooltipPositions: { arrow: number; tooltip: number } | undefined;\n  tooltipText: string | undefined;\n  @ViewChild('selectorContainer') private selectorContainer:\n    | ElementRef<HTMLElement>\n    | undefined;\n  @ViewChild('selectorTooltip') private selectorTooltip:\n    | ElementRef<HTMLElement>\n    | undefined;\n  currentTooltipTarget: HTMLElement | undefined;\n  popperTriggerHover = NgxPopperjsTriggers.hover;\n  popperPlacementAuto = NgxPopperjsPlacements.AUTO;\n\n  constructor(\n    private cdRef: ChangeDetectorRef,\n    private channelService: ChannelService,\n    private messageReactionsService: MessageReactionsService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.isSelectorOpen) {\n      this.isSelectorOpen\n        ? setTimeout(() => this.watchForOutsideClicks()) // setTimeout: wait for current click to bubble up, and only watch for clicks after that\n        : this.stopWatchForOutsideClicks();\n    }\n  }\n\n  ngAfterViewChecked(): void {\n    if (this.tooltipText && !this.tooltipPositions) {\n      this.setTooltipPosition();\n      this.cdRef.detectChanges();\n    }\n  }\n\n  get existingReactions(): MessageReactionType[] {\n    return Object.keys(this.messageReactionCounts)\n      .filter((k) => this.reactionOptions.indexOf(k) !== -1)\n      .filter((k) => this.messageReactionCounts[k]! > 0);\n  }\n\n  get reactionsCount() {\n    return this.existingReactions.reduce(\n      (total, reaction) => total + this.messageReactionCounts[reaction]!,\n      0\n    );\n  }\n\n  get reactionOptions(): MessageReactionType[] {\n    return Object.keys(this.messageReactionsService.reactions);\n  }\n\n  getLatestUserByReaction(reactionType: MessageReactionType) {\n    return this.latestReactions.find((r) => r.type === reactionType && r.user)\n      ?.user;\n  }\n\n  getEmojiByReaction(reactionType: MessageReactionType) {\n    return this.messageReactionsService.reactions[reactionType];\n  }\n\n  getUsersByReaction(reactionType: MessageReactionType) {\n    return this.latestReactions\n      .filter((r) => r.type === reactionType)\n      .map((r) => r.user?.name || r.user?.id)\n      .filter((i) => !!i)\n      .join(', ');\n  }\n\n  showTooltip(event: Event, reactionType: MessageReactionType) {\n    this.currentTooltipTarget = event.target as HTMLElement;\n    this.tooltipText = this.getUsersByReaction(reactionType);\n  }\n\n  hideTooltip() {\n    this.tooltipText = undefined;\n    this.currentTooltipTarget = undefined;\n    this.tooltipPositions = undefined;\n  }\n\n  trackByMessageReaction(index: number, item: MessageReactionType) {\n    return item;\n  }\n\n  react(type: MessageReactionType) {\n    this.ownReactions.find((r) => r.type === type)\n      ? void this.channelService.removeReaction(this.messageId!, type)\n      : void this.channelService.addReaction(this.messageId!, type);\n  }\n\n  isOwnReaction(reactionType: MessageReactionType) {\n    return !!this.ownReactions.find((r) => r.type === reactionType);\n  }\n\n  private eventHandler = (event: Event) => {\n    if (!this.selectorContainer?.nativeElement.contains(event.target as Node)) {\n      this.isSelectorOpenChange.emit(false);\n    }\n  };\n\n  private watchForOutsideClicks() {\n    window.addEventListener('click', this.eventHandler);\n  }\n\n  private stopWatchForOutsideClicks() {\n    window.removeEventListener('click', this.eventHandler);\n  }\n\n  private setTooltipPosition() {\n    const tooltip = this.selectorTooltip?.nativeElement.getBoundingClientRect();\n    const target = this.currentTooltipTarget?.getBoundingClientRect();\n\n    const container =\n      this.selectorContainer?.nativeElement.getBoundingClientRect();\n\n    if (!tooltip || !target || !container) return;\n\n    const tooltipPosition =\n      tooltip.width === container.width || tooltip.x < container.x\n        ? 0\n        : target.left + target.width / 2 - container.left - tooltip.width / 2;\n\n    const arrowPosition =\n      target.x - tooltip.x + target.width / 2 - tooltipPosition;\n\n    this.tooltipPositions = {\n      tooltip: tooltipPosition,\n      arrow: arrowPosition,\n    };\n  }\n}\n","<div\n  *ngIf=\"existingReactions.length > 0\"\n  class=\"str-chat__reaction-list str-chat__message-reactions-container\"\n  [class.str-chat__reaction-list--reverse]=\"true\"\n  [class.str-chat__reaction-list-hidden]=\"isSelectorOpen\"\n  data-testid=\"reaction-list\"\n>\n  <ul class=\"str-chat__message-reactions\">\n    <li\n      class=\"str-chat__message-reaction\"\n      *ngFor=\"\n        let reactionType of existingReactions;\n        trackBy: trackByMessageReaction\n      \"\n      [class.str-chat__message-reaction-own]=\"isOwnReaction(reactionType)\"\n      data-testclass=\"emoji\"\n      [popper]=\"popperContent\"\n      [popperTrigger]=\"popperTriggerHover\"\n      [popperPlacement]=\"popperPlacementAuto\"\n      [popperHideOnScroll]=\"false\"\n      [popperHideOnClickOutside]=\"false\"\n      (hover)=\"$event.stopPropagation()\"\n    >\n      <popper-content #popperContent>\n        <div\n          data-testclass=\"tooltip\"\n          class=\"str-chat__tooltip str-chat__tooltip-angular\"\n        >\n          <span>\n            {{ getUsersByReaction(reactionType) }}\n          </span>\n        </div>\n      </popper-content>\n      <span class=\"emoji str-chat__message-reaction-emoji\">\n        {{ getEmojiByReaction(reactionType) }}&nbsp;\n      </span>\n      <span\n        data-testclass=\"reaction-list-reaction-count\"\n        class=\"str-chat__message-reaction-count\"\n      >\n        {{ messageReactionCounts[reactionType] }}\n      </span>\n    </li>\n    <li>\n      <span\n        data-testid=\"reactions-count\"\n        class=\"str-chat__reaction-list--counter\"\n        >{{ reactionsCount }}</span\n      >\n    </li>\n  </ul>\n</div>\n\n<div\n  #selectorContainer\n  class=\"str-chat__reaction-selector str-chat__message-reaction-selector\"\n  *ngIf=\"isSelectorOpen\"\n  data-testid=\"reaction-selector\"\n>\n  <div\n    *ngIf=\"tooltipText\"\n    data-testid=\"tooltip\"\n    #selectorTooltip\n    class=\"str-chat__reaction-selector-tooltip\"\n    [ngStyle]=\"{\n      left: tooltipPositions?.tooltip + 'px',\n      visibility: tooltipPositions ? 'visible' : 'hidden'\n    }\"\n  >\n    <div\n      class=\"arrow\"\n      [ngStyle]=\"{ left: tooltipPositions?.arrow + 'px' }\"\n    ></div>\n    <span class=\"latest-user-username\">\n      {{ tooltipText }}\n    </span>\n  </div>\n  <ul\n    class=\"str-chat__message-reactions-list str-chat__message-reactions-options\"\n  >\n    <li\n      class=\"\n        str-chat__message-reactions-option\n        str-chat__message-reactions-list-item\n        str-chat__emoji\n      \"\n      *ngFor=\"\n        let reactionType of reactionOptions;\n        trackBy: trackByMessageReaction\n      \"\n      [class.str-chat__message-reactions-option-selected]=\"\n        isOwnReaction(reactionType)\n      \"\n      data-testclass=\"emoji-option\"\n      (click)=\"react(reactionType)\"\n      (keyup.enter)=\"react(reactionType)\"\n    >\n      <div\n        *ngIf=\"getLatestUserByReaction(reactionType) as user\"\n        class=\"latest-user str-chat__message-reactions-last-user\"\n        (click)=\"hideTooltip()\"\n        (keyup.enter)=\"hideTooltip()\"\n        (mouseenter)=\"showTooltip($event, reactionType)\"\n        (mouseleave)=\"hideTooltip()\"\n        attr.data-testid=\"{{ reactionType }}-last-user\"\n      >\n        <stream-avatar-placeholder\n          attr.data-testid=\"{{ reactionType }}-avatar\"\n          [imageUrl]=\"user.image\"\n          [name]=\"user.name || user.id\"\n          [size]=\"20\"\n        ></stream-avatar-placeholder>\n      </div>\n      <span\n        class=\"\n          emoji\n          str-chat__emoji-selector-emoji-angular\n          str-chat__message-reaction-emoji\n        \"\n      >\n        {{ getEmojiByReaction(reactionType) }}\n      </span>\n      <span\n        *ngIf=\"\n          messageReactionCounts[reactionType] &&\n          messageReactionCounts[reactionType]! > 0\n        \"\n        class=\"str-chat__message-reactions-list-item__count\"\n        attr.data-testid=\"{{ reactionType }}-reaction-count\"\n      >\n        {{ messageReactionCounts[reactionType] }}\n      </span>\n    </li>\n  </ul>\n</div>\n"]}
|
|
225
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-reactions.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewChecked,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  EventEmitter,\n  Input,\n  OnChanges,\n  Output,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { ReactionResponse, UserResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { MessageReactionType, DefaultStreamChatGenerics } from '../types';\nimport { NgxPopperjsTriggers, NgxPopperjsPlacements } from 'ngx-popperjs';\nimport { MessageReactionsService } from '../message-reactions.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { ThemeService } from '../theme.service';\n\n/**\n * The `MessageReactions` component displays the reactions of a message, the current user can add and remove reactions. You can read more about [message reactions](https://getstream.io/chat/docs/javascript/send_reaction/?language=javascript) in the platform documentation.\n */\n@Component({\n  selector: 'stream-message-reactions',\n  templateUrl: './message-reactions.component.html',\n  styles: ['.emoji {position: relative; display: inline-block; }'],\n})\nexport class MessageReactionsComponent implements AfterViewChecked, OnChanges {\n  /**\n   * The id of the message the reactions belong to\n   */\n  @Input() messageId: string | undefined;\n  /**\n   * The number of reactions grouped by [reaction types](https://github.com/GetStream/stream-chat-angular/tree/master/projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts)\n   */\n  @Input() messageReactionCounts: { [key in MessageReactionType]?: number } =\n    {};\n  /**\n   * Indicates if the selector should be opened or closed. Adding a UI element to open and close the selector is the parent's component responsibility.\n   */\n  @Input() isSelectorOpen: boolean = false;\n  /**\n   * List of reactions of a [message](../types/stream-message.mdx), used to display the users of a reaction type.\n   */\n  @Input() latestReactions: ReactionResponse<DefaultStreamChatGenerics>[] = [];\n  /**\n   * List of the user's own reactions of a [message](../types/stream-message.mdx), used to display the users of a reaction type.\n   */\n  @Input() ownReactions: ReactionResponse<DefaultStreamChatGenerics>[] = [];\n  /**\n   * Indicates if the selector should be opened or closed. Adding a UI element to open and close the selector is the parent's component responsibility.\n   */\n  @Output() readonly isSelectorOpenChange = new EventEmitter<boolean>();\n  tooltipPositions: { arrow: number; tooltip: number } | undefined;\n  tooltipText: string | undefined;\n  @ViewChild('selectorContainer') private selectorContainer:\n    | ElementRef<HTMLElement>\n    | undefined;\n  @ViewChild('selectorTooltip') private selectorTooltip:\n    | ElementRef<HTMLElement>\n    | undefined;\n  currentTooltipTarget: HTMLElement | undefined;\n  popperTriggerHover = NgxPopperjsTriggers.hover;\n  popperPlacementAuto = NgxPopperjsPlacements.AUTO;\n  selectedReactionType: string | undefined;\n  isLoading = true;\n  reactions: ReactionResponse[] = [];\n\n  constructor(\n    private cdRef: ChangeDetectorRef,\n    private channelService: ChannelService,\n    private messageReactionsService: MessageReactionsService,\n    public customTemplatesService: CustomTemplatesService,\n    private themeService: ThemeService\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes.isSelectorOpen) {\n      this.isSelectorOpen\n        ? setTimeout(() => this.watchForOutsideClicks()) // setTimeout: wait for current click to bubble up, and only watch for clicks after that\n        : this.stopWatchForOutsideClicks();\n    }\n  }\n\n  ngAfterViewChecked(): void {\n    if (this.tooltipText && !this.tooltipPositions) {\n      this.setTooltipPosition();\n      this.cdRef.detectChanges();\n    }\n  }\n\n  get existingReactions(): MessageReactionType[] {\n    return Object.keys(this.messageReactionCounts)\n      .filter((k) => this.reactionOptions.indexOf(k) !== -1)\n      .filter((k) => this.messageReactionCounts[k]! > 0);\n  }\n\n  get reactionsCount() {\n    return this.existingReactions.reduce(\n      (total, reaction) => total + this.messageReactionCounts[reaction]!,\n      0\n    );\n  }\n\n  get reactionOptions(): MessageReactionType[] {\n    return Object.keys(this.messageReactionsService.reactions);\n  }\n\n  getLatestUserByReaction(reactionType: MessageReactionType) {\n    return this.latestReactions.find((r) => r.type === reactionType && r.user)\n      ?.user;\n  }\n\n  getEmojiByReaction(reactionType: MessageReactionType) {\n    return this.messageReactionsService.reactions[reactionType];\n  }\n\n  reactionSelected(event: Event, reactionType: string) {\n    event.stopPropagation();\n    if (this.themeService.themeVersion === '1') {\n      return;\n    }\n    this.selectedReactionType = reactionType;\n    void this.fetchAllReactions();\n  }\n\n  getUsersByReaction(reactionType: MessageReactionType) {\n    return this.latestReactions\n      .filter((r) => r.type === reactionType)\n      .map((r) => r.user?.name || r.user?.id)\n      .filter((i) => !!i)\n      .join(', ');\n  }\n\n  getAllUsersByReaction(\n    reactionType?: MessageReactionType\n  ): UserResponse<DefaultStreamChatGenerics>[] {\n    if (!reactionType) {\n      return [];\n    }\n\n    const users = this.reactions\n      .filter((r) => r.type === reactionType)\n      .map((r) => r.user)\n      .filter((i) => !!i) as UserResponse[];\n\n    users.sort((u1, u2) => {\n      const name1 = u1.name?.toLowerCase();\n      const name2 = u2.name?.toLowerCase();\n\n      if (!name1) {\n        return 1;\n      }\n\n      if (!name2) {\n        return -1;\n      }\n\n      if (name1 === name2) {\n        return 0;\n      }\n\n      if (name1 < name2) {\n        return -1;\n      } else {\n        return 1;\n      }\n    });\n\n    return users;\n  }\n\n  showTooltip(event: Event, reactionType: MessageReactionType) {\n    this.currentTooltipTarget = event.target as HTMLElement;\n    this.tooltipText = this.getUsersByReaction(reactionType);\n  }\n\n  hideTooltip() {\n    this.tooltipText = undefined;\n    this.currentTooltipTarget = undefined;\n    this.tooltipPositions = undefined;\n  }\n\n  trackByMessageReaction(index: number, item: MessageReactionType) {\n    return item;\n  }\n\n  trackByUserId(index: number, item: UserResponse) {\n    return item.id;\n  }\n\n  react(type: MessageReactionType) {\n    this.ownReactions.find((r) => r.type === type)\n      ? void this.channelService.removeReaction(this.messageId!, type)\n      : void this.channelService.addReaction(this.messageId!, type);\n  }\n\n  isOwnReaction(reactionType: MessageReactionType) {\n    return !!this.ownReactions.find((r) => r.type === reactionType);\n  }\n\n  isOpenChange = (isOpen: boolean) => {\n    this.selectedReactionType = isOpen ? this.selectedReactionType : undefined;\n  };\n\n  private eventHandler = (event: Event) => {\n    if (!this.selectorContainer?.nativeElement.contains(event.target as Node)) {\n      this.isSelectorOpenChange.emit(false);\n    }\n  };\n\n  private watchForOutsideClicks() {\n    window.addEventListener('click', this.eventHandler);\n  }\n\n  private stopWatchForOutsideClicks() {\n    window.removeEventListener('click', this.eventHandler);\n  }\n\n  private setTooltipPosition() {\n    const tooltip = this.selectorTooltip?.nativeElement.getBoundingClientRect();\n    const target = this.currentTooltipTarget?.getBoundingClientRect();\n\n    const container =\n      this.selectorContainer?.nativeElement.getBoundingClientRect();\n\n    if (!tooltip || !target || !container) return;\n\n    const tooltipPosition =\n      tooltip.width === container.width || tooltip.x < container.x\n        ? 0\n        : target.left + target.width / 2 - container.left - tooltip.width / 2;\n\n    const arrowPosition =\n      target.x - tooltip.x + target.width / 2 - tooltipPosition;\n\n    this.tooltipPositions = {\n      tooltip: tooltipPosition,\n      arrow: arrowPosition,\n    };\n  }\n\n  private async fetchAllReactions() {\n    if (!this.messageId) {\n      return;\n    }\n    this.isLoading = true;\n    try {\n      this.reactions = await this.channelService.getMessageReactions(\n        this.messageId\n      );\n    } catch (error) {\n      this.selectedReactionType = undefined;\n    } finally {\n      this.isLoading = false;\n      this.cdRef.detectChanges();\n    }\n  }\n}\n","<div\n  *ngIf=\"existingReactions.length > 0\"\n  class=\"str-chat__reaction-list str-chat__message-reactions-container\"\n  [class.str-chat__reaction-list--reverse]=\"true\"\n  [class.str-chat__reaction-list-hidden]=\"isSelectorOpen\"\n  data-testid=\"reaction-list\"\n>\n  <ul class=\"str-chat__message-reactions\">\n    <li\n      class=\"str-chat__message-reaction\"\n      *ngFor=\"\n        let reactionType of existingReactions;\n        trackBy: trackByMessageReaction\n      \"\n      [class.str-chat__message-reaction-own]=\"isOwnReaction(reactionType)\"\n      data-testclass=\"emoji\"\n      (click)=\"reactionSelected($event, reactionType)\"\n      (keyup.enter)=\"reactionSelected($event, reactionType)\"\n    >\n      <span class=\"emoji str-chat__message-reaction-emoji\">\n        {{ getEmojiByReaction(reactionType) }}&nbsp;\n      </span>\n      <span\n        data-testclass=\"reaction-list-reaction-count\"\n        class=\"str-chat__message-reaction-count\"\n      >\n        {{ messageReactionCounts[reactionType] }}\n      </span>\n    </li>\n    <li>\n      <span\n        data-testid=\"reactions-count\"\n        class=\"str-chat__reaction-list--counter\"\n        >{{ reactionsCount }}</span\n      >\n    </li>\n  </ul>\n</div>\n\n<div\n  #selectorContainer\n  class=\"str-chat__reaction-selector str-chat__message-reaction-selector\"\n  *ngIf=\"isSelectorOpen\"\n  data-testid=\"reaction-selector\"\n>\n  <div\n    *ngIf=\"tooltipText\"\n    data-testid=\"tooltip\"\n    #selectorTooltip\n    class=\"str-chat__reaction-selector-tooltip\"\n    [ngStyle]=\"{\n      left: tooltipPositions?.tooltip + 'px',\n      visibility: tooltipPositions ? 'visible' : 'hidden'\n    }\"\n  >\n    <div\n      class=\"arrow\"\n      [ngStyle]=\"{ left: tooltipPositions?.arrow + 'px' }\"\n    ></div>\n    <span class=\"latest-user-username\">\n      {{ tooltipText }}\n    </span>\n  </div>\n  <ul\n    class=\"str-chat__message-reactions-list str-chat__message-reactions-options\"\n  >\n    <li\n      class=\"\n        str-chat__message-reactions-option\n        str-chat__message-reactions-list-item\n        str-chat__emoji\n      \"\n      *ngFor=\"\n        let reactionType of reactionOptions;\n        trackBy: trackByMessageReaction\n      \"\n      [class.str-chat__message-reactions-option-selected]=\"\n        isOwnReaction(reactionType)\n      \"\n      data-testclass=\"emoji-option\"\n      (click)=\"react(reactionType)\"\n      (keyup.enter)=\"react(reactionType)\"\n    >\n      <div\n        *ngIf=\"getLatestUserByReaction(reactionType) as user\"\n        class=\"latest-user str-chat__message-reactions-last-user\"\n        (click)=\"hideTooltip()\"\n        (keyup.enter)=\"hideTooltip()\"\n        (mouseenter)=\"showTooltip($event, reactionType)\"\n        (mouseleave)=\"hideTooltip()\"\n        attr.data-testid=\"{{ reactionType }}-last-user\"\n      >\n        <stream-avatar-placeholder\n          attr.data-testid=\"{{ reactionType }}-avatar\"\n          [imageUrl]=\"user.image\"\n          [name]=\"user.name || user.id\"\n          [size]=\"20\"\n          location=\"reaction\"\n        ></stream-avatar-placeholder>\n      </div>\n      <span\n        class=\"\n          emoji\n          str-chat__emoji-selector-emoji-angular\n          str-chat__message-reaction-emoji\n        \"\n      >\n        {{ getEmojiByReaction(reactionType) }}\n      </span>\n      <span\n        *ngIf=\"\n          messageReactionCounts[reactionType] &&\n          messageReactionCounts[reactionType]! > 0\n        \"\n        class=\"str-chat__message-reactions-list-item__count\"\n        attr.data-testid=\"{{ reactionType }}-reaction-count\"\n      >\n        {{ messageReactionCounts[reactionType] }}\n      </span>\n    </li>\n  </ul>\n</div>\n\n<ng-container *ngIf=\"selectedReactionType\">\n  <ng-container\n    *ngTemplateOutlet=\"\n      (customTemplatesService.modalTemplate$ | async) || defaultModal;\n      context: {\n        isOpen: !!selectedReactionType,\n        messageId: messageId,\n        reactionType: selectedReactionType,\n        isOpenChangeHandler: isOpenChange\n      }\n    \"\n  ></ng-container>\n</ng-container>\n\n<ng-template\n  #defaultModal\n  let-isOpen=\"isOpen\"\n  let-messageId=\"messageId\"\n  let-reactionType=\"reactionType\"\n  let-isOpenChangeHandler=\"isOpenChangeHandler\"\n>\n  <stream-modal\n    class=\"str-chat__message-reactions-details-modal\"\n    [isOpen]=\"isOpen\"\n    (isOpenChange)=\"isOpenChangeHandler($event)\"\n  >\n    <div class=\"str-chat__message-reactions-details\">\n      <div class=\"str-chat__message-reactions-details-reaction-types\">\n        <div\n          class=\"str-chat__message-reactions-details-reaction-type\"\n          *ngFor=\"\n            let reactionType of existingReactions;\n            trackBy: trackByMessageReaction\n          \"\n          attr.data-testid=\"reaction-details-selector-{{ reactionType }}\"\n          [class.str-chat__message-reactions-details-reaction-type--selected]=\"\n            reactionType === selectedReactionType\n          \"\n          (click)=\"selectedReactionType = reactionType; allUsers.scrollTop = 0\"\n          (keyup.enter)=\"\n            selectedReactionType = reactionType; allUsers.scrollTop = 0\n          \"\n        >\n          <span class=\"emoji str-chat__message-reaction-emoji\">\n            {{ getEmojiByReaction(reactionType) }}&nbsp;\n          </span>\n          <span class=\"str-chat__message-reaction-count\">\n            {{ messageReactionCounts[reactionType] }}\n          </span>\n        </div>\n      </div>\n      <div\n        class=\"\n          emoji\n          str-chat__message-reaction-emoji str-chat__message-reaction-emoji-big\n        \"\n      >\n        {{ getEmojiByReaction(selectedReactionType!) }}\n      </div>\n      <div\n        #allUsers\n        data-testid=\"all-reacting-users\"\n        class=\"str-chat__message-reactions-details-reacting-users\"\n      >\n        <stream-loading-indicator\n          *ngIf=\"isLoading; else reactions\"\n        ></stream-loading-indicator>\n        <ng-template #reactions>\n          <div\n            class=\"str-chat__message-reactions-details-reacting-user\"\n            *ngFor=\"\n              let user of getAllUsersByReaction(selectedReactionType);\n              trackBy: trackByUserId\n            \"\n          >\n            <stream-avatar-placeholder\n              data-testclass=\"avatar\"\n              class=\"str-chat__avatar str-chat__avatar--circle\"\n              style=\"max-height: 100%; overflow-y: hidden\"\n              [size]=\"30\"\n              [imageUrl]=\"user.image\"\n              [name]=\"user.name\"\n              type=\"user\"\n              location=\"reaction\"\n              [user]=\"user\"\n            ></stream-avatar-placeholder>\n            <span\n              data-testclass=\"reaction-user-username\"\n              class=\"str-chat__user-item--name\"\n              >{{ user.name }}</span\n            >\n          </div>\n        </ng-template>\n      </div>\n    </div>\n  </stream-modal>\n</ng-template>\n"]}
|