stream-chat-angular 5.3.2 → 5.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/assets/i18n/en.d.ts +4 -0
- package/assets/version.d.ts +1 -1
- package/esm2020/assets/i18n/en.mjs +5 -1
- package/esm2020/assets/version.mjs +2 -2
- package/esm2020/lib/attachment-list/attachment-list.component.mjs +4 -4
- package/esm2020/lib/attachment-preview-list/attachment-preview-list.component.mjs +5 -5
- package/esm2020/lib/attachment.service.mjs +44 -10
- package/esm2020/lib/channel-list/channel-list.component.mjs +3 -3
- package/esm2020/lib/channel-preview/channel-preview.component.mjs +1 -1
- package/esm2020/lib/file-utils.mjs +35 -0
- package/esm2020/lib/format-duration.mjs +16 -0
- package/esm2020/lib/icon/icon-placeholder/icon-placeholder.component.mjs +28 -0
- package/esm2020/lib/icon/icon.component.mjs +1 -1
- package/esm2020/lib/icon/icon.module.mjs +37 -0
- package/esm2020/lib/{loading-indicator → icon/loading-indicator}/loading-indicator.component.mjs +1 -1
- package/esm2020/lib/{loading-indicator-placeholder → icon/loading-indicator-placeholder}/loading-indicator-placeholder.component.mjs +2 -2
- package/esm2020/lib/is-safari.mjs +2 -0
- package/esm2020/lib/message/message.component.mjs +6 -6
- package/esm2020/lib/message-input/message-input-config.service.mjs +6 -1
- package/esm2020/lib/message-input/message-input.component.mjs +57 -14
- package/esm2020/lib/message-input/voice-recorder.service.mjs +27 -0
- package/esm2020/lib/message-list/message-list.component.mjs +9 -9
- package/esm2020/lib/modal/modal.component.mjs +1 -1
- package/esm2020/lib/paginated-list/paginated-list.component.mjs +1 -1
- package/esm2020/lib/stream-chat.module.mjs +21 -35
- package/esm2020/lib/thread/thread.component.mjs +1 -1
- package/esm2020/lib/types.mjs +1 -1
- package/esm2020/lib/voice-recorder/amplitude-recorder.service.mjs +119 -0
- package/esm2020/lib/voice-recorder/audio-recorder.service.mjs +79 -0
- package/esm2020/lib/voice-recorder/media-recorder.mjs +190 -0
- 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/esm2020/lib/voice-recorder/voice-recorder.module.mjs +34 -0
- package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs +4 -75
- package/esm2020/lib/voice-recording/voice-recording.component.mjs +4 -15
- package/esm2020/lib/voice-recording/voice-recording.module.mjs +21 -0
- package/esm2020/lib/wave-form-sampler.mjs +72 -0
- package/esm2020/public-api.mjs +18 -5
- package/fesm2015/stream-chat-angular.mjs +1039 -145
- package/fesm2015/stream-chat-angular.mjs.map +1 -1
- package/fesm2020/stream-chat-angular.mjs +990 -140
- package/fesm2020/stream-chat-angular.mjs.map +1 -1
- package/lib/attachment.service.d.ts +7 -1
- package/lib/file-utils.d.ts +9 -0
- package/lib/format-duration.d.ts +1 -0
- package/lib/{icon-placeholder → icon/icon-placeholder}/icon-placeholder.component.d.ts +3 -3
- package/lib/icon/icon.component.d.ts +1 -1
- package/lib/icon/icon.module.d.ts +11 -0
- package/lib/{loading-indicator-placeholder → icon/loading-indicator-placeholder}/loading-indicator-placeholder.component.d.ts +1 -1
- package/lib/is-safari.d.ts +1 -0
- package/lib/message-input/message-input-config.service.d.ts +5 -0
- package/lib/message-input/message-input.component.d.ts +19 -5
- package/lib/message-input/voice-recorder.service.d.ts +19 -0
- package/lib/message-list/message-list.component.d.ts +0 -1
- package/lib/stream-chat.module.d.ts +20 -24
- package/lib/types.d.ts +11 -1
- package/lib/voice-recorder/amplitude-recorder.service.d.ts +71 -0
- package/lib/voice-recorder/audio-recorder.service.d.ts +46 -0
- package/lib/voice-recorder/media-recorder.d.ts +46 -0
- package/lib/voice-recorder/mp3-transcoder.d.ts +1 -0
- package/lib/voice-recorder/transcoder.service.d.ts +40 -0
- package/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.d.ts +21 -0
- package/lib/voice-recorder/voice-recorder.component.d.ts +30 -0
- package/lib/voice-recorder/voice-recorder.module.d.ts +12 -0
- package/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.d.ts +0 -7
- package/lib/voice-recording/voice-recording.component.d.ts +0 -1
- package/lib/voice-recording/voice-recording.module.d.ts +11 -0
- package/lib/wave-form-sampler.d.ts +1 -0
- package/package.json +8 -1
- package/public-api.d.ts +17 -4
- package/src/assets/assets/icons/stream-chat-icons.eot +0 -0
- package/src/assets/assets/icons/stream-chat-icons.svg +4 -0
- package/src/assets/assets/icons/stream-chat-icons.ttf +0 -0
- package/src/assets/assets/icons/stream-chat-icons.woff +0 -0
- package/src/assets/assets/icons/stream-chat-icons.woff2 +0 -0
- package/src/assets/i18n/en.ts +6 -0
- package/src/assets/styles/css/index.css +1 -1
- package/src/assets/styles/css/index.layout.css +1 -1
- package/src/assets/styles/scss/AudioRecorder/AudioRecorder-layout.scss +64 -14
- package/src/assets/styles/scss/AudioRecorder/AudioRecorder-theme.scss +11 -1
- package/src/assets/styles/scss/Icon/Icon-layout.scss +6 -1
- package/src/assets/styles/scss/MessageInput/MessageInput-layout.scss +1 -0
- package/src/assets/styles/scss/MessageInput/MessageInput-theme.scss +1 -0
- package/src/assets/version.ts +1 -1
- package/esm2020/lib/icon-placeholder/icon-placeholder.component.mjs +0 -28
- package/esm2020/lib/is-image-file.mjs +0 -5
- package/lib/is-image-file.d.ts +0 -1
- /package/lib/{loading-indicator → icon/loading-indicator}/loading-indicator.component.d.ts +0 -0
|
@@ -7,8 +7,6 @@ import { MessageComponent } from './message/message.component';
|
|
|
7
7
|
import { MessageInputComponent } from './message-input/message-input.component';
|
|
8
8
|
import { MessageListComponent } from './message-list/message-list.component';
|
|
9
9
|
import { CommonModule } from '@angular/common';
|
|
10
|
-
import { LoadingIndicatorComponent } from './loading-indicator/loading-indicator.component';
|
|
11
|
-
import { IconComponent } from './icon/icon.component';
|
|
12
10
|
import { MessageActionsBoxComponent } from './message-actions-box/message-actions-box.component';
|
|
13
11
|
import { AttachmentListComponent } from './attachment-list/attachment-list.component';
|
|
14
12
|
import { MessageReactionsComponent } from './message-reactions/message-reactions.component';
|
|
@@ -19,16 +17,15 @@ import { ModalComponent } from './modal/modal.component';
|
|
|
19
17
|
import { TextareaDirective } from './message-input/textarea.directive';
|
|
20
18
|
import { StreamAvatarModule } from './stream-avatar.module';
|
|
21
19
|
import { ThreadComponent } from './thread/thread.component';
|
|
22
|
-
import { IconPlaceholderComponent } from './icon-placeholder/icon-placeholder.component';
|
|
23
|
-
import { LoadingIndicatorPlaceholderComponent } from './loading-indicator-placeholder/loading-indicator-placeholder.component';
|
|
24
20
|
import { MessageBouncePromptComponent } from './message-bounce-prompt/message-bounce-prompt.component';
|
|
25
|
-
import { VoiceRecordingComponent } from './voice-recording/voice-recording.component';
|
|
26
|
-
import { VoiceRecordingWavebarComponent } from './voice-recording/voice-recording-wavebar/voice-recording-wavebar.component';
|
|
27
21
|
import { NgxFloatUiModule } from 'ngx-float-ui';
|
|
28
22
|
import { TranslateModule } from '@ngx-translate/core';
|
|
29
23
|
import { MessageReactionsSelectorComponent } from './message-reactions-selector/message-reactions-selector.component';
|
|
30
24
|
import { PaginatedListComponent } from './paginated-list/paginated-list.component';
|
|
31
25
|
import { UserListComponent } from './user-list/user-list.component';
|
|
26
|
+
import { VoiceRecordingModule } from './voice-recording/voice-recording.module';
|
|
27
|
+
import { IconModule } from './icon/icon.module';
|
|
28
|
+
import { VoiceRecorderService } from './message-input/voice-recorder.service';
|
|
32
29
|
import * as i0 from "@angular/core";
|
|
33
30
|
export class StreamChatModule {
|
|
34
31
|
}
|
|
@@ -40,8 +37,6 @@ StreamChatModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", versio
|
|
|
40
37
|
MessageComponent,
|
|
41
38
|
MessageInputComponent,
|
|
42
39
|
MessageListComponent,
|
|
43
|
-
LoadingIndicatorComponent,
|
|
44
|
-
IconComponent,
|
|
45
40
|
MessageActionsBoxComponent,
|
|
46
41
|
AttachmentListComponent,
|
|
47
42
|
MessageReactionsComponent,
|
|
@@ -51,25 +46,21 @@ StreamChatModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", versio
|
|
|
51
46
|
ModalComponent,
|
|
52
47
|
TextareaDirective,
|
|
53
48
|
ThreadComponent,
|
|
54
|
-
IconPlaceholderComponent,
|
|
55
|
-
LoadingIndicatorPlaceholderComponent,
|
|
56
49
|
MessageBouncePromptComponent,
|
|
57
|
-
VoiceRecordingComponent,
|
|
58
|
-
VoiceRecordingWavebarComponent,
|
|
59
50
|
MessageReactionsSelectorComponent,
|
|
60
51
|
UserListComponent,
|
|
61
52
|
PaginatedListComponent], imports: [CommonModule,
|
|
62
53
|
NgxFloatUiModule,
|
|
63
54
|
StreamAvatarModule,
|
|
64
|
-
TranslateModule
|
|
55
|
+
TranslateModule,
|
|
56
|
+
VoiceRecordingModule,
|
|
57
|
+
IconModule], exports: [ChannelComponent,
|
|
65
58
|
ChannelHeaderComponent,
|
|
66
59
|
ChannelListComponent,
|
|
67
60
|
ChannelPreviewComponent,
|
|
68
61
|
MessageComponent,
|
|
69
62
|
MessageInputComponent,
|
|
70
63
|
MessageListComponent,
|
|
71
|
-
LoadingIndicatorComponent,
|
|
72
|
-
IconComponent,
|
|
73
64
|
MessageActionsBoxComponent,
|
|
74
65
|
AttachmentListComponent,
|
|
75
66
|
MessageReactionsComponent,
|
|
@@ -79,18 +70,20 @@ StreamChatModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", versio
|
|
|
79
70
|
ModalComponent,
|
|
80
71
|
StreamAvatarModule,
|
|
81
72
|
ThreadComponent,
|
|
82
|
-
IconPlaceholderComponent,
|
|
83
|
-
LoadingIndicatorPlaceholderComponent,
|
|
84
73
|
MessageBouncePromptComponent,
|
|
85
|
-
|
|
86
|
-
VoiceRecordingWavebarComponent,
|
|
74
|
+
VoiceRecordingModule,
|
|
87
75
|
MessageReactionsSelectorComponent,
|
|
88
76
|
UserListComponent,
|
|
89
|
-
PaginatedListComponent
|
|
90
|
-
|
|
77
|
+
PaginatedListComponent,
|
|
78
|
+
IconModule] });
|
|
79
|
+
StreamChatModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: StreamChatModule, providers: [VoiceRecorderService], imports: [CommonModule,
|
|
91
80
|
NgxFloatUiModule,
|
|
92
81
|
StreamAvatarModule,
|
|
93
|
-
TranslateModule,
|
|
82
|
+
TranslateModule,
|
|
83
|
+
VoiceRecordingModule,
|
|
84
|
+
IconModule, StreamAvatarModule,
|
|
85
|
+
VoiceRecordingModule,
|
|
86
|
+
IconModule] });
|
|
94
87
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: StreamChatModule, decorators: [{
|
|
95
88
|
type: NgModule,
|
|
96
89
|
args: [{
|
|
@@ -102,8 +95,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImpor
|
|
|
102
95
|
MessageComponent,
|
|
103
96
|
MessageInputComponent,
|
|
104
97
|
MessageListComponent,
|
|
105
|
-
LoadingIndicatorComponent,
|
|
106
|
-
IconComponent,
|
|
107
98
|
MessageActionsBoxComponent,
|
|
108
99
|
AttachmentListComponent,
|
|
109
100
|
MessageReactionsComponent,
|
|
@@ -113,11 +104,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImpor
|
|
|
113
104
|
ModalComponent,
|
|
114
105
|
TextareaDirective,
|
|
115
106
|
ThreadComponent,
|
|
116
|
-
IconPlaceholderComponent,
|
|
117
|
-
LoadingIndicatorPlaceholderComponent,
|
|
118
107
|
MessageBouncePromptComponent,
|
|
119
|
-
VoiceRecordingComponent,
|
|
120
|
-
VoiceRecordingWavebarComponent,
|
|
121
108
|
MessageReactionsSelectorComponent,
|
|
122
109
|
UserListComponent,
|
|
123
110
|
PaginatedListComponent,
|
|
@@ -127,6 +114,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImpor
|
|
|
127
114
|
NgxFloatUiModule,
|
|
128
115
|
StreamAvatarModule,
|
|
129
116
|
TranslateModule,
|
|
117
|
+
VoiceRecordingModule,
|
|
118
|
+
IconModule,
|
|
130
119
|
],
|
|
131
120
|
exports: [
|
|
132
121
|
ChannelComponent,
|
|
@@ -136,8 +125,6 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImpor
|
|
|
136
125
|
MessageComponent,
|
|
137
126
|
MessageInputComponent,
|
|
138
127
|
MessageListComponent,
|
|
139
|
-
LoadingIndicatorComponent,
|
|
140
|
-
IconComponent,
|
|
141
128
|
MessageActionsBoxComponent,
|
|
142
129
|
AttachmentListComponent,
|
|
143
130
|
MessageReactionsComponent,
|
|
@@ -147,15 +134,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImpor
|
|
|
147
134
|
ModalComponent,
|
|
148
135
|
StreamAvatarModule,
|
|
149
136
|
ThreadComponent,
|
|
150
|
-
IconPlaceholderComponent,
|
|
151
|
-
LoadingIndicatorPlaceholderComponent,
|
|
152
137
|
MessageBouncePromptComponent,
|
|
153
|
-
|
|
154
|
-
VoiceRecordingWavebarComponent,
|
|
138
|
+
VoiceRecordingModule,
|
|
155
139
|
MessageReactionsSelectorComponent,
|
|
156
140
|
UserListComponent,
|
|
157
141
|
PaginatedListComponent,
|
|
142
|
+
IconModule,
|
|
158
143
|
],
|
|
144
|
+
providers: [VoiceRecorderService],
|
|
159
145
|
}]
|
|
160
146
|
}] });
|
|
161
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stream-chat.module.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/stream-chat.module.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgModule } from '@angular/core';\nimport { ChannelComponent } from './channel/channel.component';\nimport { ChannelHeaderComponent } from './channel-header/channel-header.component';\nimport { ChannelListComponent } from './channel-list/channel-list.component';\nimport { ChannelPreviewComponent } from './channel-preview/channel-preview.component';\nimport { MessageComponent } from './message/message.component';\nimport { MessageInputComponent } from './message-input/message-input.component';\nimport { MessageListComponent } from './message-list/message-list.component';\nimport { CommonModule } from '@angular/common';\nimport { LoadingIndicatorComponent } from './loading-indicator/loading-indicator.component';\nimport { IconComponent } from './icon/icon.component';\nimport { MessageActionsBoxComponent } from './message-actions-box/message-actions-box.component';\nimport { AttachmentListComponent } from './attachment-list/attachment-list.component';\nimport { MessageReactionsComponent } from './message-reactions/message-reactions.component';\nimport { NotificationComponent } from './notification/notification.component';\nimport { NotificationListComponent } from './notification-list/notification-list.component';\nimport { AttachmentPreviewListComponent } from './attachment-preview-list/attachment-preview-list.component';\nimport { ModalComponent } from './modal/modal.component';\nimport { TextareaDirective } from './message-input/textarea.directive';\nimport { StreamAvatarModule } from './stream-avatar.module';\nimport { ThreadComponent } from './thread/thread.component';\nimport { IconPlaceholderComponent } from './icon-placeholder/icon-placeholder.component';\nimport { LoadingIndicatorPlaceholderComponent } from './loading-indicator-placeholder/loading-indicator-placeholder.component';\nimport { MessageBouncePromptComponent } from './message-bounce-prompt/message-bounce-prompt.component';\nimport { VoiceRecordingComponent } from './voice-recording/voice-recording.component';\nimport { VoiceRecordingWavebarComponent } from './voice-recording/voice-recording-wavebar/voice-recording-wavebar.component';\nimport { NgxFloatUiModule } from 'ngx-float-ui';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { MessageReactionsSelectorComponent } from './message-reactions-selector/message-reactions-selector.component';\nimport { PaginatedListComponent } from './paginated-list/paginated-list.component';\nimport { UserListComponent } from './user-list/user-list.component';\n\n@NgModule({\n  declarations: [\n    ChannelComponent,\n    ChannelHeaderComponent,\n    ChannelListComponent,\n    ChannelPreviewComponent,\n    MessageComponent,\n    MessageInputComponent,\n    MessageListComponent,\n    LoadingIndicatorComponent,\n    IconComponent,\n    MessageActionsBoxComponent,\n    AttachmentListComponent,\n    MessageReactionsComponent,\n    NotificationComponent,\n    NotificationListComponent,\n    AttachmentPreviewListComponent,\n    ModalComponent,\n    TextareaDirective,\n    ThreadComponent,\n    IconPlaceholderComponent,\n    LoadingIndicatorPlaceholderComponent,\n    MessageBouncePromptComponent,\n    VoiceRecordingComponent,\n    VoiceRecordingWavebarComponent,\n    MessageReactionsSelectorComponent,\n    UserListComponent,\n    PaginatedListComponent,\n  ],\n  imports: [\n    CommonModule,\n    NgxFloatUiModule,\n    StreamAvatarModule,\n    TranslateModule,\n  ],\n  exports: [\n    ChannelComponent,\n    ChannelHeaderComponent,\n    ChannelListComponent,\n    ChannelPreviewComponent,\n    MessageComponent,\n    MessageInputComponent,\n    MessageListComponent,\n    LoadingIndicatorComponent,\n    IconComponent,\n    MessageActionsBoxComponent,\n    AttachmentListComponent,\n    MessageReactionsComponent,\n    NotificationComponent,\n    NotificationListComponent,\n    AttachmentPreviewListComponent,\n    ModalComponent,\n    StreamAvatarModule,\n    ThreadComponent,\n    IconPlaceholderComponent,\n    LoadingIndicatorPlaceholderComponent,\n    MessageBouncePromptComponent,\n    VoiceRecordingComponent,\n    VoiceRecordingWavebarComponent,\n    MessageReactionsSelectorComponent,\n    UserListComponent,\n    PaginatedListComponent,\n  ],\n})\nexport class StreamChatModule {}\n"]}
|
|
147
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"stream-chat.module.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/stream-chat.module.ts"],"names":[],"mappings":"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","sourcesContent":["import { NgModule } from '@angular/core';\nimport { ChannelComponent } from './channel/channel.component';\nimport { ChannelHeaderComponent } from './channel-header/channel-header.component';\nimport { ChannelListComponent } from './channel-list/channel-list.component';\nimport { ChannelPreviewComponent } from './channel-preview/channel-preview.component';\nimport { MessageComponent } from './message/message.component';\nimport { MessageInputComponent } from './message-input/message-input.component';\nimport { MessageListComponent } from './message-list/message-list.component';\nimport { CommonModule } from '@angular/common';\nimport { MessageActionsBoxComponent } from './message-actions-box/message-actions-box.component';\nimport { AttachmentListComponent } from './attachment-list/attachment-list.component';\nimport { MessageReactionsComponent } from './message-reactions/message-reactions.component';\nimport { NotificationComponent } from './notification/notification.component';\nimport { NotificationListComponent } from './notification-list/notification-list.component';\nimport { AttachmentPreviewListComponent } from './attachment-preview-list/attachment-preview-list.component';\nimport { ModalComponent } from './modal/modal.component';\nimport { TextareaDirective } from './message-input/textarea.directive';\nimport { StreamAvatarModule } from './stream-avatar.module';\nimport { ThreadComponent } from './thread/thread.component';\nimport { MessageBouncePromptComponent } from './message-bounce-prompt/message-bounce-prompt.component';\nimport { NgxFloatUiModule } from 'ngx-float-ui';\nimport { TranslateModule } from '@ngx-translate/core';\nimport { MessageReactionsSelectorComponent } from './message-reactions-selector/message-reactions-selector.component';\nimport { PaginatedListComponent } from './paginated-list/paginated-list.component';\nimport { UserListComponent } from './user-list/user-list.component';\nimport { VoiceRecordingModule } from './voice-recording/voice-recording.module';\nimport { IconModule } from './icon/icon.module';\nimport { VoiceRecorderService } from './message-input/voice-recorder.service';\n\n@NgModule({\n  declarations: [\n    ChannelComponent,\n    ChannelHeaderComponent,\n    ChannelListComponent,\n    ChannelPreviewComponent,\n    MessageComponent,\n    MessageInputComponent,\n    MessageListComponent,\n    MessageActionsBoxComponent,\n    AttachmentListComponent,\n    MessageReactionsComponent,\n    NotificationComponent,\n    NotificationListComponent,\n    AttachmentPreviewListComponent,\n    ModalComponent,\n    TextareaDirective,\n    ThreadComponent,\n    MessageBouncePromptComponent,\n    MessageReactionsSelectorComponent,\n    UserListComponent,\n    PaginatedListComponent,\n  ],\n  imports: [\n    CommonModule,\n    NgxFloatUiModule,\n    StreamAvatarModule,\n    TranslateModule,\n    VoiceRecordingModule,\n    IconModule,\n  ],\n  exports: [\n    ChannelComponent,\n    ChannelHeaderComponent,\n    ChannelListComponent,\n    ChannelPreviewComponent,\n    MessageComponent,\n    MessageInputComponent,\n    MessageListComponent,\n    MessageActionsBoxComponent,\n    AttachmentListComponent,\n    MessageReactionsComponent,\n    NotificationComponent,\n    NotificationListComponent,\n    AttachmentPreviewListComponent,\n    ModalComponent,\n    StreamAvatarModule,\n    ThreadComponent,\n    MessageBouncePromptComponent,\n    VoiceRecordingModule,\n    MessageReactionsSelectorComponent,\n    UserListComponent,\n    PaginatedListComponent,\n    IconModule,\n  ],\n  providers: [VoiceRecorderService],\n})\nexport class StreamChatModule {}\n"]}
|
|
@@ -6,7 +6,7 @@ import * as i2 from "../channel.service";
|
|
|
6
6
|
import * as i3 from "../chat-client.service";
|
|
7
7
|
import * as i4 from "@angular/common";
|
|
8
8
|
import * as i5 from "@ngx-translate/core";
|
|
9
|
-
import * as i6 from "../icon-placeholder/icon-placeholder.component";
|
|
9
|
+
import * as i6 from "../icon/icon-placeholder/icon-placeholder.component";
|
|
10
10
|
/**
|
|
11
11
|
* The `Thread` component represents a [message thread](https://getstream.io/chat/docs/javascript/threads/?language=javascript), it is a container component that displays a thread with a header, [`MessageList`](./MessageListComponent.mdx) and [`MessageInput`](./MessageInputComponent.mdx) components.
|
|
12
12
|
*/
|
package/esm2020/lib/types.mjs
CHANGED
|
@@ -1,2 +1,2 @@
|
|
|
1
1
|
export {};
|
|
2
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/types.ts"],"names":[],"mappings":"","sourcesContent":["import { TemplateRef } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\nimport type {\n  Attachment,\n  Channel,\n  ChannelFilters,\n  ChannelMemberResponse,\n  CommandResponse,\n  Event,\n  ExtendableGenerics,\n  FormatMessageResponse,\n  LiteralStringForUnion,\n  MessageResponseBase,\n  Mute,\n  ReactionGroupResponse,\n  ReactionResponse,\n  User,\n  UserResponse,\n} from 'stream-chat';\nimport { AttachmentService } from './attachment.service';\nimport { Icon } from './icon/icon.component';\n\nexport type UnknownType = Record<string, unknown>;\n\nexport type CustomTrigger = {\n  [key: string]: {\n    componentProps: UnknownType;\n    data: UnknownType;\n  };\n};\n\nexport type DefaultStreamChatGenerics = ExtendableGenerics & {\n  attachmentType: DefaultAttachmentType;\n  channelType: DefaultChannelType;\n  commandType: LiteralStringForUnion;\n  eventType: UnknownType;\n  messageType: DefaultMessageType;\n  reactionType: UnknownType;\n  userType: DefaultUserType;\n};\n\nexport type DefaultAttachmentType = UnknownType & {\n  asset_url?: string;\n  id?: string;\n  images?: Array<Attachment<DefaultStreamChatGenerics>>;\n  mime_type?: string;\n  isCustomAttachment?: boolean;\n};\n\nexport type DefaultChannelType = UnknownType & {\n  image?: string;\n  member_count?: number;\n  subtitle?: string;\n};\n\nexport type DefaultCommandType = LiteralStringForUnion;\n\nexport type DefaultMessageType = UnknownType & {\n  customType?: 'channel.intro' | 'message.date';\n  date?: string | Date;\n  errorStatusCode?: number;\n  event?: Event<DefaultStreamChatGenerics>;\n  unread?: boolean;\n  readBy: UserResponse<DefaultStreamChatGenerics>[];\n  translation?: string;\n  quoted_message?: MessageResponseBase<DefaultStreamChatGenerics>;\n};\n\nexport type DefaultUserTypeInternal = {\n  image?: string;\n  status?: string;\n};\n\nexport type DefaultUserType = UnknownType &\n  DefaultUserTypeInternal & {\n    mutes?: Array<Mute<DefaultStreamChatGenerics>>;\n  };\n\nexport type StreamMessage<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = FormatMessageResponse<T>;\n\nexport type AttachmentUploadErrorReason =\n  | 'file-size'\n  | 'file-extension'\n  | 'unknown';\n\nexport type AttachmentUpload<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  file: File;\n  state: 'error' | 'success' | 'uploading';\n  errorReason?: AttachmentUploadErrorReason;\n  errorExtraInfo?: { param: string }[];\n  url?: string;\n  type: 'image' | 'file' | 'video';\n  previewUri?: string | ArrayBuffer;\n  thumb_url?: string;\n  fromAttachment?: Attachment<T>;\n};\n\nexport type MentionAutcompleteListItemContext = {\n  item: MentionAutcompleteListItem;\n};\n\nexport type CommandAutocompleteListItemContext = {\n  item: ComandAutocompleteListItem;\n};\n\nexport type MentionAutcompleteListItem = (\n  | ChannelMemberResponse\n  | UserResponse\n) & {\n  autocompleteLabel: string;\n};\n\nexport type ComandAutocompleteListItem = CommandResponse & {\n  autocompleteLabel: string;\n};\n\nexport type NotificationType = 'success' | 'error' | 'info';\n\nexport type NotificationPayload<T = object> = {\n  id: string;\n  type: NotificationType;\n  text?: string;\n  translateParams?: object;\n  template?: TemplateRef<T>;\n  templateContext?: T;\n  dismissFn: () => void;\n};\n\nexport type ChannelPreviewContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  channel: Channel<T>;\n};\n\nexport type ChannelPreviewInfoContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = ChannelPreviewContext & {\n  latestMessage?: StreamMessage<T>;\n  /**\n   * The text of the latest message, or some meta information (for example: \"Nothing yet\")\n   */\n  latestMessageText: string;\n  /**\n   * The title of the channel, or the name of the channel members\n   */\n  channelDisplayTitle: string;\n  /**\n   * The status of the last message (only available if the last message was sent by the current user)\n   */\n  latestMessageStatus?: 'delivered' | 'read';\n  /**\n   * The time of the last message (formatted to a user-friendly string)\n   */\n  latestMessageTime?: string;\n  unreadCount: number;\n};\n\nexport type MentionTemplateContext = {\n  content: string;\n  user: UserResponse;\n};\n\nexport type EmojiPickerContext = {\n  emojiInput$: Subject<string>;\n};\n\nexport type TypingIndicatorContext = {\n  usersTyping$: Observable<UserResponse<DefaultStreamChatGenerics>[]>;\n};\n\nexport type MessageContext = {\n  message: StreamMessage | undefined;\n  enabledMessageActions: string[];\n  isLastSentMessage: boolean | undefined;\n  mode: 'thread' | 'main';\n  isHighlighted: boolean;\n  customActions: CustomMessageActionItem[];\n  scroll$?: Observable<void>;\n};\n\nexport type ChannelActionsContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = { channel: Channel<T> };\n\nexport type AttachmentListContext = {\n  messageId: string;\n  attachments: Attachment<DefaultStreamChatGenerics>[];\n  parentMessageId?: string;\n  imageModalStateChangeHandler?: (state: 'opened' | 'closed') => void;\n};\n\nexport type AvatarType = 'channel' | 'user';\n\nexport type AvatarLocation =\n  | 'channel-preview'\n  | 'channel-header'\n  | 'message-sender'\n  | 'message-reader'\n  | 'quoted-message-sender'\n  | 'autocomplete-item'\n  | 'typing-indicator'\n  /**\n   * @deprecated this will be renamed to user-list in the next major release\n   */\n  | 'reaction';\n\nexport type AvatarContext = {\n  name: string | undefined;\n  imageUrl: string | undefined;\n  type: AvatarType | undefined;\n  location: AvatarLocation | undefined;\n  channel?: Channel<DefaultStreamChatGenerics>;\n  user?: User<DefaultStreamChatGenerics>;\n  initialsType?: 'first-letter-of-first-word' | 'first-letter-of-each-word';\n  showOnlineIndicator?: boolean;\n};\n\nexport type AttachmentPreviewListContext = {\n  attachmentUploads$: Observable<AttachmentUpload[]> | undefined;\n  retryUploadHandler: (f: File) => void;\n  deleteUploadHandler: (u: AttachmentUpload) => void;\n};\n\nexport type IconContext = {\n  icon: Icon | undefined;\n};\n\nexport type MessageActionsBoxContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  isMine: boolean;\n  message: StreamMessage<T> | undefined;\n  enabledActions: string[];\n  messageTextHtmlElement: HTMLElement | undefined;\n};\n\nexport type MessageActionHandlerExtraParams = {\n  isMine: boolean;\n  messageTextHtmlElement?: HTMLElement;\n};\n\nexport type MessageActionHandler<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = (\n  message: StreamMessage<T>,\n  params: MessageActionHandlerExtraParams\n) => void;\n\nexport type MessageActionBoxItemContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  actionName: string;\n  actionLabelOrTranslationKey: ((message: StreamMessage<T>) => string) | string;\n  message: StreamMessage<T>;\n  actionHandlerExtraParams: MessageActionHandlerExtraParams;\n  actionHandler: MessageActionHandler<T>;\n};\n\nexport type MessageReactionActionItem<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  actionName: 'react';\n  isVisible: (\n    enabledActions: string[],\n    isMine: boolean,\n    message: StreamMessage<T>\n  ) => boolean;\n};\n\ntype MessageActionItemBase<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  actionLabelOrTranslationKey: ((message: StreamMessage<T>) => string) | string;\n  isVisible: (\n    enabledActions: string[],\n    isMine: boolean,\n    message: StreamMessage<T>\n  ) => boolean;\n  actionHandler: MessageActionHandler;\n};\n\nexport type MessageActionItem<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = MessageActionItemBase<T> & {\n  actionName:\n    | 'quote'\n    | 'pin'\n    | 'flag'\n    | 'edit'\n    | 'delete'\n    | 'mark-unread'\n    | 'thread-reply'\n    | 'copy-message-text';\n};\n\nexport type CustomMessageActionItem<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = MessageActionItemBase<T> & {\n  actionName: string;\n};\n\nexport type MessageReactionsSelectorContext = {\n  messageId: string | undefined;\n  ownReactions: ReactionResponse<DefaultStreamChatGenerics>[];\n};\n\nexport type MessageReactionsContext = {\n  messageId: string | undefined;\n  /** @deprecated use `messageReactionGroups` */\n  messageReactionCounts: { [key in MessageReactionType]?: number };\n  /** @deprecated you can fetch the reactions using [`chatService.chatClient.queryReactions()`](https://getstream.io/chat/docs/javascript/send_reaction/?language=javascript&q=queryReactions#query-reactions) */\n  latestReactions: ReactionResponse<DefaultStreamChatGenerics>[];\n  ownReactions: ReactionResponse<DefaultStreamChatGenerics>[];\n  messageReactionGroups: {\n    [key in MessageReactionType]: ReactionGroupResponse;\n  };\n};\n\nexport type ModalContext = {\n  isOpen: boolean;\n  isOpenChangeHandler: (isOpen: boolean) => void;\n  content: TemplateRef<void>;\n};\n\nexport type NotificationContext = {\n  type: NotificationType | undefined;\n  content: TemplateRef<void> | undefined;\n};\n\nexport type ThreadHeaderContext = {\n  parentMessage: StreamMessage | undefined;\n  closeThreadHandler: () => void;\n};\n\nexport type MessageReactionType = string;\n\nexport type AttachmentConfigration = {\n  url: string;\n  height: string;\n  width: string;\n};\n\nexport type ImageAttachmentConfiguration = AttachmentConfigration & {\n  originalHeight: number;\n  originalWidth: number;\n};\n\nexport type VideoAttachmentConfiguration = ImageAttachmentConfiguration & {\n  thumbUrl?: string;\n};\n\nexport type DeliveredStatusContext = {\n  message: StreamMessage;\n};\n\nexport type SendingStatusContext = {\n  message: StreamMessage;\n};\n\nexport type CustomMetadataContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  message: StreamMessage<T>;\n};\n\nexport type ReadStatusContext = {\n  message: StreamMessage;\n  readByText: string;\n};\n\nexport type ChannelHeaderInfoContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = { channel: Channel<T> };\n\nexport type CustomAttachmentUploadContext = {\n  isMultipleFileUploadEnabled: boolean | undefined;\n  attachmentService: AttachmentService;\n};\n\nexport type AttachmentContext = {\n  attachment: Attachment<DefaultStreamChatGenerics>;\n};\n\nexport type SystemMessageContext = MessageContext & {\n  parsedDate: string | undefined;\n};\n\nexport type DateSeparatorContext = {\n  date: Date;\n  parsedDate: string;\n};\n\nexport type UnreadMessagesIndicatorContext = {\n  unreadCount: number;\n};\n\nexport type UnreadMessagesNotificationContext =\n  UnreadMessagesIndicatorContext & {\n    onJump: () => void;\n    onDismiss: () => void;\n  };\n\nexport type ChannelQueryState = {\n  state: 'in-progress' | 'success' | 'error';\n  // No type def from stream-chat\n  error?: unknown;\n};\n\nexport type MessageInput<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  text: string;\n  attachments: Attachment<T>[];\n  mentionedUsers: UserResponse<T>[];\n  parentId: string | undefined;\n  quotedMessageId: string | undefined;\n  customData: undefined | Partial<T['messageType']>;\n};\n\nexport type OffsetNextPageConfiguration = {\n  type: 'offset';\n  offset: number;\n};\n\nexport type FiltertNextPageConfiguration<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  type: 'filter';\n  paginationFilter: ChannelFilters<T>;\n};\n\nexport type NextPageConfiguration =\n  | OffsetNextPageConfiguration\n  | FiltertNextPageConfiguration\n  | undefined;\n\nexport type MessageReactionClickDetails = {\n  messageId: string;\n  reactionType: string;\n};\n\nexport type MessageActionsClickDetails<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = MessageActionsBoxContext<T> & { customActions: CustomMessageActionItem[] };\n\nexport type GroupStyleOptions = {\n  noGroupByUser?: boolean;\n  lastReadMessageId?: string;\n  noGroupByReadState?: boolean;\n};\n\nexport type ChannelQueryType = 'first-page' | 'next-page' | 'recover-state';\n\nexport type ChannelQueryResult<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  channels: Channel<T>[];\n  hasMorePage: boolean;\n};\n\nexport type VirtualizedListScrollPosition = 'top' | 'bottom' | 'middle';\n\nexport type VirtualizedListQueryState = {\n  state: 'loading-top' | 'loading-bottom' | 'success' | 'error';\n  error?: unknown;\n};\n\nexport type VirtualizedListQueryDirection = 'top' | 'bottom';\n\nexport type VirtualizedListVerticalItemPosition = 'top' | 'bottom' | 'middle';\n"]}
|
|
2
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../projects/stream-chat-angular/src/lib/types.ts"],"names":[],"mappings":"","sourcesContent":["import { TemplateRef } from '@angular/core';\nimport { Observable, Subject } from 'rxjs';\nimport type {\n  Attachment,\n  Channel,\n  ChannelFilters,\n  ChannelMemberResponse,\n  CommandResponse,\n  Event,\n  ExtendableGenerics,\n  FormatMessageResponse,\n  LiteralStringForUnion,\n  MessageResponseBase,\n  Mute,\n  ReactionGroupResponse,\n  ReactionResponse,\n  User,\n  UserResponse,\n} from 'stream-chat';\nimport { AttachmentService } from './attachment.service';\nimport { Icon } from './icon/icon.component';\n\nexport type UnknownType = Record<string, unknown>;\n\nexport type CustomTrigger = {\n  [key: string]: {\n    componentProps: UnknownType;\n    data: UnknownType;\n  };\n};\n\nexport type DefaultStreamChatGenerics = ExtendableGenerics & {\n  attachmentType: DefaultAttachmentType;\n  channelType: DefaultChannelType;\n  commandType: LiteralStringForUnion;\n  eventType: UnknownType;\n  messageType: DefaultMessageType;\n  reactionType: UnknownType;\n  userType: DefaultUserType;\n};\n\nexport type DefaultAttachmentType = UnknownType & {\n  asset_url?: string;\n  id?: string;\n  images?: Array<Attachment<DefaultStreamChatGenerics>>;\n  mime_type?: string;\n  isCustomAttachment?: boolean;\n};\n\nexport type DefaultChannelType = UnknownType & {\n  image?: string;\n  member_count?: number;\n  subtitle?: string;\n};\n\nexport type DefaultCommandType = LiteralStringForUnion;\n\nexport type DefaultMessageType = UnknownType & {\n  customType?: 'channel.intro' | 'message.date';\n  date?: string | Date;\n  errorStatusCode?: number;\n  event?: Event<DefaultStreamChatGenerics>;\n  unread?: boolean;\n  readBy: UserResponse<DefaultStreamChatGenerics>[];\n  translation?: string;\n  quoted_message?: MessageResponseBase<DefaultStreamChatGenerics>;\n};\n\nexport type DefaultUserTypeInternal = {\n  image?: string;\n  status?: string;\n};\n\nexport type DefaultUserType = UnknownType &\n  DefaultUserTypeInternal & {\n    mutes?: Array<Mute<DefaultStreamChatGenerics>>;\n  };\n\nexport type StreamMessage<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = FormatMessageResponse<T>;\n\nexport type AttachmentUploadErrorReason =\n  | 'file-size'\n  | 'file-extension'\n  | 'unknown';\n\nexport type AttachmentUpload<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  file: File;\n  state: 'error' | 'success' | 'uploading';\n  errorReason?: AttachmentUploadErrorReason;\n  errorExtraInfo?: { param: string }[];\n  url?: string;\n  type: 'image' | 'file' | 'video' | 'voiceRecording';\n  previewUri?: string | ArrayBuffer;\n  thumb_url?: string;\n  extraData?: Partial<Attachment<T>>;\n  fromAttachment?: Attachment<T>;\n};\n\nexport type MentionAutcompleteListItemContext = {\n  item: MentionAutcompleteListItem;\n};\n\nexport type CommandAutocompleteListItemContext = {\n  item: ComandAutocompleteListItem;\n};\n\nexport type MentionAutcompleteListItem = (\n  | ChannelMemberResponse\n  | UserResponse\n) & {\n  autocompleteLabel: string;\n};\n\nexport type ComandAutocompleteListItem = CommandResponse & {\n  autocompleteLabel: string;\n};\n\nexport type NotificationType = 'success' | 'error' | 'info';\n\nexport type NotificationPayload<T = object> = {\n  id: string;\n  type: NotificationType;\n  text?: string;\n  translateParams?: object;\n  template?: TemplateRef<T>;\n  templateContext?: T;\n  dismissFn: () => void;\n};\n\nexport type ChannelPreviewContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  channel: Channel<T>;\n};\n\nexport type ChannelPreviewInfoContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = ChannelPreviewContext & {\n  latestMessage?: StreamMessage<T>;\n  /**\n   * The text of the latest message, or some meta information (for example: \"Nothing yet\")\n   */\n  latestMessageText: string;\n  /**\n   * The title of the channel, or the name of the channel members\n   */\n  channelDisplayTitle: string;\n  /**\n   * The status of the last message (only available if the last message was sent by the current user)\n   */\n  latestMessageStatus?: 'delivered' | 'read';\n  /**\n   * The time of the last message (formatted to a user-friendly string)\n   */\n  latestMessageTime?: string;\n  unreadCount: number;\n};\n\nexport type MentionTemplateContext = {\n  content: string;\n  user: UserResponse;\n};\n\nexport type EmojiPickerContext = {\n  emojiInput$: Subject<string>;\n};\n\nexport type TypingIndicatorContext = {\n  usersTyping$: Observable<UserResponse<DefaultStreamChatGenerics>[]>;\n};\n\nexport type MessageContext = {\n  message: StreamMessage | undefined;\n  enabledMessageActions: string[];\n  isLastSentMessage: boolean | undefined;\n  mode: 'thread' | 'main';\n  isHighlighted: boolean;\n  customActions: CustomMessageActionItem[];\n  scroll$?: Observable<void>;\n};\n\nexport type ChannelActionsContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = { channel: Channel<T> };\n\nexport type AttachmentListContext = {\n  messageId: string;\n  attachments: Attachment<DefaultStreamChatGenerics>[];\n  parentMessageId?: string;\n  imageModalStateChangeHandler?: (state: 'opened' | 'closed') => void;\n};\n\nexport type AvatarType = 'channel' | 'user';\n\nexport type AvatarLocation =\n  | 'channel-preview'\n  | 'channel-header'\n  | 'message-sender'\n  | 'message-reader'\n  | 'quoted-message-sender'\n  | 'autocomplete-item'\n  | 'typing-indicator'\n  /**\n   * @deprecated this will be renamed to user-list in the next major release\n   */\n  | 'reaction';\n\nexport type AvatarContext = {\n  name: string | undefined;\n  imageUrl: string | undefined;\n  type: AvatarType | undefined;\n  location: AvatarLocation | undefined;\n  channel?: Channel<DefaultStreamChatGenerics>;\n  user?: User<DefaultStreamChatGenerics>;\n  initialsType?: 'first-letter-of-first-word' | 'first-letter-of-each-word';\n  showOnlineIndicator?: boolean;\n};\n\nexport type AttachmentPreviewListContext = {\n  attachmentUploads$: Observable<AttachmentUpload[]> | undefined;\n  retryUploadHandler: (f: File) => void;\n  deleteUploadHandler: (u: AttachmentUpload) => void;\n};\n\nexport type IconContext = {\n  icon: Icon | undefined;\n};\n\nexport type MessageActionsBoxContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  isMine: boolean;\n  message: StreamMessage<T> | undefined;\n  enabledActions: string[];\n  messageTextHtmlElement: HTMLElement | undefined;\n};\n\nexport type MessageActionHandlerExtraParams = {\n  isMine: boolean;\n  messageTextHtmlElement?: HTMLElement;\n};\n\nexport type MessageActionHandler<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = (\n  message: StreamMessage<T>,\n  params: MessageActionHandlerExtraParams\n) => void;\n\nexport type MessageActionBoxItemContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  actionName: string;\n  actionLabelOrTranslationKey: ((message: StreamMessage<T>) => string) | string;\n  message: StreamMessage<T>;\n  actionHandlerExtraParams: MessageActionHandlerExtraParams;\n  actionHandler: MessageActionHandler<T>;\n};\n\nexport type MessageReactionActionItem<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  actionName: 'react';\n  isVisible: (\n    enabledActions: string[],\n    isMine: boolean,\n    message: StreamMessage<T>\n  ) => boolean;\n};\n\ntype MessageActionItemBase<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  actionLabelOrTranslationKey: ((message: StreamMessage<T>) => string) | string;\n  isVisible: (\n    enabledActions: string[],\n    isMine: boolean,\n    message: StreamMessage<T>\n  ) => boolean;\n  actionHandler: MessageActionHandler;\n};\n\nexport type MessageActionItem<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = MessageActionItemBase<T> & {\n  actionName:\n    | 'quote'\n    | 'pin'\n    | 'flag'\n    | 'edit'\n    | 'delete'\n    | 'mark-unread'\n    | 'thread-reply'\n    | 'copy-message-text';\n};\n\nexport type CustomMessageActionItem<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = MessageActionItemBase<T> & {\n  actionName: string;\n};\n\nexport type MessageReactionsSelectorContext = {\n  messageId: string | undefined;\n  ownReactions: ReactionResponse<DefaultStreamChatGenerics>[];\n};\n\nexport type MessageReactionsContext = {\n  messageId: string | undefined;\n  /** @deprecated use `messageReactionGroups` */\n  messageReactionCounts: { [key in MessageReactionType]?: number };\n  /** @deprecated you can fetch the reactions using [`chatService.chatClient.queryReactions()`](https://getstream.io/chat/docs/javascript/send_reaction/?language=javascript&q=queryReactions#query-reactions) */\n  latestReactions: ReactionResponse<DefaultStreamChatGenerics>[];\n  ownReactions: ReactionResponse<DefaultStreamChatGenerics>[];\n  messageReactionGroups: {\n    [key in MessageReactionType]: ReactionGroupResponse;\n  };\n};\n\nexport type ModalContext = {\n  isOpen: boolean;\n  isOpenChangeHandler: (isOpen: boolean) => void;\n  content: TemplateRef<void>;\n};\n\nexport type NotificationContext = {\n  type: NotificationType | undefined;\n  content: TemplateRef<void> | undefined;\n};\n\nexport type ThreadHeaderContext = {\n  parentMessage: StreamMessage | undefined;\n  closeThreadHandler: () => void;\n};\n\nexport type MessageReactionType = string;\n\nexport type AttachmentConfigration = {\n  url: string;\n  height: string;\n  width: string;\n};\n\nexport type ImageAttachmentConfiguration = AttachmentConfigration & {\n  originalHeight: number;\n  originalWidth: number;\n};\n\nexport type VideoAttachmentConfiguration = ImageAttachmentConfiguration & {\n  thumbUrl?: string;\n};\n\nexport type DeliveredStatusContext = {\n  message: StreamMessage;\n};\n\nexport type SendingStatusContext = {\n  message: StreamMessage;\n};\n\nexport type CustomMetadataContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  message: StreamMessage<T>;\n};\n\nexport type ReadStatusContext = {\n  message: StreamMessage;\n  readByText: string;\n};\n\nexport type ChannelHeaderInfoContext<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = { channel: Channel<T> };\n\nexport type CustomAttachmentUploadContext = {\n  isMultipleFileUploadEnabled: boolean | undefined;\n  attachmentService: AttachmentService;\n};\n\nexport type AttachmentContext = {\n  attachment: Attachment<DefaultStreamChatGenerics>;\n};\n\nexport type SystemMessageContext = MessageContext & {\n  parsedDate: string | undefined;\n};\n\nexport type DateSeparatorContext = {\n  date: Date;\n  parsedDate: string;\n};\n\nexport type UnreadMessagesIndicatorContext = {\n  unreadCount: number;\n};\n\nexport type UnreadMessagesNotificationContext =\n  UnreadMessagesIndicatorContext & {\n    onJump: () => void;\n    onDismiss: () => void;\n  };\n\nexport type ChannelQueryState = {\n  state: 'in-progress' | 'success' | 'error';\n  // No type def from stream-chat\n  error?: unknown;\n};\n\nexport type MessageInput<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  text: string;\n  attachments: Attachment<T>[];\n  mentionedUsers: UserResponse<T>[];\n  parentId: string | undefined;\n  quotedMessageId: string | undefined;\n  customData: undefined | Partial<T['messageType']>;\n};\n\nexport type OffsetNextPageConfiguration = {\n  type: 'offset';\n  offset: number;\n};\n\nexport type FiltertNextPageConfiguration<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  type: 'filter';\n  paginationFilter: ChannelFilters<T>;\n};\n\nexport type NextPageConfiguration =\n  | OffsetNextPageConfiguration\n  | FiltertNextPageConfiguration\n  | undefined;\n\nexport type MessageReactionClickDetails = {\n  messageId: string;\n  reactionType: string;\n};\n\nexport type MessageActionsClickDetails<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = MessageActionsBoxContext<T> & { customActions: CustomMessageActionItem[] };\n\nexport type GroupStyleOptions = {\n  noGroupByUser?: boolean;\n  lastReadMessageId?: string;\n  noGroupByReadState?: boolean;\n};\n\nexport type ChannelQueryType = 'first-page' | 'next-page' | 'recover-state';\n\nexport type ChannelQueryResult<\n  T extends DefaultStreamChatGenerics = DefaultStreamChatGenerics\n> = {\n  channels: Channel<T>[];\n  hasMorePage: boolean;\n};\n\nexport type VirtualizedListScrollPosition = 'top' | 'bottom' | 'middle';\n\nexport type VirtualizedListQueryState = {\n  state: 'loading-top' | 'loading-bottom' | 'success' | 'error';\n  error?: unknown;\n};\n\nexport type VirtualizedListQueryDirection = 'top' | 'bottom';\n\nexport type VirtualizedListVerticalItemPosition = 'top' | 'bottom' | 'middle';\n\nexport type AudioRecording = MediaRecording & { waveform_data: number[] };\n\nexport type MediaRecording = {\n  recording: File;\n  duration: number;\n  mime_type: string;\n  asset_url: string | ArrayBuffer | undefined;\n};\n"]}
|
|
@@ -0,0 +1,119 @@
|
|
|
1
|
+
import { Injectable, NgModule } from '@angular/core';
|
|
2
|
+
import { BehaviorSubject } from 'rxjs';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../chat-client.service";
|
|
5
|
+
const MAX_FREQUENCY_AMPLITUDE = 255;
|
|
6
|
+
const rootMeanSquare = (values) => Math.sqrt(values.reduce((acc, val) => acc + Math.pow(val, 2), 0) / values.length);
|
|
7
|
+
export const DEFAULT_AMPLITUDE_RECORDER_CONFIG = {
|
|
8
|
+
analyserConfig: {
|
|
9
|
+
fftSize: 32,
|
|
10
|
+
maxDecibels: 0,
|
|
11
|
+
minDecibels: -100,
|
|
12
|
+
},
|
|
13
|
+
sampleCount: 100,
|
|
14
|
+
samplingFrequencyMs: 60,
|
|
15
|
+
};
|
|
16
|
+
/**
|
|
17
|
+
* The `AmplitudeRecorderService` is a utility service used to create amplitude values for voice recordings, making it possible to display a wave bar
|
|
18
|
+
*/
|
|
19
|
+
export class AmplitudeRecorderService {
|
|
20
|
+
constructor(chatService) {
|
|
21
|
+
this.chatService = chatService;
|
|
22
|
+
this.config = DEFAULT_AMPLITUDE_RECORDER_CONFIG;
|
|
23
|
+
this.amplitudesSubject = new BehaviorSubject([]);
|
|
24
|
+
this.errorSubject = new BehaviorSubject(undefined);
|
|
25
|
+
/**
|
|
26
|
+
* Start amplitude recording for the given media stream
|
|
27
|
+
* @param stream
|
|
28
|
+
*/
|
|
29
|
+
this.start = (stream) => {
|
|
30
|
+
this.stop();
|
|
31
|
+
this.stream = stream;
|
|
32
|
+
this.init();
|
|
33
|
+
this.resume();
|
|
34
|
+
};
|
|
35
|
+
this.amplitudes$ = this.amplitudesSubject.asObservable();
|
|
36
|
+
this.error$ = this.errorSubject.asObservable();
|
|
37
|
+
}
|
|
38
|
+
/**
|
|
39
|
+
* The recorded amplitudes
|
|
40
|
+
*/
|
|
41
|
+
get amplitudes() {
|
|
42
|
+
return this.amplitudesSubject.value;
|
|
43
|
+
}
|
|
44
|
+
/**
|
|
45
|
+
* Temporarily pause amplitude recording, recording can be resumed with `resume`
|
|
46
|
+
*/
|
|
47
|
+
pause() {
|
|
48
|
+
clearInterval(this.amplitudeSamplingInterval);
|
|
49
|
+
this.amplitudeSamplingInterval = undefined;
|
|
50
|
+
}
|
|
51
|
+
/**
|
|
52
|
+
* Resume amplited recording after it was pasued
|
|
53
|
+
*/
|
|
54
|
+
resume() {
|
|
55
|
+
this.amplitudeSamplingInterval = setInterval(() => {
|
|
56
|
+
if (!this.analyserNode) {
|
|
57
|
+
return;
|
|
58
|
+
}
|
|
59
|
+
const frequencyBins = new Uint8Array(this.analyserNode.frequencyBinCount);
|
|
60
|
+
try {
|
|
61
|
+
this.analyserNode.getByteFrequencyData(frequencyBins);
|
|
62
|
+
}
|
|
63
|
+
catch (e) {
|
|
64
|
+
this.logError(e);
|
|
65
|
+
this.errorSubject.next(e);
|
|
66
|
+
return;
|
|
67
|
+
}
|
|
68
|
+
const normalizedSignalStrength = rootMeanSquare(frequencyBins) / MAX_FREQUENCY_AMPLITUDE;
|
|
69
|
+
this.amplitudesSubject.next([
|
|
70
|
+
...this.amplitudesSubject.value,
|
|
71
|
+
normalizedSignalStrength,
|
|
72
|
+
]);
|
|
73
|
+
}, this.config.samplingFrequencyMs);
|
|
74
|
+
}
|
|
75
|
+
/**
|
|
76
|
+
* Stop the amplitude recording and frees up used resources
|
|
77
|
+
*/
|
|
78
|
+
stop() {
|
|
79
|
+
if (!this.stream) {
|
|
80
|
+
return;
|
|
81
|
+
}
|
|
82
|
+
this.stream = undefined;
|
|
83
|
+
clearInterval(this.amplitudeSamplingInterval);
|
|
84
|
+
this.amplitudeSamplingInterval = undefined;
|
|
85
|
+
this.amplitudesSubject.next([]);
|
|
86
|
+
this.errorSubject.next(undefined);
|
|
87
|
+
this.microphone?.disconnect();
|
|
88
|
+
this.analyserNode?.disconnect();
|
|
89
|
+
if (this.audioContext?.state !== 'closed') {
|
|
90
|
+
void this.audioContext?.close();
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
init() {
|
|
94
|
+
if (!this.stream) {
|
|
95
|
+
return;
|
|
96
|
+
}
|
|
97
|
+
this.audioContext = new AudioContext();
|
|
98
|
+
this.analyserNode = this.audioContext.createAnalyser();
|
|
99
|
+
const { analyserConfig } = this.config;
|
|
100
|
+
this.analyserNode.fftSize = analyserConfig.fftSize;
|
|
101
|
+
this.analyserNode.maxDecibels = analyserConfig.maxDecibels;
|
|
102
|
+
this.analyserNode.minDecibels = analyserConfig.minDecibels;
|
|
103
|
+
this.microphone = this.audioContext.createMediaStreamSource(this.stream);
|
|
104
|
+
this.microphone.connect(this.analyserNode);
|
|
105
|
+
}
|
|
106
|
+
logError(error) {
|
|
107
|
+
this.chatService.chatClient?.logger('error', error.message, {
|
|
108
|
+
error: error,
|
|
109
|
+
tag: ['AmplitudeRecorderService'],
|
|
110
|
+
});
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
AmplitudeRecorderService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AmplitudeRecorderService, deps: [{ token: i1.ChatClientService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
114
|
+
AmplitudeRecorderService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AmplitudeRecorderService, providedIn: NgModule });
|
|
115
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AmplitudeRecorderService, decorators: [{
|
|
116
|
+
type: Injectable,
|
|
117
|
+
args: [{ providedIn: NgModule }]
|
|
118
|
+
}], ctorParameters: function () { return [{ type: i1.ChatClientService }]; } });
|
|
119
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"amplitude-recorder.service.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/voice-recorder/amplitude-recorder.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgModule } from '@angular/core';\nimport { BehaviorSubject, Observable } from 'rxjs';\nimport { ChatClientService } from '../chat-client.service';\n\nconst MAX_FREQUENCY_AMPLITUDE = 255 as const;\n\nconst rootMeanSquare = (values: Uint8Array) =>\n  Math.sqrt(\n    values.reduce((acc, val) => acc + Math.pow(val, 2), 0) / values.length\n  );\n\n/**\n * fftSize\n * An unsigned integer, representing the window size of the FFT, given in number of samples.\n * A higher value will result in more details in the frequency domain but fewer details\n * in the amplitude domain.\n *\n * Must be a power of 2 between 2^5 and 2^15, so one of: 32, 64, 128, 256, 512, 1024, 2048, 4096, 8192, 16384, and 32768.\n * Defaults to 32.\n *\n * maxDecibels\n * A double, representing the maximum decibel value for scaling the FFT analysis data,\n * where 0 dB is the loudest possible sound, -10 dB is a 10th of that, etc.\n * The default value is -30 dB.\n *\n * minDecibels\n * A double, representing the minimum decibel value for scaling the FFT analysis data,\n * where 0 dB is the loudest possible sound, -10 dB is a 10th of that, etc.\n * The default value is -100 dB.\n */\nexport type AmplitudeAnalyserConfig = Pick<\n  AnalyserNode,\n  'fftSize' | 'maxDecibels' | 'minDecibels'\n>;\nexport type AmplitudeRecorderConfig = {\n  analyserConfig: AmplitudeAnalyserConfig;\n  sampleCount: number;\n  samplingFrequencyMs: number;\n};\n\nexport const DEFAULT_AMPLITUDE_RECORDER_CONFIG: AmplitudeRecorderConfig = {\n  analyserConfig: {\n    fftSize: 32,\n    maxDecibels: 0,\n    minDecibels: -100,\n  } as AmplitudeAnalyserConfig,\n  sampleCount: 100,\n  samplingFrequencyMs: 60,\n};\n\n/**\n * The `AmplitudeRecorderService` is a utility service used to create amplitude values for voice recordings, making it possible to display a wave bar\n */\n@Injectable({ providedIn: NgModule })\nexport class AmplitudeRecorderService {\n  config = DEFAULT_AMPLITUDE_RECORDER_CONFIG;\n  amplitudes$: Observable<number[]>;\n  error$: Observable<Error | undefined>;\n\n  private amplitudesSubject = new BehaviorSubject<number[]>([]);\n  private errorSubject = new BehaviorSubject<Error | undefined>(undefined);\n  private audioContext: AudioContext | undefined;\n  private analyserNode: AnalyserNode | undefined;\n  private microphone: MediaStreamAudioSourceNode | undefined;\n  private stream: MediaStream | undefined;\n  private amplitudeSamplingInterval: ReturnType<typeof setInterval> | undefined;\n\n  constructor(private chatService: ChatClientService) {\n    this.amplitudes$ = this.amplitudesSubject.asObservable();\n    this.error$ = this.errorSubject.asObservable();\n  }\n\n  /**\n   * The recorded amplitudes\n   */\n  get amplitudes() {\n    return this.amplitudesSubject.value;\n  }\n\n  /**\n   * Start amplitude recording for the given media stream\n   * @param stream\n   */\n  start = (stream: MediaStream) => {\n    this.stop();\n\n    this.stream = stream;\n    this.init();\n\n    this.resume();\n  };\n\n  /**\n   * Temporarily pause amplitude recording, recording can be resumed with `resume`\n   */\n  pause() {\n    clearInterval(this.amplitudeSamplingInterval);\n    this.amplitudeSamplingInterval = undefined;\n  }\n\n  /**\n   * Resume amplited recording after it was pasued\n   */\n  resume() {\n    this.amplitudeSamplingInterval = setInterval(() => {\n      if (!this.analyserNode) {\n        return;\n      }\n      const frequencyBins = new Uint8Array(this.analyserNode.frequencyBinCount);\n      try {\n        this.analyserNode.getByteFrequencyData(frequencyBins);\n      } catch (e) {\n        this.logError(e as Error);\n        this.errorSubject.next(e as Error);\n        return;\n      }\n      const normalizedSignalStrength =\n        rootMeanSquare(frequencyBins) / MAX_FREQUENCY_AMPLITUDE;\n      this.amplitudesSubject.next([\n        ...this.amplitudesSubject.value,\n        normalizedSignalStrength,\n      ]);\n    }, this.config.samplingFrequencyMs);\n  }\n\n  /**\n   * Stop the amplitude recording and frees up used resources\n   */\n  stop() {\n    if (!this.stream) {\n      return;\n    }\n    this.stream = undefined;\n    clearInterval(this.amplitudeSamplingInterval);\n    this.amplitudeSamplingInterval = undefined;\n    this.amplitudesSubject.next([]);\n    this.errorSubject.next(undefined);\n    this.microphone?.disconnect();\n    this.analyserNode?.disconnect();\n    if (this.audioContext?.state !== 'closed') {\n      void this.audioContext?.close();\n    }\n  }\n\n  private init() {\n    if (!this.stream) {\n      return;\n    }\n\n    this.audioContext = new AudioContext();\n    this.analyserNode = this.audioContext.createAnalyser();\n    const { analyserConfig } = this.config;\n    this.analyserNode.fftSize = analyserConfig.fftSize;\n    this.analyserNode.maxDecibels = analyserConfig.maxDecibels;\n    this.analyserNode.minDecibels = analyserConfig.minDecibels;\n\n    this.microphone = this.audioContext.createMediaStreamSource(this.stream);\n    this.microphone.connect(this.analyserNode);\n  }\n\n  private logError(error: Error) {\n    this.chatService.chatClient?.logger('error', error.message, {\n      error: error,\n      tag: ['AmplitudeRecorderService'],\n    });\n  }\n}\n"]}
|
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { isSafari } from '../is-safari';
|
|
3
|
+
import { MultimediaRecorder } from './media-recorder';
|
|
4
|
+
import { resampleWaveForm } from '../wave-form-sampler';
|
|
5
|
+
import { NgModel } from '@angular/forms';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "../notification.service";
|
|
8
|
+
import * as i2 from "../chat-client.service";
|
|
9
|
+
import * as i3 from "./transcoder.service";
|
|
10
|
+
import * as i4 from "./amplitude-recorder.service";
|
|
11
|
+
/**
|
|
12
|
+
* The `AudioRecorderService` can record an audio file, the SDK uses this to record a voice message
|
|
13
|
+
*/
|
|
14
|
+
export class AudioRecorderService extends MultimediaRecorder {
|
|
15
|
+
constructor(notificationService, chatService, transcoder, amplitudeRecorder) {
|
|
16
|
+
super(notificationService, chatService, transcoder);
|
|
17
|
+
this.amplitudeRecorder = amplitudeRecorder;
|
|
18
|
+
/**
|
|
19
|
+
* Due to browser restrictions the following config is used:
|
|
20
|
+
* - In Safari we record in audio/mp4
|
|
21
|
+
* - For all other browsers we use audio/webm (which is then transcoded to wav)
|
|
22
|
+
*/
|
|
23
|
+
this.config = {
|
|
24
|
+
mimeType: isSafari ? 'audio/mp4;codecs=mp4a.40.2' : 'audio/webm',
|
|
25
|
+
};
|
|
26
|
+
}
|
|
27
|
+
enrichWithExtraData() {
|
|
28
|
+
const waveformData = resampleWaveForm(this.amplitudeRecorder.amplitudes, this.amplitudeRecorder.config.sampleCount);
|
|
29
|
+
return { waveform_data: waveformData };
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Start audio recording
|
|
33
|
+
*/
|
|
34
|
+
async start() {
|
|
35
|
+
const result = await super.start();
|
|
36
|
+
if (this.mediaRecorder?.stream) {
|
|
37
|
+
this.amplitudeRecorder.start(this.mediaRecorder?.stream);
|
|
38
|
+
}
|
|
39
|
+
return result;
|
|
40
|
+
}
|
|
41
|
+
/**
|
|
42
|
+
* Pause audio recording, it can be restarted using `resume`
|
|
43
|
+
*/
|
|
44
|
+
pause() {
|
|
45
|
+
const result = super.pause();
|
|
46
|
+
this.amplitudeRecorder.pause();
|
|
47
|
+
return result;
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Resume a previously paused recording
|
|
51
|
+
*/
|
|
52
|
+
resume() {
|
|
53
|
+
const result = super.resume();
|
|
54
|
+
this.amplitudeRecorder.resume();
|
|
55
|
+
return result;
|
|
56
|
+
}
|
|
57
|
+
/**
|
|
58
|
+
* Stop the recording and free up used resources
|
|
59
|
+
* @param options
|
|
60
|
+
* @param options.cancel if this is `true` no recording will be created, but resources will be freed
|
|
61
|
+
* @returns the recording
|
|
62
|
+
*/
|
|
63
|
+
async stop(options) {
|
|
64
|
+
try {
|
|
65
|
+
const result = await super.stop(options);
|
|
66
|
+
return result;
|
|
67
|
+
}
|
|
68
|
+
finally {
|
|
69
|
+
this.amplitudeRecorder.stop();
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
}
|
|
73
|
+
AudioRecorderService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AudioRecorderService, deps: [{ token: i1.NotificationService }, { token: i2.ChatClientService }, { token: i3.TranscoderService }, { token: i4.AmplitudeRecorderService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
74
|
+
AudioRecorderService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AudioRecorderService, providedIn: NgModel });
|
|
75
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: AudioRecorderService, decorators: [{
|
|
76
|
+
type: Injectable,
|
|
77
|
+
args: [{ providedIn: NgModel }]
|
|
78
|
+
}], ctorParameters: function () { return [{ type: i1.NotificationService }, { type: i2.ChatClientService }, { type: i3.TranscoderService }, { type: i4.AmplitudeRecorderService }]; } });
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXVkaW8tcmVjb3JkZXIuc2VydmljZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi92b2ljZS1yZWNvcmRlci9hdWRpby1yZWNvcmRlci5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFFM0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGNBQWMsQ0FBQztBQUN4QyxPQUFPLEVBQXVCLGtCQUFrQixFQUFFLE1BQU0sa0JBQWtCLENBQUM7QUFJM0UsT0FBTyxFQUFFLGdCQUFnQixFQUFFLE1BQU0sc0JBQXNCLENBQUM7QUFFeEQsT0FBTyxFQUFFLE9BQU8sRUFBRSxNQUFNLGdCQUFnQixDQUFDOzs7Ozs7QUFFekM7O0dBRUc7QUFFSCxNQUFNLE9BQU8sb0JBQXFCLFNBQVEsa0JBRXpDO0lBVUMsWUFDRSxtQkFBd0MsRUFDeEMsV0FBOEIsRUFDOUIsVUFBNkIsRUFDckIsaUJBQTJDO1FBRW5ELEtBQUssQ0FBQyxtQkFBbUIsRUFBRSxXQUFXLEVBQUUsVUFBVSxDQUFDLENBQUM7UUFGNUMsc0JBQWlCLEdBQWpCLGlCQUFpQixDQUEwQjtRQWJyRDs7OztXQUlHO1FBQ0gsV0FBTSxHQUF3QjtZQUM1QixRQUFRLEVBQUUsUUFBUSxDQUFDLENBQUMsQ0FBQyw0QkFBNEIsQ0FBQyxDQUFDLENBQUMsWUFBWTtTQUNqRSxDQUFDO0lBU0YsQ0FBQztJQUVTLG1CQUFtQjtRQUMzQixNQUFNLFlBQVksR0FBRyxnQkFBZ0IsQ0FDbkMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLFVBQVUsRUFDakMsSUFBSSxDQUFDLGlCQUFpQixDQUFDLE1BQU0sQ0FBQyxXQUFXLENBQzFDLENBQUM7UUFFRixPQUFPLEVBQUUsYUFBYSxFQUFFLFlBQVksRUFBRSxDQUFDO0lBQ3pDLENBQUM7SUFFRDs7T0FFRztJQUNILEtBQUssQ0FBQyxLQUFLO1FBQ1QsTUFBTSxNQUFNLEdBQUcsTUFBTSxLQUFLLENBQUMsS0FBSyxFQUFFLENBQUM7UUFFbkMsSUFBSSxJQUFJLENBQUMsYUFBYSxFQUFFLE1BQU0sRUFBRTtZQUM5QixJQUFJLENBQUMsaUJBQWlCLENBQUMsS0FBSyxDQUFDLElBQUksQ0FBQyxhQUFhLEVBQUUsTUFBTSxDQUFDLENBQUM7U0FDMUQ7UUFFRCxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxLQUFLO1FBQ0gsTUFBTSxNQUFNLEdBQUcsS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO1FBRTdCLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxLQUFLLEVBQUUsQ0FBQztRQUUvQixPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxNQUFNO1FBQ0osTUFBTSxNQUFNLEdBQUcsS0FBSyxDQUFDLE1BQU0sRUFBRSxDQUFDO1FBRTlCLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxNQUFNLEVBQUUsQ0FBQztRQUVoQyxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDO0lBRUQ7Ozs7O09BS0c7SUFDSCxLQUFLLENBQUMsSUFBSSxDQUFDLE9BQTZCO1FBQ3RDLElBQUk7WUFDRixNQUFNLE1BQU0sR0FBRyxNQUFNLEtBQUssQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7WUFFekMsT0FBTyxNQUFNLENBQUM7U0FDZjtnQkFBUztZQUNSLElBQUksQ0FBQyxpQkFBaUIsQ0FBQyxJQUFJLEVBQUUsQ0FBQztTQUMvQjtJQUNILENBQUM7O2lIQS9FVSxvQkFBb0I7cUhBQXBCLG9CQUFvQixjQURQLE9BQU87MkZBQ3BCLG9CQUFvQjtrQkFEaEMsVUFBVTttQkFBQyxFQUFFLFVBQVUsRUFBRSxPQUFPLEVBQUUiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBBbXBsaXR1ZGVSZWNvcmRlclNlcnZpY2UgfSBmcm9tICcuL2FtcGxpdHVkZS1yZWNvcmRlci5zZXJ2aWNlJztcbmltcG9ydCB7IGlzU2FmYXJpIH0gZnJvbSAnLi4vaXMtc2FmYXJpJztcbmltcG9ydCB7IE1lZGlhUmVjb3JkZXJDb25maWcsIE11bHRpbWVkaWFSZWNvcmRlciB9IGZyb20gJy4vbWVkaWEtcmVjb3JkZXInO1xuaW1wb3J0IHsgTm90aWZpY2F0aW9uU2VydmljZSB9IGZyb20gJy4uL25vdGlmaWNhdGlvbi5zZXJ2aWNlJztcbmltcG9ydCB7IENoYXRDbGllbnRTZXJ2aWNlIH0gZnJvbSAnLi4vY2hhdC1jbGllbnQuc2VydmljZSc7XG5pbXBvcnQgeyBUcmFuc2NvZGVyU2VydmljZSB9IGZyb20gJy4vdHJhbnNjb2Rlci5zZXJ2aWNlJztcbmltcG9ydCB7IHJlc2FtcGxlV2F2ZUZvcm0gfSBmcm9tICcuLi93YXZlLWZvcm0tc2FtcGxlcic7XG5pbXBvcnQgeyBBdWRpb1JlY29yZGluZywgTWVkaWFSZWNvcmRpbmcgfSBmcm9tICcuLi90eXBlcyc7XG5pbXBvcnQgeyBOZ01vZGVsIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuXG4vKipcbiAqIFRoZSBgQXVkaW9SZWNvcmRlclNlcnZpY2VgIGNhbiByZWNvcmQgYW4gYXVkaW8gZmlsZSwgdGhlIFNESyB1c2VzIHRoaXMgdG8gcmVjb3JkIGEgdm9pY2UgbWVzc2FnZVxuICovXG5ASW5qZWN0YWJsZSh7IHByb3ZpZGVkSW46IE5nTW9kZWwgfSlcbmV4cG9ydCBjbGFzcyBBdWRpb1JlY29yZGVyU2VydmljZSBleHRlbmRzIE11bHRpbWVkaWFSZWNvcmRlcjxcbiAgT21pdDxBdWRpb1JlY29yZGluZywga2V5b2YgTWVkaWFSZWNvcmRpbmc+XG4+IHtcbiAgLyoqXG4gICAqIER1ZSB0byBicm93c2VyIHJlc3RyaWN0aW9ucyB0aGUgZm9sbG93aW5nIGNvbmZpZyBpcyB1c2VkOlxuICAgKiAtIEluIFNhZmFyaSB3ZSByZWNvcmQgaW4gYXVkaW8vbXA0XG4gICAqIC0gRm9yIGFsbCBvdGhlciBicm93c2VycyB3ZSB1c2UgYXVkaW8vd2VibSAod2hpY2ggaXMgdGhlbiB0cmFuc2NvZGVkIHRvIHdhdilcbiAgICovXG4gIGNvbmZpZzogTWVkaWFSZWNvcmRlckNvbmZpZyA9IHtcbiAgICBtaW1lVHlwZTogaXNTYWZhcmkgPyAnYXVkaW8vbXA0O2NvZGVjcz1tcDRhLjQwLjInIDogJ2F1ZGlvL3dlYm0nLFxuICB9O1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIG5vdGlmaWNhdGlvblNlcnZpY2U6IE5vdGlmaWNhdGlvblNlcnZpY2UsXG4gICAgY2hhdFNlcnZpY2U6IENoYXRDbGllbnRTZXJ2aWNlLFxuICAgIHRyYW5zY29kZXI6IFRyYW5zY29kZXJTZXJ2aWNlLFxuICAgIHByaXZhdGUgYW1wbGl0dWRlUmVjb3JkZXI6IEFtcGxpdHVkZVJlY29yZGVyU2VydmljZVxuICApIHtcbiAgICBzdXBlcihub3RpZmljYXRpb25TZXJ2aWNlLCBjaGF0U2VydmljZSwgdHJhbnNjb2Rlcik7XG4gIH1cblxuICBwcm90ZWN0ZWQgZW5yaWNoV2l0aEV4dHJhRGF0YSgpIHtcbiAgICBjb25zdCB3YXZlZm9ybURhdGEgPSByZXNhbXBsZVdhdmVGb3JtKFxuICAgICAgdGhpcy5hbXBsaXR1ZGVSZWNvcmRlci5hbXBsaXR1ZGVzLFxuICAgICAgdGhpcy5hbXBsaXR1ZGVSZWNvcmRlci5jb25maWcuc2FtcGxlQ291bnRcbiAgICApO1xuXG4gICAgcmV0dXJuIHsgd2F2ZWZvcm1fZGF0YTogd2F2ZWZvcm1EYXRhIH07XG4gIH1cblxuICAvKipcbiAgICogU3RhcnQgYXVkaW8gcmVjb3JkaW5nXG4gICAqL1xuICBhc3luYyBzdGFydCgpIHtcbiAgICBjb25zdCByZXN1bHQgPSBhd2FpdCBzdXBlci5zdGFydCgpO1xuXG4gICAgaWYgKHRoaXMubWVkaWFSZWNvcmRlcj8uc3RyZWFtKSB7XG4gICAgICB0aGlzLmFtcGxpdHVkZVJlY29yZGVyLnN0YXJ0KHRoaXMubWVkaWFSZWNvcmRlcj8uc3RyZWFtKTtcbiAgICB9XG5cbiAgICByZXR1cm4gcmVzdWx0O1xuICB9XG5cbiAgLyoqXG4gICAqIFBhdXNlIGF1ZGlvIHJlY29yZGluZywgaXQgY2FuIGJlIHJlc3RhcnRlZCB1c2luZyBgcmVzdW1lYFxuICAgKi9cbiAgcGF1c2UoKSB7XG4gICAgY29uc3QgcmVzdWx0ID0gc3VwZXIucGF1c2UoKTtcblxuICAgIHRoaXMuYW1wbGl0dWRlUmVjb3JkZXIucGF1c2UoKTtcblxuICAgIHJldHVybiByZXN1bHQ7XG4gIH1cblxuICAvKipcbiAgICogUmVzdW1lIGEgcHJldmlvdXNseSBwYXVzZWQgcmVjb3JkaW5nXG4gICAqL1xuICByZXN1bWUoKSB7XG4gICAgY29uc3QgcmVzdWx0ID0gc3VwZXIucmVzdW1lKCk7XG5cbiAgICB0aGlzLmFtcGxpdHVkZVJlY29yZGVyLnJlc3VtZSgpO1xuXG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gIC8qKlxuICAgKiBTdG9wIHRoZSByZWNvcmRpbmcgYW5kIGZyZWUgdXAgdXNlZCByZXNvdXJjZXNcbiAgICogQHBhcmFtIG9wdGlvbnNcbiAgICogQHBhcmFtIG9wdGlvbnMuY2FuY2VsIGlmIHRoaXMgaXMgYHRydWVgIG5vIHJlY29yZGluZyB3aWxsIGJlIGNyZWF0ZWQsIGJ1dCByZXNvdXJjZXMgd2lsbCBiZSBmcmVlZFxuICAgKiBAcmV0dXJucyB0aGUgcmVjb3JkaW5nXG4gICAqL1xuICBhc3luYyBzdG9wKG9wdGlvbnM/OiB7IGNhbmNlbDogYm9vbGVhbiB9KSB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHN1cGVyLnN0b3Aob3B0aW9ucyk7XG5cbiAgICAgIHJldHVybiByZXN1bHQ7XG4gICAgfSBmaW5hbGx5IHtcbiAgICAgIHRoaXMuYW1wbGl0dWRlUmVjb3JkZXIuc3RvcCgpO1xuICAgIH1cbiAgfVxufVxuIl19
|