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.
- package/README.md +2 -2
- package/assets/i18n/en.d.ts +0 -1
- package/assets/version.d.ts +1 -1
- package/{esm2022 → esm2020}/assets/i18n/en.mjs +1 -2
- package/{esm2022 → esm2020}/assets/version.mjs +2 -2
- package/esm2020/lib/attachment-configuration.service.mjs +182 -0
- package/esm2020/lib/attachment-list/attachment-list.component.mjs +232 -0
- package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +55 -0
- package/esm2020/lib/attachment.service.mjs +481 -0
- package/esm2020/lib/avatar/avatar.component.mjs +160 -0
- package/esm2020/lib/avatar-placeholder/avatar-placeholder.component.mjs +66 -0
- package/esm2020/lib/channel/channel.component.mjs +45 -0
- package/esm2020/lib/channel-header/channel-header.component.mjs +72 -0
- package/esm2020/lib/channel-list/channel-list.component.mjs +47 -0
- package/esm2020/lib/channel-preview/channel-preview.component.mjs +155 -0
- package/esm2020/lib/channel-query.mjs +77 -0
- package/esm2020/lib/channel.service.mjs +1561 -0
- package/esm2020/lib/chat-client.service.mjs +233 -0
- package/esm2020/lib/custom-templates.service.mjs +244 -0
- package/{esm2022 → esm2020}/lib/date-parser.service.mjs +5 -5
- package/esm2020/lib/file-utils.mjs +35 -0
- package/{esm2022 → esm2020}/lib/get-channel-display-text.mjs +1 -1
- package/esm2020/lib/get-message-translation.mjs +12 -0
- package/esm2020/lib/icon/icon-placeholder/icon-placeholder.component.mjs +28 -0
- package/{esm2022 → esm2020}/lib/icon/icon.component.mjs +5 -5
- package/{esm2022 → esm2020}/lib/icon/icon.module.mjs +11 -11
- package/{esm2022 → esm2020}/lib/icon/loading-indicator/loading-indicator.component.mjs +5 -5
- package/esm2020/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +20 -0
- package/{esm2022 → esm2020}/lib/list-users.mjs +1 -1
- package/esm2020/lib/message/message.component.mjs +486 -0
- package/esm2020/lib/message-actions-box/message-actions-box.component.mjs +123 -0
- package/esm2020/lib/message-actions.service.mjs +187 -0
- package/esm2020/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +71 -0
- package/esm2020/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +333 -0
- package/{esm2022 → esm2020}/lib/message-input/emoji-input.service.mjs +7 -7
- package/esm2020/lib/message-input/message-input-config.service.mjs +50 -0
- package/esm2020/lib/message-input/message-input.component.mjs +507 -0
- package/{esm2022 → esm2020}/lib/message-input/textarea/textarea.component.mjs +6 -6
- package/{esm2022 → esm2020}/lib/message-input/textarea.directive.mjs +5 -5
- package/{esm2022 → esm2020}/lib/message-input/voice-recorder.service.mjs +5 -5
- package/{esm2022 → esm2020}/lib/message-list/group-styles.mjs +1 -1
- package/esm2020/lib/message-list/message-list.component.mjs +717 -0
- package/esm2020/lib/message-preview.mjs +21 -0
- package/esm2020/lib/message-reactions/message-reactions.component.mjs +168 -0
- package/esm2020/lib/message-reactions-selector/message-reactions-selector.component.mjs +61 -0
- package/{esm2022 → esm2020}/lib/message-reactions.service.mjs +6 -6
- package/esm2020/lib/message-text/message-text.component.mjs +143 -0
- package/esm2020/lib/message.service.mjs +43 -0
- package/{esm2022 → esm2020}/lib/modal/modal.component.mjs +6 -6
- package/esm2020/lib/notification/notification.component.mjs +20 -0
- package/esm2020/lib/notification-list/notification-list.component.mjs +36 -0
- package/{esm2022 → esm2020}/lib/notification.service.mjs +6 -6
- package/esm2020/lib/paginated-list/paginated-list.component.mjs +94 -0
- package/{esm2022 → esm2020}/lib/parse-date.mjs +1 -1
- package/esm2020/lib/read-by.mjs +12 -0
- package/esm2020/lib/stream-autocomplete-textarea.module.mjs +33 -0
- package/{esm2022 → esm2020}/lib/stream-avatar.module.mjs +5 -5
- package/{esm2022 → esm2020}/lib/stream-chat.module.mjs +59 -59
- package/{esm2022 → esm2020}/lib/stream-i18n.service.mjs +6 -6
- package/esm2020/lib/stream-textarea.module.mjs +31 -0
- package/{esm2022 → esm2020}/lib/theme.service.mjs +6 -6
- package/esm2020/lib/thread/thread.component.mjs +51 -0
- package/{esm2022 → esm2020}/lib/transliteration.service.mjs +5 -5
- package/esm2020/lib/types-custom.mjs +2 -0
- package/esm2020/lib/types.mjs +2 -0
- package/esm2020/lib/user-list/user-list.component.mjs +47 -0
- package/esm2020/lib/virtualized-list.service.mjs +271 -0
- package/{esm2022 → esm2020}/lib/virtualized-message-list.service.mjs +1 -1
- package/{esm2022 → esm2020}/lib/voice-recorder/amplitude-recorder.service.mjs +5 -5
- package/{esm2022 → esm2020}/lib/voice-recorder/audio-recorder.service.mjs +5 -5
- package/{esm2022 → esm2020}/lib/voice-recorder/media-recorder.mjs +1 -1
- package/esm2020/lib/voice-recorder/mp3-transcoder.mjs +61 -0
- package/esm2020/lib/voice-recorder/transcoder.service.mjs +121 -0
- package/esm2020/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +35 -0
- package/esm2020/lib/voice-recorder/voice-recorder.component.mjs +80 -0
- package/{esm2022 → esm2020}/lib/voice-recorder/voice-recorder.module.mjs +9 -9
- package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +112 -0
- package/esm2020/lib/voice-recording/voice-recording.component.mjs +91 -0
- package/{esm2022 → esm2020}/lib/voice-recording/voice-recording.module.mjs +5 -5
- package/{esm2022 → esm2020}/lib/wave-form-sampler.mjs +1 -1
- package/esm2020/public-api.mjs +86 -0
- package/esm2020/stream-chat.mjs +2 -0
- package/fesm2015/stream-chat-angular.mjs +9171 -0
- package/fesm2015/stream-chat-angular.mjs.map +1 -0
- package/{fesm2022 → fesm2020}/stream-chat-angular.mjs +1251 -961
- package/fesm2020/stream-chat-angular.mjs.map +1 -0
- package/lib/attachment-configuration.service.d.ts +12 -12
- package/lib/attachment-list/attachment-list.component.d.ts +15 -10
- package/lib/attachment-preview-list/attachment-preview-list.component.d.ts +1 -1
- package/lib/attachment.service.d.ts +12 -10
- package/lib/avatar/avatar.component.d.ts +7 -7
- package/lib/avatar-placeholder/avatar-placeholder.component.d.ts +5 -5
- package/lib/channel/channel.component.d.ts +1 -1
- package/lib/channel-header/channel-header.component.d.ts +2 -2
- package/lib/channel-list/channel-list.component.d.ts +3 -4
- package/lib/channel-preview/channel-preview.component.d.ts +6 -6
- package/lib/channel-query.d.ts +26 -0
- package/lib/channel.service.d.ts +146 -154
- package/lib/chat-client.service.d.ts +17 -15
- package/lib/custom-templates.service.d.ts +50 -50
- package/lib/get-channel-display-text.d.ts +1 -2
- package/lib/get-message-translation.d.ts +3 -3
- package/lib/icon/icon-placeholder/icon-placeholder.component.d.ts +2 -2
- package/lib/icon/icon.component.d.ts +2 -2
- package/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.d.ts +1 -1
- package/lib/message/message.component.d.ts +6 -6
- package/lib/message-actions-box/message-actions-box.component.d.ts +5 -4
- package/lib/message-actions.service.d.ts +10 -10
- package/lib/message-bounce-prompt/message-bounce-prompt.component.d.ts +1 -1
- package/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.d.ts +5 -5
- package/lib/message-input/emoji-input.service.d.ts +2 -2
- package/lib/message-input/message-input-config.service.d.ts +3 -3
- package/lib/message-input/message-input.component.d.ts +10 -33
- package/lib/message-input/textarea/textarea.component.d.ts +3 -3
- package/lib/message-input/textarea.directive.d.ts +1 -1
- package/lib/message-list/group-styles.d.ts +1 -1
- package/lib/message-list/message-list.component.d.ts +3 -2
- package/lib/message-preview.d.ts +2 -3
- package/lib/message-reactions/message-reactions.component.d.ts +9 -8
- package/lib/message-reactions-selector/message-reactions-selector.component.d.ts +6 -5
- package/lib/message-reactions.service.d.ts +4 -4
- package/lib/message-text/message-text.component.d.ts +5 -5
- package/lib/message.service.d.ts +6 -7
- package/lib/modal/modal.component.d.ts +1 -1
- package/lib/notification/notification.component.d.ts +2 -2
- package/lib/notification-list/notification-list.component.d.ts +1 -0
- package/lib/notification.service.d.ts +1 -1
- package/lib/paginated-list/paginated-list.component.d.ts +2 -5
- package/lib/read-by.d.ts +1 -2
- package/lib/stream-i18n.service.d.ts +1 -1
- package/lib/theme.service.d.ts +1 -1
- package/lib/thread/thread.component.d.ts +3 -3
- package/lib/types-custom.d.ts +15 -0
- package/lib/types.d.ts +116 -155
- package/lib/user-list/user-list.component.d.ts +2 -3
- package/lib/virtualized-message-list.service.d.ts +1 -1
- package/lib/voice-recorder/amplitude-recorder.service.d.ts +2 -2
- package/lib/voice-recorder/media-recorder.d.ts +2 -2
- package/lib/voice-recorder/transcoder.service.d.ts +4 -4
- package/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.d.ts +1 -0
- package/lib/voice-recorder/voice-recorder.component.d.ts +2 -2
- package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +3 -3
- package/lib/voice-recording/voice-recording.component.d.ts +2 -3
- package/package.json +21 -15
- package/public-api.d.ts +3 -0
- package/src/assets/i18n/en.ts +0 -1
- package/src/assets/version.ts +1 -1
- package/esm2022/lib/attachment-configuration.service.mjs +0 -185
- package/esm2022/lib/attachment-list/attachment-list.component.mjs +0 -212
- package/esm2022/lib/attachment-preview-list/attachment-preview-list.component.mjs +0 -55
- package/esm2022/lib/attachment.service.mjs +0 -479
- package/esm2022/lib/avatar/avatar.component.mjs +0 -157
- package/esm2022/lib/avatar-placeholder/avatar-placeholder.component.mjs +0 -66
- package/esm2022/lib/channel/channel.component.mjs +0 -45
- package/esm2022/lib/channel-header/channel-header.component.mjs +0 -72
- package/esm2022/lib/channel-list/channel-list.component.mjs +0 -50
- package/esm2022/lib/channel-preview/channel-preview.component.mjs +0 -150
- package/esm2022/lib/channel.service.mjs +0 -1393
- package/esm2022/lib/chat-client.service.mjs +0 -227
- package/esm2022/lib/custom-templates.service.mjs +0 -244
- package/esm2022/lib/file-utils.mjs +0 -35
- package/esm2022/lib/get-message-translation.mjs +0 -12
- package/esm2022/lib/icon/icon-placeholder/icon-placeholder.component.mjs +0 -28
- package/esm2022/lib/icon/loading-indicator-placeholder/loading-indicator-placeholder.component.mjs +0 -20
- package/esm2022/lib/message/message.component.mjs +0 -486
- package/esm2022/lib/message-actions-box/message-actions-box.component.mjs +0 -120
- package/esm2022/lib/message-actions.service.mjs +0 -187
- package/esm2022/lib/message-bounce-prompt/message-bounce-prompt.component.mjs +0 -71
- package/esm2022/lib/message-input/autocomplete-textarea/autocomplete-textarea.component.mjs +0 -333
- package/esm2022/lib/message-input/message-input-config.service.mjs +0 -50
- package/esm2022/lib/message-input/message-input.component.mjs +0 -507
- package/esm2022/lib/message-list/message-list.component.mjs +0 -715
- package/esm2022/lib/message-preview.mjs +0 -21
- package/esm2022/lib/message-reactions/message-reactions.component.mjs +0 -165
- package/esm2022/lib/message-reactions-selector/message-reactions-selector.component.mjs +0 -57
- package/esm2022/lib/message-text/message-text.component.mjs +0 -143
- package/esm2022/lib/message.service.mjs +0 -43
- package/esm2022/lib/notification/notification.component.mjs +0 -20
- package/esm2022/lib/notification-list/notification-list.component.mjs +0 -33
- package/esm2022/lib/paginated-list/paginated-list.component.mjs +0 -94
- package/esm2022/lib/read-by.mjs +0 -12
- package/esm2022/lib/stream-autocomplete-textarea.module.mjs +0 -33
- package/esm2022/lib/stream-textarea.module.mjs +0 -31
- package/esm2022/lib/thread/thread.component.mjs +0 -51
- package/esm2022/lib/types.mjs +0 -2
- package/esm2022/lib/user-list/user-list.component.mjs +0 -47
- package/esm2022/lib/virtualized-list.service.mjs +0 -273
- package/esm2022/lib/voice-recorder/mp3-transcoder.mjs +0 -61
- package/esm2022/lib/voice-recorder/transcoder.service.mjs +0 -121
- package/esm2022/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs +0 -32
- package/esm2022/lib/voice-recorder/voice-recorder.component.mjs +0 -80
- package/esm2022/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +0 -112
- package/esm2022/lib/voice-recording/voice-recording.component.mjs +0 -91
- package/esm2022/public-api.mjs +0 -82
- package/fesm2022/stream-chat-angular.mjs.map +0 -1
- /package/{esm2022 → esm2020}/lib/format-duration.mjs +0 -0
- /package/{esm2022 → esm2020}/lib/injection-tokens.mjs +0 -0
- /package/{esm2022 → esm2020}/lib/is-image-attachment.mjs +0 -0
- /package/{esm2022 → esm2020}/lib/is-on-separate-date.mjs +0 -0
- /package/{esm2022 → esm2020}/lib/is-safari.mjs +0 -0
- /package/{esm2022 → esm2020}/lib/message-input/textarea.interface.mjs +0 -0
- /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"]}
|