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,61 @@
|
|
|
1
|
+
const ENCODING_BIT_RATE = 128; // kbps;
|
|
2
|
+
const COUNT_SAMPLES_PER_ENCODED_BLOCK = 1152;
|
|
3
|
+
const SAMPLE_RATE = 16000;
|
|
4
|
+
const readFileAsArrayBuffer = (blob) => new Promise((resolve, reject) => {
|
|
5
|
+
const blobReader = new FileReader();
|
|
6
|
+
blobReader.onload = () => {
|
|
7
|
+
resolve(blobReader.result);
|
|
8
|
+
};
|
|
9
|
+
blobReader.onerror = () => {
|
|
10
|
+
reject(blobReader.error);
|
|
11
|
+
};
|
|
12
|
+
blobReader.readAsArrayBuffer(blob);
|
|
13
|
+
});
|
|
14
|
+
const toAudioBuffer = async (blob) => {
|
|
15
|
+
const audioCtx = new AudioContext();
|
|
16
|
+
const arrayBuffer = await readFileAsArrayBuffer(blob);
|
|
17
|
+
const decodedData = await audioCtx.decodeAudioData(arrayBuffer);
|
|
18
|
+
if (audioCtx.state !== 'closed')
|
|
19
|
+
await audioCtx.close();
|
|
20
|
+
return decodedData;
|
|
21
|
+
};
|
|
22
|
+
const renderAudio = async (audioBuffer, sampleRate) => {
|
|
23
|
+
const offlineAudioCtx = new OfflineAudioContext(audioBuffer.numberOfChannels, audioBuffer.duration * sampleRate, sampleRate);
|
|
24
|
+
const source = offlineAudioCtx.createBufferSource();
|
|
25
|
+
source.buffer = audioBuffer;
|
|
26
|
+
source.connect(offlineAudioCtx.destination);
|
|
27
|
+
source.start();
|
|
28
|
+
return await offlineAudioCtx.startRendering();
|
|
29
|
+
};
|
|
30
|
+
const float32ArrayToInt16Array = (float32Arr) => {
|
|
31
|
+
const int16Arr = new Int16Array(float32Arr.length);
|
|
32
|
+
for (let i = 0; i < float32Arr.length; i++) {
|
|
33
|
+
const float32Value = float32Arr[i];
|
|
34
|
+
// Clamp the float value between -1 and 1
|
|
35
|
+
const clampedValue = Math.max(-1, Math.min(1, float32Value));
|
|
36
|
+
// Convert the float value to a signed 16-bit integer
|
|
37
|
+
int16Arr[i] = Math.round(clampedValue * 32767);
|
|
38
|
+
}
|
|
39
|
+
return int16Arr;
|
|
40
|
+
};
|
|
41
|
+
const splitDataByChannel = (audioBuffer) => Array.from({ length: audioBuffer.numberOfChannels }, (_, i) => audioBuffer.getChannelData(i)).map(float32ArrayToInt16Array);
|
|
42
|
+
/* eslint-disable @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-argument */
|
|
43
|
+
export async function encodeWebmToMp3(blob, lameJs) {
|
|
44
|
+
const audioBuffer = await renderAudio(await toAudioBuffer(blob), SAMPLE_RATE);
|
|
45
|
+
const channelCount = audioBuffer.numberOfChannels;
|
|
46
|
+
const dataByChannel = splitDataByChannel(audioBuffer);
|
|
47
|
+
const mp3Encoder = new lameJs.Mp3Encoder(channelCount, SAMPLE_RATE, ENCODING_BIT_RATE);
|
|
48
|
+
const dataBuffer = [];
|
|
49
|
+
let remaining = dataByChannel[0].length;
|
|
50
|
+
for (let i = 0; remaining >= COUNT_SAMPLES_PER_ENCODED_BLOCK; i += COUNT_SAMPLES_PER_ENCODED_BLOCK) {
|
|
51
|
+
const [leftChannelBlock, rightChannelBlock] = dataByChannel.map((channel) => channel.subarray(i, i + COUNT_SAMPLES_PER_ENCODED_BLOCK));
|
|
52
|
+
dataBuffer.push(new Int8Array(mp3Encoder.encodeBuffer(leftChannelBlock, rightChannelBlock)));
|
|
53
|
+
remaining -= COUNT_SAMPLES_PER_ENCODED_BLOCK;
|
|
54
|
+
}
|
|
55
|
+
const lastBlock = mp3Encoder.flush();
|
|
56
|
+
if (lastBlock.length)
|
|
57
|
+
dataBuffer.push(new Int8Array(lastBlock));
|
|
58
|
+
return new Blob(dataBuffer, { type: 'audio/mp3;sbu_type=voice' });
|
|
59
|
+
}
|
|
60
|
+
/* eslint-enable @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-argument */
|
|
61
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mp3-transcoder.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/voice-recorder/mp3-transcoder.ts"],"names":[],"mappings":"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","sourcesContent":["const ENCODING_BIT_RATE = 128; // kbps;\nconst COUNT_SAMPLES_PER_ENCODED_BLOCK = 1152;\nconst SAMPLE_RATE = 16000;\n\nconst readFileAsArrayBuffer = (blob: Blob): Promise<ArrayBuffer> =>\n  new Promise((resolve, reject) => {\n    const blobReader = new FileReader();\n    blobReader.onload = () => {\n      resolve(blobReader.result as ArrayBuffer);\n    };\n\n    blobReader.onerror = () => {\n      reject(blobReader.error);\n    };\n\n    blobReader.readAsArrayBuffer(blob);\n  });\n\nconst toAudioBuffer = async (blob: Blob) => {\n  const audioCtx = new AudioContext();\n\n  const arrayBuffer = await readFileAsArrayBuffer(blob);\n  const decodedData = await audioCtx.decodeAudioData(arrayBuffer);\n  if (audioCtx.state !== 'closed') await audioCtx.close();\n  return decodedData;\n};\n\nconst renderAudio = async (audioBuffer: AudioBuffer, sampleRate: number) => {\n  const offlineAudioCtx = new OfflineAudioContext(\n    audioBuffer.numberOfChannels,\n    audioBuffer.duration * sampleRate,\n    sampleRate\n  );\n  const source = offlineAudioCtx.createBufferSource();\n  source.buffer = audioBuffer;\n  source.connect(offlineAudioCtx.destination);\n  source.start();\n\n  return await offlineAudioCtx.startRendering();\n};\n\nconst float32ArrayToInt16Array = (float32Arr: Float32Array) => {\n  const int16Arr = new Int16Array(float32Arr.length);\n  for (let i = 0; i < float32Arr.length; i++) {\n    const float32Value = float32Arr[i];\n    // Clamp the float value between -1 and 1\n    const clampedValue = Math.max(-1, Math.min(1, float32Value));\n    // Convert the float value to a signed 16-bit integer\n    int16Arr[i] = Math.round(clampedValue * 32767);\n  }\n  return int16Arr;\n};\n\nconst splitDataByChannel = (audioBuffer: AudioBuffer) =>\n  Array.from({ length: audioBuffer.numberOfChannels }, (_, i) =>\n    audioBuffer.getChannelData(i)\n  ).map(float32ArrayToInt16Array);\n\n/* eslint-disable @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-argument */\nexport async function encodeWebmToMp3(blob: Blob, lameJs: any) {\n  const audioBuffer = await renderAudio(await toAudioBuffer(blob), SAMPLE_RATE);\n  const channelCount = audioBuffer.numberOfChannels;\n  const dataByChannel = splitDataByChannel(audioBuffer);\n  const mp3Encoder = new lameJs.Mp3Encoder(\n    channelCount,\n    SAMPLE_RATE,\n    ENCODING_BIT_RATE\n  );\n\n  const dataBuffer: Int8Array[] = [];\n  let remaining = dataByChannel[0].length;\n  for (\n    let i = 0;\n    remaining >= COUNT_SAMPLES_PER_ENCODED_BLOCK;\n    i += COUNT_SAMPLES_PER_ENCODED_BLOCK\n  ) {\n    const [leftChannelBlock, rightChannelBlock] = dataByChannel.map((channel) =>\n      channel.subarray(i, i + COUNT_SAMPLES_PER_ENCODED_BLOCK)\n    );\n    dataBuffer.push(\n      new Int8Array(\n        mp3Encoder.encodeBuffer(leftChannelBlock, rightChannelBlock)\n      )\n    );\n    remaining -= COUNT_SAMPLES_PER_ENCODED_BLOCK;\n  }\n\n  const lastBlock = mp3Encoder.flush();\n  if (lastBlock.length) dataBuffer.push(new Int8Array(lastBlock));\n  return new Blob(dataBuffer, { type: 'audio/mp3;sbu_type=voice' });\n}\n/* eslint-enable @typescript-eslint/no-explicit-any, @typescript-eslint/no-unsafe-assignment, @typescript-eslint/no-unsafe-call, @typescript-eslint/no-unsafe-member-access, @typescript-eslint/no-unsafe-argument */\n"]}
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
import { Injectable, NgModule } from '@angular/core';
|
|
2
|
+
import { readBlobAsArrayBuffer } from '../file-utils';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
const WAV_HEADER_LENGTH_BYTES = 44;
|
|
5
|
+
const BYTES_PER_SAMPLE = 2;
|
|
6
|
+
const RIFF_FILE_MAX_BYTES = 4294967295;
|
|
7
|
+
const HEADER = {
|
|
8
|
+
AUDIO_FORMAT: { offset: 20, value: 1 },
|
|
9
|
+
BITS_PER_SAMPLE: { offset: 34, value: BYTES_PER_SAMPLE * 8 },
|
|
10
|
+
BLOCK_ALIGN: { offset: 32 },
|
|
11
|
+
BYTE_RATE: { offset: 28 },
|
|
12
|
+
CHANNEL_COUNT: { offset: 22 },
|
|
13
|
+
CHUNK_ID: { offset: 0, value: 0x52494646 },
|
|
14
|
+
CHUNK_SIZE: { offset: 4 },
|
|
15
|
+
FILE_FORMAT: { offset: 8, value: 0x57415645 },
|
|
16
|
+
SAMPLE_RATE: { offset: 24 },
|
|
17
|
+
SUBCHUNK1_ID: { offset: 12, value: 0x666d7420 },
|
|
18
|
+
SUBCHUNK1_SIZE: { offset: 16, value: 16 },
|
|
19
|
+
SUBCHUNK2_ID: { offset: 36, value: 0x64617461 },
|
|
20
|
+
SUBCHUNK2_SIZE: { offset: 40 }, // actual audio data size
|
|
21
|
+
};
|
|
22
|
+
/**
|
|
23
|
+
* The `TranscoderService` is used to transcibe audio recording to a format that's supported by all major browsers. The SDK uses this to create voice messages.
|
|
24
|
+
*
|
|
25
|
+
* If you want to use your own transcoder you can provide a `customTranscoder`.
|
|
26
|
+
*/
|
|
27
|
+
export class TranscoderService {
|
|
28
|
+
constructor() {
|
|
29
|
+
this.config = {
|
|
30
|
+
sampleRate: 16000,
|
|
31
|
+
};
|
|
32
|
+
this.splitDataByChannel = (audioBuffer) => Array.from({ length: audioBuffer.numberOfChannels }, (_, i) => audioBuffer.getChannelData(i));
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* The default transcoder will leave audio/mp4 files as is, and transcode webm files to wav. If you want to customize this, you can provide your own transcoder using the `customTranscoder` field
|
|
36
|
+
* @param blob
|
|
37
|
+
* @returns the transcoded file
|
|
38
|
+
*/
|
|
39
|
+
async transcode(blob) {
|
|
40
|
+
if (this.customTranscoder) {
|
|
41
|
+
return this.customTranscoder(blob);
|
|
42
|
+
}
|
|
43
|
+
if (blob.type.includes('audio/mp4')) {
|
|
44
|
+
return blob;
|
|
45
|
+
}
|
|
46
|
+
const audioBuffer = await this.renderAudio(await this.toAudioBuffer(blob), this.config.sampleRate);
|
|
47
|
+
const numberOfSamples = audioBuffer.duration * this.config.sampleRate;
|
|
48
|
+
const fileSizeBytes = numberOfSamples * audioBuffer.numberOfChannels * BYTES_PER_SAMPLE +
|
|
49
|
+
WAV_HEADER_LENGTH_BYTES;
|
|
50
|
+
const arrayBuffer = new ArrayBuffer(fileSizeBytes);
|
|
51
|
+
this.writeWavHeader({
|
|
52
|
+
arrayBuffer,
|
|
53
|
+
channelCount: audioBuffer.numberOfChannels,
|
|
54
|
+
sampleRate: this.config.sampleRate,
|
|
55
|
+
});
|
|
56
|
+
this.writeWavAudioData({
|
|
57
|
+
arrayBuffer,
|
|
58
|
+
dataByChannel: this.splitDataByChannel(audioBuffer),
|
|
59
|
+
});
|
|
60
|
+
return new Blob([arrayBuffer], { type: 'audio/wav' });
|
|
61
|
+
}
|
|
62
|
+
async renderAudio(audioBuffer, sampleRate) {
|
|
63
|
+
const offlineAudioCtx = new OfflineAudioContext(audioBuffer.numberOfChannels, audioBuffer.duration * sampleRate, sampleRate);
|
|
64
|
+
const source = offlineAudioCtx.createBufferSource();
|
|
65
|
+
source.buffer = audioBuffer;
|
|
66
|
+
source.connect(offlineAudioCtx.destination);
|
|
67
|
+
source.start();
|
|
68
|
+
return await offlineAudioCtx.startRendering();
|
|
69
|
+
}
|
|
70
|
+
async toAudioBuffer(blob) {
|
|
71
|
+
const audioCtx = new AudioContext();
|
|
72
|
+
const arrayBuffer = await readBlobAsArrayBuffer(blob);
|
|
73
|
+
const decodedData = await audioCtx.decodeAudioData(arrayBuffer);
|
|
74
|
+
if (audioCtx.state !== 'closed')
|
|
75
|
+
await audioCtx.close();
|
|
76
|
+
return decodedData;
|
|
77
|
+
}
|
|
78
|
+
writeWavAudioData({ arrayBuffer, dataByChannel, }) {
|
|
79
|
+
const dataView = new DataView(arrayBuffer);
|
|
80
|
+
const channelCount = dataByChannel.length;
|
|
81
|
+
dataByChannel.forEach((channelData, channelIndex) => {
|
|
82
|
+
let writeOffset = WAV_HEADER_LENGTH_BYTES + channelCount * channelIndex;
|
|
83
|
+
channelData.forEach((float32Value) => {
|
|
84
|
+
dataView.setInt16(writeOffset, float32Value < 0
|
|
85
|
+
? Math.max(-1, float32Value) * 32768
|
|
86
|
+
: Math.min(1, float32Value) * 32767, true);
|
|
87
|
+
writeOffset += channelCount * BYTES_PER_SAMPLE;
|
|
88
|
+
});
|
|
89
|
+
});
|
|
90
|
+
}
|
|
91
|
+
writeWavHeader({ arrayBuffer, channelCount, sampleRate, }) {
|
|
92
|
+
const byteRate = sampleRate * channelCount * BYTES_PER_SAMPLE; // bytes/sec
|
|
93
|
+
const blockAlign = channelCount * BYTES_PER_SAMPLE;
|
|
94
|
+
const dataView = new DataView(arrayBuffer);
|
|
95
|
+
/*
|
|
96
|
+
* The maximum size of a RIFF file is 4294967295 bytes and since the header takes up 44 bytes there are 4294967251 bytes left for the
|
|
97
|
+
* data chunk.
|
|
98
|
+
*/
|
|
99
|
+
const dataChunkSize = Math.min(dataView.byteLength - WAV_HEADER_LENGTH_BYTES, RIFF_FILE_MAX_BYTES - WAV_HEADER_LENGTH_BYTES);
|
|
100
|
+
dataView.setUint32(HEADER.CHUNK_ID.offset, HEADER.CHUNK_ID.value); // "RIFF"
|
|
101
|
+
dataView.setUint32(HEADER.CHUNK_SIZE.offset, arrayBuffer.byteLength - 8, true); // adjustment for the first two headers - chunk id + file size
|
|
102
|
+
dataView.setUint32(HEADER.FILE_FORMAT.offset, HEADER.FILE_FORMAT.value); // "WAVE"
|
|
103
|
+
dataView.setUint32(HEADER.SUBCHUNK1_ID.offset, HEADER.SUBCHUNK1_ID.value); // "fmt "
|
|
104
|
+
dataView.setUint32(HEADER.SUBCHUNK1_SIZE.offset, HEADER.SUBCHUNK1_SIZE.value, true);
|
|
105
|
+
dataView.setUint16(HEADER.AUDIO_FORMAT.offset, HEADER.AUDIO_FORMAT.value, true);
|
|
106
|
+
dataView.setUint16(HEADER.CHANNEL_COUNT.offset, channelCount, true);
|
|
107
|
+
dataView.setUint32(HEADER.SAMPLE_RATE.offset, sampleRate, true);
|
|
108
|
+
dataView.setUint32(HEADER.BYTE_RATE.offset, byteRate, true);
|
|
109
|
+
dataView.setUint16(HEADER.BLOCK_ALIGN.offset, blockAlign, true);
|
|
110
|
+
dataView.setUint16(HEADER.BITS_PER_SAMPLE.offset, HEADER.BITS_PER_SAMPLE.value, true);
|
|
111
|
+
dataView.setUint32(HEADER.SUBCHUNK2_ID.offset, HEADER.SUBCHUNK2_ID.value); // "data"
|
|
112
|
+
dataView.setUint32(HEADER.SUBCHUNK2_SIZE.offset, dataChunkSize, true);
|
|
113
|
+
}
|
|
114
|
+
}
|
|
115
|
+
TranscoderService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: TranscoderService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
116
|
+
TranscoderService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: TranscoderService, providedIn: NgModule });
|
|
117
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: TranscoderService, decorators: [{
|
|
118
|
+
type: Injectable,
|
|
119
|
+
args: [{ providedIn: NgModule }]
|
|
120
|
+
}], ctorParameters: function () { return []; } });
|
|
121
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"transcoder.service.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/voice-recorder/transcoder.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable, NgModule } from '@angular/core';\nimport { readBlobAsArrayBuffer } from '../file-utils';\n\nexport type TranscoderConfig = {\n  sampleRate: number;\n};\n\nexport type TranscodeParams = TranscoderConfig & {\n  blob: Blob;\n};\n\nconst WAV_HEADER_LENGTH_BYTES = 44 as const;\nconst BYTES_PER_SAMPLE = 2 as const;\nconst RIFF_FILE_MAX_BYTES = 4294967295 as const;\n\nconst HEADER = {\n  AUDIO_FORMAT: { offset: 20, value: 1 }, // PCM = 1\n  BITS_PER_SAMPLE: { offset: 34, value: BYTES_PER_SAMPLE * 8 }, // 16 bits encoding\n  BLOCK_ALIGN: { offset: 32 },\n  BYTE_RATE: { offset: 28 },\n  CHANNEL_COUNT: { offset: 22 }, // 1 - mono, 2 - stereo\n  CHUNK_ID: { offset: 0, value: 0x52494646 }, // hex representation of string \"RIFF\" (Resource Interchange File Format) - identifies the file structure that defines a class of more specific file formats, e.g. WAVE\n  CHUNK_SIZE: { offset: 4 },\n  FILE_FORMAT: { offset: 8, value: 0x57415645 }, // hex representation of string \"WAVE\"\n  SAMPLE_RATE: { offset: 24 },\n  SUBCHUNK1_ID: { offset: 12, value: 0x666d7420 }, // hex representation of string \"fmt \" - identifies the start of \"format\" section of the header\n  SUBCHUNK1_SIZE: { offset: 16, value: 16 }, // Subchunk1 Size without SUBCHUNK1_ID and SUBCHUNK1_SIZE fields\n  SUBCHUNK2_ID: { offset: 36, value: 0x64617461 }, // hex representation of string \"data\" - identifies the start of actual audio data section\n  SUBCHUNK2_SIZE: { offset: 40 }, // actual audio data size\n} as const;\n\ntype WriteWaveHeaderParams = {\n  arrayBuffer: ArrayBuffer;\n  // 1 - mono, 2 - stereo\n  channelCount: number;\n  // Number of samples per second, e.g. 44100Hz\n  sampleRate: number;\n};\n\ntype WriteAudioDataParams = {\n  arrayBuffer: ArrayBuffer;\n  dataByChannel: Float32Array[];\n};\n\n/**\n * The `TranscoderService` is used to transcibe audio recording to a format that's supported by all major browsers. The SDK uses this to create voice messages.\n *\n * If you want to use your own transcoder you can provide a `customTranscoder`.\n */\n@Injectable({ providedIn: NgModule })\nexport class TranscoderService {\n  config: TranscoderConfig = {\n    sampleRate: 16000,\n  };\n  customTranscoder?: (blob: Blob) => Blob | Promise<Blob>;\n  constructor() {}\n\n  /**\n   * The default transcoder will leave audio/mp4 files as is, and transcode webm files to wav. If you want to customize this, you can provide your own transcoder using the `customTranscoder` field\n   * @param blob\n   * @returns the transcoded file\n   */\n  async transcode(blob: Blob) {\n    if (this.customTranscoder) {\n      return this.customTranscoder(blob);\n    }\n    if (blob.type.includes('audio/mp4')) {\n      return blob;\n    }\n    const audioBuffer = await this.renderAudio(\n      await this.toAudioBuffer(blob),\n      this.config.sampleRate\n    );\n    const numberOfSamples = audioBuffer.duration * this.config.sampleRate;\n    const fileSizeBytes =\n      numberOfSamples * audioBuffer.numberOfChannels * BYTES_PER_SAMPLE +\n      WAV_HEADER_LENGTH_BYTES;\n\n    const arrayBuffer = new ArrayBuffer(fileSizeBytes);\n    this.writeWavHeader({\n      arrayBuffer,\n      channelCount: audioBuffer.numberOfChannels,\n      sampleRate: this.config.sampleRate,\n    });\n    this.writeWavAudioData({\n      arrayBuffer,\n      dataByChannel: this.splitDataByChannel(audioBuffer),\n    });\n    return new Blob([arrayBuffer], { type: 'audio/wav' });\n  }\n\n  protected async renderAudio(audioBuffer: AudioBuffer, sampleRate: number) {\n    const offlineAudioCtx = new OfflineAudioContext(\n      audioBuffer.numberOfChannels,\n      audioBuffer.duration * sampleRate,\n      sampleRate\n    );\n    const source = offlineAudioCtx.createBufferSource();\n    source.buffer = audioBuffer;\n    source.connect(offlineAudioCtx.destination);\n    source.start();\n\n    return await offlineAudioCtx.startRendering();\n  }\n\n  protected async toAudioBuffer(blob: Blob) {\n    const audioCtx = new AudioContext();\n\n    const arrayBuffer = await readBlobAsArrayBuffer(blob);\n    const decodedData = await audioCtx.decodeAudioData(arrayBuffer);\n    if (audioCtx.state !== 'closed') await audioCtx.close();\n    return decodedData;\n  }\n\n  protected writeWavAudioData({\n    arrayBuffer,\n    dataByChannel,\n  }: WriteAudioDataParams) {\n    const dataView = new DataView(arrayBuffer);\n    const channelCount = dataByChannel.length;\n\n    dataByChannel.forEach((channelData, channelIndex) => {\n      let writeOffset = WAV_HEADER_LENGTH_BYTES + channelCount * channelIndex;\n\n      channelData.forEach((float32Value) => {\n        dataView.setInt16(\n          writeOffset,\n          float32Value < 0\n            ? Math.max(-1, float32Value) * 32768\n            : Math.min(1, float32Value) * 32767,\n          true\n        );\n        writeOffset += channelCount * BYTES_PER_SAMPLE;\n      });\n    });\n  }\n\n  protected writeWavHeader({\n    arrayBuffer,\n    channelCount,\n    sampleRate,\n  }: WriteWaveHeaderParams) {\n    const byteRate = sampleRate * channelCount * BYTES_PER_SAMPLE; // bytes/sec\n    const blockAlign = channelCount * BYTES_PER_SAMPLE;\n\n    const dataView = new DataView(arrayBuffer);\n    /*\n     * The maximum size of a RIFF file is 4294967295 bytes and since the header takes up 44 bytes there are 4294967251 bytes left for the\n     * data chunk.\n     */\n    const dataChunkSize = Math.min(\n      dataView.byteLength - WAV_HEADER_LENGTH_BYTES,\n      RIFF_FILE_MAX_BYTES - WAV_HEADER_LENGTH_BYTES\n    );\n\n    dataView.setUint32(HEADER.CHUNK_ID.offset, HEADER.CHUNK_ID.value); // \"RIFF\"\n    dataView.setUint32(\n      HEADER.CHUNK_SIZE.offset,\n      arrayBuffer.byteLength - 8,\n      true\n    ); // adjustment for the first two headers - chunk id + file size\n    dataView.setUint32(HEADER.FILE_FORMAT.offset, HEADER.FILE_FORMAT.value); // \"WAVE\"\n\n    dataView.setUint32(HEADER.SUBCHUNK1_ID.offset, HEADER.SUBCHUNK1_ID.value); // \"fmt \"\n    dataView.setUint32(\n      HEADER.SUBCHUNK1_SIZE.offset,\n      HEADER.SUBCHUNK1_SIZE.value,\n      true\n    );\n    dataView.setUint16(\n      HEADER.AUDIO_FORMAT.offset,\n      HEADER.AUDIO_FORMAT.value,\n      true\n    );\n    dataView.setUint16(HEADER.CHANNEL_COUNT.offset, channelCount, true);\n    dataView.setUint32(HEADER.SAMPLE_RATE.offset, sampleRate, true);\n    dataView.setUint32(HEADER.BYTE_RATE.offset, byteRate, true);\n    dataView.setUint16(HEADER.BLOCK_ALIGN.offset, blockAlign, true);\n    dataView.setUint16(\n      HEADER.BITS_PER_SAMPLE.offset,\n      HEADER.BITS_PER_SAMPLE.value,\n      true\n    );\n\n    dataView.setUint32(HEADER.SUBCHUNK2_ID.offset, HEADER.SUBCHUNK2_ID.value); // \"data\"\n    dataView.setUint32(HEADER.SUBCHUNK2_SIZE.offset, dataChunkSize, true);\n  }\n\n  protected splitDataByChannel = (audioBuffer: AudioBuffer) =>\n    Array.from({ length: audioBuffer.numberOfChannels }, (_, i) =>\n      audioBuffer.getChannelData(i)\n    );\n}\n"]}
|
package/esm2020/lib/voice-recorder/voice-recorder-wavebar/voice-recorder-wavebar.component.mjs
ADDED
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
import { Component } from '@angular/core';
|
|
2
|
+
import { formatDuration } from '../../format-duration';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "../amplitude-recorder.service";
|
|
5
|
+
import * as i2 from "../audio-recorder.service";
|
|
6
|
+
import * as i3 from "@angular/common";
|
|
7
|
+
/**
|
|
8
|
+
* The `VoiceRecorderWavebarComponent` displays the amplitudes of the recording while the recoding is in progress
|
|
9
|
+
*/
|
|
10
|
+
export class VoiceRecorderWavebarComponent {
|
|
11
|
+
constructor(amplitudeRecorder, audioRecorder) {
|
|
12
|
+
this.amplitudeRecorder = amplitudeRecorder;
|
|
13
|
+
this.audioRecorder = audioRecorder;
|
|
14
|
+
this.isLongerThanOneHour = false;
|
|
15
|
+
this.amplitudes$ = this.amplitudeRecorder.amplitudes$;
|
|
16
|
+
this.formattedDuration = formatDuration(this.audioRecorder.durationMs / 1000);
|
|
17
|
+
this.durationComputeInterval = setInterval(() => {
|
|
18
|
+
this.isLongerThanOneHour = this.audioRecorder.durationMs / 1000 > 3600;
|
|
19
|
+
this.formattedDuration = formatDuration(this.audioRecorder.durationMs / 1000);
|
|
20
|
+
}, 1000);
|
|
21
|
+
}
|
|
22
|
+
trackByIndex(i) {
|
|
23
|
+
return i;
|
|
24
|
+
}
|
|
25
|
+
ngOnDestroy() {
|
|
26
|
+
clearInterval(this.durationComputeInterval);
|
|
27
|
+
}
|
|
28
|
+
}
|
|
29
|
+
VoiceRecorderWavebarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderWavebarComponent, deps: [{ token: i1.AmplitudeRecorderService }, { token: i2.AudioRecorderService }], target: i0.ɵɵFactoryTarget.Component });
|
|
30
|
+
VoiceRecorderWavebarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: VoiceRecorderWavebarComponent, selector: "stream-voice-recorder-wavebar", ngImport: i0, template: "<div\n class=\"str-chat__recording-timer\"\n [class.str-chat__recording-timer--hours]=\"isLongerThanOneHour\"\n>\n {{ formattedDuration }}\n</div>\n<div class=\"str-chat__waveform-box-container\">\n <div class=\"str-chat__audio_recorder__waveform-box\">\n <div\n *ngFor=\"let amplitude of amplitudes$ | async; trackBy: trackByIndex\"\n class=\"str-chat__wave-progress-bar__amplitude-bar\"\n [style.--str-chat__wave-progress-bar__amplitude-bar-height]=\"\n (amplitude ?? 0) * 100 + '%'\n \"\n ></div>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "pipe", type: i3.AsyncPipe, name: "async" }] });
|
|
31
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderWavebarComponent, decorators: [{
|
|
32
|
+
type: Component,
|
|
33
|
+
args: [{ selector: 'stream-voice-recorder-wavebar', template: "<div\n class=\"str-chat__recording-timer\"\n [class.str-chat__recording-timer--hours]=\"isLongerThanOneHour\"\n>\n {{ formattedDuration }}\n</div>\n<div class=\"str-chat__waveform-box-container\">\n <div class=\"str-chat__audio_recorder__waveform-box\">\n <div\n *ngFor=\"let amplitude of amplitudes$ | async; trackBy: trackByIndex\"\n class=\"str-chat__wave-progress-bar__amplitude-bar\"\n [style.--str-chat__wave-progress-bar__amplitude-bar-height]=\"\n (amplitude ?? 0) * 100 + '%'\n \"\n ></div>\n </div>\n</div>\n" }]
|
|
34
|
+
}], ctorParameters: function () { return [{ type: i1.AmplitudeRecorderService }, { type: i2.AudioRecorderService }]; } });
|
|
35
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtcmVjb3JkZXItd2F2ZWJhci5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zdHJlYW0tY2hhdC1hbmd1bGFyL3NyYy9saWIvdm9pY2UtcmVjb3JkZXIvdm9pY2UtcmVjb3JkZXItd2F2ZWJhci92b2ljZS1yZWNvcmRlci13YXZlYmFyLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3N0cmVhbS1jaGF0LWFuZ3VsYXIvc3JjL2xpYi92b2ljZS1yZWNvcmRlci92b2ljZS1yZWNvcmRlci13YXZlYmFyL3ZvaWNlLXJlY29yZGVyLXdhdmViYXIuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBYSxNQUFNLGVBQWUsQ0FBQztBQUlyRCxPQUFPLEVBQUUsY0FBYyxFQUFFLE1BQU0sdUJBQXVCLENBQUM7Ozs7O0FBRXZEOztHQUVHO0FBTUgsTUFBTSxPQUFPLDZCQUE2QjtJQU14QyxZQUNVLGlCQUEyQyxFQUMzQyxhQUFtQztRQURuQyxzQkFBaUIsR0FBakIsaUJBQWlCLENBQTBCO1FBQzNDLGtCQUFhLEdBQWIsYUFBYSxDQUFzQjtRQUo3Qyx3QkFBbUIsR0FBRyxLQUFLLENBQUM7UUFNMUIsSUFBSSxDQUFDLFdBQVcsR0FBRyxJQUFJLENBQUMsaUJBQWlCLENBQUMsV0FBVyxDQUFDO1FBQ3RELElBQUksQ0FBQyxpQkFBaUIsR0FBRyxjQUFjLENBQ3JDLElBQUksQ0FBQyxhQUFhLENBQUMsVUFBVSxHQUFHLElBQUksQ0FDckMsQ0FBQztRQUNGLElBQUksQ0FBQyx1QkFBdUIsR0FBRyxXQUFXLENBQUMsR0FBRyxFQUFFO1lBQzlDLElBQUksQ0FBQyxtQkFBbUIsR0FBRyxJQUFJLENBQUMsYUFBYSxDQUFDLFVBQVUsR0FBRyxJQUFJLEdBQUcsSUFBSSxDQUFDO1lBQ3ZFLElBQUksQ0FBQyxpQkFBaUIsR0FBRyxjQUFjLENBQ3JDLElBQUksQ0FBQyxhQUFhLENBQUMsVUFBVSxHQUFHLElBQUksQ0FDckMsQ0FBQztRQUNKLENBQUMsRUFBRSxJQUFJLENBQUMsQ0FBQztJQUNYLENBQUM7SUFFRCxZQUFZLENBQUMsQ0FBUztRQUNwQixPQUFPLENBQUMsQ0FBQztJQUNYLENBQUM7SUFFRCxXQUFXO1FBQ1QsYUFBYSxDQUFDLElBQUksQ0FBQyx1QkFBdUIsQ0FBQyxDQUFDO0lBQzlDLENBQUM7OzBIQTVCVSw2QkFBNkI7OEdBQTdCLDZCQUE2QixxRUNkMUMsOGlCQWlCQTsyRkRIYSw2QkFBNkI7a0JBTHpDLFNBQVM7K0JBQ0UsK0JBQStCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgQ29tcG9uZW50LCBPbkRlc3Ryb3kgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEFtcGxpdHVkZVJlY29yZGVyU2VydmljZSB9IGZyb20gJy4uL2FtcGxpdHVkZS1yZWNvcmRlci5zZXJ2aWNlJztcbmltcG9ydCB7IE9ic2VydmFibGUgfSBmcm9tICdyeGpzJztcbmltcG9ydCB7IEF1ZGlvUmVjb3JkZXJTZXJ2aWNlIH0gZnJvbSAnLi4vYXVkaW8tcmVjb3JkZXIuc2VydmljZSc7XG5pbXBvcnQgeyBmb3JtYXREdXJhdGlvbiB9IGZyb20gJy4uLy4uL2Zvcm1hdC1kdXJhdGlvbic7XG5cbi8qKlxuICogVGhlIGBWb2ljZVJlY29yZGVyV2F2ZWJhckNvbXBvbmVudGAgZGlzcGxheXMgdGhlIGFtcGxpdHVkZXMgb2YgdGhlIHJlY29yZGluZyB3aGlsZSB0aGUgcmVjb2RpbmcgaXMgaW4gcHJvZ3Jlc3NcbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnc3RyZWFtLXZvaWNlLXJlY29yZGVyLXdhdmViYXInLFxuICB0ZW1wbGF0ZVVybDogJy4vdm9pY2UtcmVjb3JkZXItd2F2ZWJhci5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlczogW10sXG59KVxuZXhwb3J0IGNsYXNzIFZvaWNlUmVjb3JkZXJXYXZlYmFyQ29tcG9uZW50IGltcGxlbWVudHMgT25EZXN0cm95IHtcbiAgYW1wbGl0dWRlcyQ6IE9ic2VydmFibGU8bnVtYmVyW10+O1xuICBmb3JtYXR0ZWREdXJhdGlvbjogc3RyaW5nO1xuICBkdXJhdGlvbkNvbXB1dGVJbnRlcnZhbDogUmV0dXJuVHlwZTx0eXBlb2Ygc2V0SW50ZXJ2YWw+O1xuICBpc0xvbmdlclRoYW5PbmVIb3VyID0gZmFsc2U7XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJpdmF0ZSBhbXBsaXR1ZGVSZWNvcmRlcjogQW1wbGl0dWRlUmVjb3JkZXJTZXJ2aWNlLFxuICAgIHByaXZhdGUgYXVkaW9SZWNvcmRlcjogQXVkaW9SZWNvcmRlclNlcnZpY2VcbiAgKSB7XG4gICAgdGhpcy5hbXBsaXR1ZGVzJCA9IHRoaXMuYW1wbGl0dWRlUmVjb3JkZXIuYW1wbGl0dWRlcyQ7XG4gICAgdGhpcy5mb3JtYXR0ZWREdXJhdGlvbiA9IGZvcm1hdER1cmF0aW9uKFxuICAgICAgdGhpcy5hdWRpb1JlY29yZGVyLmR1cmF0aW9uTXMgLyAxMDAwXG4gICAgKTtcbiAgICB0aGlzLmR1cmF0aW9uQ29tcHV0ZUludGVydmFsID0gc2V0SW50ZXJ2YWwoKCkgPT4ge1xuICAgICAgdGhpcy5pc0xvbmdlclRoYW5PbmVIb3VyID0gdGhpcy5hdWRpb1JlY29yZGVyLmR1cmF0aW9uTXMgLyAxMDAwID4gMzYwMDtcbiAgICAgIHRoaXMuZm9ybWF0dGVkRHVyYXRpb24gPSBmb3JtYXREdXJhdGlvbihcbiAgICAgICAgdGhpcy5hdWRpb1JlY29yZGVyLmR1cmF0aW9uTXMgLyAxMDAwXG4gICAgICApO1xuICAgIH0sIDEwMDApO1xuICB9XG5cbiAgdHJhY2tCeUluZGV4KGk6IG51bWJlcikge1xuICAgIHJldHVybiBpO1xuICB9XG5cbiAgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgY2xlYXJJbnRlcnZhbCh0aGlzLmR1cmF0aW9uQ29tcHV0ZUludGVydmFsKTtcbiAgfVxufVxuIiwiPGRpdlxuICBjbGFzcz1cInN0ci1jaGF0X19yZWNvcmRpbmctdGltZXJcIlxuICBbY2xhc3Muc3RyLWNoYXRfX3JlY29yZGluZy10aW1lci0taG91cnNdPVwiaXNMb25nZXJUaGFuT25lSG91clwiXG4+XG4gIHt7IGZvcm1hdHRlZER1cmF0aW9uIH19XG48L2Rpdj5cbjxkaXYgY2xhc3M9XCJzdHItY2hhdF9fd2F2ZWZvcm0tYm94LWNvbnRhaW5lclwiPlxuICA8ZGl2IGNsYXNzPVwic3RyLWNoYXRfX2F1ZGlvX3JlY29yZGVyX193YXZlZm9ybS1ib3hcIj5cbiAgICA8ZGl2XG4gICAgICAqbmdGb3I9XCJsZXQgYW1wbGl0dWRlIG9mIGFtcGxpdHVkZXMkIHwgYXN5bmM7IHRyYWNrQnk6IHRyYWNrQnlJbmRleFwiXG4gICAgICBjbGFzcz1cInN0ci1jaGF0X193YXZlLXByb2dyZXNzLWJhcl9fYW1wbGl0dWRlLWJhclwiXG4gICAgICBbc3R5bGUuLS1zdHItY2hhdF9fd2F2ZS1wcm9ncmVzcy1iYXJfX2FtcGxpdHVkZS1iYXItaGVpZ2h0XT1cIlxuICAgICAgICAoYW1wbGl0dWRlID8/IDApICogMTAwICsgJyUnXG4gICAgICBcIlxuICAgID48L2Rpdj5cbiAgPC9kaXY+XG48L2Rpdj5cbiJdfQ==
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
import { Component, Input, } from '@angular/core';
|
|
2
|
+
import { MediaRecordingState } from './media-recorder';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "./audio-recorder.service";
|
|
5
|
+
import * as i2 from "@angular/common";
|
|
6
|
+
import * as i3 from "../voice-recording/voice-recording.component";
|
|
7
|
+
import * as i4 from "../icon/icon-placeholder/icon-placeholder.component";
|
|
8
|
+
import * as i5 from "../icon/loading-indicator/loading-indicator.component";
|
|
9
|
+
import * as i6 from "./voice-recorder-wavebar/voice-recorder-wavebar.component";
|
|
10
|
+
/**
|
|
11
|
+
* The `VoiceRecorderComponent` makes it possible to record audio, and then upload it as a voice recording attachment
|
|
12
|
+
*/
|
|
13
|
+
export class VoiceRecorderComponent {
|
|
14
|
+
constructor(recorder) {
|
|
15
|
+
this.recorder = recorder;
|
|
16
|
+
this.recordState = MediaRecordingState.STOPPED;
|
|
17
|
+
this.isLoading = false;
|
|
18
|
+
this.MediaRecordingState = MediaRecordingState;
|
|
19
|
+
this.subscriptions = [];
|
|
20
|
+
}
|
|
21
|
+
ngOnInit() {
|
|
22
|
+
this.subscriptions.push(this.recorder.recordingState$.subscribe((s) => {
|
|
23
|
+
this.recordState = s;
|
|
24
|
+
if (this.recordState === MediaRecordingState.ERROR) {
|
|
25
|
+
this.voiceRecorderService?.isRecorderVisible$.next(false);
|
|
26
|
+
}
|
|
27
|
+
}));
|
|
28
|
+
}
|
|
29
|
+
ngOnChanges(changes) {
|
|
30
|
+
if (changes.voiceRecorderService && this.voiceRecorderService) {
|
|
31
|
+
this.isVisibleSubscription =
|
|
32
|
+
this.voiceRecorderService.isRecorderVisible$.subscribe((isVisible) => {
|
|
33
|
+
if (!isVisible) {
|
|
34
|
+
this.recording = undefined;
|
|
35
|
+
this.isLoading = false;
|
|
36
|
+
}
|
|
37
|
+
});
|
|
38
|
+
}
|
|
39
|
+
else {
|
|
40
|
+
this.isVisibleSubscription?.unsubscribe();
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
ngOnDestroy() {
|
|
44
|
+
this.subscriptions.forEach((s) => s.unsubscribe());
|
|
45
|
+
}
|
|
46
|
+
cancel() {
|
|
47
|
+
if (this.recording) {
|
|
48
|
+
this.recording = undefined;
|
|
49
|
+
}
|
|
50
|
+
else {
|
|
51
|
+
void this.recorder.stop({ cancel: true });
|
|
52
|
+
}
|
|
53
|
+
this.voiceRecorderService?.isRecorderVisible$.next(false);
|
|
54
|
+
}
|
|
55
|
+
async stop() {
|
|
56
|
+
this.recording = await this.recorder.stop();
|
|
57
|
+
}
|
|
58
|
+
pause() {
|
|
59
|
+
this.recorder.pause();
|
|
60
|
+
}
|
|
61
|
+
resume() {
|
|
62
|
+
this.recorder.resume();
|
|
63
|
+
}
|
|
64
|
+
uploadRecording() {
|
|
65
|
+
if (!this.recording) {
|
|
66
|
+
return;
|
|
67
|
+
}
|
|
68
|
+
this.isLoading = true;
|
|
69
|
+
this.voiceRecorderService?.recording$.next(this.recording);
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
VoiceRecorderComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderComponent, deps: [{ token: i1.AudioRecorderService }], target: i0.ɵɵFactoryTarget.Component });
|
|
73
|
+
VoiceRecorderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: VoiceRecorderComponent, selector: "stream-voice-recorder", inputs: { voiceRecorderService: "voiceRecorderService" }, providers: [], usesOnChanges: true, ngImport: i0, template: "<div\n class=\"str-chat__audio_recorder-container\"\n *ngIf=\"voiceRecorderService?.isRecorderVisible$ | async\"\n>\n <div class=\"str-chat__audio_recorder\" data-testid=\"audio-recorder\">\n <button\n class=\"str-chat__audio_recorder__cancel-button\"\n data-testid=\"cancel-recording-audio-button\"\n [disabled]=\"isLoading\"\n (click)=\"cancel()\"\n (keyup.enter)=\"cancel()\"\n >\n <stream-icon-placeholder icon=\"bin\"></stream-icon-placeholder>\n </button>\n <stream-voice-recorder-wavebar\n *ngIf=\"\n (recordState === MediaRecordingState.RECORDING ||\n recordState === MediaRecordingState.PAUSED) &&\n !recording\n \"\n ></stream-voice-recorder-wavebar>\n <!-- eslint-disable @angular-eslint/template/no-any -->\n <stream-voice-recording\n [attachment]=\"$any(recording)\"\n *ngIf=\"!!recording\"\n ></stream-voice-recording>\n <!-- eslint-enable @angular-eslint/template/no-any -->\n <button\n *ngIf=\"recordState === MediaRecordingState.PAUSED && !recording\"\n class=\"str-chat__audio_recorder__resume-recording-button\"\n (click)=\"resume()\"\n (keyup.enter)=\"resume()\"\n >\n <stream-icon-placeholder icon=\"mic\"></stream-icon-placeholder>\n </button>\n <button\n *ngIf=\"recordState === MediaRecordingState.RECORDING && !recording\"\n class=\"str-chat__audio_recorder__pause-recording-button\"\n data-testid=\"pause-recording-audio-button\"\n (click)=\"pause()\"\n (keyup.enter)=\"pause()\"\n >\n <stream-icon-placeholder icon=\"pause\"></stream-icon-placeholder>\n </button>\n <ng-container\n *ngIf=\"recordState === MediaRecordingState.STOPPED; else stopButton\"\n >\n <button\n class=\"str-chat__audio_recorder__complete-button\"\n data-testid=\"audio-recorder-complete-button\"\n [disabled]=\"!recording\"\n (click)=\"uploadRecording()\"\n (keyup.enter)=\"uploadRecording()\"\n >\n <stream-loading-indicator\n *ngIf=\"isLoading; else sendIcon\"\n ></stream-loading-indicator>\n <ng-template #sendIcon>\n <stream-icon-placeholder icon=\"send\"></stream-icon-placeholder>\n </ng-template>\n </button>\n </ng-container>\n <ng-template #stopButton>\n <button\n class=\"str-chat__audio_recorder__stop-button\"\n data-testid=\"audio-recorder-stop-button\"\n [disabled]=\"recordState === MediaRecordingState.STOPPED\"\n (click)=\"stop()\"\n (keyup.enter)=\"stop()\"\n >\n <stream-icon-placeholder icon=\"delivered\"></stream-icon-placeholder>\n </button>\n </ng-template>\n </div>\n</div>\n", dependencies: [{ kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i3.VoiceRecordingComponent, selector: "stream-voice-recording", inputs: ["attachment"] }, { kind: "component", type: i4.IconPlaceholderComponent, selector: "stream-icon-placeholder", inputs: ["icon"] }, { kind: "component", type: i5.LoadingIndicatorComponent, selector: "stream-loading-indicator" }, { kind: "component", type: i6.VoiceRecorderWavebarComponent, selector: "stream-voice-recorder-wavebar" }, { kind: "pipe", type: i2.AsyncPipe, name: "async" }] });
|
|
74
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderComponent, decorators: [{
|
|
75
|
+
type: Component,
|
|
76
|
+
args: [{ selector: 'stream-voice-recorder', providers: [], template: "<div\n class=\"str-chat__audio_recorder-container\"\n *ngIf=\"voiceRecorderService?.isRecorderVisible$ | async\"\n>\n <div class=\"str-chat__audio_recorder\" data-testid=\"audio-recorder\">\n <button\n class=\"str-chat__audio_recorder__cancel-button\"\n data-testid=\"cancel-recording-audio-button\"\n [disabled]=\"isLoading\"\n (click)=\"cancel()\"\n (keyup.enter)=\"cancel()\"\n >\n <stream-icon-placeholder icon=\"bin\"></stream-icon-placeholder>\n </button>\n <stream-voice-recorder-wavebar\n *ngIf=\"\n (recordState === MediaRecordingState.RECORDING ||\n recordState === MediaRecordingState.PAUSED) &&\n !recording\n \"\n ></stream-voice-recorder-wavebar>\n <!-- eslint-disable @angular-eslint/template/no-any -->\n <stream-voice-recording\n [attachment]=\"$any(recording)\"\n *ngIf=\"!!recording\"\n ></stream-voice-recording>\n <!-- eslint-enable @angular-eslint/template/no-any -->\n <button\n *ngIf=\"recordState === MediaRecordingState.PAUSED && !recording\"\n class=\"str-chat__audio_recorder__resume-recording-button\"\n (click)=\"resume()\"\n (keyup.enter)=\"resume()\"\n >\n <stream-icon-placeholder icon=\"mic\"></stream-icon-placeholder>\n </button>\n <button\n *ngIf=\"recordState === MediaRecordingState.RECORDING && !recording\"\n class=\"str-chat__audio_recorder__pause-recording-button\"\n data-testid=\"pause-recording-audio-button\"\n (click)=\"pause()\"\n (keyup.enter)=\"pause()\"\n >\n <stream-icon-placeholder icon=\"pause\"></stream-icon-placeholder>\n </button>\n <ng-container\n *ngIf=\"recordState === MediaRecordingState.STOPPED; else stopButton\"\n >\n <button\n class=\"str-chat__audio_recorder__complete-button\"\n data-testid=\"audio-recorder-complete-button\"\n [disabled]=\"!recording\"\n (click)=\"uploadRecording()\"\n (keyup.enter)=\"uploadRecording()\"\n >\n <stream-loading-indicator\n *ngIf=\"isLoading; else sendIcon\"\n ></stream-loading-indicator>\n <ng-template #sendIcon>\n <stream-icon-placeholder icon=\"send\"></stream-icon-placeholder>\n </ng-template>\n </button>\n </ng-container>\n <ng-template #stopButton>\n <button\n class=\"str-chat__audio_recorder__stop-button\"\n data-testid=\"audio-recorder-stop-button\"\n [disabled]=\"recordState === MediaRecordingState.STOPPED\"\n (click)=\"stop()\"\n (keyup.enter)=\"stop()\"\n >\n <stream-icon-placeholder icon=\"delivered\"></stream-icon-placeholder>\n </button>\n </ng-template>\n </div>\n</div>\n" }]
|
|
77
|
+
}], ctorParameters: function () { return [{ type: i1.AudioRecorderService }]; }, propDecorators: { voiceRecorderService: [{
|
|
78
|
+
type: Input
|
|
79
|
+
}] } });
|
|
80
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-recorder.component.js","sourceRoot":"","sources":["../../../../../projects/stream-chat-angular/src/lib/voice-recorder/voice-recorder.component.ts","../../../../../projects/stream-chat-angular/src/lib/voice-recorder/voice-recorder.component.html"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,KAAK,GAKN,MAAM,eAAe,CAAC;AAEvB,OAAO,EAAE,mBAAmB,EAAE,MAAM,kBAAkB,CAAC;;;;;;;;AAKvD;;GAEG;AAOH,MAAM,OAAO,sBAAsB;IASjC,YAA4B,QAA8B;QAA9B,aAAQ,GAAR,QAAQ,CAAsB;QAP1D,gBAAW,GAAwB,mBAAmB,CAAC,OAAO,CAAC;QAC/D,cAAS,GAAG,KAAK,CAAC;QAET,wBAAmB,GAAG,mBAAmB,CAAC;QAC3C,kBAAa,GAAmB,EAAE,CAAC;IAGkB,CAAC;IAE9D,QAAQ;QACN,IAAI,CAAC,aAAa,CAAC,IAAI,CACrB,IAAI,CAAC,QAAQ,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC,CAAC,EAAE,EAAE;YAC5C,IAAI,CAAC,WAAW,GAAG,CAAC,CAAC;YACrB,IAAI,IAAI,CAAC,WAAW,KAAK,mBAAmB,CAAC,KAAK,EAAE;gBAClD,IAAI,CAAC,oBAAoB,EAAE,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;aAC3D;QACH,CAAC,CAAC,CACH,CAAC;IACJ,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,oBAAoB,IAAI,IAAI,CAAC,oBAAoB,EAAE;YAC7D,IAAI,CAAC,qBAAqB;gBACxB,IAAI,CAAC,oBAAoB,CAAC,kBAAkB,CAAC,SAAS,CAAC,CAAC,SAAS,EAAE,EAAE;oBACnE,IAAI,CAAC,SAAS,EAAE;wBACd,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;wBAC3B,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;qBACxB;gBACH,CAAC,CAAC,CAAC;SACN;aAAM;YACL,IAAI,CAAC,qBAAqB,EAAE,WAAW,EAAE,CAAC;SAC3C;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC;IACrD,CAAC;IAED,MAAM;QACJ,IAAI,IAAI,CAAC,SAAS,EAAE;YAClB,IAAI,CAAC,SAAS,GAAG,SAAS,CAAC;SAC5B;aAAM;YACL,KAAK,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CAAC;SAC3C;QACD,IAAI,CAAC,oBAAoB,EAAE,kBAAkB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC5D,CAAC;IAED,KAAK,CAAC,IAAI;QACR,IAAI,CAAC,SAAS,GAAG,MAAM,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;IAC9C,CAAC;IAED,KAAK;QACH,IAAI,CAAC,QAAQ,CAAC,KAAK,EAAE,CAAC;IACxB,CAAC;IAED,MAAM;QACJ,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;IACzB,CAAC;IAED,eAAe;QACb,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;YACnB,OAAO;SACR;QACD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,oBAAoB,EAAE,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IAC7D,CAAC;;mHAnEU,sBAAsB;uGAAtB,sBAAsB,0GAFtB,EAAE,+CCrBf,ksFA2EA;2FDpDa,sBAAsB;kBANlC,SAAS;+BACE,uBAAuB,aAGtB,EAAE;2GAGJ,oBAAoB;sBAA5B,KAAK","sourcesContent":["import {\n  Component,\n  Input,\n  OnChanges,\n  OnDestroy,\n  OnInit,\n  SimpleChanges,\n} from '@angular/core';\nimport { AudioRecorderService } from './audio-recorder.service';\nimport { MediaRecordingState } from './media-recorder';\nimport { Subscription } from 'rxjs';\nimport { AudioRecording } from '../types';\nimport { VoiceRecorderService } from '../message-input/voice-recorder.service';\n\n/**\n * The `VoiceRecorderComponent` makes it possible to record audio, and then upload it as a voice recording attachment\n */\n@Component({\n  selector: 'stream-voice-recorder',\n  templateUrl: './voice-recorder.component.html',\n  styles: [],\n  providers: [],\n})\nexport class VoiceRecorderComponent implements OnInit, OnDestroy, OnChanges {\n  @Input() voiceRecorderService?: VoiceRecorderService;\n  recordState: MediaRecordingState = MediaRecordingState.STOPPED;\n  isLoading = false;\n  recording?: AudioRecording;\n  readonly MediaRecordingState = MediaRecordingState;\n  private subscriptions: Subscription[] = [];\n  private isVisibleSubscription?: Subscription;\n\n  constructor(public readonly recorder: AudioRecorderService) {}\n\n  ngOnInit(): void {\n    this.subscriptions.push(\n      this.recorder.recordingState$.subscribe((s) => {\n        this.recordState = s;\n        if (this.recordState === MediaRecordingState.ERROR) {\n          this.voiceRecorderService?.isRecorderVisible$.next(false);\n        }\n      })\n    );\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.voiceRecorderService && this.voiceRecorderService) {\n      this.isVisibleSubscription =\n        this.voiceRecorderService.isRecorderVisible$.subscribe((isVisible) => {\n          if (!isVisible) {\n            this.recording = undefined;\n            this.isLoading = false;\n          }\n        });\n    } else {\n      this.isVisibleSubscription?.unsubscribe();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.subscriptions.forEach((s) => s.unsubscribe());\n  }\n\n  cancel() {\n    if (this.recording) {\n      this.recording = undefined;\n    } else {\n      void this.recorder.stop({ cancel: true });\n    }\n    this.voiceRecorderService?.isRecorderVisible$.next(false);\n  }\n\n  async stop() {\n    this.recording = await this.recorder.stop();\n  }\n\n  pause() {\n    this.recorder.pause();\n  }\n\n  resume() {\n    this.recorder.resume();\n  }\n\n  uploadRecording() {\n    if (!this.recording) {\n      return;\n    }\n    this.isLoading = true;\n    this.voiceRecorderService?.recording$.next(this.recording);\n  }\n}\n","<div\n  class=\"str-chat__audio_recorder-container\"\n  *ngIf=\"voiceRecorderService?.isRecorderVisible$ | async\"\n>\n  <div class=\"str-chat__audio_recorder\" data-testid=\"audio-recorder\">\n    <button\n      class=\"str-chat__audio_recorder__cancel-button\"\n      data-testid=\"cancel-recording-audio-button\"\n      [disabled]=\"isLoading\"\n      (click)=\"cancel()\"\n      (keyup.enter)=\"cancel()\"\n    >\n      <stream-icon-placeholder icon=\"bin\"></stream-icon-placeholder>\n    </button>\n    <stream-voice-recorder-wavebar\n      *ngIf=\"\n        (recordState === MediaRecordingState.RECORDING ||\n          recordState === MediaRecordingState.PAUSED) &&\n        !recording\n      \"\n    ></stream-voice-recorder-wavebar>\n    <!-- eslint-disable @angular-eslint/template/no-any -->\n    <stream-voice-recording\n      [attachment]=\"$any(recording)\"\n      *ngIf=\"!!recording\"\n    ></stream-voice-recording>\n    <!-- eslint-enable @angular-eslint/template/no-any -->\n    <button\n      *ngIf=\"recordState === MediaRecordingState.PAUSED && !recording\"\n      class=\"str-chat__audio_recorder__resume-recording-button\"\n      (click)=\"resume()\"\n      (keyup.enter)=\"resume()\"\n    >\n      <stream-icon-placeholder icon=\"mic\"></stream-icon-placeholder>\n    </button>\n    <button\n      *ngIf=\"recordState === MediaRecordingState.RECORDING && !recording\"\n      class=\"str-chat__audio_recorder__pause-recording-button\"\n      data-testid=\"pause-recording-audio-button\"\n      (click)=\"pause()\"\n      (keyup.enter)=\"pause()\"\n    >\n      <stream-icon-placeholder icon=\"pause\"></stream-icon-placeholder>\n    </button>\n    <ng-container\n      *ngIf=\"recordState === MediaRecordingState.STOPPED; else stopButton\"\n    >\n      <button\n        class=\"str-chat__audio_recorder__complete-button\"\n        data-testid=\"audio-recorder-complete-button\"\n        [disabled]=\"!recording\"\n        (click)=\"uploadRecording()\"\n        (keyup.enter)=\"uploadRecording()\"\n      >\n        <stream-loading-indicator\n          *ngIf=\"isLoading; else sendIcon\"\n        ></stream-loading-indicator>\n        <ng-template #sendIcon>\n          <stream-icon-placeholder icon=\"send\"></stream-icon-placeholder>\n        </ng-template>\n      </button>\n    </ng-container>\n    <ng-template #stopButton>\n      <button\n        class=\"str-chat__audio_recorder__stop-button\"\n        data-testid=\"audio-recorder-stop-button\"\n        [disabled]=\"recordState === MediaRecordingState.STOPPED\"\n        (click)=\"stop()\"\n        (keyup.enter)=\"stop()\"\n      >\n        <stream-icon-placeholder icon=\"delivered\"></stream-icon-placeholder>\n      </button>\n    </ng-template>\n  </div>\n</div>\n"]}
|
|
@@ -10,15 +10,15 @@ import { AmplitudeRecorderService } from './amplitude-recorder.service';
|
|
|
10
10
|
import { VoiceRecorderWavebarComponent } from './voice-recorder-wavebar/voice-recorder-wavebar.component';
|
|
11
11
|
import * as i0 from "@angular/core";
|
|
12
12
|
export class VoiceRecorderModule {
|
|
13
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: VoiceRecorderModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule }); }
|
|
14
|
-
static { this.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "17.0.0", ngImport: i0, type: VoiceRecorderModule, declarations: [VoiceRecorderComponent, VoiceRecorderWavebarComponent], imports: [CommonModule, VoiceRecordingModule, IconModule, TranslateModule], exports: [VoiceRecorderComponent, VoiceRecorderWavebarComponent] }); }
|
|
15
|
-
static { this.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "17.0.0", ngImport: i0, type: VoiceRecorderModule, providers: [
|
|
16
|
-
AudioRecorderService,
|
|
17
|
-
TranscoderService,
|
|
18
|
-
AmplitudeRecorderService,
|
|
19
|
-
], imports: [CommonModule, VoiceRecordingModule, IconModule, TranslateModule] }); }
|
|
20
13
|
}
|
|
21
|
-
i0.ɵɵ
|
|
14
|
+
VoiceRecorderModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
|
|
15
|
+
VoiceRecorderModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderModule, declarations: [VoiceRecorderComponent, VoiceRecorderWavebarComponent], imports: [CommonModule, VoiceRecordingModule, IconModule, TranslateModule], exports: [VoiceRecorderComponent, VoiceRecorderWavebarComponent] });
|
|
16
|
+
VoiceRecorderModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderModule, providers: [
|
|
17
|
+
AudioRecorderService,
|
|
18
|
+
TranscoderService,
|
|
19
|
+
AmplitudeRecorderService,
|
|
20
|
+
], imports: [CommonModule, VoiceRecordingModule, IconModule, TranslateModule] });
|
|
21
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecorderModule, decorators: [{
|
|
22
22
|
type: NgModule,
|
|
23
23
|
args: [{
|
|
24
24
|
declarations: [VoiceRecorderComponent, VoiceRecorderWavebarComponent],
|
|
@@ -31,4 +31,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.0", ngImpor
|
|
|
31
31
|
],
|
|
32
32
|
}]
|
|
33
33
|
}] });
|
|
34
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
34
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidm9pY2UtcmVjb3JkZXIubW9kdWxlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc3RyZWFtLWNoYXQtYW5ndWxhci9zcmMvbGliL3ZvaWNlLXJlY29yZGVyL3ZvaWNlLXJlY29yZGVyLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsc0JBQXNCLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUNwRSxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSwyQ0FBMkMsQ0FBQztBQUNqRixPQUFPLEVBQUUsVUFBVSxFQUFFLE1BQU0scUJBQXFCLENBQUM7QUFDakQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ3RELE9BQU8sRUFBRSxvQkFBb0IsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQ2hFLE9BQU8sRUFBRSxpQkFBaUIsRUFBRSxNQUFNLHNCQUFzQixDQUFDO0FBQ3pELE9BQU8sRUFBRSx3QkFBd0IsRUFBRSxNQUFNLDhCQUE4QixDQUFDO0FBQ3hFLE9BQU8sRUFBRSw2QkFBNkIsRUFBRSxNQUFNLDJEQUEyRCxDQUFDOztBQVkxRyxNQUFNLE9BQU8sbUJBQW1COztnSEFBbkIsbUJBQW1CO2lIQUFuQixtQkFBbUIsaUJBVGYsc0JBQXNCLEVBQUUsNkJBQTZCLGFBQzFELFlBQVksRUFBRSxvQkFBb0IsRUFBRSxVQUFVLEVBQUUsZUFBZSxhQUMvRCxzQkFBc0IsRUFBRSw2QkFBNkI7aUhBT3BELG1CQUFtQixhQU5uQjtRQUNULG9CQUFvQjtRQUNwQixpQkFBaUI7UUFDakIsd0JBQXdCO0tBQ3pCLFlBTlMsWUFBWSxFQUFFLG9CQUFvQixFQUFFLFVBQVUsRUFBRSxlQUFlOzJGQVE5RCxtQkFBbUI7a0JBVi9CLFFBQVE7bUJBQUM7b0JBQ1IsWUFBWSxFQUFFLENBQUMsc0JBQXNCLEVBQUUsNkJBQTZCLENBQUM7b0JBQ3JFLE9BQU8sRUFBRSxDQUFDLFlBQVksRUFBRSxvQkFBb0IsRUFBRSxVQUFVLEVBQUUsZUFBZSxDQUFDO29CQUMxRSxPQUFPLEVBQUUsQ0FBQyxzQkFBc0IsRUFBRSw2QkFBNkIsQ0FBQztvQkFDaEUsU0FBUyxFQUFFO3dCQUNULG9CQUFvQjt3QkFDcEIsaUJBQWlCO3dCQUNqQix3QkFBd0I7cUJBQ3pCO2lCQUNGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBWb2ljZVJlY29yZGVyQ29tcG9uZW50IH0gZnJvbSAnLi92b2ljZS1yZWNvcmRlci5jb21wb25lbnQnO1xuaW1wb3J0IHsgVm9pY2VSZWNvcmRpbmdNb2R1bGUgfSBmcm9tICcuLi92b2ljZS1yZWNvcmRpbmcvdm9pY2UtcmVjb3JkaW5nLm1vZHVsZSc7XG5pbXBvcnQgeyBJY29uTW9kdWxlIH0gZnJvbSAnLi4vaWNvbi9pY29uLm1vZHVsZSc7XG5pbXBvcnQgeyBUcmFuc2xhdGVNb2R1bGUgfSBmcm9tICdAbmd4LXRyYW5zbGF0ZS9jb3JlJztcbmltcG9ydCB7IEF1ZGlvUmVjb3JkZXJTZXJ2aWNlIH0gZnJvbSAnLi9hdWRpby1yZWNvcmRlci5zZXJ2aWNlJztcbmltcG9ydCB7IFRyYW5zY29kZXJTZXJ2aWNlIH0gZnJvbSAnLi90cmFuc2NvZGVyLnNlcnZpY2UnO1xuaW1wb3J0IHsgQW1wbGl0dWRlUmVjb3JkZXJTZXJ2aWNlIH0gZnJvbSAnLi9hbXBsaXR1ZGUtcmVjb3JkZXIuc2VydmljZSc7XG5pbXBvcnQgeyBWb2ljZVJlY29yZGVyV2F2ZWJhckNvbXBvbmVudCB9IGZyb20gJy4vdm9pY2UtcmVjb3JkZXItd2F2ZWJhci92b2ljZS1yZWNvcmRlci13YXZlYmFyLmNvbXBvbmVudCc7XG5cbkBOZ01vZHVsZSh7XG4gIGRlY2xhcmF0aW9uczogW1ZvaWNlUmVjb3JkZXJDb21wb25lbnQsIFZvaWNlUmVjb3JkZXJXYXZlYmFyQ29tcG9uZW50XSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgVm9pY2VSZWNvcmRpbmdNb2R1bGUsIEljb25Nb2R1bGUsIFRyYW5zbGF0ZU1vZHVsZV0sXG4gIGV4cG9ydHM6IFtWb2ljZVJlY29yZGVyQ29tcG9uZW50LCBWb2ljZVJlY29yZGVyV2F2ZWJhckNvbXBvbmVudF0sXG4gIHByb3ZpZGVyczogW1xuICAgIEF1ZGlvUmVjb3JkZXJTZXJ2aWNlLFxuICAgIFRyYW5zY29kZXJTZXJ2aWNlLFxuICAgIEFtcGxpdHVkZVJlY29yZGVyU2VydmljZSxcbiAgXSxcbn0pXG5leHBvcnQgY2xhc3MgVm9pY2VSZWNvcmRlck1vZHVsZSB7fVxuIl19
|
package/esm2020/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.mjs
ADDED
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
import { Component, Input, ViewChild, } from '@angular/core';
|
|
2
|
+
import { resampleWaveForm } from '../../wave-form-sampler';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
import * as i1 from "@angular/common";
|
|
5
|
+
/**
|
|
6
|
+
* This component can be used to visualize the wave bar of a voice recording
|
|
7
|
+
*/
|
|
8
|
+
export class VoiceRecordingWavebarComponent {
|
|
9
|
+
constructor(ngZone, cdRef) {
|
|
10
|
+
this.ngZone = ngZone;
|
|
11
|
+
this.cdRef = cdRef;
|
|
12
|
+
/**
|
|
13
|
+
* The waveform data to visualize
|
|
14
|
+
*/
|
|
15
|
+
this.waveFormData = [];
|
|
16
|
+
this.resampledWaveFormData = [];
|
|
17
|
+
this.progress = 0;
|
|
18
|
+
this.isDragging = false;
|
|
19
|
+
this.sampleSize = 40;
|
|
20
|
+
this.isViewInited = false;
|
|
21
|
+
}
|
|
22
|
+
ngOnInit() {
|
|
23
|
+
this.containerSizeChanged();
|
|
24
|
+
if (this.container?.nativeElement) {
|
|
25
|
+
this.ngZone.runOutsideAngular(() => {
|
|
26
|
+
new ResizeObserver(() => {
|
|
27
|
+
this.containerSizeChanged();
|
|
28
|
+
}).observe(this.container.nativeElement);
|
|
29
|
+
});
|
|
30
|
+
}
|
|
31
|
+
}
|
|
32
|
+
ngOnChanges(changes) {
|
|
33
|
+
if (changes.waveFormData) {
|
|
34
|
+
this.resampledWaveFormData = resampleWaveForm(this.waveFormData, this.sampleSize);
|
|
35
|
+
}
|
|
36
|
+
if (changes.audioElement) {
|
|
37
|
+
this.ngZone.runOutsideAngular(() => {
|
|
38
|
+
this.audioElement?.addEventListener('timeupdate', () => {
|
|
39
|
+
const progress = (this.audioElement?.currentTime || 0) / (this.duration || 0) || 0;
|
|
40
|
+
if (Math.abs(progress - this.progress) >= 0.02) {
|
|
41
|
+
this.ngZone.run(() => {
|
|
42
|
+
this.progress = progress;
|
|
43
|
+
this.cdRef.detectChanges();
|
|
44
|
+
});
|
|
45
|
+
}
|
|
46
|
+
});
|
|
47
|
+
});
|
|
48
|
+
}
|
|
49
|
+
}
|
|
50
|
+
ngAfterViewInit() {
|
|
51
|
+
this.isViewInited = true;
|
|
52
|
+
}
|
|
53
|
+
seek(event) {
|
|
54
|
+
const containerWidth = this.container?.nativeElement?.getBoundingClientRect().width || 0;
|
|
55
|
+
const containerStart = this.container?.nativeElement?.getBoundingClientRect()?.x || 0;
|
|
56
|
+
// eslint-disable-next-line @typescript-eslint/no-unsafe-member-access
|
|
57
|
+
const progress = (event.x - containerStart) / containerWidth;
|
|
58
|
+
if (!isNaN(progress) && this.audioElement) {
|
|
59
|
+
const duration = this.duration || 0;
|
|
60
|
+
const time = duration * progress;
|
|
61
|
+
this.audioElement.currentTime = time;
|
|
62
|
+
}
|
|
63
|
+
}
|
|
64
|
+
trackByIndex(index) {
|
|
65
|
+
return index;
|
|
66
|
+
}
|
|
67
|
+
containerSizeChanged() {
|
|
68
|
+
if (!this.container?.nativeElement) {
|
|
69
|
+
return;
|
|
70
|
+
}
|
|
71
|
+
const containerWidth = this.container.nativeElement.clientWidth;
|
|
72
|
+
if (containerWidth === 0) {
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
const barWidth = +getComputedStyle(this.container.nativeElement)
|
|
76
|
+
.getPropertyValue('--str-chat__voice-recording-amplitude-bar-width')
|
|
77
|
+
.replace('px', '');
|
|
78
|
+
const barGap = +getComputedStyle(this.container.nativeElement)
|
|
79
|
+
.getPropertyValue('--str-chat__voice-recording-amplitude-bar-gap-width')
|
|
80
|
+
.replace('px', '');
|
|
81
|
+
if (!isNaN(barWidth) && !isNaN(barGap)) {
|
|
82
|
+
const sampleSize = Math.floor(containerWidth / (barWidth + barGap));
|
|
83
|
+
if (sampleSize !== this.sampleSize &&
|
|
84
|
+
!isNaN(sampleSize) &&
|
|
85
|
+
sampleSize !== Infinity) {
|
|
86
|
+
this.ngZone.run(() => {
|
|
87
|
+
this.sampleSize = sampleSize;
|
|
88
|
+
this.resampledWaveFormData = resampleWaveForm(this.waveFormData, this.sampleSize);
|
|
89
|
+
if (this.isViewInited) {
|
|
90
|
+
this.cdRef.detectChanges();
|
|
91
|
+
}
|
|
92
|
+
});
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
VoiceRecordingWavebarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecordingWavebarComponent, deps: [{ token: i0.NgZone }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
98
|
+
VoiceRecordingWavebarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.0.4", type: VoiceRecordingWavebarComponent, selector: "stream-voice-recording-wavebar", inputs: { audioElement: "audioElement", waveFormData: "waveFormData", duration: "duration" }, viewQueries: [{ propertyName: "container", first: true, predicate: ["container"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<!--eslint-disable @angular-eslint/template/click-events-have-key-events-->\n<div\n #container\n class=\"str-chat__wave-progress-bar__track\"\n data-testid=\"wave-progress-bar-track\"\n role=\"progressbar\"\n (mousedown)=\"isDragging = true\"\n (mouseup)=\"isDragging = false\"\n (mouseleave)=\"isDragging = false\"\n (mousemove)=\"isDragging ? seek($event) : null\"\n (click)=\"seek($event)\"\n>\n <!--eslint-enable @angular-eslint/template/click-events-have-key-events-->\n <div\n *ngFor=\"\n let dataPoint of resampledWaveFormData;\n let i = index;\n trackBy: trackByIndex\n \"\n class=\"str-chat__wave-progress-bar__amplitude-bar\"\n [class.str-chat__wave-progress-bar__amplitude-bar--active]=\"\n progress > i / resampledWaveFormData.length\n \"\n [style.--str-chat__wave-progress-bar__amplitude-bar-height]=\"\n dataPoint ? dataPoint * 100 + '%' : '0%'\n \"\n ></div>\n <div\n class=\"str-chat__wave-progress-bar__progress-indicator\"\n data-testid=\"wave-progress-bar-progress-indicator\"\n [ngStyle]=\"{ 'inset-inline-start': progress * 100 + '%' }\"\n ></div>\n</div>\n", dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }] });
|
|
99
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.0.4", ngImport: i0, type: VoiceRecordingWavebarComponent, decorators: [{
|
|
100
|
+
type: Component,
|
|
101
|
+
args: [{ selector: 'stream-voice-recording-wavebar', template: "<!--eslint-disable @angular-eslint/template/click-events-have-key-events-->\n<div\n #container\n class=\"str-chat__wave-progress-bar__track\"\n data-testid=\"wave-progress-bar-track\"\n role=\"progressbar\"\n (mousedown)=\"isDragging = true\"\n (mouseup)=\"isDragging = false\"\n (mouseleave)=\"isDragging = false\"\n (mousemove)=\"isDragging ? seek($event) : null\"\n (click)=\"seek($event)\"\n>\n <!--eslint-enable @angular-eslint/template/click-events-have-key-events-->\n <div\n *ngFor=\"\n let dataPoint of resampledWaveFormData;\n let i = index;\n trackBy: trackByIndex\n \"\n class=\"str-chat__wave-progress-bar__amplitude-bar\"\n [class.str-chat__wave-progress-bar__amplitude-bar--active]=\"\n progress > i / resampledWaveFormData.length\n \"\n [style.--str-chat__wave-progress-bar__amplitude-bar-height]=\"\n dataPoint ? dataPoint * 100 + '%' : '0%'\n \"\n ></div>\n <div\n class=\"str-chat__wave-progress-bar__progress-indicator\"\n data-testid=\"wave-progress-bar-progress-indicator\"\n [ngStyle]=\"{ 'inset-inline-start': progress * 100 + '%' }\"\n ></div>\n</div>\n" }]
|
|
102
|
+
}], ctorParameters: function () { return [{ type: i0.NgZone }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { audioElement: [{
|
|
103
|
+
type: Input
|
|
104
|
+
}], waveFormData: [{
|
|
105
|
+
type: Input
|
|
106
|
+
}], duration: [{
|
|
107
|
+
type: Input
|
|
108
|
+
}], container: [{
|
|
109
|
+
type: ViewChild,
|
|
110
|
+
args: ['container', { static: true }]
|
|
111
|
+
}] } });
|
|
112
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"voice-recording-wavebar.component.js","sourceRoot":"","sources":["../../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.ts","../../../../../../projects/stream-chat-angular/src/lib/voice-recording/voice-recording-wavebar/voice-recording-wavebar.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  Input,\n  NgZone,\n  OnChanges,\n  OnInit,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { resampleWaveForm } from '../../wave-form-sampler';\n\n/**\n * This component can be used to visualize the wave bar of a voice recording\n */\n@Component({\n  selector: 'stream-voice-recording-wavebar',\n  templateUrl: './voice-recording-wavebar.component.html',\n  styles: [],\n})\nexport class VoiceRecordingWavebarComponent\n  implements OnInit, OnChanges, AfterViewInit\n{\n  /**\n   * The audio element that plays the voice recording\n   */\n  @Input() audioElement?: HTMLAudioElement;\n  /**\n   * The waveform data to visualize\n   */\n  @Input() waveFormData: number[] = [];\n  /**\n   * The duration of the voice recording in seconds\n   */\n  @Input() duration?: number;\n  resampledWaveFormData: number[] = [];\n  progress: number = 0;\n  isDragging = false;\n  private sampleSize: number = 40;\n  @ViewChild('container', { static: true })\n  private container?: ElementRef<HTMLElement>;\n  private isViewInited = false;\n\n  constructor(private ngZone: NgZone, private cdRef: ChangeDetectorRef) {}\n\n  ngOnInit(): void {\n    this.containerSizeChanged();\n    if (this.container?.nativeElement) {\n      this.ngZone.runOutsideAngular(() => {\n        new ResizeObserver(() => {\n          this.containerSizeChanged();\n        }).observe(this.container!.nativeElement);\n      });\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes.waveFormData) {\n      this.resampledWaveFormData = resampleWaveForm(\n        this.waveFormData,\n        this.sampleSize\n      );\n    }\n    if (changes.audioElement) {\n      this.ngZone.runOutsideAngular(() => {\n        this.audioElement?.addEventListener('timeupdate', () => {\n          const progress =\n            (this.audioElement?.currentTime || 0) / (this.duration || 0) || 0;\n          if (Math.abs(progress - this.progress) >= 0.02) {\n            this.ngZone.run(() => {\n              this.progress = progress;\n              this.cdRef.detectChanges();\n            });\n          }\n        });\n      });\n    }\n  }\n\n  ngAfterViewInit(): void {\n    this.isViewInited = true;\n  }\n\n  seek(event: MouseEvent) {\n    const containerWidth =\n      this.container?.nativeElement?.getBoundingClientRect().width || 0;\n    const containerStart =\n      this.container?.nativeElement?.getBoundingClientRect()?.x || 0;\n    // eslint-disable-next-line @typescript-eslint/no-unsafe-member-access\n    const progress = (event.x - containerStart) / containerWidth;\n\n    if (!isNaN(progress) && this.audioElement) {\n      const duration = this.duration || 0;\n      const time = duration * progress;\n      this.audioElement.currentTime = time;\n    }\n  }\n\n  trackByIndex(index: number) {\n    return index;\n  }\n\n  private containerSizeChanged() {\n    if (!this.container?.nativeElement) {\n      return;\n    }\n    const containerWidth = this.container.nativeElement.clientWidth;\n    if (containerWidth === 0) {\n      return;\n    }\n    const barWidth = +getComputedStyle(this.container.nativeElement)\n      .getPropertyValue('--str-chat__voice-recording-amplitude-bar-width')\n      .replace('px', '');\n    const barGap = +getComputedStyle(this.container.nativeElement)\n      .getPropertyValue('--str-chat__voice-recording-amplitude-bar-gap-width')\n      .replace('px', '');\n    if (!isNaN(barWidth) && !isNaN(barGap)) {\n      const sampleSize = Math.floor(containerWidth / (barWidth + barGap));\n      if (\n        sampleSize !== this.sampleSize &&\n        !isNaN(sampleSize) &&\n        sampleSize !== Infinity\n      ) {\n        this.ngZone.run(() => {\n          this.sampleSize = sampleSize;\n          this.resampledWaveFormData = resampleWaveForm(\n            this.waveFormData,\n            this.sampleSize\n          );\n          if (this.isViewInited) {\n            this.cdRef.detectChanges();\n          }\n        });\n      }\n    }\n  }\n}\n","<!--eslint-disable @angular-eslint/template/click-events-have-key-events-->\n<div\n  #container\n  class=\"str-chat__wave-progress-bar__track\"\n  data-testid=\"wave-progress-bar-track\"\n  role=\"progressbar\"\n  (mousedown)=\"isDragging = true\"\n  (mouseup)=\"isDragging = false\"\n  (mouseleave)=\"isDragging = false\"\n  (mousemove)=\"isDragging ? seek($event) : null\"\n  (click)=\"seek($event)\"\n>\n  <!--eslint-enable @angular-eslint/template/click-events-have-key-events-->\n  <div\n    *ngFor=\"\n      let dataPoint of resampledWaveFormData;\n      let i = index;\n      trackBy: trackByIndex\n    \"\n    class=\"str-chat__wave-progress-bar__amplitude-bar\"\n    [class.str-chat__wave-progress-bar__amplitude-bar--active]=\"\n      progress > i / resampledWaveFormData.length\n    \"\n    [style.--str-chat__wave-progress-bar__amplitude-bar-height]=\"\n      dataPoint ? dataPoint * 100 + '%' : '0%'\n    \"\n  ></div>\n  <div\n    class=\"str-chat__wave-progress-bar__progress-indicator\"\n    data-testid=\"wave-progress-bar-progress-indicator\"\n    [ngStyle]=\"{ 'inset-inline-start': progress * 100 + '%' }\"\n  ></div>\n</div>\n"]}
|