speechrecorderng 3.0.0 → 3.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2020/lib/action/action.mjs +2 -1
- package/esm2020/lib/audio/array_audio_buffer.mjs +65 -2
- package/esm2020/lib/audio/array_audio_buffer_input_stream.mjs +2 -2
- package/esm2020/lib/audio/array_audio_buffer_random_access_stream.mjs +16 -0
- package/esm2020/lib/audio/audio_data_holder.mjs +203 -48
- package/esm2020/lib/audio/audio_display.mjs +10 -34
- package/esm2020/lib/audio/audio_player.mjs +18 -45
- package/esm2020/lib/audio/capture/capture.mjs +293 -69
- package/esm2020/lib/audio/dsp/level_measure.mjs +211 -88
- package/esm2020/lib/audio/impl/wavformat.mjs +1 -1
- package/esm2020/lib/audio/impl/wavreader.mjs +134 -0
- package/esm2020/lib/audio/impl/wavwriter.mjs +10 -9
- package/esm2020/lib/audio/inddb_audio_buffer.mjs +508 -0
- package/esm2020/lib/audio/net_audio_buffer.mjs +297 -0
- package/esm2020/lib/audio/persistor.mjs +8 -2
- package/esm2020/lib/audio/playback/array_audio_buffer_source_node.mjs +15 -154
- package/esm2020/lib/audio/playback/audio_source_node.mjs +18 -0
- package/esm2020/lib/audio/playback/audio_source_worklet_module_loader.mjs +167 -0
- package/esm2020/lib/audio/playback/inddb_audio_buffer_source_node.mjs +167 -0
- package/esm2020/lib/audio/playback/net_audio_buffer_source_node.mjs +218 -0
- package/esm2020/lib/audio/playback/player.mjs +190 -171
- package/esm2020/lib/audio/ui/audio_canvas_layer_comp.mjs +35 -76
- package/esm2020/lib/audio/ui/audio_display_control.mjs +12 -24
- package/esm2020/lib/audio/ui/audio_display_scroll_pane.mjs +14 -45
- package/esm2020/lib/audio/ui/audiosignal.mjs +333 -267
- package/esm2020/lib/audio/ui/container.mjs +40 -57
- package/esm2020/lib/audio/ui/livelevel.mjs +53 -52
- package/esm2020/lib/audio/ui/scroll_pane_horizontal.mjs +5 -19
- package/esm2020/lib/audio/ui/sonagram.mjs +386 -339
- package/esm2020/lib/db/inddb.mjs +120 -0
- package/esm2020/lib/io/BinaryReader.mjs +85 -0
- package/esm2020/lib/io/stream.mjs +101 -1
- package/esm2020/lib/net/uploader.mjs +111 -5
- package/esm2020/lib/recorder_component.mjs +59 -1
- package/esm2020/lib/speechrecorder/project/project.mjs +10 -1
- package/esm2020/lib/speechrecorder/project/project.service.mjs +3 -3
- package/esm2020/lib/speechrecorder/recording.mjs +30 -6
- package/esm2020/lib/speechrecorder/recordings/basic_recording.service.mjs +213 -0
- package/esm2020/lib/speechrecorder/recordings/recordings.service.mjs +732 -65
- package/esm2020/lib/speechrecorder/script/script.service.mjs +3 -3
- package/esm2020/lib/speechrecorder/session/audiorecorder.mjs +358 -184
- package/esm2020/lib/speechrecorder/session/basicrecorder.mjs +181 -28
- package/esm2020/lib/speechrecorder/session/controlpanel.mjs +47 -129
- package/esm2020/lib/speechrecorder/session/item.mjs +13 -1
- package/esm2020/lib/speechrecorder/session/progress.mjs +26 -55
- package/esm2020/lib/speechrecorder/session/prompting.mjs +33 -176
- package/esm2020/lib/speechrecorder/session/recorder_combi_pane.mjs +6 -6
- package/esm2020/lib/speechrecorder/session/recording_file_cache.mjs +28 -15
- package/esm2020/lib/speechrecorder/session/recording_list.mjs +92 -55
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +9 -13
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs +9 -18
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +10 -36
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +10 -37
- package/esm2020/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +80 -56
- package/esm2020/lib/speechrecorder/session/session.service.mjs +3 -3
- package/esm2020/lib/speechrecorder/session/session_finished_dialog.mjs +4 -4
- package/esm2020/lib/speechrecorder/session/sessionmanager.mjs +409 -165
- package/esm2020/lib/speechrecorder/session/warning_bar.mjs +6 -29
- package/esm2020/lib/speechrecorder/spruploader.mjs +3 -3
- package/esm2020/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.mjs +6 -41
- package/esm2020/lib/speechrecorderng.component.mjs +42 -31
- package/esm2020/lib/speechrecorderng.module.mjs +5 -5
- package/esm2020/lib/spr.config.mjs +3 -3
- package/esm2020/lib/spr.module.version.mjs +2 -2
- package/esm2020/lib/ui/canvas_layer_comp.mjs +3 -3
- package/esm2020/lib/ui/message_dialog.mjs +7 -7
- package/esm2020/lib/ui/recordingitem_display.mjs +26 -59
- package/esm2020/lib/utils/scrollIntoViewToBottom.mjs +3 -3
- package/esm2020/lib/utils/ua-parser.mjs +28 -4
- package/esm2020/lib/utils/utils.mjs +29 -1
- package/fesm2015/speechrecorderng.mjs +12755 -9195
- package/fesm2015/speechrecorderng.mjs.map +1 -1
- package/fesm2020/speechrecorderng.mjs +12652 -9101
- package/fesm2020/speechrecorderng.mjs.map +1 -1
- package/{speechrecorderng.d.ts → index.d.ts} +0 -0
- package/lib/audio/array_audio_buffer.d.ts +17 -1
- package/lib/audio/array_audio_buffer_random_access_stream.d.ts +9 -0
- package/lib/audio/audio_data_holder.d.ts +69 -14
- package/lib/audio/audio_display.d.ts +1 -1
- package/lib/audio/audio_player.d.ts +3 -7
- package/lib/audio/capture/capture.d.ts +24 -4
- package/lib/audio/dsp/level_measure.d.ts +2 -23
- package/lib/audio/impl/wavformat.d.ts +3 -3
- package/lib/audio/impl/wavreader.d.ts +16 -0
- package/lib/audio/impl/wavwriter.d.ts +1 -4
- package/lib/audio/inddb_audio_buffer.d.ts +68 -0
- package/lib/audio/net_audio_buffer.d.ts +59 -0
- package/lib/audio/persistor.d.ts +3 -9
- package/lib/audio/playback/array_audio_buffer_source_node.d.ts +2 -8
- package/lib/audio/playback/audio_source_node.d.ts +10 -0
- package/lib/audio/playback/audio_source_worklet_module_loader.d.ts +4 -0
- package/lib/audio/playback/inddb_audio_buffer_source_node.d.ts +21 -0
- package/lib/audio/playback/net_audio_buffer_source_node.d.ts +27 -0
- package/lib/audio/playback/player.d.ts +19 -16
- package/lib/audio/ui/audio_canvas_layer_comp.d.ts +3 -3
- package/lib/audio/ui/audio_display_control.d.ts +3 -6
- package/lib/audio/ui/audio_display_scroll_pane.d.ts +1 -1
- package/lib/audio/ui/audiosignal.d.ts +4 -3
- package/lib/audio/ui/container.d.ts +1 -1
- package/lib/audio/ui/livelevel.d.ts +11 -4
- package/lib/audio/ui/scroll_pane_horizontal.d.ts +1 -1
- package/lib/audio/ui/sonagram.d.ts +4 -3
- package/lib/db/inddb.d.ts +21 -0
- package/lib/io/BinaryReader.d.ts +18 -0
- package/lib/io/stream.d.ts +18 -0
- package/lib/net/uploader.d.ts +20 -2
- package/lib/recorder_component.d.ts +5 -0
- package/lib/speechrecorder/project/project.d.ts +9 -0
- package/lib/speechrecorder/recording.d.ts +8 -2
- package/lib/speechrecorder/recordings/basic_recording.service.d.ts +27 -0
- package/lib/speechrecorder/recordings/recordings.service.d.ts +21 -6
- package/lib/speechrecorder/session/audiorecorder.d.ts +7 -18
- package/lib/speechrecorder/session/basicrecorder.d.ts +28 -4
- package/lib/speechrecorder/session/controlpanel.d.ts +7 -7
- package/lib/speechrecorder/session/item.d.ts +1 -0
- package/lib/speechrecorder/session/progress.d.ts +2 -1
- package/lib/speechrecorder/session/prompting.d.ts +5 -5
- package/lib/speechrecorder/session/recorder_combi_pane.d.ts +1 -1
- package/lib/speechrecorder/session/recording_file_cache.d.ts +7 -4
- package/lib/speechrecorder/session/recording_list.d.ts +2 -1
- package/lib/speechrecorder/session/recordingfile/recording-file-meta.component.d.ts +1 -1
- package/lib/speechrecorder/session/recordingfile/recording-file-navi.component.d.ts +1 -1
- package/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.d.ts +1 -1
- package/lib/speechrecorder/session/recordingfile/recording-file-view.component.d.ts +1 -1
- package/lib/speechrecorder/session/recordingfile/recordingfile-service.d.ts +4 -5
- package/lib/speechrecorder/session/session_finished_dialog.d.ts +1 -1
- package/lib/speechrecorder/session/sessionmanager.d.ts +10 -10
- package/lib/speechrecorder/session/warning_bar.d.ts +1 -1
- package/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.d.ts +1 -1
- package/lib/speechrecorderng.component.d.ts +2 -1
- package/lib/spr.module.version.d.ts +1 -1
- package/lib/ui/canvas_layer_comp.d.ts +1 -1
- package/lib/ui/message_dialog.d.ts +1 -1
- package/lib/ui/recordingitem_display.d.ts +3 -2
- package/lib/utils/scrollIntoViewToBottom.d.ts +1 -1
- package/lib/utils/ua-parser.d.ts +4 -1
- package/lib/utils/utils.d.ts +6 -0
- package/package.json +5 -4
- package/esm2020/lib/math/utils.mjs +0 -14
- package/esm2020/lib/utils/css_utils.mjs +0 -7
- package/lib/math/utils.d.ts +0 -3
- package/lib/utils/css_utils.d.ts +0 -3
|
@@ -0,0 +1,213 @@
|
|
|
1
|
+
import { __decorate, __param } from "tslib";
|
|
2
|
+
import { HttpHeaders } from "@angular/common/http";
|
|
3
|
+
import { Inject } from "@angular/core";
|
|
4
|
+
import { ApiType, SPEECHRECORDER_CONFIG } from "../../spr.config";
|
|
5
|
+
import { Observable } from "rxjs";
|
|
6
|
+
import { NetAudioBuffer } from "../../audio/net_audio_buffer";
|
|
7
|
+
import { UUID } from "../../utils/utils";
|
|
8
|
+
import { WavReader } from "../../audio/impl/wavreader";
|
|
9
|
+
export class ChunkDownload {
|
|
10
|
+
constructor(_orgPCMAudioFormat, _orgFrameLength, _decodedAudioBuffer) {
|
|
11
|
+
this._orgPCMAudioFormat = _orgPCMAudioFormat;
|
|
12
|
+
this._orgFrameLength = _orgFrameLength;
|
|
13
|
+
this._decodedAudioBuffer = _decodedAudioBuffer;
|
|
14
|
+
}
|
|
15
|
+
get orgPCMAudioFormat() {
|
|
16
|
+
return this._orgPCMAudioFormat;
|
|
17
|
+
}
|
|
18
|
+
get orgFrameLength() {
|
|
19
|
+
return this._orgFrameLength;
|
|
20
|
+
}
|
|
21
|
+
get decodedAudioBuffer() {
|
|
22
|
+
return this._decodedAudioBuffer;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
let BasicRecordingService = class BasicRecordingService {
|
|
26
|
+
constructor(http, config) {
|
|
27
|
+
this.http = http;
|
|
28
|
+
this.config = config;
|
|
29
|
+
this._maxAutoNetMemStoreSamples = BasicRecordingService.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;
|
|
30
|
+
this.withCredentials = false;
|
|
31
|
+
this.apiEndPoint = '';
|
|
32
|
+
if (config && config.apiEndPoint) {
|
|
33
|
+
this.apiEndPoint = config.apiEndPoint;
|
|
34
|
+
}
|
|
35
|
+
if (this.apiEndPoint !== '') {
|
|
36
|
+
this.apiEndPoint = this.apiEndPoint + '/';
|
|
37
|
+
}
|
|
38
|
+
if (config != null && config.withCredentials != null) {
|
|
39
|
+
this.withCredentials = config.withCredentials;
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
audioRequestByURL(audioBaseUrl, audioURLSearchParams) {
|
|
43
|
+
let audioUrl = audioBaseUrl;
|
|
44
|
+
if (this.config && this.config.apiType === ApiType.FILES) {
|
|
45
|
+
// for development and demo
|
|
46
|
+
audioUrl = audioUrl + '.wav';
|
|
47
|
+
// append UUID to make request URL unique to avoid localhost server caching
|
|
48
|
+
audioURLSearchParams.set('requestUUID', UUID.generate());
|
|
49
|
+
//audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();
|
|
50
|
+
}
|
|
51
|
+
audioUrl = audioUrl + '?' + audioURLSearchParams.toString();
|
|
52
|
+
let headers = new HttpHeaders();
|
|
53
|
+
headers = headers.set('Accept', 'audio/wav');
|
|
54
|
+
return this.http.get(audioUrl, {
|
|
55
|
+
headers: headers,
|
|
56
|
+
observe: 'response',
|
|
57
|
+
responseType: 'arraybuffer',
|
|
58
|
+
withCredentials: this.withCredentials
|
|
59
|
+
});
|
|
60
|
+
}
|
|
61
|
+
chunkAudioRequest(aCtx, baseAudioUrl, startFrame = 0, frameLength) {
|
|
62
|
+
let ausps = new URLSearchParams();
|
|
63
|
+
ausps.set('startFrame', startFrame.toString());
|
|
64
|
+
ausps.set('frameLength', frameLength.toString());
|
|
65
|
+
if (this.config && this.config.apiType === ApiType.FILES) {
|
|
66
|
+
// for development and demo
|
|
67
|
+
// append UUID to make request URL unique to avoid localhost server caching
|
|
68
|
+
//audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();
|
|
69
|
+
ausps.set('requestUUID', UUID.generate());
|
|
70
|
+
}
|
|
71
|
+
let obs = new Observable(observer => {
|
|
72
|
+
this.audioRequestByURL(baseAudioUrl, ausps).subscribe(resp => {
|
|
73
|
+
// Do not use Promise version, which does not work with Safari 13 (13.0.5)
|
|
74
|
+
if (resp.body) {
|
|
75
|
+
//console.debug("chunkAudioRequest: observer.closed: "+observer.closed);
|
|
76
|
+
//console.debug("Audio file bytes: "+resp.body.byteLength);
|
|
77
|
+
// Check original audio format
|
|
78
|
+
let wr = new WavReader(resp.body);
|
|
79
|
+
const pcmFmt = wr.readFormat();
|
|
80
|
+
const orgFl = wr.frameLength();
|
|
81
|
+
// if(pcmFmt){
|
|
82
|
+
// console.debug("Original WAVE format of download chunk: "+pcmFmt);
|
|
83
|
+
// }else{
|
|
84
|
+
// console.error("Could not read WAVE format of original download chunk!");
|
|
85
|
+
// }
|
|
86
|
+
// if(orgFl){
|
|
87
|
+
// console.debug("Original frame length of download chunk: "+orgFl);
|
|
88
|
+
// }else{
|
|
89
|
+
// console.error("Could not read WAVE format of original download chunk!");
|
|
90
|
+
// }
|
|
91
|
+
if (pcmFmt && orgFl) {
|
|
92
|
+
aCtx.decodeAudioData(resp.body, ab => {
|
|
93
|
+
//console.debug("Decoded audio chunk frames: "+ab.length);
|
|
94
|
+
let chDl = new ChunkDownload(pcmFmt, orgFl, ab);
|
|
95
|
+
observer.next(chDl);
|
|
96
|
+
observer.complete();
|
|
97
|
+
}, error => {
|
|
98
|
+
//if(error instanceof HttpErrorResponse) {
|
|
99
|
+
// if (error.status == 404) {
|
|
100
|
+
// // Interpret not as an error, the file ist not recorded yet
|
|
101
|
+
// observer.next(null);
|
|
102
|
+
// observer.complete()
|
|
103
|
+
// } else {
|
|
104
|
+
// // all other states are errors
|
|
105
|
+
console.error("Recordings service chunkAudioRequest error decoding audio data: " + error.name + ": " + error.message);
|
|
106
|
+
observer.error(error);
|
|
107
|
+
// }
|
|
108
|
+
// }
|
|
109
|
+
});
|
|
110
|
+
}
|
|
111
|
+
else {
|
|
112
|
+
const errMsg = 'Could not parse audio header for format and/or frame length of download.';
|
|
113
|
+
console.error(errMsg);
|
|
114
|
+
observer.error(errMsg);
|
|
115
|
+
}
|
|
116
|
+
}
|
|
117
|
+
else {
|
|
118
|
+
const errMsg = 'Fetching audio file: response has no body';
|
|
119
|
+
console.error(errMsg);
|
|
120
|
+
observer.error(errMsg);
|
|
121
|
+
}
|
|
122
|
+
}, (error) => {
|
|
123
|
+
// all other states are errors
|
|
124
|
+
//const errMsg='Fetching audio file HTTP error: '+error;
|
|
125
|
+
//console.error(errMsg);
|
|
126
|
+
observer.error(error);
|
|
127
|
+
//observer.complete();
|
|
128
|
+
});
|
|
129
|
+
});
|
|
130
|
+
return obs;
|
|
131
|
+
}
|
|
132
|
+
chunkAudioRequestToNetAudioBuffer(aCtx, baseAudioUrl, startFrame = 0, orgSampleRate, seconds, frames) {
|
|
133
|
+
//let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;
|
|
134
|
+
//let audioUrl=new URL(baseAudioUrl);
|
|
135
|
+
// if(orgSampleRate!=null && frameLength%orgSampleRate>0){
|
|
136
|
+
// const errMsg='frameLength must be equal or multiple of original samplerate.';
|
|
137
|
+
// console.error(errMsg+' ('+frameLength+'%'+orgSampleRate+'=='+(frameLength%orgSampleRate)+')');
|
|
138
|
+
// throw Error(errMsg)
|
|
139
|
+
// }
|
|
140
|
+
let frameLength = orgSampleRate * Math.round(seconds);
|
|
141
|
+
let ausps = new URLSearchParams();
|
|
142
|
+
ausps.set('startFrame', startFrame.toString());
|
|
143
|
+
ausps.set('frameLength', frameLength.toString());
|
|
144
|
+
if (this.config && this.config.apiType === ApiType.FILES) {
|
|
145
|
+
// for development and demo
|
|
146
|
+
// append UUID to make request URL unique to avoid localhost server caching
|
|
147
|
+
//audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();
|
|
148
|
+
ausps.set('requestUUID', UUID.generate());
|
|
149
|
+
}
|
|
150
|
+
let obs = new Observable(subscriber => {
|
|
151
|
+
this.audioRequestByURL(baseAudioUrl, ausps).subscribe({ next: (resp) => {
|
|
152
|
+
// Do not use Promise version, which does not work with Safari 13 (13.0.5)
|
|
153
|
+
if (resp.body) {
|
|
154
|
+
//console.debug("chunkAudioRequestTonetAb: subscriber.closed: "+subscriber.closed);
|
|
155
|
+
//console.debug("chunkAudioRequestTonetAb: Audio file bytes: "+resp.body.byteLength);
|
|
156
|
+
aCtx.decodeAudioData(resp.body, ab => {
|
|
157
|
+
//console.debug("chunkAudioRequestTonetAb: Decoded audio chunk frames for netAb: "+ab.length);
|
|
158
|
+
//console.debug("chunkAudioRequestTonetAb: Create netAb ab from chunk ab...");
|
|
159
|
+
if (frames === null) {
|
|
160
|
+
subscriber.error(new Error('Could not get frame length from recording file object'));
|
|
161
|
+
}
|
|
162
|
+
else {
|
|
163
|
+
let fl = frames;
|
|
164
|
+
if (ab.sampleRate !== orgSampleRate) {
|
|
165
|
+
if (orgSampleRate && frames) {
|
|
166
|
+
fl = Math.round(ab.sampleRate * frames / orgSampleRate);
|
|
167
|
+
//console.debug("Platform sr: "+ab.sampleRate+", file sr: "+orgSampleRate+", decoded/org frame length: "+fl+"/"+frames+", ab.length: "+ab.length);
|
|
168
|
+
}
|
|
169
|
+
}
|
|
170
|
+
let nab = NetAudioBuffer.fromChunkAudioBuffer(aCtx, this, baseAudioUrl, ab, fl, frameLength);
|
|
171
|
+
//let rp=new ReadyProvider();
|
|
172
|
+
//nab.readyProvider=rp;
|
|
173
|
+
//rp.ready();
|
|
174
|
+
nab.ready();
|
|
175
|
+
if (nab.frameLen < frameLength) {
|
|
176
|
+
//console.debug("chunkAudioRequestTonetAb: Built netAb ab from chunk ab: First chunk shorter tha frameLength ("+netAbAudioBuffer.frameLen+"<"+frameLength+"), assuming end of data, sealing netAb ab.");
|
|
177
|
+
nab.seal();
|
|
178
|
+
}
|
|
179
|
+
subscriber.next(nab);
|
|
180
|
+
subscriber.complete();
|
|
181
|
+
}
|
|
182
|
+
}, error => {
|
|
183
|
+
console.error('chunkAudioRequestToNetAb: error: ' + error);
|
|
184
|
+
//if(error instanceof HttpErrorResponse) {
|
|
185
|
+
subscriber.error(error);
|
|
186
|
+
//}
|
|
187
|
+
});
|
|
188
|
+
}
|
|
189
|
+
else {
|
|
190
|
+
console.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');
|
|
191
|
+
subscriber.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');
|
|
192
|
+
}
|
|
193
|
+
},
|
|
194
|
+
error: (error) => {
|
|
195
|
+
console.error('chunkAudioRequestToNetAb: error: ' + error);
|
|
196
|
+
subscriber.error(error);
|
|
197
|
+
//subscriber.complete();
|
|
198
|
+
}
|
|
199
|
+
});
|
|
200
|
+
});
|
|
201
|
+
return obs;
|
|
202
|
+
}
|
|
203
|
+
};
|
|
204
|
+
// iPad 9th generation, iOS 15.7.1, sometimes:
|
|
205
|
+
// Failed to load resource: WebKit hat einen internen Fehler festgestellt
|
|
206
|
+
// Firefox on Windows crashes
|
|
207
|
+
BasicRecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS = 10;
|
|
208
|
+
BasicRecordingService.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES = 2880000 * 5; // Default 5 minutes one channel at 48kHz
|
|
209
|
+
BasicRecordingService = __decorate([
|
|
210
|
+
__param(1, Inject(SPEECHRECORDER_CONFIG))
|
|
211
|
+
], BasicRecordingService);
|
|
212
|
+
export { BasicRecordingService };
|
|
213
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"basic_recording.service.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/recordings/basic_recording.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {HttpClient, HttpHeaders, HttpResponse} from \"@angular/common/http\";\r\nimport {Inject} from \"@angular/core\";\r\nimport {ApiType, SPEECHRECORDER_CONFIG, SpeechRecorderConfig} from \"../../spr.config\";\r\nimport {Observable} from \"rxjs\";\r\nimport {NetAudioBuffer} from \"../../audio/net_audio_buffer\";\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {WavReader} from \"../../audio/impl/wavreader\";\r\nimport {PCMAudioFormat} from \"../../audio/format\";\r\n\r\nexport class ChunkDownload{\r\n  get orgPCMAudioFormat(): PCMAudioFormat {\r\n    return this._orgPCMAudioFormat;\r\n  }\r\n\r\n  get orgFrameLength(): number {\r\n    return this._orgFrameLength;\r\n  }\r\n\r\n  get decodedAudioBuffer(): AudioBuffer {\r\n    return this._decodedAudioBuffer;\r\n  }\r\n  constructor(private _orgPCMAudioFormat:PCMAudioFormat,private _orgFrameLength:number,private _decodedAudioBuffer:AudioBuffer){}\r\n}\r\n\r\nexport class BasicRecordingService{\r\n  // iPad 9th generation, iOS 15.7.1, sometimes:\r\n  // Failed to load resource: WebKit hat einen internen Fehler festgestellt\r\n  // Firefox on Windows crashes\r\n  public static readonly DEFAULT_CHUNKED_DOWNLOAD_SECONDS:number=10;\r\n  public static readonly DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES:number=2880000*5; // Default 5 minutes one channel at 48kHz\r\n  protected _maxAutoNetMemStoreSamples:number=BasicRecordingService.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;\r\n    protected apiEndPoint: string;\r\n    protected withCredentials: boolean = false;\r\n    constructor(protected http: HttpClient, @Inject(SPEECHRECORDER_CONFIG) protected config?: SpeechRecorderConfig) {\r\n        this.apiEndPoint = ''\r\n\r\n        if (config && config.apiEndPoint) {\r\n            this.apiEndPoint = config.apiEndPoint;\r\n        }\r\n        if (this.apiEndPoint !== '') {\r\n            this.apiEndPoint = this.apiEndPoint + '/'\r\n        }\r\n        if (config != null && config.withCredentials != null) {\r\n            this.withCredentials = config.withCredentials;\r\n        }\r\n    }\r\n\r\n    protected audioRequestByURL(audioBaseUrl:string,audioURLSearchParams:URLSearchParams): Observable<HttpResponse<ArrayBuffer>> {\r\n        let audioUrl=audioBaseUrl;\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            audioUrl=audioUrl+'.wav';\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            audioURLSearchParams.set('requestUUID',UUID.generate());\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n        }\r\n\r\n        audioUrl=audioUrl+'?'+audioURLSearchParams.toString();\r\n\r\n        let headers = new HttpHeaders();\r\n        headers = headers.set('Accept', 'audio/wav');\r\n        return this.http.get(audioUrl, {\r\n            headers: headers,\r\n            observe: 'response',\r\n            responseType: 'arraybuffer',\r\n            withCredentials: this.withCredentials\r\n        });\r\n\r\n    }\r\n\r\n    public chunkAudioRequest(aCtx:AudioContext,baseAudioUrl:string,startFrame:number=0,frameLength:number): Observable<ChunkDownload|null> {\r\n\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n\r\n        let obs=new Observable<ChunkDownload|null>(observer=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe(resp => {\r\n                    // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                    if (resp.body) {\r\n                        //console.debug(\"chunkAudioRequest: observer.closed: \"+observer.closed);\r\n                        //console.debug(\"Audio file bytes: \"+resp.body.byteLength);\r\n\r\n                        // Check original audio format\r\n                        let wr=new WavReader(resp.body);\r\n                        const pcmFmt=wr.readFormat();\r\n                        const orgFl=wr.frameLength();\r\n                        // if(pcmFmt){\r\n                        //   console.debug(\"Original WAVE format of download chunk: \"+pcmFmt);\r\n                        // }else{\r\n                        //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                        // }\r\n                        // if(orgFl){\r\n                        //   console.debug(\"Original frame length of download chunk: \"+orgFl);\r\n                        // }else{\r\n                        //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                        // }\r\n\r\n                        if(pcmFmt && orgFl) {\r\n                            aCtx.decodeAudioData(resp.body, ab => {\r\n                                    //console.debug(\"Decoded audio chunk frames: \"+ab.length);\r\n                                    let chDl = new ChunkDownload(pcmFmt, orgFl, ab);\r\n                                    observer.next(chDl);\r\n                                    observer.complete();\r\n                                }\r\n                                , error => {\r\n                                    //if(error instanceof HttpErrorResponse) {\r\n                                    // if (error.status == 404) {\r\n                                    //   // Interpret not as an error, the file ist not recorded yet\r\n                                    //   observer.next(null);\r\n                                    //   observer.complete()\r\n                                    // } else {\r\n                                    //   // all other states are errors\r\n                                    console.error(\"Recordings service chunkAudioRequest error decoding audio data: \"+error.name+\": \"+error.message);\r\n                                    observer.error(error);\r\n                                    // }\r\n                                    // }\r\n                                });\r\n                        }else{\r\n                            const errMsg='Could not parse audio header for format and/or frame length of download.';\r\n                            console.error(errMsg);\r\n                            observer.error(errMsg);\r\n                        }\r\n                    } else {\r\n                        const errMsg='Fetching audio file: response has no body';\r\n                        console.error(errMsg);\r\n                        observer.error(errMsg);\r\n                    }\r\n                },\r\n                (error) => {\r\n                    // all other states are errors\r\n                    //const errMsg='Fetching audio file HTTP error: '+error;\r\n                    //console.error(errMsg);\r\n                    observer.error(error);\r\n                    //observer.complete();\r\n\r\n                });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n\r\n    protected chunkAudioRequestToNetAudioBuffer(aCtx: AudioContext, baseAudioUrl: string, startFrame: number = 0, orgSampleRate: number, seconds:number,frames: number | null): Observable<NetAudioBuffer | null> {\r\n        //let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;\r\n        //let audioUrl=new URL(baseAudioUrl);\r\n        // if(orgSampleRate!=null && frameLength%orgSampleRate>0){\r\n        //   const errMsg='frameLength must be equal or multiple of original samplerate.';\r\n        //   console.error(errMsg+' ('+frameLength+'%'+orgSampleRate+'=='+(frameLength%orgSampleRate)+')');\r\n        //   throw Error(errMsg)\r\n        // }\r\n        let frameLength:number=orgSampleRate*Math.round(seconds);\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n        let obs=new Observable<NetAudioBuffer|null>(subscriber=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe({next:(resp) => {\r\n                    // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                    if (resp.body) {\r\n                        //console.debug(\"chunkAudioRequestTonetAb: subscriber.closed: \"+subscriber.closed);\r\n                        //console.debug(\"chunkAudioRequestTonetAb: Audio file bytes: \"+resp.body.byteLength);\r\n                        aCtx.decodeAudioData(resp.body, ab => {\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Decoded audio chunk frames for netAb: \"+ab.length);\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Create netAb ab from chunk ab...\");\r\n                                if(frames===null){\r\n                                    subscriber.error(new Error('Could not get frame length from recording file object'));\r\n                                }else {\r\n                                    let fl = frames;\r\n                                    if (ab.sampleRate !== orgSampleRate) {\r\n                                        if (orgSampleRate && frames) {\r\n                                            fl = Math.round(ab.sampleRate * frames / orgSampleRate);\r\n                                            //console.debug(\"Platform sr: \"+ab.sampleRate+\", file sr: \"+orgSampleRate+\", decoded/org frame length: \"+fl+\"/\"+frames+\", ab.length: \"+ab.length);\r\n                                        }\r\n                                    }\r\n\r\n                                    let nab = NetAudioBuffer.fromChunkAudioBuffer(aCtx, this,baseAudioUrl, ab, fl,frameLength);\r\n                                    //let rp=new ReadyProvider();\r\n                                    //nab.readyProvider=rp;\r\n                                    //rp.ready();\r\n                                    nab.ready();\r\n                                    if (nab.frameLen < frameLength) {\r\n                                        //console.debug(\"chunkAudioRequestTonetAb: Built netAb ab from chunk ab: First chunk shorter tha frameLength (\"+netAbAudioBuffer.frameLen+\"<\"+frameLength+\"), assuming end of data, sealing netAb ab.\");\r\n                                        nab.seal();\r\n                                    }\r\n                                    subscriber.next(nab);\r\n\r\n                                    subscriber.complete();\r\n                                }\r\n                            }\r\n                            , error => {\r\n                                console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                                //if(error instanceof HttpErrorResponse) {\r\n                                subscriber.error(error);\r\n                                //}\r\n                            })\r\n                    } else {\r\n                        console.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                        subscriber.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                    }\r\n                },\r\n                error:(error) => {\r\n                    console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                    subscriber.error(error);\r\n                    //subscriber.complete();\r\n                }\r\n            });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n}\r\n"]}
|