stream-chat-angular 6.0.0-beta.5 → 6.0.0-rc.2

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 (202) hide show
  1. package/README.md +2 -2
  2. package/assets/i18n/en.d.ts +0 -1
  3. package/assets/version.d.ts +1 -1
  4. package/{esm2022 → esm2020}/assets/i18n/en.mjs +1 -2
  5. package/{esm2022 → esm2020}/assets/version.mjs +2 -2
  6. package/esm2020/lib/attachment-configuration.service.mjs +182 -0
  7. package/esm2020/lib/attachment-list/attachment-list.component.mjs +232 -0
  8. package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +55 -0
  9. package/esm2020/lib/attachment.service.mjs +481 -0
  10. package/esm2020/lib/avatar/avatar.component.mjs +160 -0
  11. package/esm2020/lib/avatar-placeholder/avatar-placeholder.component.mjs +66 -0
  12. package/esm2020/lib/channel/channel.component.mjs +45 -0
  13. package/esm2020/lib/channel-header/channel-header.component.mjs +72 -0
  14. package/esm2020/lib/channel-list/channel-list.component.mjs +47 -0
  15. package/esm2020/lib/channel-preview/channel-preview.component.mjs +155 -0
  16. package/esm2020/lib/channel-query.mjs +77 -0
  17. package/esm2020/lib/channel.service.mjs +1561 -0
  18. package/esm2020/lib/chat-client.service.mjs +233 -0
  19. package/esm2020/lib/custom-templates.service.mjs +244 -0
  20. package/{esm2022 → esm2020}/lib/date-parser.service.mjs +5 -5
  21. package/esm2020/lib/file-utils.mjs +35 -0
  22. package/{esm2022 → esm2020}/lib/get-channel-display-text.mjs +1 -1
  23. package/esm2020/lib/get-message-translation.mjs +12 -0
  24. package/esm2020/lib/icon/icon-placeholder/icon-placeholder.component.mjs +28 -0
  25. package/{esm2022 → esm2020}/lib/icon/icon.component.mjs +5 -5
  26. package/{esm2022 → esm2020}/lib/icon/icon.module.mjs +11 -11
  27. package/{esm2022 → esm2020}/lib/icon/loading-indicator/loading-indicator.component.mjs +5 -5
  28. package/esm2020/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +20 -0
  29. package/{esm2022 → esm2020}/lib/list-users.mjs +1 -1
  30. package/esm2020/lib/message/message.component.mjs +486 -0
  31. package/esm2020/lib/message-actions-box/message-actions-box.component.mjs +123 -0
  32. package/esm2020/lib/message-actions.service.mjs +187 -0
  33. package/esm2020/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +71 -0
  34. package/esm2020/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +333 -0
  35. package/{esm2022 → esm2020}/lib/message-input/emoji-input.service.mjs +7 -7
  36. package/esm2020/lib/message-input/message-input-config.service.mjs +50 -0
  37. package/esm2020/lib/message-input/message-input.component.mjs +507 -0
  38. package/{esm2022 → esm2020}/lib/message-input/textarea/textarea.component.mjs +6 -6
  39. package/{esm2022 → esm2020}/lib/message-input/textarea.directive.mjs +5 -5
  40. package/{esm2022 → esm2020}/lib/message-input/voice-recorder.service.mjs +5 -5
  41. package/{esm2022 → esm2020}/lib/message-list/group-styles.mjs +1 -1
  42. package/esm2020/lib/message-list/message-list.component.mjs +717 -0
  43. package/esm2020/lib/message-preview.mjs +21 -0
  44. package/esm2020/lib/message-reactions/message-reactions.component.mjs +168 -0
  45. package/esm2020/lib/message-reactions-selector/message-reactions-selector.component.mjs +61 -0
  46. package/{esm2022 → esm2020}/lib/message-reactions.service.mjs +6 -6
  47. package/esm2020/lib/message-text/message-text.component.mjs +143 -0
  48. package/esm2020/lib/message.service.mjs +43 -0
  49. package/{esm2022 → esm2020}/lib/modal/modal.component.mjs +6 -6
  50. package/esm2020/lib/notification/notification.component.mjs +20 -0
  51. package/esm2020/lib/notification-list/notification-list.component.mjs +36 -0
  52. package/{esm2022 → esm2020}/lib/notification.service.mjs +6 -6
  53. package/esm2020/lib/paginated-list/paginated-list.component.mjs +94 -0
  54. package/{esm2022 → esm2020}/lib/parse-date.mjs +1 -1
  55. package/esm2020/lib/read-by.mjs +12 -0
  56. package/esm2020/lib/stream-autocomplete-textarea.module.mjs +33 -0
  57. package/{esm2022 → esm2020}/lib/stream-avatar.module.mjs +5 -5
  58. package/{esm2022 → esm2020}/lib/stream-chat.module.mjs +59 -59
  59. package/{esm2022 → esm2020}/lib/stream-i18n.service.mjs +6 -6
  60. package/esm2020/lib/stream-textarea.module.mjs +31 -0
  61. package/{esm2022 → esm2020}/lib/theme.service.mjs +6 -6
  62. package/esm2020/lib/thread/thread.component.mjs +51 -0
  63. package/{esm2022 → esm2020}/lib/transliteration.service.mjs +5 -5
  64. package/esm2020/lib/types-custom.mjs +2 -0
  65. package/esm2020/lib/types.mjs +2 -0
  66. package/esm2020/lib/user-list/user-list.component.mjs +47 -0
  67. package/esm2020/lib/virtualized-list.service.mjs +271 -0
  68. package/{esm2022 → esm2020}/lib/virtualized-message-list.service.mjs +1 -1
  69. package/{esm2022 → esm2020}/lib/voice-recorder/amplitude-recorder.service.mjs +5 -5
  70. package/{esm2022 → esm2020}/lib/voice-recorder/audio-recorder.service.mjs +5 -5
  71. package/{esm2022 → esm2020}/lib/voice-recorder/media-recorder.mjs +1 -1
  72. package/esm2020/lib/voice-recorder/mp3-transcoder.mjs +61 -0
  73. package/esm2020/lib/voice-recorder/transcoder.service.mjs +121 -0
  74. package/esm2020/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +35 -0
  75. package/esm2020/lib/voice-recorder/voice-recorder.component.mjs +80 -0
  76. package/{esm2022 → esm2020}/lib/voice-recorder/voice-recorder.module.mjs +9 -9
  77. package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +112 -0
  78. package/esm2020/lib/voice-recording/voice-recording.component.mjs +91 -0
  79. package/{esm2022 → esm2020}/lib/voice-recording/voice-recording.module.mjs +5 -5
  80. package/{esm2022 → esm2020}/lib/wave-form-sampler.mjs +1 -1
  81. package/esm2020/public-api.mjs +86 -0
  82. package/esm2020/stream-chat.mjs +2 -0
  83. package/fesm2015/stream-chat-angular.mjs +9171 -0
  84. package/fesm2015/stream-chat-angular.mjs.map +1 -0
  85. package/{fesm2022 → fesm2020}/stream-chat-angular.mjs +1251 -961
  86. package/fesm2020/stream-chat-angular.mjs.map +1 -0
  87. package/lib/attachment-configuration.service.d.ts +12 -12
  88. package/lib/attachment-list/attachment-list.component.d.ts +15 -10
  89. package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +1 -1
  90. package/lib/attachment.service.d.ts +12 -10
  91. package/lib/avatar/avatar.component.d.ts +7 -7
  92. package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +5 -5
  93. package/lib/channel/channel.component.d.ts +1 -1
  94. package/lib/channel-header/channel-header.component.d.ts +2 -2
  95. package/lib/channel-list/channel-list.component.d.ts +3 -4
  96. package/lib/channel-preview/channel-preview.component.d.ts +6 -6
  97. package/lib/channel-query.d.ts +26 -0
  98. package/lib/channel.service.d.ts +146 -154
  99. package/lib/chat-client.service.d.ts +17 -15
  100. package/lib/custom-templates.service.d.ts +50 -50
  101. package/lib/get-channel-display-text.d.ts +1 -2
  102. package/lib/get-message-translation.d.ts +3 -3
  103. package/lib/icon/icon-placeholder/icon-placeholder.component.d.ts +2 -2
  104. package/lib/icon/icon.component.d.ts +2 -2
  105. package/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.d.ts +1 -1
  106. package/lib/message/message.component.d.ts +6 -6
  107. package/lib/message-actions-box/message-actions-box.component.d.ts +5 -4
  108. package/lib/message-actions.service.d.ts +10 -10
  109. package/lib/message-bounce-prompt/message-bounce-prompt.component.d.ts +1 -1
  110. package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +5 -5
  111. package/lib/message-input/emoji-input.service.d.ts +2 -2
  112. package/lib/message-input/message-input-config.service.d.ts +3 -3
  113. package/lib/message-input/message-input.component.d.ts +10 -33
  114. package/lib/message-input/textarea/textarea.component.d.ts +3 -3
  115. package/lib/message-input/textarea.directive.d.ts +1 -1
  116. package/lib/message-list/group-styles.d.ts +1 -1
  117. package/lib/message-list/message-list.component.d.ts +3 -2
  118. package/lib/message-preview.d.ts +2 -3
  119. package/lib/message-reactions/message-reactions.component.d.ts +9 -8
  120. package/lib/message-reactions-selector/message-reactions-selector.component.d.ts +6 -5
  121. package/lib/message-reactions.service.d.ts +4 -4
  122. package/lib/message-text/message-text.component.d.ts +5 -5
  123. package/lib/message.service.d.ts +6 -7
  124. package/lib/modal/modal.component.d.ts +1 -1
  125. package/lib/notification/notification.component.d.ts +2 -2
  126. package/lib/notification-list/notification-list.component.d.ts +1 -0
  127. package/lib/notification.service.d.ts +1 -1
  128. package/lib/paginated-list/paginated-list.component.d.ts +2 -5
  129. package/lib/read-by.d.ts +1 -2
  130. package/lib/stream-i18n.service.d.ts +1 -1
  131. package/lib/theme.service.d.ts +1 -1
  132. package/lib/thread/thread.component.d.ts +3 -3
  133. package/lib/types-custom.d.ts +15 -0
  134. package/lib/types.d.ts +116 -155
  135. package/lib/user-list/user-list.component.d.ts +2 -3
  136. package/lib/virtualized-message-list.service.d.ts +1 -1
  137. package/lib/voice-recorder/amplitude-recorder.service.d.ts +2 -2
  138. package/lib/voice-recorder/media-recorder.d.ts +2 -2
  139. package/lib/voice-recorder/transcoder.service.d.ts +4 -4
  140. package/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.d.ts +1 -0
  141. package/lib/voice-recorder/voice-recorder.component.d.ts +2 -2
  142. package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +3 -3
  143. package/lib/voice-recording/voice-recording.component.d.ts +2 -3
  144. package/package.json +21 -15
  145. package/public-api.d.ts +3 -0
  146. package/src/assets/i18n/en.ts +0 -1
  147. package/src/assets/version.ts +1 -1
  148. package/esm2022/lib/attachment-configuration.service.mjs +0 -185
  149. package/esm2022/lib/attachment-list/attachment-list.component.mjs +0 -212
  150. package/esm2022/lib/attachment-preview-list/attachment-preview-list.component.mjs +0 -55
  151. package/esm2022/lib/attachment.service.mjs +0 -479
  152. package/esm2022/lib/avatar/avatar.component.mjs +0 -157
  153. package/esm2022/lib/avatar-placeholder/avatar-placeholder.component.mjs +0 -66
  154. package/esm2022/lib/channel/channel.component.mjs +0 -45
  155. package/esm2022/lib/channel-header/channel-header.component.mjs +0 -72
  156. package/esm2022/lib/channel-list/channel-list.component.mjs +0 -50
  157. package/esm2022/lib/channel-preview/channel-preview.component.mjs +0 -150
  158. package/esm2022/lib/channel.service.mjs +0 -1393
  159. package/esm2022/lib/chat-client.service.mjs +0 -227
  160. package/esm2022/lib/custom-templates.service.mjs +0 -244
  161. package/esm2022/lib/file-utils.mjs +0 -35
  162. package/esm2022/lib/get-message-translation.mjs +0 -12
  163. package/esm2022/lib/icon/icon-placeholder/icon-placeholder.component.mjs +0 -28
  164. package/esm2022/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +0 -20
  165. package/esm2022/lib/message/message.component.mjs +0 -486
  166. package/esm2022/lib/message-actions-box/message-actions-box.component.mjs +0 -120
  167. package/esm2022/lib/message-actions.service.mjs +0 -187
  168. package/esm2022/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +0 -71
  169. package/esm2022/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +0 -333
  170. package/esm2022/lib/message-input/message-input-config.service.mjs +0 -50
  171. package/esm2022/lib/message-input/message-input.component.mjs +0 -507
  172. package/esm2022/lib/message-list/message-list.component.mjs +0 -715
  173. package/esm2022/lib/message-preview.mjs +0 -21
  174. package/esm2022/lib/message-reactions/message-reactions.component.mjs +0 -165
  175. package/esm2022/lib/message-reactions-selector/message-reactions-selector.component.mjs +0 -57
  176. package/esm2022/lib/message-text/message-text.component.mjs +0 -143
  177. package/esm2022/lib/message.service.mjs +0 -43
  178. package/esm2022/lib/notification/notification.component.mjs +0 -20
  179. package/esm2022/lib/notification-list/notification-list.component.mjs +0 -33
  180. package/esm2022/lib/paginated-list/paginated-list.component.mjs +0 -94
  181. package/esm2022/lib/read-by.mjs +0 -12
  182. package/esm2022/lib/stream-autocomplete-textarea.module.mjs +0 -33
  183. package/esm2022/lib/stream-textarea.module.mjs +0 -31
  184. package/esm2022/lib/thread/thread.component.mjs +0 -51
  185. package/esm2022/lib/types.mjs +0 -2
  186. package/esm2022/lib/user-list/user-list.component.mjs +0 -47
  187. package/esm2022/lib/virtualized-list.service.mjs +0 -273
  188. package/esm2022/lib/voice-recorder/mp3-transcoder.mjs +0 -61
  189. package/esm2022/lib/voice-recorder/transcoder.service.mjs +0 -121
  190. package/esm2022/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +0 -32
  191. package/esm2022/lib/voice-recorder/voice-recorder.component.mjs +0 -80
  192. package/esm2022/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +0 -112
  193. package/esm2022/lib/voice-recording/voice-recording.component.mjs +0 -91
  194. package/esm2022/public-api.mjs +0 -82
  195. package/fesm2022/stream-chat-angular.mjs.map +0 -1
  196. /package/{esm2022 → esm2020}/lib/format-duration.mjs +0 -0
  197. /package/{esm2022 → esm2020}/lib/injection-tokens.mjs +0 -0
  198. /package/{esm2022 → esm2020}/lib/is-image-attachment.mjs +0 -0
  199. /package/{esm2022 → esm2020}/lib/is-on-separate-date.mjs +0 -0
  200. /package/{esm2022 → esm2020}/lib/is-safari.mjs +0 -0
  201. /package/{esm2022 → esm2020}/lib/message-input/textarea.interface.mjs +0 -0
  202. /package/{esm2022 → esm2020}/stream-chat-angular.mjs +0 -0
