stream-chat-angular 4.66.0 → 5.0.0-v5.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.
Files changed (157) hide show
  1. package/assets/version.d.ts +1 -1
  2. package/{esm2015/assets/version.js → esm2020/assets/version.mjs} +2 -2
  3. package/esm2020/lib/attachment-configuration.service.mjs +185 -0
  4. package/esm2020/lib/attachment-list/attachment-list.component.mjs +205 -0
  5. package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +45 -0
  6. package/esm2020/lib/attachment.service.mjs +262 -0
  7. package/esm2020/lib/avatar/avatar.component.mjs +163 -0
  8. package/esm2020/lib/avatar-placeholder/avatar-placeholder.component.mjs +74 -0
  9. package/esm2020/lib/channel/channel.component.mjs +46 -0
  10. package/esm2020/lib/channel-header/channel-header.component.mjs +79 -0
  11. package/esm2020/lib/channel-list/channel-list-toggle.service.mjs +72 -0
  12. package/esm2020/lib/channel-list/channel-list.component.mjs +60 -0
  13. package/esm2020/lib/channel-preview/channel-preview.component.mjs +155 -0
  14. package/esm2020/lib/channel.service.mjs +1460 -0
  15. package/esm2020/lib/chat-client.service.mjs +206 -0
  16. package/{esm2015/lib/custom-templates.service.js → esm2020/lib/custom-templates.service.mjs} +3 -3
  17. package/{esm2015/lib/date-parser.service.js → esm2020/lib/date-parser.service.mjs} +3 -3
  18. package/esm2020/lib/edit-message-form/edit-message-form.component.mjs +83 -0
  19. package/esm2020/lib/get-channel-display-text.mjs +14 -0
  20. package/esm2020/lib/get-message-translation.mjs +12 -0
  21. package/esm2020/lib/icon/icon.component.mjs +21 -0
  22. package/esm2020/lib/icon-placeholder/icon-placeholder.component.mjs +31 -0
  23. package/esm2020/lib/loading-indicator/loading-indicator.component.mjs +31 -0
  24. package/esm2020/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +38 -0
  25. package/esm2020/lib/message/message.component.mjs +422 -0
  26. package/esm2020/lib/message-actions-box/message-actions-box.component.mjs +130 -0
  27. package/esm2020/lib/message-actions.service.mjs +119 -0
  28. package/esm2020/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +71 -0
  29. package/esm2020/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +262 -0
  30. package/{esm2015/lib/message-input/emoji-input.service.js → esm2020/lib/message-input/emoji-input.service.mjs} +3 -3
  31. package/{esm2015/lib/message-input/message-input-config.service.js → esm2020/lib/message-input/message-input-config.service.mjs} +3 -3
  32. package/esm2020/lib/message-input/message-input.component.mjs +443 -0
  33. package/{esm2015/lib/message-input/textarea/textarea.component.js → esm2020/lib/message-input/textarea/textarea.component.mjs} +5 -9
  34. package/esm2020/lib/message-input/textarea.directive.mjs +89 -0
  35. package/esm2020/lib/message-list/group-styles.mjs +52 -0
  36. package/{esm2015/lib/message-list/image-load.service.js → esm2020/lib/message-list/image-load.service.mjs} +3 -3
  37. package/esm2020/lib/message-list/message-list.component.mjs +699 -0
  38. package/esm2020/lib/message-preview.mjs +21 -0
  39. package/esm2020/lib/message-reactions/message-reactions.component.mjs +255 -0
  40. package/{esm2015/lib/message-reactions.service.js → esm2020/lib/message-reactions.service.mjs} +3 -3
  41. package/{esm2015/lib/message.service.js → esm2020/lib/message.service.mjs} +4 -4
  42. package/esm2020/lib/modal/modal.component.mjs +69 -0
  43. package/esm2020/lib/notification/notification.component.mjs +20 -0
  44. package/esm2020/lib/notification-list/notification-list.component.mjs +37 -0
  45. package/esm2020/lib/notification.service.mjs +79 -0
  46. package/esm2020/lib/read-by.mjs +12 -0
  47. package/{esm2015/lib/stream-autocomplete-textarea.module.js → esm2020/lib/stream-autocomplete-textarea.module.mjs} +6 -6
  48. package/{esm2015/lib/stream-avatar.module.js → esm2020/lib/stream-avatar.module.mjs} +5 -5
  49. package/{esm2015/lib/stream-chat.module.js → esm2020/lib/stream-chat.module.mjs} +8 -10
  50. package/{esm2015/lib/stream-i18n.service.js → esm2020/lib/stream-i18n.service.mjs} +5 -5
  51. package/{esm2015/lib/stream-textarea.module.js → esm2020/lib/stream-textarea.module.mjs} +6 -6
  52. package/esm2020/lib/theme.service.mjs +123 -0
  53. package/esm2020/lib/thread/thread.component.mjs +51 -0
  54. package/{esm2015/lib/transliteration.service.js → esm2020/lib/transliteration.service.mjs} +3 -3
  55. package/esm2020/lib/types.mjs +2 -0
  56. package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +183 -0
  57. package/esm2020/lib/voice-recording/voice-recording.component.mjs +102 -0
  58. package/fesm2015/{stream-chat-angular.js → stream-chat-angular.mjs} +345 -433
  59. package/fesm2015/stream-chat-angular.mjs.map +1 -0
  60. package/fesm2020/stream-chat-angular.mjs +7128 -0
  61. package/fesm2020/stream-chat-angular.mjs.map +1 -0
  62. package/lib/attachment-list/attachment-list.component.d.ts +3 -3
  63. package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +1 -1
  64. package/lib/attachment.service.d.ts +0 -1
  65. package/lib/avatar/avatar.component.d.ts +1 -1
  66. package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +1 -1
  67. package/lib/channel/channel.component.d.ts +1 -1
  68. package/lib/channel-header/channel-header.component.d.ts +1 -1
  69. package/lib/channel-list/channel-list-toggle.service.d.ts +0 -1
  70. package/lib/channel-list/channel-list.component.d.ts +1 -1
  71. package/lib/channel-preview/channel-preview.component.d.ts +1 -1
  72. package/lib/channel.service.d.ts +7 -7
  73. package/lib/chat-client.service.d.ts +1 -1
  74. package/lib/edit-message-form/edit-message-form.component.d.ts +1 -1
  75. package/lib/get-message-translation.d.ts +1 -1
  76. package/lib/icon/icon.component.d.ts +1 -1
  77. package/lib/icon-placeholder/icon-placeholder.component.d.ts +1 -1
  78. package/lib/loading-indicator/loading-indicator.component.d.ts +1 -1
  79. package/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.d.ts +1 -1
  80. package/lib/message/message.component.d.ts +1 -2
  81. package/lib/message-actions-box/message-actions-box.component.d.ts +2 -4
  82. package/lib/message-actions.service.d.ts +0 -1
  83. package/lib/message-bounce-prompt/message-bounce-prompt.component.d.ts +1 -1
  84. package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +1 -1
  85. package/lib/message-input/message-input.component.d.ts +2 -2
  86. package/lib/message-input/textarea/textarea.component.d.ts +1 -1
  87. package/lib/message-input/textarea.directive.d.ts +1 -1
  88. package/lib/message-list/group-styles.d.ts +1 -1
  89. package/lib/message-list/message-list.component.d.ts +4 -5
  90. package/lib/message-reactions/message-reactions.component.d.ts +1 -1
  91. package/lib/message.service.d.ts +0 -1
  92. package/lib/modal/modal.component.d.ts +1 -1
  93. package/lib/notification/notification.component.d.ts +1 -1
  94. package/lib/notification-list/notification-list.component.d.ts +2 -2
  95. package/lib/notification.service.d.ts +2 -5
  96. package/lib/theme.service.d.ts +1 -2
  97. package/lib/thread/thread.component.d.ts +1 -1
  98. package/lib/types.d.ts +18 -18
  99. package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +2 -2
  100. package/lib/voice-recording/voice-recording.component.d.ts +1 -1
  101. package/package.json +28 -15
  102. package/src/assets/version.ts +1 -1
  103. package/bundles/stream-chat-angular.umd.js +0 -8425
  104. package/bundles/stream-chat-angular.umd.js.map +0 -1
  105. package/esm2015/lib/attachment-configuration.service.js +0 -166
  106. package/esm2015/lib/attachment-list/attachment-list.component.js +0 -209
  107. package/esm2015/lib/attachment-preview-list/attachment-preview-list.component.js +0 -49
  108. package/esm2015/lib/attachment.service.js +0 -276
  109. package/esm2015/lib/avatar/avatar.component.js +0 -172
  110. package/esm2015/lib/avatar-placeholder/avatar-placeholder.component.js +0 -78
  111. package/esm2015/lib/channel/channel.component.js +0 -50
  112. package/esm2015/lib/channel-header/channel-header.component.js +0 -86
  113. package/esm2015/lib/channel-list/channel-list-toggle.service.js +0 -73
  114. package/esm2015/lib/channel-list/channel-list.component.js +0 -67
  115. package/esm2015/lib/channel-preview/channel-preview.component.js +0 -167
  116. package/esm2015/lib/channel.service.js +0 -1487
  117. package/esm2015/lib/chat-client.service.js +0 -211
  118. package/esm2015/lib/edit-message-form/edit-message-form.component.js +0 -87
  119. package/esm2015/lib/get-channel-display-text.js +0 -15
  120. package/esm2015/lib/get-message-translation.js +0 -13
  121. package/esm2015/lib/icon/icon.component.js +0 -25
  122. package/esm2015/lib/icon-placeholder/icon-placeholder.component.js +0 -35
  123. package/esm2015/lib/loading-indicator/loading-indicator.component.js +0 -35
  124. package/esm2015/lib/loading-indicator-placeholder/loading-indicator-placeholder.component.js +0 -42
  125. package/esm2015/lib/message/message.component.js +0 -436
  126. package/esm2015/lib/message-actions-box/message-actions-box.component.js +0 -137
  127. package/esm2015/lib/message-actions.service.js +0 -114
  128. package/esm2015/lib/message-bounce-prompt/message-bounce-prompt.component.js +0 -80
  129. package/esm2015/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.js +0 -262
  130. package/esm2015/lib/message-input/message-input.component.js +0 -455
  131. package/esm2015/lib/message-input/textarea.directive.js +0 -90
  132. package/esm2015/lib/message-list/group-styles.js +0 -53
  133. package/esm2015/lib/message-list/message-list.component.js +0 -726
  134. package/esm2015/lib/message-preview.js +0 -7
  135. package/esm2015/lib/message-reactions/message-reactions.component.js +0 -266
  136. package/esm2015/lib/modal/modal.component.js +0 -74
  137. package/esm2015/lib/notification/notification.component.js +0 -24
  138. package/esm2015/lib/notification-list/notification-list.component.js +0 -38
  139. package/esm2015/lib/notification.service.js +0 -79
  140. package/esm2015/lib/read-by.js +0 -13
  141. package/esm2015/lib/theme.service.js +0 -122
  142. package/esm2015/lib/thread/thread.component.js +0 -55
  143. package/esm2015/lib/types.js +0 -2
  144. package/esm2015/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.js +0 -192
  145. package/esm2015/lib/voice-recording/voice-recording.component.js +0 -115
  146. package/fesm2015/stream-chat-angular.js.map +0 -1
  147. /package/{esm2015/assets/i18n/en.js → esm2020/assets/i18n/en.mjs} +0 -0
  148. /package/{esm2015/lib/injection-tokens.js → esm2020/lib/injection-tokens.mjs} +0 -0
  149. /package/{esm2015/lib/is-image-attachment.js → esm2020/lib/is-image-attachment.mjs} +0 -0
  150. /package/{esm2015/lib/is-image-file.js → esm2020/lib/is-image-file.mjs} +0 -0
  151. /package/{esm2015/lib/is-on-separate-date.js → esm2020/lib/is-on-separate-date.mjs} +0 -0
  152. /package/{esm2015/lib/list-users.js → esm2020/lib/list-users.mjs} +0 -0
  153. /package/{esm2015/lib/message-input/textarea.interface.js → esm2020/lib/message-input/textarea.interface.mjs} +0 -0
  154. /package/{esm2015/lib/parse-date.js → esm2020/lib/parse-date.mjs} +0 -0
  155. /package/{esm2015/public-api.js → esm2020/public-api.mjs} +0 -0
  156. /package/{esm2015/stream-chat-angular.js → esm2020/stream-chat-angular.mjs} +0 -0
  157. /package/{stream-chat-angular.d.ts → index.d.ts} +0 -0
