stream-chat-angular 5.13.0 → 6.0.0-beta.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/i18n/en.d.ts +1 -0
- package/assets/version.d.ts +1 -1
- package/{esm2020 → esm2022}/assets/i18n/en.mjs +2 -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 +212 -0
- package/esm2022/lib/attachment-preview-list/attachment-preview-list.component.mjs +55 -0
- package/{esm2020 → esm2022}/lib/attachment.service.mjs +5 -5
- package/esm2022/lib/avatar/avatar.component.mjs +157 -0
- package/{esm2020 → esm2022}/lib/avatar-placeholder/avatar-placeholder.component.mjs +6 -6
- package/esm2022/lib/channel/channel.component.mjs +45 -0
- package/esm2022/lib/channel-header/channel-header.component.mjs +72 -0
- package/esm2022/lib/channel-list/channel-list.component.mjs +50 -0
- package/esm2022/lib/channel-preview/channel-preview.component.mjs +150 -0
- package/esm2022/lib/channel.service.mjs +1381 -0
- package/esm2022/lib/chat-client.service.mjs +227 -0
- package/{esm2020 → esm2022}/lib/custom-templates.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/date-parser.service.mjs +5 -5
- package/esm2022/lib/file-utils.mjs +35 -0
- package/{esm2020 → esm2022}/lib/get-channel-display-text.mjs +1 -1
- package/{esm2020 → esm2022}/lib/get-message-translation.mjs +1 -1
- 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/{esm2020 → esm2022}/lib/list-users.mjs +1 -1
- package/esm2022/lib/message/message.component.mjs +486 -0
- package/esm2022/lib/message-actions-box/message-actions-box.component.mjs +120 -0
- package/{esm2020 → esm2022}/lib/message-actions.service.mjs +5 -5
- package/esm2022/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +71 -0
- 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/esm2022/lib/message-input/message-input.component.mjs +507 -0
- 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/{esm2020 → esm2022}/lib/message-list/group-styles.mjs +1 -1
- package/esm2022/lib/message-list/message-list.component.mjs +715 -0
- package/{esm2020 → esm2022}/lib/message-preview.mjs +1 -1
- package/esm2022/lib/message-reactions/message-reactions.component.mjs +165 -0
- package/esm2022/lib/message-reactions-selector/message-reactions-selector.component.mjs +57 -0
- package/{esm2020 → esm2022}/lib/message-reactions.service.mjs +5 -5
- package/{esm2020 → esm2022}/lib/message-text/message-text.component.mjs +6 -6
- package/esm2022/lib/message.service.mjs +43 -0
- package/{esm2020 → esm2022}/lib/modal/modal.component.mjs +6 -6
- package/{esm2020 → esm2022}/lib/notification/notification.component.mjs +6 -6
- package/esm2022/lib/notification-list/notification-list.component.mjs +33 -0
- package/{esm2020 → esm2022}/lib/notification.service.mjs +5 -5
- package/esm2022/lib/paginated-list/paginated-list.component.mjs +94 -0
- package/{esm2020 → esm2022}/lib/parse-date.mjs +1 -1
- package/{esm2020 → esm2022}/lib/read-by.mjs +1 -1
- 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 +59 -59
- package/{esm2020 → esm2022}/lib/stream-i18n.service.mjs +5 -5
- 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/esm2022/lib/types.mjs +2 -0
- package/{esm2020 → esm2022}/lib/user-list/user-list.component.mjs +5 -5
- package/esm2022/lib/virtualized-list.service.mjs +273 -0
- package/{esm2020 → esm2022}/lib/virtualized-message-list.service.mjs +1 -1
- 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/media-recorder.mjs +1 -1
- package/esm2022/lib/voice-recorder/mp3-transcoder.mjs +61 -0
- package/esm2022/lib/voice-recorder/transcoder.service.mjs +121 -0
- package/esm2022/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +32 -0
- package/esm2022/lib/voice-recorder/voice-recorder.component.mjs +80 -0
- package/{esm2020 → esm2022}/lib/voice-recorder/voice-recorder.module.mjs +9 -9
- package/esm2022/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +112 -0
- package/esm2022/lib/voice-recording/voice-recording.component.mjs +91 -0
- package/{esm2020 → esm2022}/lib/voice-recording/voice-recording.module.mjs +5 -5
- package/{esm2020 → esm2022}/lib/wave-form-sampler.mjs +1 -1
- package/esm2022/public-api.mjs +82 -0
- package/{fesm2020 → fesm2022}/stream-chat-angular.mjs +857 -1140
- package/fesm2022/stream-chat-angular.mjs.map +1 -0
- package/lib/attachment-list/attachment-list.component.d.ts +2 -5
- package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +2 -2
- package/lib/attachment.service.d.ts +1 -1
- package/lib/avatar/avatar.component.d.ts +4 -4
- package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +1 -1
- package/lib/channel-list/channel-list.component.d.ts +1 -0
- package/lib/channel-preview/channel-preview.component.d.ts +3 -4
- package/lib/channel.service.d.ts +40 -106
- package/lib/chat-client.service.d.ts +1 -4
- package/lib/custom-templates.service.d.ts +10 -10
- 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 +2 -2
- package/lib/message-actions-box/message-actions-box.component.d.ts +2 -3
- package/lib/message-actions.service.d.ts +1 -1
- package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +2 -2
- package/lib/message-input/message-input.component.d.ts +2 -2
- package/lib/message-input/textarea/textarea.component.d.ts +1 -1
- package/lib/message-input/textarea.directive.d.ts +2 -2
- package/lib/message-list/group-styles.d.ts +1 -1
- package/lib/message-list/message-list.component.d.ts +2 -3
- package/lib/message-reactions/message-reactions.component.d.ts +2 -3
- package/lib/message-reactions-selector/message-reactions-selector.component.d.ts +1 -2
- 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/notification-list/notification-list.component.d.ts +0 -1
- package/lib/paginated-list/paginated-list.component.d.ts +5 -2
- package/lib/read-by.d.ts +1 -1
- package/lib/types.d.ts +98 -84
- 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-wavebar/voice-recorder-wavebar.component.d.ts +0 -1
- package/lib/voice-recorder/voice-recorder.component.d.ts +2 -2
- package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +3 -3
- package/lib/voice-recording/voice-recording.component.d.ts +1 -1
- package/package.json +15 -21
- package/public-api.d.ts +0 -1
- package/src/assets/i18n/en.ts +1 -0
- package/src/assets/version.ts +1 -1
- package/esm2020/lib/attachment-list/attachment-list.component.mjs +0 -224
- package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +0 -55
- package/esm2020/lib/avatar/avatar.component.mjs +0 -160
- package/esm2020/lib/channel/channel.component.mjs +0 -45
- package/esm2020/lib/channel-header/channel-header.component.mjs +0 -72
- package/esm2020/lib/channel-list/channel-list.component.mjs +0 -47
- package/esm2020/lib/channel-preview/channel-preview.component.mjs +0 -155
- package/esm2020/lib/channel-query.mjs +0 -77
- package/esm2020/lib/channel.service.mjs +0 -1546
- package/esm2020/lib/chat-client.service.mjs +0 -238
- package/esm2020/lib/file-utils.mjs +0 -35
- package/esm2020/lib/message/message.component.mjs +0 -486
- package/esm2020/lib/message-actions-box/message-actions-box.component.mjs +0 -123
- package/esm2020/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +0 -71
- package/esm2020/lib/message-input/message-input.component.mjs +0 -507
- package/esm2020/lib/message-list/message-list.component.mjs +0 -717
- package/esm2020/lib/message-reactions/message-reactions.component.mjs +0 -168
- package/esm2020/lib/message-reactions-selector/message-reactions-selector.component.mjs +0 -61
- package/esm2020/lib/message.service.mjs +0 -43
- package/esm2020/lib/notification-list/notification-list.component.mjs +0 -36
- package/esm2020/lib/paginated-list/paginated-list.component.mjs +0 -94
- package/esm2020/lib/stream-autocomplete-textarea.module.mjs +0 -33
- package/esm2020/lib/stream-textarea.module.mjs +0 -31
- package/esm2020/lib/types.mjs +0 -2
- package/esm2020/lib/virtualized-list.service.mjs +0 -271
- package/esm2020/lib/voice-recorder/mp3-transcoder.mjs +0 -61
- package/esm2020/lib/voice-recorder/transcoder.service.mjs +0 -121
- package/esm2020/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +0 -35
- package/esm2020/lib/voice-recorder/voice-recorder.component.mjs +0 -80
- package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +0 -112
- package/esm2020/lib/voice-recording/voice-recording.component.mjs +0 -91
- package/esm2020/public-api.mjs +0 -83
- package/fesm2015/stream-chat-angular.mjs +0 -9152
- package/fesm2015/stream-chat-angular.mjs.map +0 -1
- package/fesm2020/stream-chat-angular.mjs.map +0 -1
- package/lib/channel-query.d.ts +0 -26
- /package/{esm2020 → esm2022}/lib/format-duration.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/message-input/textarea.interface.mjs +0 -0
- /package/{esm2020 → esm2022}/stream-chat-angular.mjs +0 -0
|
@@ -18,4 +18,4 @@ export const createMessagePreview = (user, text, attachments = [], mentionedUser
|
|
|
18
18
|
...customData,
|
|
19
19
|
};
|
|
20
20
|
};
|
|
21
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
21
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1wcmV2aWV3LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2UtcHJldmlldy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxPQUFPLEVBQUUsRUFBRSxJQUFJLE1BQU0sRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUdwQyxNQUFNLENBQUMsTUFBTSxvQkFBb0IsR0FBRyxDQUdsQyxJQUFrQixFQUNsQixJQUFZLEVBQ1osY0FBK0IsRUFBRSxFQUNqQyxpQkFBb0MsRUFBRSxFQUN0QyxXQUErQixTQUFTLEVBQ3hDLGtCQUFzQyxTQUFTLEVBQy9DLFVBQWlELEVBQ2pELEVBQUU7SUFDRixNQUFNLFlBQVksR0FBRyxHQUFHLElBQUksQ0FBQyxFQUFFLElBQUksTUFBTSxFQUFFLEVBQUUsQ0FBQztJQUU5QyxPQUFPO1FBQ0wsTUFBTSxFQUFFLElBQUk7UUFDWixVQUFVLEVBQUUsSUFBSSxJQUFJLEVBQUU7UUFDdEIsSUFBSSxFQUFFLElBQUk7UUFDVixFQUFFLEVBQUUsWUFBWTtRQUNoQixTQUFTLEVBQUUsRUFBRTtRQUNiLE1BQU0sRUFBRSxTQUFTO1FBQ2pCLElBQUk7UUFDSixJQUFJLEVBQUUsU0FBUztRQUNmLElBQUk7UUFDSixXQUFXO1FBQ1gsZUFBZSxFQUFFLGNBQWM7UUFDL0IsU0FBUyxFQUFFLFFBQVE7UUFDbkIsaUJBQWlCLEVBQUUsZUFBZTtRQUNsQyxHQUFHLFVBQVU7S0FDbUIsQ0FBQztBQUNyQyxDQUFDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBBdHRhY2htZW50LCBNZXNzYWdlUmVzcG9uc2UsIFVzZXJSZXNwb25zZSB9IGZyb20gJ3N0cmVhbS1jaGF0JztcbmltcG9ydCB7IHY0IGFzIHV1aWR2NCB9IGZyb20gJ3V1aWQnO1xuaW1wb3J0IHsgRGVmYXVsdFN0cmVhbUNoYXRHZW5lcmljcyB9IGZyb20gJy4vdHlwZXMnO1xuXG5leHBvcnQgY29uc3QgY3JlYXRlTWVzc2FnZVByZXZpZXcgPSA8XG4gIFQgZXh0ZW5kcyBEZWZhdWx0U3RyZWFtQ2hhdEdlbmVyaWNzID0gRGVmYXVsdFN0cmVhbUNoYXRHZW5lcmljcyxcbj4oXG4gIHVzZXI6IFVzZXJSZXNwb25zZSxcbiAgdGV4dDogc3RyaW5nLFxuICBhdHRhY2htZW50czogQXR0YWNobWVudDxUPltdID0gW10sXG4gIG1lbnRpb25lZFVzZXJzOiBVc2VyUmVzcG9uc2U8VD5bXSA9IFtdLFxuICBwYXJlbnRJZDogdW5kZWZpbmVkIHwgc3RyaW5nID0gdW5kZWZpbmVkLFxuICBxdW90ZWRNZXNzYWdlSWQ6IHVuZGVmaW5lZCB8IHN0cmluZyA9IHVuZGVmaW5lZCxcbiAgY3VzdG9tRGF0YTogdW5kZWZpbmVkIHwgUGFydGlhbDxUWydtZXNzYWdlVHlwZSddPixcbikgPT4ge1xuICBjb25zdCBjbGllbnRTaWRlSWQgPSBgJHt1c2VyLmlkfS0ke3V1aWR2NCgpfWA7XG5cbiAgcmV0dXJuIHtcbiAgICBfX2h0bWw6IHRleHQsXG4gICAgY3JlYXRlZF9hdDogbmV3IERhdGUoKSxcbiAgICBodG1sOiB0ZXh0LFxuICAgIGlkOiBjbGllbnRTaWRlSWQsXG4gICAgcmVhY3Rpb25zOiBbXSxcbiAgICBzdGF0dXM6ICdzZW5kaW5nJyxcbiAgICB0ZXh0LFxuICAgIHR5cGU6ICdyZWd1bGFyJyxcbiAgICB1c2VyLFxuICAgIGF0dGFjaG1lbnRzLFxuICAgIG1lbnRpb25lZF91c2VyczogbWVudGlvbmVkVXNlcnMsXG4gICAgcGFyZW50X2lkOiBwYXJlbnRJZCxcbiAgICBxdW90ZWRfbWVzc2FnZV9pZDogcXVvdGVkTWVzc2FnZUlkLFxuICAgIC4uLmN1c3RvbURhdGEsXG4gIH0gYXMgdW5rbm93biBhcyBNZXNzYWdlUmVzcG9uc2U8VD47XG59O1xuIl19
|
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
import { Component, Input, ViewChild, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "../message-reactions.service";
|
|
4
|
+
import * as i2 from "../custom-templates.service";
|
|
5
|
+
import * as i3 from "@angular/common";
|
|
6
|
+
import * as i4 from "../modal/modal.component";
|
|
7
|
+
import * as i5 from "../user-list/user-list.component";
|
|
8
|
+
/**
|
|
9
|
+
* The `MessageReactions` component displays the reactions of a message. You can read more about [message reactions](/chat/docs/javascript/send_reaction/) in the platform documentation.
|
|
10
|
+
*/
|
|
11
|
+
export class MessageReactionsComponent {
|
|
12
|
+
constructor(cdRef, messageReactionsService, customTemplatesService) {
|
|
13
|
+
this.cdRef = cdRef;
|
|
14
|
+
this.messageReactionsService = messageReactionsService;
|
|
15
|
+
this.customTemplatesService = customTemplatesService;
|
|
16
|
+
/**
|
|
17
|
+
* The number of reactions grouped by [reaction types](https://github.com/GetStream/stream-chat-angular/tree/master/projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts)
|
|
18
|
+
*/
|
|
19
|
+
this.messageReactionGroups = undefined;
|
|
20
|
+
/**
|
|
21
|
+
* The number of reactions grouped by [reaction types](https://github.com/GetStream/stream-chat-angular/tree/master/projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts)
|
|
22
|
+
* @deprecated use `messageReactionGroups`
|
|
23
|
+
*/
|
|
24
|
+
this.messageReactionCounts = {};
|
|
25
|
+
/**
|
|
26
|
+
* List of reactions of a [message](/chat/docs/sdk/angular/types/stream-message/), used to display the users of a reaction type.
|
|
27
|
+
* @deprecated you can fetch the reactions using [`messageReactionsService.queryReactions()`](/chat/docs/sdk/angular/services/MessageReactionsService/#queryreactions)
|
|
28
|
+
*/
|
|
29
|
+
this.latestReactions = [];
|
|
30
|
+
/**
|
|
31
|
+
* List of the user's own reactions of a [message](/chat/docs/sdk/angular/types/stream-message/), used to display the users of a reaction type.
|
|
32
|
+
*/
|
|
33
|
+
this.ownReactions = [];
|
|
34
|
+
this.isLoading = true;
|
|
35
|
+
this.reactions = [];
|
|
36
|
+
this.shouldHandleReactionClick = true;
|
|
37
|
+
this.existingReactions = [];
|
|
38
|
+
this.reactionOptions = [];
|
|
39
|
+
this.usersByReactions = {};
|
|
40
|
+
this.subscriptions = [];
|
|
41
|
+
this.isViewInited = false;
|
|
42
|
+
this.isOpenChange = (isOpen) => {
|
|
43
|
+
this.selectedReactionType = isOpen ? this.selectedReactionType : undefined;
|
|
44
|
+
if (!isOpen) {
|
|
45
|
+
this.usersByReactions = {};
|
|
46
|
+
}
|
|
47
|
+
};
|
|
48
|
+
}
|
|
49
|
+
ngOnChanges(changes) {
|
|
50
|
+
if (changes.messageReactionCounts || changes.messageReactionGroups) {
|
|
51
|
+
if (this.messageReactionCounts && !this.messageReactionGroups) {
|
|
52
|
+
this.messageReactionGroups = {};
|
|
53
|
+
Object.keys(this.messageReactionCounts).forEach((k) => {
|
|
54
|
+
this.messageReactionGroups[k] = {
|
|
55
|
+
count: this.messageReactionCounts?.[k] ?? 0,
|
|
56
|
+
sum_scores: this.messageReactionCounts?.[k] ?? 0,
|
|
57
|
+
first_reaction_at: undefined,
|
|
58
|
+
last_reaction_at: undefined,
|
|
59
|
+
};
|
|
60
|
+
});
|
|
61
|
+
}
|
|
62
|
+
this.setExistingReactions();
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
ngOnInit() {
|
|
66
|
+
this.subscriptions.push(this.messageReactionsService.reactions$.subscribe((reactions) => {
|
|
67
|
+
this.reactionOptions = Object.keys(reactions);
|
|
68
|
+
this.setExistingReactions();
|
|
69
|
+
if (this.isViewInited) {
|
|
70
|
+
this.cdRef.detectChanges();
|
|
71
|
+
}
|
|
72
|
+
}));
|
|
73
|
+
}
|
|
74
|
+
ngAfterViewInit() {
|
|
75
|
+
this.isViewInited = true;
|
|
76
|
+
}
|
|
77
|
+
ngOnDestroy() {
|
|
78
|
+
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
79
|
+
}
|
|
80
|
+
getEmojiByReaction(reactionType) {
|
|
81
|
+
return this.messageReactionsService.reactions[reactionType];
|
|
82
|
+
}
|
|
83
|
+
async reactionSelected(reactionType) {
|
|
84
|
+
if (!this.messageId) {
|
|
85
|
+
return;
|
|
86
|
+
}
|
|
87
|
+
if (this.messageReactionsService.customReactionClickHandler) {
|
|
88
|
+
this.messageReactionsService.customReactionClickHandler({
|
|
89
|
+
messageId: this.messageId,
|
|
90
|
+
reactionType: reactionType,
|
|
91
|
+
});
|
|
92
|
+
}
|
|
93
|
+
else {
|
|
94
|
+
this.selectedReactionType = reactionType;
|
|
95
|
+
if (!this.usersByReactions[this.selectedReactionType]) {
|
|
96
|
+
this.usersByReactions[this.selectedReactionType] = {
|
|
97
|
+
users: [],
|
|
98
|
+
};
|
|
99
|
+
await this.loadNextPageOfReactions();
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
async loadNextPageOfReactions() {
|
|
104
|
+
if (!this.messageId || !this.selectedReactionType) {
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
this.isLoading = true;
|
|
108
|
+
try {
|
|
109
|
+
const response = await this.messageReactionsService.queryReactions(this.messageId, this.selectedReactionType, this.usersByReactions[this.selectedReactionType].next);
|
|
110
|
+
this.usersByReactions[this.selectedReactionType].users = [
|
|
111
|
+
...this.usersByReactions[this.selectedReactionType].users,
|
|
112
|
+
...response.reactions
|
|
113
|
+
.map((r) => r.user)
|
|
114
|
+
.filter((u) => !!u),
|
|
115
|
+
];
|
|
116
|
+
this.usersByReactions[this.selectedReactionType].next = response.next;
|
|
117
|
+
}
|
|
118
|
+
catch (error) {
|
|
119
|
+
this.selectedReactionType = undefined;
|
|
120
|
+
}
|
|
121
|
+
finally {
|
|
122
|
+
this.isLoading = false;
|
|
123
|
+
}
|
|
124
|
+
if (this.isViewInited) {
|
|
125
|
+
this.cdRef.detectChanges();
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
isOwnReaction(reactionType) {
|
|
129
|
+
return !!this.ownReactions.find((r) => r.type === reactionType);
|
|
130
|
+
}
|
|
131
|
+
setExistingReactions() {
|
|
132
|
+
this.existingReactions = Object.keys(this.messageReactionGroups ?? {})
|
|
133
|
+
.filter((k) => this.reactionOptions.indexOf(k) !== -1)
|
|
134
|
+
.filter((k) => this.messageReactionGroups[k].count > 0)
|
|
135
|
+
.sort((r1, r2) => {
|
|
136
|
+
const date1 = this.messageReactionGroups[r1].first_reaction_at
|
|
137
|
+
? new Date(this.messageReactionGroups[r1].first_reaction_at)
|
|
138
|
+
: new Date();
|
|
139
|
+
const date2 = this.messageReactionGroups[r2].first_reaction_at
|
|
140
|
+
? new Date(this.messageReactionGroups[r2].first_reaction_at)
|
|
141
|
+
: new Date();
|
|
142
|
+
return date1.getTime() - date2.getTime();
|
|
143
|
+
});
|
|
144
|
+
}
|
|
145
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageReactionsComponent, deps: [{ token: i0.ChangeDetectorRef }, { token: i1.MessageReactionsService }, { token: i2.CustomTemplatesService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
146
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.0", type: MessageReactionsComponent, selector: "stream-message-reactions", inputs: { messageId: "messageId", messageReactionGroups: "messageReactionGroups", messageReactionCounts: "messageReactionCounts", latestReactions: "latestReactions", ownReactions: "ownReactions" }, viewQueries: [{ propertyName: "selectorContainer", first: true, predicate: ["selectorContainer"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "@if (existingReactions.length > 0) {\n <div\n data-testid=\"reaction-list\"\n class=\"str-chat__reaction-list str-chat__message-reactions-container\"\n [class.str-chat__reaction-list--reverse]=\"true\"\n >\n <ul class=\"str-chat__message-reactions\">\n @for (reactionType of existingReactions; track reactionType) {\n <li\n class=\"str-chat__message-reaction\"\n data-testclass=\"emoji\"\n [class.str-chat__message-reaction-own]=\"isOwnReaction(reactionType)\"\n (click)=\"reactionSelected(reactionType)\"\n (keyup.enter)=\"reactionSelected(reactionType)\"\n >\n <span class=\"emoji str-chat__message-reaction-emoji\">\n {{ getEmojiByReaction(reactionType) }} \n </span>\n <span\n data-testclass=\"reaction-list-reaction-count\"\n class=\"str-chat__message-reaction-count\"\n >\n {{ messageReactionGroups?.[reactionType]?.count ?? 0 }}\n </span>\n </li>\n }\n </ul>\n </div>\n}\n\n@if (selectedReactionType) {\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.modalTemplate$ | async) || defaultModal;\n context: {\n isOpen: !!selectedReactionType,\n messageId: messageId,\n reactionType: selectedReactionType,\n isOpenChangeHandler: isOpenChange,\n content: modalContent,\n }\n \"\n />\n}\n\n<ng-template\n #defaultModal\n let-isOpen=\"isOpen\"\n let-messageId=\"messageId\"\n let-reactionType=\"reactionType\"\n let-isOpenChangeHandler=\"isOpenChangeHandler\"\n let-content=\"content\"\n>\n <stream-modal\n class=\"str-chat__message-reactions-details-modal\"\n [isOpen]=\"isOpen\"\n [content]=\"content\"\n (isOpenChange)=\"isOpenChangeHandler($event)\"\n />\n</ng-template>\n\n<ng-template #modalContent>\n <div class=\"str-chat__message-reactions-details\">\n <div class=\"str-chat__message-reactions-details-reaction-types\">\n @for (reactionType of existingReactions; track reactionType) {\n <div\n class=\"str-chat__message-reactions-details-reaction-type\"\n attr.data-testid=\"reaction-details-selector-{{ reactionType }}\"\n [class.str-chat__message-reactions-details-reaction-type--selected]=\"\n reactionType === selectedReactionType\n \"\n (click)=\"reactionSelected(reactionType)\"\n (keyup.enter)=\"reactionSelected(reactionType)\"\n >\n <span class=\"emoji str-chat__message-reaction-emoji\">\n {{ getEmojiByReaction(reactionType) }} \n </span>\n <span class=\"str-chat__message-reaction-count\">\n {{ messageReactionGroups?.[reactionType]?.count ?? 0 }}\n </span>\n </div>\n }\n </div>\n <div\n class=\"emoji str-chat__message-reaction-emoji str-chat__message-reaction-emoji-big\"\n >\n {{ getEmojiByReaction(selectedReactionType!) }}\n </div>\n <div\n data-testid=\"all-reacting-users\"\n class=\"str-chat__message-reactions-details-reacting-users\"\n >\n @for (reactionType of existingReactions; track reactionType) {\n <stream-user-list\n attr.data-testid=\"{{ reactionType }}-user-list\"\n [style.display]=\"\n selectedReactionType === reactionType ? 'block' : 'none'\n \"\n [users]=\"usersByReactions[reactionType]?.users || []\"\n [isLoading]=\"isLoading\"\n [hasMore]=\"!!usersByReactions[reactionType]?.next || false\"\n (loadMore)=\"loadNextPageOfReactions()\"\n />\n }\n </div>\n </div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i4.ModalComponent, selector: "stream-modal", inputs: ["isOpen", "content"], outputs: ["isOpenChange"] }, { kind: "component", type: i5.UserListComponent, selector: "stream-user-list", inputs: ["users", "isLoading", "hasMore"], outputs: ["loadMore"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }] }); }
|
|
147
|
+
}
|
|
148
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageReactionsComponent, decorators: [{
|
|
149
|
+
type: Component,
|
|
150
|
+
args: [{ selector: 'stream-message-reactions', template: "@if (existingReactions.length > 0) {\n <div\n data-testid=\"reaction-list\"\n class=\"str-chat__reaction-list str-chat__message-reactions-container\"\n [class.str-chat__reaction-list--reverse]=\"true\"\n >\n <ul class=\"str-chat__message-reactions\">\n @for (reactionType of existingReactions; track reactionType) {\n <li\n class=\"str-chat__message-reaction\"\n data-testclass=\"emoji\"\n [class.str-chat__message-reaction-own]=\"isOwnReaction(reactionType)\"\n (click)=\"reactionSelected(reactionType)\"\n (keyup.enter)=\"reactionSelected(reactionType)\"\n >\n <span class=\"emoji str-chat__message-reaction-emoji\">\n {{ getEmojiByReaction(reactionType) }} \n </span>\n <span\n data-testclass=\"reaction-list-reaction-count\"\n class=\"str-chat__message-reaction-count\"\n >\n {{ messageReactionGroups?.[reactionType]?.count ?? 0 }}\n </span>\n </li>\n }\n </ul>\n </div>\n}\n\n@if (selectedReactionType) {\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.modalTemplate$ | async) || defaultModal;\n context: {\n isOpen: !!selectedReactionType,\n messageId: messageId,\n reactionType: selectedReactionType,\n isOpenChangeHandler: isOpenChange,\n content: modalContent,\n }\n \"\n />\n}\n\n<ng-template\n #defaultModal\n let-isOpen=\"isOpen\"\n let-messageId=\"messageId\"\n let-reactionType=\"reactionType\"\n let-isOpenChangeHandler=\"isOpenChangeHandler\"\n let-content=\"content\"\n>\n <stream-modal\n class=\"str-chat__message-reactions-details-modal\"\n [isOpen]=\"isOpen\"\n [content]=\"content\"\n (isOpenChange)=\"isOpenChangeHandler($event)\"\n />\n</ng-template>\n\n<ng-template #modalContent>\n <div class=\"str-chat__message-reactions-details\">\n <div class=\"str-chat__message-reactions-details-reaction-types\">\n @for (reactionType of existingReactions; track reactionType) {\n <div\n class=\"str-chat__message-reactions-details-reaction-type\"\n attr.data-testid=\"reaction-details-selector-{{ reactionType }}\"\n [class.str-chat__message-reactions-details-reaction-type--selected]=\"\n reactionType === selectedReactionType\n \"\n (click)=\"reactionSelected(reactionType)\"\n (keyup.enter)=\"reactionSelected(reactionType)\"\n >\n <span class=\"emoji str-chat__message-reaction-emoji\">\n {{ getEmojiByReaction(reactionType) }} \n </span>\n <span class=\"str-chat__message-reaction-count\">\n {{ messageReactionGroups?.[reactionType]?.count ?? 0 }}\n </span>\n </div>\n }\n </div>\n <div\n class=\"emoji str-chat__message-reaction-emoji str-chat__message-reaction-emoji-big\"\n >\n {{ getEmojiByReaction(selectedReactionType!) }}\n </div>\n <div\n data-testid=\"all-reacting-users\"\n class=\"str-chat__message-reactions-details-reacting-users\"\n >\n @for (reactionType of existingReactions; track reactionType) {\n <stream-user-list\n attr.data-testid=\"{{ reactionType }}-user-list\"\n [style.display]=\"\n selectedReactionType === reactionType ? 'block' : 'none'\n \"\n [users]=\"usersByReactions[reactionType]?.users || []\"\n [isLoading]=\"isLoading\"\n [hasMore]=\"!!usersByReactions[reactionType]?.next || false\"\n (loadMore)=\"loadNextPageOfReactions()\"\n />\n }\n </div>\n </div>\n</ng-template>\n" }]
|
|
151
|
+
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }, { type: i1.MessageReactionsService }, { type: i2.CustomTemplatesService }], propDecorators: { messageId: [{
|
|
152
|
+
type: Input
|
|
153
|
+
}], messageReactionGroups: [{
|
|
154
|
+
type: Input
|
|
155
|
+
}], messageReactionCounts: [{
|
|
156
|
+
type: Input
|
|
157
|
+
}], latestReactions: [{
|
|
158
|
+
type: Input
|
|
159
|
+
}], ownReactions: [{
|
|
160
|
+
type: Input
|
|
161
|
+
}], selectorContainer: [{
|
|
162
|
+
type: ViewChild,
|
|
163
|
+
args: ['selectorContainer']
|
|
164
|
+
}] } });
|
|
165
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-reactions.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport {\n  ReactionGroupResponse,\n  ReactionResponse,\n  UserResponse,\n} from 'stream-chat';\nimport { MessageReactionType, DefaultStreamChatGenerics } from '../types';\nimport { MessageReactionsService } from '../message-reactions.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { Subscription } from 'rxjs';\n\n/**\n * The `MessageReactions` component displays the reactions of a message. You can read more about [message reactions](/chat/docs/javascript/send_reaction/) in the platform documentation.\n */\n@Component({\n  selector: 'stream-message-reactions',\n  templateUrl: './message-reactions.component.html',\n})\nexport class MessageReactionsComponent\n  implements OnChanges, OnInit, AfterViewInit, OnDestroy\n{\n  /**\n   * The id of the message the reactions belong to\n   */\n  @Input() messageId: string | undefined;\n  /**\n   * The number of reactions grouped by [reaction types](https://github.com/GetStream/stream-chat-angular/tree/master/projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts)\n   */\n  @Input() messageReactionGroups:\n    | { [key: string]: ReactionGroupResponse }\n    | undefined = undefined;\n  /**\n   * The number of reactions grouped by [reaction types](https://github.com/GetStream/stream-chat-angular/tree/master/projects/stream-chat-angular/src/lib/message-reactions/message-reactions.component.ts)\n   * @deprecated use `messageReactionGroups`\n   */\n  @Input() messageReactionCounts: { [key in MessageReactionType]?: number } =\n    {};\n  /**\n   * List of reactions of a [message](/chat/docs/sdk/angular/types/stream-message/), used to display the users of a reaction type.\n   * @deprecated you can fetch the reactions using [`messageReactionsService.queryReactions()`](/chat/docs/sdk/angular/services/MessageReactionsService/#queryreactions)\n   */\n  @Input() latestReactions: ReactionResponse<DefaultStreamChatGenerics>[] = [];\n  /**\n   * List of the user's own reactions of a [message](/chat/docs/sdk/angular/types/stream-message/), used to display the users of a reaction type.\n   */\n  @Input() ownReactions: ReactionResponse<DefaultStreamChatGenerics>[] = [];\n  @ViewChild('selectorContainer') private selectorContainer:\n    | ElementRef<HTMLElement>\n    | undefined;\n  selectedReactionType: string | undefined;\n  isLoading = true;\n  reactions: ReactionResponse[] = [];\n  shouldHandleReactionClick = true;\n  existingReactions: string[] = [];\n  reactionOptions: string[] = [];\n  usersByReactions: {\n    [key: string]: { users: UserResponse[]; next?: string };\n  } = {};\n  private subscriptions: Subscription[] = [];\n  private isViewInited = false;\n\n  constructor(\n    private cdRef: ChangeDetectorRef,\n    private messageReactionsService: MessageReactionsService,\n    public customTemplatesService: CustomTemplatesService,\n  ) {}\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes.messageReactionCounts || changes.messageReactionGroups) {\n      if (this.messageReactionCounts && !this.messageReactionGroups) {\n        this.messageReactionGroups = {};\n        Object.keys(this.messageReactionCounts).forEach((k) => {\n          this.messageReactionGroups![k] = {\n            count: this.messageReactionCounts?.[k] ?? 0,\n            sum_scores: this.messageReactionCounts?.[k] ?? 0,\n            first_reaction_at: undefined,\n            last_reaction_at: undefined,\n          };\n        });\n      }\n      this.setExistingReactions();\n    }\n  }\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.messageReactionsService.reactions$.subscribe((reactions) => {\n        this.reactionOptions = Object.keys(reactions);\n        this.setExistingReactions();\n        if (this.isViewInited) {\n          this.cdRef.detectChanges();\n        }\n      }),\n    );\n  }\n\n  ngAfterViewInit(): void {\n    this.isViewInited = true;\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  getEmojiByReaction(reactionType: MessageReactionType) {\n    return this.messageReactionsService.reactions[reactionType];\n  }\n\n  async reactionSelected(reactionType: string) {\n    if (!this.messageId) {\n      return;\n    }\n    if (this.messageReactionsService.customReactionClickHandler) {\n      this.messageReactionsService.customReactionClickHandler({\n        messageId: this.messageId,\n        reactionType: reactionType,\n      });\n    } else {\n      this.selectedReactionType = reactionType;\n      if (!this.usersByReactions[this.selectedReactionType]) {\n        this.usersByReactions[this.selectedReactionType] = {\n          users: [],\n        };\n        await this.loadNextPageOfReactions();\n      }\n    }\n  }\n\n  async loadNextPageOfReactions() {\n    if (!this.messageId || !this.selectedReactionType) {\n      return;\n    }\n\n    this.isLoading = true;\n    try {\n      const response = await this.messageReactionsService.queryReactions(\n        this.messageId,\n        this.selectedReactionType,\n        this.usersByReactions[this.selectedReactionType].next,\n      );\n      this.usersByReactions[this.selectedReactionType].users = [\n        ...this.usersByReactions[this.selectedReactionType].users,\n        ...(response.reactions\n          .map((r) => r.user)\n          .filter((u) => !!u) as UserResponse[]),\n      ];\n      this.usersByReactions[this.selectedReactionType].next = response.next;\n    } catch (error) {\n      this.selectedReactionType = undefined;\n    } finally {\n      this.isLoading = false;\n    }\n    if (this.isViewInited) {\n      this.cdRef.detectChanges();\n    }\n  }\n\n  isOwnReaction(reactionType: MessageReactionType) {\n    return !!this.ownReactions.find((r) => r.type === reactionType);\n  }\n\n  isOpenChange = (isOpen: boolean) => {\n    this.selectedReactionType = isOpen ? this.selectedReactionType : undefined;\n    if (!isOpen) {\n      this.usersByReactions = {};\n    }\n  };\n\n  private setExistingReactions() {\n    this.existingReactions = Object.keys(this.messageReactionGroups ?? {})\n      .filter((k) => this.reactionOptions.indexOf(k) !== -1)\n      .filter((k) => this.messageReactionGroups![k].count > 0)\n      .sort((r1, r2) => {\n        const date1 = this.messageReactionGroups![r1].first_reaction_at\n          ? new Date(this.messageReactionGroups![r1].first_reaction_at!)\n          : new Date();\n        const date2 = this.messageReactionGroups![r2].first_reaction_at\n          ? new Date(this.messageReactionGroups![r2].first_reaction_at!)\n          : new Date();\n\n        return date1.getTime() - date2.getTime();\n      });\n  }\n}\n","@if (existingReactions.length > 0) {\n  <div\n    data-testid=\"reaction-list\"\n    class=\"str-chat__reaction-list str-chat__message-reactions-container\"\n    [class.str-chat__reaction-list--reverse]=\"true\"\n  >\n    <ul class=\"str-chat__message-reactions\">\n      @for (reactionType of existingReactions; track reactionType) {\n        <li\n          class=\"str-chat__message-reaction\"\n          data-testclass=\"emoji\"\n          [class.str-chat__message-reaction-own]=\"isOwnReaction(reactionType)\"\n          (click)=\"reactionSelected(reactionType)\"\n          (keyup.enter)=\"reactionSelected(reactionType)\"\n        >\n          <span class=\"emoji str-chat__message-reaction-emoji\">\n            {{ getEmojiByReaction(reactionType) }}&nbsp;\n          </span>\n          <span\n            data-testclass=\"reaction-list-reaction-count\"\n            class=\"str-chat__message-reaction-count\"\n          >\n            {{ messageReactionGroups?.[reactionType]?.count ?? 0 }}\n          </span>\n        </li>\n      }\n    </ul>\n  </div>\n}\n\n@if (selectedReactionType) {\n  <ng-container\n    *ngTemplateOutlet=\"\n      (customTemplatesService.modalTemplate$ | async) || defaultModal;\n      context: {\n        isOpen: !!selectedReactionType,\n        messageId: messageId,\n        reactionType: selectedReactionType,\n        isOpenChangeHandler: isOpenChange,\n        content: modalContent,\n      }\n    \"\n  />\n}\n\n<ng-template\n  #defaultModal\n  let-isOpen=\"isOpen\"\n  let-messageId=\"messageId\"\n  let-reactionType=\"reactionType\"\n  let-isOpenChangeHandler=\"isOpenChangeHandler\"\n  let-content=\"content\"\n>\n  <stream-modal\n    class=\"str-chat__message-reactions-details-modal\"\n    [isOpen]=\"isOpen\"\n    [content]=\"content\"\n    (isOpenChange)=\"isOpenChangeHandler($event)\"\n  />\n</ng-template>\n\n<ng-template #modalContent>\n  <div class=\"str-chat__message-reactions-details\">\n    <div class=\"str-chat__message-reactions-details-reaction-types\">\n      @for (reactionType of existingReactions; track reactionType) {\n        <div\n          class=\"str-chat__message-reactions-details-reaction-type\"\n          attr.data-testid=\"reaction-details-selector-{{ reactionType }}\"\n          [class.str-chat__message-reactions-details-reaction-type--selected]=\"\n            reactionType === selectedReactionType\n          \"\n          (click)=\"reactionSelected(reactionType)\"\n          (keyup.enter)=\"reactionSelected(reactionType)\"\n        >\n          <span class=\"emoji str-chat__message-reaction-emoji\">\n            {{ getEmojiByReaction(reactionType) }}&nbsp;\n          </span>\n          <span class=\"str-chat__message-reaction-count\">\n            {{ messageReactionGroups?.[reactionType]?.count ?? 0 }}\n          </span>\n        </div>\n      }\n    </div>\n    <div\n      class=\"emoji str-chat__message-reaction-emoji str-chat__message-reaction-emoji-big\"\n    >\n      {{ getEmojiByReaction(selectedReactionType!) }}\n    </div>\n    <div\n      data-testid=\"all-reacting-users\"\n      class=\"str-chat__message-reactions-details-reacting-users\"\n    >\n      @for (reactionType of existingReactions; track reactionType) {\n        <stream-user-list\n          attr.data-testid=\"{{ reactionType }}-user-list\"\n          [style.display]=\"\n            selectedReactionType === reactionType ? 'block' : 'none'\n          \"\n          [users]=\"usersByReactions[reactionType]?.users || []\"\n          [isLoading]=\"isLoading\"\n          [hasMore]=\"!!usersByReactions[reactionType]?.next || false\"\n          (loadMore)=\"loadNextPageOfReactions()\"\n        />\n      }\n    </div>\n  </div>\n</ng-template>\n"]}
|
|
@@ -0,0 +1,57 @@
|
|
|
1
|
+
import { Component, Input, } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
import * as i1 from "../channel.service";
|
|
4
|
+
import * as i2 from "../message-reactions.service";
|
|
5
|
+
/**
|
|
6
|
+
* The `MessageReactionsSelectorComponent` makes it possible for users to react to a message, the reaction options can be set using the [`MessageReactionsService`](/chat/docs/sdk/angular/services/MessageReactionsService/). You can read more about [message reactions](/chat/docs/javascript/send_reaction/) in the platform documentation.
|
|
7
|
+
*/
|
|
8
|
+
export class MessageReactionsSelectorComponent {
|
|
9
|
+
constructor(channelService, messageReactionsService, cdRef) {
|
|
10
|
+
this.channelService = channelService;
|
|
11
|
+
this.messageReactionsService = messageReactionsService;
|
|
12
|
+
this.cdRef = cdRef;
|
|
13
|
+
/**
|
|
14
|
+
* List of the user's own reactions of a [message](/chat/docs/sdk/angular/types/stream-message/), used to display the users of a reaction type.
|
|
15
|
+
*/
|
|
16
|
+
this.ownReactions = [];
|
|
17
|
+
this.reactionOptions = [];
|
|
18
|
+
this.subscriptions = [];
|
|
19
|
+
this.isViewInited = false;
|
|
20
|
+
}
|
|
21
|
+
ngOnInit() {
|
|
22
|
+
this.subscriptions.push(this.messageReactionsService.reactions$.subscribe((reactions) => {
|
|
23
|
+
this.reactionOptions = Object.keys(reactions);
|
|
24
|
+
if (this.isViewInited) {
|
|
25
|
+
this.cdRef.detectChanges();
|
|
26
|
+
}
|
|
27
|
+
}));
|
|
28
|
+
}
|
|
29
|
+
ngAfterViewInit() {
|
|
30
|
+
this.isViewInited = true;
|
|
31
|
+
}
|
|
32
|
+
ngOnDestroy() {
|
|
33
|
+
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
34
|
+
}
|
|
35
|
+
isOwnReaction(reactionType) {
|
|
36
|
+
return !!this.ownReactions.find((r) => r.type === reactionType);
|
|
37
|
+
}
|
|
38
|
+
getEmojiByReaction(reactionType) {
|
|
39
|
+
return this.messageReactionsService.reactions[reactionType];
|
|
40
|
+
}
|
|
41
|
+
async react(type) {
|
|
42
|
+
this.ownReactions.find((r) => r.type === type)
|
|
43
|
+
? await this.channelService.removeReaction(this.messageId, type)
|
|
44
|
+
: await this.channelService.addReaction(this.messageId, type);
|
|
45
|
+
}
|
|
46
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageReactionsSelectorComponent, deps: [{ token: i1.ChannelService }, { token: i2.MessageReactionsService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
47
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.0", type: MessageReactionsSelectorComponent, selector: "stream-message-reactions-selector", inputs: { ownReactions: "ownReactions", messageId: "messageId" }, ngImport: i0, template: "<div\n #selectorContainer\n data-testid=\"reaction-selector\"\n class=\"str-chat__reaction-selector str-chat-angular-v5__reaction-selector str-chat__message-reaction-selector\"\n>\n <ul\n class=\"str-chat__message-reactions-list str-chat__message-reactions-options\"\n >\n @for (reactionType of reactionOptions; track reactionType) {\n <li\n class=\"str-chat__message-reactions-option str-chat__message-reactions-list-item str-chat__emoji\"\n data-testclass=\"emoji-option\"\n [attr.data-testid]=\"reactionType\"\n [class.str-chat__message-reactions-option-selected]=\"\n isOwnReaction(reactionType)\n \"\n (click)=\"react(reactionType)\"\n (keyup.enter)=\"react(reactionType)\"\n >\n <span\n class=\"emoji str-chat__emoji-selector-emoji-angular str-chat__message-reaction-emoji\"\n >\n {{ getEmojiByReaction(reactionType) }}\n </span>\n </li>\n }\n </ul>\n</div>\n" }); }
|
|
48
|
+
}
|
|
49
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageReactionsSelectorComponent, decorators: [{
|
|
50
|
+
type: Component,
|
|
51
|
+
args: [{ selector: 'stream-message-reactions-selector', template: "<div\n #selectorContainer\n data-testid=\"reaction-selector\"\n class=\"str-chat__reaction-selector str-chat-angular-v5__reaction-selector str-chat__message-reaction-selector\"\n>\n <ul\n class=\"str-chat__message-reactions-list str-chat__message-reactions-options\"\n >\n @for (reactionType of reactionOptions; track reactionType) {\n <li\n class=\"str-chat__message-reactions-option str-chat__message-reactions-list-item str-chat__emoji\"\n data-testclass=\"emoji-option\"\n [attr.data-testid]=\"reactionType\"\n [class.str-chat__message-reactions-option-selected]=\"\n isOwnReaction(reactionType)\n \"\n (click)=\"react(reactionType)\"\n (keyup.enter)=\"react(reactionType)\"\n >\n <span\n class=\"emoji str-chat__emoji-selector-emoji-angular str-chat__message-reaction-emoji\"\n >\n {{ getEmojiByReaction(reactionType) }}\n </span>\n </li>\n }\n </ul>\n</div>\n" }]
|
|
52
|
+
}], ctorParameters: () => [{ type: i1.ChannelService }, { type: i2.MessageReactionsService }, { type: i0.ChangeDetectorRef }], propDecorators: { ownReactions: [{
|
|
53
|
+
type: Input
|
|
54
|
+
}], messageId: [{
|
|
55
|
+
type: Input
|
|
56
|
+
}] } });
|
|
57
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1yZWFjdGlvbnMtc2VsZWN0b3IuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2UtcmVhY3Rpb25zLXNlbGVjdG9yL21lc3NhZ2UtcmVhY3Rpb25zLXNlbGVjdG9yLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9tZXNzYWdlLXJlYWN0aW9ucy1zZWxlY3Rvci9tZXNzYWdlLXJlYWN0aW9ucy1zZWxlY3Rvci5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBR0wsU0FBUyxFQUNULEtBQUssR0FHTixNQUFNLGVBQWUsQ0FBQzs7OztBQU92Qjs7R0FFRztBQU1ILE1BQU0sT0FBTyxpQ0FBaUM7SUFlNUMsWUFDVSxjQUE4QixFQUM5Qix1QkFBZ0QsRUFDaEQsS0FBd0I7UUFGeEIsbUJBQWMsR0FBZCxjQUFjLENBQWdCO1FBQzlCLDRCQUF1QixHQUF2Qix1QkFBdUIsQ0FBeUI7UUFDaEQsVUFBSyxHQUFMLEtBQUssQ0FBbUI7UUFmbEM7O1dBRUc7UUFDTSxpQkFBWSxHQUFrRCxFQUFFLENBQUM7UUFLMUUsb0JBQWUsR0FBYSxFQUFFLENBQUM7UUFDdkIsa0JBQWEsR0FBbUIsRUFBRSxDQUFDO1FBQ25DLGlCQUFZLEdBQUcsS0FBSyxDQUFDO0lBTTFCLENBQUM7SUFDSixRQUFRO1FBQ04sSUFBSSxDQUFDLGFBQWEsQ0FBQyxJQUFJLENBQ3JCLElBQUksQ0FBQyx1QkFBdUIsQ0FBQyxVQUFVLENBQUMsU0FBUyxDQUFDLENBQUMsU0FBUyxFQUFFLEVBQUU7WUFDOUQsSUFBSSxDQUFDLGVBQWUsR0FBRyxNQUFNLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUFDO1lBQzlDLElBQUksSUFBSSxDQUFDLFlBQVksRUFBRTtnQkFDckIsSUFBSSxDQUFDLEtBQUssQ0FBQyxhQUFhLEVBQUUsQ0FBQzthQUM1QjtRQUNILENBQUMsQ0FBQyxDQUNILENBQUM7SUFDSixDQUFDO0lBRUQsZUFBZTtRQUNiLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDO0lBQzNCLENBQUM7SUFFRCxXQUFXO1FBQ1QsSUFBSSxDQUFDLGFBQWEsQ0FBQyxPQUFPLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxXQUFXLEVBQUUsQ0FBQyxDQUFDO0lBQ3JELENBQUM7SUFFRCxhQUFhLENBQUMsWUFBaUM7UUFDN0MsT0FBTyxDQUFDLENBQUMsSUFBSSxDQUFDLFlBQVksQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxJQUFJLEtBQUssWUFBWSxDQUFDLENBQUM7SUFDbEUsQ0FBQztJQUVELGtCQUFrQixDQUFDLFlBQWlDO1FBQ2xELE9BQU8sSUFBSSxDQUFDLHVCQUF1QixDQUFDLFNBQVMsQ0FBQyxZQUFZLENBQUMsQ0FBQztJQUM5RCxDQUFDO0lBRUQsS0FBSyxDQUFDLEtBQUssQ0FBQyxJQUF5QjtRQUNuQyxJQUFJLENBQUMsWUFBWSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLElBQUksS0FBSyxJQUFJLENBQUM7WUFDNUMsQ0FBQyxDQUFDLE1BQU0sSUFBSSxDQUFDLGNBQWMsQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLFNBQVUsRUFBRSxJQUFJLENBQUM7WUFDakUsQ0FBQyxDQUFDLE1BQU0sSUFBSSxDQUFDLGNBQWMsQ0FBQyxXQUFXLENBQUMsSUFBSSxDQUFDLFNBQVUsRUFBRSxJQUFJLENBQUMsQ0FBQztJQUNuRSxDQUFDOzhHQW5EVSxpQ0FBaUM7a0dBQWpDLGlDQUFpQywySUN0QjlDLGsrQkE0QkE7OzJGRE5hLGlDQUFpQztrQkFMN0MsU0FBUzsrQkFDRSxtQ0FBbUM7eUpBVXBDLFlBQVk7c0JBQXBCLEtBQUs7Z0JBSUcsU0FBUztzQkFBakIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7XG4gIEFmdGVyVmlld0luaXQsXG4gIENoYW5nZURldGVjdG9yUmVmLFxuICBDb21wb25lbnQsXG4gIElucHV0LFxuICBPbkRlc3Ryb3ksXG4gIE9uSW5pdCxcbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBEZWZhdWx0U3RyZWFtQ2hhdEdlbmVyaWNzLCBNZXNzYWdlUmVhY3Rpb25UeXBlIH0gZnJvbSAnLi4vdHlwZXMnO1xuaW1wb3J0IHsgUmVhY3Rpb25SZXNwb25zZSB9IGZyb20gJ3N0cmVhbS1jaGF0JztcbmltcG9ydCB7IENoYW5uZWxTZXJ2aWNlIH0gZnJvbSAnLi4vY2hhbm5lbC5zZXJ2aWNlJztcbmltcG9ydCB7IE1lc3NhZ2VSZWFjdGlvbnNTZXJ2aWNlIH0gZnJvbSAnLi4vbWVzc2FnZS1yZWFjdGlvbnMuc2VydmljZSc7XG5pbXBvcnQgeyBTdWJzY3JpcHRpb24gfSBmcm9tICdyeGpzJztcblxuLyoqXG4gKiBUaGUgYE1lc3NhZ2VSZWFjdGlvbnNTZWxlY3RvckNvbXBvbmVudGAgbWFrZXMgaXQgcG9zc2libGUgZm9yIHVzZXJzIHRvIHJlYWN0IHRvIGEgbWVzc2FnZSwgdGhlIHJlYWN0aW9uIG9wdGlvbnMgY2FuIGJlIHNldCB1c2luZyB0aGUgW2BNZXNzYWdlUmVhY3Rpb25zU2VydmljZWBdKC9jaGF0L2RvY3Mvc2RrL2FuZ3VsYXIvc2VydmljZXMvTWVzc2FnZVJlYWN0aW9uc1NlcnZpY2UvKS4gWW91IGNhbiByZWFkIG1vcmUgYWJvdXQgW21lc3NhZ2UgcmVhY3Rpb25zXSgvY2hhdC9kb2NzL2phdmFzY3JpcHQvc2VuZF9yZWFjdGlvbi8pIGluIHRoZSBwbGF0Zm9ybSBkb2N1bWVudGF0aW9uLlxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdzdHJlYW0tbWVzc2FnZS1yZWFjdGlvbnMtc2VsZWN0b3InLFxuICB0ZW1wbGF0ZVVybDogJy4vbWVzc2FnZS1yZWFjdGlvbnMtc2VsZWN0b3IuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZXM6IFtdLFxufSlcbmV4cG9ydCBjbGFzcyBNZXNzYWdlUmVhY3Rpb25zU2VsZWN0b3JDb21wb25lbnRcbiAgaW1wbGVtZW50cyBPbkluaXQsIE9uRGVzdHJveSwgQWZ0ZXJWaWV3SW5pdFxue1xuICAvKipcbiAgICogTGlzdCBvZiB0aGUgdXNlcidzIG93biByZWFjdGlvbnMgb2YgYSBbbWVzc2FnZV0oL2NoYXQvZG9jcy9zZGsvYW5ndWxhci90eXBlcy9zdHJlYW0tbWVzc2FnZS8pLCB1c2VkIHRvIGRpc3BsYXkgdGhlIHVzZXJzIG9mIGEgcmVhY3Rpb24gdHlwZS5cbiAgICovXG4gIEBJbnB1dCgpIG93blJlYWN0aW9uczogUmVhY3Rpb25SZXNwb25zZTxEZWZhdWx0U3RyZWFtQ2hhdEdlbmVyaWNzPltdID0gW107XG4gIC8qKlxuICAgKiBUaGUgaWQgb2YgdGhlIG1lc3NhZ2UgdGhlIHJlYWN0aW9ucyBiZWxvbmcgdG9cbiAgICovXG4gIEBJbnB1dCgpIG1lc3NhZ2VJZDogc3RyaW5nIHwgdW5kZWZpbmVkO1xuICByZWFjdGlvbk9wdGlvbnM6IHN0cmluZ1tdID0gW107XG4gIHByaXZhdGUgc3Vic2NyaXB0aW9uczogU3Vic2NyaXB0aW9uW10gPSBbXTtcbiAgcHJpdmF0ZSBpc1ZpZXdJbml0ZWQgPSBmYWxzZTtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIGNoYW5uZWxTZXJ2aWNlOiBDaGFubmVsU2VydmljZSxcbiAgICBwcml2YXRlIG1lc3NhZ2VSZWFjdGlvbnNTZXJ2aWNlOiBNZXNzYWdlUmVhY3Rpb25zU2VydmljZSxcbiAgICBwcml2YXRlIGNkUmVmOiBDaGFuZ2VEZXRlY3RvclJlZixcbiAgKSB7fVxuICBuZ09uSW5pdCgpOiB2b2lkIHtcbiAgICB0aGlzLnN1YnNjcmlwdGlvbnMucHVzaChcbiAgICAgIHRoaXMubWVzc2FnZVJlYWN0aW9uc1NlcnZpY2UucmVhY3Rpb25zJC5zdWJzY3JpYmUoKHJlYWN0aW9ucykgPT4ge1xuICAgICAgICB0aGlzLnJlYWN0aW9uT3B0aW9ucyA9IE9iamVjdC5rZXlzKHJlYWN0aW9ucyk7XG4gICAgICAgIGlmICh0aGlzLmlzVmlld0luaXRlZCkge1xuICAgICAgICAgIHRoaXMuY2RSZWYuZGV0ZWN0Q2hhbmdlcygpO1xuICAgICAgICB9XG4gICAgICB9KSxcbiAgICApO1xuICB9XG5cbiAgbmdBZnRlclZpZXdJbml0KCk6IHZvaWQge1xuICAgIHRoaXMuaXNWaWV3SW5pdGVkID0gdHJ1ZTtcbiAgfVxuXG4gIG5nT25EZXN0cm95KCk6IHZvaWQge1xuICAgIHRoaXMuc3Vic2NyaXB0aW9ucy5mb3JFYWNoKChzKSA9PiBzLnVuc3Vic2NyaWJlKCkpO1xuICB9XG5cbiAgaXNPd25SZWFjdGlvbihyZWFjdGlvblR5cGU6IE1lc3NhZ2VSZWFjdGlvblR5cGUpIHtcbiAgICByZXR1cm4gISF0aGlzLm93blJlYWN0aW9ucy5maW5kKChyKSA9PiByLnR5cGUgPT09IHJlYWN0aW9uVHlwZSk7XG4gIH1cblxuICBnZXRFbW9qaUJ5UmVhY3Rpb24ocmVhY3Rpb25UeXBlOiBNZXNzYWdlUmVhY3Rpb25UeXBlKSB7XG4gICAgcmV0dXJuIHRoaXMubWVzc2FnZVJlYWN0aW9uc1NlcnZpY2UucmVhY3Rpb25zW3JlYWN0aW9uVHlwZV07XG4gIH1cblxuICBhc3luYyByZWFjdCh0eXBlOiBNZXNzYWdlUmVhY3Rpb25UeXBlKSB7XG4gICAgdGhpcy5vd25SZWFjdGlvbnMuZmluZCgocikgPT4gci50eXBlID09PSB0eXBlKVxuICAgICAgPyBhd2FpdCB0aGlzLmNoYW5uZWxTZXJ2aWNlLnJlbW92ZVJlYWN0aW9uKHRoaXMubWVzc2FnZUlkISwgdHlwZSlcbiAgICAgIDogYXdhaXQgdGhpcy5jaGFubmVsU2VydmljZS5hZGRSZWFjdGlvbih0aGlzLm1lc3NhZ2VJZCEsIHR5cGUpO1xuICB9XG59XG4iLCI8ZGl2XG4gICNzZWxlY3RvckNvbnRhaW5lclxuICBkYXRhLXRlc3RpZD1cInJlYWN0aW9uLXNlbGVjdG9yXCJcbiAgY2xhc3M9XCJzdHItY2hhdF9fcmVhY3Rpb24tc2VsZWN0b3Igc3RyLWNoYXQtYW5ndWxhci12NV9fcmVhY3Rpb24tc2VsZWN0b3Igc3RyLWNoYXRfX21lc3NhZ2UtcmVhY3Rpb24tc2VsZWN0b3JcIlxuPlxuICA8dWxcbiAgICBjbGFzcz1cInN0ci1jaGF0X19tZXNzYWdlLXJlYWN0aW9ucy1saXN0IHN0ci1jaGF0X19tZXNzYWdlLXJlYWN0aW9ucy1vcHRpb25zXCJcbiAgPlxuICAgIEBmb3IgKHJlYWN0aW9uVHlwZSBvZiByZWFjdGlvbk9wdGlvbnM7IHRyYWNrIHJlYWN0aW9uVHlwZSkge1xuICAgICAgPGxpXG4gICAgICAgIGNsYXNzPVwic3RyLWNoYXRfX21lc3NhZ2UtcmVhY3Rpb25zLW9wdGlvbiBzdHItY2hhdF9fbWVzc2FnZS1yZWFjdGlvbnMtbGlzdC1pdGVtIHN0ci1jaGF0X19lbW9qaVwiXG4gICAgICAgIGRhdGEtdGVzdGNsYXNzPVwiZW1vamktb3B0aW9uXCJcbiAgICAgICAgW2F0dHIuZGF0YS10ZXN0aWRdPVwicmVhY3Rpb25UeXBlXCJcbiAgICAgICAgW2NsYXNzLnN0ci1jaGF0X19tZXNzYWdlLXJlYWN0aW9ucy1vcHRpb24tc2VsZWN0ZWRdPVwiXG4gICAgICAgICAgaXNPd25SZWFjdGlvbihyZWFjdGlvblR5cGUpXG4gICAgICAgIFwiXG4gICAgICAgIChjbGljayk9XCJyZWFjdChyZWFjdGlvblR5cGUpXCJcbiAgICAgICAgKGtleXVwLmVudGVyKT1cInJlYWN0KHJlYWN0aW9uVHlwZSlcIlxuICAgICAgPlxuICAgICAgICA8c3BhblxuICAgICAgICAgIGNsYXNzPVwiZW1vamkgc3RyLWNoYXRfX2Vtb2ppLXNlbGVjdG9yLWVtb2ppLWFuZ3VsYXIgc3RyLWNoYXRfX21lc3NhZ2UtcmVhY3Rpb24tZW1vamlcIlxuICAgICAgICA+XG4gICAgICAgICAge3sgZ2V0RW1vamlCeVJlYWN0aW9uKHJlYWN0aW9uVHlwZSkgfX1cbiAgICAgICAgPC9zcGFuPlxuICAgICAgPC9saT5cbiAgICB9XG4gIDwvdWw+XG48L2Rpdj5cbiJdfQ==
|
|
@@ -58,13 +58,13 @@ export class MessageReactionsService {
|
|
|
58
58
|
}
|
|
59
59
|
}
|
|
60
60
|
}
|
|
61
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageReactionsService, deps: [{ token: i1.ChatClientService }, { token: i2.NotificationService }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
62
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageReactionsService, providedIn: 'root' }); }
|
|
61
63
|
}
|
|
62
|
-
|
|
63
|
-
MessageReactionsService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageReactionsService, providedIn: 'root' });
|
|
64
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageReactionsService, decorators: [{
|
|
64
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageReactionsService, decorators: [{
|
|
65
65
|
type: Injectable,
|
|
66
66
|
args: [{
|
|
67
67
|
providedIn: 'root',
|
|
68
68
|
}]
|
|
69
|
-
}], ctorParameters:
|
|
70
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
69
|
+
}], ctorParameters: () => [{ type: i1.ChatClientService }, { type: i2.NotificationService }] });
|
|
70
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS1yZWFjdGlvbnMuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9tZXNzYWdlLXJlYWN0aW9ucy5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFM0MsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLE1BQU0sQ0FBQzs7OztBQUl2Qzs7O0dBR0c7QUFJSCxNQUFNLE9BQU8sdUJBQXVCO0lBb0JsQyxZQUNVLGlCQUFvQyxFQUNwQyxtQkFBd0M7UUFEeEMsc0JBQWlCLEdBQWpCLGlCQUFpQixDQUFtQjtRQUNwQyx3QkFBbUIsR0FBbkIsbUJBQW1CLENBQXFCO1FBckJsRDs7OztXQUlHO1FBQ0gsZUFBVSxHQUFHLElBQUksZUFBZSxDQUEyQztZQUN6RSxJQUFJLEVBQUUsSUFBSTtZQUNWLElBQUksRUFBRSxJQUFJO1lBQ1YsSUFBSSxFQUFFLElBQUk7WUFDVixHQUFHLEVBQUUsSUFBSTtZQUNULEdBQUcsRUFBRSxJQUFJO1NBQ1YsQ0FBQyxDQUFDO0lBV0EsQ0FBQztJQUVKOztPQUVHO0lBQ0gsSUFBSSxTQUFTLENBQUMsU0FBbUQ7UUFDL0QsSUFBSSxDQUFDLFVBQVUsQ0FBQyxJQUFJLENBQUMsU0FBUyxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQUVEOztPQUVHO0lBQ0gsSUFBSSxTQUFTO1FBQ1gsT0FBTyxJQUFJLENBQUMsVUFBVSxDQUFDLFFBQVEsRUFBRSxDQUFDO0lBQ3BDLENBQUM7SUFFRDs7Ozs7O09BTUc7SUFDSCxLQUFLLENBQUMsY0FBYyxDQUFDLFNBQWlCLEVBQUUsSUFBWSxFQUFFLElBQWE7UUFDakUsSUFBSSxDQUFDLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxVQUFVLEVBQUU7WUFDdEMsTUFBTSxJQUFJLEtBQUssQ0FDYiwyREFBMkQsQ0FDNUQsQ0FBQztTQUNIO2FBQU07WUFDTCxJQUFJO2dCQUNGLE1BQU0sUUFBUSxHQUFHLE1BQU0sSUFBSSxDQUFDLGlCQUFpQixDQUFDLFVBQVUsQ0FBQyxjQUFjLENBQ3JFLFNBQVMsRUFDVCxFQUFFLElBQUksRUFBRSxFQUNSLEVBQUUsVUFBVSxFQUFFLENBQUMsQ0FBQyxFQUFFLEVBQ2xCLEVBQUUsSUFBSSxFQUFFLENBQ1QsQ0FBQztnQkFFRixPQUFPLFFBQVEsQ0FBQzthQUNqQjtZQUFDLE9BQU8sS0FBSyxFQUFFO2dCQUNkLElBQUksQ0FBQyxtQkFBbUIsQ0FBQyx3QkFBd0IsQ0FDL0Msb0NBQW9DLENBQ3JDLENBQUM7Z0JBQ0YsTUFBTSxLQUFLLENBQUM7YUFDYjtTQUNGO0lBQ0gsQ0FBQzs4R0FwRVUsdUJBQXVCO2tIQUF2Qix1QkFBdUIsY0FGdEIsTUFBTTs7MkZBRVAsdUJBQXVCO2tCQUhuQyxVQUFVO21CQUFDO29CQUNWLFVBQVUsRUFBRSxNQUFNO2lCQUNuQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IE1lc3NhZ2VSZWFjdGlvbkNsaWNrRGV0YWlscywgTWVzc2FnZVJlYWN0aW9uVHlwZSB9IGZyb20gJy4vdHlwZXMnO1xuaW1wb3J0IHsgQmVoYXZpb3JTdWJqZWN0IH0gZnJvbSAncnhqcyc7XG5pbXBvcnQgeyBDaGF0Q2xpZW50U2VydmljZSB9IGZyb20gJy4vY2hhdC1jbGllbnQuc2VydmljZSc7XG5pbXBvcnQgeyBOb3RpZmljYXRpb25TZXJ2aWNlIH0gZnJvbSAnLi9ub3RpZmljYXRpb24uc2VydmljZSc7XG5cbi8qKlxuICogVGhlIGBNZXNzYWdlUmVhY3Rpb25zU2VydmljZWAgcHJvdmlkZXMgY3VzdG9taXphdGlvbiBvcHRpb25zIHRvIG1lc3NhZ2UgW3JlYWN0aW9uc10oL2NoYXQvZG9jcy9qYXZhc2NyaXB0L3NlbmRfcmVhY3Rpb24vKS5cbiAqXG4gKi9cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnLFxufSlcbmV4cG9ydCBjbGFzcyBNZXNzYWdlUmVhY3Rpb25zU2VydmljZSB7XG4gIC8qKlxuICAgKiBUaGUgZW5hYmxlZCBbcmVhY3Rpb25zXSgvY2hhdC9kb2NzL2phdmFzY3JpcHQvc2VuZF9yZWFjdGlvbi8pIGFuZCB0aGUgYXNzb2NpYXRlZCBlbW9qaVxuICAgKlxuICAgKiBZb3UgY2FuIHByb3ZpZGUgYW55IHN0cmluZyBhcyBhIHJlYWN0aW9uLiBUaGUgZW1vamkgY2FuIGJlIHByb3ZpZGVkIGFzIGEgc3RyaW5nLCBpZiB5b3Ugd2FudCB0byB1c2UgY3VzdG9tIGltYWdlcyBmb3IgcmVhY3Rpb25zIHlvdSBoYXZlIHRvIHByb3ZpZGUgYSBbY3VzdG9tIHJlYWN0aW9ucyBVSV0oL2NoYXQvZG9jcy9zZGsvYW5ndWxhci9zZXJ2aWNlcy9DdXN0b21UZW1wbGF0ZXNTZXJ2aWNlLyNtZXNzYWdlcmVhY3Rpb25zdGVtcGxhdGUvKVxuICAgKi9cbiAgcmVhY3Rpb25zJCA9IG5ldyBCZWhhdmlvclN1YmplY3Q8eyBba2V5IGluIE1lc3NhZ2VSZWFjdGlvblR5cGVdOiBzdHJpbmcgfT4oe1xuICAgIGhhaGE6ICfwn5iCJyxcbiAgICBsaWtlOiAn8J+RjScsXG4gICAgbG92ZTogJ+KdpO+4jycsXG4gICAgc2FkOiAn8J+YnicsXG4gICAgd293OiAn8J+YricsXG4gIH0pO1xuICAvKipcbiAgICogQnkgZGVmYXVsdCB0aGUgW2BNZXNzYWdlUmVhY3Rpb25zQ29tcG9uZW50YF0oL2NoYXQvZG9jcy9zZGsvYW5ndWxhci9jb21wb25lbnRzL01lc3NhZ2VSZWFjdGlvbnNDb21wb25lbnQvKSB3aWxsIGRpc3BsYXkgdGhlIHJlYWN0aW5nIHVzZXJzIHdoZW4gYSByZWFjdGlvbiBpcyBjbGlja2VkLiBZb3UgY2FuIG92ZXJyaWRlIHRoaXMgd2l0aCB5b3VyIG93biBVSSBieSBwcm92aWRpbmcgYSBjdXN0b20gZXZlbnQgaGFuZGxlci5cbiAgICpcbiAgICogVGhlIGV2ZW50IGhhbmRsZXIgY2FuIHJldHJpZXZlIGFsbCByZWFjdGlvbnMgb2YgYSBtZXNzYWdlIHVzaW5nIHRoZSBbYG1lc3NhZ2VSZWFjdGlvbnNTZXJ2aWNlLnF1ZXJ5UmVhY3Rpb25zKClgXSgvY2hhdC9kb2NzL3Nkay9hbmd1bGFyL3NlcnZpY2VzL01lc3NhZ2VSZWFjdGlvbnNTZXJ2aWNlLyNxdWVyeXJlYWN0aW9ucylcbiAgICovXG4gIGN1c3RvbVJlYWN0aW9uQ2xpY2tIYW5kbGVyPzogKGRldGFpbHM6IE1lc3NhZ2VSZWFjdGlvbkNsaWNrRGV0YWlscykgPT4gdm9pZDtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwcml2YXRlIGNoYXRDbGllbnRTZXJ2aWNlOiBDaGF0Q2xpZW50U2VydmljZSxcbiAgICBwcml2YXRlIG5vdGlmaWNhdGlvblNlcnZpY2U6IE5vdGlmaWNhdGlvblNlcnZpY2UsXG4gICkge31cblxuICAvKipcbiAgICogU2V0cyB0aGUgZW5hYmxlZCByZWFjdGlvbnNcbiAgICovXG4gIHNldCByZWFjdGlvbnMocmVhY3Rpb25zOiB7IFtrZXkgaW4gTWVzc2FnZVJlYWN0aW9uVHlwZV06IHN0cmluZyB9KSB7XG4gICAgdGhpcy5yZWFjdGlvbnMkLm5leHQocmVhY3Rpb25zKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZXQgdGhlIGN1cnJlbnRseSBlbmFibGVkIHJlYWN0aW9uc1xuICAgKi9cbiAgZ2V0IHJlYWN0aW9ucygpIHtcbiAgICByZXR1cm4gdGhpcy5yZWFjdGlvbnMkLmdldFZhbHVlKCk7XG4gIH1cblxuICAvKipcbiAgICogUXVlcnkgcmVhY3Rpb25zIG9mIGEgc3BlY2lmaWMgbWVzc2FnZSwgbW9yZSBpbmZvIGluIHRoZSBbQVBJIGRvY3VtZW50YXRpb25dKC9jaGF0L2RvY3MvamF2YXNjcmlwdC9zZW5kX3JlYWN0aW9uLyNxdWVyeS1yZWFjdGlvbnMpXG4gICAqIEBwYXJhbSBtZXNzYWdlSWRcbiAgICogQHBhcmFtIHR5cGVcbiAgICogQHBhcmFtIG5leHRcbiAgICogQHJldHVybnMgdGhlIHJlYWN0aW9ucyBhbmQgdGhlIGN1cnNvciBmb3IgdGhlIG5leHQvcHJldiBwYWdlc1xuICAgKi9cbiAgYXN5bmMgcXVlcnlSZWFjdGlvbnMobWVzc2FnZUlkOiBzdHJpbmcsIHR5cGU6IHN0cmluZywgbmV4dD86IHN0cmluZykge1xuICAgIGlmICghdGhpcy5jaGF0Q2xpZW50U2VydmljZS5jaGF0Q2xpZW50KSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoXG4gICAgICAgICdJbnRpYWxpemUgdGhlIENoYXRDbGllbnRTZXJ2aWNlIGJlZm9yZSBxdWVyeWluZyByZWFjdGlvbnMnLFxuICAgICAgKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdHJ5IHtcbiAgICAgICAgY29uc3QgcmVzcG9uc2UgPSBhd2FpdCB0aGlzLmNoYXRDbGllbnRTZXJ2aWNlLmNoYXRDbGllbnQucXVlcnlSZWFjdGlvbnMoXG4gICAgICAgICAgbWVzc2FnZUlkLFxuICAgICAgICAgIHsgdHlwZSB9LFxuICAgICAgICAgIHsgY3JlYXRlZF9hdDogLTEgfSxcbiAgICAgICAgICB7IG5leHQgfSxcbiAgICAgICAgKTtcblxuICAgICAgICByZXR1cm4gcmVzcG9uc2U7XG4gICAgICB9IGNhdGNoIChlcnJvcikge1xuICAgICAgICB0aGlzLm5vdGlmaWNhdGlvblNlcnZpY2UuYWRkVGVtcG9yYXJ5Tm90aWZpY2F0aW9uKFxuICAgICAgICAgICdzdHJlYW1DaGF0LkVycm9yIGxvYWRpbmcgcmVhY3Rpb25zJyxcbiAgICAgICAgKTtcbiAgICAgICAgdGhyb3cgZXJyb3I7XG4gICAgICB9XG4gICAgfVxuICB9XG59XG4iXX0=
|
|
@@ -127,17 +127,17 @@ export class MessageTextComponent {
|
|
|
127
127
|
});
|
|
128
128
|
return content;
|
|
129
129
|
}
|
|
130
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageTextComponent, deps: [{ token: i1.MessageService }, { token: i2.CustomTemplatesService }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
131
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.0", type: MessageTextComponent, selector: "stream-message-text", inputs: { message: "message", isQuoted: "isQuoted", shouldTranslate: "shouldTranslate" }, usesOnChanges: true, ngImport: i0, template: "@if (message?.text) {\n <p\n [class.str-chat__quoted-message-text-value]=\"isQuoted\"\n [class.str-chat__message-text-value]=\"!isQuoted\"\n data-testid=\"text\"\n >\n @if (messageTextParts) {\n @for (part of messageTextParts; track part.content) {\n @if (part.type === \"text\") {\n <span [innerHTML]=\"part.content\"></span>\n } @else {\n <ng-template #defaultMention let-content=\"content\">\n <span class=\"str-chat__message-mention\">{{ content }}</span>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.mentionTemplate$ | async) ||\n defaultMention;\n context: getMentionContext(part)\n \"\n />\n }\n }\n } @else {\n @if (displayAs === \"text\") {\n {{ messageText || \"\" }}\n } @else {\n <span data-testid=\"html-content\" [innerHTML]=\"messageText\"></span>\n }\n }\n </p>\n}\n", dependencies: [{ kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }] }); }
|
|
130
132
|
}
|
|
131
|
-
|
|
132
|
-
MessageTextComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: MessageTextComponent, selector: "stream-message-text", inputs: { message: "message", isQuoted: "isQuoted", shouldTranslate: "shouldTranslate" }, usesOnChanges: true, ngImport: i0, template: "<p\n [class.str-chat__quoted-message-text-value]=\"isQuoted\"\n [class.str-chat__message-text-value]=\"!isQuoted\"\n *ngIf=\"message?.text\"\n data-testid=\"text\"\n>\n <ng-container *ngIf=\"messageTextParts; else defaultContent\">\n <!-- eslint-disable-next-line @angular-eslint/template/use-track-by-function -->\n <ng-container *ngFor=\"let part of messageTextParts\">\n <span\n *ngIf=\"part.type === 'text'; else mention\"\n [innerHTML]=\"part.content\"\n ></span>\n <ng-template #mention>\n <ng-template #defaultMention let-content=\"content\">\n <span class=\"str-chat__message-mention\">{{ content }}</span>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.mentionTemplate$ | async) || defaultMention;\n context: getMentionContext(part)\n \"\n ></ng-container>\n </ng-template>\n </ng-container>\n </ng-container>\n <ng-template #defaultContent>\n <ng-container *ngIf=\"displayAs === 'text'; else asHTML\">\n {{ messageText || \"\" }}\n </ng-container>\n <ng-template #asHTML\n ><span data-testid=\"html-content\" [innerHTML]=\"messageText\"></span\n ></ng-template>\n </ng-template>\n</p>\n", dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }] });
|
|
133
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageTextComponent, decorators: [{
|
|
133
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageTextComponent, decorators: [{
|
|
134
134
|
type: Component,
|
|
135
|
-
args: [{ selector: 'stream-message-text', template: "<p\n
|
|
136
|
-
}], ctorParameters:
|
|
135
|
+
args: [{ selector: 'stream-message-text', template: "@if (message?.text) {\n <p\n [class.str-chat__quoted-message-text-value]=\"isQuoted\"\n [class.str-chat__message-text-value]=\"!isQuoted\"\n data-testid=\"text\"\n >\n @if (messageTextParts) {\n @for (part of messageTextParts; track part.content) {\n @if (part.type === \"text\") {\n <span [innerHTML]=\"part.content\"></span>\n } @else {\n <ng-template #defaultMention let-content=\"content\">\n <span class=\"str-chat__message-mention\">{{ content }}</span>\n </ng-template>\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.mentionTemplate$ | async) ||\n defaultMention;\n context: getMentionContext(part)\n \"\n />\n }\n }\n } @else {\n @if (displayAs === \"text\") {\n {{ messageText || \"\" }}\n } @else {\n <span data-testid=\"html-content\" [innerHTML]=\"messageText\"></span>\n }\n }\n </p>\n}\n" }]
|
|
136
|
+
}], ctorParameters: () => [{ type: i1.MessageService }, { type: i2.CustomTemplatesService }], propDecorators: { message: [{
|
|
137
137
|
type: Input
|
|
138
138
|
}], isQuoted: [{
|
|
139
139
|
type: Input
|
|
140
140
|
}], shouldTranslate: [{
|
|
141
141
|
type: Input
|
|
142
142
|
}] } });
|
|
143
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-text.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-text/message-text.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-text/message-text.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport {\n  DefaultStreamChatGenerics,\n  MentionTemplateContext,\n  StreamMessage,\n} from '../types';\nimport { MessageResponseBase, UserResponse } from 'stream-chat';\nimport emojiRegex from 'emoji-regex';\nimport { MessageService } from '../message.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\n\ntype MessagePart = {\n  content: string;\n  type: 'text' | 'mention';\n  user?: UserResponse;\n};\n\n/**\n * The `MessageTextComponent` displays the text content of a message.\n */\n@Component({\n  selector: 'stream-message-text',\n  templateUrl: './message-text.component.html',\n  styles: [],\n})\nexport class MessageTextComponent implements OnChanges {\n  /**\n   * The message which text should be displayed\n   */\n  @Input() message:\n    | StreamMessage<DefaultStreamChatGenerics>\n    | undefined\n    | MessageResponseBase<DefaultStreamChatGenerics>;\n  /**\n   * `true` if the component displayes a message quote\n   */\n  @Input() isQuoted: boolean = false;\n  /**\n   * `true` if the\n   */\n  @Input() shouldTranslate: boolean = false;\n  messageTextParts: MessagePart[] | undefined = [];\n  messageText?: string;\n  displayAs: 'text' | 'html';\n  private readonly urlRegexp: RegExp;\n  private emojiRegexp = new RegExp(emojiRegex(), 'g');\n\n  constructor(\n    private messageService: MessageService,\n    readonly customTemplatesService: CustomTemplatesService\n  ) {\n    this.displayAs = this.messageService.displayAs;\n    try {\n      this.urlRegexp = new RegExp(\n        '(?:(?:https?|ftp|file):\\\\/\\\\/|www\\\\.|ftp\\\\.|(?:[a-zA-Z0-9-]+\\\\.)+[a-zA-Z]{2,})(?![^\\\\s]*@[^\\\\s]*)(?:[^\\\\s()<>]+|\\\\([\\\\w\\\\d]+\\\\))*(?<!@[a-zA-Z0-9.-]+\\\\.[a-zA-Z]{2,})',\n        'gim'\n      );\n    } catch {\n      this.urlRegexp =\n        /(?:(?:https?|ftp|file):\\/\\/|www\\.|ftp\\.)(?:\\([-A-Z0-9+&@#/%=~_|$?!:,.]*\\)|[-A-Z0-9+&@#/%=~_|$?!:,.])*(?:\\([-A-Z0-9+&@#/%=~_|$?!:,.]*\\)|[A-Z0-9+&@#/%=~_|$])/gim;\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message || changes.shouldTranslate) {\n      this.createMessageParts();\n    }\n  }\n\n  getMentionContext(messagePart: MessagePart): MentionTemplateContext {\n    return {\n      content: messagePart.content,\n      user: messagePart.user!,\n    };\n  }\n\n  private createMessageParts() {\n    this.messageTextParts = undefined;\n    this.messageText = undefined;\n    let content = this.getMessageContent();\n    if (\n      (!this.message!.mentioned_users ||\n        this.message!.mentioned_users.length === 0) &&\n      !content?.match(this.emojiRegexp) &&\n      !content?.match(this.urlRegexp)\n    ) {\n      this.messageTextParts = undefined;\n      this.messageText = content;\n      return;\n    }\n    if (!content) {\n      return;\n    }\n    if (\n      !this.message!.mentioned_users ||\n      this.message!.mentioned_users.length === 0\n    ) {\n      content = this.fixEmojiDisplay(content);\n      content = this.wrapLinksWithAnchorTag(content);\n      this.messageTextParts = [{ content, type: 'text' }];\n    } else {\n      this.messageTextParts = [];\n      let text = content;\n      this.message!.mentioned_users.forEach((user) => {\n        const mention = `@${user.name || user.id}`;\n        const precedingText = text.substring(0, text.indexOf(mention));\n        let formattedPrecedingText = this.fixEmojiDisplay(precedingText);\n        formattedPrecedingText = this.wrapLinksWithAnchorTag(\n          formattedPrecedingText\n        );\n        this.messageTextParts!.push({\n          content: formattedPrecedingText,\n          type: 'text',\n        });\n        this.messageTextParts!.push({\n          content: mention,\n          type: 'mention',\n          user,\n        });\n        text = text.replace(precedingText + mention, '');\n      });\n      if (text) {\n        text = this.fixEmojiDisplay(text);\n        text = this.wrapLinksWithAnchorTag(text);\n        this.messageTextParts.push({ content: text, type: 'text' });\n      }\n    }\n  }\n\n  private getMessageContent() {\n    const originalContent = this.message?.text;\n    if (this.shouldTranslate) {\n      const translation = this.message?.translation;\n      return translation || originalContent;\n    } else {\n      return originalContent;\n    }\n  }\n\n  private fixEmojiDisplay(content: string) {\n    // Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223\n    // Based on this: https://stackoverflow.com/questions/4565112/javascript-how-to-find-out-if-the-user-browser-is-chrome\n    /* eslint-disable @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-explicit-any */\n    const isChrome =\n      !!(window as any).chrome && typeof (window as any).opr === 'undefined';\n    /* eslint-enable @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-explicit-any */\n    content = content.replace(\n      this.emojiRegexp,\n      (match) =>\n        `<span ${\n          isChrome ? 'class=\"str-chat__emoji-display-fix\"' : ''\n        }>${match}</span>`\n    );\n\n    return content;\n  }\n\n  private wrapLinksWithAnchorTag(content: string) {\n    if (this.displayAs === 'html') {\n      return content;\n    }\n    content = content.replace(this.urlRegexp, (match) => {\n      if (this.messageService.customLinkRenderer) {\n        return this.messageService.customLinkRenderer(match);\n      } else {\n        let href = match;\n        if (\n          !href.startsWith('http') &&\n          !href.startsWith('ftp') &&\n          !href.startsWith('file')\n        ) {\n          href = `https://${match}`;\n        }\n        return `<a href=\"${href}\" target=\"_blank\" rel=\"nofollow\">${match}</a>`;\n      }\n    });\n\n    return content;\n  }\n}\n","<p\n  [class.str-chat__quoted-message-text-value]=\"isQuoted\"\n  [class.str-chat__message-text-value]=\"!isQuoted\"\n  *ngIf=\"message?.text\"\n  data-testid=\"text\"\n>\n  <ng-container *ngIf=\"messageTextParts; else defaultContent\">\n    <!-- eslint-disable-next-line @angular-eslint/template/use-track-by-function -->\n    <ng-container *ngFor=\"let part of messageTextParts\">\n      <span\n        *ngIf=\"part.type === 'text'; else mention\"\n        [innerHTML]=\"part.content\"\n      ></span>\n      <ng-template #mention>\n        <ng-template #defaultMention let-content=\"content\">\n          <span class=\"str-chat__message-mention\">{{ content }}</span>\n        </ng-template>\n        <ng-container\n          *ngTemplateOutlet=\"\n            (customTemplatesService.mentionTemplate$ | async) || defaultMention;\n            context: getMentionContext(part)\n          \"\n        ></ng-container>\n      </ng-template>\n    </ng-container>\n  </ng-container>\n  <ng-template #defaultContent>\n    <ng-container *ngIf=\"displayAs === 'text'; else asHTML\">\n      {{ messageText || \"\" }}\n    </ng-container>\n    <ng-template #asHTML\n      ><span data-testid=\"html-content\" [innerHTML]=\"messageText\"></span\n    ></ng-template>\n  </ng-template>\n</p>\n"]}
|
|
143
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-text.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-text/message-text.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-text/message-text.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, SimpleChanges } from '@angular/core';\nimport {\n  DefaultStreamChatGenerics,\n  MentionTemplateContext,\n  StreamMessage,\n} from '../types';\nimport { MessageResponseBase, UserResponse } from 'stream-chat';\nimport emojiRegex from 'emoji-regex';\nimport { MessageService } from '../message.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\n\ntype MessagePart = {\n  content: string;\n  type: 'text' | 'mention';\n  user?: UserResponse;\n};\n\n/**\n * The `MessageTextComponent` displays the text content of a message.\n */\n@Component({\n  selector: 'stream-message-text',\n  templateUrl: './message-text.component.html',\n  styles: [],\n})\nexport class MessageTextComponent implements OnChanges {\n  /**\n   * The message which text should be displayed\n   */\n  @Input() message:\n    | StreamMessage<DefaultStreamChatGenerics>\n    | undefined\n    | MessageResponseBase<DefaultStreamChatGenerics>;\n  /**\n   * `true` if the component displayes a message quote\n   */\n  @Input() isQuoted: boolean = false;\n  /**\n   * `true` if the\n   */\n  @Input() shouldTranslate: boolean = false;\n  messageTextParts: MessagePart[] | undefined = [];\n  messageText?: string;\n  displayAs: 'text' | 'html';\n  private readonly urlRegexp: RegExp;\n  private emojiRegexp = new RegExp(emojiRegex(), 'g');\n\n  constructor(\n    private messageService: MessageService,\n    readonly customTemplatesService: CustomTemplatesService,\n  ) {\n    this.displayAs = this.messageService.displayAs;\n    try {\n      this.urlRegexp = new RegExp(\n        '(?:(?:https?|ftp|file):\\\\/\\\\/|www\\\\.|ftp\\\\.|(?:[a-zA-Z0-9-]+\\\\.)+[a-zA-Z]{2,})(?![^\\\\s]*@[^\\\\s]*)(?:[^\\\\s()<>]+|\\\\([\\\\w\\\\d]+\\\\))*(?<!@[a-zA-Z0-9.-]+\\\\.[a-zA-Z]{2,})',\n        'gim',\n      );\n    } catch {\n      this.urlRegexp =\n        /(?:(?:https?|ftp|file):\\/\\/|www\\.|ftp\\.)(?:\\([-A-Z0-9+&@#/%=~_|$?!:,.]*\\)|[-A-Z0-9+&@#/%=~_|$?!:,.])*(?:\\([-A-Z0-9+&@#/%=~_|$?!:,.]*\\)|[A-Z0-9+&@#/%=~_|$])/gim;\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.message || changes.shouldTranslate) {\n      this.createMessageParts();\n    }\n  }\n\n  getMentionContext(messagePart: MessagePart): MentionTemplateContext {\n    return {\n      content: messagePart.content,\n      user: messagePart.user!,\n    };\n  }\n\n  private createMessageParts() {\n    this.messageTextParts = undefined;\n    this.messageText = undefined;\n    let content = this.getMessageContent();\n    if (\n      (!this.message!.mentioned_users ||\n        this.message!.mentioned_users.length === 0) &&\n      !content?.match(this.emojiRegexp) &&\n      !content?.match(this.urlRegexp)\n    ) {\n      this.messageTextParts = undefined;\n      this.messageText = content;\n      return;\n    }\n    if (!content) {\n      return;\n    }\n    if (\n      !this.message!.mentioned_users ||\n      this.message!.mentioned_users.length === 0\n    ) {\n      content = this.fixEmojiDisplay(content);\n      content = this.wrapLinksWithAnchorTag(content);\n      this.messageTextParts = [{ content, type: 'text' }];\n    } else {\n      this.messageTextParts = [];\n      let text = content;\n      this.message!.mentioned_users.forEach((user) => {\n        const mention = `@${user.name || user.id}`;\n        const precedingText = text.substring(0, text.indexOf(mention));\n        let formattedPrecedingText = this.fixEmojiDisplay(precedingText);\n        formattedPrecedingText = this.wrapLinksWithAnchorTag(\n          formattedPrecedingText,\n        );\n        this.messageTextParts!.push({\n          content: formattedPrecedingText,\n          type: 'text',\n        });\n        this.messageTextParts!.push({\n          content: mention,\n          type: 'mention',\n          user,\n        });\n        text = text.replace(precedingText + mention, '');\n      });\n      if (text) {\n        text = this.fixEmojiDisplay(text);\n        text = this.wrapLinksWithAnchorTag(text);\n        this.messageTextParts.push({ content: text, type: 'text' });\n      }\n    }\n  }\n\n  private getMessageContent() {\n    const originalContent = this.message?.text;\n    if (this.shouldTranslate) {\n      const translation = this.message?.translation;\n      return translation || originalContent;\n    } else {\n      return originalContent;\n    }\n  }\n\n  private fixEmojiDisplay(content: string) {\n    // Wrap emojis in span to display emojis correctly in Chrome https://bugs.chromium.org/p/chromium/issues/detail?id=596223\n    // Based on this: https://stackoverflow.com/questions/4565112/javascript-how-to-find-out-if-the-user-browser-is-chrome\n    /* eslint-disable @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-explicit-any */\n    const isChrome =\n      !!(window as any).chrome && typeof (window as any).opr === 'undefined';\n    /* eslint-enable @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-explicit-any */\n    content = content.replace(\n      this.emojiRegexp,\n      (match) =>\n        `<span ${\n          isChrome ? 'class=\"str-chat__emoji-display-fix\"' : ''\n        }>${match}</span>`,\n    );\n\n    return content;\n  }\n\n  private wrapLinksWithAnchorTag(content: string) {\n    if (this.displayAs === 'html') {\n      return content;\n    }\n    content = content.replace(this.urlRegexp, (match) => {\n      if (this.messageService.customLinkRenderer) {\n        return this.messageService.customLinkRenderer(match);\n      } else {\n        let href = match;\n        if (\n          !href.startsWith('http') &&\n          !href.startsWith('ftp') &&\n          !href.startsWith('file')\n        ) {\n          href = `https://${match}`;\n        }\n        return `<a href=\"${href}\" target=\"_blank\" rel=\"nofollow\">${match}</a>`;\n      }\n    });\n\n    return content;\n  }\n}\n","@if (message?.text) {\n  <p\n    [class.str-chat__quoted-message-text-value]=\"isQuoted\"\n    [class.str-chat__message-text-value]=\"!isQuoted\"\n    data-testid=\"text\"\n  >\n    @if (messageTextParts) {\n      @for (part of messageTextParts; track part.content) {\n        @if (part.type === \"text\") {\n          <span [innerHTML]=\"part.content\"></span>\n        } @else {\n          <ng-template #defaultMention let-content=\"content\">\n            <span class=\"str-chat__message-mention\">{{ content }}</span>\n          </ng-template>\n          <ng-container\n            *ngTemplateOutlet=\"\n              (customTemplatesService.mentionTemplate$ | async) ||\n                defaultMention;\n              context: getMentionContext(part)\n            \"\n          />\n        }\n      }\n    } @else {\n      @if (displayAs === \"text\") {\n        {{ messageText || \"\" }}\n      } @else {\n        <span data-testid=\"html-content\" [innerHTML]=\"messageText\"></span>\n      }\n    }\n  </p>\n}\n"]}
|
|
@@ -0,0 +1,43 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import * as i0 from "@angular/core";
|
|
3
|
+
/**
|
|
4
|
+
* The message service contains configuration options related to displaying the message content
|
|
5
|
+
*/
|
|
6
|
+
export class MessageService {
|
|
7
|
+
constructor() {
|
|
8
|
+
/**
|
|
9
|
+
* Decides if the message content should be formatted as text or HTML
|
|
10
|
+
*
|
|
11
|
+
* If you display messages as text the following parts are still be displayed as HTML:
|
|
12
|
+
* - user mentions -> you can customize this with your own template using the [`customTemplatesService.mentionTemplate$`](/chat/docs/sdk/angular/services/CustomTemplatesService/#mentiontemplate)
|
|
13
|
+
* - links -> you can customize this by providing you own [`customLinkRenderer`](#customlinkrenderer) method
|
|
14
|
+
*/
|
|
15
|
+
this.displayAs = 'text';
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* Tells if an attachment is custom (you need to provide your own template to display them) or built-in (the SDK supports it out-of-the-box)
|
|
19
|
+
* @param attachment
|
|
20
|
+
* @returns `true` if the attachment is custom
|
|
21
|
+
*/
|
|
22
|
+
isCustomAttachment(attachment) {
|
|
23
|
+
if (this.filterCustomAttachment) {
|
|
24
|
+
return this.filterCustomAttachment(attachment);
|
|
25
|
+
}
|
|
26
|
+
else {
|
|
27
|
+
return (attachment.type !== 'image' &&
|
|
28
|
+
attachment.type !== 'file' &&
|
|
29
|
+
attachment.type !== 'video' &&
|
|
30
|
+
attachment.type !== 'voiceRecording' &&
|
|
31
|
+
attachment.type !== 'giphy');
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
35
|
+
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageService, providedIn: 'root' }); }
|
|
36
|
+
}
|
|
37
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: MessageService, decorators: [{
|
|
38
|
+
type: Injectable,
|
|
39
|
+
args: [{
|
|
40
|
+
providedIn: 'root',
|
|
41
|
+
}]
|
|
42
|
+
}], ctorParameters: () => [] });
|
|
43
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWVzc2FnZS5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21lc3NhZ2Uuc2VydmljZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDOztBQUkzQzs7R0FFRztBQUlILE1BQU0sT0FBTyxjQUFjO0lBMkJ6QjtRQXhCQTs7Ozs7O1dBTUc7UUFDSCxjQUFTLEdBQW9CLE1BQU0sQ0FBQztJQWlCckIsQ0FBQztJQUVoQjs7OztPQUlHO0lBQ0gsa0JBQWtCLENBQUMsVUFBeUI7UUFDMUMsSUFBSSxJQUFJLENBQUMsc0JBQXNCLEVBQUU7WUFDL0IsT0FBTyxJQUFJLENBQUMsc0JBQXNCLENBQUMsVUFBVSxDQUFDLENBQUM7U0FDaEQ7YUFBTTtZQUNMLE9BQU8sQ0FDTCxVQUFVLENBQUMsSUFBSSxLQUFLLE9BQU87Z0JBQzNCLFVBQVUsQ0FBQyxJQUFJLEtBQUssTUFBTTtnQkFDMUIsVUFBVSxDQUFDLElBQUksS0FBSyxPQUFPO2dCQUMzQixVQUFVLENBQUMsSUFBSSxLQUFLLGdCQUFnQjtnQkFDcEMsVUFBVSxDQUFDLElBQUksS0FBSyxPQUFPLENBQzVCLENBQUM7U0FDSDtJQUNILENBQUM7OEdBOUNVLGNBQWM7a0hBQWQsY0FBYyxjQUZiLE1BQU07OzJGQUVQLGNBQWM7a0JBSDFCLFVBQVU7bUJBQUM7b0JBQ1YsVUFBVSxFQUFFLE1BQU07aUJBQ25CIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQXR0YWNobWVudCB9IGZyb20gJ3N0cmVhbS1jaGF0JztcbmltcG9ydCB7IERlZmF1bHRTdHJlYW1DaGF0R2VuZXJpY3MgfSBmcm9tICcuL3R5cGVzJztcblxuLyoqXG4gKiBUaGUgbWVzc2FnZSBzZXJ2aWNlIGNvbnRhaW5zIGNvbmZpZ3VyYXRpb24gb3B0aW9ucyByZWxhdGVkIHRvIGRpc3BsYXlpbmcgdGhlIG1lc3NhZ2UgY29udGVudFxuICovXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290Jyxcbn0pXG5leHBvcnQgY2xhc3MgTWVzc2FnZVNlcnZpY2U8XG4gIFQgZXh0ZW5kcyBEZWZhdWx0U3RyZWFtQ2hhdEdlbmVyaWNzID0gRGVmYXVsdFN0cmVhbUNoYXRHZW5lcmljcyxcbj4ge1xuICAvKipcbiAgICogRGVjaWRlcyBpZiB0aGUgbWVzc2FnZSBjb250ZW50IHNob3VsZCBiZSBmb3JtYXR0ZWQgYXMgdGV4dCBvciBIVE1MXG4gICAqXG4gICAqIElmIHlvdSBkaXNwbGF5IG1lc3NhZ2VzIGFzIHRleHQgdGhlIGZvbGxvd2luZyBwYXJ0cyBhcmUgc3RpbGwgYmUgZGlzcGxheWVkIGFzIEhUTUw6XG4gICAqIC0gdXNlciBtZW50aW9ucyAtPiB5b3UgY2FuIGN1c3RvbWl6ZSB0aGlzIHdpdGggeW91ciBvd24gdGVtcGxhdGUgdXNpbmcgdGhlIFtgY3VzdG9tVGVtcGxhdGVzU2VydmljZS5tZW50aW9uVGVtcGxhdGUkYF0oL2NoYXQvZG9jcy9zZGsvYW5ndWxhci9zZXJ2aWNlcy9DdXN0b21UZW1wbGF0ZXNTZXJ2aWNlLyNtZW50aW9udGVtcGxhdGUpXG4gICAqIC0gbGlua3MgLT4geW91IGNhbiBjdXN0b21pemUgdGhpcyBieSBwcm92aWRpbmcgeW91IG93biBbYGN1c3RvbUxpbmtSZW5kZXJlcmBdKCNjdXN0b21saW5rcmVuZGVyZXIpIG1ldGhvZFxuICAgKi9cbiAgZGlzcGxheUFzOiAndGV4dCcgfCAnaHRtbCcgPSAndGV4dCc7XG5cbiAgLyoqXG4gICAqIFlvdSBjYW4gcHJvdmlkZSBhIGN1c3RvbSBtZXRob2QgdG8gZGlzcGxheSBsaW5rc1xuICAgKiBAcGFyYW0gdXJsIHRoZSB1cmwgdGhlIGxpbmsgc2hvdWxkIHJlZmVyIHRvXG4gICAqIEByZXR1cm5zIHRoZSBIVE1MIG1hcmt1cCBhcyBhIHN0cmluZyBmb3IgdGhlIGxpbmtcbiAgICovXG4gIGN1c3RvbUxpbmtSZW5kZXJlcj86ICh1cmw6IHN0cmluZykgPT4gc3RyaW5nO1xuICAvKipcbiAgICogVGhlIFNESyBzdXBwb3J0cyB0aGUgZm9sbG93aW5nIGF0dGFjaG1lbnQgdHlwZXM6IGBpbWFnZWAsIGBmaWxlYCwgYGdpcGh5YCwgYHZpZGVvYCBhbmQgYHZvaWNlUmVjb3JkaW5nYCBhdHRhY2htZW50cy5cbiAgICpcbiAgICogQWxsIG90aGVyIHR5cGVzIGFyZSB0cmVhdGVkIGFzIGN1c3RvbSBhdHRhY2htZW50cywgaG93ZXZlciBpdCdzIHBvc3NpYmxlIHRvIG92ZXJyaWRlIHRoaXMgbG9naWMsIGFuZCBwcm92aWRlIGEgY3VzdG9tIGZpbHRlcmluZyBtZXRob2Qgd2hpY2ggY2FuIGJlIHVzZWQgdG8gdHJlYXQgc29tZSBidWlsdC1pbiBhdHRhY2htZW50cyBhcyBjdXN0b20uXG4gICAqXG4gICAqIFByb3ZpZGUgYSBtZXRob2Qgd2hpY2ggcmV0cnVucyBgdHJ1ZWAgaWYgYW4gYXR0YWNobWVudCBzaG91bGQgYmUgY29uc2lkZXJlZCBhcyBjdXN0b20uXG4gICAqL1xuICBmaWx0ZXJDdXN0b21BdHRhY2htZW50PzogKGF0dGFjaG1lbnQ6IEF0dGFjaG1lbnQ8VD4pID0+IGJvb2xlYW47XG5cbiAgY29uc3RydWN0b3IoKSB7fVxuXG4gIC8qKlxuICAgKiBUZWxscyBpZiBhbiBhdHRhY2htZW50IGlzIGN1c3RvbSAoeW91IG5lZWQgdG8gcHJvdmlkZSB5b3VyIG93biB0ZW1wbGF0ZSB0byBkaXNwbGF5IHRoZW0pIG9yIGJ1aWx0LWluICh0aGUgU0RLIHN1cHBvcnRzIGl0IG91dC1vZi10aGUtYm94KVxuICAgKiBAcGFyYW0gYXR0YWNobWVudFxuICAgKiBAcmV0dXJucyBgdHJ1ZWAgaWYgdGhlIGF0dGFjaG1lbnQgaXMgY3VzdG9tXG4gICAqL1xuICBpc0N1c3RvbUF0dGFjaG1lbnQoYXR0YWNobWVudDogQXR0YWNobWVudDxUPikge1xuICAgIGlmICh0aGlzLmZpbHRlckN1c3RvbUF0dGFjaG1lbnQpIHtcbiAgICAgIHJldHVybiB0aGlzLmZpbHRlckN1c3RvbUF0dGFjaG1lbnQoYXR0YWNobWVudCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIHJldHVybiAoXG4gICAgICAgIGF0dGFjaG1lbnQudHlwZSAhPT0gJ2ltYWdlJyAmJlxuICAgICAgICBhdHRhY2htZW50LnR5cGUgIT09ICdmaWxlJyAmJlxuICAgICAgICBhdHRhY2htZW50LnR5cGUgIT09ICd2aWRlbycgJiZcbiAgICAgICAgYXR0YWNobWVudC50eXBlICE9PSAndm9pY2VSZWNvcmRpbmcnICYmXG4gICAgICAgIGF0dGFjaG1lbnQudHlwZSAhPT0gJ2dpcGh5J1xuICAgICAgKTtcbiAgICB9XG4gIH1cbn1cbiJdfQ==
|
|
@@ -50,13 +50,13 @@ export class ModalComponent {
|
|
|
50
50
|
stopWatchForOutsideClicks() {
|
|
51
51
|
window.removeEventListener('click', this.watchForOutsideClicks);
|
|
52
52
|
}
|
|
53
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: ModalComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
54
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "17.0.0", type: ModalComponent, selector: "stream-modal", inputs: { isOpen: "isOpen", content: "content" }, outputs: { isOpenChange: "isOpenChange" }, viewQueries: [{ propertyName: "innerContainer", first: true, predicate: ["modalInner"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n data-testid=\"modal\"\n class=\"str-chat__modal str-chat__modal--{{ isOpen ? 'open' : 'close' }}\"\n>\n <div\n data-testid=\"close\"\n class=\"str-chat__modal__close-button\"\n (click)=\"close()\"\n (keyup.enter)=\"close()\"\n >\n <stream-icon-placeholder icon=\"close\" />\n </div>\n <div #modalInner class=\"str-chat__modal__inner\">\n @if (isOpen && content) {\n <ng-container *ngTemplateOutlet=\"content\" />\n }\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon"] }] }); }
|
|
53
55
|
}
|
|
54
|
-
|
|
55
|
-
ModalComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: ModalComponent, selector: "stream-modal", inputs: { isOpen: "isOpen", content: "content" }, outputs: { isOpenChange: "isOpenChange" }, viewQueries: [{ propertyName: "innerContainer", first: true, predicate: ["modalInner"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<div\n data-testid=\"modal\"\n class=\"str-chat__modal str-chat__modal--{{ isOpen ? 'open' : 'close' }}\"\n>\n <div\n data-testid=\"close\"\n class=\"str-chat__modal__close-button\"\n (click)=\"close()\"\n (keyup.enter)=\"close()\"\n >\n <stream-icon-placeholder icon=\"close\"></stream-icon-placeholder>\n </div>\n <div #modalInner class=\"str-chat__modal__inner\">\n <ng-container *ngIf=\"content; else elseContent\">\n <ng-container *ngTemplateOutlet=\"content\"></ng-container>\n </ng-container>\n <ng-template #elseContent>\n <ng-content></ng-content>\n </ng-template>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i2.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon"] }] });
|
|
56
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: ModalComponent, decorators: [{
|
|
56
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: ModalComponent, decorators: [{
|
|
57
57
|
type: Component,
|
|
58
|
-
args: [{ selector: 'stream-modal', template: "<div\n data-testid=\"modal\"\n class=\"str-chat__modal str-chat__modal--{{ isOpen ? 'open' : 'close' }}\"\n>\n <div\n data-testid=\"close\"\n class=\"str-chat__modal__close-button\"\n (click)=\"close()\"\n (keyup.enter)=\"close()\"\n >\n <stream-icon-placeholder icon=\"close\"
|
|
59
|
-
}], ctorParameters:
|
|
58
|
+
args: [{ selector: 'stream-modal', template: "<div\n data-testid=\"modal\"\n class=\"str-chat__modal str-chat__modal--{{ isOpen ? 'open' : 'close' }}\"\n>\n <div\n data-testid=\"close\"\n class=\"str-chat__modal__close-button\"\n (click)=\"close()\"\n (keyup.enter)=\"close()\"\n >\n <stream-icon-placeholder icon=\"close\" />\n </div>\n <div #modalInner class=\"str-chat__modal__inner\">\n @if (isOpen && content) {\n <ng-container *ngTemplateOutlet=\"content\" />\n }\n </div>\n</div>\n" }]
|
|
59
|
+
}], ctorParameters: () => [], propDecorators: { isOpen: [{
|
|
60
60
|
type: Input
|
|
61
61
|
}], content: [{
|
|
62
62
|
type: Input
|
|
@@ -66,4 +66,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImpor
|
|
|
66
66
|
type: ViewChild,
|
|
67
67
|
args: ['modalInner']
|
|
68
68
|
}] } });
|
|
69
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
69
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibW9kYWwuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL21vZGFsL21vZGFsLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi9tb2RhbC9tb2RhbC5jb21wb25lbnQuaHRtbCJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQ0wsU0FBUyxFQUVULFlBQVksRUFDWixLQUFLLEVBRUwsTUFBTSxFQUdOLFNBQVMsR0FDVixNQUFNLGVBQWUsQ0FBQzs7OztBQUV2Qjs7R0FFRztBQU1ILE1BQU0sT0FBTyxjQUFjO0lBaUJ6QjtRQWhCQTs7V0FFRztRQUNNLFdBQU0sR0FBRyxLQUFLLENBQUM7UUFLeEI7O1dBRUc7UUFDZ0IsaUJBQVksR0FBRyxJQUFJLFlBQVksRUFBVyxDQUFDO1FBNkJ0RCxxQkFBZ0IsR0FBRyxDQUFDLEtBQW9CLEVBQUUsRUFBRTtZQUNsRCxJQUFJLEtBQUssQ0FBQyxHQUFHLEtBQUssUUFBUSxFQUFFO2dCQUMxQixJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7YUFDZDtRQUNILENBQUMsQ0FBQztRQUVNLHlCQUFvQixHQUFHLEdBQUcsRUFBRTtZQUNsQyxNQUFNLENBQUMsbUJBQW1CLENBQUMsT0FBTyxFQUFFLElBQUksQ0FBQyxnQkFBZ0IsQ0FBQyxDQUFDO1FBQzdELENBQUMsQ0FBQztRQUVNLDBCQUFxQixHQUFHLENBQUMsS0FBWSxFQUFFLEVBQUU7WUFDL0MsSUFBSSxDQUFDLElBQUksQ0FBQyxjQUFjLEVBQUUsYUFBYSxDQUFDLFFBQVEsQ0FBQyxLQUFLLENBQUMsTUFBYyxDQUFDLEVBQUU7Z0JBQ3RFLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQzthQUNkO1FBQ0gsQ0FBQyxDQUFDO0lBdENhLENBQUM7SUFFaEIsV0FBVyxDQUFDLE9BQXNCO1FBQ2hDLElBQUksT0FBTyxDQUFDLE1BQU0sRUFBRTtZQUNsQixJQUFJLElBQUksQ0FBQyxNQUFNLEVBQUU7Z0JBQ2YsTUFBTSxDQUFDLGdCQUFnQixDQUFDLE9BQU8sRUFBRSxJQUFJLENBQUMsZ0JBQWdCLENBQUMsQ0FBQztnQkFDeEQsVUFBVSxDQUNSLEdBQUcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBQyxPQUFPLEVBQUUsSUFBSSxDQUFDLHFCQUFxQixDQUFDLEVBQ2xFLENBQUMsQ0FDRixDQUFDO2FBQ0g7aUJBQU07Z0JBQ0wsSUFBSSxDQUFDLHlCQUF5QixFQUFFLENBQUM7Z0JBQ2pDLElBQUksQ0FBQyxvQkFBb0IsRUFBRSxDQUFDO2FBQzdCO1NBQ0Y7SUFDSCxDQUFDO0lBRUQsS0FBSztRQUNILElBQUksQ0FBQyxNQUFNLEdBQUcsS0FBSyxDQUFDO1FBQ3BCLElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQzlCLElBQUksQ0FBQyx5QkFBeUIsRUFBRSxDQUFDO1FBQ2pDLElBQUksQ0FBQyxvQkFBb0IsRUFBRSxDQUFDO0lBQzlCLENBQUM7SUFrQk8seUJBQXlCO1FBQy9CLE1BQU0sQ0FBQyxtQkFBbUIsQ0FBQyxPQUFPLEVBQUUsSUFBSSxDQUFDLHFCQUFxQixDQUFDLENBQUM7SUFDbEUsQ0FBQzs4R0EzRFUsY0FBYztrR0FBZCxjQUFjLG9SQ3BCM0IsNmRBa0JBOzsyRkRFYSxjQUFjO2tCQUwxQixTQUFTOytCQUNFLGNBQWM7d0RBUWYsTUFBTTtzQkFBZCxLQUFLO2dCQUlHLE9BQU87c0JBQWYsS0FBSztnQkFJYSxZQUFZO3NCQUE5QixNQUFNO2dCQUMwQixjQUFjO3NCQUE5QyxTQUFTO3VCQUFDLFlBQVkiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1xuICBDb21wb25lbnQsXG4gIEVsZW1lbnRSZWYsXG4gIEV2ZW50RW1pdHRlcixcbiAgSW5wdXQsXG4gIE9uQ2hhbmdlcyxcbiAgT3V0cHV0LFxuICBTaW1wbGVDaGFuZ2VzLFxuICBUZW1wbGF0ZVJlZixcbiAgVmlld0NoaWxkLFxufSBmcm9tICdAYW5ndWxhci9jb3JlJztcblxuLyoqXG4gKiBUaGUgYE1vZGFsYCBjb21wb25lbnQgZGlzcGxheXMgaXRzIGNvbnRlbnQgaW4gYW4gb3ZlcmxheS4gVGhlIG1vZGFsIGNhbiBiZSBjbG9zZWQgd2l0aCBhIGNsb3NlIGJ1dHRvbiwgaWYgdGhlIHVzZXIgY2xpY2tzIG91dHNpZGUgb2YgdGhlIG1vZGFsIGNvbnRlbnQsIG9yIGlmIHRoZSBlc2NhcGUgYnV0dG9uIGlzIHByZXNzZWQuIFRoZSBtb2RhbCBjYW4gYWxzbyBiZSBjbG9zZWQgZnJvbSBvdXRzaWRlLlxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdzdHJlYW0tbW9kYWwnLFxuICB0ZW1wbGF0ZVVybDogJy4vbW9kYWwuY29tcG9uZW50Lmh0bWwnLFxuICBzdHlsZXM6IFtdLFxufSlcbmV4cG9ydCBjbGFzcyBNb2RhbENvbXBvbmVudCBpbXBsZW1lbnRzIE9uQ2hhbmdlcyB7XG4gIC8qKlxuICAgKiBJZiBgdHJ1ZWAgdGhlIG1vZGFsIHdpbGwgYmUgZGlzcGxheWVkLCBpZiBgZmFsc2VgIHRoZSBtb2RhbCB3aWxsIGJlIGhpZGRlblxuICAgKi9cbiAgQElucHV0KCkgaXNPcGVuID0gZmFsc2U7XG4gIC8qKlxuICAgKiBUaGUgY29udGVudCBvZiB0aGUgbW9kYWwgIChjYW4gYWxzbyBiZSBwcm92aWRlZCB1c2luZyBgbmctY29udGVudGApXG4gICAqL1xuICBASW5wdXQoKSBjb250ZW50OiBUZW1wbGF0ZVJlZjx2b2lkPiB8IHVuZGVmaW5lZDtcbiAgLyoqXG4gICAqIEVtaXRzIGB0cnVlYCBpZiB0aGUgbW9kYWwgYmVjb21lcyB2aXNpYmxlLCBhbmQgYGZhbHNlYCBpZiB0aGUgbW9kYWwgaXMgY2xvc2VkLlxuICAgKi9cbiAgQE91dHB1dCgpIHJlYWRvbmx5IGlzT3BlbkNoYW5nZSA9IG5ldyBFdmVudEVtaXR0ZXI8Ym9vbGVhbj4oKTtcbiAgQFZpZXdDaGlsZCgnbW9kYWxJbm5lcicpIHByaXZhdGUgaW5uZXJDb250YWluZXI6XG4gICAgfCBFbGVtZW50UmVmPEhUTUxFbGVtZW50PlxuICAgIHwgdW5kZWZpbmVkO1xuXG4gIGNvbnN0cnVjdG9yKCkge31cblxuICBuZ09uQ2hhbmdlcyhjaGFuZ2VzOiBTaW1wbGVDaGFuZ2VzKTogdm9pZCB7XG4gICAgaWYgKGNoYW5nZXMuaXNPcGVuKSB7XG4gICAgICBpZiAodGhpcy5pc09wZW4pIHtcbiAgICAgICAgd2luZG93LmFkZEV2ZW50TGlzdGVuZXIoJ2tleXVwJywgdGhpcy53YXRjaEZvckVzY1ByZXNzKTtcbiAgICAgICAgc2V0VGltZW91dChcbiAgICAgICAgICAoKSA9PiB3aW5kb3cuYWRkRXZlbnRMaXN0ZW5lcignY2xpY2snLCB0aGlzLndhdGNoRm9yT3V0c2lkZUNsaWNrcyksXG4gICAgICAgICAgMCxcbiAgICAgICAgKTtcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHRoaXMuc3RvcFdhdGNoRm9yT3V0c2lkZUNsaWNrcygpO1xuICAgICAgICB0aGlzLnN0b3BXYXRjaEZvckVzY1ByZXNzKCk7XG4gICAgICB9XG4gICAgfVxuICB9XG5cbiAgY2xvc2UoKSB7XG4gICAgdGhpcy5pc09wZW4gPSBmYWxzZTtcbiAgICB0aGlzLmlzT3BlbkNoYW5nZS5lbWl0KGZhbHNlKTtcbiAgICB0aGlzLnN0b3BXYXRjaEZvck91dHNpZGVDbGlja3MoKTtcbiAgICB0aGlzLnN0b3BXYXRjaEZvckVzY1ByZXNzKCk7XG4gIH1cblxuICBwcml2YXRlIHdhdGNoRm9yRXNjUHJlc3MgPSAoZXZlbnQ6IEtleWJvYXJkRXZlbnQpID0+IHtcbiAgICBpZiAoZXZlbnQua2V5ID09PSAnRXNjYXBlJykge1xuICAgICAgdGhpcy5jbG9zZSgpO1xuICAgIH1cbiAgfTtcblxuICBwcml2YXRlIHN0b3BXYXRjaEZvckVzY1ByZXNzID0gKCkgPT4ge1xuICAgIHdpbmRvdy5yZW1vdmVFdmVudExpc3RlbmVyKCdrZXl1cCcsIHRoaXMud2F0Y2hGb3JFc2NQcmVzcyk7XG4gIH07XG5cbiAgcHJpdmF0ZSB3YXRjaEZvck91dHNpZGVDbGlja3MgPSAoZXZlbnQ6IEV2ZW50KSA9PiB7XG4gICAgaWYgKCF0aGlzLmlubmVyQ29udGFpbmVyPy5uYXRpdmVFbGVtZW50LmNvbnRhaW5zKGV2ZW50LnRhcmdldCBhcyBOb2RlKSkge1xuICAgICAgdGhpcy5jbG9zZSgpO1xuICAgIH1cbiAgfTtcblxuICBwcml2YXRlIHN0b3BXYXRjaEZvck91dHNpZGVDbGlja3MoKSB7XG4gICAgd2luZG93LnJlbW92ZUV2ZW50TGlzdGVuZXIoJ2NsaWNrJywgdGhpcy53YXRjaEZvck91dHNpZGVDbGlja3MpO1xuICB9XG59XG4iLCI8ZGl2XG4gIGRhdGEtdGVzdGlkPVwibW9kYWxcIlxuICBjbGFzcz1cInN0ci1jaGF0X19tb2RhbCBzdHItY2hhdF9fbW9kYWwtLXt7IGlzT3BlbiA/ICdvcGVuJyA6ICdjbG9zZScgfX1cIlxuPlxuICA8ZGl2XG4gICAgZGF0YS10ZXN0aWQ9XCJjbG9zZVwiXG4gICAgY2xhc3M9XCJzdHItY2hhdF9fbW9kYWxfX2Nsb3NlLWJ1dHRvblwiXG4gICAgKGNsaWNrKT1cImNsb3NlKClcIlxuICAgIChrZXl1cC5lbnRlcik9XCJjbG9zZSgpXCJcbiAgPlxuICAgIDxzdHJlYW0taWNvbi1wbGFjZWhvbGRlciBpY29uPVwiY2xvc2VcIiAvPlxuICA8L2Rpdj5cbiAgPGRpdiAjbW9kYWxJbm5lciBjbGFzcz1cInN0ci1jaGF0X19tb2RhbF9faW5uZXJcIj5cbiAgICBAaWYgKGlzT3BlbiAmJiBjb250ZW50KSB7XG4gICAgICA8bmctY29udGFpbmVyICpuZ1RlbXBsYXRlT3V0bGV0PVwiY29udGVudFwiIC8+XG4gICAgfVxuICA8L2Rpdj5cbjwvZGl2PlxuIl19
|