speechrecorderng 3.4.5 → 3.6.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +1 -1
- package/esm2020/lib/audio/audio_player.mjs +13 -26
- package/esm2020/lib/audio/capture/capture.mjs +244 -207
- package/esm2020/lib/audio/context.mjs +64 -2
- package/esm2020/lib/audio/net_audio_buffer.mjs +5 -9
- package/esm2020/lib/audio/playback/player.mjs +137 -96
- package/esm2020/lib/audio/ui/audio_display_control.mjs +1 -1
- package/esm2020/lib/speechrecorder/project/project.mjs +1 -1
- package/esm2020/lib/speechrecorder/recordings/basic_recording.service.mjs +9 -8
- package/esm2020/lib/speechrecorder/recordings/recordings.service.mjs +38 -37
- package/esm2020/lib/speechrecorder/script/script.mjs +1 -1
- package/esm2020/lib/speechrecorder/session/audiorecorder.mjs +28 -77
- package/esm2020/lib/speechrecorder/session/basicrecorder.mjs +9 -2
- package/esm2020/lib/speechrecorder/session/controlpanel.mjs +14 -7
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +3 -3
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs +1 -1
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +44 -47
- package/esm2020/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +11 -11
- package/esm2020/lib/speechrecorder/session/sessionmanager.mjs +137 -88
- package/esm2020/lib/speechrecorderng.component.mjs +9 -15
- package/esm2020/lib/spr.module.version.mjs +2 -2
- package/fesm2015/speechrecorderng.mjs +767 -635
- package/fesm2015/speechrecorderng.mjs.map +1 -1
- package/fesm2020/speechrecorderng.mjs +764 -635
- package/fesm2020/speechrecorderng.mjs.map +1 -1
- package/lib/audio/audio_player.d.ts +0 -1
- package/lib/audio/capture/capture.d.ts +5 -4
- package/lib/audio/context.d.ts +2 -0
- package/lib/audio/net_audio_buffer.d.ts +2 -4
- package/lib/audio/playback/player.d.ts +3 -2
- package/lib/speechrecorder/project/project.d.ts +1 -0
- package/lib/speechrecorder/recordings/basic_recording.service.d.ts +2 -2
- package/lib/speechrecorder/recordings/recordings.service.d.ts +8 -8
- package/lib/speechrecorder/script/script.d.ts +1 -0
- package/lib/speechrecorder/session/basicrecorder.d.ts +3 -0
- package/lib/speechrecorder/session/controlpanel.d.ts +2 -0
- package/lib/speechrecorder/session/recordingfile/recording-file-meta.component.d.ts +1 -0
- package/lib/speechrecorder/session/recordingfile/recordingfile-service.d.ts +2 -2
- package/lib/speechrecorder/session/sessionmanager.d.ts +8 -1
- package/lib/spr.module.version.d.ts +1 -1
- package/package.json +1 -1
|
@@ -3,6 +3,7 @@ import { AudioStorageType, Platform as CfgPlatform } from "../../speechrecorder/
|
|
|
3
3
|
import { ArrayAudioBuffer } from "../array_audio_buffer";
|
|
4
4
|
import { UUID } from "../../utils/utils";
|
|
5
5
|
import { IndexedDbAudioBuffer } from "../inddb_audio_buffer";
|
|
6
|
+
import { AudioContextProvider } from "../context";
|
|
6
7
|
export const CHROME_ACTIVATE_ECHO_CANCELLATION_WITH_AGC = false;
|
|
7
8
|
const DEBUG_TRACE_LEVEL = 0;
|
|
8
9
|
// Dirty way to load this module
|
|
@@ -82,8 +83,10 @@ const awpStr = "class AudioCaptureInterceptorProcessor extends AudioWorkletProce
|
|
|
82
83
|
"\n" +
|
|
83
84
|
"registerProcessor('capture-interceptor',AudioCaptureInterceptorProcessor);\n";
|
|
84
85
|
export class AudioCapture {
|
|
85
|
-
|
|
86
|
+
//private context:AudioContext|null=null;
|
|
87
|
+
constructor() {
|
|
86
88
|
this._maxAutoNetMemStoreSamples = AudioCapture.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;
|
|
89
|
+
this.context = null;
|
|
87
90
|
this._recUUID = null;
|
|
88
91
|
this.agcStatus = null;
|
|
89
92
|
this.bufferingNode = null;
|
|
@@ -98,13 +101,7 @@ export class AudioCapture {
|
|
|
98
101
|
this.persisted = true;
|
|
99
102
|
this.persistError = null;
|
|
100
103
|
this.inddbAudioBuffer = null;
|
|
101
|
-
this.context = context;
|
|
102
104
|
this.n = navigator;
|
|
103
|
-
this.context.addEventListener('statechange', () => {
|
|
104
|
-
if (this.context.state !== 'running') {
|
|
105
|
-
this.close();
|
|
106
|
-
}
|
|
107
|
-
});
|
|
108
105
|
}
|
|
109
106
|
get maxAutoNetMemStoreSamples() {
|
|
110
107
|
return this._maxAutoNetMemStoreSamples;
|
|
@@ -133,6 +130,19 @@ export class AudioCapture {
|
|
|
133
130
|
get opened() {
|
|
134
131
|
return this._opened;
|
|
135
132
|
}
|
|
133
|
+
_audioContext() {
|
|
134
|
+
if (!this.context) {
|
|
135
|
+
this.context = AudioContextProvider.audioContextInstance();
|
|
136
|
+
if (this.context) {
|
|
137
|
+
this.context.addEventListener('statechange', () => {
|
|
138
|
+
if (this.context && this.context.state !== 'running') {
|
|
139
|
+
this.close();
|
|
140
|
+
}
|
|
141
|
+
});
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
return this.context;
|
|
145
|
+
}
|
|
136
146
|
initData() {
|
|
137
147
|
if (!this._recUUID) {
|
|
138
148
|
this._recUUID = UUID.generate();
|
|
@@ -247,119 +257,137 @@ export class AudioCapture {
|
|
|
247
257
|
}
|
|
248
258
|
}
|
|
249
259
|
addCaptureInterceptor() {
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
this.listener
|
|
259
|
-
|
|
260
|
-
|
|
261
|
-
|
|
262
|
-
|
|
263
|
-
|
|
264
|
-
|
|
265
|
-
|
|
266
|
-
|
|
267
|
-
|
|
268
|
-
|
|
269
|
-
|
|
270
|
-
|
|
271
|
-
let chunk = new Array(chs);
|
|
272
|
-
const samples = this.framesRecorded * chs;
|
|
273
|
-
if ((AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this.audioStorageType || AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this.audioStorageType) && this.data && samples > this._maxAutoNetMemStoreSamples) {
|
|
274
|
-
this.data = null;
|
|
275
|
-
}
|
|
276
|
-
//console.debug("Data initialized: "+(this.data!=null));
|
|
277
|
-
for (let ch = 0; ch < chs; ch++) {
|
|
278
|
-
//console.debug("Data ch initialized: "+(this.data !=null && this.data[ch] !=null));
|
|
279
|
-
if (ch < this.channelCount) {
|
|
280
|
-
if (dt.data[ch]) {
|
|
281
|
-
let fa = new Float32Array(dt.data[ch]);
|
|
282
|
-
if (this.data && this.data[ch]) {
|
|
283
|
-
this.data[ch].push(fa);
|
|
284
|
-
}
|
|
285
|
-
chunk[ch] = fa;
|
|
286
|
-
// Use samples of channel 0 to count frames (samples)
|
|
287
|
-
if (ch == 0) {
|
|
288
|
-
this.framesRecorded += fa.length;
|
|
289
|
-
}
|
|
290
|
-
}
|
|
260
|
+
if (this.context) {
|
|
261
|
+
const awn = new AudioWorkletNode(this.context, 'capture-interceptor');
|
|
262
|
+
awn.onprocessorerror = (ev) => {
|
|
263
|
+
let msg = 'Unknwon error';
|
|
264
|
+
if (ev instanceof ErrorEvent) {
|
|
265
|
+
msg = ev.message;
|
|
266
|
+
}
|
|
267
|
+
console.error("Capture audio worklet error: " + msg);
|
|
268
|
+
if (this.listener) {
|
|
269
|
+
this.listener.error(msg);
|
|
270
|
+
}
|
|
271
|
+
};
|
|
272
|
+
let awnPt = awn.port;
|
|
273
|
+
if (awnPt) {
|
|
274
|
+
awnPt.onmessage = (ev) => {
|
|
275
|
+
if (this.capturing) {
|
|
276
|
+
let dt = ev.data;
|
|
277
|
+
let chs = dt.chs;
|
|
278
|
+
let adaLen = dt.data.length;
|
|
279
|
+
if (DEBUG_TRACE_LEVEL > 8) {
|
|
280
|
+
console.debug('Received data from worklet: ' + chs + ' ' + dt.len + ' Data chs: ' + adaLen);
|
|
291
281
|
}
|
|
292
|
-
|
|
293
|
-
|
|
294
|
-
|
|
295
|
-
this.
|
|
296
|
-
// // Random test error:
|
|
297
|
-
// if(Math.random()>0.98) {
|
|
298
|
-
// throw new Error('Test');
|
|
299
|
-
// }
|
|
282
|
+
let chunk = new Array(chs);
|
|
283
|
+
const samples = this.framesRecorded * chs;
|
|
284
|
+
if ((AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this.audioStorageType || AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this.audioStorageType) && this.data && samples > this._maxAutoNetMemStoreSamples) {
|
|
285
|
+
this.data = null;
|
|
300
286
|
}
|
|
301
|
-
|
|
302
|
-
|
|
303
|
-
|
|
304
|
-
|
|
305
|
-
|
|
306
|
-
|
|
287
|
+
//console.debug("Data initialized: "+(this.data!=null));
|
|
288
|
+
for (let ch = 0; ch < chs; ch++) {
|
|
289
|
+
//console.debug("Data ch initialized: "+(this.data !=null && this.data[ch] !=null));
|
|
290
|
+
if (ch < this.channelCount) {
|
|
291
|
+
if (dt.data[ch]) {
|
|
292
|
+
let fa = new Float32Array(dt.data[ch]);
|
|
293
|
+
if (this.data && this.data[ch]) {
|
|
294
|
+
this.data[ch].push(fa);
|
|
295
|
+
}
|
|
296
|
+
chunk[ch] = fa;
|
|
297
|
+
// Use samples of channel 0 to count frames (samples)
|
|
298
|
+
if (ch == 0) {
|
|
299
|
+
this.framesRecorded += fa.length;
|
|
300
|
+
}
|
|
301
|
+
}
|
|
307
302
|
}
|
|
308
|
-
|
|
303
|
+
}
|
|
304
|
+
if (this.audioOutStream) {
|
|
309
305
|
try {
|
|
310
|
-
this.
|
|
306
|
+
this.audioOutStream.write(chunk);
|
|
307
|
+
// // Random test error:
|
|
308
|
+
// if(Math.random()>0.98) {
|
|
309
|
+
// throw new Error('Test');
|
|
310
|
+
// }
|
|
311
311
|
}
|
|
312
|
-
catch (
|
|
313
|
-
|
|
314
|
-
|
|
315
|
-
finally {
|
|
316
|
-
if (this.listener) {
|
|
317
|
-
let errExpl = '';
|
|
318
|
-
if (err instanceof DOMException) {
|
|
319
|
-
errExpl = ': ' + err.name + ': ' + err.message;
|
|
320
|
-
}
|
|
321
|
-
this.listener.error("Could not handle recorded audio data" + errExpl, "Please try to record again.");
|
|
312
|
+
catch (err) {
|
|
313
|
+
if (err instanceof Error) {
|
|
314
|
+
this.persistError = err;
|
|
322
315
|
}
|
|
323
316
|
else {
|
|
324
|
-
this.
|
|
317
|
+
this.persistError = new Error('Error handling recorded audio data');
|
|
318
|
+
}
|
|
319
|
+
console.error("Capture error: " + err);
|
|
320
|
+
try {
|
|
321
|
+
this.stop();
|
|
322
|
+
}
|
|
323
|
+
catch (err2) {
|
|
324
|
+
console.error("Capture next error (ignored): " + err2);
|
|
325
|
+
}
|
|
326
|
+
finally {
|
|
327
|
+
if (this.listener) {
|
|
328
|
+
let errExpl = '';
|
|
329
|
+
if (err instanceof DOMException) {
|
|
330
|
+
errExpl = ': ' + err.name + ': ' + err.message;
|
|
331
|
+
}
|
|
332
|
+
this.listener.error("Could not handle recorded audio data" + errExpl, "Please try to record again.");
|
|
333
|
+
}
|
|
334
|
+
else {
|
|
335
|
+
this.close();
|
|
336
|
+
}
|
|
325
337
|
}
|
|
326
338
|
}
|
|
327
339
|
}
|
|
340
|
+
if (AudioStorageType.DB_CHUNKED === this._audioStorageType && this._persistentAudioStorageTarget) {
|
|
341
|
+
this.store();
|
|
342
|
+
}
|
|
328
343
|
}
|
|
329
|
-
|
|
330
|
-
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
338
|
-
|
|
339
|
-
|
|
340
|
-
|
|
341
|
-
|
|
342
|
-
this._opened = true;
|
|
343
|
-
if (this.listener) {
|
|
344
|
-
this.listener.opened();
|
|
344
|
+
};
|
|
345
|
+
}
|
|
346
|
+
// Tried to fix that Safari does not record the second channel
|
|
347
|
+
// Does not help
|
|
348
|
+
//awn.channelCount=this.channelCount;
|
|
349
|
+
//awn.channelCountMode='explicit';
|
|
350
|
+
//console.debug('Channel count explicitly set to '+this.channelCount);
|
|
351
|
+
this.bufferingNode = awn;
|
|
352
|
+
//this.bufferingNode.channelCount=this.channelCount;
|
|
353
|
+
this._opened = true;
|
|
354
|
+
if (this.listener) {
|
|
355
|
+
this.listener.opened();
|
|
356
|
+
}
|
|
345
357
|
}
|
|
346
358
|
}
|
|
347
|
-
open(channelCount, selDeviceId, autoGainControlConfigs) {
|
|
359
|
+
open(channelCount, selDeviceId, autoGainControlConfigs, allowEchoCancellation) {
|
|
348
360
|
//console.debug("Capture open: ctx state: "+this.context.state);
|
|
349
|
-
|
|
361
|
+
this.context = this._audioContext();
|
|
362
|
+
if (!this.context) {
|
|
363
|
+
throw new Error("Could not get audio context!");
|
|
364
|
+
}
|
|
365
|
+
if (this.context.state === 'suspended') {
|
|
350
366
|
//console.debug("Capture open: Resume context");
|
|
351
367
|
this.context.resume().then(() => {
|
|
352
368
|
//console.debug("Capture open (ctx resumed): ctx state: "+this.context.state);
|
|
353
|
-
this._open(channelCount, selDeviceId, autoGainControlConfigs);
|
|
369
|
+
this._open(channelCount, selDeviceId, autoGainControlConfigs, allowEchoCancellation);
|
|
370
|
+
}).catch((err) => {
|
|
371
|
+
console.error(err.message);
|
|
372
|
+
throw err;
|
|
354
373
|
});
|
|
355
374
|
}
|
|
375
|
+
else if (this.context.state === 'closed') {
|
|
376
|
+
const msg = 'Error on start capture: The audio context is already closed.';
|
|
377
|
+
console.error(msg);
|
|
378
|
+
throw new Error(msg);
|
|
379
|
+
}
|
|
356
380
|
else {
|
|
357
|
-
this._open(channelCount, selDeviceId, autoGainControlConfigs);
|
|
381
|
+
this._open(channelCount, selDeviceId, autoGainControlConfigs, allowEchoCancellation);
|
|
358
382
|
}
|
|
359
383
|
}
|
|
360
|
-
_open(channelCount, selDeviceId, autoGainControlConfigs) {
|
|
384
|
+
_open(channelCount, selDeviceId, autoGainControlConfigs, allowEchoCancellation) {
|
|
361
385
|
this.channelCount = channelCount;
|
|
362
386
|
this.framesRecorded = 0;
|
|
387
|
+
this.context = this._audioContext();
|
|
388
|
+
if (!this.context) {
|
|
389
|
+
throw new Error("Could not get audio context!");
|
|
390
|
+
}
|
|
363
391
|
//var msc = new AudioStreamConstr();
|
|
364
392
|
// var msc={};
|
|
365
393
|
//msc.video = false;
|
|
@@ -467,7 +495,7 @@ export class AudioCapture {
|
|
|
467
495
|
audio: {
|
|
468
496
|
deviceId: selDeviceId,
|
|
469
497
|
channelCount: channelCount,
|
|
470
|
-
|
|
498
|
+
echoCancellation: allowEchoCancellation ? undefined : false
|
|
471
499
|
},
|
|
472
500
|
video: false,
|
|
473
501
|
};
|
|
@@ -478,106 +506,109 @@ export class AudioCapture {
|
|
|
478
506
|
console.debug("Audio capture, AGC: " + this.agcStatus);
|
|
479
507
|
let ump = navigator.mediaDevices.getUserMedia(msc);
|
|
480
508
|
ump.then((s) => {
|
|
481
|
-
this.
|
|
482
|
-
|
|
483
|
-
|
|
484
|
-
let
|
|
485
|
-
|
|
486
|
-
|
|
487
|
-
|
|
488
|
-
|
|
489
|
-
|
|
490
|
-
|
|
491
|
-
|
|
492
|
-
|
|
509
|
+
if (this.context) {
|
|
510
|
+
this.stream = s;
|
|
511
|
+
let aTracks = s.getAudioTracks();
|
|
512
|
+
for (let i = 0; i < aTracks.length; i++) {
|
|
513
|
+
let aTrack = aTracks[i];
|
|
514
|
+
console.info("Track audio info: id: " + aTrack.id + " kind: " + aTrack.kind + " label: \"" + aTrack.label + "\"");
|
|
515
|
+
let mtrSts = aTrack.getSettings();
|
|
516
|
+
// Typescript lib.dom.ts MediaTrackSettings.channelCount is missing
|
|
517
|
+
// https://github.com/mdn/browser-compat-data/blob/5493d8f937e05b2ddbd41b99f5bdfad4a1f2ed85/api/MediaTrackSettings.json
|
|
518
|
+
//@ts-ignore
|
|
519
|
+
console.info("Track audio settings: Ch cnt: " + mtrSts.channelCount + ", AGC: " + mtrSts.autoGainControl + ", Echo cancell.: " + mtrSts.echoCancellation);
|
|
520
|
+
if (mtrSts.autoGainControl) {
|
|
521
|
+
this.agcStatus = mtrSts.autoGainControl;
|
|
522
|
+
}
|
|
523
|
+
console.debug("Echo cancellation: " + mtrSts.echoCancellation);
|
|
493
524
|
}
|
|
494
|
-
|
|
495
|
-
|
|
496
|
-
|
|
497
|
-
|
|
498
|
-
console.info("Track video info: id: " + vTrack.id + " kind: " + vTrack.kind + " label: " + vTrack.label);
|
|
499
|
-
}
|
|
500
|
-
this.mediaStream = this.context.createMediaStreamSource(s);
|
|
501
|
-
// stream channel count ( is always 2 !)
|
|
502
|
-
let streamChannelCount = this.mediaStream.channelCount;
|
|
503
|
-
console.info("Stream channel count: " + streamChannelCount);
|
|
504
|
-
// is not set!!
|
|
505
|
-
//this.currentSampleRate = this.mediaStream.sampleRate;
|
|
506
|
-
this.currentSampleRate = this.context.sampleRate;
|
|
507
|
-
console.info("Source audio node: channels: " + streamChannelCount + " samplerate: " + this.currentSampleRate);
|
|
508
|
-
if (this.audioOutStream) {
|
|
509
|
-
this.audioOutStream.setFormat(this.channelCount, this.currentSampleRate);
|
|
510
|
-
}
|
|
511
|
-
// W3C -> new name is createScriptProcessor
|
|
512
|
-
//
|
|
513
|
-
// Again deprecated, but AudioWorker not yet implemented in stable releases (June 2016)
|
|
514
|
-
// AudioWorker is now AudioWorkletProcessor ... (May 2017)
|
|
515
|
-
// Update 12-2020:
|
|
516
|
-
// The ScriptProcessorNode Interface - DEPRECATED
|
|
517
|
-
// Update 06-2021
|
|
518
|
-
// AudioWorkletProcessor is here to stay. Web Audio API has now Recommendation status !
|
|
519
|
-
if (this.context.audioWorklet) {
|
|
520
|
-
//const workletFileName = ('file-loader!./interceptor_worklet.js');
|
|
521
|
-
//const workletFileName = 'http://localhost:4200/assets/interceptor_worklet.js';
|
|
522
|
-
//console.log(awpStr);
|
|
523
|
-
if (AudioCapture.captureInterceptorModuleRegistered) {
|
|
524
|
-
// Required capture interceptor module already registered
|
|
525
|
-
this.addCaptureInterceptor();
|
|
525
|
+
let vTracks = s.getVideoTracks();
|
|
526
|
+
for (let i = 0; i < vTracks.length; i++) {
|
|
527
|
+
let vTrack = vTracks[i];
|
|
528
|
+
console.info("Track video info: id: " + vTrack.id + " kind: " + vTrack.kind + " label: " + vTrack.label);
|
|
526
529
|
}
|
|
527
|
-
|
|
528
|
-
|
|
529
|
-
|
|
530
|
-
|
|
531
|
-
|
|
532
|
-
|
|
530
|
+
this.mediaStream = this.context.createMediaStreamSource(s);
|
|
531
|
+
// stream channel count ( is always 2 !)
|
|
532
|
+
let streamChannelCount = this.mediaStream.channelCount;
|
|
533
|
+
console.info("Stream channel count: " + streamChannelCount);
|
|
534
|
+
// is not set!!
|
|
535
|
+
//this.currentSampleRate = this.mediaStream.sampleRate;
|
|
536
|
+
this.currentSampleRate = this.context.sampleRate;
|
|
537
|
+
console.info("Source audio node: channels: " + streamChannelCount + " samplerate: " + this.currentSampleRate);
|
|
538
|
+
if (this.audioOutStream) {
|
|
539
|
+
this.audioOutStream.setFormat(this.channelCount, this.currentSampleRate);
|
|
540
|
+
}
|
|
541
|
+
// W3C -> new name is createScriptProcessor
|
|
542
|
+
//
|
|
543
|
+
// Again deprecated, but AudioWorker not yet implemented in stable releases (June 2016)
|
|
544
|
+
// AudioWorker is now AudioWorkletProcessor ... (May 2017)
|
|
545
|
+
// Update 12-2020:
|
|
546
|
+
// The ScriptProcessorNode Interface - DEPRECATED
|
|
547
|
+
// Update 06-2021
|
|
548
|
+
// AudioWorkletProcessor is here to stay. Web Audio API has now Recommendation status !
|
|
549
|
+
if (this.context.audioWorklet) {
|
|
550
|
+
//const workletFileName = ('file-loader!./interceptor_worklet.js');
|
|
551
|
+
//const workletFileName = 'http://localhost:4200/assets/interceptor_worklet.js';
|
|
552
|
+
//console.log(awpStr);
|
|
553
|
+
if (AudioCapture.captureInterceptorModuleRegistered) {
|
|
554
|
+
// Required capture interceptor module already registered
|
|
533
555
|
this.addCaptureInterceptor();
|
|
534
|
-
}
|
|
535
|
-
|
|
536
|
-
|
|
556
|
+
}
|
|
557
|
+
else {
|
|
558
|
+
// Register capture interceptor module
|
|
559
|
+
let audioWorkletModuleBlob = new Blob([awpStr], { type: 'text/javascript' });
|
|
560
|
+
let audioWorkletModuleBlobUrl = window.URL.createObjectURL(audioWorkletModuleBlob);
|
|
561
|
+
this.context.audioWorklet.addModule(audioWorkletModuleBlobUrl).then(() => {
|
|
562
|
+
AudioCapture.captureInterceptorModuleRegistered = true;
|
|
563
|
+
this.addCaptureInterceptor();
|
|
564
|
+
}).catch((error) => {
|
|
565
|
+
console.log('Could not add module ' + error);
|
|
566
|
+
});
|
|
567
|
+
}
|
|
537
568
|
}
|
|
538
|
-
|
|
539
|
-
|
|
540
|
-
|
|
541
|
-
|
|
542
|
-
|
|
543
|
-
|
|
544
|
-
|
|
545
|
-
|
|
546
|
-
|
|
547
|
-
|
|
548
|
-
|
|
549
|
-
|
|
550
|
-
|
|
551
|
-
|
|
552
|
-
|
|
553
|
-
|
|
554
|
-
|
|
555
|
-
|
|
556
|
-
|
|
569
|
+
else if (this.context.createScriptProcessor) {
|
|
570
|
+
// The ScriptProcessorNode Interface - DEPRECATED Only as fallback
|
|
571
|
+
// TODO should we use streamChannelCount or channelCount here ?
|
|
572
|
+
let scriptProcessorNode = this.context.createScriptProcessor(AudioCapture.BUFFER_SIZE, streamChannelCount, streamChannelCount);
|
|
573
|
+
this.bufferingNode = scriptProcessorNode;
|
|
574
|
+
let c = 0;
|
|
575
|
+
if (scriptProcessorNode.onaudioprocess) {
|
|
576
|
+
scriptProcessorNode.onaudioprocess = (e) => {
|
|
577
|
+
if (this.capturing) {
|
|
578
|
+
let inBuffer = e.inputBuffer;
|
|
579
|
+
// only process requested count of channels
|
|
580
|
+
let currentBuffers = new Array(channelCount);
|
|
581
|
+
for (let ch = 0; ch < channelCount; ch++) {
|
|
582
|
+
let chSamples = inBuffer.getChannelData(ch);
|
|
583
|
+
let chSamplesCopy = chSamples.slice(0);
|
|
584
|
+
currentBuffers[ch] = chSamplesCopy.slice(0);
|
|
585
|
+
if (this.data) {
|
|
586
|
+
this.data[ch].push(chSamplesCopy);
|
|
587
|
+
}
|
|
588
|
+
if (DEBUG_TRACE_LEVEL > 8) {
|
|
589
|
+
console.debug("Process " + chSamplesCopy.length + " samples.");
|
|
590
|
+
}
|
|
591
|
+
this.framesRecorded += chSamplesCopy.length;
|
|
557
592
|
}
|
|
558
|
-
|
|
559
|
-
|
|
593
|
+
c++;
|
|
594
|
+
if (this.audioOutStream) {
|
|
595
|
+
this.audioOutStream.write(currentBuffers);
|
|
560
596
|
}
|
|
561
|
-
this.framesRecorded += chSamplesCopy.length;
|
|
562
|
-
}
|
|
563
|
-
c++;
|
|
564
|
-
if (this.audioOutStream) {
|
|
565
|
-
this.audioOutStream.write(currentBuffers);
|
|
566
597
|
}
|
|
598
|
+
};
|
|
599
|
+
this._opened = true;
|
|
600
|
+
if (this.listener) {
|
|
601
|
+
this.listener.opened();
|
|
567
602
|
}
|
|
568
|
-
}
|
|
569
|
-
|
|
570
|
-
|
|
571
|
-
this.listener.opened();
|
|
603
|
+
}
|
|
604
|
+
else {
|
|
605
|
+
this.listener.error('Browser does not support audio processing (ScriptProcessor.onaudioprocess method not found)!');
|
|
572
606
|
}
|
|
573
607
|
}
|
|
574
608
|
else {
|
|
575
|
-
this.listener.error('Browser does not support audio processing (
|
|
609
|
+
this.listener.error('Browser does not support audio processing (neither AudioWorkletProcessor nor ScriptProcessor)!');
|
|
576
610
|
}
|
|
577
611
|
}
|
|
578
|
-
else {
|
|
579
|
-
this.listener.error('Browser does not support audio processing (neither AudioWorkletProcessor nor ScriptProcessor)!');
|
|
580
|
-
}
|
|
581
612
|
}, (e) => {
|
|
582
613
|
console.error(e + " Error name: " + e.name);
|
|
583
614
|
if (this.listener) {
|
|
@@ -598,36 +629,42 @@ export class AudioCapture {
|
|
|
598
629
|
});
|
|
599
630
|
}
|
|
600
631
|
_start() {
|
|
601
|
-
this.
|
|
602
|
-
|
|
603
|
-
this.audioOutStream
|
|
604
|
-
|
|
605
|
-
|
|
606
|
-
|
|
607
|
-
|
|
608
|
-
|
|
609
|
-
|
|
610
|
-
|
|
611
|
-
this.listener
|
|
632
|
+
if (this.context) {
|
|
633
|
+
this.initData();
|
|
634
|
+
if (this.audioOutStream) {
|
|
635
|
+
this.audioOutStream.nextStream();
|
|
636
|
+
}
|
|
637
|
+
this.capturing = true;
|
|
638
|
+
if (this.bufferingNode) {
|
|
639
|
+
this.mediaStream.connect(this.bufferingNode);
|
|
640
|
+
this.bufferingNode.connect(this.context.destination);
|
|
641
|
+
}
|
|
642
|
+
if (this.listener) {
|
|
643
|
+
this.listener.started();
|
|
644
|
+
}
|
|
612
645
|
}
|
|
613
646
|
}
|
|
614
647
|
start() {
|
|
615
|
-
|
|
616
|
-
|
|
617
|
-
|
|
618
|
-
}
|
|
619
|
-
else {
|
|
620
|
-
console.debug("Capture start: audio context not running, state: " + aSt + ", resuming...");
|
|
621
|
-
this.context.resume().then(() => {
|
|
622
|
-
console.debug("Capture start: audio context resumed, starting...");
|
|
648
|
+
if (this.context) {
|
|
649
|
+
const aSt = this.context.state;
|
|
650
|
+
if (aSt === 'running') {
|
|
623
651
|
this._start();
|
|
624
|
-
}
|
|
652
|
+
}
|
|
653
|
+
else {
|
|
654
|
+
console.debug("Capture start: audio context not running, state: " + aSt + ", resuming...");
|
|
655
|
+
this.context.resume().then(() => {
|
|
656
|
+
console.debug("Capture start: audio context resumed, starting...");
|
|
657
|
+
this._start();
|
|
658
|
+
});
|
|
659
|
+
}
|
|
625
660
|
}
|
|
626
661
|
}
|
|
627
662
|
stop() {
|
|
628
663
|
if (this.disconnectStreams && this.bufferingNode) {
|
|
629
664
|
this.mediaStream.disconnect(this.bufferingNode);
|
|
630
|
-
|
|
665
|
+
if (this.context) {
|
|
666
|
+
this.bufferingNode.disconnect(this.context.destination);
|
|
667
|
+
}
|
|
631
668
|
}
|
|
632
669
|
try {
|
|
633
670
|
if (this.audioOutStream) {
|
|
@@ -750,7 +787,7 @@ export class AudioCapture {
|
|
|
750
787
|
}
|
|
751
788
|
audioBuffer() {
|
|
752
789
|
let ab = null;
|
|
753
|
-
if (this.data) {
|
|
790
|
+
if (this.context && this.data) {
|
|
754
791
|
let frameLen = 0;
|
|
755
792
|
let ch0Data = this.data[0];
|
|
756
793
|
for (let ch0Chk of ch0Data) {
|
|
@@ -815,4 +852,4 @@ export class AudioCapture {
|
|
|
815
852
|
AudioCapture.BUFFER_SIZE = 8192;
|
|
816
853
|
AudioCapture.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES = 2880000 * 5; // Default 5 minute at 48kHz
|
|
817
854
|
AudioCapture.captureInterceptorModuleRegistered = false;
|
|
818
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"capture.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/audio/capture/capture.ts"],"names":[],"mappings":"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","sourcesContent":["import {SequenceAudioFloat32OutStream} from \"../io/stream\";\r\nimport {Browser, Platform, UserAgentBuilder} from \"../../utils/ua-parser\";\r\n\r\nimport {AudioStorageType, AutoGainControlConfig, Platform as CfgPlatform} from \"../../speechrecorder/project/project\";\r\nimport {ArrayAudioBuffer} from \"../array_audio_buffer\";\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {IndexedDbAudioBuffer, PersistentAudioStorageTarget} from \"../inddb_audio_buffer\";\r\n\r\n\r\nexport const CHROME_ACTIVATE_ECHO_CANCELLATION_WITH_AGC=false;\r\n\r\nconst DEBUG_TRACE_LEVEL=0;\r\n\r\n// Dirty way to load this module\r\n// Copy content of interceptor_worklet.js to this string\r\nconst awpStr=\"class AudioCaptureInterceptorProcessor extends AudioWorkletProcessor{\\n\" +\r\n    \"\\n\" +\r\n    \"    BUFFER_QUANTUMS=64;\\n\" +\r\n    \"    QUANTUM_FRAME_LEN=128;\\n\" +\r\n    \"    BUFFER_FRAME_LEN=this.QUANTUM_FRAME_LEN*this.BUFFER_QUANTUMS;\\n\" +\r\n    \"    buffer=null;\\n\" +\r\n    \"    bufferPos=0;\\n\" +\r\n    \"    bufferPosBytes=0;\\n\" +\r\n    \"    constructor() {\\n\" +\r\n    \"        super();\\n\" +\r\n    \"\\n\" +\r\n    \"    }\\n\" +\r\n    \"\\n\" +\r\n    \" process(\\n\" +\r\n    \"      inputs,\\n\" +\r\n    \"      outputs,\\n\" +\r\n    \"      parameters\\n\" +\r\n    \"  ){\\n\" +\r\n    \"\\n\" +\r\n    \"     let inputsCnt=inputs.length;\\n\" +\r\n    \"     let channelCount=0;\\n\" +\r\n    \"     let inputLen=0;\\n\" +\r\n    \"     let inputLenBytes=0;\\n\" +\r\n    \"     if(inputsCnt>0) {\\n\" +\r\n    \"         let input0 = inputs[0];\\n\" +\r\n    \"         channelCount = input0.length;\\n\" +\r\n    \"         if (channelCount > 0) {\\n\" +\r\n    \"             let input0ch0=input0[0];\\n\" +\r\n    \"             inputLen=input0ch0.length;\\n\" +\r\n    \"             inputLenBytes=input0ch0.buffer.length;\\n\" +\r\n    \"         }\\n\" +\r\n    \"     }\\n\" +\r\n    \"     if (!this.buffer || this.buffer.length < channelCount) {\\n\" +\r\n    \"         this.buffer = new Array(channelCount);\\n\" +\r\n    \"         this.bufferPos = 0\\n\" +\r\n    \"         for (let bch = 0; bch < channelCount; bch++) {\\n\" +\r\n    \"             this.buffer[bch] = new Float32Array(this.BUFFER_FRAME_LEN);\\n\" +\r\n    \"             this.bufferPos = 0;\\n\" +\r\n    \"             this.bufferPosBytes=0;\\n\" +\r\n    \"         }\\n\" +\r\n    \"     }\\n\" +\r\n    \"     let bufAvail = this.BUFFER_FRAME_LEN - this.bufferPos;\\n\" +\r\n    \"     // check if buffer has to be transferred\\n\" +\r\n    \"     if (inputLen > bufAvail) {\\n\" +\r\n    \"         let ada=new Array(channelCount);\\n\" +\r\n    \"         for (let ch = 0; ch < channelCount; ch++) {\\n\" +\r\n    \"             ada[ch]=this.buffer[ch].buffer.slice(0);\\n\" +\r\n    \"         }\\n\" +\r\n    \"         this.port.postMessage({\\n\" +\r\n    \"             data: ada,\\n\" +\r\n    \"             chs: channelCount,\\n\" +\r\n    \"             len: this.bufferPos\\n\" +\r\n    \"         }, ada);\\n\" +\r\n    \"         // buffer transferred, reset\\n\" +\r\n    \"         this.bufferPos = 0;\\n\" +\r\n    \"         this.bufferPosBytes=0;\\n\" +\r\n    \"     }\\n\" +\r\n    \"\\n\" +\r\n    \"     for(let ii=0;ii<inputsCnt;ii++) {\\n\" +\r\n    \"         for (let ch = 0; ch < channelCount; ch++) {\\n\" +\r\n    \"             // Mute outputs\\n\" +\r\n    \"             //outputs[ii][ch].fill(0);\\n\" +\r\n    \"             let chSamples = inputs[ii][ch];\\n\" +\r\n    \"             this.buffer[ch].set(chSamples,this.bufferPos);\\n\" +\r\n    \"         }\\n\" +\r\n    \"         this.bufferPos+=inputLen;\\n\" +\r\n    \"         this.bufferPosBytes+=inputLenBytes;\\n\" +\r\n    \"     }\\n\" +\r\n    \"    \\n\" +\r\n    \"     return true;\\n\" +\r\n    \"  }\\n\" +\r\n    \"}\\n\" +\r\n    \"\\n\" +\r\n    \"registerProcessor('capture-interceptor',AudioCaptureInterceptorProcessor);\\n\";\r\n\r\n\r\n\r\n\r\nexport interface AudioCaptureListener {\r\n  opened(): void;\r\n\r\n  started(): void;\r\n\r\n  stopped(): void;\r\n\r\n  closed(): void;\r\n\r\n  error(msg?:string,advice?:string): void;\r\n}\r\n\r\nexport class AudioCapture {\r\n\r\n  get maxAutoNetMemStoreSamples(): number {\r\n    return this._maxAutoNetMemStoreSamples;\r\n  }\r\n\r\n  set maxAutoNetMemStoreSamples(value: number) {\r\n    this._maxAutoNetMemStoreSamples = value;\r\n  }\r\n  set recUUID(value: string|null) {\r\n    this._recUUID = value;\r\n  }\r\n  get recUUID(): string |null{\r\n    return this._recUUID;\r\n  }\r\n  get audioStorageType(): AudioStorageType {\r\n    return this._audioStorageType;\r\n  }\r\n\r\n  set audioStorageType(value: AudioStorageType) {\r\n    this._audioStorageType = value;\r\n  }\r\n  get persistentAudioStorageTarget(): PersistentAudioStorageTarget | null {\r\n    return this._persistentAudioStorageTarget;\r\n  }\r\n\r\n  set persistentAudioStorageTarget(value: PersistentAudioStorageTarget | null) {\r\n    this._persistentAudioStorageTarget = value;\r\n  }\r\n\r\n  get opened(): boolean {\r\n    return this._opened;\r\n  }\r\n\r\n  static BUFFER_SIZE: number = 8192;\r\n  private static readonly DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES:number=2880000*5; // Default 5 minute at 48kHz\r\n  private _maxAutoNetMemStoreSamples:number=AudioCapture.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;\r\n  private static captureInterceptorModuleRegistered=false;\r\n  context: AudioContext;\r\n  stream!: MediaStream;\r\n  channelCount!: number;\r\n  private _recUUID:string|null=null;\r\n  mediaStream: any;\r\n  agcStatus:boolean|null=null;\r\n  bufferingNode: AudioNode|null=null;\r\n  listener!: AudioCaptureListener;\r\n  data: Array<Array<Float32Array>>|null=null;\r\n  currentSampleRate!: number;\r\n  n: Navigator;\r\n  audioOutStream: SequenceAudioFloat32OutStream | null=null;\r\n  private disconnectStreams = true;\r\n  private _opened=false;\r\n  private capturing = false;\r\n\r\n  framesRecorded: number=0;\r\n\r\n  private _audioStorageType:AudioStorageType=AudioStorageType.MEM_ENTIRE;\r\n  private _persistentAudioStorageTarget:PersistentAudioStorageTarget|null=null;\r\n\r\n  private persisted=true;\r\n  private persistError:Error|null=null;\r\n  private inddbAudioBuffer:IndexedDbAudioBuffer|null=null;\r\n\r\n  constructor(context: AudioContext) {\r\n    this.context = context;\r\n    this.n = navigator;\r\n    this.context.addEventListener('statechange', () => {\r\n      if (this.context.state !== 'running') {\r\n        this.close();\r\n      }\r\n    });\r\n  }\r\n\r\n  private initData() {\r\n    if(!this._recUUID) {\r\n      this._recUUID = UUID.generate();\r\n    }\r\n    this.persistError=null;\r\n    //console.debug(\"Audio capture initialize storage for type: \"+this._audioStorageType);\r\n    if(AudioStorageType.DB_CHUNKED === this._audioStorageType && this._persistentAudioStorageTarget && this._recUUID) {\r\n      //console.debug(\"Create indexed db audio buffer.\");\r\n      this.inddbAudioBuffer = new IndexedDbAudioBuffer(this._persistentAudioStorageTarget, this.channelCount,this.currentSampleRate,AudioCapture.BUFFER_SIZE,0,this._recUUID)\r\n    }\r\n    if(!(AudioStorageType.NET_CHUNKED === this._audioStorageType)) {\r\n      // Initialize audio data array except for net audio buffer mode\r\n      this.data = new Array<Array<Float32Array>>();\r\n      for (let i = 0; i < this.channelCount; i++) {\r\n        this.data.push(new Array<Float32Array>());\r\n      }\r\n    }\r\n    this.framesRecorded = 0;\r\n  }\r\n\r\n  listDevices() {\r\n    navigator.mediaDevices.enumerateDevices().then((l: MediaDeviceInfo[]) => this.printDevices(l));\r\n  }\r\n\r\n  private dummySession():Promise<MediaStream>{\r\n    // workaround to request permissions:\r\n    // Start a dummy session\r\n    let mediaStrCnstrs = <MediaStreamConstraints>{audio:\r\n        {echoCancelation: false}\r\n    };\r\n    return navigator.mediaDevices.getUserMedia(mediaStrCnstrs);\r\n\r\n  }\r\n\r\n\r\n  private stopAllSessionTracks(mediaStream:MediaStream){\r\n      let ats = mediaStream.getTracks();\r\n      for (let atIdx = 0; atIdx < ats.length; atIdx++) {\r\n        //console.debug(\"Stop dummy session track: #\" + atIdx)\r\n        ats[atIdx].stop();\r\n      }\r\n  }\r\n\r\n  deviceInfos(cb: (deviceInfos: MediaDeviceInfo[] | null) => any, retry = true,dummyStream?:MediaStream) {\r\n\r\n    navigator.mediaDevices.enumerateDevices().then((l: MediaDeviceInfo[]) => {\r\n      let labelsAvailable = false;\r\n      for (let i = 0; i < l.length; i++) {\r\n        let di = l[i];\r\n        if (di.label) {\r\n          labelsAvailable = true;\r\n        }\r\n      }\r\n      if (!labelsAvailable) {\r\n        //console.debug(\"Media device enumeration: No labels.\")\r\n        if (retry) {\r\n            console.info(\"Starting dummy session to request audio permissions...\")\r\n\r\n            this.dummySession().then((s: MediaStream) => {\r\n            // and stop it immediately\r\n\r\n            if(s) {\r\n              //console.debug(\"Got dummy session stream: \" + s + \" .\")\r\n            }else{\r\n              //console.debug(\"No dummy stream\")\r\n            }\r\n            // retry (only once)\r\n            this.deviceInfos(cb, false,s);\r\n          },reason => {\r\n            //console.debug(\"Dummy session rejected.\")\r\n            // TODO error callback\r\n            cb(null);\r\n          });\r\n        } else {\r\n          cb(null);\r\n        }\r\n      } else {\r\n        // success\r\n        cb(l);\r\n      }\r\n      if(dummyStream){\r\n        this.stopAllSessionTracks(dummyStream);\r\n      }\r\n    },(reason)=> {\r\n      //rejected\r\n      //console.debug(\"Media device enumeration rejected.\")\r\n      if (retry) {\r\n        //console.debug(\"Starting dummy session to request audio permissions...\")\r\n        this.dummySession().then((s: MediaStream) => {\r\n          // and stop it immediately\r\n          //console.debug(\"Dummy session.\")\r\n          if(s) {\r\n            //console.debug(\"Got dummy session stream: \" + s + \" .\")\r\n          }else{\r\n            //console.debug(\"No dummy stream\")\r\n          }\r\n          // retry (only once)\r\n          this.deviceInfos(cb, false,s);\r\n        }, reason => {\r\n          //console.debug(\"Dummy session rejected.\")\r\n          // TODO error callback\r\n          cb(null);\r\n        });\r\n      } else {\r\n        cb(null);\r\n      }\r\n      if(dummyStream){\r\n        this.stopAllSessionTracks(dummyStream);\r\n      }\r\n    });\r\n\r\n\r\n\r\n  }\r\n\r\n\r\n  printDevices(l: MediaDeviceInfo[]): void {\r\n    //let selDeviceId = '___dummy___';\r\n    for (let i = 0; i < l.length; i++) {\r\n      let di = l[i];\r\n      console.log(\"Audio device: Id: \" + di.deviceId + \" groupId: \" + di.groupId + \" label: \" + di.label + \" kind: \" + di.kind);\r\n    }\r\n  }\r\n\r\n  addCaptureInterceptor() {\r\n    const awn = new AudioWorkletNode(this.context, 'capture-interceptor');\r\n    awn.onprocessorerror = (ev: Event) => {\r\n      let msg = 'Unknwon error';\r\n      if (ev instanceof ErrorEvent) {\r\n        msg = ev.message;\r\n      }\r\n      console.error(\"Capture audio worklet error: \" + msg);\r\n      if (this.listener) {\r\n        this.listener.error(msg);\r\n      }\r\n    }\r\n    let awnPt = awn.port;\r\n    if (awnPt) {\r\n      awnPt.onmessage = (ev: MessageEvent) => {\r\n        if (this.capturing) {\r\n          let dt = ev.data;\r\n          let chs = dt.chs;\r\n          let adaLen = dt.data.length;\r\n          if (DEBUG_TRACE_LEVEL > 8) {\r\n            console.debug('Received data from worklet: ' + chs + ' ' + dt.len + ' Data chs: ' + adaLen);\r\n          }\r\n          let chunk = new Array<Float32Array>(chs);\r\n          const samples = this.framesRecorded * chs;\r\n          if ((AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this.audioStorageType || AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this.audioStorageType) && this.data && samples > this._maxAutoNetMemStoreSamples) {\r\n            this.data = null;\r\n          }\r\n\r\n          //console.debug(\"Data initialized: \"+(this.data!=null));\r\n          for (let ch = 0; ch < chs; ch++) {\r\n            //console.debug(\"Data ch initialized: \"+(this.data !=null && this.data[ch] !=null));\r\n            if (ch < this.channelCount) {\r\n              if (dt.data[ch]) {\r\n                let fa = new Float32Array(dt.data[ch]);\r\n                if (this.data && this.data[ch]) {\r\n                  this.data[ch].push(fa);\r\n                }\r\n                chunk[ch] = fa;\r\n                // Use samples of channel 0 to count frames (samples)\r\n                if (ch == 0) {\r\n                  this.framesRecorded += fa.length;\r\n                }\r\n              }\r\n            }\r\n          }\r\n          if (this.audioOutStream) {\r\n            try {\r\n              this.audioOutStream.write(chunk);\r\n              // // Random test error:\r\n              // if(Math.random()>0.98) {\r\n              //   throw new Error('Test');\r\n              // }\r\n            } catch (err) {\r\n              if (err instanceof Error) {\r\n                this.persistError = err;\r\n              } else {\r\n                this.persistError = new Error('Error handling recorded audio data');\r\n              }\r\n\r\n              console.error(\"Capture error: \" + err);\r\n              try {\r\n                this.stop();\r\n              } catch (err2) {\r\n                console.error(\"Capture next error (ignored): \" + err2);\r\n              } finally {\r\n                if (this.listener) {\r\n                  let errExpl = '';\r\n                  if (err instanceof DOMException) {\r\n                    errExpl = ': ' + err.name + ': ' + err.message;\r\n                  }\r\n                  this.listener.error(\"Could not handle recorded audio data\" + errExpl, \"Please try to record again.\");\r\n                } else {\r\n                  this.close();\r\n                }\r\n              }\r\n            }\r\n          }\r\n          if (AudioStorageType.DB_CHUNKED === this._audioStorageType && this._persistentAudioStorageTarget) {\r\n            this.store();\r\n          }\r\n        }\r\n      };\r\n    }\r\n    // Tried to fix that Safari does not record the second channel\r\n    // Does not help\r\n    //awn.channelCount=this.channelCount;\r\n    //awn.channelCountMode='explicit';\r\n    //console.debug('Channel count explicitly set to '+this.channelCount);\r\n\r\n    this.bufferingNode = awn;\r\n    //this.bufferingNode.channelCount=this.channelCount;\r\n    this._opened = true;\r\n    if (this.listener) {\r\n      this.listener.opened();\r\n    }\r\n  }\r\n\r\n  open(channelCount: number, selDeviceId?: ConstrainDOMString|undefined,autoGainControlConfigs?:Array<AutoGainControlConfig>|null|undefined){\r\n    //console.debug(\"Capture open: ctx state: \"+this.context.state);\r\n    if(this.context.state!=='running'){\r\n      //console.debug(\"Capture open: Resume context\");\r\n      this.context.resume().then(()=>{\r\n        //console.debug(\"Capture open (ctx resumed): ctx state: \"+this.context.state);\r\n        this._open(channelCount, selDeviceId, autoGainControlConfigs);\r\n      })\r\n    }else{\r\n      this._open(channelCount, selDeviceId, autoGainControlConfigs);\r\n    }\r\n\r\n  }\r\n\r\n  _open(channelCount: number, selDeviceId?: ConstrainDOMString|undefined,autoGainControlConfigs?:Array<AutoGainControlConfig>|null|undefined) {\r\n    this.channelCount = channelCount;\r\n    this.framesRecorded = 0;\r\n\r\n\r\n    //var msc = new AudioStreamConstr();\r\n    // var msc={};\r\n    //msc.video = false;\r\n    //msc.audio = true;\r\n\r\n    // Chrome and Firefox stereo channels are identical !!\r\n    // And even worse: The data coming from the source is already preprocessed on FF and Chrome. It contains DSP artifacts!!\r\n\r\n    // https://bugs.chromium.org/p/chromium/issues/detail?id=387737\r\n    // The workaround to set these constraints does _NOT_ help:\r\n    //var msc={audio: {echoCancellation: false,channelCount: 2, googAudioMirroring: false},video: false};\r\n\r\n    // Safari at least version 11: Support for media streams\r\n    // TODO test if input is unprocessed\r\n\r\n    let msc:MediaStreamConstraints;\r\n    console.info('User agent: '+navigator.userAgent);\r\n\r\n\r\n      let ua=UserAgentBuilder.userAgent();\r\n\r\n      // ua.components.forEach((c)=>{\r\n      //   console.info(\"UA_Comp: \"+c.toString());\r\n      // })\r\n\r\n     let agcCfg:AutoGainControlConfig|null=null;\r\n\r\n    let autoGainControl=false;\r\n    let chromeEchoCancellation=false;\r\n    if(autoGainControlConfigs){\r\n      for(let agcc of autoGainControlConfigs){\r\n        if(agcc.platform===CfgPlatform.Android && ua.detectedPlatform===Platform.Android){\r\n            agcCfg=agcc;\r\n            break;\r\n        }\r\n        if(agcc.platform===CfgPlatform.Windows && ua.detectedPlatform===Platform.Windows){\r\n          agcCfg=agcc;\r\n          break;\r\n        }\r\n      }\r\n      if(agcCfg){\r\n        // TODO use EXACT/IDEAL constraint\r\n        autoGainControl=agcCfg.value;\r\n        if(CHROME_ACTIVATE_ECHO_CANCELLATION_WITH_AGC){\r\n          chromeEchoCancellation=agcCfg.value;\r\n        }\r\n        // TODO query real AGC status\r\n        this.agcStatus=agcCfg.value;\r\n      }else{\r\n        this.agcStatus=false;\r\n      }\r\n    }\r\n\r\n    // default\r\n    msc = {\r\n      audio: {\r\n        deviceId: selDeviceId,\r\n        echoCancellation: false,\r\n        channelCount: channelCount,\r\n        autoGainControl: autoGainControl\r\n      },\r\n      video: false\r\n    };\r\n\r\n    if (ua.detectedBrowser===Browser.Edge) {\r\n\r\n      // Microsoft Edge sends unmodified audio\r\n      // The constraint can follow the specification\r\n      console.info(\"Setting media track constraints for Microsoft Edge.\");\r\n      msc = {\r\n        audio: {\r\n          deviceId: selDeviceId,\r\n          echoCancellation: false,\r\n          channelCount: channelCount,\r\n          autoGainControl: autoGainControl\r\n        },\r\n        video: false\r\n      };\r\n    } else if (ua.detectedBrowser===Browser.Chrome) {\r\n      // Google Chrome: we need to switch of each of the preprocessing units including the\r\n      console.info(\"Setting media track constraints for Google Chrome.\");\r\n\r\n      // Chrome 60 -> 61 changed\r\n      // it works now without mandatory/optional sub-objects\r\n\r\n\r\n      // Requires at least Chrome 61\r\n      msc = {\r\n        audio: {\r\n          deviceId: selDeviceId,\r\n          channelCount: channelCount,\r\n          echoCancellation: {exact:chromeEchoCancellation},\r\n          autoGainControl: {exact:autoGainControl},\r\n          sampleSize:{min: 16},\r\n        },\r\n        video: false,\r\n      }\r\n\r\n    } else if (ua.detectedBrowser===Browser.Firefox) {\r\n      console.info(\"Setting media track constraints for Mozilla Firefox.\");\r\n      // Firefox\r\n      msc = {\r\n        audio: {\r\n            deviceId: selDeviceId,\r\n            channelCount: channelCount,\r\n          echoCancellation: false,\r\n            autoGainControl: autoGainControl,\r\n          noiseSuppression: false\r\n        },\r\n        video: false,\r\n      }\r\n\r\n    } else if (ua.detectedBrowser===Browser.Safari) {\r\n      console.info(\"Setting media track constraints for Safari browser.\")\r\n      //console.info(\"Apply workaround for Safari: Avoid disconnect of streams.\");\r\n\r\n      this.disconnectStreams = true;\r\n      msc = {\r\n        audio: {\r\n          deviceId: selDeviceId,\r\n          channelCount: channelCount,\r\n          //echoCancellation: false\r\n        },\r\n        video: false,\r\n      }\r\n\r\n    } else {\r\n\r\n      // TODO default constraints or error Browser not supported\r\n    }\r\n\r\n\r\n\r\n    console.debug(\"Audio capture, AGC: \"+this.agcStatus)\r\n\r\n    let ump = navigator.mediaDevices.getUserMedia(msc);\r\n    ump.then((s) => {\r\n        this.stream = s;\r\n\r\n        let aTracks = s.getAudioTracks();\r\n\r\n        for (let i = 0; i < aTracks.length; i++) {\r\n          let aTrack = aTracks[i];\r\n\r\n          console.info(\"Track audio info: id: \" + aTrack.id + \" kind: \" + aTrack.kind + \" label: \\\"\" + aTrack.label + \"\\\"\");\r\n          let mtrSts=aTrack.getSettings();\r\n\r\n          // Typescript lib.dom.ts MediaTrackSettings.channelCount is missing\r\n          // https://github.com/mdn/browser-compat-data/blob/5493d8f937e05b2ddbd41b99f5bdfad4a1f2ed85/api/MediaTrackSettings.json\r\n          //@ts-ignore\r\n          console.info(\"Track audio settings: Ch cnt: \"+mtrSts.channelCount+\", AGC: \"+mtrSts.autoGainControl+\", Echo cancell.: \"+mtrSts.echoCancellation);\r\n          if(mtrSts.autoGainControl){\r\n            this.agcStatus=mtrSts.autoGainControl;\r\n          }\r\n        }\r\n\r\n        let vTracks = s.getVideoTracks();\r\n        for (let i = 0; i < vTracks.length; i++) {\r\n          let vTrack = vTracks[i];\r\n          console.info(\"Track video info: id: \" + vTrack.id + \" kind: \" + vTrack.kind + \" label: \" + vTrack.label);\r\n        }\r\n        this.mediaStream = this.context.createMediaStreamSource(s);\r\n        // stream channel count ( is always 2 !)\r\n        let streamChannelCount: number = this.mediaStream.channelCount;\r\n        console.info(\"Stream channel count: \"+streamChannelCount);\r\n        // is not set!!\r\n        //this.currentSampleRate = this.mediaStream.sampleRate;\r\n        this.currentSampleRate = this.context.sampleRate;\r\n        console.info(\"Source audio node: channels: \" + streamChannelCount + \" samplerate: \" + this.currentSampleRate);\r\n        if (this.audioOutStream) {\r\n          this.audioOutStream.setFormat(this.channelCount, this.currentSampleRate);\r\n        }\r\n        // W3C  -> new name is createScriptProcessor\r\n        //\r\n        // Again deprecated, but AudioWorker not yet implemented in stable releases (June 2016)\r\n        // AudioWorker is now AudioWorkletProcessor ... (May 2017)\r\n\r\n      // Update 12-2020:\r\n       // The ScriptProcessorNode Interface - DEPRECATED\r\n\r\n      // Update 06-2021\r\n      //  AudioWorkletProcessor is here to stay. Web Audio API has now Recommendation status !\r\n\r\n          if(this.context.audioWorklet){\r\n            //const workletFileName = ('file-loader!./interceptor_worklet.js');\r\n            //const workletFileName = 'http://localhost:4200/assets/interceptor_worklet.js';\r\n            //console.log(awpStr);\r\n            if(AudioCapture.captureInterceptorModuleRegistered) {\r\n              // Required capture interceptor module already registered\r\n              this.addCaptureInterceptor();\r\n            }else{\r\n\r\n              // Register capture interceptor module\r\n              let audioWorkletModuleBlob = new Blob([awpStr], {type: 'text/javascript'});\r\n\r\n              let audioWorkletModuleBlobUrl = window.URL.createObjectURL(audioWorkletModuleBlob);\r\n\r\n              this.context.audioWorklet.addModule(audioWorkletModuleBlobUrl).then(() => {\r\n                  AudioCapture.captureInterceptorModuleRegistered = true;\r\n                  this.addCaptureInterceptor();\r\n                }\r\n              ).catch((error: any) => {\r\n                console.log('Could not add module ' + error);\r\n              });\r\n            }\r\n          }else if(this.context.createScriptProcessor) {\r\n            // The ScriptProcessorNode Interface - DEPRECATED Only as fallback\r\n            // TODO should we use streamChannelCount or channelCount here ?\r\n            let scriptProcessorNode= this.context.createScriptProcessor(AudioCapture.BUFFER_SIZE, streamChannelCount, streamChannelCount);\r\n            this.bufferingNode=scriptProcessorNode;\r\n            let c = 0;\r\n            if(scriptProcessorNode.onaudioprocess){\r\n              scriptProcessorNode.onaudioprocess = (e: AudioProcessingEvent) => {\r\n\r\n                if (this.capturing) {\r\n                  let inBuffer = e.inputBuffer;\r\n                  // only process requested count of channels\r\n                  let currentBuffers = new Array<Float32Array>(channelCount);\r\n                  for (let ch: number = 0; ch < channelCount; ch++) {\r\n                    let chSamples = inBuffer.getChannelData(ch);\r\n                    let chSamplesCopy = chSamples.slice(0);\r\n                    currentBuffers[ch] = chSamplesCopy.slice(0);\r\n                    if(this.data) {\r\n                      this.data[ch].push(chSamplesCopy);\r\n                    }\r\n                    if(DEBUG_TRACE_LEVEL>8){\r\n                      console.debug(\"Process \"+chSamplesCopy.length+\" samples.\");\r\n                    }\r\n                    this.framesRecorded += chSamplesCopy.length;\r\n                  }\r\n                  c++;\r\n                  if (this.audioOutStream) {\r\n                    this.audioOutStream.write(currentBuffers);\r\n                  }\r\n                }\r\n              };\r\n              this._opened = true;\r\n              if (this.listener) {\r\n                this.listener.opened();\r\n              }\r\n            }else{\r\n              this.listener.error('Browser does not support audio processing (ScriptProcessor.onaudioprocess method not found)!');\r\n            }\r\n          }else{\r\n            this.listener.error('Browser does not support audio processing (neither AudioWorkletProcessor nor ScriptProcessor)!');\r\n          }\r\n        }, (e) => {\r\n          console.error(e + \" Error name: \" +e.name);\r\n          if (this.listener) {\r\n            if('NotAllowedError' === e.name){\r\n              this.listener.error('Not allowed to use your microphone.','Please make sure that microphone access is allowed for this web page and reload the page.');\r\n            }else if('NotReadableError' === e.name){\r\n              this.listener.error('Could not read from your audio device.','Please make sure your audio device is working.');\r\n            }else if('OverconstrainedError' === e.name){\r\n              let eMsg=e.msg?e.msg:'Overconstrained media device request error.';\r\n              this.listener.error(eMsg);\r\n            } else {\r\n              this.listener.error();\r\n            }\r\n          }\r\n        }\r\n    )\r\n  }\r\n\r\n  private _start(){\r\n    this.initData();\r\n    if (this.audioOutStream) {\r\n      this.audioOutStream.nextStream()\r\n    }\r\n    this.capturing = true;\r\n    if(this.bufferingNode) {\r\n      this.mediaStream.connect(this.bufferingNode);\r\n      this.bufferingNode.connect(this.context.destination);\r\n    }\r\n    if (this.listener) {\r\n      this.listener.started();\r\n    }\r\n  }\r\n\r\n  start() {\r\n    const aSt=this.context.state;\r\n    if(aSt==='running') {\r\n      this._start();\r\n    }else{\r\n      console.debug(\"Capture start: audio context not running, state: \"+aSt+\", resuming...\");\r\n      this.context.resume().then(()=>{\r\n        console.debug(\"Capture start: audio context resumed, starting...\");\r\n        this._start();\r\n      })\r\n    }\r\n  }\r\n\r\n  stop() {\r\n\r\n    if (this.disconnectStreams && this.bufferingNode) {\r\n      this.mediaStream.disconnect(this.bufferingNode);\r\n      this.bufferingNode.disconnect(this.context.destination);\r\n    }\r\n\r\n    try {\r\n      if (this.audioOutStream) {\r\n        this.audioOutStream.flush();\r\n      }\r\n    }catch(err){\r\n      console.error(\"Could not flush capture stream.\");\r\n     throw err;\r\n    }finally {\r\n      this.capturing = false;\r\n      if (this.inddbAudioBuffer && this.persistError) {\r\n        // Delete invalid persistent audio data and hope that it will be completely uploaded to the server\r\n        this.inddbAudioBuffer.releaseAudioData();\r\n        this.inddbAudioBuffer = null;\r\n      }\r\n      if (this.listener && (this.persistError || this.persisted)) {\r\n        //console.debug(\"Stopped by stop() method call\");\r\n        this.listener.stopped();\r\n      }\r\n    }\r\n  }\r\n\r\n\r\n  store(){\r\n   // if(this.db && this.recUUID){\r\n   //    let tr= this.db.transaction(SprDb.RECORDING_FILE_CHUNKS_OBJECT_STORE_NAME, 'readwrite');\r\n   //    let recFileObjStore = tr.objectStore(SprDb.RECORDING_FILE_CHUNKS_OBJECT_STORE_NAME);\r\n   //\r\n   //      try {\r\n   //        let ch0Data=this.data[0];\r\n   //        let dataChkCnt=ch0Data.length;\r\n   //          let pos = 0;\r\n   //          for (let chCkIdx = 0; chCkIdx < dataChkCnt; chCkIdx++) {\r\n   //            let bufLen=0;\r\n   //            for (let ch = 0; ch < this.channelCount; ch++) {\r\n   //              let chChk = this.data[ch][chCkIdx];\r\n   //              bufLen = chChk.length;\r\n   //              //let cacheId = uuid + '_' + ch + '_' + chCkIdx;\r\n   //              let chkDbId = [this.recUUID, this.indDbChkIdx + chCkIdx, ch];\r\n   //              let cr = recFileObjStore.add(chChk, chkDbId);\r\n   //              //console.debug(\"Added: \"+ch+\" \"+(this.indDbChkIdx+chCkIdx));\r\n   //              cr.onsuccess=()=>{\r\n   //                //console.debug(\"Stored audio data to indexed db\");\r\n   //              }\r\n   //              cr.onerror=()=>{\r\n   //                console.error(\"Error storing audio data to indexed db\");\r\n   //              }\r\n   //            }\r\n   //            pos += bufLen;\r\n   //        }\r\n   //        this.indDbChkIdx+=dataChkCnt;\r\n   //\r\n   //        tr.onerror = (err) => {\r\n   //          console.error('Failed to cache audio data to indexed db: ' + err)\r\n   //        }\r\n   //        tr.oncomplete = () => {\r\n   //          //console.debug('Transferred capture audio data to indexed db, deleting original data from memory...');\r\n   //\r\n   //          /// Audio data saved to index db delete from in memory data array\r\n   //          for (let ch = 0; ch < this.channelCount; ch++) {\r\n   //           this.data[ch].splice(0);\r\n   //            //console.debug(\"Spliced/removed ch: \"+ch);\r\n   //          }\r\n   //\r\n   //          this.persisted=true;\r\n   //          if(this.listener && !this.capturing){\r\n   //            //console.debug(\"Stopped by indexed db hook\");\r\n   //            this.listener.stopped();\r\n   //          }\r\n   //        }\r\n   //        // Commit chunks\r\n   //        this.persisted=false;\r\n   //        tr.commit();\r\n   //      } catch (err) {\r\n   //        console.error('Transfer capture audio data error: '+err);\r\n   //      }\r\n   //    }\r\n\r\n    // if(!this.inddbAudioBuffer && this._persistentAudioStorageTarget && this.recUUID) {\r\n    //   this.inddbAudioBuffer = new IndexedDbAudioBuffer(this._persistentAudioStorageTarget, this.channelCount,this.currentSampleRate,AudioCapture.BUFFER_SIZE,0,this.recUUID)\r\n    // }\r\n    if(this.inddbAudioBuffer && this.data){\r\n\r\n      // Try to append to\r\n      this.inddbAudioBuffer.appendRawAudioData(this.data).subscribe({\r\n        complete: () => {\r\n          //console.debug('Transferred capture audio data to indexed db, deleting original data from memory...');\r\n\r\n          /// Audio data saved to index db delete from in memory data array\r\n          if(this.data) {\r\n            for (let ch = 0; ch < this.channelCount; ch++) {\r\n              this.data[ch].splice(0);\r\n              //console.debug(\"Spliced/removed ch: \"+ch);\r\n            }\r\n          }\r\n          this.persisted = true;\r\n          if (this.listener && !this.capturing) {\r\n            //console.debug(\"Stopped by indexed db hook\");\r\n            this.listener.stopped();\r\n          }\r\n        },error:(err)=>{\r\n          // Only log the first error\r\n          if(!this.persistError) {\r\n            this.persistError = err;\r\n            console.error(\"Error persisting audio data: \" + err);\r\n          }\r\n        }\r\n      });\r\n      this.persisted=false;\r\n    }\r\n  }\r\n\r\n\r\n  close() {\r\n    if(this.mediaStream) {\r\n      this.mediaStream.disconnect();\r\n    }\r\n    if (this.stream) {\r\n      const mts = this.stream.getTracks();\r\n      for (let i = 0; i < mts.length; i++) {\r\n        mts[i].stop();\r\n      }\r\n    }\r\n    this._opened=false;\r\n  }\r\n\r\n  audioBuffer(): AudioBuffer |null{\r\n    let ab: AudioBuffer|null=null;\r\n    if(this.data) {\r\n      let frameLen: number = 0;\r\n\r\n      let ch0Data = this.data[0];\r\n\r\n      for (let ch0Chk of ch0Data) {\r\n        frameLen += ch0Chk.length;\r\n      }\r\n\r\n      try {\r\n        ab = this.context.createBuffer(this.channelCount, frameLen, this.context.sampleRate);\r\n      } catch (err) {\r\n        if (err instanceof DOMException) {\r\n          if (err.name === 'NotSupportedError') {\r\n            if (frameLen == 0) {\r\n              // Empty buffers are not supported by Chromium\r\n              // Create dummy buffer with one sample\r\n              ab = this.context.createBuffer(this.channelCount, 1, this.context.sampleRate);\r\n            } else {\r\n              throw err;\r\n            }\r\n          } else {\r\n            throw err;\r\n          }\r\n        } else if (err instanceof RangeError) {\r\n          // Out of memory\r\n          // TODO What to do ??\r\n          throw err;\r\n        } else {\r\n          throw err;\r\n        }\r\n      }\r\n      for (let ch = 0; ch < this.channelCount; ch++) {\r\n        let chD = ab.getChannelData(ch);\r\n        let pos = 0;\r\n        for (let chChk of this.data[ch]) {\r\n          let bufLen = chChk.length;\r\n          chD.set(chChk, pos);\r\n          pos += bufLen;\r\n        }\r\n      }\r\n    }\r\n    return ab;\r\n  }\r\n\r\n  audioBufferArray():ArrayAudioBuffer|null{\r\n      let arrAb:ArrayAudioBuffer|null=null;\r\n      if(this.data) {\r\n        arrAb=new ArrayAudioBuffer(this.channelCount, this.currentSampleRate, this.data);\r\n      }\r\n      return arrAb;\r\n  }\r\n\r\n  inddbAudioBufferArray():IndexedDbAudioBuffer|null{\r\n    if(this.persistError){\r\n      return null;\r\n    }else {\r\n      return this.inddbAudioBuffer;\r\n    }\r\n  }\r\n\r\n\r\n}\r\n\r\n"]}
|
|
855
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"capture.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/audio/capture/capture.ts"],"names":[],"mappings":"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","sourcesContent":["import {SequenceAudioFloat32OutStream} from \"../io/stream\";\r\nimport {Browser, Platform, UserAgentBuilder} from \"../../utils/ua-parser\";\r\n\r\nimport {AudioStorageType, AutoGainControlConfig, Platform as CfgPlatform} from \"../../speechrecorder/project/project\";\r\nimport {ArrayAudioBuffer} from \"../array_audio_buffer\";\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {IndexedDbAudioBuffer, PersistentAudioStorageTarget} from \"../inddb_audio_buffer\";\r\nimport {AudioContextProvider} from \"../context\";\r\n\r\n\r\nexport const CHROME_ACTIVATE_ECHO_CANCELLATION_WITH_AGC=false;\r\n\r\nconst DEBUG_TRACE_LEVEL=0;\r\n\r\n// Dirty way to load this module\r\n// Copy content of interceptor_worklet.js to this string\r\nconst awpStr=\"class AudioCaptureInterceptorProcessor extends AudioWorkletProcessor{\\n\" +\r\n    \"\\n\" +\r\n    \"    BUFFER_QUANTUMS=64;\\n\" +\r\n    \"    QUANTUM_FRAME_LEN=128;\\n\" +\r\n    \"    BUFFER_FRAME_LEN=this.QUANTUM_FRAME_LEN*this.BUFFER_QUANTUMS;\\n\" +\r\n    \"    buffer=null;\\n\" +\r\n    \"    bufferPos=0;\\n\" +\r\n    \"    bufferPosBytes=0;\\n\" +\r\n    \"    constructor() {\\n\" +\r\n    \"        super();\\n\" +\r\n    \"\\n\" +\r\n    \"    }\\n\" +\r\n    \"\\n\" +\r\n    \" process(\\n\" +\r\n    \"      inputs,\\n\" +\r\n    \"      outputs,\\n\" +\r\n    \"      parameters\\n\" +\r\n    \"  ){\\n\" +\r\n    \"\\n\" +\r\n    \"     let inputsCnt=inputs.length;\\n\" +\r\n    \"     let channelCount=0;\\n\" +\r\n    \"     let inputLen=0;\\n\" +\r\n    \"     let inputLenBytes=0;\\n\" +\r\n    \"     if(inputsCnt>0) {\\n\" +\r\n    \"         let input0 = inputs[0];\\n\" +\r\n    \"         channelCount = input0.length;\\n\" +\r\n    \"         if (channelCount > 0) {\\n\" +\r\n    \"             let input0ch0=input0[0];\\n\" +\r\n    \"             inputLen=input0ch0.length;\\n\" +\r\n    \"             inputLenBytes=input0ch0.buffer.length;\\n\" +\r\n    \"         }\\n\" +\r\n    \"     }\\n\" +\r\n    \"     if (!this.buffer || this.buffer.length < channelCount) {\\n\" +\r\n    \"         this.buffer = new Array(channelCount);\\n\" +\r\n    \"         this.bufferPos = 0\\n\" +\r\n    \"         for (let bch = 0; bch < channelCount; bch++) {\\n\" +\r\n    \"             this.buffer[bch] = new Float32Array(this.BUFFER_FRAME_LEN);\\n\" +\r\n    \"             this.bufferPos = 0;\\n\" +\r\n    \"             this.bufferPosBytes=0;\\n\" +\r\n    \"         }\\n\" +\r\n    \"     }\\n\" +\r\n    \"     let bufAvail = this.BUFFER_FRAME_LEN - this.bufferPos;\\n\" +\r\n    \"     // check if buffer has to be transferred\\n\" +\r\n    \"     if (inputLen > bufAvail) {\\n\" +\r\n    \"         let ada=new Array(channelCount);\\n\" +\r\n    \"         for (let ch = 0; ch < channelCount; ch++) {\\n\" +\r\n    \"             ada[ch]=this.buffer[ch].buffer.slice(0);\\n\" +\r\n    \"         }\\n\" +\r\n    \"         this.port.postMessage({\\n\" +\r\n    \"             data: ada,\\n\" +\r\n    \"             chs: channelCount,\\n\" +\r\n    \"             len: this.bufferPos\\n\" +\r\n    \"         }, ada);\\n\" +\r\n    \"         // buffer transferred, reset\\n\" +\r\n    \"         this.bufferPos = 0;\\n\" +\r\n    \"         this.bufferPosBytes=0;\\n\" +\r\n    \"     }\\n\" +\r\n    \"\\n\" +\r\n    \"     for(let ii=0;ii<inputsCnt;ii++) {\\n\" +\r\n    \"         for (let ch = 0; ch < channelCount; ch++) {\\n\" +\r\n    \"             // Mute outputs\\n\" +\r\n    \"             //outputs[ii][ch].fill(0);\\n\" +\r\n    \"             let chSamples = inputs[ii][ch];\\n\" +\r\n    \"             this.buffer[ch].set(chSamples,this.bufferPos);\\n\" +\r\n    \"         }\\n\" +\r\n    \"         this.bufferPos+=inputLen;\\n\" +\r\n    \"         this.bufferPosBytes+=inputLenBytes;\\n\" +\r\n    \"     }\\n\" +\r\n    \"    \\n\" +\r\n    \"     return true;\\n\" +\r\n    \"  }\\n\" +\r\n    \"}\\n\" +\r\n    \"\\n\" +\r\n    \"registerProcessor('capture-interceptor',AudioCaptureInterceptorProcessor);\\n\";\r\n\r\n\r\n\r\n\r\nexport interface AudioCaptureListener {\r\n  opened(): void;\r\n\r\n  started(): void;\r\n\r\n  stopped(): void;\r\n\r\n  closed(): void;\r\n\r\n  error(msg?:string,advice?:string): void;\r\n}\r\n\r\nexport class AudioCapture {\r\n\r\n  get maxAutoNetMemStoreSamples(): number {\r\n    return this._maxAutoNetMemStoreSamples;\r\n  }\r\n\r\n  set maxAutoNetMemStoreSamples(value: number) {\r\n    this._maxAutoNetMemStoreSamples = value;\r\n  }\r\n  set recUUID(value: string|null) {\r\n    this._recUUID = value;\r\n  }\r\n  get recUUID(): string |null{\r\n    return this._recUUID;\r\n  }\r\n  get audioStorageType(): AudioStorageType {\r\n    return this._audioStorageType;\r\n  }\r\n\r\n  set audioStorageType(value: AudioStorageType) {\r\n    this._audioStorageType = value;\r\n  }\r\n  get persistentAudioStorageTarget(): PersistentAudioStorageTarget | null {\r\n    return this._persistentAudioStorageTarget;\r\n  }\r\n\r\n  set persistentAudioStorageTarget(value: PersistentAudioStorageTarget | null) {\r\n    this._persistentAudioStorageTarget = value;\r\n  }\r\n\r\n\r\n  get opened(): boolean {\r\n    return this._opened;\r\n  }\r\n\r\n  static BUFFER_SIZE: number = 8192;\r\n  private static readonly DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES:number=2880000*5; // Default 5 minute at 48kHz\r\n  private _maxAutoNetMemStoreSamples:number=AudioCapture.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;\r\n  private static captureInterceptorModuleRegistered=false;\r\n  context: AudioContext|null=null;\r\n  stream!: MediaStream;\r\n  channelCount!: number;\r\n  private _recUUID:string|null=null;\r\n  mediaStream: any;\r\n  agcStatus:boolean|null=null;\r\n  bufferingNode: AudioNode|null=null;\r\n  listener!: AudioCaptureListener;\r\n  data: Array<Array<Float32Array>>|null=null;\r\n  currentSampleRate!: number;\r\n  n: Navigator;\r\n  audioOutStream: SequenceAudioFloat32OutStream | null=null;\r\n  private disconnectStreams = true;\r\n  private _opened=false;\r\n  private capturing = false;\r\n\r\n  framesRecorded: number=0;\r\n\r\n  private _audioStorageType:AudioStorageType=AudioStorageType.MEM_ENTIRE;\r\n  private _persistentAudioStorageTarget:PersistentAudioStorageTarget|null=null;\r\n\r\n  private persisted=true;\r\n  private persistError:Error|null=null;\r\n  private inddbAudioBuffer:IndexedDbAudioBuffer|null=null;\r\n\r\n  //private context:AudioContext|null=null;\r\n\r\n  constructor() {\r\n    this.n = navigator;\r\n  }\r\n\r\n  private _audioContext():AudioContext|null{\r\n    if(!this.context){\r\n      this.context=AudioContextProvider.audioContextInstance();\r\n      if(this.context) {\r\n        this.context.addEventListener('statechange', () => {\r\n          if (this.context && this.context.state !== 'running') {\r\n            this.close();\r\n          }\r\n        });\r\n      }\r\n    }\r\n    return this.context;\r\n  }\r\n\r\n  private initData() {\r\n    if(!this._recUUID) {\r\n      this._recUUID = UUID.generate();\r\n    }\r\n    this.persistError=null;\r\n    //console.debug(\"Audio capture initialize storage for type: \"+this._audioStorageType);\r\n    if(AudioStorageType.DB_CHUNKED === this._audioStorageType && this._persistentAudioStorageTarget && this._recUUID) {\r\n      //console.debug(\"Create indexed db audio buffer.\");\r\n      this.inddbAudioBuffer = new IndexedDbAudioBuffer(this._persistentAudioStorageTarget, this.channelCount,this.currentSampleRate,AudioCapture.BUFFER_SIZE,0,this._recUUID)\r\n    }\r\n    if(!(AudioStorageType.NET_CHUNKED === this._audioStorageType)) {\r\n      // Initialize audio data array except for net audio buffer mode\r\n      this.data = new Array<Array<Float32Array>>();\r\n      for (let i = 0; i < this.channelCount; i++) {\r\n        this.data.push(new Array<Float32Array>());\r\n      }\r\n    }\r\n    this.framesRecorded = 0;\r\n  }\r\n\r\n  listDevices() {\r\n    navigator.mediaDevices.enumerateDevices().then((l: MediaDeviceInfo[]) => this.printDevices(l));\r\n  }\r\n\r\n  private dummySession():Promise<MediaStream>{\r\n    // workaround to request permissions:\r\n    // Start a dummy session\r\n    let mediaStrCnstrs = <MediaStreamConstraints>{audio:\r\n        {echoCancelation: false}\r\n    };\r\n    return navigator.mediaDevices.getUserMedia(mediaStrCnstrs);\r\n\r\n  }\r\n\r\n\r\n  private stopAllSessionTracks(mediaStream:MediaStream){\r\n      let ats = mediaStream.getTracks();\r\n      for (let atIdx = 0; atIdx < ats.length; atIdx++) {\r\n        //console.debug(\"Stop dummy session track: #\" + atIdx)\r\n        ats[atIdx].stop();\r\n      }\r\n  }\r\n\r\n  deviceInfos(cb: (deviceInfos: MediaDeviceInfo[] | null) => any, retry = true,dummyStream?:MediaStream) {\r\n\r\n    navigator.mediaDevices.enumerateDevices().then((l: MediaDeviceInfo[]) => {\r\n      let labelsAvailable = false;\r\n      for (let i = 0; i < l.length; i++) {\r\n        let di = l[i];\r\n        if (di.label) {\r\n          labelsAvailable = true;\r\n        }\r\n      }\r\n      if (!labelsAvailable) {\r\n        //console.debug(\"Media device enumeration: No labels.\")\r\n        if (retry) {\r\n            console.info(\"Starting dummy session to request audio permissions...\")\r\n\r\n            this.dummySession().then((s: MediaStream) => {\r\n            // and stop it immediately\r\n\r\n            if(s) {\r\n              //console.debug(\"Got dummy session stream: \" + s + \" .\")\r\n            }else{\r\n              //console.debug(\"No dummy stream\")\r\n            }\r\n            // retry (only once)\r\n            this.deviceInfos(cb, false,s);\r\n          },reason => {\r\n            //console.debug(\"Dummy session rejected.\")\r\n            // TODO error callback\r\n            cb(null);\r\n          });\r\n        } else {\r\n          cb(null);\r\n        }\r\n      } else {\r\n        // success\r\n        cb(l);\r\n      }\r\n      if(dummyStream){\r\n        this.stopAllSessionTracks(dummyStream);\r\n      }\r\n    },(reason)=> {\r\n      //rejected\r\n      //console.debug(\"Media device enumeration rejected.\")\r\n      if (retry) {\r\n        //console.debug(\"Starting dummy session to request audio permissions...\")\r\n        this.dummySession().then((s: MediaStream) => {\r\n          // and stop it immediately\r\n          //console.debug(\"Dummy session.\")\r\n          if(s) {\r\n            //console.debug(\"Got dummy session stream: \" + s + \" .\")\r\n          }else{\r\n            //console.debug(\"No dummy stream\")\r\n          }\r\n          // retry (only once)\r\n          this.deviceInfos(cb, false,s);\r\n        }, reason => {\r\n          //console.debug(\"Dummy session rejected.\")\r\n          // TODO error callback\r\n          cb(null);\r\n        });\r\n      } else {\r\n        cb(null);\r\n      }\r\n      if(dummyStream){\r\n        this.stopAllSessionTracks(dummyStream);\r\n      }\r\n    });\r\n\r\n\r\n\r\n  }\r\n\r\n\r\n  printDevices(l: MediaDeviceInfo[]): void {\r\n    //let selDeviceId = '___dummy___';\r\n    for (let i = 0; i < l.length; i++) {\r\n      let di = l[i];\r\n      console.log(\"Audio device: Id: \" + di.deviceId + \" groupId: \" + di.groupId + \" label: \" + di.label + \" kind: \" + di.kind);\r\n    }\r\n  }\r\n\r\n  addCaptureInterceptor() {\r\n    if (this.context) {\r\n\r\n    const awn = new AudioWorkletNode(this.context, 'capture-interceptor');\r\n    awn.onprocessorerror = (ev: Event) => {\r\n      let msg = 'Unknwon error';\r\n      if (ev instanceof ErrorEvent) {\r\n        msg = ev.message;\r\n      }\r\n      console.error(\"Capture audio worklet error: \" + msg);\r\n      if (this.listener) {\r\n        this.listener.error(msg);\r\n      }\r\n    }\r\n    let awnPt = awn.port;\r\n    if (awnPt) {\r\n      awnPt.onmessage = (ev: MessageEvent) => {\r\n        if (this.capturing) {\r\n          let dt = ev.data;\r\n          let chs = dt.chs;\r\n          let adaLen = dt.data.length;\r\n          if (DEBUG_TRACE_LEVEL > 8) {\r\n            console.debug('Received data from worklet: ' + chs + ' ' + dt.len + ' Data chs: ' + adaLen);\r\n          }\r\n          let chunk = new Array<Float32Array>(chs);\r\n          const samples = this.framesRecorded * chs;\r\n          if ((AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this.audioStorageType || AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this.audioStorageType) && this.data && samples > this._maxAutoNetMemStoreSamples) {\r\n            this.data = null;\r\n          }\r\n\r\n          //console.debug(\"Data initialized: \"+(this.data!=null));\r\n          for (let ch = 0; ch < chs; ch++) {\r\n            //console.debug(\"Data ch initialized: \"+(this.data !=null && this.data[ch] !=null));\r\n            if (ch < this.channelCount) {\r\n              if (dt.data[ch]) {\r\n                let fa = new Float32Array(dt.data[ch]);\r\n                if (this.data && this.data[ch]) {\r\n                  this.data[ch].push(fa);\r\n                }\r\n                chunk[ch] = fa;\r\n                // Use samples of channel 0 to count frames (samples)\r\n                if (ch == 0) {\r\n                  this.framesRecorded += fa.length;\r\n                }\r\n              }\r\n            }\r\n          }\r\n          if (this.audioOutStream) {\r\n            try {\r\n              this.audioOutStream.write(chunk);\r\n              // // Random test error:\r\n              // if(Math.random()>0.98) {\r\n              //   throw new Error('Test');\r\n              // }\r\n            } catch (err) {\r\n              if (err instanceof Error) {\r\n                this.persistError = err;\r\n              } else {\r\n                this.persistError = new Error('Error handling recorded audio data');\r\n              }\r\n\r\n              console.error(\"Capture error: \" + err);\r\n              try {\r\n                this.stop();\r\n              } catch (err2) {\r\n                console.error(\"Capture next error (ignored): \" + err2);\r\n              } finally {\r\n                if (this.listener) {\r\n                  let errExpl = '';\r\n                  if (err instanceof DOMException) {\r\n                    errExpl = ': ' + err.name + ': ' + err.message;\r\n                  }\r\n                  this.listener.error(\"Could not handle recorded audio data\" + errExpl, \"Please try to record again.\");\r\n                } else {\r\n                  this.close();\r\n                }\r\n              }\r\n            }\r\n          }\r\n          if (AudioStorageType.DB_CHUNKED === this._audioStorageType && this._persistentAudioStorageTarget) {\r\n            this.store();\r\n          }\r\n        }\r\n      };\r\n    }\r\n    // Tried to fix that Safari does not record the second channel\r\n    // Does not help\r\n    //awn.channelCount=this.channelCount;\r\n    //awn.channelCountMode='explicit';\r\n    //console.debug('Channel count explicitly set to '+this.channelCount);\r\n\r\n    this.bufferingNode = awn;\r\n    //this.bufferingNode.channelCount=this.channelCount;\r\n    this._opened = true;\r\n    if (this.listener) {\r\n      this.listener.opened();\r\n    }\r\n  }\r\n  }\r\n\r\n  open(channelCount: number, selDeviceId?: ConstrainDOMString|undefined,autoGainControlConfigs?:Array<AutoGainControlConfig>|null|undefined,allowEchoCancellation?:boolean){\r\n    //console.debug(\"Capture open: ctx state: \"+this.context.state);\r\n    this.context=this._audioContext();\r\n    if(!this.context){\r\n      throw new Error(\"Could not get audio context!\");\r\n    }\r\n    if(this.context.state==='suspended'){\r\n      //console.debug(\"Capture open: Resume context\");\r\n      this.context.resume().then(()=>{\r\n        //console.debug(\"Capture open (ctx resumed): ctx state: \"+this.context.state);\r\n        this._open(channelCount, selDeviceId, autoGainControlConfigs,allowEchoCancellation);\r\n      }).catch((err)=>{\r\n        console.error(err.message);\r\n        throw err;\r\n      })\r\n    }else if(this.context.state==='closed') {\r\n        const msg='Error on start capture: The audio context is already closed.';\r\n        console.error(msg);\r\n        throw new Error(msg);\r\n    }else {\r\n      this._open(channelCount, selDeviceId, autoGainControlConfigs,allowEchoCancellation);\r\n    }\r\n\r\n  }\r\n\r\n  _open(channelCount: number, selDeviceId?: ConstrainDOMString|undefined,autoGainControlConfigs?:Array<AutoGainControlConfig>|null|undefined,allowEchoCancellation?:boolean) {\r\n    this.channelCount = channelCount;\r\n    this.framesRecorded = 0;\r\n\r\n    this.context=this._audioContext();\r\n\r\n    if(!this.context){\r\n      throw new Error(\"Could not get audio context!\");\r\n    }\r\n\r\n    //var msc = new AudioStreamConstr();\r\n    // var msc={};\r\n    //msc.video = false;\r\n    //msc.audio = true;\r\n\r\n    // Chrome and Firefox stereo channels are identical !!\r\n    // And even worse: The data coming from the source is already preprocessed on FF and Chrome. It contains DSP artifacts!!\r\n\r\n    // https://bugs.chromium.org/p/chromium/issues/detail?id=387737\r\n    // The workaround to set these constraints does _NOT_ help:\r\n    //var msc={audio: {echoCancellation: false,channelCount: 2, googAudioMirroring: false},video: false};\r\n\r\n    // Safari at least version 11: Support for media streams\r\n    // TODO test if input is unprocessed\r\n\r\n    let msc:MediaStreamConstraints;\r\n    console.info('User agent: '+navigator.userAgent);\r\n\r\n\r\n      let ua=UserAgentBuilder.userAgent();\r\n\r\n      // ua.components.forEach((c)=>{\r\n      //   console.info(\"UA_Comp: \"+c.toString());\r\n      // })\r\n\r\n     let agcCfg:AutoGainControlConfig|null=null;\r\n\r\n    let autoGainControl=false;\r\n    let chromeEchoCancellation=false;\r\n    if(autoGainControlConfigs){\r\n      for(let agcc of autoGainControlConfigs){\r\n        if(agcc.platform===CfgPlatform.Android && ua.detectedPlatform===Platform.Android){\r\n            agcCfg=agcc;\r\n            break;\r\n        }\r\n        if(agcc.platform===CfgPlatform.Windows && ua.detectedPlatform===Platform.Windows){\r\n          agcCfg=agcc;\r\n          break;\r\n        }\r\n      }\r\n      if(agcCfg){\r\n        // TODO use EXACT/IDEAL constraint\r\n        autoGainControl=agcCfg.value;\r\n        if(CHROME_ACTIVATE_ECHO_CANCELLATION_WITH_AGC){\r\n          chromeEchoCancellation=agcCfg.value;\r\n        }\r\n        // TODO query real AGC status\r\n        this.agcStatus=agcCfg.value;\r\n      }else{\r\n        this.agcStatus=false;\r\n      }\r\n    }\r\n\r\n    // default\r\n    msc = {\r\n      audio: {\r\n        deviceId: selDeviceId,\r\n        echoCancellation: false,\r\n        channelCount: channelCount,\r\n        autoGainControl: autoGainControl\r\n      },\r\n      video: false\r\n    };\r\n\r\n    if (ua.detectedBrowser===Browser.Edge) {\r\n\r\n      // Microsoft Edge sends unmodified audio\r\n      // The constraint can follow the specification\r\n      console.info(\"Setting media track constraints for Microsoft Edge.\");\r\n      msc = {\r\n        audio: {\r\n          deviceId: selDeviceId,\r\n          echoCancellation: false,\r\n          channelCount: channelCount,\r\n          autoGainControl: autoGainControl\r\n        },\r\n        video: false\r\n      };\r\n    } else if (ua.detectedBrowser===Browser.Chrome) {\r\n      // Google Chrome: we need to switch of each of the preprocessing units including the\r\n      console.info(\"Setting media track constraints for Google Chrome.\");\r\n\r\n      // Chrome 60 -> 61 changed\r\n      // it works now without mandatory/optional sub-objects\r\n\r\n\r\n      // Requires at least Chrome 61\r\n      msc = {\r\n        audio: {\r\n          deviceId: selDeviceId,\r\n          channelCount: channelCount,\r\n          echoCancellation: {exact:chromeEchoCancellation},\r\n          autoGainControl: {exact:autoGainControl},\r\n          sampleSize:{min: 16},\r\n        },\r\n        video: false,\r\n      }\r\n\r\n    } else if (ua.detectedBrowser===Browser.Firefox) {\r\n      console.info(\"Setting media track constraints for Mozilla Firefox.\");\r\n      // Firefox\r\n      msc = {\r\n        audio: {\r\n            deviceId: selDeviceId,\r\n            channelCount: channelCount,\r\n          echoCancellation: false,\r\n            autoGainControl: autoGainControl,\r\n          noiseSuppression: false\r\n        },\r\n        video: false,\r\n      }\r\n\r\n    } else if (ua.detectedBrowser===Browser.Safari) {\r\n      console.info(\"Setting media track constraints for Safari browser.\")\r\n      //console.info(\"Apply workaround for Safari: Avoid disconnect of streams.\");\r\n\r\n      this.disconnectStreams = true;\r\n\r\n      msc = {\r\n        audio: {\r\n          deviceId: selDeviceId,\r\n          channelCount: channelCount,\r\n          echoCancellation: allowEchoCancellation?undefined:false\r\n        },\r\n        video: false,\r\n      }\r\n\r\n    } else {\r\n\r\n      // TODO default constraints or error Browser not supported\r\n    }\r\n\r\n\r\n\r\n    console.debug(\"Audio capture, AGC: \"+this.agcStatus)\r\n\r\n    let ump = navigator.mediaDevices.getUserMedia(msc);\r\n    ump.then((s) => {\r\n\r\n      if (this.context) {\r\n      this.stream = s;\r\n\r\n      let aTracks = s.getAudioTracks();\r\n\r\n      for (let i = 0; i < aTracks.length; i++) {\r\n        let aTrack = aTracks[i];\r\n\r\n        console.info(\"Track audio info: id: \" + aTrack.id + \" kind: \" + aTrack.kind + \" label: \\\"\" + aTrack.label + \"\\\"\");\r\n        let mtrSts = aTrack.getSettings();\r\n\r\n        // Typescript lib.dom.ts MediaTrackSettings.channelCount is missing\r\n        // https://github.com/mdn/browser-compat-data/blob/5493d8f937e05b2ddbd41b99f5bdfad4a1f2ed85/api/MediaTrackSettings.json\r\n        //@ts-ignore\r\n        console.info(\"Track audio settings: Ch cnt: \" + mtrSts.channelCount + \", AGC: \" + mtrSts.autoGainControl + \", Echo cancell.: \" + mtrSts.echoCancellation);\r\n        if (mtrSts.autoGainControl) {\r\n          this.agcStatus = mtrSts.autoGainControl;\r\n        }\r\n\r\n        console.debug(\"Echo cancellation: \"+mtrSts.echoCancellation);\r\n\r\n      }\r\n\r\n      let vTracks = s.getVideoTracks();\r\n      for (let i = 0; i < vTracks.length; i++) {\r\n        let vTrack = vTracks[i];\r\n        console.info(\"Track video info: id: \" + vTrack.id + \" kind: \" + vTrack.kind + \" label: \" + vTrack.label);\r\n      }\r\n      this.mediaStream = this.context.createMediaStreamSource(s);\r\n      // stream channel count ( is always 2 !)\r\n      let streamChannelCount: number = this.mediaStream.channelCount;\r\n      console.info(\"Stream channel count: \" + streamChannelCount);\r\n      // is not set!!\r\n      //this.currentSampleRate = this.mediaStream.sampleRate;\r\n      this.currentSampleRate = this.context.sampleRate;\r\n      console.info(\"Source audio node: channels: \" + streamChannelCount + \" samplerate: \" + this.currentSampleRate);\r\n      if (this.audioOutStream) {\r\n        this.audioOutStream.setFormat(this.channelCount, this.currentSampleRate);\r\n      }\r\n      // W3C  -> new name is createScriptProcessor\r\n      //\r\n      // Again deprecated, but AudioWorker not yet implemented in stable releases (June 2016)\r\n      // AudioWorker is now AudioWorkletProcessor ... (May 2017)\r\n\r\n      // Update 12-2020:\r\n      // The ScriptProcessorNode Interface - DEPRECATED\r\n\r\n      // Update 06-2021\r\n      //  AudioWorkletProcessor is here to stay. Web Audio API has now Recommendation status !\r\n\r\n      if (this.context.audioWorklet) {\r\n        //const workletFileName = ('file-loader!./interceptor_worklet.js');\r\n        //const workletFileName = 'http://localhost:4200/assets/interceptor_worklet.js';\r\n        //console.log(awpStr);\r\n        if (AudioCapture.captureInterceptorModuleRegistered) {\r\n          // Required capture interceptor module already registered\r\n          this.addCaptureInterceptor();\r\n        } else {\r\n\r\n          // Register capture interceptor module\r\n          let audioWorkletModuleBlob = new Blob([awpStr], {type: 'text/javascript'});\r\n\r\n          let audioWorkletModuleBlobUrl = window.URL.createObjectURL(audioWorkletModuleBlob);\r\n\r\n          this.context.audioWorklet.addModule(audioWorkletModuleBlobUrl).then(() => {\r\n                AudioCapture.captureInterceptorModuleRegistered = true;\r\n                this.addCaptureInterceptor();\r\n              }\r\n          ).catch((error: any) => {\r\n            console.log('Could not add module ' + error);\r\n          });\r\n        }\r\n      } else if (this.context.createScriptProcessor) {\r\n        // The ScriptProcessorNode Interface - DEPRECATED Only as fallback\r\n        // TODO should we use streamChannelCount or channelCount here ?\r\n        let scriptProcessorNode = this.context.createScriptProcessor(AudioCapture.BUFFER_SIZE, streamChannelCount, streamChannelCount);\r\n        this.bufferingNode = scriptProcessorNode;\r\n        let c = 0;\r\n        if (scriptProcessorNode.onaudioprocess) {\r\n          scriptProcessorNode.onaudioprocess = (e: AudioProcessingEvent) => {\r\n\r\n            if (this.capturing) {\r\n              let inBuffer = e.inputBuffer;\r\n              // only process requested count of channels\r\n              let currentBuffers = new Array<Float32Array>(channelCount);\r\n              for (let ch: number = 0; ch < channelCount; ch++) {\r\n                let chSamples = inBuffer.getChannelData(ch);\r\n                let chSamplesCopy = chSamples.slice(0);\r\n                currentBuffers[ch] = chSamplesCopy.slice(0);\r\n                if (this.data) {\r\n                  this.data[ch].push(chSamplesCopy);\r\n                }\r\n                if (DEBUG_TRACE_LEVEL > 8) {\r\n                  console.debug(\"Process \" + chSamplesCopy.length + \" samples.\");\r\n                }\r\n                this.framesRecorded += chSamplesCopy.length;\r\n              }\r\n              c++;\r\n              if (this.audioOutStream) {\r\n                this.audioOutStream.write(currentBuffers);\r\n              }\r\n            }\r\n          };\r\n          this._opened = true;\r\n          if (this.listener) {\r\n            this.listener.opened();\r\n          }\r\n        } else {\r\n          this.listener.error('Browser does not support audio processing (ScriptProcessor.onaudioprocess method not found)!');\r\n        }\r\n      } else {\r\n        this.listener.error('Browser does not support audio processing (neither AudioWorkletProcessor nor ScriptProcessor)!');\r\n      }\r\n    }\r\n        }, (e) => {\r\n          console.error(e + \" Error name: \" +e.name);\r\n          if (this.listener) {\r\n            if('NotAllowedError' === e.name){\r\n              this.listener.error('Not allowed to use your microphone.','Please make sure that microphone access is allowed for this web page and reload the page.');\r\n            }else if('NotReadableError' === e.name){\r\n              this.listener.error('Could not read from your audio device.','Please make sure your audio device is working.');\r\n            }else if('OverconstrainedError' === e.name){\r\n              let eMsg=e.msg?e.msg:'Overconstrained media device request error.';\r\n              this.listener.error(eMsg);\r\n            } else {\r\n              this.listener.error();\r\n            }\r\n          }\r\n        }\r\n    )\r\n  }\r\n\r\n  private _start(){\r\n    if(this.context) {\r\n      this.initData();\r\n      if (this.audioOutStream) {\r\n        this.audioOutStream.nextStream()\r\n      }\r\n      this.capturing = true;\r\n      if (this.bufferingNode) {\r\n        this.mediaStream.connect(this.bufferingNode);\r\n        this.bufferingNode.connect(this.context.destination);\r\n      }\r\n      if (this.listener) {\r\n        this.listener.started();\r\n      }\r\n    }\r\n  }\r\n\r\n  start() {\r\n    if(this.context) {\r\n      const aSt = this.context.state;\r\n      if (aSt === 'running') {\r\n        this._start();\r\n      } else {\r\n        console.debug(\"Capture start: audio context not running, state: \" + aSt + \", resuming...\");\r\n        this.context.resume().then(() => {\r\n          console.debug(\"Capture start: audio context resumed, starting...\");\r\n          this._start();\r\n        })\r\n      }\r\n    }\r\n  }\r\n\r\n  stop() {\r\n\r\n    if (this.disconnectStreams && this.bufferingNode) {\r\n      this.mediaStream.disconnect(this.bufferingNode);\r\n      if(this.context) {\r\n        this.bufferingNode.disconnect(this.context.destination);\r\n      }\r\n    }\r\n\r\n    try {\r\n      if (this.audioOutStream) {\r\n        this.audioOutStream.flush();\r\n      }\r\n    }catch(err){\r\n      console.error(\"Could not flush capture stream.\");\r\n     throw err;\r\n    }finally {\r\n      this.capturing = false;\r\n      if (this.inddbAudioBuffer && this.persistError) {\r\n        // Delete invalid persistent audio data and hope that it will be completely uploaded to the server\r\n        this.inddbAudioBuffer.releaseAudioData();\r\n        this.inddbAudioBuffer = null;\r\n      }\r\n      if (this.listener && (this.persistError || this.persisted)) {\r\n        //console.debug(\"Stopped by stop() method call\");\r\n        this.listener.stopped();\r\n      }\r\n    }\r\n  }\r\n\r\n\r\n  store(){\r\n   // if(this.db && this.recUUID){\r\n   //    let tr= this.db.transaction(SprDb.RECORDING_FILE_CHUNKS_OBJECT_STORE_NAME, 'readwrite');\r\n   //    let recFileObjStore = tr.objectStore(SprDb.RECORDING_FILE_CHUNKS_OBJECT_STORE_NAME);\r\n   //\r\n   //      try {\r\n   //        let ch0Data=this.data[0];\r\n   //        let dataChkCnt=ch0Data.length;\r\n   //          let pos = 0;\r\n   //          for (let chCkIdx = 0; chCkIdx < dataChkCnt; chCkIdx++) {\r\n   //            let bufLen=0;\r\n   //            for (let ch = 0; ch < this.channelCount; ch++) {\r\n   //              let chChk = this.data[ch][chCkIdx];\r\n   //              bufLen = chChk.length;\r\n   //              //let cacheId = uuid + '_' + ch + '_' + chCkIdx;\r\n   //              let chkDbId = [this.recUUID, this.indDbChkIdx + chCkIdx, ch];\r\n   //              let cr = recFileObjStore.add(chChk, chkDbId);\r\n   //              //console.debug(\"Added: \"+ch+\" \"+(this.indDbChkIdx+chCkIdx));\r\n   //              cr.onsuccess=()=>{\r\n   //                //console.debug(\"Stored audio data to indexed db\");\r\n   //              }\r\n   //              cr.onerror=()=>{\r\n   //                console.error(\"Error storing audio data to indexed db\");\r\n   //              }\r\n   //            }\r\n   //            pos += bufLen;\r\n   //        }\r\n   //        this.indDbChkIdx+=dataChkCnt;\r\n   //\r\n   //        tr.onerror = (err) => {\r\n   //          console.error('Failed to cache audio data to indexed db: ' + err)\r\n   //        }\r\n   //        tr.oncomplete = () => {\r\n   //          //console.debug('Transferred capture audio data to indexed db, deleting original data from memory...');\r\n   //\r\n   //          /// Audio data saved to index db delete from in memory data array\r\n   //          for (let ch = 0; ch < this.channelCount; ch++) {\r\n   //           this.data[ch].splice(0);\r\n   //            //console.debug(\"Spliced/removed ch: \"+ch);\r\n   //          }\r\n   //\r\n   //          this.persisted=true;\r\n   //          if(this.listener && !this.capturing){\r\n   //            //console.debug(\"Stopped by indexed db hook\");\r\n   //            this.listener.stopped();\r\n   //          }\r\n   //        }\r\n   //        // Commit chunks\r\n   //        this.persisted=false;\r\n   //        tr.commit();\r\n   //      } catch (err) {\r\n   //        console.error('Transfer capture audio data error: '+err);\r\n   //      }\r\n   //    }\r\n\r\n    // if(!this.inddbAudioBuffer && this._persistentAudioStorageTarget && this.recUUID) {\r\n    //   this.inddbAudioBuffer = new IndexedDbAudioBuffer(this._persistentAudioStorageTarget, this.channelCount,this.currentSampleRate,AudioCapture.BUFFER_SIZE,0,this.recUUID)\r\n    // }\r\n    if(this.inddbAudioBuffer && this.data){\r\n\r\n      // Try to append to\r\n      this.inddbAudioBuffer.appendRawAudioData(this.data).subscribe({\r\n        complete: () => {\r\n          //console.debug('Transferred capture audio data to indexed db, deleting original data from memory...');\r\n\r\n          /// Audio data saved to index db delete from in memory data array\r\n          if(this.data) {\r\n            for (let ch = 0; ch < this.channelCount; ch++) {\r\n              this.data[ch].splice(0);\r\n              //console.debug(\"Spliced/removed ch: \"+ch);\r\n            }\r\n          }\r\n          this.persisted = true;\r\n          if (this.listener && !this.capturing) {\r\n            //console.debug(\"Stopped by indexed db hook\");\r\n            this.listener.stopped();\r\n          }\r\n        },error:(err)=>{\r\n          // Only log the first error\r\n          if(!this.persistError) {\r\n            this.persistError = err;\r\n            console.error(\"Error persisting audio data: \" + err);\r\n          }\r\n        }\r\n      });\r\n      this.persisted=false;\r\n    }\r\n  }\r\n\r\n\r\n  close() {\r\n    if(this.mediaStream) {\r\n      this.mediaStream.disconnect();\r\n    }\r\n    if (this.stream) {\r\n      const mts = this.stream.getTracks();\r\n      for (let i = 0; i < mts.length; i++) {\r\n        mts[i].stop();\r\n      }\r\n    }\r\n    this._opened=false;\r\n  }\r\n\r\n  audioBuffer(): AudioBuffer |null{\r\n    let ab: AudioBuffer|null=null;\r\n    if(this.context && this.data) {\r\n      let frameLen: number = 0;\r\n\r\n      let ch0Data = this.data[0];\r\n\r\n      for (let ch0Chk of ch0Data) {\r\n        frameLen += ch0Chk.length;\r\n      }\r\n\r\n      try {\r\n        ab = this.context.createBuffer(this.channelCount, frameLen, this.context.sampleRate);\r\n      } catch (err) {\r\n        if (err instanceof DOMException) {\r\n          if (err.name === 'NotSupportedError') {\r\n            if (frameLen == 0) {\r\n              // Empty buffers are not supported by Chromium\r\n              // Create dummy buffer with one sample\r\n              ab = this.context.createBuffer(this.channelCount, 1, this.context.sampleRate);\r\n            } else {\r\n              throw err;\r\n            }\r\n          } else {\r\n            throw err;\r\n          }\r\n        } else if (err instanceof RangeError) {\r\n          // Out of memory\r\n          // TODO What to do ??\r\n          throw err;\r\n        } else {\r\n          throw err;\r\n        }\r\n      }\r\n      for (let ch = 0; ch < this.channelCount; ch++) {\r\n        let chD = ab.getChannelData(ch);\r\n        let pos = 0;\r\n        for (let chChk of this.data[ch]) {\r\n          let bufLen = chChk.length;\r\n          chD.set(chChk, pos);\r\n          pos += bufLen;\r\n        }\r\n      }\r\n    }\r\n    return ab;\r\n  }\r\n\r\n  audioBufferArray():ArrayAudioBuffer|null{\r\n      let arrAb:ArrayAudioBuffer|null=null;\r\n      if(this.data) {\r\n        arrAb=new ArrayAudioBuffer(this.channelCount, this.currentSampleRate, this.data);\r\n      }\r\n      return arrAb;\r\n  }\r\n\r\n  inddbAudioBufferArray():IndexedDbAudioBuffer|null{\r\n    if(this.persistError){\r\n      return null;\r\n    }else {\r\n      return this.inddbAudioBuffer;\r\n    }\r\n  }\r\n\r\n\r\n}\r\n\r\n"]}
|