@@ -0,0 +1,130 @@
1
+ import { Component, EventEmitter, Input, Output, } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "../custom-templates.service";
4
+ import * as i2 from "../message-actions.service";
5
+ import * as i3 from "@angular/common";
6
+ import * as i4 from "@ngx-translate/core";
7
+ /**
8
+ * The `MessageActionsBox` component displays a list of message actions (i.e edit), that can be opened or closed. You can find the [list of the supported actions](../concepts/message-interactions.mdx) in the message interaction guide.
9
+ */
10
+ export class MessageActionsBoxComponent {
11
+ constructor(customTemplatesService, messageActionsService, cdRef) {
12
+ this.customTemplatesService = customTemplatesService;
13
+ this.messageActionsService = messageActionsService;
14
+ this.cdRef = cdRef;
15
+ /**
16
+ * Indicates if the list should be opened or closed. Adding a UI element to open and close the list is the parent's component responsibility.
17
+ * @deprecated No need for this since [theme-v2](../theming/introduction.mdx)
18
+ */
19
+ this.isOpen = false;
20
+ /**
21
+ * Indicates if the message actions are belonging to a message that was sent by the current user or not.
22
+ */
23
+ this.isMine = false;
24
+ /**
25
+ * The list of [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) that are enabled for the current user, the list of [supported interactions](../concepts/message-interactions.mdx) can be found in our message interaction guide. Unathorized actions won't be displayed on the UI.
26
+ */
27
+ this.enabledActions = [];
28
+ /**
29
+ * A list of custom message actions to be displayed in the action box
30
+ *
31
+ * In the next major release this will be released with `messageReactionsService.customActions$`
32
+ *
33
+ * More information: https://getstream.io/chat/docs/sdk/angular/services/MessageActionsService
34
+ */
35
+ this.customActions = [];
36
+ /**
37
+ * The number of authorized actions (it can be less or equal than the number of enabled actions)
38
+ * @deprecated components should use `messageReactionsService.getAuthorizedMessageActionsCount` method
39
+ *
40
+ * More information: https://getstream.io/chat/docs/sdk/angular/services/MessageActionsService
41
+ */
42
+ this.displayedActionsCount = new EventEmitter();
43
+ /**
44
+ * An event which emits `true` if the edit message modal is open, and `false` when it is closed.
45
+ * @deprecated components should use `messageReactionsService.messageToEdit$` Observable
46
+ *
47
+ * More information: https://getstream.io/chat/docs/sdk/angular/services/MessageActionsService
48
+ */
49
+ this.isEditing = new EventEmitter();
50
+ this.visibleMessageActionItems = [];
51
+ this.isEditModalOpen = false;
52
+ this.subscriptions = [];
53
+ this.isViewInited = false;
54
+ this.messageActionItems = this.messageActionsService.defaultActions;
55
+ }
56
+ ngOnInit() {
57
+ this.subscriptions.push(this.messageActionsService.messageToEdit$.subscribe((m) => {
58
+ let isEditModalOpen = false;
59
+ if (m && m.id === this.message?.id) {
60
+ isEditModalOpen = true;
61
+ }
62
+ if (isEditModalOpen !== this.isEditModalOpen) {
63
+ this.isEditModalOpen = isEditModalOpen;
64
+ this.isEditing.emit(this.isEditModalOpen);
65
+ if (this.isViewInited) {
66
+ this.cdRef.detectChanges();
67
+ }
68
+ }
69
+ }));
70
+ }
71
+ ngOnChanges(changes) {
72
+ if (changes.isMine ||
73
+ changes.enabledActions ||
74
+ changes.message ||
75
+ changes.customActions) {
76
+ this.setVisibleActions();
77
+ }
78
+ }
79
+ ngAfterViewInit() {
80
+ this.isViewInited = true;
81
+ }
82
+ ngOnDestroy() {
83
+ this.subscriptions.forEach((s) => s.unsubscribe());
84
+ }
85
+ getActionLabel(actionLabelOrTranslationKey) {
86
+ return typeof actionLabelOrTranslationKey === 'string'
87
+ ? actionLabelOrTranslationKey
88
+ : actionLabelOrTranslationKey(this.message);
89
+ }
90
+ getMessageActionTemplateContext(item) {
91
+ return {
92
+ actionHandler: item.actionHandler,
93
+ isMine: this.isMine,
94
+ actionName: item.actionName,
95
+ message: this.message,
96
+ actionLabelOrTranslationKey: item.actionLabelOrTranslationKey,
97
+ };
98
+ }
99
+ trackByActionName(_, item) {
100
+ return item.actionName;
101
+ }
102
+ setVisibleActions() {
103
+ this.visibleMessageActionItems = [
104
+ ...this.messageActionItems,
105
+ ...this.customActions,
106
+ ].filter((item) => item.isVisible(this.enabledActions, this.isMine, this.message));
107
+ this.displayedActionsCount.emit(this.visibleMessageActionItems.length);
108
+ }
109
+ }
110
+ MessageActionsBoxComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageActionsBoxComponent, deps: [{ token: i1.CustomTemplatesService }, { token: i2.MessageActionsService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
111
+ MessageActionsBoxComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: MessageActionsBoxComponent, selector: "stream-message-actions-box", inputs: { isOpen: "isOpen", isMine: "isMine", message: "message", enabledActions: "enabledActions", customActions: "customActions" }, outputs: { displayedActionsCount: "displayedActionsCount", isEditing: "isEditing" }, usesOnChanges: true, ngImport: i0, template: "<div\n #actionBox\n data-testid=\"action-box\"\n class=\"str-chat__message-actions-box str-chat__message-actions-box-angular\"\n [class.str-chat__message-actions-box--open]=\"true\"\n>\n <ul class=\"str-chat__message-actions-list\">\n <ng-container\n *ngFor=\"let item of visibleMessageActionItems; trackBy: trackByActionName\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageActionsBoxItemTemplate$ | async) ||\n defaultMessageActionItem;\n context: getMessageActionTemplateContext(item)\n \"\n ></ng-container>\n </ng-container>\n </ul>\n</div>\n\n<ng-template\n #defaultMessageActionItem\n let-actionName=\"actionName\"\n let-actionHandler=\"actionHandler\"\n let-actionLabelOrTranslationKey=\"actionLabelOrTranslationKey\"\n>\n <button\n class=\"str-chat__message-actions-list-item-button\"\n [attr.data-testid]=\"actionName + '-action'\"\n (click)=\"actionHandler(message, isMine)\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ getActionLabel(actionLabelOrTranslationKey) | translate }}\n </li>\n </button>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }] });
112
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageActionsBoxComponent, decorators: [{
113
+ type: Component,
114
+ args: [{ selector: 'stream-message-actions-box', template: "<div\n #actionBox\n data-testid=\"action-box\"\n class=\"str-chat__message-actions-box str-chat__message-actions-box-angular\"\n [class.str-chat__message-actions-box--open]=\"true\"\n>\n <ul class=\"str-chat__message-actions-list\">\n <ng-container\n *ngFor=\"let item of visibleMessageActionItems; trackBy: trackByActionName\"\n >\n <ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.messageActionsBoxItemTemplate$ | async) ||\n defaultMessageActionItem;\n context: getMessageActionTemplateContext(item)\n \"\n ></ng-container>\n </ng-container>\n </ul>\n</div>\n\n<ng-template\n #defaultMessageActionItem\n let-actionName=\"actionName\"\n let-actionHandler=\"actionHandler\"\n let-actionLabelOrTranslationKey=\"actionLabelOrTranslationKey\"\n>\n <button\n class=\"str-chat__message-actions-list-item-button\"\n [attr.data-testid]=\"actionName + '-action'\"\n (click)=\"actionHandler(message, isMine)\"\n >\n <li class=\"str-chat__message-actions-list-item\">\n {{ getActionLabel(actionLabelOrTranslationKey) | translate }}\n </li>\n </button>\n</ng-template>\n" }]
115
+ }], ctorParameters: function () { return [{ type: i1.CustomTemplatesService }, { type: i2.MessageActionsService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { isOpen: [{
116
+ type: Input
117
+ }], isMine: [{
118
+ type: Input
119
+ }], message: [{
120
+ type: Input
121
+ }], enabledActions: [{
122
+ type: Input
123
+ }], customActions: [{
124
+ type: Input
125
+ }], displayedActionsCount: [{
126
+ type: Output
127
+ }], isEditing: [{
128
+ type: Output
129
+ }] } });
130
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-actions-box.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-actions-box/message-actions-box.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-actions-box/message-actions-box.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  Output,\n  SimpleChanges,\n  TemplateRef,\n} from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport {\n  CustomMessageActionItem,\n  MessageActionBoxItemContext,\n  MessageActionItem,\n  MessageInputContext,\n  StreamMessage,\n} from '../types';\nimport { MessageActionsService } from '../message-actions.service';\n/**\n * The `MessageActionsBox` component displays a list of message actions (i.e edit), that can be opened or closed. You can find the [list of the supported actions](../concepts/message-interactions.mdx) in the message interaction guide.\n */\n@Component({\n  selector: 'stream-message-actions-box',\n  templateUrl: './message-actions-box.component.html',\n  styles: [],\n})\nexport class MessageActionsBoxComponent\n  implements OnInit, OnChanges, OnDestroy, AfterViewInit\n{\n  /**\n   * Indicates if the list should be opened or closed. Adding a UI element to open and close the list is the parent's component responsibility.\n   * @deprecated No need for this since [theme-v2](../theming/introduction.mdx)\n   */\n  @Input() isOpen = false;\n  /**\n   * Indicates if the message actions are belonging to a message that was sent by the current user or not.\n   */\n  @Input() isMine = false;\n  /**\n   * The message the actions will be executed on\n   */\n  @Input() message: StreamMessage | undefined;\n  /**\n   * The list of [channel capabilities](https://getstream.io/chat/docs/javascript/channel_capabilities/?language=javascript) that are enabled for the current user, the list of [supported interactions](../concepts/message-interactions.mdx) can be found in our message interaction guide. Unathorized actions won't be displayed on the UI.\n   */\n  @Input() enabledActions: string[] = [];\n  /**\n   * A list of custom message actions to be displayed in the action box\n   *\n   * In the next major release this will be released with `messageReactionsService.customActions$`\n   *\n   * More information: https://getstream.io/chat/docs/sdk/angular/services/MessageActionsService\n   */\n  @Input() customActions: CustomMessageActionItem[] = [];\n  /**\n   * The number of authorized actions (it can be less or equal than the number of enabled actions)\n   * @deprecated components should use `messageReactionsService.getAuthorizedMessageActionsCount` method\n   *\n   * More information: https://getstream.io/chat/docs/sdk/angular/services/MessageActionsService\n   */\n  @Output() readonly displayedActionsCount = new EventEmitter<number>();\n  /**\n   * An event which emits `true` if the edit message modal is open, and `false` when it is closed.\n   * @deprecated components should use `messageReactionsService.messageToEdit$` Observable\n   *\n   * More information: https://getstream.io/chat/docs/sdk/angular/services/MessageActionsService\n   */\n  @Output() readonly isEditing = new EventEmitter<boolean>();\n  messageInputTemplate: TemplateRef<MessageInputContext> | undefined;\n  messageActionItemTemplate:\n    | TemplateRef<MessageActionBoxItemContext>\n    | undefined;\n  visibleMessageActionItems: (MessageActionItem | CustomMessageActionItem)[] =\n    [];\n  isEditModalOpen = false;\n  private readonly messageActionItems: MessageActionItem[];\n  private subscriptions: Subscription[] = [];\n  private isViewInited = false;\n  constructor(\n    public readonly customTemplatesService: CustomTemplatesService,\n    private messageActionsService: MessageActionsService,\n    private cdRef: ChangeDetectorRef\n  ) {\n    this.messageActionItems = this.messageActionsService.defaultActions;\n  }\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.messageActionsService.messageToEdit$.subscribe((m) => {\n        let isEditModalOpen = false;\n        if (m && m.id === this.message?.id) {\n          isEditModalOpen = true;\n        }\n        if (isEditModalOpen !== this.isEditModalOpen) {\n          this.isEditModalOpen = isEditModalOpen;\n          this.isEditing.emit(this.isEditModalOpen);\n          if (this.isViewInited) {\n            this.cdRef.detectChanges();\n          }\n        }\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (\n      changes.isMine ||\n      changes.enabledActions ||\n      changes.message ||\n      changes.customActions\n    ) {\n      this.setVisibleActions();\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  getActionLabel(\n    actionLabelOrTranslationKey: ((message: StreamMessage) => string) | string\n  ) {\n    return typeof actionLabelOrTranslationKey === 'string'\n      ? actionLabelOrTranslationKey\n      : actionLabelOrTranslationKey(this.message!);\n  }\n\n  getMessageActionTemplateContext(\n    item: MessageActionItem | CustomMessageActionItem\n  ): MessageActionBoxItemContext {\n    return {\n      actionHandler: item.actionHandler,\n      isMine: this.isMine,\n      actionName: item.actionName,\n      message: this.message!,\n      actionLabelOrTranslationKey: item.actionLabelOrTranslationKey,\n    };\n  }\n\n  trackByActionName(\n    _: number,\n    item: MessageActionItem | CustomMessageActionItem\n  ) {\n    return item.actionName;\n  }\n\n  private setVisibleActions() {\n    this.visibleMessageActionItems = [\n      ...this.messageActionItems,\n      ...this.customActions,\n    ].filter((item) =>\n      item.isVisible(this.enabledActions, this.isMine, this.message!)\n    );\n    this.displayedActionsCount.emit(this.visibleMessageActionItems.length);\n  }\n}\n","<div\n  #actionBox\n  data-testid=\"action-box\"\n  class=\"str-chat__message-actions-box str-chat__message-actions-box-angular\"\n  [class.str-chat__message-actions-box--open]=\"true\"\n>\n  <ul class=\"str-chat__message-actions-list\">\n    <ng-container\n      *ngFor=\"let item of visibleMessageActionItems; trackBy: trackByActionName\"\n    >\n      <ng-container\n        *ngTemplateOutlet=\"\n          (customTemplatesService.messageActionsBoxItemTemplate$ | async) ||\n            defaultMessageActionItem;\n          context: getMessageActionTemplateContext(item)\n        \"\n      ></ng-container>\n    </ng-container>\n  </ul>\n</div>\n\n<ng-template\n  #defaultMessageActionItem\n  let-actionName=\"actionName\"\n  let-actionHandler=\"actionHandler\"\n  let-actionLabelOrTranslationKey=\"actionLabelOrTranslationKey\"\n>\n  <button\n    class=\"str-chat__message-actions-list-item-button\"\n    [attr.data-testid]=\"actionName + '-action'\"\n    (click)=\"actionHandler(message, isMine)\"\n  >\n    <li class=\"str-chat__message-actions-list-item\">\n      {{ getActionLabel(actionLabelOrTranslationKey) | translate }}\n    </li>\n  </button>\n</ng-template>\n"]}
@@ -0,0 +1,119 @@
1
+ import { Injectable } from '@angular/core';
2
+ import { BehaviorSubject } from 'rxjs';
3
+ import * as i0 from "@angular/core";
4
+ import * as i1 from "./chat-client.service";
5
+ import * as i2 from "./notification.service";
6
+ import * as i3 from "./channel.service";
7
+ /**
8
+ * The message actions service provides customization options for the [message actions](../../components/MessageActionsBoxComponent)
9
+ */
10
+ export class MessageActionsService {
11
+ constructor(chatClientService, notificationService, channelService) {
12
+ this.chatClientService = chatClientService;
13
+ this.notificationService = notificationService;
14
+ this.channelService = channelService;
15
+ /**
16
+ * Default actions - these are the actions that are handled by the built-in component
17
+ */
18
+ this.defaultActions = [
19
+ {
20
+ actionName: 'mark-unread',
21
+ actionLabelOrTranslationKey: 'streamChat.Mark as unread',
22
+ actionHandler: (message) => {
23
+ void this.channelService.markMessageUnread(message.id);
24
+ },
25
+ isVisible: (enabledActions, isMine, message) => enabledActions.indexOf('read-events') !== -1 && !message.parent_id,
26
+ },
27
+ {
28
+ actionName: 'quote',
29
+ actionLabelOrTranslationKey: 'streamChat.Reply',
30
+ actionHandler: (message) => {
31
+ this.channelService.selectMessageToQuote(message);
32
+ },
33
+ isVisible: (enabledActions) => enabledActions.indexOf('quote-message') !== -1,
34
+ },
35
+ {
36
+ actionName: 'pin',
37
+ actionLabelOrTranslationKey: (message) => message.pinned ? 'streamChat.Unpin' : 'streamChat.Pin',
38
+ actionHandler: (message) => {
39
+ message.pinned
40
+ ? void this.channelService.unpinMessage(message)
41
+ : void this.channelService.pinMessage(message);
42
+ },
43
+ isVisible: (enabledActions) => enabledActions.indexOf('pin-message') !== -1,
44
+ },
45
+ {
46
+ actionName: 'flag',
47
+ actionLabelOrTranslationKey: 'streamChat.Flag',
48
+ // eslint-disable-next-line @typescript-eslint/no-misused-promises
49
+ actionHandler: async (message) => {
50
+ try {
51
+ await this.chatClientService.flagMessage(message.id);
52
+ this.notificationService.addTemporaryNotification('streamChat.Message has been successfully flagged', 'success');
53
+ }
54
+ catch (err) {
55
+ this.notificationService.addTemporaryNotification('streamChat.Error adding flag');
56
+ }
57
+ },
58
+ isVisible: (enabledActions, isMine) => enabledActions.indexOf('flag-message') !== -1 && !isMine,
59
+ },
60
+ {
61
+ actionName: 'edit',
62
+ actionLabelOrTranslationKey: 'streamChat.Edit Message',
63
+ actionHandler: (message) => {
64
+ this.messageToEdit$.next(message);
65
+ },
66
+ isVisible: (enabledActions, isMine) => (enabledActions.indexOf('update-own-message') !== -1 && isMine) ||
67
+ enabledActions.indexOf('update-any-message') !== -1,
68
+ },
69
+ {
70
+ actionName: 'delete',
71
+ actionLabelOrTranslationKey: 'streamChat.Delete',
72
+ // eslint-disable-next-line @typescript-eslint/no-misused-promises
73
+ actionHandler: async (message) => {
74
+ try {
75
+ await this.channelService.deleteMessage(message);
76
+ }
77
+ catch (error) {
78
+ this.notificationService.addTemporaryNotification('streamChat.Error deleting message');
79
+ }
80
+ },
81
+ isVisible: (enabledActions, isMine) => ((enabledActions.indexOf('delete') !== -1 ||
82
+ enabledActions.indexOf('delete-own-message') !== -1) &&
83
+ isMine) ||
84
+ enabledActions.indexOf('delete-any') !== -1 ||
85
+ enabledActions.indexOf('delete-any-message') !== -1,
86
+ },
87
+ ];
88
+ /**
89
+ * The built-in components will handle changes to this observable.
90
+ */
91
+ this.messageToEdit$ = new BehaviorSubject(undefined);
92
+ /**
93
+ * You can pass your own custom actions that will be displayed inside the built-in message actions component
94
+ */
95
+ this.customActions$ = new BehaviorSubject([]);
96
+ }
97
+ /**
98
+ * This method returns how many authorized actions are available to the given message
99
+ * @param message
100
+ * @param enabledActions
101
+ * @returns the count
102
+ */
103
+ getAuthorizedMessageActionsCount(message, enabledActions) {
104
+ const customActions = this.customActions$.getValue() || [];
105
+ const allActions = [...this.defaultActions, ...customActions];
106
+ const currentUserId = this.chatClientService.chatClient.user?.id;
107
+ const isMine = message.user_id === currentUserId;
108
+ return allActions.filter((item) => item.isVisible(enabledActions, isMine, message)).length;
109
+ }
110
+ }
111
+ MessageActionsService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageActionsService, deps: [{ token: i1.ChatClientService }, { token: i2.NotificationService }, { token: i3.ChannelService }], target: i0.ɵɵFactoryTarget.Injectable });
112
+ MessageActionsService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageActionsService, providedIn: 'root' });
113
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageActionsService, decorators: [{
114
+ type: Injectable,
115
+ args: [{
116
+ providedIn: 'root',
117
+ }]
118
+ }], ctorParameters: function () { return [{ type: i1.ChatClientService }, { type: i2.NotificationService }, { type: i3.ChannelService }]; } });
119
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-actions.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/message-actions.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport {\n  CustomMessageActionItem,\n  DefaultStreamChatGenerics,\n  MessageActionItem,\n  MessageActionsClickDetails,\n  StreamMessage,\n} from './types';\nimport { ChatClientService } from './chat-client.service';\nimport { NotificationService } from './notification.service';\nimport { ChannelService } from './channel.service';\n\n/**\n * The message actions service provides customization options for the [message actions](../../components/MessageActionsBoxComponent)\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class MessageActionsService<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> {\n  /**\n   * Default actions - these are the actions that are handled by the built-in component\n   */\n  readonly defaultActions: MessageActionItem<T>[] = [\n    {\n      actionName: 'mark-unread',\n      actionLabelOrTranslationKey: 'streamChat.Mark as unread',\n      actionHandler: (message: StreamMessage<T>) => {\n        void this.channelService.markMessageUnread(message.id);\n      },\n      isVisible: (\n        enabledActions: string[],\n        isMine: boolean,\n        message: StreamMessage<T>\n      ) => enabledActions.indexOf('read-events') !== -1 && !message.parent_id,\n    },\n    {\n      actionName: 'quote',\n      actionLabelOrTranslationKey: 'streamChat.Reply',\n      actionHandler: (message: StreamMessage<T>) => {\n        this.channelService.selectMessageToQuote(message);\n      },\n      isVisible: (enabledActions: string[]) =>\n        enabledActions.indexOf('quote-message') !== -1,\n    },\n    {\n      actionName: 'pin',\n      actionLabelOrTranslationKey: (message: StreamMessage<T>) =>\n        message.pinned ? 'streamChat.Unpin' : 'streamChat.Pin',\n      actionHandler: (message: StreamMessage<T>) => {\n        message.pinned\n          ? void this.channelService.unpinMessage(message)\n          : void this.channelService.pinMessage(message);\n      },\n      isVisible: (enabledActions: string[]) =>\n        enabledActions.indexOf('pin-message') !== -1,\n    },\n    {\n      actionName: 'flag',\n      actionLabelOrTranslationKey: 'streamChat.Flag',\n      // eslint-disable-next-line @typescript-eslint/no-misused-promises\n      actionHandler: async (message: StreamMessage<T>) => {\n        try {\n          await this.chatClientService.flagMessage(message.id);\n          this.notificationService.addTemporaryNotification(\n            'streamChat.Message has been successfully flagged',\n            'success'\n          );\n        } catch (err) {\n          this.notificationService.addTemporaryNotification(\n            'streamChat.Error adding flag'\n          );\n        }\n      },\n      isVisible: (enabledActions: string[], isMine: boolean) =>\n        enabledActions.indexOf('flag-message') !== -1 && !isMine,\n    },\n    {\n      actionName: 'edit',\n      actionLabelOrTranslationKey: 'streamChat.Edit Message',\n      actionHandler: (message: StreamMessage<T>) => {\n        this.messageToEdit$.next(message);\n      },\n      isVisible: (enabledActions: string[], isMine: boolean) =>\n        (enabledActions.indexOf('update-own-message') !== -1 && isMine) ||\n        enabledActions.indexOf('update-any-message') !== -1,\n    },\n    {\n      actionName: 'delete',\n      actionLabelOrTranslationKey: 'streamChat.Delete',\n      // eslint-disable-next-line @typescript-eslint/no-misused-promises\n      actionHandler: async (message: StreamMessage<T>) => {\n        try {\n          await this.channelService.deleteMessage(message);\n        } catch (error) {\n          this.notificationService.addTemporaryNotification(\n            'streamChat.Error deleting message'\n          );\n        }\n      },\n      isVisible: (enabledActions: string[], isMine: boolean) =>\n        ((enabledActions.indexOf('delete') !== -1 ||\n          enabledActions.indexOf('delete-own-message') !== -1) &&\n          isMine) ||\n        enabledActions.indexOf('delete-any') !== -1 ||\n        enabledActions.indexOf('delete-any-message') !== -1,\n    },\n  ];\n  /**\n   * The built-in components will handle changes to this observable.\n   */\n  messageToEdit$ = new BehaviorSubject<StreamMessage<T> | undefined>(undefined);\n  /**\n   * You can pass your own custom actions that will be displayed inside the built-in message actions component\n   */\n  customActions$ = new BehaviorSubject<CustomMessageActionItem[]>([]);\n  /**\n   * By default the [`MessageComponent`](../../components/MessageComponent) will display the [`MessageActionsBoxComponent`](../../components/MessageActionsBoxComponent). You can override that behavior by providing your own event handler.\n   */\n  customActionClickHandler?: (details: MessageActionsClickDetails<T>) => void;\n\n  constructor(\n    private chatClientService: ChatClientService,\n    private notificationService: NotificationService,\n    private channelService: ChannelService\n  ) {}\n\n  /**\n   * This method returns how many authorized actions are available to the given message\n   * @param message\n   * @param enabledActions\n   * @returns the count\n   */\n  getAuthorizedMessageActionsCount(\n    message: StreamMessage<T>,\n    enabledActions: string[]\n  ) {\n    const customActions = this.customActions$.getValue() || [];\n    const allActions = [...this.defaultActions, ...customActions];\n    const currentUserId = this.chatClientService.chatClient.user?.id;\n    const isMine = message.user_id === currentUserId;\n\n    return allActions.filter((item) =>\n      item.isVisible(enabledActions, isMine, message)\n    ).length;\n  }\n}\n"]}
@@ -0,0 +1,71 @@
1
+ import { Component, HostBinding } from '@angular/core';
2
+ import * as i0 from "@angular/core";
3
+ import * as i1 from "../channel.service";
4
+ import * as i2 from "../custom-templates.service";
5
+ import * as i3 from "../message-actions.service";
6
+ import * as i4 from "@angular/common";
7
+ import * as i5 from "../modal/modal.component";
8
+ import * as i6 from "@ngx-translate/core";
9
+ /**
10
+ * The component watches for the [`channelService.bouncedMessage$` stream](../../services/ChannelService/#bouncedmessage) and opens the bounce modal if a message is emitted.
11
+ *
12
+ * To bounce messages, you need to set up [semantic filters for moderation](https://getstream.io/automated-moderation/docs/automod_configuration/?q=semantic%20filters).
13
+ */
14
+ export class MessageBouncePromptComponent {
15
+ constructor(channelService, customTemplatesService, messageActionsService) {
16
+ this.channelService = channelService;
17
+ this.customTemplatesService = customTemplatesService;
18
+ this.messageActionsService = messageActionsService;
19
+ this.class = 'str-chat__message-bounce-prompt';
20
+ this.isModalOpen = false;
21
+ this.subscriptions = [];
22
+ this.messageBounceModalOpenChanged = (isOpen) => {
23
+ this.isModalOpen = isOpen;
24
+ if (!isOpen) {
25
+ this.message = undefined;
26
+ this.channelService.bouncedMessage$.next(undefined);
27
+ }
28
+ };
29
+ this.subscriptions.push(this.channelService.bouncedMessage$.subscribe((m) => {
30
+ if (m !== this.message) {
31
+ this.message = m;
32
+ if (this.message) {
33
+ this.isModalOpen = true;
34
+ }
35
+ }
36
+ }));
37
+ }
38
+ ngOnDestroy() {
39
+ this.subscriptions.forEach((s) => s.unsubscribe());
40
+ }
41
+ async resendMessage() {
42
+ this.isModalOpen = false;
43
+ await this.channelService.resendMessage(this.message);
44
+ this.message = undefined;
45
+ this.channelService.bouncedMessage$.next(undefined);
46
+ }
47
+ async deleteMessage() {
48
+ if (!this.message) {
49
+ return;
50
+ }
51
+ this.isModalOpen = false;
52
+ await this.channelService.deleteMessage(this.message, true);
53
+ this.message = undefined;
54
+ this.channelService.bouncedMessage$.next(undefined);
55
+ }
56
+ editMessage() {
57
+ this.isModalOpen = false;
58
+ this.messageActionsService.messageToEdit$.next(this.message);
59
+ this.message = undefined;
60
+ this.channelService.bouncedMessage$.next(undefined);
61
+ }
62
+ }
63
+ MessageBouncePromptComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageBouncePromptComponent, deps: [{ token: i1.ChannelService }, { token: i2.CustomTemplatesService }, { token: i3.MessageActionsService }], target: i0.ɵɵFactoryTarget.Component });
64
+ MessageBouncePromptComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: MessageBouncePromptComponent, selector: "stream-message-bounce-prompt", host: { properties: { "class": "this.class" } }, ngImport: i0, template: "<ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.modalTemplate$ | async) || defaultModal;\n context: {\n message: message,\n isOpen: isModalOpen,\n isOpenChangeHandler: messageBounceModalOpenChanged,\n content: modalContent\n }\n \"\n></ng-container>\n\n<ng-template\n #defaultModal\n let-isOpen=\"isOpen\"\n let-isOpenChangeHandler=\"isOpenChangeHandler\"\n let-content=\"content\"\n>\n <stream-modal\n *ngIf=\"isOpen\"\n [isOpen]=\"isOpen\"\n [content]=\"content\"\n (isOpenChange)=\"isOpenChangeHandler($event)\"\n >\n </stream-modal>\n</ng-template>\n\n<ng-template #modalContent>\n <div\n class=\"str-chat__message-bounce-prompt\"\n data-testid=\"message-bounce-prompt\"\n >\n <div class=\"str-chat__message-bounce-prompt-header\">\n {{\n \"streamChat.This message did not meet our content guidelines\"\n | translate\n }}\n </div>\n <div class=\"str-chat__message-bounce-actions\">\n <button\n class=\"str-chat__message-bounce-edit\"\n data-testid=\"message-bounce-edit\"\n type=\"button\"\n (click)=\"editMessage()\"\n (keyup.enter)=\"editMessage()\"\n >\n {{ \"streamChat.Edit Message\" | translate }}\n </button>\n <button\n class=\"str-chat__message-bounce-send\"\n data-testid=\"message-bounce-send\"\n (click)=\"resendMessage()\"\n (keyup.enter)=\"resendMessage()\"\n >\n {{ \"streamChat.Send Anyway\" | translate }}\n </button>\n <button\n class=\"str-chat__message-bounce-delete\"\n data-testid=\"message-bounce-delete\"\n (click)=\"deleteMessage()\"\n (keyup.enter)=\"deleteMessage()\"\n >\n {{ \"streamChat.Delete\" | translate }}\n </button>\n </div>\n </div>\n</ng-template>\n", dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "component", type: i5.ModalComponent, selector: "stream-modal", inputs: ["isOpen", "content"], outputs: ["isOpenChange"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i6.TranslatePipe, name: "translate" }] });
65
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: MessageBouncePromptComponent, decorators: [{
66
+ type: Component,
67
+ args: [{ selector: 'stream-message-bounce-prompt', template: "<ng-container\n *ngTemplateOutlet=\"\n (customTemplatesService.modalTemplate$ | async) || defaultModal;\n context: {\n message: message,\n isOpen: isModalOpen,\n isOpenChangeHandler: messageBounceModalOpenChanged,\n content: modalContent\n }\n \"\n></ng-container>\n\n<ng-template\n #defaultModal\n let-isOpen=\"isOpen\"\n let-isOpenChangeHandler=\"isOpenChangeHandler\"\n let-content=\"content\"\n>\n <stream-modal\n *ngIf=\"isOpen\"\n [isOpen]=\"isOpen\"\n [content]=\"content\"\n (isOpenChange)=\"isOpenChangeHandler($event)\"\n >\n </stream-modal>\n</ng-template>\n\n<ng-template #modalContent>\n <div\n class=\"str-chat__message-bounce-prompt\"\n data-testid=\"message-bounce-prompt\"\n >\n <div class=\"str-chat__message-bounce-prompt-header\">\n {{\n \"streamChat.This message did not meet our content guidelines\"\n | translate\n }}\n </div>\n <div class=\"str-chat__message-bounce-actions\">\n <button\n class=\"str-chat__message-bounce-edit\"\n data-testid=\"message-bounce-edit\"\n type=\"button\"\n (click)=\"editMessage()\"\n (keyup.enter)=\"editMessage()\"\n >\n {{ \"streamChat.Edit Message\" | translate }}\n </button>\n <button\n class=\"str-chat__message-bounce-send\"\n data-testid=\"message-bounce-send\"\n (click)=\"resendMessage()\"\n (keyup.enter)=\"resendMessage()\"\n >\n {{ \"streamChat.Send Anyway\" | translate }}\n </button>\n <button\n class=\"str-chat__message-bounce-delete\"\n data-testid=\"message-bounce-delete\"\n (click)=\"deleteMessage()\"\n (keyup.enter)=\"deleteMessage()\"\n >\n {{ \"streamChat.Delete\" | translate }}\n </button>\n </div>\n </div>\n</ng-template>\n" }]
68
+ }], ctorParameters: function () { return [{ type: i1.ChannelService }, { type: i2.CustomTemplatesService }, { type: i3.MessageActionsService }]; }, propDecorators: { class: [{
69
+ type: HostBinding
70
+ }] } });
71
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"message-bounce-prompt.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/message-bounce-prompt/message-bounce-prompt.component.ts","../../../../../projects/stream-chat-angular/src/lib/message-bounce-prompt/message-bounce-prompt.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,WAAW,EAAa,MAAM,eAAe,CAAC;;;;;;;;AAOlE;;;;GAIG;AAMH,MAAM,OAAO,4BAA4B;IAMvC,YACU,cAA8B,EAC7B,sBAA8C,EAC/C,qBAA4C;QAF5C,mBAAc,GAAd,cAAc,CAAgB;QAC7B,2BAAsB,GAAtB,sBAAsB,CAAwB;QAC/C,0BAAqB,GAArB,qBAAqB,CAAuB;QARvC,UAAK,GAAG,iCAAiC,CAAC;QACzD,gBAAW,GAAG,KAAK,CAAC;QAEZ,kBAAa,GAAmB,EAAE,CAAC;QAmB3C,kCAA6B,GAAG,CAAC,MAAe,EAAE,EAAE;YAClD,IAAI,CAAC,WAAW,GAAG,MAAM,CAAC;YAC1B,IAAI,CAAC,MAAM,EAAE;gBACX,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC;gBACzB,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;aACrD;QACH,CAAC,CAAC;QAlBA,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;YAClD,IAAI,CAAC,KAAK,IAAI,CAAC,OAAO,EAAE;gBACtB,IAAI,CAAC,OAAO,GAAG,CAAC,CAAC;gBACjB,IAAI,IAAI,CAAC,OAAO,EAAE;oBAChB,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC;iBACzB;aACF;QACH,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAUD,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;IACrD,CAAC;IAED,KAAK,CAAC,aAAa;QACjB,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,MAAM,IAAI,CAAC,cAAc,CAAC,aAAa,CAAC,IAAI,CAAC,OAAQ,CAAC,CAAC;QACvD,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC;QACzB,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACtD,CAAC;IAED,KAAK,CAAC,aAAa;QACjB,IAAI,CAAC,IAAI,CAAC,OAAO,EAAE;YACjB,OAAO;SACR;QACD,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,MAAM,IAAI,CAAC,cAAc,CAAC,aAAa,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC,CAAC;QAC5D,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC;QACzB,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACtD,CAAC;IAED,WAAW;QACT,IAAI,CAAC,WAAW,GAAG,KAAK,CAAC;QACzB,IAAI,CAAC,qBAAqB,CAAC,cAAc,CAAC,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAC7D,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC;QACzB,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACtD,CAAC;;yHAzDU,4BAA4B;6GAA5B,4BAA4B,qHCjBzC,u0DAmEA;2FDlDa,4BAA4B;kBALxC,SAAS;+BACE,8BAA8B;8KAKzB,KAAK;sBAAnB,WAAW","sourcesContent":["import { Component, HostBinding, OnDestroy } from '@angular/core';\nimport { ChannelService } from '../channel.service';\nimport { CustomTemplatesService } from '../custom-templates.service';\nimport { Subscription } from 'rxjs';\nimport { MessageActionsService } from '../message-actions.service';\nimport { StreamMessage } from '../types';\n\n/**\n * The component watches for the [`channelService.bouncedMessage$` stream](../../services/ChannelService/#bouncedmessage) and opens the bounce modal if a message is emitted.\n *\n * To bounce messages, you need to set up [semantic filters for moderation](https://getstream.io/automated-moderation/docs/automod_configuration/?q=semantic%20filters).\n */\n@Component({\n  selector: 'stream-message-bounce-prompt',\n  templateUrl: './message-bounce-prompt.component.html',\n  styles: [],\n})\nexport class MessageBouncePromptComponent implements OnDestroy {\n  @HostBinding() class = 'str-chat__message-bounce-prompt';\n  isModalOpen = false;\n  message?: StreamMessage;\n  private subscriptions: Subscription[] = [];\n\n  constructor(\n    private channelService: ChannelService,\n    readonly customTemplatesService: CustomTemplatesService,\n    private messageActionsService: MessageActionsService\n  ) {\n    this.subscriptions.push(\n      this.channelService.bouncedMessage$.subscribe((m) => {\n        if (m !== this.message) {\n          this.message = m;\n          if (this.message) {\n            this.isModalOpen = true;\n          }\n        }\n      })\n    );\n  }\n\n  messageBounceModalOpenChanged = (isOpen: boolean) => {\n    this.isModalOpen = isOpen;\n    if (!isOpen) {\n      this.message = undefined;\n      this.channelService.bouncedMessage$.next(undefined);\n    }\n  };\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  async resendMessage() {\n    this.isModalOpen = false;\n    await this.channelService.resendMessage(this.message!);\n    this.message = undefined;\n    this.channelService.bouncedMessage$.next(undefined);\n  }\n\n  async deleteMessage() {\n    if (!this.message) {\n      return;\n    }\n    this.isModalOpen = false;\n    await this.channelService.deleteMessage(this.message, true);\n    this.message = undefined;\n    this.channelService.bouncedMessage$.next(undefined);\n  }\n\n  editMessage() {\n    this.isModalOpen = false;\n    this.messageActionsService.messageToEdit$.next(this.message);\n    this.message = undefined;\n    this.channelService.bouncedMessage$.next(undefined);\n  }\n}\n","<ng-container\n  *ngTemplateOutlet=\"\n    (customTemplatesService.modalTemplate$ | async) || defaultModal;\n    context: {\n      message: message,\n      isOpen: isModalOpen,\n      isOpenChangeHandler: messageBounceModalOpenChanged,\n      content: modalContent\n    }\n  \"\n></ng-container>\n\n<ng-template\n  #defaultModal\n  let-isOpen=\"isOpen\"\n  let-isOpenChangeHandler=\"isOpenChangeHandler\"\n  let-content=\"content\"\n>\n  <stream-modal\n    *ngIf=\"isOpen\"\n    [isOpen]=\"isOpen\"\n    [content]=\"content\"\n    (isOpenChange)=\"isOpenChangeHandler($event)\"\n  >\n  </stream-modal>\n</ng-template>\n\n<ng-template #modalContent>\n  <div\n    class=\"str-chat__message-bounce-prompt\"\n    data-testid=\"message-bounce-prompt\"\n  >\n    <div class=\"str-chat__message-bounce-prompt-header\">\n      {{\n        \"streamChat.This message did not meet our content guidelines\"\n          | translate\n      }}\n    </div>\n    <div class=\"str-chat__message-bounce-actions\">\n      <button\n        class=\"str-chat__message-bounce-edit\"\n        data-testid=\"message-bounce-edit\"\n        type=\"button\"\n        (click)=\"editMessage()\"\n        (keyup.enter)=\"editMessage()\"\n      >\n        {{ \"streamChat.Edit Message\" | translate }}\n      </button>\n      <button\n        class=\"str-chat__message-bounce-send\"\n        data-testid=\"message-bounce-send\"\n        (click)=\"resendMessage()\"\n        (keyup.enter)=\"resendMessage()\"\n      >\n        {{ \"streamChat.Send Anyway\" | translate }}\n      </button>\n      <button\n        class=\"str-chat__message-bounce-delete\"\n        data-testid=\"message-bounce-delete\"\n        (click)=\"deleteMessage()\"\n        (keyup.enter)=\"deleteMessage()\"\n      >\n        {{ \"streamChat.Delete\" | translate }}\n      </button>\n    </div>\n  </div>\n</ng-template>\n"]}