stream-chat-angular 6.4.1 → 7.0.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/README.md +12 -18
- package/assets/version.d.ts +1 -1
- package/{esm2020 → esm2022}/assets/version.mjs +2 -2
- package/{esm2020 → esm2022}/lib/attachment-configuration.service.mjs +4 -4
- package/esm2022/lib/attachment-list/attachment-list.component.mjs +232 -0
- package/esm2022/lib/attachment-preview-list/attachment-preview-list.component.mjs +55 -0
- package/{esm2020 → esm2022}/lib/attachment.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/avatar/avatar.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/avatar-placeholder/avatar-placeholder.component.mjs +6 -6
- package/esm2022/lib/channel/channel.component.mjs +45 -0
- package/{esm2020 → esm2022}/lib/channel-header/channel-header.component.mjs +6 -6
- package/esm2022/lib/channel-list/channel-list.component.mjs +47 -0
- package/{esm2020 → esm2022}/lib/channel-preview/channel-preview.component.mjs +6 -6
- package/esm2022/lib/channel.service.mjs +1575 -0
- package/{esm2020 → esm2022}/lib/chat-client.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/custom-templates.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/date-parser.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/icon/icon-placeholder/icon-placeholder.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/icon/icon.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/icon/icon.module.mjs +11 -11
- package/{esm2020 → esm2022}/lib/icon/loading-indicator/loading-indicator.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +6 -6
- package/esm2022/lib/message/message.component.mjs +501 -0
- package/{esm2020 → esm2022}/lib/message-actions-box/message-actions-box.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/message-actions.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-blocked/message-blocked.component.mjs +4 -4
- package/{esm2020 → esm2022}/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/message-input/emoji-input.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-input/message-input-config.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-input/message-input.component.mjs +7 -7
- package/{esm2020 → esm2022}/lib/message-input/textarea/textarea.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-input/textarea.directive.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-input/voice-recorder.service.mjs +5 -5
- package/esm2022/lib/message-list/message-list.component.mjs +717 -0
- package/{esm2020 → esm2022}/lib/message-reactions/message-reactions.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/message-reactions-selector/message-reactions-selector.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-reactions.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-text/message-text.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/message.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/modal/modal.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/modal/stream-modal.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/notification/notification.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/notification-list/notification-list.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/notification-list/stream-notification.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/notification.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/paginated-list/paginated-list.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/paginated-list/stream-paginated-list.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/polls/base-poll.component.mjs +5 -5
- package/esm2022/lib/polls/poll/poll.component.mjs +34 -0
- package/{esm2020 → esm2022}/lib/polls/poll-actions/add-option/add-option.component.mjs +4 -4
- package/{esm2020 → esm2022}/lib/polls/poll-actions/poll-actions.component.mjs +6 -6
- package/esm2022/lib/polls/poll-actions/poll-answers-list/poll-answers-list.component.mjs +82 -0
- package/esm2022/lib/polls/poll-actions/poll-results/poll-results-list/poll-results-list.component.mjs +63 -0
- package/{esm2020 → esm2022}/lib/polls/poll-actions/poll-results/poll-vote/poll-vote.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/polls/poll-actions/poll-results/poll-vote-results-list/poll-vote-results-list.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/polls/poll-actions/upsert-answer/upsert-answer.component.mjs +4 -4
- package/esm2022/lib/polls/poll-composer/poll-composer.component.mjs +134 -0
- package/{esm2020 → esm2022}/lib/polls/poll-header/poll-header.component.mjs +4 -4
- package/{esm2020 → esm2022}/lib/polls/poll-option-selector/poll-option-selector.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/polls/poll-options-list/poll-options-list.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/polls/poll-preview/poll-preview.component.mjs +4 -4
- package/{esm2020 → esm2022}/lib/polls/stream-polls.module.mjs +41 -41
- package/esm2022/lib/stream-autocomplete-textarea.module.mjs +33 -0
- package/{esm2020 → esm2022}/lib/stream-avatar.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/stream-chat.module.mjs +67 -67
- package/esm2022/lib/stream-i18n.service.mjs +34 -0
- package/esm2022/lib/stream-textarea.module.mjs +31 -0
- package/{esm2020 → esm2022}/lib/theme.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/thread/thread.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/transliteration.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/types.mjs +1 -1
- package/{esm2020 → esm2022}/lib/user-list/user-list.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/voice-recorder/amplitude-recorder.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/voice-recorder/audio-recorder.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/voice-recorder/transcoder.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/voice-recorder/voice-recorder.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/voice-recorder/voice-recorder.module.mjs +9 -9
- package/{esm2020 → esm2022}/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +5 -5
- package/{esm2020 → esm2022}/lib/voice-recording/voice-recording.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/voice-recording/voice-recording.module.mjs +5 -5
- package/{fesm2020 → fesm2022}/stream-chat-angular.mjs +463 -470
- package/fesm2022/stream-chat-angular.mjs.map +1 -0
- package/lib/attachment-list/attachment-list.component.d.ts +1 -1
- package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +1 -1
- package/lib/avatar/avatar.component.d.ts +1 -1
- package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +1 -1
- package/lib/channel-preview/channel-preview.component.d.ts +1 -1
- package/lib/chat-client.service.d.ts +1 -1
- package/lib/custom-templates.service.d.ts +1 -0
- package/lib/icon/icon-placeholder/icon-placeholder.component.d.ts +1 -1
- package/lib/icon/icon.component.d.ts +2 -2
- package/lib/message/message.component.d.ts +1 -1
- package/lib/message-actions-box/message-actions-box.component.d.ts +1 -1
- package/lib/message-blocked/message-blocked.component.d.ts +1 -1
- package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +1 -1
- package/lib/message-input/message-input.component.d.ts +1 -1
- package/lib/message-input/textarea/textarea.component.d.ts +1 -1
- package/lib/message-input/textarea.directive.d.ts +1 -1
- package/lib/message-list/group-styles.d.ts +1 -1
- package/lib/message-list/message-list.component.d.ts +1 -1
- package/lib/message-reactions/message-reactions.component.d.ts +1 -1
- package/lib/message-reactions-selector/message-reactions-selector.component.d.ts +1 -1
- package/lib/message-text/message-text.component.d.ts +2 -2
- package/lib/modal/modal.component.d.ts +1 -1
- package/lib/notification/notification.component.d.ts +1 -1
- package/lib/paginated-list/paginated-list.component.d.ts +5 -2
- package/lib/polls/base-poll.component.d.ts +1 -1
- package/lib/polls/poll-actions/add-option/add-option.component.d.ts +1 -1
- package/lib/polls/poll-actions/poll-actions.component.d.ts +2 -2
- package/lib/polls/poll-actions/poll-answers-list/poll-answers-list.component.d.ts +1 -1
- package/lib/polls/poll-actions/poll-results/poll-vote/poll-vote.component.d.ts +1 -1
- package/lib/polls/poll-actions/poll-results/poll-vote-results-list/poll-vote-results-list.component.d.ts +1 -1
- package/lib/polls/poll-actions/upsert-answer/upsert-answer.component.d.ts +1 -1
- package/lib/polls/poll-composer/poll-composer.component.d.ts +2 -2
- package/lib/polls/poll-header/poll-header.component.d.ts +1 -1
- package/lib/polls/poll-option-selector/poll-option-selector.component.d.ts +1 -1
- package/lib/polls/poll-options-list/poll-options-list.component.d.ts +1 -1
- package/lib/types.d.ts +79 -79
- package/lib/user-list/user-list.component.d.ts +1 -1
- package/lib/voice-recorder/amplitude-recorder.service.d.ts +2 -2
- package/lib/voice-recorder/media-recorder.d.ts +2 -2
- package/lib/voice-recorder/transcoder.service.d.ts +4 -4
- package/lib/voice-recorder/voice-recorder.component.d.ts +1 -1
- package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +1 -1
- package/lib/voice-recording/voice-recording.component.d.ts +1 -1
- package/package.json +11 -17
- package/src/assets/version.ts +1 -1
- package/esm2020/lib/attachment-list/attachment-list.component.mjs +0 -232
- package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +0 -55
- package/esm2020/lib/channel/channel.component.mjs +0 -45
- package/esm2020/lib/channel-list/channel-list.component.mjs +0 -47
- package/esm2020/lib/channel.service.mjs +0 -1575
- package/esm2020/lib/message/message.component.mjs +0 -501
- package/esm2020/lib/message-list/message-list.component.mjs +0 -717
- package/esm2020/lib/polls/poll/poll.component.mjs +0 -34
- package/esm2020/lib/polls/poll-actions/poll-answers-list/poll-answers-list.component.mjs +0 -82
- package/esm2020/lib/polls/poll-actions/poll-results/poll-results-list/poll-results-list.component.mjs +0 -63
- package/esm2020/lib/polls/poll-composer/poll-composer.component.mjs +0 -134
- package/esm2020/lib/stream-autocomplete-textarea.module.mjs +0 -33
- package/esm2020/lib/stream-i18n.service.mjs +0 -41
- package/esm2020/lib/stream-textarea.module.mjs +0 -31
- package/fesm2015/stream-chat-angular.mjs +0 -10596
- package/fesm2015/stream-chat-angular.mjs.map +0 -1
- package/fesm2020/stream-chat-angular.mjs.map +0 -1
- /package/{esm2020 → esm2022}/assets/i18n/en.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/channel-query.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/file-utils.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/format-duration.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/get-channel-display-text.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/get-message-translation.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/injection-tokens.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/is-image-attachment.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/is-on-separate-date.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/is-safari.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/list-users.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/message-input/textarea.interface.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/message-list/group-styles.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/message-preview.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/parse-date.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/polls/poll-composer/validators.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/polls/unique.validator.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/read-by.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/types-custom.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/virtualized-list.service.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/virtualized-message-list.service.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/voice-recorder/media-recorder.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/voice-recorder/mp3-transcoder.mjs +0 -0
- /package/{esm2020 → esm2022}/lib/wave-form-sampler.mjs +0 -0
- /package/{esm2020 → esm2022}/public-api.mjs +0 -0
- /package/{esm2020 → esm2022}/stream-chat-angular.mjs +0 -0
- /package/{esm2020 → esm2022}/stream-chat.mjs +0 -0
|
@@ -0,0 +1,501 @@
|
|
|
1
|
+
import { Component, Input, ChangeDetectionStrategy, ViewChild, } from '@angular/core';
|
|
2
|
+
import { pairwise, take } from 'rxjs';
|
|
3
|
+
import { listUsers } from '../list-users';
|
|
4
|
+
import { isSafari } from '../is-safari';
|
|
5
|
+
import * as i0 from "@angular/core";
|
|
6
|
+
import * as i1 from "../chat-client.service";
|
|
7
|
+
import * as i2 from "../channel.service";
|
|
8
|
+
import * as i3 from "../custom-templates.service";
|
|
9
|
+
import * as i4 from "../date-parser.service";
|
|
10
|
+
import * as i5 from "../message-actions.service";
|
|
11
|
+
import * as i6 from "@ngx-translate/core";
|
|
12
|
+
import * as i7 from "@angular/common";
|
|
13
|
+
import * as i8 from "ngx-float-ui";
|
|
14
|
+
import * as i9 from "../avatar-placeholder/avatar-placeholder.component";
|
|
15
|
+
import * as i10 from "../icon/icon-placeholder/icon-placeholder.component";
|
|
16
|
+
import * as i11 from "../icon/loading-indicator-placeholder/loading-indicator-placeholder.component";
|
|
17
|
+
import * as i12 from "../message-actions-box/message-actions-box.component";
|
|
18
|
+
import * as i13 from "../attachment-list/attachment-list.component";
|
|
19
|
+
import * as i14 from "../message-reactions/message-reactions.component";
|
|
20
|
+
import * as i15 from "../message-text/message-text.component";
|
|
21
|
+
import * as i16 from "../message-blocked/message-blocked.component";
|
|
22
|
+
/**
|
|
23
|
+
* The `Message` component displays a message with additional information such as sender and date, and enables [interaction with the message (i.e. edit or react)](/chat/docs/sdk/angular/concepts/message-interactions/).
|
|
24
|
+
*/
|
|
25
|
+
export class MessageComponent {
|
|
26
|
+
constructor(chatClientService, channelService, customTemplatesService, cdRef, dateParser, messageActionsService, ngZone, translateService) {
|
|
27
|
+
this.chatClientService = chatClientService;
|
|
28
|
+
this.channelService = channelService;
|
|
29
|
+
this.customTemplatesService = customTemplatesService;
|
|
30
|
+
this.cdRef = cdRef;
|
|
31
|
+
this.dateParser = dateParser;
|
|
32
|
+
this.messageActionsService = messageActionsService;
|
|
33
|
+
this.ngZone = ngZone;
|
|
34
|
+
this.translateService = translateService;
|
|
35
|
+
/**
|
|
36
|
+
* The list of [channel capabilities](/chat/docs/javascript/channel_capabilities/) that are enabled for the current user, the list of [supported interactions](/chat/docs/sdk/angular/concepts/message-interactions/) can be found in our message interaction guide. Unathorized actions won't be displayed on the UI. The [`MessageList`](/chat/docs/sdk/angular/components/MessageListComponent/) component automatically sets this based on [channel capabilities](/chat/docs/javascript/channel_capabilities/).
|
|
37
|
+
*/
|
|
38
|
+
this.enabledMessageActions = [];
|
|
39
|
+
/**
|
|
40
|
+
* Determines if the message is being dispalyed in a channel or in a [thread](/chat/docs/javascript/threads/).
|
|
41
|
+
*/
|
|
42
|
+
this.mode = 'main';
|
|
43
|
+
/**
|
|
44
|
+
* Highlighting is used to add visual emphasize to a message when jumping to the message
|
|
45
|
+
*/
|
|
46
|
+
this.isHighlighted = false;
|
|
47
|
+
this.isEditedFlagOpened = false;
|
|
48
|
+
this.shouldDisplayTranslationNotice = false;
|
|
49
|
+
this.displayedMessageTextContent = 'original';
|
|
50
|
+
this.nestedModalState = 'closed';
|
|
51
|
+
this.shouldDisplayThreadLink = false;
|
|
52
|
+
this.isSentByCurrentUser = false;
|
|
53
|
+
this.readByText = '';
|
|
54
|
+
this.lastReadUser = undefined;
|
|
55
|
+
this.isOnlyReadByMe = false;
|
|
56
|
+
this.isReadByMultipleUsers = false;
|
|
57
|
+
this.isMessageDeliveredAndRead = false;
|
|
58
|
+
this.parsedDate = '';
|
|
59
|
+
this.pasedEditedDate = '';
|
|
60
|
+
this.areOptionsVisible = false;
|
|
61
|
+
this.hasAttachment = false;
|
|
62
|
+
this.hasReactions = false;
|
|
63
|
+
this.replyCountParam = {
|
|
64
|
+
replyCount: undefined,
|
|
65
|
+
};
|
|
66
|
+
this.areMessageOptionsOpen = false;
|
|
67
|
+
this.canDisplayReadStatus = false;
|
|
68
|
+
this.hasTouchSupport = isSafari()
|
|
69
|
+
? 'ontouchstart' in window || navigator.maxTouchPoints > 0
|
|
70
|
+
: window.matchMedia('(any-hover: none)').matches;
|
|
71
|
+
this.subscriptions = [];
|
|
72
|
+
this.isViewInited = false;
|
|
73
|
+
this.shouldPreventMessageMenuClose = false;
|
|
74
|
+
this._visibleMessageActionsCount = 0;
|
|
75
|
+
}
|
|
76
|
+
get visibleMessageActionsCount() {
|
|
77
|
+
return this._visibleMessageActionsCount;
|
|
78
|
+
}
|
|
79
|
+
set visibleMessageActionsCount(count) {
|
|
80
|
+
this._visibleMessageActionsCount = count;
|
|
81
|
+
if (this.areOptionsVisible && this._visibleMessageActionsCount === 0) {
|
|
82
|
+
this.areOptionsVisible = false;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
ngOnInit() {
|
|
86
|
+
this.subscriptions.push(this.chatClientService.user$.subscribe((u) => {
|
|
87
|
+
if (u?.id !== this.userId) {
|
|
88
|
+
this.userId = u?.id;
|
|
89
|
+
this.setIsSentByCurrentUser();
|
|
90
|
+
this.setLastReadUser();
|
|
91
|
+
if (this.isViewInited) {
|
|
92
|
+
this.cdRef.detectChanges();
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}));
|
|
96
|
+
this.subscriptions.push(this.messageActionsService.customActions$.subscribe(() => {
|
|
97
|
+
if (this.message) {
|
|
98
|
+
const numberOfEnabledActions = this.messageActionsService.getAuthorizedMessageActionsCount(this.message, this.enabledMessageActions);
|
|
99
|
+
if (numberOfEnabledActions !== this.visibleMessageActionsCount) {
|
|
100
|
+
this.visibleMessageActionsCount = numberOfEnabledActions;
|
|
101
|
+
if (this.isViewInited) {
|
|
102
|
+
this.cdRef.detectChanges();
|
|
103
|
+
}
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
}));
|
|
107
|
+
this.subscriptions.push(this.channelService.activeChannel$.subscribe((activeChannel) => {
|
|
108
|
+
const newChannelMemberCount = activeChannel?.data?.member_count;
|
|
109
|
+
if (newChannelMemberCount !== this.channelMemberCount) {
|
|
110
|
+
const shouldUpdateText = this.channelMemberCount !== undefined &&
|
|
111
|
+
newChannelMemberCount != undefined &&
|
|
112
|
+
((this.channelMemberCount <= 1000 && newChannelMemberCount > 100) ||
|
|
113
|
+
(this.channelMemberCount > 100 && newChannelMemberCount <= 100));
|
|
114
|
+
this.channelMemberCount = activeChannel?.data?.member_count;
|
|
115
|
+
if (this.message &&
|
|
116
|
+
this.message.cid === activeChannel?.cid &&
|
|
117
|
+
shouldUpdateText) {
|
|
118
|
+
this.updateReadByText();
|
|
119
|
+
}
|
|
120
|
+
}
|
|
121
|
+
}));
|
|
122
|
+
this.subscriptions.push(this.messageActionsService.modalOpenedForMessage.subscribe((messageId) => {
|
|
123
|
+
if (messageId === this.message?.id) {
|
|
124
|
+
this.nestedModalState = 'opened';
|
|
125
|
+
}
|
|
126
|
+
}));
|
|
127
|
+
}
|
|
128
|
+
ngOnChanges(changes) {
|
|
129
|
+
if (changes.message) {
|
|
130
|
+
this.displayedMessageTextContent = this.message?.translation
|
|
131
|
+
? 'translation'
|
|
132
|
+
: 'original';
|
|
133
|
+
this.shouldDisplayTranslationNotice =
|
|
134
|
+
this.displayedMessageTextContent === 'translation';
|
|
135
|
+
const originalAttachments = this.message?.quoted_message?.attachments;
|
|
136
|
+
this.quotedMessageAttachments =
|
|
137
|
+
originalAttachments && originalAttachments.length
|
|
138
|
+
? [originalAttachments[0]]
|
|
139
|
+
: [];
|
|
140
|
+
this.setIsSentByCurrentUser();
|
|
141
|
+
this.setLastReadUser();
|
|
142
|
+
this.updateReadByText();
|
|
143
|
+
this.isOnlyReadByMe = !!(this.message &&
|
|
144
|
+
this.message.readBy &&
|
|
145
|
+
this.message.readBy.length === 0);
|
|
146
|
+
this.isReadByMultipleUsers = !!(this.message &&
|
|
147
|
+
this.message.readBy &&
|
|
148
|
+
this.message.readBy.length > 1);
|
|
149
|
+
this.isMessageDeliveredAndRead = !!(this.message &&
|
|
150
|
+
this.message.readBy &&
|
|
151
|
+
this.message.status === 'received' &&
|
|
152
|
+
this.message.readBy.length > 0);
|
|
153
|
+
this.parsedDate =
|
|
154
|
+
(this.message &&
|
|
155
|
+
this.message.created_at &&
|
|
156
|
+
this.dateParser.parseDateTime(this.message.created_at)) ||
|
|
157
|
+
'';
|
|
158
|
+
this.pasedEditedDate =
|
|
159
|
+
(this.message &&
|
|
160
|
+
this.message.message_text_updated_at &&
|
|
161
|
+
this.dateParser.parseDateTime(new Date(this.message.message_text_updated_at))) ||
|
|
162
|
+
'';
|
|
163
|
+
this.hasAttachment =
|
|
164
|
+
!!this.message?.attachments && !!this.message.attachments.length;
|
|
165
|
+
this.hasReactions =
|
|
166
|
+
!!this.message?.reaction_groups &&
|
|
167
|
+
Object.keys(this.message.reaction_groups).length > 0;
|
|
168
|
+
this.replyCountParam = { replyCount: this.message?.reply_count };
|
|
169
|
+
}
|
|
170
|
+
if (changes.enabledMessageActions) {
|
|
171
|
+
this.canReactToMessage =
|
|
172
|
+
this.enabledMessageActions.indexOf('send-reaction') !== -1;
|
|
173
|
+
this.canReceiveReadEvents =
|
|
174
|
+
this.enabledMessageActions.indexOf('read-events') !== -1;
|
|
175
|
+
this.canDisplayReadStatus =
|
|
176
|
+
this.canReceiveReadEvents !== false &&
|
|
177
|
+
this.enabledMessageActions.indexOf('read-events') !== -1;
|
|
178
|
+
}
|
|
179
|
+
if (changes.message || changes.enabledMessageActions || changes.mode) {
|
|
180
|
+
this.shouldDisplayThreadLink =
|
|
181
|
+
!!this.message?.reply_count && this.mode !== 'thread';
|
|
182
|
+
}
|
|
183
|
+
if (changes.message || changes.mode || changes.enabledMessageActions) {
|
|
184
|
+
this.areOptionsVisible = this.message
|
|
185
|
+
? !(!this.message.type ||
|
|
186
|
+
this.message.type === 'error' ||
|
|
187
|
+
this.message.type === 'system' ||
|
|
188
|
+
this.message.type === 'deleted' ||
|
|
189
|
+
this.message.type === 'ephemeral' ||
|
|
190
|
+
this.message.status === 'failed' ||
|
|
191
|
+
this.message.status === 'sending' ||
|
|
192
|
+
(this.mode === 'thread' && !this.message.parent_id) ||
|
|
193
|
+
this.message.deleted_at ||
|
|
194
|
+
this.enabledMessageActions.length === 0)
|
|
195
|
+
: false;
|
|
196
|
+
}
|
|
197
|
+
if (changes.message || changes.enabledMessageActions) {
|
|
198
|
+
if (this.message) {
|
|
199
|
+
this.visibleMessageActionsCount =
|
|
200
|
+
this.messageActionsService.getAuthorizedMessageActionsCount(this.message, this.enabledMessageActions);
|
|
201
|
+
}
|
|
202
|
+
else {
|
|
203
|
+
this.visibleMessageActionsCount = 0;
|
|
204
|
+
}
|
|
205
|
+
}
|
|
206
|
+
}
|
|
207
|
+
ngAfterViewInit() {
|
|
208
|
+
this.isViewInited = true;
|
|
209
|
+
if (this.hasTouchSupport && this.messageBubble?.nativeElement) {
|
|
210
|
+
this.ngZone.runOutsideAngular(() => {
|
|
211
|
+
this.registerMenuTriggerEventHandlers();
|
|
212
|
+
});
|
|
213
|
+
}
|
|
214
|
+
this.subscriptions.push(this.messageActionsService.messageMenuOpenedFor$.subscribe((id) => {
|
|
215
|
+
if ((id === undefined || this.message?.id !== id) &&
|
|
216
|
+
this.areMessageOptionsOpen) {
|
|
217
|
+
this.messageMenuTrigger?.hide();
|
|
218
|
+
}
|
|
219
|
+
}));
|
|
220
|
+
this.subscriptions.push(this.messageActionsService.modalOpenedForMessage
|
|
221
|
+
.pipe(pairwise())
|
|
222
|
+
.subscribe(([oldId, newId]) => {
|
|
223
|
+
if (newId === this.message?.id) {
|
|
224
|
+
this.nestedModalState = 'opened';
|
|
225
|
+
}
|
|
226
|
+
else if (oldId === this.message?.id && newId === undefined) {
|
|
227
|
+
this.nestedModalState = 'closed';
|
|
228
|
+
}
|
|
229
|
+
}));
|
|
230
|
+
}
|
|
231
|
+
ngOnDestroy() {
|
|
232
|
+
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
233
|
+
if (this.showMessageMenuTimeout) {
|
|
234
|
+
clearTimeout(this.showMessageMenuTimeout);
|
|
235
|
+
}
|
|
236
|
+
}
|
|
237
|
+
mousePushedDown(event) {
|
|
238
|
+
if (!this.hasTouchSupport ||
|
|
239
|
+
event.button !== 0 ||
|
|
240
|
+
!this.areOptionsVisible) {
|
|
241
|
+
return;
|
|
242
|
+
}
|
|
243
|
+
this.startMessageMenuShowTimer({ fromTouch: false });
|
|
244
|
+
}
|
|
245
|
+
mouseReleased() {
|
|
246
|
+
this.stopMessageMenuShowTimer();
|
|
247
|
+
}
|
|
248
|
+
touchStarted() {
|
|
249
|
+
if (!this.areOptionsVisible) {
|
|
250
|
+
return;
|
|
251
|
+
}
|
|
252
|
+
this.startMessageMenuShowTimer({ fromTouch: true });
|
|
253
|
+
}
|
|
254
|
+
touchEnded() {
|
|
255
|
+
this.stopMessageMenuShowTimer();
|
|
256
|
+
}
|
|
257
|
+
messageBubbleClicked(event) {
|
|
258
|
+
if (!this.hasTouchSupport) {
|
|
259
|
+
return;
|
|
260
|
+
}
|
|
261
|
+
if (this.shouldPreventMessageMenuClose) {
|
|
262
|
+
event.stopPropagation();
|
|
263
|
+
this.shouldPreventMessageMenuClose = false;
|
|
264
|
+
}
|
|
265
|
+
else if (this.areMessageOptionsOpen) {
|
|
266
|
+
this.messageMenuTrigger?.hide();
|
|
267
|
+
}
|
|
268
|
+
}
|
|
269
|
+
messageOptionsButtonClicked() {
|
|
270
|
+
if (!this.message) {
|
|
271
|
+
return;
|
|
272
|
+
}
|
|
273
|
+
if (this.messageActionsService.customActionClickHandler) {
|
|
274
|
+
this.messageActionsService.customActionClickHandler({
|
|
275
|
+
message: this.message,
|
|
276
|
+
enabledActions: this.enabledMessageActions,
|
|
277
|
+
customActions: this.messageActionsService.customActions$.getValue(),
|
|
278
|
+
isMine: this.isSentByCurrentUser,
|
|
279
|
+
messageTextHtmlElement: this.messageTextElement?.nativeElement,
|
|
280
|
+
});
|
|
281
|
+
}
|
|
282
|
+
else {
|
|
283
|
+
this.areMessageOptionsOpen = !this.areMessageOptionsOpen;
|
|
284
|
+
}
|
|
285
|
+
}
|
|
286
|
+
messageActionsBoxClicked(floatingContent) {
|
|
287
|
+
floatingContent.hide();
|
|
288
|
+
}
|
|
289
|
+
getAttachmentListContext() {
|
|
290
|
+
return {
|
|
291
|
+
messageId: this.message?.id || '',
|
|
292
|
+
attachments: this.message?.attachments || [],
|
|
293
|
+
parentMessageId: this.message?.parent_id,
|
|
294
|
+
imageModalStateChangeHandler: (state) => (this.nestedModalState = state),
|
|
295
|
+
};
|
|
296
|
+
}
|
|
297
|
+
getMessageContext() {
|
|
298
|
+
return {
|
|
299
|
+
message: this.message,
|
|
300
|
+
enabledMessageActions: this.enabledMessageActions,
|
|
301
|
+
isHighlighted: this.isHighlighted,
|
|
302
|
+
isLastSentMessage: this.isLastSentMessage,
|
|
303
|
+
mode: this.mode,
|
|
304
|
+
customActions: this.messageActionsService.customActions$.getValue(),
|
|
305
|
+
parsedDate: this.parsedDate,
|
|
306
|
+
scroll$: this.scroll$,
|
|
307
|
+
};
|
|
308
|
+
}
|
|
309
|
+
getMessageTextContext() {
|
|
310
|
+
return {
|
|
311
|
+
message: this.message,
|
|
312
|
+
isQuoted: false,
|
|
313
|
+
shouldTranslate: this.displayedMessageTextContent === 'translation',
|
|
314
|
+
};
|
|
315
|
+
}
|
|
316
|
+
getQuotedMessageTextContext() {
|
|
317
|
+
return {
|
|
318
|
+
message: this.message?.quoted_message,
|
|
319
|
+
isQuoted: true,
|
|
320
|
+
shouldTranslate: this.displayedMessageTextContent === 'translation',
|
|
321
|
+
};
|
|
322
|
+
}
|
|
323
|
+
getQuotedMessageAttachmentListContext() {
|
|
324
|
+
return {
|
|
325
|
+
messageId: this.message?.quoted_message?.id || '',
|
|
326
|
+
attachments: this.quotedMessageAttachments,
|
|
327
|
+
parentMessageId: this?.message?.quoted_message?.parent_id,
|
|
328
|
+
};
|
|
329
|
+
}
|
|
330
|
+
getMessageReactionsContext() {
|
|
331
|
+
return {
|
|
332
|
+
messageReactionCounts: this.message?.reaction_counts || {},
|
|
333
|
+
latestReactions: this.message?.latest_reactions || [],
|
|
334
|
+
messageId: this.message?.id,
|
|
335
|
+
ownReactions: this.message?.own_reactions || [],
|
|
336
|
+
messageReactionGroups: this.message?.reaction_groups || {},
|
|
337
|
+
};
|
|
338
|
+
}
|
|
339
|
+
messageClicked() {
|
|
340
|
+
if (this.message?.status === 'failed' &&
|
|
341
|
+
this.message?.errorStatusCode !== 403) {
|
|
342
|
+
this.resendMessage();
|
|
343
|
+
}
|
|
344
|
+
else if (this.message?.type === 'error' &&
|
|
345
|
+
this.message?.moderation_details) {
|
|
346
|
+
this.openMessageBouncePrompt();
|
|
347
|
+
}
|
|
348
|
+
else {
|
|
349
|
+
this.isEditedFlagOpened = !this.isEditedFlagOpened;
|
|
350
|
+
}
|
|
351
|
+
}
|
|
352
|
+
resendMessage() {
|
|
353
|
+
void this.channelService.resendMessage(this.message);
|
|
354
|
+
}
|
|
355
|
+
setAsActiveParentMessage() {
|
|
356
|
+
void this.channelService.setAsActiveParentMessage(this.message);
|
|
357
|
+
}
|
|
358
|
+
getMessageActionsBoxContext() {
|
|
359
|
+
return {
|
|
360
|
+
isMine: this.isSentByCurrentUser,
|
|
361
|
+
enabledActions: this.enabledMessageActions,
|
|
362
|
+
message: this.message,
|
|
363
|
+
messageTextHtmlElement: this.messageTextElement?.nativeElement,
|
|
364
|
+
};
|
|
365
|
+
}
|
|
366
|
+
getDeliveredStatusContext() {
|
|
367
|
+
return {
|
|
368
|
+
message: this.message,
|
|
369
|
+
};
|
|
370
|
+
}
|
|
371
|
+
getSendingStatusContext() {
|
|
372
|
+
return {
|
|
373
|
+
message: this.message,
|
|
374
|
+
};
|
|
375
|
+
}
|
|
376
|
+
getReadStatusContext() {
|
|
377
|
+
return {
|
|
378
|
+
message: this.message,
|
|
379
|
+
readByText: this.readByText,
|
|
380
|
+
};
|
|
381
|
+
}
|
|
382
|
+
getMessageMetadataContext() {
|
|
383
|
+
return { message: this.message };
|
|
384
|
+
}
|
|
385
|
+
getMessageBlockedContext() {
|
|
386
|
+
return {
|
|
387
|
+
message: this.message,
|
|
388
|
+
isMyMessage: this.isSentByCurrentUser,
|
|
389
|
+
};
|
|
390
|
+
}
|
|
391
|
+
jumpToMessage(messageId, parentMessageId) {
|
|
392
|
+
void this.channelService.jumpToMessage(messageId, parentMessageId);
|
|
393
|
+
}
|
|
394
|
+
openMessageBouncePrompt() {
|
|
395
|
+
this.channelService.bouncedMessage$.next(this.message);
|
|
396
|
+
}
|
|
397
|
+
displayTranslatedMessage() {
|
|
398
|
+
this.shouldDisplayTranslationNotice = true;
|
|
399
|
+
this.displayedMessageTextContent = 'translation';
|
|
400
|
+
}
|
|
401
|
+
displayOriginalMessage() {
|
|
402
|
+
this.displayedMessageTextContent = 'original';
|
|
403
|
+
}
|
|
404
|
+
updateReadByText() {
|
|
405
|
+
const others = this.translateService.instant('streamChat.and others');
|
|
406
|
+
const hasMoreThan100Members = (this.channelMemberCount ?? 0) > 100;
|
|
407
|
+
this.readByText = this.message?.readBy
|
|
408
|
+
? listUsers(this.message.readBy, !hasMoreThan100Members, others)
|
|
409
|
+
: '';
|
|
410
|
+
}
|
|
411
|
+
setIsSentByCurrentUser() {
|
|
412
|
+
this.isSentByCurrentUser = this.message?.user?.id === this.userId;
|
|
413
|
+
}
|
|
414
|
+
setLastReadUser() {
|
|
415
|
+
this.lastReadUser = this.message?.readBy?.filter((u) => u.id !== this.userId)[0];
|
|
416
|
+
}
|
|
417
|
+
startMessageMenuShowTimer(options) {
|
|
418
|
+
this.stopMessageMenuShowTimer();
|
|
419
|
+
if (this.scroll$) {
|
|
420
|
+
this.subscriptions.push(this.scroll$.pipe(take(1)).subscribe(() => {
|
|
421
|
+
this.stopMessageMenuShowTimer();
|
|
422
|
+
}));
|
|
423
|
+
}
|
|
424
|
+
this.showMessageMenuTimeout = setTimeout(() => {
|
|
425
|
+
if (!this.message) {
|
|
426
|
+
return;
|
|
427
|
+
}
|
|
428
|
+
this.ngZone.run(() => {
|
|
429
|
+
if (this.messageActionsService.customActionClickHandler) {
|
|
430
|
+
this.messageActionsService.customActionClickHandler({
|
|
431
|
+
message: this.message,
|
|
432
|
+
enabledActions: this.enabledMessageActions,
|
|
433
|
+
customActions: this.messageActionsService.customActions$.getValue(),
|
|
434
|
+
isMine: this.isSentByCurrentUser,
|
|
435
|
+
messageTextHtmlElement: this.messageTextElement?.nativeElement,
|
|
436
|
+
});
|
|
437
|
+
return;
|
|
438
|
+
}
|
|
439
|
+
else {
|
|
440
|
+
this.shouldPreventMessageMenuClose = !options.fromTouch;
|
|
441
|
+
// Fix for iOS Safari: iOS Safari won't close the input if we open message menu
|
|
442
|
+
// The virtual keyboard can hide parts of the message menu, so we close the input here
|
|
443
|
+
if (document.activeElement &&
|
|
444
|
+
typeof document.activeElement.blur !==
|
|
445
|
+
'undefined')
|
|
446
|
+
document.activeElement.blur();
|
|
447
|
+
this.messageMenuTrigger.show();
|
|
448
|
+
this.messageActionsService.messageMenuOpenedFor$.next(this.message?.id);
|
|
449
|
+
}
|
|
450
|
+
if (this.isViewInited) {
|
|
451
|
+
this.cdRef.detectChanges();
|
|
452
|
+
}
|
|
453
|
+
this.showMessageMenuTimeout = undefined;
|
|
454
|
+
});
|
|
455
|
+
}, 400);
|
|
456
|
+
}
|
|
457
|
+
registerMenuTriggerEventHandlers() {
|
|
458
|
+
this.messageBubble.nativeElement.addEventListener('touchstart', () => this.touchStarted());
|
|
459
|
+
this.messageBubble.nativeElement.addEventListener('touchend', () => this.touchEnded());
|
|
460
|
+
this.messageBubble.nativeElement.addEventListener('mousedown', (e) => this.mousePushedDown(e));
|
|
461
|
+
this.messageBubble.nativeElement.addEventListener('mouseup', () => this.mouseReleased());
|
|
462
|
+
this.messageBubble.nativeElement.addEventListener('click', (e) => this.messageBubbleClicked(e));
|
|
463
|
+
}
|
|
464
|
+
stopMessageMenuShowTimer() {
|
|
465
|
+
if (this.showMessageMenuTimeout) {
|
|
466
|
+
clearTimeout(this.showMessageMenuTimeout);
|
|
467
|
+
this.showMessageMenuTimeout = undefined;
|
|
468
|
+
}
|
|
469
|
+
}
|
|
470
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MessageComponent, deps: [{ token: i1.ChatClientService }, { token: i2.ChannelService }, { token: i3.CustomTemplatesService }, { token: i0.ChangeDetectorRef }, { token: i4.DateParserService }, { token: i5.MessageActionsService }, { token: i0.NgZone }, { token: i6.TranslateService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
471
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: MessageComponent, selector: "stream-message", inputs: { message: "message", enabledMessageActions: "enabledMessageActions", isLastSentMessage: "isLastSentMessage", mode: "mode", isHighlighted: "isHighlighted", scroll$: "scroll$" }, viewQueries: [{ propertyName: "messageMenuTrigger", first: true, predicate: ["messageMenuTrigger"], descendants: true }, { propertyName: "messageMenuFloat", first: true, predicate: ["messageMenuFloat"], descendants: true }, { propertyName: "messageTextElement", first: true, predicate: ["messageTextElement"], descendants: true }, { propertyName: "messageBubble", first: true, predicate: ["messageBubble"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<ng-container\n *ngIf=\"\n message?.type === 'error' &&\n (message?.moderation_details?.action ===\n 'MESSAGE_RESPONSE_ACTION_REMOVE' ||\n message?.moderation?.action === 'remove');\n else notBlockedMessage\n \"\n>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageBlockedTemplate$ | async) ||\n defaultBlockedMessage;\n context: getMessageBlockedContext()\n \"\n />\n <ng-template #defaultBlockedMessage>\n <stream-message-blocked\n [message]=\"message\"\n [isMyMessage]=\"isSentByCurrentUser\"\n />\n </ng-template>\n</ng-container>\n<ng-template #notBlockedMessage>\n <div\n data-testid=\"message-container\"\n class=\"str-chat__message-simple str-chat__message str-chat__message--{{\n message?.type\n }} str-chat__message--{{ message?.status }} {{\n message?.text ? 'str-chat__message--has-text' : 'has-no-text'\n }} str-chat__message-menu-{{ areMessageOptionsOpen ? 'opened' : 'closed' }}\"\n [class.str-chat__message--me]=\"isSentByCurrentUser\"\n [class.str-chat__message--other]=\"!isSentByCurrentUser\"\n [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n [class.str-chat__message--has-attachment]=\"hasAttachment\"\n [class.str-chat__message--with-reactions]=\"hasReactions\"\n [class.str-chat__message--highlighted]=\"isHighlighted\"\n [class.str-chat__message-with-thread-link]=\"shouldDisplayThreadLink\"\n [class.str-chat__message-send-can-be-retried]=\"\n (message?.status === 'failed' && message?.errorStatusCode !== 403) ||\n (message?.type === 'error' && message?.moderation_details)\n \"\n [class.str-chat__message-with-touch-support]=\"hasTouchSupport\"\n [class.str-chat__message-without-touch-support]=\"!hasTouchSupport\"\n >\n <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n <stream-avatar-placeholder\n data-testid=\"avatar\"\n class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\n type=\"user\"\n location=\"message-sender\"\n [imageUrl]=\"message?.user?.image\"\n [name]=\"message?.user?.name || message?.user?.id\"\n [user]=\"message?.user || undefined\"\n />\n <div class=\"str-chat__message-inner\">\n <div\n *ngIf=\"!hasTouchSupport && areOptionsVisible\"\n class=\"str-chat__message-simple__actions str-chat__message-options\"\n data-testid=\"message-options\"\n [class.str-chat__message-actions-open]=\"areMessageOptionsOpen\"\n >\n <div\n #messageActionsToggle\n data-testid=\"message-actions-container\"\n class=\"str-chat__message-actions-container str-chat__message-simple__actions__action str-chat__message-simple__actions__action--options\"\n [floatUiLoose]=\"messageMenuFloat\"\n [looseTrigger]=\"\n messageActionsService.customActionClickHandler\n ? 'none'\n : 'click'\n \"\n [hideOnScroll]=\"false\"\n [hideOnClickOutside]=\"true\"\n [hideOnMouseLeave]=\"false\"\n [disableAnimation]=\"false\"\n [preventOverflow]=\"true\"\n [positionFixed]=\"true\"\n (onSHown)=\"areMessageOptionsOpen = true\"\n (onHidden)=\"areMessageOptionsOpen = false\"\n >\n <div\n *ngIf=\"visibleMessageActionsCount > 0\"\n class=\"str-chat__message-actions-box-button\"\n data-testid=\"message-options-button\"\n (click)=\"messageOptionsButtonClicked()\"\n (keyup.enter)=\"messageOptionsButtonClicked()\"\n >\n <stream-icon-placeholder\n icon=\"action\"\n class=\"str-chat__message-action-icon\"\n />\n </div>\n </div>\n </div>\n <ng-container\n *ngIf=\"\n customTemplatesService.customMessageMetadataInsideBubbleTemplate$\n | async\n \"\n >\n <ng-template\n *ngTemplateOutlet=\"\n (customTemplatesService.customMessageMetadataInsideBubbleTemplate$\n | async)!;\n context: { message: message }\n \"\n ></ng-template>\n </ng-container>\n <div class=\"str-chat__message-reactions-host\">\n <ng-template\n #defaultMessageReactions\n let-messageReactionCounts=\"messageReactionCounts\"\n let-latestReactions=\"latestReactions\"\n let-messageId=\"messageId\"\n let-ownReactions=\"ownReactions\"\n let-messageReactionGroups=\"messageReactionGroups\"\n >\n <stream-message-reactions\n [messageReactionCounts]=\"messageReactionCounts\"\n [latestReactions]=\"latestReactions\"\n [messageId]=\"messageId\"\n [ownReactions]=\"ownReactions\"\n [messageReactionGroups]=\"messageReactionGroups\"\n />\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageReactionsTemplate$ | async) ||\n defaultMessageReactions;\n context: getMessageReactionsContext()\n \"\n />\n </div>\n <float-ui-content #messageMenuFloat>\n <ng-template\n #defaultMessageActionsBox\n let-isMine=\"isMine\"\n let-messageInput=\"message\"\n let-enabledActions=\"enabledActions\"\n let-messageTextHtmlElement=\"messageTextHtmlElement\"\n >\n <stream-message-actions-box\n [isMine]=\"isMine\"\n [message]=\"messageInput\"\n [enabledActions]=\"enabledActions\"\n [messageTextHtmlElement]=\"messageTextHtmlElement\"\n />\n </ng-template>\n <ng-container>\n <div\n (click)=\"messageActionsBoxClicked(messageMenuFloat)\"\n (keyup.enter)=\"messageActionsBoxClicked(messageMenuFloat)\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageActionsBoxTemplate$\n | async) || defaultMessageActionsBox;\n context: getMessageActionsBoxContext()\n \"\n />\n </div>\n </ng-container>\n </float-ui-content>\n <div\n class=\"str-chat__message-bubble str-chat-angular__message-bubble\"\n [class.str-chat-angular__message-bubble--attachment-modal-open]=\"\n nestedModalState === 'opened'\n \"\n data-testid=\"message-bubble\"\n [floatUiLoose]=\"messageMenuFloat\"\n #messageMenuTrigger=\"floatUiLoose\"\n #messageBubble\n looseTrigger=\"none\"\n [hideOnScroll]=\"false\"\n [hideOnClickOutside]=\"true\"\n [hideOnMouseLeave]=\"false\"\n [disableAnimation]=\"true\"\n [preventOverflow]=\"true\"\n (onShown)=\"areMessageOptionsOpen = true\"\n (onHidden)=\"areMessageOptionsOpen = false\"\n [positionFixed]=\"true\"\n >\n <ng-container *ngIf=\"hasAttachment && !message?.quoted_message\">\n <div class=\"str-chat__attachments-container\">\n <ng-container *ngTemplateOutlet=\"attachmentsTemplate\" />\n </div>\n </ng-container>\n <ng-container\n *ngIf=\"\n message?.poll_id &&\n (customTemplatesService.pollTemplate$ | async)\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.pollTemplate$ | async)!;\n context: {\n pollId: message!.poll_id!,\n messageId: message!.id,\n isQuote: false\n }\n \"\n />\n </ng-container>\n <div\n *ngIf=\"\n message?.text || (message?.quoted_message && hasAttachment)\n \"\n class=\"str-chat__message-text\"\n tabindex=\"0\"\n [class.str-chat__message-text--pointer-cursor]=\"\n (message?.status === 'failed' &&\n message?.errorStatusCode !== 403) ||\n (this.message?.type === 'error' &&\n this.message?.moderation_details) ||\n message?.message_text_updated_at\n \"\n (click)=\"messageClicked()\"\n (keyup.enter)=\"messageClicked()\"\n >\n <div\n data-testid=\"inner-message\"\n class=\"str-chat__message-text-inner str-chat__message-simple-text-inner\"\n [class.str-chat__message-light-text-inner--has-attachment]=\"\n hasAttachment\n \"\n >\n <ng-container *ngTemplateOutlet=\"quotedMessage\" />\n <ng-container *ngIf=\"hasAttachment && message?.quoted_message\">\n <ng-container *ngTemplateOutlet=\"attachmentsTemplate\" />\n </ng-container>\n <div\n *ngIf=\"message?.type === 'error'\"\n data-testid=\"client-error-message\"\n class=\"str-chat__simple-message--error-message str-chat__message--error-message\"\n >\n <ng-container *ngIf=\"!message?.moderation_details\">{{\n \"streamChat.Error \u00B7 Unsent\" | translate\n }}</ng-container>\n </div>\n <div\n *ngIf=\"message?.status === 'failed'\"\n data-testid=\"error-message\"\n class=\"str-chat__simple-message--error-message str-chat__message--error-message\"\n >\n {{\n (message?.errorStatusCode === 403\n ? \"streamChat.Message Failed \u00B7 Unauthorized\"\n : \"streamChat.Message Failed \u00B7 Click to try again\"\n ) | translate\n }}\n </div>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageTextTemplate$ | async) ||\n defaultText;\n context: getMessageTextContext()\n \"\n />\n <ng-template\n #defaultText\n let-message=\"message\"\n let-isQuoted=\"isQuoted\"\n let-shouldTranslate=\"shouldTranslate\"\n >\n <stream-message-text\n [message]=\"message\"\n [isQuoted]=\"isQuoted\"\n [shouldTranslate]=\"shouldTranslate\"\n data-testid=\"quoted-message-text\"\n />\n </ng-template>\n </div>\n </div>\n <stream-icon-placeholder\n class=\"str-chat__message-error-icon\"\n icon=\"error\"\n />\n </div>\n </div>\n <ng-container\n *ngTemplateOutlet=\"replyCountButton; context: { message: message }\"\n />\n\n <ng-container *ngTemplateOutlet=\"messageDateAndSender\" />\n </ng-container>\n </ng-container>\n </div>\n\n <ng-template #deletedMessage>\n <div data-testid=\"message-deleted-component\">\n <div class=\"str-chat__message--deleted-inner\" translate>\n streamChat.This message was deleted...\n </div>\n </div>\n </ng-template>\n\n <ng-template #systemMessage>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.systemMessageTemplate$ | async) ||\n defaultSystemMessage;\n context: getMessageContext()\n \"\n />\n <ng-template #defaultSystemMessage let-messageInput=\"message\">\n <div data-testid=\"system-message\" class=\"str-chat__message--system\">\n <div class=\"str-chat__message--system__text\">\n <div class=\"str-chat__message--system__line\"></div>\n <p>{{ messageInput?.text }}</p>\n <div class=\"str-chat__message--system__line\"></div>\n </div>\n <div class=\"str-chat__message--system__date\">\n {{ parsedDate }}\n </div>\n </div>\n </ng-template>\n </ng-template>\n\n <ng-template #quotedMessage>\n <div\n *ngIf=\"message?.quoted_message\"\n class=\"quoted-message str-chat__quoted-message-preview\"\n data-testid=\"quoted-message-container\"\n [class.mine]=\"isSentByCurrentUser\"\n (click)=\"\n jumpToMessage(\n (message?.quoted_message)!.id,\n message?.quoted_message?.parent_id\n )\n \"\n (keyup.enter)=\"\n jumpToMessage(\n (message?.quoted_message)!.id,\n message?.quoted_message?.parent_id\n )\n \"\n >\n <stream-avatar-placeholder\n data-testid=\"qouted-message-avatar\"\n class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\n type=\"user\"\n location=\"quoted-message-sender\"\n [imageUrl]=\"message?.quoted_message?.user?.image\"\n [name]=\"\n message?.quoted_message?.user?.name ||\n message?.quoted_message?.user?.id\n \"\n [user]=\"message?.quoted_message?.user || undefined\"\n />\n <div class=\"quoted-message-inner str-chat__quoted-message-bubble\">\n <ng-container\n *ngIf=\"\n message?.quoted_message?.attachments &&\n message?.quoted_message?.attachments?.length\n \"\n >\n <ng-template\n #defaultAttachments\n let-messageId=\"messageId\"\n let-attachments=\"attachments\"\n let-parentMessageId=\"parentMessageId\"\n let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n >\n <stream-attachment-list\n [messageId]=\"messageId\"\n [attachments]=\"attachments\"\n [parentMessageId]=\"parentMessageId\"\n (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n />\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.attachmentListTemplate$ | async) ||\n defaultAttachments;\n context: getQuotedMessageAttachmentListContext()\n \"\n />\n </ng-container>\n <ng-container\n *ngIf=\"\n message?.quoted_message?.poll_id &&\n (customTemplatesService.pollTemplate$ | async)\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.pollTemplate$ | async)!;\n context: {\n pollId: message?.quoted_message?.poll_id,\n messageId: message?.quoted_message?.id,\n isQuote: true\n }\n \"\n />\n </ng-container>\n <ng-container *ngIf=\"message?.quoted_message\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageTextTemplate$ | async) ||\n defaultText;\n context: getQuotedMessageTextContext()\n \"\n />\n <ng-template\n #defaultText\n let-message=\"message\"\n let-isQuoted=\"isQuoted\"\n let-shouldTranslate=\"shouldTranslate\"\n >\n <stream-message-text\n [message]=\"message\"\n [isQuoted]=\"isQuoted\"\n [shouldTranslate]=\"shouldTranslate\"\n data-testid=\"quoted-message-text\"\n />\n </ng-template>\n </ng-container>\n </div>\n </div>\n </ng-template>\n\n <ng-template #messageDateAndSender>\n <ng-container>\n <div\n *ngIf=\"shouldDisplayTranslationNotice\"\n class=\"str-chat__translation-notice\"\n data-testid=\"translation-notice\"\n >\n <button\n *ngIf=\"displayedMessageTextContent === 'translation'\"\n data-testid=\"see-original\"\n translate\n (click)=\"displayOriginalMessage()\"\n (keyup.enter)=\"displayOriginalMessage()\"\n >\n streamChat.See original (automatically translated)\n </button>\n <button\n *ngIf=\"displayedMessageTextContent === 'original'\"\n data-testid=\"see-translation\"\n translate\n (click)=\"displayTranslatedMessage()\"\n (keyup.enter)=\"displayTranslatedMessage()\"\n >\n streamChat.See translation\n </button>\n </div>\n <ng-container\n *ngIf=\"customTemplatesService.customMessageMetadataTemplate$ | async\"\n >\n <div class=\"str-chat__custom-message-metadata\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.customMessageMetadataTemplate$ | async)!;\n context: getMessageMetadataContext()\n \"\n />\n </div>\n </ng-container>\n <div\n class=\"str-chat__message-data str-chat__message-simple-data str-chat__message-metadata\"\n >\n <ng-container *ngTemplateOutlet=\"messageStatus\" />\n\n <span\n *ngIf=\"!isSentByCurrentUser\"\n data-testid=\"sender\"\n class=\"str-chat__message-simple-name str-chat__message-sender-name\"\n >\n {{ message?.user?.name ? message?.user?.name : message?.user?.id }}\n </span>\n <span\n data-testid=\"date\"\n class=\"str-chat__message-simple-timestamp str-chat__message-simple-time\"\n >\n {{ parsedDate }}\n </span>\n <ng-container *ngIf=\"message?.message_text_updated_at\">\n <span\n data-testid=\"edited-flag\"\n class=\"str-chat__mesage-simple-edited\"\n translate\n >streamChat.Edited</span\n >\n <div\n data-testid=\"edited-timestamp\"\n class=\"str-chat__message-edited-timestamp\"\n [ngClass]=\"{\n 'str-chat__message-edited-timestamp--open': isEditedFlagOpened,\n 'str-chat__message-edited-timestamp--collapsed':\n !isEditedFlagOpened\n }\"\n >\n <span translate>streamChat.Edited</span>\n <time\n dateTime=\"{{ message?.message_text_updated_at }}\"\n title=\"{{ message?.message_text_updated_at }}\"\n >\n {{ pasedEditedDate }}\n </time>\n </div>\n </ng-container>\n </div>\n </ng-container>\n </ng-template>\n\n <ng-template #messageStatus>\n <ng-container\n *ngIf=\"\n isSentByCurrentUser &&\n ((isLastSentMessage && message?.status === 'received') ||\n message?.status === 'sending')\n \"\n >\n <ng-container *ngIf=\"message?.status === 'sending'; else sentStatus\">\n <ng-container *ngTemplateOutlet=\"sendingStatus\" />\n </ng-container>\n <ng-template #sentStatus>\n <ng-container\n *ngIf=\"\n mode === 'main' &&\n isMessageDeliveredAndRead &&\n canDisplayReadStatus;\n else deliveredStatus\n \"\n >\n <ng-container *ngTemplateOutlet=\"readStatus\" />\n </ng-container>\n </ng-template>\n <ng-template #deliveredStatus>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.deliveredStatusTemplate$ | async) ||\n defaultDeliveredStatus;\n context: getDeliveredStatusContext()\n \"\n />\n </ng-template>\n <ng-template #defaultDeliveredStatus>\n <span\n *ngIf=\"mode === 'main'\"\n class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n data-testid=\"delivered-indicator\"\n tabindex=\"0\"\n [floatUiLoose]=\"floatingContent\"\n loosePlacement=\"top\"\n [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n [disableAnimation]=\"true\"\n [hideOnClickOutside]=\"true\"\n [positionFixed]=\"true\"\n [preventOverflow]=\"true\"\n >\n <float-ui-content #floatingContent>\n <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n {{ \"streamChat.Delivered\" | translate }}\n </div>\n </float-ui-content>\n <stream-icon-placeholder\n data-testid=\"delivered-icon\"\n icon=\"delivered\"\n />\n </span>\n </ng-template>\n <ng-template #sendingStatus>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.sendingStatusTemplate$ | async) ||\n defaultSendingStatus;\n context: getSendingStatusContext()\n \"\n />\n </ng-template>\n <ng-template #defaultSendingStatus>\n <span\n class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n data-testid=\"sending-indicator\"\n tabindex=\"0\"\n [floatUiLoose]=\"floatingContent\"\n loosePlacement=\"top\"\n [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n [disableAnimation]=\"true\"\n [hideOnClickOutside]=\"true\"\n [positionFixed]=\"true\"\n [preventOverflow]=\"true\"\n >\n <float-ui-content #floatingContent>\n <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n {{ \"streamChat.Sending...\" | translate }}\n </div>\n </float-ui-content>\n <stream-loading-indicator-placeholder\n data-testid=\"loading-indicator\"\n />\n </span>\n </ng-template>\n <ng-template #readStatus>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.readStatusTemplate$ | async) ||\n defaultReadStatus;\n context: getReadStatusContext()\n \"\n />\n </ng-template>\n <ng-template #defaultReadStatus let-readByText=\"readByText\">\n <span\n class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n data-testid=\"read-indicator\"\n tabindex=\"0\"\n [floatUiLoose]=\"floatingContent\"\n loosePlacement=\"top\"\n [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n [disableAnimation]=\"true\"\n [hideOnClickOutside]=\"true\"\n [positionFixed]=\"true\"\n [preventOverflow]=\"true\"\n >\n <float-ui-content #floatingContent>\n <div\n class=\"str-chat__tooltip str-chat__tooltip-angular\"\n data-testid=\"read-by-tooltip\"\n >\n {{ readByText }}\n </div>\n </float-ui-content>\n <stream-icon-placeholder icon=\"read\" />\n </span>\n </ng-template>\n </ng-container>\n </ng-template>\n\n <ng-template #replyCountButton>\n <div\n class=\"str-chat__message-simple-reply-button str-chat__message-replies-count-button-wrapper\"\n >\n <ng-container *ngIf=\"shouldDisplayThreadLink\">\n <ng-template\n *ngTemplateOutlet=\"\n (customTemplatesService.threadLinkButton$ | async) || defaultButton;\n context: { message: message }\n \"\n ></ng-template>\n </ng-container>\n <ng-template #defaultButton let-message=\"message\">\n <button\n class=\"str-chat__message-replies-count-button\"\n data-testid=\"reply-count-button\"\n (click)=\"setAsActiveParentMessage()\"\n >\n {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n replies' | translate:replyCountParam)}}\n </button>\n </ng-template>\n </div>\n </ng-template>\n\n <ng-template #attachmentsTemplate>\n <ng-template\n #defaultAttachments\n let-messageId=\"messageId\"\n let-attachments=\"attachments\"\n let-parentMessageId=\"parentMessageId\"\n let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n >\n <stream-attachment-list\n [messageId]=\"messageId\"\n [attachments]=\"attachments\"\n [parentMessageId]=\"parentMessageId\"\n (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n />\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.attachmentListTemplate$ | async) ||\n defaultAttachments;\n context: getAttachmentListContext()\n \"\n />\n </ng-template>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i7.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i7.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i7.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i8.NgxFloatUiContentComponent, selector: "float-ui-content", exportAs: ["ngxFloatUiContent"] }, { kind: "directive", type: i8.NgxFloatUiLooseDirective, selector: "[floatUiLoose]", inputs: ["floatUiLoose", "loosePlacement", "looseTrigger"], exportAs: ["floatUiLoose"] }, { kind: "component", type: i9.AvatarPlaceholderComponent, selector: "stream-avatar-placeholder", inputs: ["name", "imageUrl", "location", "channel", "user", "type", "initialsType", "showOnlineIndicator"] }, { kind: "directive", type: i6.TranslateDirective, selector: "[translate],[ngx-translate]", inputs: ["translate", "translateParams"] }, { kind: "component", type: i10.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon"] }, { kind: "component", type: i11.LoadingIndicatorPlaceholderComponent, selector: "stream-loading-indicator-placeholder" }, { kind: "component", type: i12.MessageActionsBoxComponent, selector: "stream-message-actions-box", inputs: ["isMine", "message", "messageTextHtmlElement", "enabledActions"] }, { kind: "component", type: i13.AttachmentListComponent, selector: "stream-attachment-list", inputs: ["messageId", "parentMessageId", "attachments"], outputs: ["imageModalStateChange"] }, { kind: "component", type: i14.MessageReactionsComponent, selector: "stream-message-reactions", inputs: ["messageId", "messageReactionGroups", "messageReactionCounts", "latestReactions", "ownReactions"] }, { kind: "component", type: i15.MessageTextComponent, selector: "stream-message-text", inputs: ["message", "isQuoted", "shouldTranslate"] }, { kind: "component", type: i16.MessageBlockedComponent, selector: "stream-message-blocked", inputs: ["message", "isMyMessage"] }, { kind: "pipe", type: i7.AsyncPipe, name: "async" }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
472
|
+
}
|
|
473
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MessageComponent, decorators: [{
|
|
474
|
+
type: Component,
|
|
475
|
+
args: [{ selector: 'stream-message', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container\n *ngIf=\"\n message?.type === 'error' &&\n (message?.moderation_details?.action ===\n 'MESSAGE_RESPONSE_ACTION_REMOVE' ||\n message?.moderation?.action === 'remove');\n else notBlockedMessage\n \"\n>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageBlockedTemplate$ | async) ||\n defaultBlockedMessage;\n context: getMessageBlockedContext()\n \"\n />\n <ng-template #defaultBlockedMessage>\n <stream-message-blocked\n [message]=\"message\"\n [isMyMessage]=\"isSentByCurrentUser\"\n />\n </ng-template>\n</ng-container>\n<ng-template #notBlockedMessage>\n <div\n data-testid=\"message-container\"\n class=\"str-chat__message-simple str-chat__message str-chat__message--{{\n message?.type\n }} str-chat__message--{{ message?.status }} {{\n message?.text ? 'str-chat__message--has-text' : 'has-no-text'\n }} str-chat__message-menu-{{ areMessageOptionsOpen ? 'opened' : 'closed' }}\"\n [class.str-chat__message--me]=\"isSentByCurrentUser\"\n [class.str-chat__message--other]=\"!isSentByCurrentUser\"\n [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n [class.str-chat__message--has-attachment]=\"hasAttachment\"\n [class.str-chat__message--with-reactions]=\"hasReactions\"\n [class.str-chat__message--highlighted]=\"isHighlighted\"\n [class.str-chat__message-with-thread-link]=\"shouldDisplayThreadLink\"\n [class.str-chat__message-send-can-be-retried]=\"\n (message?.status === 'failed' && message?.errorStatusCode !== 403) ||\n (message?.type === 'error' && message?.moderation_details)\n \"\n [class.str-chat__message-with-touch-support]=\"hasTouchSupport\"\n [class.str-chat__message-without-touch-support]=\"!hasTouchSupport\"\n >\n <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n <stream-avatar-placeholder\n data-testid=\"avatar\"\n class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\n type=\"user\"\n location=\"message-sender\"\n [imageUrl]=\"message?.user?.image\"\n [name]=\"message?.user?.name || message?.user?.id\"\n [user]=\"message?.user || undefined\"\n />\n <div class=\"str-chat__message-inner\">\n <div\n *ngIf=\"!hasTouchSupport && areOptionsVisible\"\n class=\"str-chat__message-simple__actions str-chat__message-options\"\n data-testid=\"message-options\"\n [class.str-chat__message-actions-open]=\"areMessageOptionsOpen\"\n >\n <div\n #messageActionsToggle\n data-testid=\"message-actions-container\"\n class=\"str-chat__message-actions-container str-chat__message-simple__actions__action str-chat__message-simple__actions__action--options\"\n [floatUiLoose]=\"messageMenuFloat\"\n [looseTrigger]=\"\n messageActionsService.customActionClickHandler\n ? 'none'\n : 'click'\n \"\n [hideOnScroll]=\"false\"\n [hideOnClickOutside]=\"true\"\n [hideOnMouseLeave]=\"false\"\n [disableAnimation]=\"false\"\n [preventOverflow]=\"true\"\n [positionFixed]=\"true\"\n (onSHown)=\"areMessageOptionsOpen = true\"\n (onHidden)=\"areMessageOptionsOpen = false\"\n >\n <div\n *ngIf=\"visibleMessageActionsCount > 0\"\n class=\"str-chat__message-actions-box-button\"\n data-testid=\"message-options-button\"\n (click)=\"messageOptionsButtonClicked()\"\n (keyup.enter)=\"messageOptionsButtonClicked()\"\n >\n <stream-icon-placeholder\n icon=\"action\"\n class=\"str-chat__message-action-icon\"\n />\n </div>\n </div>\n </div>\n <ng-container\n *ngIf=\"\n customTemplatesService.customMessageMetadataInsideBubbleTemplate$\n | async\n \"\n >\n <ng-template\n *ngTemplateOutlet=\"\n (customTemplatesService.customMessageMetadataInsideBubbleTemplate$\n | async)!;\n context: { message: message }\n \"\n ></ng-template>\n </ng-container>\n <div class=\"str-chat__message-reactions-host\">\n <ng-template\n #defaultMessageReactions\n let-messageReactionCounts=\"messageReactionCounts\"\n let-latestReactions=\"latestReactions\"\n let-messageId=\"messageId\"\n let-ownReactions=\"ownReactions\"\n let-messageReactionGroups=\"messageReactionGroups\"\n >\n <stream-message-reactions\n [messageReactionCounts]=\"messageReactionCounts\"\n [latestReactions]=\"latestReactions\"\n [messageId]=\"messageId\"\n [ownReactions]=\"ownReactions\"\n [messageReactionGroups]=\"messageReactionGroups\"\n />\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageReactionsTemplate$ | async) ||\n defaultMessageReactions;\n context: getMessageReactionsContext()\n \"\n />\n </div>\n <float-ui-content #messageMenuFloat>\n <ng-template\n #defaultMessageActionsBox\n let-isMine=\"isMine\"\n let-messageInput=\"message\"\n let-enabledActions=\"enabledActions\"\n let-messageTextHtmlElement=\"messageTextHtmlElement\"\n >\n <stream-message-actions-box\n [isMine]=\"isMine\"\n [message]=\"messageInput\"\n [enabledActions]=\"enabledActions\"\n [messageTextHtmlElement]=\"messageTextHtmlElement\"\n />\n </ng-template>\n <ng-container>\n <div\n (click)=\"messageActionsBoxClicked(messageMenuFloat)\"\n (keyup.enter)=\"messageActionsBoxClicked(messageMenuFloat)\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageActionsBoxTemplate$\n | async) || defaultMessageActionsBox;\n context: getMessageActionsBoxContext()\n \"\n />\n </div>\n </ng-container>\n </float-ui-content>\n <div\n class=\"str-chat__message-bubble str-chat-angular__message-bubble\"\n [class.str-chat-angular__message-bubble--attachment-modal-open]=\"\n nestedModalState === 'opened'\n \"\n data-testid=\"message-bubble\"\n [floatUiLoose]=\"messageMenuFloat\"\n #messageMenuTrigger=\"floatUiLoose\"\n #messageBubble\n looseTrigger=\"none\"\n [hideOnScroll]=\"false\"\n [hideOnClickOutside]=\"true\"\n [hideOnMouseLeave]=\"false\"\n [disableAnimation]=\"true\"\n [preventOverflow]=\"true\"\n (onShown)=\"areMessageOptionsOpen = true\"\n (onHidden)=\"areMessageOptionsOpen = false\"\n [positionFixed]=\"true\"\n >\n <ng-container *ngIf=\"hasAttachment && !message?.quoted_message\">\n <div class=\"str-chat__attachments-container\">\n <ng-container *ngTemplateOutlet=\"attachmentsTemplate\" />\n </div>\n </ng-container>\n <ng-container\n *ngIf=\"\n message?.poll_id &&\n (customTemplatesService.pollTemplate$ | async)\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.pollTemplate$ | async)!;\n context: {\n pollId: message!.poll_id!,\n messageId: message!.id,\n isQuote: false\n }\n \"\n />\n </ng-container>\n <div\n *ngIf=\"\n message?.text || (message?.quoted_message && hasAttachment)\n \"\n class=\"str-chat__message-text\"\n tabindex=\"0\"\n [class.str-chat__message-text--pointer-cursor]=\"\n (message?.status === 'failed' &&\n message?.errorStatusCode !== 403) ||\n (this.message?.type === 'error' &&\n this.message?.moderation_details) ||\n message?.message_text_updated_at\n \"\n (click)=\"messageClicked()\"\n (keyup.enter)=\"messageClicked()\"\n >\n <div\n data-testid=\"inner-message\"\n class=\"str-chat__message-text-inner str-chat__message-simple-text-inner\"\n [class.str-chat__message-light-text-inner--has-attachment]=\"\n hasAttachment\n \"\n >\n <ng-container *ngTemplateOutlet=\"quotedMessage\" />\n <ng-container *ngIf=\"hasAttachment && message?.quoted_message\">\n <ng-container *ngTemplateOutlet=\"attachmentsTemplate\" />\n </ng-container>\n <div\n *ngIf=\"message?.type === 'error'\"\n data-testid=\"client-error-message\"\n class=\"str-chat__simple-message--error-message str-chat__message--error-message\"\n >\n <ng-container *ngIf=\"!message?.moderation_details\">{{\n \"streamChat.Error \u00B7 Unsent\" | translate\n }}</ng-container>\n </div>\n <div\n *ngIf=\"message?.status === 'failed'\"\n data-testid=\"error-message\"\n class=\"str-chat__simple-message--error-message str-chat__message--error-message\"\n >\n {{\n (message?.errorStatusCode === 403\n ? \"streamChat.Message Failed \u00B7 Unauthorized\"\n : \"streamChat.Message Failed \u00B7 Click to try again\"\n ) | translate\n }}\n </div>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageTextTemplate$ | async) ||\n defaultText;\n context: getMessageTextContext()\n \"\n />\n <ng-template\n #defaultText\n let-message=\"message\"\n let-isQuoted=\"isQuoted\"\n let-shouldTranslate=\"shouldTranslate\"\n >\n <stream-message-text\n [message]=\"message\"\n [isQuoted]=\"isQuoted\"\n [shouldTranslate]=\"shouldTranslate\"\n data-testid=\"quoted-message-text\"\n />\n </ng-template>\n </div>\n </div>\n <stream-icon-placeholder\n class=\"str-chat__message-error-icon\"\n icon=\"error\"\n />\n </div>\n </div>\n <ng-container\n *ngTemplateOutlet=\"replyCountButton; context: { message: message }\"\n />\n\n <ng-container *ngTemplateOutlet=\"messageDateAndSender\" />\n </ng-container>\n </ng-container>\n </div>\n\n <ng-template #deletedMessage>\n <div data-testid=\"message-deleted-component\">\n <div class=\"str-chat__message--deleted-inner\" translate>\n streamChat.This message was deleted...\n </div>\n </div>\n </ng-template>\n\n <ng-template #systemMessage>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.systemMessageTemplate$ | async) ||\n defaultSystemMessage;\n context: getMessageContext()\n \"\n />\n <ng-template #defaultSystemMessage let-messageInput=\"message\">\n <div data-testid=\"system-message\" class=\"str-chat__message--system\">\n <div class=\"str-chat__message--system__text\">\n <div class=\"str-chat__message--system__line\"></div>\n <p>{{ messageInput?.text }}</p>\n <div class=\"str-chat__message--system__line\"></div>\n </div>\n <div class=\"str-chat__message--system__date\">\n {{ parsedDate }}\n </div>\n </div>\n </ng-template>\n </ng-template>\n\n <ng-template #quotedMessage>\n <div\n *ngIf=\"message?.quoted_message\"\n class=\"quoted-message str-chat__quoted-message-preview\"\n data-testid=\"quoted-message-container\"\n [class.mine]=\"isSentByCurrentUser\"\n (click)=\"\n jumpToMessage(\n (message?.quoted_message)!.id,\n message?.quoted_message?.parent_id\n )\n \"\n (keyup.enter)=\"\n jumpToMessage(\n (message?.quoted_message)!.id,\n message?.quoted_message?.parent_id\n )\n \"\n >\n <stream-avatar-placeholder\n data-testid=\"qouted-message-avatar\"\n class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\n type=\"user\"\n location=\"quoted-message-sender\"\n [imageUrl]=\"message?.quoted_message?.user?.image\"\n [name]=\"\n message?.quoted_message?.user?.name ||\n message?.quoted_message?.user?.id\n \"\n [user]=\"message?.quoted_message?.user || undefined\"\n />\n <div class=\"quoted-message-inner str-chat__quoted-message-bubble\">\n <ng-container\n *ngIf=\"\n message?.quoted_message?.attachments &&\n message?.quoted_message?.attachments?.length\n \"\n >\n <ng-template\n #defaultAttachments\n let-messageId=\"messageId\"\n let-attachments=\"attachments\"\n let-parentMessageId=\"parentMessageId\"\n let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n >\n <stream-attachment-list\n [messageId]=\"messageId\"\n [attachments]=\"attachments\"\n [parentMessageId]=\"parentMessageId\"\n (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n />\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.attachmentListTemplate$ | async) ||\n defaultAttachments;\n context: getQuotedMessageAttachmentListContext()\n \"\n />\n </ng-container>\n <ng-container\n *ngIf=\"\n message?.quoted_message?.poll_id &&\n (customTemplatesService.pollTemplate$ | async)\n \"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.pollTemplate$ | async)!;\n context: {\n pollId: message?.quoted_message?.poll_id,\n messageId: message?.quoted_message?.id,\n isQuote: true\n }\n \"\n />\n </ng-container>\n <ng-container *ngIf=\"message?.quoted_message\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageTextTemplate$ | async) ||\n defaultText;\n context: getQuotedMessageTextContext()\n \"\n />\n <ng-template\n #defaultText\n let-message=\"message\"\n let-isQuoted=\"isQuoted\"\n let-shouldTranslate=\"shouldTranslate\"\n >\n <stream-message-text\n [message]=\"message\"\n [isQuoted]=\"isQuoted\"\n [shouldTranslate]=\"shouldTranslate\"\n data-testid=\"quoted-message-text\"\n />\n </ng-template>\n </ng-container>\n </div>\n </div>\n </ng-template>\n\n <ng-template #messageDateAndSender>\n <ng-container>\n <div\n *ngIf=\"shouldDisplayTranslationNotice\"\n class=\"str-chat__translation-notice\"\n data-testid=\"translation-notice\"\n >\n <button\n *ngIf=\"displayedMessageTextContent === 'translation'\"\n data-testid=\"see-original\"\n translate\n (click)=\"displayOriginalMessage()\"\n (keyup.enter)=\"displayOriginalMessage()\"\n >\n streamChat.See original (automatically translated)\n </button>\n <button\n *ngIf=\"displayedMessageTextContent === 'original'\"\n data-testid=\"see-translation\"\n translate\n (click)=\"displayTranslatedMessage()\"\n (keyup.enter)=\"displayTranslatedMessage()\"\n >\n streamChat.See translation\n </button>\n </div>\n <ng-container\n *ngIf=\"customTemplatesService.customMessageMetadataTemplate$ | async\"\n >\n <div class=\"str-chat__custom-message-metadata\">\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.customMessageMetadataTemplate$ | async)!;\n context: getMessageMetadataContext()\n \"\n />\n </div>\n </ng-container>\n <div\n class=\"str-chat__message-data str-chat__message-simple-data str-chat__message-metadata\"\n >\n <ng-container *ngTemplateOutlet=\"messageStatus\" />\n\n <span\n *ngIf=\"!isSentByCurrentUser\"\n data-testid=\"sender\"\n class=\"str-chat__message-simple-name str-chat__message-sender-name\"\n >\n {{ message?.user?.name ? message?.user?.name : message?.user?.id }}\n </span>\n <span\n data-testid=\"date\"\n class=\"str-chat__message-simple-timestamp str-chat__message-simple-time\"\n >\n {{ parsedDate }}\n </span>\n <ng-container *ngIf=\"message?.message_text_updated_at\">\n <span\n data-testid=\"edited-flag\"\n class=\"str-chat__mesage-simple-edited\"\n translate\n >streamChat.Edited</span\n >\n <div\n data-testid=\"edited-timestamp\"\n class=\"str-chat__message-edited-timestamp\"\n [ngClass]=\"{\n 'str-chat__message-edited-timestamp--open': isEditedFlagOpened,\n 'str-chat__message-edited-timestamp--collapsed':\n !isEditedFlagOpened\n }\"\n >\n <span translate>streamChat.Edited</span>\n <time\n dateTime=\"{{ message?.message_text_updated_at }}\"\n title=\"{{ message?.message_text_updated_at }}\"\n >\n {{ pasedEditedDate }}\n </time>\n </div>\n </ng-container>\n </div>\n </ng-container>\n </ng-template>\n\n <ng-template #messageStatus>\n <ng-container\n *ngIf=\"\n isSentByCurrentUser &&\n ((isLastSentMessage && message?.status === 'received') ||\n message?.status === 'sending')\n \"\n >\n <ng-container *ngIf=\"message?.status === 'sending'; else sentStatus\">\n <ng-container *ngTemplateOutlet=\"sendingStatus\" />\n </ng-container>\n <ng-template #sentStatus>\n <ng-container\n *ngIf=\"\n mode === 'main' &&\n isMessageDeliveredAndRead &&\n canDisplayReadStatus;\n else deliveredStatus\n \"\n >\n <ng-container *ngTemplateOutlet=\"readStatus\" />\n </ng-container>\n </ng-template>\n <ng-template #deliveredStatus>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.deliveredStatusTemplate$ | async) ||\n defaultDeliveredStatus;\n context: getDeliveredStatusContext()\n \"\n />\n </ng-template>\n <ng-template #defaultDeliveredStatus>\n <span\n *ngIf=\"mode === 'main'\"\n class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n data-testid=\"delivered-indicator\"\n tabindex=\"0\"\n [floatUiLoose]=\"floatingContent\"\n loosePlacement=\"top\"\n [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n [disableAnimation]=\"true\"\n [hideOnClickOutside]=\"true\"\n [positionFixed]=\"true\"\n [preventOverflow]=\"true\"\n >\n <float-ui-content #floatingContent>\n <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n {{ \"streamChat.Delivered\" | translate }}\n </div>\n </float-ui-content>\n <stream-icon-placeholder\n data-testid=\"delivered-icon\"\n icon=\"delivered\"\n />\n </span>\n </ng-template>\n <ng-template #sendingStatus>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.sendingStatusTemplate$ | async) ||\n defaultSendingStatus;\n context: getSendingStatusContext()\n \"\n />\n </ng-template>\n <ng-template #defaultSendingStatus>\n <span\n class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n data-testid=\"sending-indicator\"\n tabindex=\"0\"\n [floatUiLoose]=\"floatingContent\"\n loosePlacement=\"top\"\n [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n [disableAnimation]=\"true\"\n [hideOnClickOutside]=\"true\"\n [positionFixed]=\"true\"\n [preventOverflow]=\"true\"\n >\n <float-ui-content #floatingContent>\n <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n {{ \"streamChat.Sending...\" | translate }}\n </div>\n </float-ui-content>\n <stream-loading-indicator-placeholder\n data-testid=\"loading-indicator\"\n />\n </span>\n </ng-template>\n <ng-template #readStatus>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.readStatusTemplate$ | async) ||\n defaultReadStatus;\n context: getReadStatusContext()\n \"\n />\n </ng-template>\n <ng-template #defaultReadStatus let-readByText=\"readByText\">\n <span\n class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n data-testid=\"read-indicator\"\n tabindex=\"0\"\n [floatUiLoose]=\"floatingContent\"\n loosePlacement=\"top\"\n [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n [disableAnimation]=\"true\"\n [hideOnClickOutside]=\"true\"\n [positionFixed]=\"true\"\n [preventOverflow]=\"true\"\n >\n <float-ui-content #floatingContent>\n <div\n class=\"str-chat__tooltip str-chat__tooltip-angular\"\n data-testid=\"read-by-tooltip\"\n >\n {{ readByText }}\n </div>\n </float-ui-content>\n <stream-icon-placeholder icon=\"read\" />\n </span>\n </ng-template>\n </ng-container>\n </ng-template>\n\n <ng-template #replyCountButton>\n <div\n class=\"str-chat__message-simple-reply-button str-chat__message-replies-count-button-wrapper\"\n >\n <ng-container *ngIf=\"shouldDisplayThreadLink\">\n <ng-template\n *ngTemplateOutlet=\"\n (customTemplatesService.threadLinkButton$ | async) || defaultButton;\n context: { message: message }\n \"\n ></ng-template>\n </ng-container>\n <ng-template #defaultButton let-message=\"message\">\n <button\n class=\"str-chat__message-replies-count-button\"\n data-testid=\"reply-count-button\"\n (click)=\"setAsActiveParentMessage()\"\n >\n {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n replies' | translate:replyCountParam)}}\n </button>\n </ng-template>\n </div>\n </ng-template>\n\n <ng-template #attachmentsTemplate>\n <ng-template\n #defaultAttachments\n let-messageId=\"messageId\"\n let-attachments=\"attachments\"\n let-parentMessageId=\"parentMessageId\"\n let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n >\n <stream-attachment-list\n [messageId]=\"messageId\"\n [attachments]=\"attachments\"\n [parentMessageId]=\"parentMessageId\"\n (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n />\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.attachmentListTemplate$ | async) ||\n defaultAttachments;\n context: getAttachmentListContext()\n \"\n />\n </ng-template>\n</ng-template>\n" }]
|
|
476
|
+
}], ctorParameters: () => [{ type: i1.ChatClientService }, { type: i2.ChannelService }, { type: i3.CustomTemplatesService }, { type: i0.ChangeDetectorRef }, { type: i4.DateParserService }, { type: i5.MessageActionsService }, { type: i0.NgZone }, { type: i6.TranslateService }], propDecorators: { message: [{
|
|
477
|
+
type: Input
|
|
478
|
+
}], enabledMessageActions: [{
|
|
479
|
+
type: Input
|
|
480
|
+
}], isLastSentMessage: [{
|
|
481
|
+
type: Input
|
|
482
|
+
}], mode: [{
|
|
483
|
+
type: Input
|
|
484
|
+
}], isHighlighted: [{
|
|
485
|
+
type: Input
|
|
486
|
+
}], scroll$: [{
|
|
487
|
+
type: Input
|
|
488
|
+
}], messageMenuTrigger: [{
|
|
489
|
+
type: ViewChild,
|
|
490
|
+
args: ['messageMenuTrigger']
|
|
491
|
+
}], messageMenuFloat: [{
|
|
492
|
+
type: ViewChild,
|
|
493
|
+
args: ['messageMenuFloat']
|
|
494
|
+
}], messageTextElement: [{
|
|
495
|
+
type: ViewChild,
|
|
496
|
+
args: ['messageTextElement']
|
|
497
|
+
}], messageBubble: [{
|
|
498
|
+
type: ViewChild,
|
|
499
|
+
args: ['messageBubble']
|
|
500
|
+
}] } });
|
|
501
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message/message.component.ts","../../../../../projects/stream-chat-angular/src/lib/message/message.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  OnChanges,\n  SimpleChanges,\n  OnDestroy,\n  OnInit,\n  ChangeDetectorRef,\n  ChangeDetectionStrategy,\n  AfterViewInit,\n  ViewChild,\n  ElementRef,\n  NgZone,\n} from '@angular/core';\nimport { Attachment, UserResponse } from 'stream-chat';\nimport { ChannelService } from '../channel.service';\nimport { ChatClientService } from '../chat-client.service';\nimport {\n  AttachmentListContext,\n  MessageActionsBoxContext,\n  MessageReactionsContext,\n  StreamMessage,\n  DeliveredStatusContext,\n  SendingStatusContext,\n  ReadStatusContext,\n  SystemMessageContext,\n  CustomMetadataContext,\n  MessageTextContext,\n  MessageBlockedContext,\n} from '../types';\nimport { Observable, pairwise, Subscription, take } from 'rxjs';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { listUsers } from '../list-users';\nimport { DateParserService } from '../date-parser.service';\nimport { MessageActionsService } from '../message-actions.service';\nimport {\n  NgxFloatUiContentComponent,\n  NgxFloatUiLooseDirective,\n} from 'ngx-float-ui';\nimport { TranslateService } from '@ngx-translate/core';\nimport { isSafari } from '../is-safari';\n\n/**\n * The `Message` component displays a message with additional information such as sender and date, and enables [interaction with the message (i.e. edit or react)](/chat/docs/sdk/angular/concepts/message-interactions/).\n */\n@Component({\n  selector: 'stream-message',\n  templateUrl: './message.component.html',\n  styles: [],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MessageComponent\n  implements OnInit, OnChanges, OnDestroy, AfterViewInit\n{\n  /**\n   * The message to be displayed\n   */\n  @Input() message: StreamMessage | undefined;\n  /**\n   * The list of [channel capabilities](/chat/docs/javascript/channel_capabilities/) that are enabled for the current user, the list of [supported interactions](/chat/docs/sdk/angular/concepts/message-interactions/) can be found in our message interaction guide. Unathorized actions won't be displayed on the UI. The [`MessageList`](/chat/docs/sdk/angular/components/MessageListComponent/) component automatically sets this based on [channel capabilities](/chat/docs/javascript/channel_capabilities/).\n   */\n  @Input() enabledMessageActions: string[] = [];\n  /**\n   * If `true`, the message status (sending, sent, who read the message) is displayed.\n   */\n  @Input() isLastSentMessage: boolean | undefined;\n  /**\n   * Determines if the message is being dispalyed in a channel or in a [thread](/chat/docs/javascript/threads/).\n   */\n  @Input() mode: 'thread' | 'main' = 'main';\n  /**\n   * Highlighting is used to add visual emphasize to a message when jumping to the message\n   */\n  @Input() isHighlighted = false;\n  /**\n   * An Observable that emits when the message list is scrolled, it's used to prevent opening the message menu while scroll is in progress\n   */\n  @Input() scroll$?: Observable<void>;\n  canReceiveReadEvents: boolean | undefined;\n  canReactToMessage: boolean | undefined;\n  isEditedFlagOpened = false;\n  shouldDisplayTranslationNotice = false;\n  displayedMessageTextContent: 'original' | 'translation' = 'original';\n  nestedModalState: 'opened' | 'closed' = 'closed';\n  shouldDisplayThreadLink = false;\n  isSentByCurrentUser = false;\n  readByText = '';\n  lastReadUser: UserResponse | undefined = undefined;\n  isOnlyReadByMe = false;\n  isReadByMultipleUsers = false;\n  isMessageDeliveredAndRead = false;\n  parsedDate = '';\n  pasedEditedDate = '';\n  areOptionsVisible = false;\n  hasAttachment = false;\n  hasReactions = false;\n  replyCountParam: { replyCount: number | undefined } = {\n    replyCount: undefined,\n  };\n  areMessageOptionsOpen = false;\n  canDisplayReadStatus = false;\n  hasTouchSupport = isSafari()\n    ? 'ontouchstart' in window || navigator.maxTouchPoints > 0\n    : window.matchMedia('(any-hover: none)').matches;\n  private quotedMessageAttachments: Attachment[] | undefined;\n  private subscriptions: Subscription[] = [];\n  private isViewInited = false;\n  private userId?: string;\n  @ViewChild('messageMenuTrigger')\n  messageMenuTrigger!: NgxFloatUiLooseDirective;\n  @ViewChild('messageMenuFloat')\n  messageMenuFloat!: NgxFloatUiContentComponent;\n  @ViewChild('messageTextElement') messageTextElement?: ElementRef<HTMLElement>;\n  @ViewChild('messageBubble') messageBubble?: ElementRef<HTMLElement>;\n  private showMessageMenuTimeout?: ReturnType<typeof setTimeout>;\n  private shouldPreventMessageMenuClose = false;\n  private _visibleMessageActionsCount = 0;\n  private channelMemberCount?: number;\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private channelService: ChannelService,\n    public customTemplatesService: CustomTemplatesService,\n    private cdRef: ChangeDetectorRef,\n    private dateParser: DateParserService,\n    public messageActionsService: MessageActionsService,\n    private ngZone: NgZone,\n    private translateService: TranslateService\n  ) {}\n\n  get visibleMessageActionsCount() {\n    return this._visibleMessageActionsCount;\n  }\n\n  set visibleMessageActionsCount(count: number) {\n    this._visibleMessageActionsCount = count;\n    if (this.areOptionsVisible && this._visibleMessageActionsCount === 0) {\n      this.areOptionsVisible = false;\n    }\n  }\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.chatClientService.user$.subscribe((u) => {\n        if (u?.id !== this.userId) {\n          this.userId = u?.id;\n          this.setIsSentByCurrentUser();\n          this.setLastReadUser();\n          if (this.isViewInited) {\n            this.cdRef.detectChanges();\n          }\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.messageActionsService.customActions$.subscribe(() => {\n        if (this.message) {\n          const numberOfEnabledActions =\n            this.messageActionsService.getAuthorizedMessageActionsCount(\n              this.message,\n              this.enabledMessageActions\n            );\n          if (numberOfEnabledActions !== this.visibleMessageActionsCount) {\n            this.visibleMessageActionsCount = numberOfEnabledActions;\n            if (this.isViewInited) {\n              this.cdRef.detectChanges();\n            }\n          }\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.channelService.activeChannel$.subscribe((activeChannel) => {\n        const newChannelMemberCount = activeChannel?.data?.member_count;\n        if (newChannelMemberCount !== this.channelMemberCount) {\n          const shouldUpdateText =\n            this.channelMemberCount !== undefined &&\n            newChannelMemberCount != undefined &&\n            ((this.channelMemberCount <= 1000 && newChannelMemberCount > 100) ||\n              (this.channelMemberCount > 100 && newChannelMemberCount <= 100));\n          this.channelMemberCount = activeChannel?.data?.member_count;\n          if (\n            this.message &&\n            this.message.cid === activeChannel?.cid &&\n            shouldUpdateText\n          ) {\n            this.updateReadByText();\n          }\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.messageActionsService.modalOpenedForMessage.subscribe(\n        (messageId) => {\n          if (messageId === this.message?.id) {\n            this.nestedModalState = 'opened';\n          }\n        }\n      )\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message) {\n      this.displayedMessageTextContent = this.message?.translation\n        ? 'translation'\n        : 'original';\n      this.shouldDisplayTranslationNotice =\n        this.displayedMessageTextContent === 'translation';\n      const originalAttachments = this.message?.quoted_message?.attachments;\n      this.quotedMessageAttachments =\n        originalAttachments && originalAttachments.length\n          ? [originalAttachments[0]]\n          : [];\n      this.setIsSentByCurrentUser();\n      this.setLastReadUser();\n      this.updateReadByText();\n      this.isOnlyReadByMe = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.readBy.length === 0\n      );\n      this.isReadByMultipleUsers = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.readBy.length > 1\n      );\n      this.isMessageDeliveredAndRead = !!(\n        this.message &&\n        this.message.readBy &&\n        this.message.status === 'received' &&\n        this.message.readBy.length > 0\n      );\n      this.parsedDate =\n        (this.message &&\n          this.message.created_at &&\n          this.dateParser.parseDateTime(this.message.created_at)) ||\n        '';\n      this.pasedEditedDate =\n        (this.message &&\n          this.message.message_text_updated_at &&\n          this.dateParser.parseDateTime(\n            new Date(this.message.message_text_updated_at)\n          )) ||\n        '';\n      this.hasAttachment =\n        !!this.message?.attachments && !!this.message.attachments.length;\n      this.hasReactions =\n        !!this.message?.reaction_groups &&\n        Object.keys(this.message.reaction_groups).length > 0;\n      this.replyCountParam = { replyCount: this.message?.reply_count };\n    }\n    if (changes.enabledMessageActions) {\n      this.canReactToMessage =\n        this.enabledMessageActions.indexOf('send-reaction') !== -1;\n      this.canReceiveReadEvents =\n        this.enabledMessageActions.indexOf('read-events') !== -1;\n      this.canDisplayReadStatus =\n        this.canReceiveReadEvents !== false &&\n        this.enabledMessageActions.indexOf('read-events') !== -1;\n    }\n    if (changes.message || changes.enabledMessageActions || changes.mode) {\n      this.shouldDisplayThreadLink =\n        !!this.message?.reply_count && this.mode !== 'thread';\n    }\n    if (changes.message || changes.mode || changes.enabledMessageActions) {\n      this.areOptionsVisible = this.message\n        ? !(\n            !this.message.type ||\n            this.message.type === 'error' ||\n            this.message.type === 'system' ||\n            this.message.type === 'deleted' ||\n            this.message.type === 'ephemeral' ||\n            this.message.status === 'failed' ||\n            this.message.status === 'sending' ||\n            (this.mode === 'thread' && !this.message.parent_id) ||\n            this.message.deleted_at ||\n            this.enabledMessageActions.length === 0\n          )\n        : false;\n    }\n    if (changes.message || changes.enabledMessageActions) {\n      if (this.message) {\n        this.visibleMessageActionsCount =\n          this.messageActionsService.getAuthorizedMessageActionsCount(\n            this.message,\n            this.enabledMessageActions\n          );\n      } else {\n        this.visibleMessageActionsCount = 0;\n      }\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.isViewInited = true;\n    if (this.hasTouchSupport && this.messageBubble?.nativeElement) {\n      this.ngZone.runOutsideAngular(() => {\n        this.registerMenuTriggerEventHandlers();\n      });\n    }\n    this.subscriptions.push(\n      this.messageActionsService.messageMenuOpenedFor$.subscribe((id) => {\n        if (\n          (id === undefined || this.message?.id !== id) &&\n          this.areMessageOptionsOpen\n        ) {\n          this.messageMenuTrigger?.hide();\n        }\n      })\n    );\n    this.subscriptions.push(\n      this.messageActionsService.modalOpenedForMessage\n        .pipe(pairwise())\n        .subscribe(([oldId, newId]) => {\n          if (newId === this.message?.id) {\n            this.nestedModalState = 'opened';\n          } else if (oldId === this.message?.id && newId === undefined) {\n            this.nestedModalState = 'closed';\n          }\n        })\n    );\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n    if (this.showMessageMenuTimeout) {\n      clearTimeout(this.showMessageMenuTimeout);\n    }\n  }\n\n  mousePushedDown(event: MouseEvent) {\n    if (\n      !this.hasTouchSupport ||\n      event.button !== 0 ||\n      !this.areOptionsVisible\n    ) {\n      return;\n    }\n    this.startMessageMenuShowTimer({ fromTouch: false });\n  }\n\n  mouseReleased() {\n    this.stopMessageMenuShowTimer();\n  }\n\n  touchStarted() {\n    if (!this.areOptionsVisible) {\n      return;\n    }\n    this.startMessageMenuShowTimer({ fromTouch: true });\n  }\n\n  touchEnded() {\n    this.stopMessageMenuShowTimer();\n  }\n\n  messageBubbleClicked(event: Event) {\n    if (!this.hasTouchSupport) {\n      return;\n    }\n    if (this.shouldPreventMessageMenuClose) {\n      event.stopPropagation();\n      this.shouldPreventMessageMenuClose = false;\n    } else if (this.areMessageOptionsOpen) {\n      this.messageMenuTrigger?.hide();\n    }\n  }\n\n  messageOptionsButtonClicked() {\n    if (!this.message) {\n      return;\n    }\n    if (this.messageActionsService.customActionClickHandler) {\n      this.messageActionsService.customActionClickHandler({\n        message: this.message,\n        enabledActions: this.enabledMessageActions,\n        customActions: this.messageActionsService.customActions$.getValue(),\n        isMine: this.isSentByCurrentUser,\n        messageTextHtmlElement: this.messageTextElement?.nativeElement,\n      });\n    } else {\n      this.areMessageOptionsOpen = !this.areMessageOptionsOpen;\n    }\n  }\n\n  messageActionsBoxClicked(floatingContent: NgxFloatUiContentComponent) {\n    floatingContent.hide();\n  }\n\n  getAttachmentListContext(): AttachmentListContext {\n    return {\n      messageId: this.message?.id || '',\n      attachments: this.message?.attachments || [],\n      parentMessageId: this.message?.parent_id,\n      imageModalStateChangeHandler: (state) => (this.nestedModalState = state),\n    };\n  }\n\n  getMessageContext(): SystemMessageContext {\n    return {\n      message: this.message,\n      enabledMessageActions: this.enabledMessageActions,\n      isHighlighted: this.isHighlighted,\n      isLastSentMessage: this.isLastSentMessage,\n      mode: this.mode,\n      customActions: this.messageActionsService.customActions$.getValue(),\n      parsedDate: this.parsedDate,\n      scroll$: this.scroll$,\n    };\n  }\n\n  getMessageTextContext(): MessageTextContext {\n    return {\n      message: this.message,\n      isQuoted: false,\n      shouldTranslate: this.displayedMessageTextContent === 'translation',\n    };\n  }\n\n  getQuotedMessageTextContext(): MessageTextContext {\n    return {\n      message: this.message?.quoted_message,\n      isQuoted: true,\n      shouldTranslate: this.displayedMessageTextContent === 'translation',\n    };\n  }\n\n  getQuotedMessageAttachmentListContext(): AttachmentListContext {\n    return {\n      messageId: this.message?.quoted_message?.id || '',\n      attachments: this.quotedMessageAttachments!,\n      parentMessageId: this?.message?.quoted_message?.parent_id,\n    };\n  }\n\n  getMessageReactionsContext(): MessageReactionsContext {\n    return {\n      messageReactionCounts: this.message?.reaction_counts || {},\n      latestReactions: this.message?.latest_reactions || [],\n      messageId: this.message?.id,\n      ownReactions: this.message?.own_reactions || [],\n      messageReactionGroups: this.message?.reaction_groups || {},\n    };\n  }\n\n  messageClicked() {\n    if (\n      this.message?.status === 'failed' &&\n      this.message?.errorStatusCode !== 403\n    ) {\n      this.resendMessage();\n    } else if (\n      this.message?.type === 'error' &&\n      this.message?.moderation_details\n    ) {\n      this.openMessageBouncePrompt();\n    } else {\n      this.isEditedFlagOpened = !this.isEditedFlagOpened;\n    }\n  }\n\n  resendMessage() {\n    void this.channelService.resendMessage(this.message!);\n  }\n\n  setAsActiveParentMessage() {\n    void this.channelService.setAsActiveParentMessage(this.message);\n  }\n\n  getMessageActionsBoxContext(): MessageActionsBoxContext {\n    return {\n      isMine: this.isSentByCurrentUser,\n      enabledActions: this.enabledMessageActions,\n      message: this.message,\n      messageTextHtmlElement: this.messageTextElement?.nativeElement,\n    };\n  }\n\n  getDeliveredStatusContext(): DeliveredStatusContext {\n    return {\n      message: this.message!,\n    };\n  }\n\n  getSendingStatusContext(): SendingStatusContext {\n    return {\n      message: this.message!,\n    };\n  }\n\n  getReadStatusContext(): ReadStatusContext {\n    return {\n      message: this.message!,\n      readByText: this.readByText,\n    };\n  }\n\n  getMessageMetadataContext(): CustomMetadataContext {\n    return { message: this.message! };\n  }\n\n  getMessageBlockedContext(): MessageBlockedContext {\n    return {\n      message: this.message!,\n      isMyMessage: this.isSentByCurrentUser,\n    };\n  }\n\n  jumpToMessage(messageId: string, parentMessageId?: string) {\n    void this.channelService.jumpToMessage(messageId, parentMessageId);\n  }\n\n  openMessageBouncePrompt() {\n    this.channelService.bouncedMessage$.next(this.message);\n  }\n\n  displayTranslatedMessage() {\n    this.shouldDisplayTranslationNotice = true;\n    this.displayedMessageTextContent = 'translation';\n  }\n\n  displayOriginalMessage() {\n    this.displayedMessageTextContent = 'original';\n  }\n\n  private updateReadByText() {\n    const others = this.translateService.instant(\n      'streamChat.and others'\n    ) as string;\n    const hasMoreThan100Members = (this.channelMemberCount ?? 0) > 100;\n    this.readByText = this.message?.readBy\n      ? listUsers(this.message.readBy, !hasMoreThan100Members, others)\n      : '';\n  }\n\n  private setIsSentByCurrentUser() {\n    this.isSentByCurrentUser = this.message?.user?.id === this.userId;\n  }\n\n  private setLastReadUser() {\n    this.lastReadUser = this.message?.readBy?.filter(\n      (u) => u.id !== this.userId\n    )[0];\n  }\n\n  private startMessageMenuShowTimer(options: { fromTouch: boolean }) {\n    this.stopMessageMenuShowTimer();\n    if (this.scroll$) {\n      this.subscriptions.push(\n        this.scroll$.pipe(take(1)).subscribe(() => {\n          this.stopMessageMenuShowTimer();\n        })\n      );\n    }\n    this.showMessageMenuTimeout = setTimeout(() => {\n      if (!this.message) {\n        return;\n      }\n      this.ngZone.run(() => {\n        if (this.messageActionsService.customActionClickHandler) {\n          this.messageActionsService.customActionClickHandler({\n            message: this.message,\n            enabledActions: this.enabledMessageActions,\n            customActions: this.messageActionsService.customActions$.getValue(),\n            isMine: this.isSentByCurrentUser,\n            messageTextHtmlElement: this.messageTextElement?.nativeElement,\n          });\n          return;\n        } else {\n          this.shouldPreventMessageMenuClose = !options.fromTouch;\n          // Fix for iOS Safari: iOS Safari won't close the input if we open message menu\n          // The virtual keyboard can hide parts of the message menu, so we close the input here\n          if (\n            document.activeElement &&\n            typeof (document.activeElement as HTMLInputElement).blur !==\n              'undefined'\n          )\n            (document.activeElement as HTMLInputElement).blur();\n          this.messageMenuTrigger.show();\n          this.messageActionsService.messageMenuOpenedFor$.next(\n            this.message?.id\n          );\n        }\n        if (this.isViewInited) {\n          this.cdRef.detectChanges();\n        }\n        this.showMessageMenuTimeout = undefined;\n      });\n    }, 400);\n  }\n\n  private registerMenuTriggerEventHandlers() {\n    this.messageBubble!.nativeElement.addEventListener('touchstart', () =>\n      this.touchStarted()\n    );\n    this.messageBubble!.nativeElement.addEventListener('touchend', () =>\n      this.touchEnded()\n    );\n    this.messageBubble!.nativeElement.addEventListener('mousedown', (e) =>\n      this.mousePushedDown(e)\n    );\n    this.messageBubble!.nativeElement.addEventListener('mouseup', () =>\n      this.mouseReleased()\n    );\n    this.messageBubble!.nativeElement.addEventListener('click', (e) =>\n      this.messageBubbleClicked(e)\n    );\n  }\n\n  private stopMessageMenuShowTimer() {\n    if (this.showMessageMenuTimeout) {\n      clearTimeout(this.showMessageMenuTimeout);\n      this.showMessageMenuTimeout = undefined;\n    }\n  }\n}\n","<ng-container\n  *ngIf=\"\n    message?.type === 'error' &&\n      (message?.moderation_details?.action ===\n        'MESSAGE_RESPONSE_ACTION_REMOVE' ||\n        message?.moderation?.action === 'remove');\n    else notBlockedMessage\n  \"\n>\n  <ng-container\n    *ngTemplateOutlet=\"\n      (customTemplatesService.messageBlockedTemplate$ | async) ||\n        defaultBlockedMessage;\n      context: getMessageBlockedContext()\n    \"\n  />\n  <ng-template #defaultBlockedMessage>\n    <stream-message-blocked\n      [message]=\"message\"\n      [isMyMessage]=\"isSentByCurrentUser\"\n    />\n  </ng-template>\n</ng-container>\n<ng-template #notBlockedMessage>\n  <div\n    data-testid=\"message-container\"\n    class=\"str-chat__message-simple str-chat__message str-chat__message--{{\n      message?.type\n    }} str-chat__message--{{ message?.status }} {{\n      message?.text ? 'str-chat__message--has-text' : 'has-no-text'\n    }} str-chat__message-menu-{{ areMessageOptionsOpen ? 'opened' : 'closed' }}\"\n    [class.str-chat__message--me]=\"isSentByCurrentUser\"\n    [class.str-chat__message--other]=\"!isSentByCurrentUser\"\n    [class.str-chat__message-simple--me]=\"isSentByCurrentUser\"\n    [class.str-chat__message--has-attachment]=\"hasAttachment\"\n    [class.str-chat__message--with-reactions]=\"hasReactions\"\n    [class.str-chat__message--highlighted]=\"isHighlighted\"\n    [class.str-chat__message-with-thread-link]=\"shouldDisplayThreadLink\"\n    [class.str-chat__message-send-can-be-retried]=\"\n      (message?.status === 'failed' && message?.errorStatusCode !== 403) ||\n      (message?.type === 'error' && message?.moderation_details)\n    \"\n    [class.str-chat__message-with-touch-support]=\"hasTouchSupport\"\n    [class.str-chat__message-without-touch-support]=\"!hasTouchSupport\"\n  >\n    <ng-container *ngIf=\"!message?.deleted_at; else deletedMessage\">\n      <ng-container *ngIf=\"message?.type !== 'system'; else systemMessage\">\n        <stream-avatar-placeholder\n          data-testid=\"avatar\"\n          class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\n          type=\"user\"\n          location=\"message-sender\"\n          [imageUrl]=\"message?.user?.image\"\n          [name]=\"message?.user?.name || message?.user?.id\"\n          [user]=\"message?.user || undefined\"\n        />\n        <div class=\"str-chat__message-inner\">\n          <div\n            *ngIf=\"!hasTouchSupport && areOptionsVisible\"\n            class=\"str-chat__message-simple__actions str-chat__message-options\"\n            data-testid=\"message-options\"\n            [class.str-chat__message-actions-open]=\"areMessageOptionsOpen\"\n          >\n            <div\n              #messageActionsToggle\n              data-testid=\"message-actions-container\"\n              class=\"str-chat__message-actions-container str-chat__message-simple__actions__action str-chat__message-simple__actions__action--options\"\n              [floatUiLoose]=\"messageMenuFloat\"\n              [looseTrigger]=\"\n                messageActionsService.customActionClickHandler\n                  ? 'none'\n                  : 'click'\n              \"\n              [hideOnScroll]=\"false\"\n              [hideOnClickOutside]=\"true\"\n              [hideOnMouseLeave]=\"false\"\n              [disableAnimation]=\"false\"\n              [preventOverflow]=\"true\"\n              [positionFixed]=\"true\"\n              (onSHown)=\"areMessageOptionsOpen = true\"\n              (onHidden)=\"areMessageOptionsOpen = false\"\n            >\n              <div\n                *ngIf=\"visibleMessageActionsCount > 0\"\n                class=\"str-chat__message-actions-box-button\"\n                data-testid=\"message-options-button\"\n                (click)=\"messageOptionsButtonClicked()\"\n                (keyup.enter)=\"messageOptionsButtonClicked()\"\n              >\n                <stream-icon-placeholder\n                  icon=\"action\"\n                  class=\"str-chat__message-action-icon\"\n                />\n              </div>\n            </div>\n          </div>\n          <ng-container\n            *ngIf=\"\n              customTemplatesService.customMessageMetadataInsideBubbleTemplate$\n                | async\n            \"\n          >\n            <ng-template\n              *ngTemplateOutlet=\"\n                (customTemplatesService.customMessageMetadataInsideBubbleTemplate$\n                  | async)!;\n                context: { message: message }\n              \"\n            ></ng-template>\n          </ng-container>\n          <div class=\"str-chat__message-reactions-host\">\n            <ng-template\n              #defaultMessageReactions\n              let-messageReactionCounts=\"messageReactionCounts\"\n              let-latestReactions=\"latestReactions\"\n              let-messageId=\"messageId\"\n              let-ownReactions=\"ownReactions\"\n              let-messageReactionGroups=\"messageReactionGroups\"\n            >\n              <stream-message-reactions\n                [messageReactionCounts]=\"messageReactionCounts\"\n                [latestReactions]=\"latestReactions\"\n                [messageId]=\"messageId\"\n                [ownReactions]=\"ownReactions\"\n                [messageReactionGroups]=\"messageReactionGroups\"\n              />\n            </ng-template>\n            <ng-container\n              *ngTemplateOutlet=\"\n                (customTemplatesService.messageReactionsTemplate$ | async) ||\n                  defaultMessageReactions;\n                context: getMessageReactionsContext()\n              \"\n            />\n          </div>\n          <float-ui-content #messageMenuFloat>\n            <ng-template\n              #defaultMessageActionsBox\n              let-isMine=\"isMine\"\n              let-messageInput=\"message\"\n              let-enabledActions=\"enabledActions\"\n              let-messageTextHtmlElement=\"messageTextHtmlElement\"\n            >\n              <stream-message-actions-box\n                [isMine]=\"isMine\"\n                [message]=\"messageInput\"\n                [enabledActions]=\"enabledActions\"\n                [messageTextHtmlElement]=\"messageTextHtmlElement\"\n              />\n            </ng-template>\n            <ng-container>\n              <div\n                (click)=\"messageActionsBoxClicked(messageMenuFloat)\"\n                (keyup.enter)=\"messageActionsBoxClicked(messageMenuFloat)\"\n              >\n                <ng-container\n                  *ngTemplateOutlet=\"\n                    (customTemplatesService.messageActionsBoxTemplate$\n                      | async) || defaultMessageActionsBox;\n                    context: getMessageActionsBoxContext()\n                  \"\n                />\n              </div>\n            </ng-container>\n          </float-ui-content>\n          <div\n            class=\"str-chat__message-bubble str-chat-angular__message-bubble\"\n            [class.str-chat-angular__message-bubble--attachment-modal-open]=\"\n              nestedModalState === 'opened'\n            \"\n            data-testid=\"message-bubble\"\n            [floatUiLoose]=\"messageMenuFloat\"\n            #messageMenuTrigger=\"floatUiLoose\"\n            #messageBubble\n            looseTrigger=\"none\"\n            [hideOnScroll]=\"false\"\n            [hideOnClickOutside]=\"true\"\n            [hideOnMouseLeave]=\"false\"\n            [disableAnimation]=\"true\"\n            [preventOverflow]=\"true\"\n            (onShown)=\"areMessageOptionsOpen = true\"\n            (onHidden)=\"areMessageOptionsOpen = false\"\n            [positionFixed]=\"true\"\n          >\n            <ng-container *ngIf=\"hasAttachment && !message?.quoted_message\">\n              <div class=\"str-chat__attachments-container\">\n                <ng-container *ngTemplateOutlet=\"attachmentsTemplate\" />\n              </div>\n            </ng-container>\n            <ng-container\n              *ngIf=\"\n                message?.poll_id &&\n                (customTemplatesService.pollTemplate$ | async)\n              \"\n            >\n              <ng-container\n                *ngTemplateOutlet=\"\n                  (customTemplatesService.pollTemplate$ | async)!;\n                  context: {\n                    pollId: message!.poll_id!,\n                    messageId: message!.id,\n                    isQuote: false\n                  }\n                \"\n              />\n            </ng-container>\n            <div\n              *ngIf=\"\n                message?.text || (message?.quoted_message && hasAttachment)\n              \"\n              class=\"str-chat__message-text\"\n              tabindex=\"0\"\n              [class.str-chat__message-text--pointer-cursor]=\"\n                (message?.status === 'failed' &&\n                  message?.errorStatusCode !== 403) ||\n                (this.message?.type === 'error' &&\n                  this.message?.moderation_details) ||\n                message?.message_text_updated_at\n              \"\n              (click)=\"messageClicked()\"\n              (keyup.enter)=\"messageClicked()\"\n            >\n              <div\n                data-testid=\"inner-message\"\n                class=\"str-chat__message-text-inner str-chat__message-simple-text-inner\"\n                [class.str-chat__message-light-text-inner--has-attachment]=\"\n                  hasAttachment\n                \"\n              >\n                <ng-container *ngTemplateOutlet=\"quotedMessage\" />\n                <ng-container *ngIf=\"hasAttachment && message?.quoted_message\">\n                  <ng-container *ngTemplateOutlet=\"attachmentsTemplate\" />\n                </ng-container>\n                <div\n                  *ngIf=\"message?.type === 'error'\"\n                  data-testid=\"client-error-message\"\n                  class=\"str-chat__simple-message--error-message str-chat__message--error-message\"\n                >\n                  <ng-container *ngIf=\"!message?.moderation_details\">{{\n                    \"streamChat.Error · Unsent\" | translate\n                  }}</ng-container>\n                </div>\n                <div\n                  *ngIf=\"message?.status === 'failed'\"\n                  data-testid=\"error-message\"\n                  class=\"str-chat__simple-message--error-message str-chat__message--error-message\"\n                >\n                  {{\n                    (message?.errorStatusCode === 403\n                      ? \"streamChat.Message Failed · Unauthorized\"\n                      : \"streamChat.Message Failed · Click to try again\"\n                    ) | translate\n                  }}\n                </div>\n                <ng-container\n                  *ngTemplateOutlet=\"\n                    (customTemplatesService.messageTextTemplate$ | async) ||\n                      defaultText;\n                    context: getMessageTextContext()\n                  \"\n                />\n                <ng-template\n                  #defaultText\n                  let-message=\"message\"\n                  let-isQuoted=\"isQuoted\"\n                  let-shouldTranslate=\"shouldTranslate\"\n                >\n                  <stream-message-text\n                    [message]=\"message\"\n                    [isQuoted]=\"isQuoted\"\n                    [shouldTranslate]=\"shouldTranslate\"\n                    data-testid=\"quoted-message-text\"\n                  />\n                </ng-template>\n              </div>\n            </div>\n            <stream-icon-placeholder\n              class=\"str-chat__message-error-icon\"\n              icon=\"error\"\n            />\n          </div>\n        </div>\n        <ng-container\n          *ngTemplateOutlet=\"replyCountButton; context: { message: message }\"\n        />\n\n        <ng-container *ngTemplateOutlet=\"messageDateAndSender\" />\n      </ng-container>\n    </ng-container>\n  </div>\n\n  <ng-template #deletedMessage>\n    <div data-testid=\"message-deleted-component\">\n      <div class=\"str-chat__message--deleted-inner\" translate>\n        streamChat.This message was deleted...\n      </div>\n    </div>\n  </ng-template>\n\n  <ng-template #systemMessage>\n    <ng-container\n      *ngTemplateOutlet=\"\n        (customTemplatesService.systemMessageTemplate$ | async) ||\n          defaultSystemMessage;\n        context: getMessageContext()\n      \"\n    />\n    <ng-template #defaultSystemMessage let-messageInput=\"message\">\n      <div data-testid=\"system-message\" class=\"str-chat__message--system\">\n        <div class=\"str-chat__message--system__text\">\n          <div class=\"str-chat__message--system__line\"></div>\n          <p>{{ messageInput?.text }}</p>\n          <div class=\"str-chat__message--system__line\"></div>\n        </div>\n        <div class=\"str-chat__message--system__date\">\n          {{ parsedDate }}\n        </div>\n      </div>\n    </ng-template>\n  </ng-template>\n\n  <ng-template #quotedMessage>\n    <div\n      *ngIf=\"message?.quoted_message\"\n      class=\"quoted-message str-chat__quoted-message-preview\"\n      data-testid=\"quoted-message-container\"\n      [class.mine]=\"isSentByCurrentUser\"\n      (click)=\"\n        jumpToMessage(\n          (message?.quoted_message)!.id,\n          message?.quoted_message?.parent_id\n        )\n      \"\n      (keyup.enter)=\"\n        jumpToMessage(\n          (message?.quoted_message)!.id,\n          message?.quoted_message?.parent_id\n        )\n      \"\n    >\n      <stream-avatar-placeholder\n        data-testid=\"qouted-message-avatar\"\n        class=\"str-chat-angular__avatar-host str-chat__message-sender-avatar\"\n        type=\"user\"\n        location=\"quoted-message-sender\"\n        [imageUrl]=\"message?.quoted_message?.user?.image\"\n        [name]=\"\n          message?.quoted_message?.user?.name ||\n          message?.quoted_message?.user?.id\n        \"\n        [user]=\"message?.quoted_message?.user || undefined\"\n      />\n      <div class=\"quoted-message-inner str-chat__quoted-message-bubble\">\n        <ng-container\n          *ngIf=\"\n            message?.quoted_message?.attachments &&\n            message?.quoted_message?.attachments?.length\n          \"\n        >\n          <ng-template\n            #defaultAttachments\n            let-messageId=\"messageId\"\n            let-attachments=\"attachments\"\n            let-parentMessageId=\"parentMessageId\"\n            let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n          >\n            <stream-attachment-list\n              [messageId]=\"messageId\"\n              [attachments]=\"attachments\"\n              [parentMessageId]=\"parentMessageId\"\n              (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n            />\n          </ng-template>\n          <ng-container\n            *ngTemplateOutlet=\"\n              (customTemplatesService.attachmentListTemplate$ | async) ||\n                defaultAttachments;\n              context: getQuotedMessageAttachmentListContext()\n            \"\n          />\n        </ng-container>\n        <ng-container\n          *ngIf=\"\n            message?.quoted_message?.poll_id &&\n            (customTemplatesService.pollTemplate$ | async)\n          \"\n        >\n          <ng-container\n            *ngTemplateOutlet=\"\n              (customTemplatesService.pollTemplate$ | async)!;\n              context: {\n                pollId: message?.quoted_message?.poll_id,\n                messageId: message?.quoted_message?.id,\n                isQuote: true\n              }\n            \"\n          />\n        </ng-container>\n        <ng-container *ngIf=\"message?.quoted_message\">\n          <ng-container\n            *ngTemplateOutlet=\"\n              (customTemplatesService.messageTextTemplate$ | async) ||\n                defaultText;\n              context: getQuotedMessageTextContext()\n            \"\n          />\n          <ng-template\n            #defaultText\n            let-message=\"message\"\n            let-isQuoted=\"isQuoted\"\n            let-shouldTranslate=\"shouldTranslate\"\n          >\n            <stream-message-text\n              [message]=\"message\"\n              [isQuoted]=\"isQuoted\"\n              [shouldTranslate]=\"shouldTranslate\"\n              data-testid=\"quoted-message-text\"\n            />\n          </ng-template>\n        </ng-container>\n      </div>\n    </div>\n  </ng-template>\n\n  <ng-template #messageDateAndSender>\n    <ng-container>\n      <div\n        *ngIf=\"shouldDisplayTranslationNotice\"\n        class=\"str-chat__translation-notice\"\n        data-testid=\"translation-notice\"\n      >\n        <button\n          *ngIf=\"displayedMessageTextContent === 'translation'\"\n          data-testid=\"see-original\"\n          translate\n          (click)=\"displayOriginalMessage()\"\n          (keyup.enter)=\"displayOriginalMessage()\"\n        >\n          streamChat.See original (automatically translated)\n        </button>\n        <button\n          *ngIf=\"displayedMessageTextContent === 'original'\"\n          data-testid=\"see-translation\"\n          translate\n          (click)=\"displayTranslatedMessage()\"\n          (keyup.enter)=\"displayTranslatedMessage()\"\n        >\n          streamChat.See translation\n        </button>\n      </div>\n      <ng-container\n        *ngIf=\"customTemplatesService.customMessageMetadataTemplate$ | async\"\n      >\n        <div class=\"str-chat__custom-message-metadata\">\n          <ng-container\n            *ngTemplateOutlet=\"\n              (customTemplatesService.customMessageMetadataTemplate$ | async)!;\n              context: getMessageMetadataContext()\n            \"\n          />\n        </div>\n      </ng-container>\n      <div\n        class=\"str-chat__message-data str-chat__message-simple-data str-chat__message-metadata\"\n      >\n        <ng-container *ngTemplateOutlet=\"messageStatus\" />\n\n        <span\n          *ngIf=\"!isSentByCurrentUser\"\n          data-testid=\"sender\"\n          class=\"str-chat__message-simple-name str-chat__message-sender-name\"\n        >\n          {{ message?.user?.name ? message?.user?.name : message?.user?.id }}\n        </span>\n        <span\n          data-testid=\"date\"\n          class=\"str-chat__message-simple-timestamp str-chat__message-simple-time\"\n        >\n          {{ parsedDate }}\n        </span>\n        <ng-container *ngIf=\"message?.message_text_updated_at\">\n          <span\n            data-testid=\"edited-flag\"\n            class=\"str-chat__mesage-simple-edited\"\n            translate\n            >streamChat.Edited</span\n          >\n          <div\n            data-testid=\"edited-timestamp\"\n            class=\"str-chat__message-edited-timestamp\"\n            [ngClass]=\"{\n              'str-chat__message-edited-timestamp--open': isEditedFlagOpened,\n              'str-chat__message-edited-timestamp--collapsed':\n                !isEditedFlagOpened\n            }\"\n          >\n            <span translate>streamChat.Edited</span>\n            <time\n              dateTime=\"{{ message?.message_text_updated_at }}\"\n              title=\"{{ message?.message_text_updated_at }}\"\n            >\n              {{ pasedEditedDate }}\n            </time>\n          </div>\n        </ng-container>\n      </div>\n    </ng-container>\n  </ng-template>\n\n  <ng-template #messageStatus>\n    <ng-container\n      *ngIf=\"\n        isSentByCurrentUser &&\n        ((isLastSentMessage && message?.status === 'received') ||\n          message?.status === 'sending')\n      \"\n    >\n      <ng-container *ngIf=\"message?.status === 'sending'; else sentStatus\">\n        <ng-container *ngTemplateOutlet=\"sendingStatus\" />\n      </ng-container>\n      <ng-template #sentStatus>\n        <ng-container\n          *ngIf=\"\n            mode === 'main' &&\n              isMessageDeliveredAndRead &&\n              canDisplayReadStatus;\n            else deliveredStatus\n          \"\n        >\n          <ng-container *ngTemplateOutlet=\"readStatus\" />\n        </ng-container>\n      </ng-template>\n      <ng-template #deliveredStatus>\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.deliveredStatusTemplate$ | async) ||\n              defaultDeliveredStatus;\n            context: getDeliveredStatusContext()\n          \"\n        />\n      </ng-template>\n      <ng-template #defaultDeliveredStatus>\n        <span\n          *ngIf=\"mode === 'main'\"\n          class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n          data-testid=\"delivered-indicator\"\n          tabindex=\"0\"\n          [floatUiLoose]=\"floatingContent\"\n          loosePlacement=\"top\"\n          [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n          [disableAnimation]=\"true\"\n          [hideOnClickOutside]=\"true\"\n          [positionFixed]=\"true\"\n          [preventOverflow]=\"true\"\n        >\n          <float-ui-content #floatingContent>\n            <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n              {{ \"streamChat.Delivered\" | translate }}\n            </div>\n          </float-ui-content>\n          <stream-icon-placeholder\n            data-testid=\"delivered-icon\"\n            icon=\"delivered\"\n          />\n        </span>\n      </ng-template>\n      <ng-template #sendingStatus>\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.sendingStatusTemplate$ | async) ||\n              defaultSendingStatus;\n            context: getSendingStatusContext()\n          \"\n        />\n      </ng-template>\n      <ng-template #defaultSendingStatus>\n        <span\n          class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n          data-testid=\"sending-indicator\"\n          tabindex=\"0\"\n          [floatUiLoose]=\"floatingContent\"\n          loosePlacement=\"top\"\n          [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n          [disableAnimation]=\"true\"\n          [hideOnClickOutside]=\"true\"\n          [positionFixed]=\"true\"\n          [preventOverflow]=\"true\"\n        >\n          <float-ui-content #floatingContent>\n            <div class=\"str-chat__tooltip str-chat__tooltip-angular\">\n              {{ \"streamChat.Sending...\" | translate }}\n            </div>\n          </float-ui-content>\n          <stream-loading-indicator-placeholder\n            data-testid=\"loading-indicator\"\n          />\n        </span>\n      </ng-template>\n      <ng-template #readStatus>\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.readStatusTemplate$ | async) ||\n              defaultReadStatus;\n            context: getReadStatusContext()\n          \"\n        />\n      </ng-template>\n      <ng-template #defaultReadStatus let-readByText=\"readByText\">\n        <span\n          class=\"str-chat__message-simple-status str-chat__message-simple-status-angular str-chat__message-status\"\n          data-testid=\"read-indicator\"\n          tabindex=\"0\"\n          [floatUiLoose]=\"floatingContent\"\n          loosePlacement=\"top\"\n          [looseTrigger]=\"hasTouchSupport ? 'click' : 'hover'\"\n          [disableAnimation]=\"true\"\n          [hideOnClickOutside]=\"true\"\n          [positionFixed]=\"true\"\n          [preventOverflow]=\"true\"\n        >\n          <float-ui-content #floatingContent>\n            <div\n              class=\"str-chat__tooltip str-chat__tooltip-angular\"\n              data-testid=\"read-by-tooltip\"\n            >\n              {{ readByText }}\n            </div>\n          </float-ui-content>\n          <stream-icon-placeholder icon=\"read\" />\n        </span>\n      </ng-template>\n    </ng-container>\n  </ng-template>\n\n  <ng-template #replyCountButton>\n    <div\n      class=\"str-chat__message-simple-reply-button str-chat__message-replies-count-button-wrapper\"\n    >\n      <ng-container *ngIf=\"shouldDisplayThreadLink\">\n        <ng-template\n          *ngTemplateOutlet=\"\n            (customTemplatesService.threadLinkButton$ | async) || defaultButton;\n            context: { message: message }\n          \"\n        ></ng-template>\n      </ng-container>\n      <ng-template #defaultButton let-message=\"message\">\n        <button\n          class=\"str-chat__message-replies-count-button\"\n          data-testid=\"reply-count-button\"\n          (click)=\"setAsActiveParentMessage()\"\n        >\n          {{message?.reply_count === 1 ? ('streamChat.1 reply' | translate) : ('streamChat.{{ replyCount }}\n          replies' | translate:replyCountParam)}}\n        </button>\n      </ng-template>\n    </div>\n  </ng-template>\n\n  <ng-template #attachmentsTemplate>\n    <ng-template\n      #defaultAttachments\n      let-messageId=\"messageId\"\n      let-attachments=\"attachments\"\n      let-parentMessageId=\"parentMessageId\"\n      let-imageModalStateChangeHandler=\"imageModalStateChangeHandler\"\n    >\n      <stream-attachment-list\n        [messageId]=\"messageId\"\n        [attachments]=\"attachments\"\n        [parentMessageId]=\"parentMessageId\"\n        (imageModalStateChange)=\"imageModalStateChangeHandler($event)\"\n      />\n    </ng-template>\n    <ng-container\n      *ngTemplateOutlet=\"\n        (customTemplatesService.attachmentListTemplate$ | async) ||\n          defaultAttachments;\n        context: getAttachmentListContext()\n      \"\n    />\n  </ng-template>\n</ng-template>\n"]}
|