@@ -0,0 +1,481 @@
1
+ import { Injectable } from '@angular/core';
2
+ import { createUriFromBlob, isImageFile } from './file-utils';
3
+ import { BehaviorSubject, combineLatest, map, shareReplay, take, } from 'rxjs';
4
+ import { isImageAttachment } from './is-image-attachment';
5
+ import * as i0 from "@angular/core";
6
+ import * as i1 from "./channel.service";
7
+ import * as i2 from "./notification.service";
8
+ import * as i3 from "./chat-client.service";
9
+ import * as i4 from "./message.service";
10
+ /**
11
+ * The `AttachmentService` manages the uploads of a message input.
12
+ *
13
+ * You can read more about [uploads](/chat/docs/javascript/file_uploads/) in the Stream API documentation. You can use Stream's API or the dashboard to customize the [file](/chat/docs/javascript/app_setting_overview/#file-uploads) and [image upload](/chat/docs/javascript/app_setting_overview/#image-uploads) configuration.
14
+ */
15
+ export class AttachmentService {
16
+ constructor(channelService, notificationService, chatClientService, messageService) {
17
+ this.channelService = channelService;
18
+ this.notificationService = notificationService;
19
+ this.chatClientService = chatClientService;
20
+ this.messageService = messageService;
21
+ /**
22
+ * Emits the number of uploads in progress.
23
+ *
24
+ * You can increment and decrement this counter if you're using custom attachments and want to disable message sending until all attachments are uploaded.
25
+ *
26
+ * The SDK will handle updating this counter for built-in attachments, but for custom attachments you should take care of this.
27
+ */
28
+ this.attachmentUploadInProgressCounter$ = new BehaviorSubject(0);
29
+ /**
30
+ * You can get and set the list if uploaded custom attachments
31
+ *
32
+ * By default the SDK components won't display these, but you can provide your own `customAttachmentPreviewListTemplate$` and `customAttachmentListTemplate$` for the [`CustomTemplatesService`](/chat/docs/sdk/angular/v6-rc/services/CustomTemplatesService/).
33
+ */
34
+ this.customAttachments$ = new BehaviorSubject([]);
35
+ /**
36
+ * The maximum number of attachments allowed for a message.
37
+ *
38
+ * The maximum is 30, you can set it to lower, but not higher.
39
+ */
40
+ this.maxNumberOfAttachments = 30;
41
+ this.attachmentUploadsSubject = new BehaviorSubject([]);
42
+ this.attachmentUploads$ = this.attachmentUploadsSubject.asObservable();
43
+ this.chatClientService.appSettings$.subscribe((appSettings) => (this.appSettings = appSettings));
44
+ this.attachmentsCounter$ = combineLatest([
45
+ this.attachmentUploads$,
46
+ this.customAttachments$,
47
+ ]).pipe(map(([attchmentUploads, customAttachments]) => {
48
+ return (attchmentUploads.filter((u) => u.state === 'success').length +
49
+ customAttachments.length);
50
+ }), shareReplay(1));
51
+ this.attachmentsCounter$.subscribe((count) => {
52
+ if (count > this.maxNumberOfAttachments) {
53
+ this.attachmentLimitNotificationHide =
54
+ this.notificationService.addPermanentNotification('streamChat.You currently have {{count}} attachments, the maximum is {{max}}', 'error', { count, max: this.maxNumberOfAttachments });
55
+ }
56
+ else {
57
+ this.attachmentLimitNotificationHide?.();
58
+ }
59
+ });
60
+ }
61
+ /**
62
+ * Resets the attachments uploads (for example after the message with the attachments sent successfully)
63
+ */
64
+ resetAttachmentUploads() {
65
+ this.attachmentUploadsSubject.next([]);
66
+ this.customAttachments$.next([]);
67
+ this.attachmentLimitNotificationHide?.();
68
+ }
69
+ /**
70
+ * Upload a voice recording
71
+ * @param audioRecording
72
+ * @returns A promise with true or false. If false is returned the upload was canceled because of a client side error. The error is emitted via the `NotificationService`.
73
+ */
74
+ async uploadVoiceRecording(audioRecording) {
75
+ if (!this.isWithinLimit(1)) {
76
+ return false;
77
+ }
78
+ if (!(await this.areAttachmentsHaveValidExtension([audioRecording.recording]))) {
79
+ return false;
80
+ }
81
+ if (!(await this.areAttachmentsHaveValidSize([audioRecording.recording]))) {
82
+ return false;
83
+ }
84
+ const upload = {
85
+ file: audioRecording.recording,
86
+ previewUri: audioRecording.asset_url,
87
+ extraData: {
88
+ duration: audioRecording.duration,
89
+ waveform_data: audioRecording.waveform_data,
90
+ },
91
+ state: 'uploading',
92
+ type: 'voiceRecording',
93
+ };
94
+ this.attachmentUploadsSubject.next([
95
+ ...this.attachmentUploadsSubject.getValue(),
96
+ upload,
97
+ ]);
98
+ await this.uploadAttachments([upload]);
99
+ return true;
100
+ }
101
+ /**
102
+ * Uploads the selected files, and creates preview for image files. The result is propagated throught the `attachmentUploads$` stream.
103
+ * @param fileList The files selected by the user, if you have Blobs instead of Files, you can convert them with this method: https://developer.mozilla.org/en-US/docs/Web/API/File/File
104
+ * @returns A promise with true or false. If false is returned the upload was canceled because of a client side error. The error is emitted via the `NotificationService`.
105
+ */
106
+ async filesSelected(fileList) {
107
+ if (!fileList) {
108
+ return;
109
+ }
110
+ const files = Array.from(fileList);
111
+ if (!this.isWithinLimit(files.length)) {
112
+ return false;
113
+ }
114
+ if (!(await this.areAttachmentsHaveValidExtension(files))) {
115
+ return false;
116
+ }
117
+ if (!(await this.areAttachmentsHaveValidSize(files))) {
118
+ return false;
119
+ }
120
+ const imageFiles = [];
121
+ const dataFiles = [];
122
+ const videoFiles = [];
123
+ files.forEach((file) => {
124
+ if (isImageFile(file)) {
125
+ imageFiles.push(file);
126
+ }
127
+ else if (file.type.startsWith('video/')) {
128
+ videoFiles.push(file);
129
+ }
130
+ else {
131
+ dataFiles.push(file);
132
+ }
133
+ });
134
+ imageFiles.forEach((f) => void this.createPreview(f));
135
+ const newUploads = [
136
+ ...imageFiles.map((file) => ({
137
+ file,
138
+ state: 'uploading',
139
+ type: 'image',
140
+ })),
141
+ ...videoFiles.map((file) => ({
142
+ file,
143
+ state: 'uploading',
144
+ type: 'video',
145
+ })),
146
+ ...dataFiles.map((file) => ({
147
+ file,
148
+ state: 'uploading',
149
+ type: 'file',
150
+ })),
151
+ ];
152
+ this.attachmentUploadsSubject.next([
153
+ ...this.attachmentUploadsSubject.getValue(),
154
+ ...newUploads,
155
+ ]);
156
+ await this.uploadAttachments(newUploads);
157
+ return true;
158
+ }
159
+ /**
160
+ * You can add custom `image`, `video` and `file` attachments using this method.
161
+ *
162
+ * Note: If you just want to use your own CDN for file uploads, you don't necessary need this method, you can just specify you own upload function in the [`ChannelService`](/chat/docs/sdk/angular/v6-rc/services/ChannelService/)
163
+ * @param attachment
164
+ *
165
+ * Will set `isCustomAttachment` to `true` on the attachment. This is a non-standard field, other SDKs will ignore this property.
166
+ */
167
+ addAttachment(attachment) {
168
+ attachment.isCustomAttachment = true;
169
+ this.createFromAttachments([attachment]);
170
+ }
171
+ /**
172
+ * Retries to upload an attachment.
173
+ * @param file
174
+ * @returns A promise with the result
175
+ */
176
+ async retryAttachmentUpload(file) {
177
+ const attachmentUploads = this.attachmentUploadsSubject.getValue();
178
+ const upload = attachmentUploads.find((u) => u.file === file);
179
+ if (!upload) {
180
+ return;
181
+ }
182
+ upload.state = 'uploading';
183
+ this.attachmentUploadsSubject.next([...attachmentUploads]);
184
+ await this.uploadAttachments([upload]);
185
+ }
186
+ /**
187
+ * Deletes an attachment, the attachment can have any state (`error`, `uploading` or `success`).
188
+ * @param upload
189
+ */
190
+ async deleteAttachment(upload) {
191
+ const attachmentUploads = this.attachmentUploadsSubject.getValue();
192
+ let result;
193
+ if (upload.state === 'success' &&
194
+ !upload.fromAttachment?.isCustomAttachment) {
195
+ try {
196
+ await this.channelService.deleteAttachment(upload);
197
+ result = [...attachmentUploads];
198
+ const index = attachmentUploads.indexOf(upload);
199
+ result.splice(index, 1);
200
+ }
201
+ catch (error) {
202
+ result = attachmentUploads;
203
+ this.notificationService.addTemporaryNotification('streamChat.Error deleting attachment');
204
+ }
205
+ }
206
+ else {
207
+ result = [...attachmentUploads];
208
+ const index = attachmentUploads.indexOf(upload);
209
+ result.splice(index, 1);
210
+ }
211
+ this.attachmentUploadsSubject.next([...result]);
212
+ }
213
+ /**
214
+ * Maps the current uploads to a format that can be sent along with the message to the Stream API.
215
+ * @returns the attachments
216
+ */
217
+ mapToAttachments() {
218
+ const attachmentUploads = this.attachmentUploadsSubject.getValue();
219
+ const builtInAttachments = attachmentUploads
220
+ .filter((r) => r.state === 'success')
221
+ .map((r) => {
222
+ let attachment = {
223
+ type: r.type,
224
+ };
225
+ if (r.fromAttachment) {
226
+ return r.fromAttachment;
227
+ }
228
+ else {
229
+ attachment.mime_type = r.file?.type;
230
+ if (r.type === 'image') {
231
+ attachment.fallback = r.file?.name;
232
+ attachment.image_url = r.url;
233
+ }
234
+ else {
235
+ attachment.asset_url = r.url;
236
+ attachment.title = r.file?.name;
237
+ attachment.file_size = r.file?.size;
238
+ attachment.thumb_url = r.thumb_url;
239
+ }
240
+ if (r.extraData) {
241
+ attachment = { ...attachment, ...r.extraData };
242
+ }
243
+ }
244
+ return attachment;
245
+ });
246
+ return [...builtInAttachments, ...this.customAttachments$.value];
247
+ }
248
+ /**
249
+ * Maps attachments received from the Stream API to uploads. This is useful when editing a message.
250
+ * @param attachments Attachemnts received with the message
251
+ */
252
+ createFromAttachments(attachments) {
253
+ const attachmentUploads = [];
254
+ const builtInAttachments = [];
255
+ const customAttachments = [];
256
+ attachments.forEach((attachment) => {
257
+ if (this.messageService.isCustomAttachment(attachment)) {
258
+ customAttachments.push(attachment);
259
+ }
260
+ else {
261
+ builtInAttachments.push(attachment);
262
+ }
263
+ });
264
+ builtInAttachments.forEach((attachment) => {
265
+ if (isImageAttachment(attachment)) {
266
+ attachmentUploads.push({
267
+ url: (attachment.img_url ||
268
+ attachment.thumb_url ||
269
+ attachment.image_url),
270
+ state: 'success',
271
+ type: 'image',
272
+ file: {
273
+ name: attachment.fallback,
274
+ type: attachment.mime_type,
275
+ },
276
+ fromAttachment: attachment,
277
+ });
278
+ }
279
+ else if (attachment.type === 'file' || attachment.type === 'video') {
280
+ attachmentUploads.push({
281
+ url: attachment.asset_url,
282
+ state: 'success',
283
+ file: {
284
+ name: attachment.title,
285
+ size: attachment.file_size,
286
+ type: attachment.mime_type,
287
+ },
288
+ type: attachment.type,
289
+ thumb_url: attachment.thumb_url,
290
+ fromAttachment: attachment,
291
+ });
292
+ }
293
+ else if (attachment.type === 'voiceRecording') {
294
+ attachmentUploads.push({
295
+ url: attachment.asset_url,
296
+ state: 'success',
297
+ file: {
298
+ name: attachment.title,
299
+ size: attachment.file_size,
300
+ type: attachment.mime_type,
301
+ },
302
+ type: 'voiceRecording',
303
+ extraData: {
304
+ duration: attachment.duration,
305
+ waveform_data: attachment.waveform_data,
306
+ },
307
+ });
308
+ }
309
+ });
310
+ if (attachmentUploads.length > 0) {
311
+ this.attachmentUploadsSubject.next([
312
+ ...this.attachmentUploadsSubject.getValue(),
313
+ ...attachmentUploads,
314
+ ]);
315
+ }
316
+ if (customAttachments.length > 0) {
317
+ this.customAttachments$.next(customAttachments);
318
+ }
319
+ }
320
+ async createPreview(file) {
321
+ try {
322
+ const uri = await createUriFromBlob(file);
323
+ const attachmentUploads = this.attachmentUploadsSubject.getValue();
324
+ const upload = attachmentUploads.find((upload) => upload.file === file);
325
+ if (!upload) {
326
+ return;
327
+ }
328
+ upload.previewUri = uri;
329
+ this.attachmentUploadsSubject.next([...attachmentUploads]);
330
+ }
331
+ catch (e) {
332
+ this.chatClientService?.chatClient?.logger('error', e instanceof Error ? e.message : `Can't create image preview`, { error: e, tag: ['AttachmentService'] });
333
+ }
334
+ }
335
+ async uploadAttachments(uploads) {
336
+ this.attachmentUploadInProgressCounter$.next(this.attachmentUploadInProgressCounter$.value + 1);
337
+ const result = await this.channelService.uploadAttachments(uploads);
338
+ const attachmentUploads = this.attachmentUploadsSubject.getValue();
339
+ result.forEach((r) => {
340
+ const upload = attachmentUploads.find((upload) => upload.file === r.file);
341
+ if (!upload) {
342
+ if (r.url) {
343
+ void this.channelService.deleteAttachment(r);
344
+ }
345
+ return;
346
+ }
347
+ upload.state = r.state;
348
+ upload.url = r.url;
349
+ upload.thumb_url = r.thumb_url;
350
+ if (upload.state === 'error') {
351
+ upload.errorReason = r.errorReason;
352
+ upload.errorExtraInfo = r.errorExtraInfo;
353
+ let errorKey;
354
+ const translateParams = { name: upload.file.name };
355
+ switch (upload.errorReason) {
356
+ case 'file-extension':
357
+ errorKey =
358
+ 'streamChat.Error uploading file, extension not supported';
359
+ translateParams.ext = upload.errorExtraInfo?.[0]?.param;
360
+ break;
361
+ case 'file-size':
362
+ errorKey =
363
+ 'streamChat.Error uploading file, maximum file size exceeded';
364
+ translateParams.limit = upload.errorExtraInfo?.[0]?.param;
365
+ break;
366
+ default:
367
+ errorKey = 'streamChat.Error uploading file';
368
+ }
369
+ this.notificationService.addTemporaryNotification(errorKey, 'error', undefined, translateParams);
370
+ }
371
+ });
372
+ this.attachmentUploadInProgressCounter$.next(this.attachmentUploadInProgressCounter$.value - 1);
373
+ this.attachmentUploadsSubject.next([...attachmentUploads]);
374
+ }
375
+ async areAttachmentsHaveValidExtension(files) {
376
+ if (!this.appSettings) {
377
+ try {
378
+ await this.chatClientService.getAppSettings();
379
+ }
380
+ catch (error) {
381
+ return true;
382
+ }
383
+ }
384
+ let isValid = true;
385
+ files.forEach((f) => {
386
+ let hasBlockedExtension;
387
+ let hasBlockedMimeType;
388
+ let hasNotAllowedExtension;
389
+ let hasNotAllowedMimeType;
390
+ if (isImageFile(f)) {
391
+ hasBlockedExtension =
392
+ !!this.appSettings?.image_upload_config?.blocked_file_extensions?.find((ext) => f.name.endsWith(ext));
393
+ hasBlockedMimeType =
394
+ !!this.appSettings?.image_upload_config?.blocked_mime_types?.find((type) => f.type === type);
395
+ hasNotAllowedExtension =
396
+ !!this.appSettings?.image_upload_config?.allowed_file_extensions
397
+ ?.length &&
398
+ !this.appSettings?.image_upload_config?.allowed_file_extensions?.find((ext) => f.name.endsWith(ext));
399
+ hasNotAllowedMimeType =
400
+ !!this.appSettings?.image_upload_config?.allowed_mime_types?.length &&
401
+ !this.appSettings?.image_upload_config?.allowed_mime_types?.find((type) => f.type === type);
402
+ }
403
+ else {
404
+ hasBlockedExtension =
405
+ !!this.appSettings?.file_upload_config?.blocked_file_extensions?.find((ext) => f.name.endsWith(ext));
406
+ hasBlockedMimeType =
407
+ !!this.appSettings?.file_upload_config?.blocked_mime_types?.find((type) => f.type === type);
408
+ hasNotAllowedExtension =
409
+ !!this.appSettings?.file_upload_config?.allowed_file_extensions
410
+ ?.length &&
411
+ !this.appSettings?.file_upload_config?.allowed_file_extensions?.find((ext) => f.name.endsWith(ext));
412
+ hasNotAllowedMimeType =
413
+ !!this.appSettings?.file_upload_config?.allowed_mime_types?.length &&
414
+ !this.appSettings?.file_upload_config?.allowed_mime_types?.find((type) => f.type === type);
415
+ }
416
+ if (hasBlockedExtension ||
417
+ hasBlockedMimeType ||
418
+ hasNotAllowedExtension ||
419
+ hasNotAllowedMimeType) {
420
+ this.notificationService.addTemporaryNotification('streamChat.Error uploading file, extension not supported', undefined, undefined, { name: f.name, ext: f.type });
421
+ isValid = false;
422
+ }
423
+ });
424
+ return isValid;
425
+ }
426
+ async areAttachmentsHaveValidSize(files) {
427
+ if (!this.appSettings) {
428
+ try {
429
+ await this.chatClientService.getAppSettings();
430
+ }
431
+ catch (error) {
432
+ return true;
433
+ }
434
+ }
435
+ const imageSizeLimitInBytes = this.appSettings?.image_upload_config?.size_limit || 0;
436
+ const imageSizeLimiString = `${imageSizeLimitInBytes / (1024 * 1024)}MB`;
437
+ const fileSizeLimitInBytes = this.appSettings?.file_upload_config?.size_limit || 0;
438
+ const fileSizeLimitInString = `${fileSizeLimitInBytes / (1024 * 1024)}MB`;
439
+ let isValid = true;
440
+ files.forEach((f) => {
441
+ let isOverSized = false;
442
+ let limit = '';
443
+ if (isImageFile(f) && imageSizeLimitInBytes > 0) {
444
+ isOverSized = f.size > imageSizeLimitInBytes;
445
+ limit = imageSizeLimiString;
446
+ }
447
+ else if (fileSizeLimitInBytes > 0) {
448
+ isOverSized = f.size > fileSizeLimitInBytes;
449
+ limit = fileSizeLimitInString;
450
+ }
451
+ if (isOverSized) {
452
+ this.notificationService.addTemporaryNotification('streamChat.Error uploading file, maximum file size exceeded', undefined, undefined, { name: f.name, limit: limit });
453
+ isValid = false;
454
+ }
455
+ });
456
+ return isValid;
457
+ }
458
+ isWithinLimit(numberOfNewAttachments) {
459
+ let currentNumberOfAttachments = 0;
460
+ this.attachmentsCounter$
461
+ .pipe(take(1))
462
+ .subscribe((counter) => (currentNumberOfAttachments = counter));
463
+ if (currentNumberOfAttachments + numberOfNewAttachments >
464
+ this.maxNumberOfAttachments) {
465
+ this.notificationService.addTemporaryNotification(`streamChat.You can't uplod more than {{max}} attachments`, 'error', undefined, { max: this.maxNumberOfAttachments });
466
+ return false;
467
+ }
468
+ else {
469
+ return true;
470
+ }
471
+ }
472
+ }
473
+ AttachmentService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AttachmentService, deps: [{ token: i1.ChannelService }, { token: i2.NotificationService }, { token: i3.ChatClientService }, { token: i4.MessageService }], target: i0.ɵɵFactoryTarget.Injectable });
474
+ AttachmentService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AttachmentService, providedIn: 'root' });
475
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AttachmentService, decorators: [{
476
+ type: Injectable,
477
+ args: [{
478
+ providedIn: 'root',
479
+ }]
480
+ }], ctorParameters: function () { return [{ type: i1.ChannelService }, { type: i2.NotificationService }, { type: i3.ChatClientService }, { type: i4.MessageService }]; } });
481
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"attachment.service.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/attachment.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { createUriFromBlob, isImageFile } from './file-utils';\nimport {\n  BehaviorSubject,\n  combineLatest,\n  map,\n  Observable,\n  shareReplay,\n  take,\n} from 'rxjs';\nimport { AppSettings, Attachment } from 'stream-chat';\nimport { ChannelService } from './channel.service';\nimport { isImageAttachment } from './is-image-attachment';\nimport { NotificationService } from './notification.service';\nimport { AttachmentUpload, AudioRecording } from './types';\nimport { ChatClientService } from './chat-client.service';\nimport { MessageService } from './message.service';\n\n/**\n * The `AttachmentService` manages the uploads of a message input.\n *\n * You can read more about [uploads](/chat/docs/javascript/file_uploads/) in the Stream API documentation. You can use Stream's API or the dashboard to customize the [file](/chat/docs/javascript/app_setting_overview/#file-uploads) and [image upload](/chat/docs/javascript/app_setting_overview/#image-uploads) configuration.\n */\n@Injectable({\n  providedIn: 'root',\n})\nexport class AttachmentService {\n  /**\n   * Emits the number of uploads in progress.\n   *\n   * You can increment and decrement this counter if you're using custom attachments and want to disable message sending until all attachments are uploaded.\n   *\n   * The SDK will handle updating this counter for built-in attachments, but for custom attachments you should take care of this.\n   */\n  attachmentUploadInProgressCounter$ = new BehaviorSubject<number>(0);\n  /**\n   * Emits the state of the uploads ([`AttachmentUpload[]`](https://github.com/GetStream/stream-chat-angular/blob/master/projects/stream-chat-angular/src/lib/types.ts)), it adds a state (`success`, `error` or `uploading`) to each file the user selects for upload. It is used by the [`AttachmentPreviewList`](/chat/docs/sdk/angular/v6-rc/components/AttachmentPreviewListComponent/) to display the attachment previews.\n   */\n  attachmentUploads$: Observable<AttachmentUpload[]>;\n  /**\n   * You can get and set the list if uploaded custom attachments\n   *\n   * By default the SDK components won't display these, but you can provide your own `customAttachmentPreviewListTemplate$` and `customAttachmentListTemplate$` for the [`CustomTemplatesService`](/chat/docs/sdk/angular/v6-rc/services/CustomTemplatesService/).\n   */\n  customAttachments$ = new BehaviorSubject<Attachment[]>([]);\n  /**\n   * The current number of attachments\n   */\n  attachmentsCounter$: Observable<number>;\n  /**\n   * The maximum number of attachments allowed for a message.\n   *\n   * The maximum is 30, you can set it to lower, but not higher.\n   */\n  maxNumberOfAttachments = 30;\n  private attachmentUploadsSubject = new BehaviorSubject<AttachmentUpload[]>(\n    []\n  );\n  private appSettings: AppSettings | undefined;\n  private attachmentLimitNotificationHide?: () => void;\n\n  constructor(\n    private channelService: ChannelService,\n    private notificationService: NotificationService,\n    private chatClientService: ChatClientService,\n    private messageService: MessageService\n  ) {\n    this.attachmentUploads$ = this.attachmentUploadsSubject.asObservable();\n    this.chatClientService.appSettings$.subscribe(\n      (appSettings) => (this.appSettings = appSettings)\n    );\n    this.attachmentsCounter$ = combineLatest([\n      this.attachmentUploads$,\n      this.customAttachments$,\n    ]).pipe(\n      map(([attchmentUploads, customAttachments]) => {\n        return (\n          attchmentUploads.filter((u) => u.state === 'success').length +\n          customAttachments.length\n        );\n      }),\n      shareReplay(1)\n    );\n    this.attachmentsCounter$.subscribe((count) => {\n      if (count > this.maxNumberOfAttachments) {\n        this.attachmentLimitNotificationHide =\n          this.notificationService.addPermanentNotification(\n            'streamChat.You currently have {{count}} attachments, the maximum is {{max}}',\n            'error',\n            { count, max: this.maxNumberOfAttachments }\n          );\n      } else {\n        this.attachmentLimitNotificationHide?.();\n      }\n    });\n  }\n\n  /**\n   * Resets the attachments uploads (for example after the message with the attachments sent successfully)\n   */\n  resetAttachmentUploads() {\n    this.attachmentUploadsSubject.next([]);\n    this.customAttachments$.next([]);\n    this.attachmentLimitNotificationHide?.();\n  }\n\n  /**\n   * Upload a voice recording\n   * @param audioRecording\n   * @returns A promise with true or false. If false is returned the upload was canceled because of a client side error. The error is emitted via the `NotificationService`.\n   */\n  async uploadVoiceRecording(audioRecording: AudioRecording) {\n    if (!this.isWithinLimit(1)) {\n      return false;\n    }\n    if (\n      !(await this.areAttachmentsHaveValidExtension([audioRecording.recording]))\n    ) {\n      return false;\n    }\n    if (!(await this.areAttachmentsHaveValidSize([audioRecording.recording]))) {\n      return false;\n    }\n\n    const upload = {\n      file: audioRecording.recording,\n      previewUri: audioRecording.asset_url,\n      extraData: {\n        duration: audioRecording.duration,\n        waveform_data: audioRecording.waveform_data,\n      },\n      state: 'uploading' as const,\n      type: 'voiceRecording' as const,\n    };\n    this.attachmentUploadsSubject.next([\n      ...this.attachmentUploadsSubject.getValue(),\n      upload,\n    ]);\n    await this.uploadAttachments([upload]);\n    return true;\n  }\n\n  /**\n   * Uploads the selected files, and creates preview for image files. The result is propagated throught the `attachmentUploads$` stream.\n   * @param fileList The files selected by the user, if you have Blobs instead of Files, you can convert them with this method: https://developer.mozilla.org/en-US/docs/Web/API/File/File\n   * @returns A promise with true or false. If false is returned the upload was canceled because of a client side error. The error is emitted via the `NotificationService`.\n   */\n  async filesSelected(fileList: FileList | File[] | null) {\n    if (!fileList) {\n      return;\n    }\n\n    const files = Array.from(fileList);\n\n    if (!this.isWithinLimit(files.length)) {\n      return false;\n    }\n\n    if (!(await this.areAttachmentsHaveValidExtension(files))) {\n      return false;\n    }\n    if (!(await this.areAttachmentsHaveValidSize(files))) {\n      return false;\n    }\n    const imageFiles: File[] = [];\n    const dataFiles: File[] = [];\n    const videoFiles: File[] = [];\n\n    files.forEach((file) => {\n      if (isImageFile(file)) {\n        imageFiles.push(file);\n      } else if (file.type.startsWith('video/')) {\n        videoFiles.push(file);\n      } else {\n        dataFiles.push(file);\n      }\n    });\n    imageFiles.forEach((f) => void this.createPreview(f));\n    const newUploads = [\n      ...imageFiles.map((file) => ({\n        file,\n        state: 'uploading' as const,\n        type: 'image' as const,\n      })),\n      ...videoFiles.map((file) => ({\n        file,\n        state: 'uploading' as const,\n        type: 'video' as const,\n      })),\n      ...dataFiles.map((file) => ({\n        file,\n        state: 'uploading' as const,\n        type: 'file' as const,\n      })),\n    ];\n    this.attachmentUploadsSubject.next([\n      ...this.attachmentUploadsSubject.getValue(),\n      ...newUploads,\n    ]);\n    await this.uploadAttachments(newUploads);\n    return true;\n  }\n\n  /**\n   * You can add custom `image`, `video` and `file` attachments using this method.\n   *\n   * Note: If you just want to use your own CDN for file uploads, you don't necessary need this method, you can just specify you own upload function in the [`ChannelService`](/chat/docs/sdk/angular/v6-rc/services/ChannelService/)\n   * @param attachment\n   *\n   * Will set `isCustomAttachment` to `true` on the attachment. This is a non-standard field, other SDKs will ignore this property.\n   */\n  addAttachment(attachment: Attachment) {\n    attachment.isCustomAttachment = true;\n    this.createFromAttachments([attachment]);\n  }\n\n  /**\n   * Retries to upload an attachment.\n   * @param file\n   * @returns A promise with the result\n   */\n  async retryAttachmentUpload(file: File) {\n    const attachmentUploads = this.attachmentUploadsSubject.getValue();\n    const upload = attachmentUploads.find((u) => u.file === file);\n    if (!upload) {\n      return;\n    }\n    upload.state = 'uploading';\n    this.attachmentUploadsSubject.next([...attachmentUploads]);\n    await this.uploadAttachments([upload]);\n  }\n\n  /**\n   * Deletes an attachment, the attachment can have any state (`error`, `uploading` or `success`).\n   * @param upload\n   */\n  async deleteAttachment(upload: AttachmentUpload) {\n    const attachmentUploads = this.attachmentUploadsSubject.getValue();\n    let result!: AttachmentUpload[];\n    if (\n      upload.state === 'success' &&\n      !upload.fromAttachment?.isCustomAttachment\n    ) {\n      try {\n        await this.channelService.deleteAttachment(upload);\n        result = [...attachmentUploads];\n        const index = attachmentUploads.indexOf(upload);\n        result.splice(index, 1);\n      } catch (error) {\n        result = attachmentUploads;\n        this.notificationService.addTemporaryNotification(\n          'streamChat.Error deleting attachment'\n        );\n      }\n    } else {\n      result = [...attachmentUploads];\n      const index = attachmentUploads.indexOf(upload);\n      result.splice(index, 1);\n    }\n    this.attachmentUploadsSubject.next([...result]);\n  }\n\n  /**\n   * Maps the current uploads to a format that can be sent along with the message to the Stream API.\n   * @returns the attachments\n   */\n  mapToAttachments() {\n    const attachmentUploads = this.attachmentUploadsSubject.getValue();\n    const builtInAttachments = attachmentUploads\n      .filter((r) => r.state === 'success')\n      .map((r) => {\n        let attachment: Attachment = {\n          type: r.type,\n        };\n        if (r.fromAttachment) {\n          return r.fromAttachment;\n        } else {\n          attachment.mime_type = r.file?.type;\n          if (r.type === 'image') {\n            attachment.fallback = r.file?.name;\n            attachment.image_url = r.url;\n          } else {\n            attachment.asset_url = r.url;\n            attachment.title = r.file?.name;\n            attachment.file_size = r.file?.size;\n            attachment.thumb_url = r.thumb_url;\n          }\n          if (r.extraData) {\n            attachment = { ...attachment, ...r.extraData };\n          }\n        }\n\n        return attachment;\n      });\n    return [...builtInAttachments, ...this.customAttachments$.value];\n  }\n\n  /**\n   * Maps attachments received from the Stream API to uploads. This is useful when editing a message.\n   * @param attachments Attachemnts received with the message\n   */\n  createFromAttachments(attachments: Attachment[]) {\n    const attachmentUploads: AttachmentUpload[] = [];\n    const builtInAttachments: Attachment[] = [];\n    const customAttachments: Attachment[] = [];\n    attachments.forEach((attachment) => {\n      if (this.messageService.isCustomAttachment(attachment)) {\n        customAttachments.push(attachment);\n      } else {\n        builtInAttachments.push(attachment);\n      }\n    });\n    builtInAttachments.forEach((attachment) => {\n      if (isImageAttachment(attachment)) {\n        attachmentUploads.push({\n          url: (attachment.img_url ||\n            attachment.thumb_url ||\n            attachment.image_url) as string,\n          state: 'success',\n          type: 'image',\n          file: {\n            name: attachment.fallback,\n            type: attachment.mime_type,\n          } as File,\n          fromAttachment: attachment,\n        });\n      } else if (attachment.type === 'file' || attachment.type === 'video') {\n        attachmentUploads.push({\n          url: attachment.asset_url,\n          state: 'success',\n          file: {\n            name: attachment.title,\n            size: attachment.file_size,\n            type: attachment.mime_type,\n          } as File,\n          type: attachment.type,\n          thumb_url: attachment.thumb_url,\n          fromAttachment: attachment,\n        });\n      } else if (attachment.type === 'voiceRecording') {\n        attachmentUploads.push({\n          url: attachment.asset_url,\n          state: 'success',\n          file: {\n            name: attachment.title,\n            size: attachment.file_size,\n            type: attachment.mime_type,\n          } as File,\n          type: 'voiceRecording',\n          extraData: {\n            duration: attachment.duration,\n            waveform_data: attachment.waveform_data,\n          },\n        });\n      }\n    });\n\n    if (attachmentUploads.length > 0) {\n      this.attachmentUploadsSubject.next([\n        ...this.attachmentUploadsSubject.getValue(),\n        ...attachmentUploads,\n      ]);\n    }\n\n    if (customAttachments.length > 0) {\n      this.customAttachments$.next(customAttachments);\n    }\n  }\n\n  private async createPreview(file: File | Blob) {\n    try {\n      const uri = await createUriFromBlob(file);\n      const attachmentUploads = this.attachmentUploadsSubject.getValue();\n      const upload = attachmentUploads.find((upload) => upload.file === file);\n      if (!upload) {\n        return;\n      }\n      upload.previewUri = uri;\n      this.attachmentUploadsSubject.next([...attachmentUploads]);\n    } catch (e: unknown) {\n      this.chatClientService?.chatClient?.logger(\n        'error',\n        e instanceof Error ? e.message : `Can't create image preview`,\n        { error: e, tag: ['AttachmentService'] }\n      );\n    }\n  }\n\n  private async uploadAttachments(uploads: AttachmentUpload[]) {\n    this.attachmentUploadInProgressCounter$.next(\n      this.attachmentUploadInProgressCounter$.value + 1\n    );\n    const result = await this.channelService.uploadAttachments(uploads);\n    const attachmentUploads = this.attachmentUploadsSubject.getValue();\n    result.forEach((r) => {\n      const upload = attachmentUploads.find((upload) => upload.file === r.file);\n      if (!upload) {\n        if (r.url) {\n          void this.channelService.deleteAttachment(r);\n        }\n        return;\n      }\n      upload.state = r.state;\n      upload.url = r.url;\n      upload.thumb_url = r.thumb_url;\n      if (upload.state === 'error') {\n        upload.errorReason = r.errorReason;\n        upload.errorExtraInfo = r.errorExtraInfo;\n        let errorKey;\n        const translateParams: { name: string; ext?: string; limit?: string } =\n          { name: upload.file.name };\n        switch (upload.errorReason) {\n          case 'file-extension':\n            errorKey =\n              'streamChat.Error uploading file, extension not supported';\n            translateParams.ext = upload.errorExtraInfo?.[0]?.param;\n            break;\n          case 'file-size':\n            errorKey =\n              'streamChat.Error uploading file, maximum file size exceeded';\n            translateParams.limit = upload.errorExtraInfo?.[0]?.param;\n            break;\n          default:\n            errorKey = 'streamChat.Error uploading file';\n        }\n        this.notificationService.addTemporaryNotification(\n          errorKey,\n          'error',\n          undefined,\n          translateParams\n        );\n      }\n    });\n    this.attachmentUploadInProgressCounter$.next(\n      this.attachmentUploadInProgressCounter$.value - 1\n    );\n    this.attachmentUploadsSubject.next([...attachmentUploads]);\n  }\n\n  private async areAttachmentsHaveValidExtension(files: File[]) {\n    if (!this.appSettings) {\n      try {\n        await this.chatClientService.getAppSettings();\n      } catch (error) {\n        return true;\n      }\n    }\n    let isValid = true;\n    files.forEach((f) => {\n      let hasBlockedExtension: boolean;\n      let hasBlockedMimeType: boolean;\n      let hasNotAllowedExtension: boolean;\n      let hasNotAllowedMimeType: boolean;\n      if (isImageFile(f)) {\n        hasBlockedExtension =\n          !!this.appSettings?.image_upload_config?.blocked_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasBlockedMimeType =\n          !!this.appSettings?.image_upload_config?.blocked_mime_types?.find(\n            (type) => f.type === type\n          );\n        hasNotAllowedExtension =\n          !!this.appSettings?.image_upload_config?.allowed_file_extensions\n            ?.length &&\n          !this.appSettings?.image_upload_config?.allowed_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasNotAllowedMimeType =\n          !!this.appSettings?.image_upload_config?.allowed_mime_types?.length &&\n          !this.appSettings?.image_upload_config?.allowed_mime_types?.find(\n            (type) => f.type === type\n          );\n      } else {\n        hasBlockedExtension =\n          !!this.appSettings?.file_upload_config?.blocked_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasBlockedMimeType =\n          !!this.appSettings?.file_upload_config?.blocked_mime_types?.find(\n            (type) => f.type === type\n          );\n        hasNotAllowedExtension =\n          !!this.appSettings?.file_upload_config?.allowed_file_extensions\n            ?.length &&\n          !this.appSettings?.file_upload_config?.allowed_file_extensions?.find(\n            (ext) => f.name.endsWith(ext)\n          );\n        hasNotAllowedMimeType =\n          !!this.appSettings?.file_upload_config?.allowed_mime_types?.length &&\n          !this.appSettings?.file_upload_config?.allowed_mime_types?.find(\n            (type) => f.type === type\n          );\n      }\n      if (\n        hasBlockedExtension ||\n        hasBlockedMimeType ||\n        hasNotAllowedExtension ||\n        hasNotAllowedMimeType\n      ) {\n        this.notificationService.addTemporaryNotification(\n          'streamChat.Error uploading file, extension not supported',\n          undefined,\n          undefined,\n          { name: f.name, ext: f.type }\n        );\n        isValid = false;\n      }\n    });\n    return isValid;\n  }\n\n  private async areAttachmentsHaveValidSize(files: File[]) {\n    if (!this.appSettings) {\n      try {\n        await this.chatClientService.getAppSettings();\n      } catch (error) {\n        return true;\n      }\n    }\n    const imageSizeLimitInBytes =\n      this.appSettings?.image_upload_config?.size_limit || 0;\n    const imageSizeLimiString = `${imageSizeLimitInBytes / (1024 * 1024)}MB`;\n    const fileSizeLimitInBytes =\n      this.appSettings?.file_upload_config?.size_limit || 0;\n    const fileSizeLimitInString = `${fileSizeLimitInBytes / (1024 * 1024)}MB`;\n    let isValid = true;\n    files.forEach((f) => {\n      let isOverSized = false;\n      let limit = '';\n      if (isImageFile(f) && imageSizeLimitInBytes > 0) {\n        isOverSized = f.size > imageSizeLimitInBytes;\n        limit = imageSizeLimiString;\n      } else if (fileSizeLimitInBytes > 0) {\n        isOverSized = f.size > fileSizeLimitInBytes;\n        limit = fileSizeLimitInString;\n      }\n      if (isOverSized) {\n        this.notificationService.addTemporaryNotification(\n          'streamChat.Error uploading file, maximum file size exceeded',\n          undefined,\n          undefined,\n          { name: f.name, limit: limit }\n        );\n        isValid = false;\n      }\n    });\n    return isValid;\n  }\n\n  private isWithinLimit(numberOfNewAttachments: number) {\n    let currentNumberOfAttachments: number = 0;\n    this.attachmentsCounter$\n      .pipe(take(1))\n      .subscribe((counter) => (currentNumberOfAttachments = counter));\n    if (\n      currentNumberOfAttachments + numberOfNewAttachments >\n      this.maxNumberOfAttachments\n    ) {\n      this.notificationService.addTemporaryNotification(\n        `streamChat.You can't uplod more than {{max}} attachments`,\n        'error',\n        undefined,\n        { max: this.maxNumberOfAttachments }\n      );\n      return false;\n    } else {\n      return true;\n    }\n  }\n}\n"]}