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
|
@@ -2,35 +2,27 @@
|
|
|
2
2
|
* Created by klausj on 17.06.2017.
|
|
3
3
|
*/
|
|
4
4
|
import { Inject, Injectable } from '@angular/core';
|
|
5
|
-
import { HttpHeaders } from "@angular/common/http";
|
|
5
|
+
import { HttpErrorResponse, HttpHeaders } from "@angular/common/http";
|
|
6
6
|
import { ApiType, SPEECHRECORDER_CONFIG } from "../../spr.config";
|
|
7
7
|
import { UUID } from "../../utils/utils";
|
|
8
8
|
import { SprRecordingFile, RecordingFileUtils } from "../recording";
|
|
9
9
|
import { ProjectService } from "../project/project.service";
|
|
10
10
|
import { SessionService } from "../session/session.service";
|
|
11
|
-
import { Observable } from "rxjs";
|
|
12
|
-
import { AudioDataHolder } from "../../audio/audio_data_holder";
|
|
11
|
+
import { EMPTY, expand, map, Observable } from "rxjs";
|
|
12
|
+
import { AudioBufferSource, AudioDataHolder } from "../../audio/audio_data_holder";
|
|
13
|
+
import { ArrayAudioBuffer } from "../../audio/array_audio_buffer";
|
|
14
|
+
import { IndexedDbAudioBuffer } from "../../audio/inddb_audio_buffer";
|
|
15
|
+
import { BasicRecordingService } from "./basic_recording.service";
|
|
13
16
|
import * as i0 from "@angular/core";
|
|
14
17
|
import * as i1 from "@angular/common/http";
|
|
15
18
|
import * as i2 from "../../spr.config";
|
|
16
|
-
export
|
|
17
|
-
export class RecordingService {
|
|
19
|
+
export class RecordingService extends BasicRecordingService {
|
|
18
20
|
constructor(http, config) {
|
|
21
|
+
super(http, config);
|
|
19
22
|
this.http = http;
|
|
20
23
|
this.config = config;
|
|
21
|
-
this.withCredentials = false;
|
|
22
24
|
//private debugDelay:number=10000;
|
|
23
25
|
this.debugDelay = 0;
|
|
24
|
-
this.apiEndPoint = '';
|
|
25
|
-
if (config && config.apiEndPoint) {
|
|
26
|
-
this.apiEndPoint = config.apiEndPoint;
|
|
27
|
-
}
|
|
28
|
-
if (this.apiEndPoint !== '') {
|
|
29
|
-
this.apiEndPoint = this.apiEndPoint + '/';
|
|
30
|
-
}
|
|
31
|
-
if (config != null && config.withCredentials != null) {
|
|
32
|
-
this.withCredentials = config.withCredentials;
|
|
33
|
-
}
|
|
34
26
|
}
|
|
35
27
|
recordingFilesUrl() {
|
|
36
28
|
let recFilesUrl = this.apiEndPoint + RecordingService.RECORDING_API_CTX;
|
|
@@ -43,6 +35,15 @@ export class RecordingService {
|
|
|
43
35
|
SessionService.SESSION_API_CTX + '/' + encSessId + '/' + RecordingService.RECORDING_API_CTX;
|
|
44
36
|
return recFilesUrl;
|
|
45
37
|
}
|
|
38
|
+
// private sessionRecordingFilesUrl(projectName: string, sessId: string | number):URL{
|
|
39
|
+
// let encPrjName=encodeURIComponent(projectName);
|
|
40
|
+
// let encSessId=encodeURIComponent(sessId);
|
|
41
|
+
//
|
|
42
|
+
// let recFilesUrlStr = this.apiEndPoint + ProjectService.PROJECT_API_CTX + '/' + encPrjName + '/' +
|
|
43
|
+
// SessionService.SESSION_API_CTX + '/' + encSessId + '/' + RecordingService.RECORDING_API_CTX;
|
|
44
|
+
// let recFilesUrl=new URL(recFilesUrlStr);
|
|
45
|
+
// return recFilesUrl;
|
|
46
|
+
// }
|
|
46
47
|
sessionRecFilesUrl(projectName, sessId) {
|
|
47
48
|
let encPrjName = encodeURIComponent(projectName);
|
|
48
49
|
let encSessId = encodeURIComponent(sessId);
|
|
@@ -89,18 +90,405 @@ export class RecordingService {
|
|
|
89
90
|
withCredentials: this.withCredentials
|
|
90
91
|
});
|
|
91
92
|
}
|
|
92
|
-
|
|
93
|
+
chunkAudioRequestToIndDb(aCtx, persistentAudioStorageTarget, inddbAudioBuffer, baseAudioUrl, startFrame = 0, orgSampleRate, seconds) {
|
|
94
|
+
//let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;
|
|
95
|
+
//let audioUrl=new URL(baseAudioUrl);
|
|
96
|
+
let frameLength = orgSampleRate * Math.round(seconds); // Important: multiple of original sample rate to prevent numeric rounding errors on resampling. 10 seconds is a good value for ind db storage.
|
|
97
|
+
let ausps = new URLSearchParams();
|
|
98
|
+
ausps.set('startFrame', startFrame.toString());
|
|
99
|
+
ausps.set('frameLength', frameLength.toString());
|
|
100
|
+
if (this.config && this.config.apiType === ApiType.FILES) {
|
|
101
|
+
// for development and demo
|
|
102
|
+
// append UUID to make request URL unique to avoid localhost server caching
|
|
103
|
+
//audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();
|
|
104
|
+
ausps.set('requestUUID', UUID.generate());
|
|
105
|
+
}
|
|
106
|
+
let obs = new Observable(subscriber => {
|
|
107
|
+
this.audioRequestByURL(baseAudioUrl, ausps).subscribe({ next: (resp) => {
|
|
108
|
+
// Do not use Promise version, which does not work with Safari 13 (13.0.5)
|
|
109
|
+
if (resp.body) {
|
|
110
|
+
//console.debug("chunkAudioRequestToIndDb: subscriber.closed: "+subscriber.closed);
|
|
111
|
+
//console.debug("chunkAudioRequestToIndDb: Audio file bytes: "+resp.body.byteLength);
|
|
112
|
+
aCtx.decodeAudioData(resp.body, ab => {
|
|
113
|
+
//console.debug("chunkAudioRequestToIndDb: Decoded audio chunk frames for inddb: "+ab.length);
|
|
114
|
+
if (!inddbAudioBuffer) {
|
|
115
|
+
//console.debug("chunkAudioRequestToIndDb: Create inddb ab from chunk ab...");
|
|
116
|
+
IndexedDbAudioBuffer.fromChunkAudioBuffer(persistentAudioStorageTarget, ab).subscribe({
|
|
117
|
+
next: (iab) => {
|
|
118
|
+
//console.debug("chunkAudioRequestToIndDb: Built inddb ab from chunk ab: "+iab);
|
|
119
|
+
inddbAudioBuffer = iab;
|
|
120
|
+
if (inddbAudioBuffer.frameLen < frameLength) {
|
|
121
|
+
//console.debug("chunkAudioRequestToIndDb: Built inddb ab from chunk ab: First chunk shorter tha frameLength ("+inddbAudioBuffer.frameLen+"<"+frameLength+"), assuming end of data, sealing inddb ab.");
|
|
122
|
+
inddbAudioBuffer.seal();
|
|
123
|
+
}
|
|
124
|
+
subscriber.next(iab);
|
|
125
|
+
},
|
|
126
|
+
complete: () => {
|
|
127
|
+
//console.debug("chunkAudioRequestToIndDb: Built inddb ab from chunk ab complete.");
|
|
128
|
+
subscriber.complete();
|
|
129
|
+
},
|
|
130
|
+
error: (err) => {
|
|
131
|
+
// iPad may throw QuotaExceededError here
|
|
132
|
+
console.error("chunkAudioRequestToIndDb: Built inddb ab from chunk ab error: " + err);
|
|
133
|
+
subscriber.error(err);
|
|
134
|
+
}
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
else {
|
|
138
|
+
//console.debug("chunkAudioRequestToIndDb: Append audio chunk to inddb ab...");
|
|
139
|
+
inddbAudioBuffer.appendChunkAudioBuffer(ab).subscribe({
|
|
140
|
+
next: (iab) => {
|
|
141
|
+
// if(inddbAudioBuffer) {
|
|
142
|
+
// console.debug("chunkAudioRequestToIndDb: Appended audio chunk to inddb audio buffer: "+inddbAudioBuffer);
|
|
143
|
+
// }else{
|
|
144
|
+
// console.debug("chunkAudioRequestToIndDb: Append audio chunk returned null");
|
|
145
|
+
// }
|
|
146
|
+
subscriber.next(inddbAudioBuffer);
|
|
147
|
+
},
|
|
148
|
+
complete: () => {
|
|
149
|
+
subscriber.complete();
|
|
150
|
+
},
|
|
151
|
+
error: (err) => {
|
|
152
|
+
subscriber.error(err);
|
|
153
|
+
}
|
|
154
|
+
});
|
|
155
|
+
}
|
|
156
|
+
}, error => {
|
|
157
|
+
console.error('chunkAudioRequestToIndDb: error: ' + error);
|
|
158
|
+
//if(error instanceof HttpErrorResponse) {
|
|
159
|
+
subscriber.error(error);
|
|
160
|
+
//}
|
|
161
|
+
});
|
|
162
|
+
}
|
|
163
|
+
else {
|
|
164
|
+
console.error('chunkAudioRequestToIndDb: Fetching audio file: response has no body');
|
|
165
|
+
subscriber.error('chunkAudioRequestToIndDb: Fetching audio file: response has no body');
|
|
166
|
+
}
|
|
167
|
+
},
|
|
168
|
+
error: (error) => {
|
|
169
|
+
console.error('chunkAudioRequestToIndDb: error: ' + error);
|
|
170
|
+
subscriber.error(error);
|
|
171
|
+
//subscriber.complete();
|
|
172
|
+
}
|
|
173
|
+
});
|
|
174
|
+
});
|
|
175
|
+
return obs;
|
|
176
|
+
}
|
|
177
|
+
chunkedAudioRequestToArrayBuffer(aCtx, baseAudioUrl, orgSampleRate, seconds) {
|
|
178
|
+
let obs = new Observable(subscriber => {
|
|
179
|
+
let arrayAudioBuffer = null;
|
|
180
|
+
let startFrame = 0;
|
|
181
|
+
let frameLength = orgSampleRate * Math.round(seconds); // Important: multiple of original sample rate to prevent numeric rounding errors on resampling.
|
|
182
|
+
//console.debug("Chunk audio request startFrame 0");
|
|
183
|
+
let subscr = this.chunkAudioRequest(aCtx, baseAudioUrl, startFrame, frameLength).pipe(expand(value => {
|
|
184
|
+
if (subscriber.closed) {
|
|
185
|
+
subscr.unsubscribe();
|
|
186
|
+
}
|
|
187
|
+
if (value) {
|
|
188
|
+
if (arrayAudioBuffer) {
|
|
189
|
+
if (arrayAudioBuffer?.sealed()) {
|
|
190
|
+
return EMPTY;
|
|
191
|
+
}
|
|
192
|
+
else {
|
|
193
|
+
//let nextStartFrame=arrayAudioBuffer.frameLen + 1;
|
|
194
|
+
// We cannot use the frame length of the array audio buffer to determine the next start frame because
|
|
195
|
+
// AudioContext.decodeAudioData might change the sample rate of the original file. (E.g. Recordings from Mac/iOS with SR 44100 and listen to the on Windows platform will resmaple the audio data to 48000.)
|
|
196
|
+
// Frame count then differs on client and server
|
|
197
|
+
// Simply proceed in steps of frameLength
|
|
198
|
+
// More advanced method would be to parse the WAV header to find out the rela frame length of the chunk audio file
|
|
199
|
+
startFrame += frameLength;
|
|
200
|
+
//console.debug("Next start frame: "+startFrame);
|
|
201
|
+
//console.debug("chunkedAudioRequest: expand() subscriber.closed: "+subscriber.closed);
|
|
202
|
+
return this.chunkAudioRequest(aCtx, baseAudioUrl, startFrame, frameLength);
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
else {
|
|
206
|
+
return EMPTY;
|
|
207
|
+
}
|
|
208
|
+
}
|
|
209
|
+
else {
|
|
210
|
+
return EMPTY;
|
|
211
|
+
}
|
|
212
|
+
}), map(chDl => {
|
|
213
|
+
let ab = null;
|
|
214
|
+
if (chDl === null) {
|
|
215
|
+
return null;
|
|
216
|
+
}
|
|
217
|
+
else {
|
|
218
|
+
ab = chDl.decodedAudioBuffer;
|
|
219
|
+
}
|
|
220
|
+
if (ab) {
|
|
221
|
+
if (arrayAudioBuffer) {
|
|
222
|
+
arrayAudioBuffer.appendAudioBuffer(ab);
|
|
223
|
+
}
|
|
224
|
+
else {
|
|
225
|
+
arrayAudioBuffer = ArrayAudioBuffer.fromAudioBuffer(ab, ab.length);
|
|
226
|
+
}
|
|
227
|
+
// Following code is wrong if original samplerate is less-then platform samplerate
|
|
228
|
+
// E.g. Loading 44100Hz file on Safari
|
|
229
|
+
// console.debug("Array ab seal?: "+ab.length+" < "+frameLength);
|
|
230
|
+
// if (ab.length < frameLength) {
|
|
231
|
+
// arrayAudioBuffer.seal();
|
|
232
|
+
// console.debug("Array ab sealed.");
|
|
233
|
+
// }
|
|
234
|
+
}
|
|
235
|
+
return arrayAudioBuffer;
|
|
236
|
+
})).subscribe({
|
|
237
|
+
next: (aab) => {
|
|
238
|
+
//console.debug("chunkedAudioRequest: subscriber.closed: "+subscriber.closed);
|
|
239
|
+
subscriber.next(aab);
|
|
240
|
+
},
|
|
241
|
+
complete: () => {
|
|
242
|
+
subscriber.complete();
|
|
243
|
+
},
|
|
244
|
+
error: (err) => {
|
|
245
|
+
if (err instanceof HttpErrorResponse) {
|
|
246
|
+
if (err.status == 404) {
|
|
247
|
+
if (arrayAudioBuffer) {
|
|
248
|
+
arrayAudioBuffer.seal();
|
|
249
|
+
subscriber.next(arrayAudioBuffer);
|
|
250
|
+
subscriber.complete();
|
|
251
|
+
}
|
|
252
|
+
else {
|
|
253
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
254
|
+
subscriber.next(null);
|
|
255
|
+
subscriber.complete();
|
|
256
|
+
}
|
|
257
|
+
// observer.complete()
|
|
258
|
+
}
|
|
259
|
+
else {
|
|
260
|
+
// all other states are errors
|
|
261
|
+
subscriber.error(err);
|
|
262
|
+
}
|
|
263
|
+
}
|
|
264
|
+
else {
|
|
265
|
+
subscriber.error(err);
|
|
266
|
+
}
|
|
267
|
+
}
|
|
268
|
+
});
|
|
269
|
+
});
|
|
270
|
+
return obs;
|
|
271
|
+
}
|
|
272
|
+
chunkedInddbAudioRequest(aCtx, persistentAudioStorageTarget, baseAudioUrl, orgSampleRate, seconds) {
|
|
273
|
+
let obs = new Observable(subscriber => {
|
|
274
|
+
let inddbAudioBuffer = null;
|
|
275
|
+
let startFrame = 0;
|
|
276
|
+
//let frameLength = DEFAULT_CHUNKED_DOWNLOAD_FRAMELENGTH;
|
|
277
|
+
let frameLength = orgSampleRate * Math.round(seconds);
|
|
278
|
+
//console.debug("chunkedInddbAudioRequest: Chunk audio request for inddb. startFrame: "+startFrame);
|
|
279
|
+
let subscr = this.chunkAudioRequestToIndDb(aCtx, persistentAudioStorageTarget, null, baseAudioUrl, startFrame, orgSampleRate, seconds).pipe(expand(iab => {
|
|
280
|
+
// console.debug("chunkedInddbAudioRequest (pipe/expand): Got inddb ab: "+iab);
|
|
281
|
+
if (subscriber.closed) {
|
|
282
|
+
subscr.unsubscribe();
|
|
283
|
+
}
|
|
284
|
+
if (iab) {
|
|
285
|
+
inddbAudioBuffer = iab;
|
|
286
|
+
//if (inddbAudioBuffer) {
|
|
287
|
+
if (inddbAudioBuffer?.sealed()) {
|
|
288
|
+
return EMPTY;
|
|
289
|
+
}
|
|
290
|
+
else {
|
|
291
|
+
//let nextStartFrame=inddbAudioBuffer.frameLen + 1;
|
|
292
|
+
// We cannot use the frame length of the array audio buffer to determine the next start frame because
|
|
293
|
+
// AudioContext.decodeAudioData might change the sample rate of the original file. (E.g. Recordings from Mac/iOS with SR 44100 and listen to the on Windows platform will resmaple the audio data to 48000.)
|
|
294
|
+
// Frame count then differs on client and server
|
|
295
|
+
// Simply proceed in steps of frameLength
|
|
296
|
+
// More advanced method would be to parse the WAV header to find out the real frame length of the chunk audio file
|
|
297
|
+
startFrame += frameLength;
|
|
298
|
+
//console.debug("Next start frame: "+startFrame);
|
|
299
|
+
//console.debug("chunkedInddbAudioRequest: expand() subscriber.closed: "+subscriber.closed);
|
|
300
|
+
return this.chunkAudioRequestToIndDb(aCtx, persistentAudioStorageTarget, inddbAudioBuffer, baseAudioUrl, startFrame, orgSampleRate, seconds);
|
|
301
|
+
}
|
|
302
|
+
// } else {
|
|
303
|
+
// return EMPTY;
|
|
304
|
+
// }
|
|
305
|
+
}
|
|
306
|
+
else {
|
|
307
|
+
return EMPTY;
|
|
308
|
+
}
|
|
309
|
+
})).subscribe({
|
|
310
|
+
next: (aab) => {
|
|
311
|
+
//console.debug("chunkedInddbAudioRequest: subscriber.closed: "+subscriber.closed);
|
|
312
|
+
subscriber.next(aab);
|
|
313
|
+
},
|
|
314
|
+
complete: () => {
|
|
315
|
+
subscriber.complete();
|
|
316
|
+
},
|
|
317
|
+
error: (err) => {
|
|
318
|
+
if (err instanceof HttpErrorResponse) {
|
|
319
|
+
if (err.status == 404) {
|
|
320
|
+
//console.debug("chunkedInddbAudioRequest: Received HTTP 404 not found.");
|
|
321
|
+
if (inddbAudioBuffer) {
|
|
322
|
+
inddbAudioBuffer.seal();
|
|
323
|
+
//console.debug("chunkedInddbAudioRequest: Sealed inddb audio buffer.");
|
|
324
|
+
subscriber.next(inddbAudioBuffer);
|
|
325
|
+
subscriber.complete();
|
|
326
|
+
}
|
|
327
|
+
else {
|
|
328
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
329
|
+
//console.debug("chunkedInddbAudioRequest: Interpret HTTP 404 as not recorded yet.");
|
|
330
|
+
subscriber.next(null);
|
|
331
|
+
subscriber.complete();
|
|
332
|
+
}
|
|
333
|
+
// observer.complete()
|
|
334
|
+
}
|
|
335
|
+
else {
|
|
336
|
+
// all other states are (real) errors
|
|
337
|
+
console.error("chunkedInddbAudioRequest: Error: " + err.message);
|
|
338
|
+
subscriber.error(err);
|
|
339
|
+
}
|
|
340
|
+
}
|
|
341
|
+
else {
|
|
342
|
+
// all other errors are (real) errors
|
|
343
|
+
console.error("chunkedInddbAudioRequest: Error: " + err);
|
|
344
|
+
subscriber.error(err);
|
|
345
|
+
}
|
|
346
|
+
}
|
|
347
|
+
});
|
|
348
|
+
});
|
|
349
|
+
return obs;
|
|
350
|
+
}
|
|
351
|
+
// private chunkedNetAudioRequest(aCtx:AudioContext,baseAudioUrl:string): Observable<NetAudioBuffer|null> {
|
|
352
|
+
// let obs=new Observable<NetAudioBuffer|null>(subscriber => {
|
|
353
|
+
// let inddbAudioBuffer: NetAudioBuffer | null = null;
|
|
354
|
+
// let startFrame=0;
|
|
355
|
+
// let frameLength = DEFAULT_CHUNKED_DOWNLOAD_FRAMELENGTH;
|
|
356
|
+
// //console.debug("chunkedInddbAudioRequest: Chunk audio request for inddb. startFrame: "+startFrame);
|
|
357
|
+
// let subscr=this.chunkAudioRequestToIndDb(aCtx,persistentAudioStorageTarget,null, baseAudioUrl, startFrame, frameLength).pipe(
|
|
358
|
+
//
|
|
359
|
+
// expand(iab => {
|
|
360
|
+
// // console.debug("chunkedInddbAudioRequest (pipe/expand): Got inddb ab: "+iab);
|
|
361
|
+
// if(subscriber.closed){
|
|
362
|
+
// subscr.unsubscribe();
|
|
363
|
+
// }
|
|
364
|
+
// if (iab) {
|
|
365
|
+
// inddbAudioBuffer=iab;
|
|
366
|
+
// //if (inddbAudioBuffer) {
|
|
367
|
+
// if (inddbAudioBuffer?.sealed()) {
|
|
368
|
+
// return EMPTY;
|
|
369
|
+
// } else {
|
|
370
|
+
// //let nextStartFrame=inddbAudioBuffer.frameLen + 1;
|
|
371
|
+
// // We cannot use the frame length of the array audio buffer to determine the next start frame because
|
|
372
|
+
// // AudioContext.decodeAudioData might change the sample rate of the original file. (E.g. Recordings from Mac/iOS with SR 44100 and listen to the on Windows platform will resmaple the audio data to 48000.)
|
|
373
|
+
// // Frame count then differs on client and server
|
|
374
|
+
//
|
|
375
|
+
// // Simply proceed in steps of frameLength
|
|
376
|
+
// // More advanced method would be to parse the WAV header to find out the rela frame length of the chunk audio file
|
|
377
|
+
// startFrame+=frameLength;
|
|
378
|
+
// //console.debug("Next start frame: "+startFrame);
|
|
379
|
+
// //console.debug("chunkedInddbAudioRequest: expand() subscriber.closed: "+subscriber.closed);
|
|
380
|
+
// return this.chunkAudioRequestToIndDb(aCtx,persistentAudioStorageTarget,inddbAudioBuffer, baseAudioUrl, startFrame, frameLength);
|
|
381
|
+
// }
|
|
382
|
+
// // } else {
|
|
383
|
+
// // return EMPTY;
|
|
384
|
+
// // }
|
|
385
|
+
// }else{
|
|
386
|
+
// return EMPTY;
|
|
387
|
+
// }
|
|
388
|
+
// }
|
|
389
|
+
// )
|
|
390
|
+
// ).subscribe({
|
|
391
|
+
// next: (aab)=>{
|
|
392
|
+
// //console.debug("chunkedInddbAudioRequest: subscriber.closed: "+subscriber.closed);
|
|
393
|
+
// subscriber.next(aab)
|
|
394
|
+
// },
|
|
395
|
+
// complete: ()=>{
|
|
396
|
+
// subscriber.complete()
|
|
397
|
+
// },
|
|
398
|
+
// error: (err)=>{
|
|
399
|
+
// if(err instanceof HttpErrorResponse){
|
|
400
|
+
// if (err.status == 404) {
|
|
401
|
+
// //console.debug("chunkedInddbAudioRequest: Received HTTP 404 not found.");
|
|
402
|
+
// if(inddbAudioBuffer){
|
|
403
|
+
// inddbAudioBuffer.seal();
|
|
404
|
+
// //console.debug("chunkedInddbAudioRequest: Sealed inddb audio buffer.");
|
|
405
|
+
// subscriber.next(inddbAudioBuffer);
|
|
406
|
+
// subscriber.complete();
|
|
407
|
+
// }else {
|
|
408
|
+
// // Interpret not as an error, the file ist not recorded yet
|
|
409
|
+
// //console.debug("chunkedInddbAudioRequest: Interpret HTTP 404 as not recorded yet.");
|
|
410
|
+
// subscriber.next(null);
|
|
411
|
+
// subscriber.complete();
|
|
412
|
+
// }
|
|
413
|
+
// // observer.complete()
|
|
414
|
+
// } else {
|
|
415
|
+
// // all other states are (real) errors
|
|
416
|
+
// console.error("chunkedInddbAudioRequest: Error: "+err.message);
|
|
417
|
+
// subscriber.error(err);
|
|
418
|
+
// }
|
|
419
|
+
// }else {
|
|
420
|
+
// // all other errors are (real) errors
|
|
421
|
+
// console.error("chunkedInddbAudioRequest: Error: "+err);
|
|
422
|
+
// subscriber.error(err);
|
|
423
|
+
// }
|
|
424
|
+
// }
|
|
425
|
+
// });
|
|
426
|
+
// });
|
|
427
|
+
// return obs;
|
|
428
|
+
// }
|
|
429
|
+
audioFileUrlById(projectName, sessId, recFileId) {
|
|
93
430
|
let recFilesUrl = this.sessionRecordingFilesUrl(projectName, sessId);
|
|
94
431
|
let encRecFileId = encodeURIComponent(recFileId);
|
|
95
|
-
|
|
432
|
+
return recFilesUrl + '/' + encRecFileId;
|
|
433
|
+
}
|
|
434
|
+
fetchAudiofile(projectName, sessId, recFileId) {
|
|
435
|
+
let recUrl = this.audioFileUrlById(projectName, sessId, recFileId);
|
|
96
436
|
return this.audioRequest(recUrl);
|
|
97
437
|
}
|
|
438
|
+
audioFileUrl(projectName, recordingFile) {
|
|
439
|
+
let url = null;
|
|
440
|
+
let recFileId = recordingFile.recordingFileId;
|
|
441
|
+
if (!recFileId) {
|
|
442
|
+
recFileId = recordingFile.uuid;
|
|
443
|
+
}
|
|
444
|
+
if (recordingFile.session && recFileId) {
|
|
445
|
+
url = this.audioFileUrlById(projectName, recordingFile.session, recFileId);
|
|
446
|
+
}
|
|
447
|
+
return url;
|
|
448
|
+
}
|
|
449
|
+
audioFileUrlByUUID(projectName, session, uuid) {
|
|
450
|
+
let url = this.audioFileUrlById(projectName, session, uuid);
|
|
451
|
+
return url;
|
|
452
|
+
}
|
|
98
453
|
fetchSprAudiofile(projectName, sessId, itemcode, version) {
|
|
99
454
|
let recFilesUrl = this.sessionRecFilesUrl(projectName, sessId);
|
|
100
455
|
let encItemcode = encodeURIComponent(itemcode);
|
|
101
456
|
let recUrl = recFilesUrl + '/' + encItemcode + '/' + version;
|
|
102
457
|
return this.audioRequest(recUrl);
|
|
103
458
|
}
|
|
459
|
+
sprAudioFileUrlByItemcode(projectName, sessId, itemcode, version) {
|
|
460
|
+
let recFilesUrl = this.sessionRecFilesUrl(projectName, sessId);
|
|
461
|
+
let encItemcode = encodeURIComponent(itemcode);
|
|
462
|
+
let recUrl = recFilesUrl + '/' + encItemcode + '/' + version;
|
|
463
|
+
return recUrl;
|
|
464
|
+
}
|
|
465
|
+
sprAudioFileUrl(projectName, recordingFile) {
|
|
466
|
+
let url = null;
|
|
467
|
+
if (recordingFile.session) {
|
|
468
|
+
url = this.sprAudioFileUrlByItemcode(projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);
|
|
469
|
+
}
|
|
470
|
+
return url;
|
|
471
|
+
}
|
|
472
|
+
// private fetchSprAudiofileArrayBuffer(aCtx:AudioContext,projectName: string, sessId: string | number, itemcode: string,version:number): Observable<ArrayAudioBuffer|null>{
|
|
473
|
+
// let recFilesUrl=this.sessionRecFilesUrl(projectName,sessId);
|
|
474
|
+
// let encItemcode=encodeURIComponent(itemcode);
|
|
475
|
+
// let recUrl = recFilesUrl + '/' + encItemcode +'/'+version;
|
|
476
|
+
// return this.chunkedAudioRequest(aCtx,recUrl);
|
|
477
|
+
// }
|
|
478
|
+
// private fetchSprAudiofileInddbBuffer(aCtx:AudioContext, persistentAudioStorageTarget:PersistentAudioStorageTarget,projectName: string, sessId: string | number, itemcode: string,version:number): Observable<IndexedDbAudioBuffer|null>{
|
|
479
|
+
// let recFilesUrlStr=this.sessionRecFilesUrl(projectName,sessId);
|
|
480
|
+
// let encItemcode=encodeURIComponent(itemcode);
|
|
481
|
+
// let recUrlStr = recFilesUrlStr + '/' + encItemcode +'/'+version;
|
|
482
|
+
// //let recUrl=new URL(recUrlStr);
|
|
483
|
+
// return this.chunkedInddbAudioRequest(aCtx,persistentAudioStorageTarget,recUrlStr);
|
|
484
|
+
// }
|
|
485
|
+
// private sprNetAudiofileBuffer(aCtx:AudioContext,projectName: string, sessId: string | number, itemcode: string,version:number): Observable<IndexedDbAudioBuffer|null>{
|
|
486
|
+
// let recFilesUrlStr=this.sessionRecFilesUrl(projectName,sessId);
|
|
487
|
+
// let encItemcode=encodeURIComponent(itemcode);
|
|
488
|
+
// let recUrlStr = recFilesUrlStr + '/' + encItemcode +'/'+version;
|
|
489
|
+
// //let recUrl=new URL(recUrlStr);
|
|
490
|
+
// return this.chunkedInddbAudioRequest(aCtx,recUrlStr);
|
|
491
|
+
// }
|
|
104
492
|
fetchRecordingFileAudioBuffer(aCtx, projectName, recordingFile) {
|
|
105
493
|
let wobs = new Observable(observer => {
|
|
106
494
|
let recFileId = recordingFile.recordingFileId;
|
|
@@ -109,34 +497,36 @@ export class RecordingService {
|
|
|
109
497
|
}
|
|
110
498
|
if (recordingFile.session && recFileId) {
|
|
111
499
|
let obs = this.fetchAudiofile(projectName, recordingFile.session, recFileId);
|
|
112
|
-
obs.subscribe({
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
if (error.status == 404) {
|
|
500
|
+
obs.subscribe(resp => {
|
|
501
|
+
// Do not use Promise version, which does not work with Safari 13 (13.0.5)
|
|
502
|
+
if (resp.body) {
|
|
503
|
+
aCtx.decodeAudioData(resp.body, ab => {
|
|
504
|
+
observer.next(ab);
|
|
505
|
+
observer.complete();
|
|
506
|
+
}, error => {
|
|
507
|
+
observer.error(error);
|
|
508
|
+
observer.complete();
|
|
509
|
+
});
|
|
510
|
+
}
|
|
511
|
+
else {
|
|
512
|
+
observer.error('Fetching audio file: response has no body');
|
|
513
|
+
}
|
|
514
|
+
}, (err) => {
|
|
515
|
+
if (err instanceof HttpErrorResponse) {
|
|
516
|
+
if (err.status == 404) {
|
|
130
517
|
// Interpret not as an error, the file ist not recorded yet
|
|
131
518
|
observer.next(null);
|
|
132
519
|
observer.complete();
|
|
133
520
|
}
|
|
134
521
|
else {
|
|
135
522
|
// all other states are errors
|
|
136
|
-
observer.error(
|
|
523
|
+
observer.error(err);
|
|
137
524
|
observer.complete();
|
|
138
525
|
}
|
|
139
526
|
}
|
|
527
|
+
else {
|
|
528
|
+
observer.error(err);
|
|
529
|
+
}
|
|
140
530
|
});
|
|
141
531
|
}
|
|
142
532
|
else {
|
|
@@ -158,7 +548,8 @@ export class RecordingService {
|
|
|
158
548
|
// Do not use Promise version, which does not work with Safari 13 (13.0.5)
|
|
159
549
|
if (resp.body) {
|
|
160
550
|
aCtx.decodeAudioData(resp.body, ab => {
|
|
161
|
-
let
|
|
551
|
+
let abs = new AudioBufferSource(ab);
|
|
552
|
+
let adh = new AudioDataHolder(abs);
|
|
162
553
|
RecordingFileUtils.setAudioData(recordingFile, adh);
|
|
163
554
|
if (this.debugDelay > 0) {
|
|
164
555
|
window.setTimeout(() => {
|
|
@@ -178,16 +569,21 @@ export class RecordingService {
|
|
|
178
569
|
else {
|
|
179
570
|
observer.error('Fetching audio file: response has no body');
|
|
180
571
|
}
|
|
181
|
-
}, (
|
|
182
|
-
if (
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
572
|
+
}, (err) => {
|
|
573
|
+
if (err instanceof HttpErrorResponse) {
|
|
574
|
+
if (err.status == 404) {
|
|
575
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
576
|
+
observer.next(null);
|
|
577
|
+
observer.complete();
|
|
578
|
+
}
|
|
579
|
+
else {
|
|
580
|
+
// all other states are errors
|
|
581
|
+
observer.error(err);
|
|
582
|
+
observer.complete();
|
|
583
|
+
}
|
|
186
584
|
}
|
|
187
585
|
else {
|
|
188
|
-
|
|
189
|
-
observer.error(error);
|
|
190
|
-
observer.complete();
|
|
586
|
+
observer.error(err);
|
|
191
587
|
}
|
|
192
588
|
});
|
|
193
589
|
}
|
|
@@ -218,16 +614,15 @@ export class RecordingService {
|
|
|
218
614
|
observer.error('Fetching audio file: response has no body');
|
|
219
615
|
}
|
|
220
616
|
},
|
|
221
|
-
error: (
|
|
222
|
-
if (
|
|
617
|
+
error: (err) => {
|
|
618
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
223
619
|
// Interpret not as an error, the file ist not recorded yet
|
|
224
620
|
observer.next(null);
|
|
225
621
|
observer.complete();
|
|
226
622
|
}
|
|
227
623
|
else {
|
|
228
624
|
// all other states are errors
|
|
229
|
-
observer.error(
|
|
230
|
-
observer.complete();
|
|
625
|
+
observer.error(err);
|
|
231
626
|
}
|
|
232
627
|
}
|
|
233
628
|
});
|
|
@@ -238,6 +633,276 @@ export class RecordingService {
|
|
|
238
633
|
});
|
|
239
634
|
return wobs;
|
|
240
635
|
}
|
|
636
|
+
fetchSprRecordingFileArrayAudioBuffer(aCtx, projectName, recordingFile) {
|
|
637
|
+
let wobs = new Observable(observer => {
|
|
638
|
+
if (recordingFile.session) {
|
|
639
|
+
let baseUrl = this.sprAudioFileUrl(projectName, recordingFile);
|
|
640
|
+
if (baseUrl) {
|
|
641
|
+
if (recordingFile.samplerate) {
|
|
642
|
+
let lengthInSeconds = RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;
|
|
643
|
+
let obs = this.chunkedAudioRequestToArrayBuffer(aCtx, baseUrl, recordingFile.samplerate, lengthInSeconds);
|
|
644
|
+
//let obs = this.fetchSprAudiofileArrayBuffer(aCtx,projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);
|
|
645
|
+
let subscr = obs.subscribe({
|
|
646
|
+
next: aab => {
|
|
647
|
+
//console.debug("fetchSprRecordingFileArrayAudioBuffer: observer.closed: "+observer.closed);
|
|
648
|
+
if (observer.closed) {
|
|
649
|
+
subscr.unsubscribe();
|
|
650
|
+
}
|
|
651
|
+
observer.next(aab);
|
|
652
|
+
},
|
|
653
|
+
complete: () => {
|
|
654
|
+
observer.complete();
|
|
655
|
+
},
|
|
656
|
+
error: (err) => {
|
|
657
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
658
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
659
|
+
observer.next(null);
|
|
660
|
+
observer.complete();
|
|
661
|
+
}
|
|
662
|
+
else {
|
|
663
|
+
// all other states are errors
|
|
664
|
+
observer.error(err);
|
|
665
|
+
}
|
|
666
|
+
}
|
|
667
|
+
});
|
|
668
|
+
}
|
|
669
|
+
else {
|
|
670
|
+
observer.error(new Error('Unknown sample rate of recording file ID: ' + recordingFile.recordingFileId));
|
|
671
|
+
}
|
|
672
|
+
}
|
|
673
|
+
}
|
|
674
|
+
else {
|
|
675
|
+
observer.error(new Error('Could not get session ID of recording file'));
|
|
676
|
+
}
|
|
677
|
+
});
|
|
678
|
+
return wobs;
|
|
679
|
+
}
|
|
680
|
+
fetchRecordingFileArrayAudioBuffer(aCtx, projectName, recordingFile) {
|
|
681
|
+
let wobs = new Observable(observer => {
|
|
682
|
+
if (recordingFile.session) {
|
|
683
|
+
let baseUrl = this.audioFileUrl(projectName, recordingFile);
|
|
684
|
+
if (baseUrl) {
|
|
685
|
+
if (recordingFile.samplerate) {
|
|
686
|
+
let lengthInSeconds = RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;
|
|
687
|
+
let obs = this.chunkedAudioRequestToArrayBuffer(aCtx, baseUrl, recordingFile.samplerate, lengthInSeconds);
|
|
688
|
+
//let obs = this.fetchSprAudiofileArrayBuffer(aCtx,projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);
|
|
689
|
+
let subscr = obs.subscribe({
|
|
690
|
+
next: aab => {
|
|
691
|
+
//console.debug("fetchSprRecordingFileArrayAudioBuffer: observer.closed: "+observer.closed);
|
|
692
|
+
if (observer.closed) {
|
|
693
|
+
subscr.unsubscribe();
|
|
694
|
+
}
|
|
695
|
+
observer.next(aab);
|
|
696
|
+
},
|
|
697
|
+
complete: () => {
|
|
698
|
+
observer.complete();
|
|
699
|
+
},
|
|
700
|
+
error: (err) => {
|
|
701
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
702
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
703
|
+
observer.next(null);
|
|
704
|
+
observer.complete();
|
|
705
|
+
}
|
|
706
|
+
else {
|
|
707
|
+
// all other states are errors
|
|
708
|
+
observer.error(err);
|
|
709
|
+
}
|
|
710
|
+
}
|
|
711
|
+
});
|
|
712
|
+
}
|
|
713
|
+
else {
|
|
714
|
+
observer.error(new Error('Unknown sample rate of recording file ID: ' + recordingFile.recordingFileId));
|
|
715
|
+
}
|
|
716
|
+
}
|
|
717
|
+
}
|
|
718
|
+
else {
|
|
719
|
+
observer.error(new Error('Could not get session ID of recording file'));
|
|
720
|
+
}
|
|
721
|
+
});
|
|
722
|
+
return wobs;
|
|
723
|
+
}
|
|
724
|
+
fetchSprRecordingFileIndDbAudioBuffer(aCtx, persistentAudioStorageTarget, projectName, recordingFile) {
|
|
725
|
+
let wobs = new Observable(observer => {
|
|
726
|
+
if (recordingFile.session) {
|
|
727
|
+
let baseUrl = this.sprAudioFileUrl(projectName, recordingFile);
|
|
728
|
+
if (baseUrl) {
|
|
729
|
+
if (recordingFile.samplerate) {
|
|
730
|
+
let lengthInSeconds = RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;
|
|
731
|
+
let obs = this.chunkedInddbAudioRequest(aCtx, persistentAudioStorageTarget, baseUrl, recordingFile.samplerate, lengthInSeconds);
|
|
732
|
+
let subscr = obs.subscribe({
|
|
733
|
+
next: aab => {
|
|
734
|
+
//console.debug("fetchSprRecordingFileIndDbAudioBuffer: observer.closed: "+observer.closed);
|
|
735
|
+
if (observer.closed) {
|
|
736
|
+
subscr.unsubscribe();
|
|
737
|
+
}
|
|
738
|
+
observer.next(aab);
|
|
739
|
+
},
|
|
740
|
+
complete: () => {
|
|
741
|
+
observer.complete();
|
|
742
|
+
},
|
|
743
|
+
error: (err) => {
|
|
744
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
745
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
746
|
+
observer.next(null);
|
|
747
|
+
observer.complete();
|
|
748
|
+
}
|
|
749
|
+
else {
|
|
750
|
+
// all other errors are real errors
|
|
751
|
+
observer.error(err);
|
|
752
|
+
}
|
|
753
|
+
}
|
|
754
|
+
});
|
|
755
|
+
}
|
|
756
|
+
else {
|
|
757
|
+
observer.error(new Error('Unknown sample rate of recording file ID: ' + recordingFile.recordingFileId));
|
|
758
|
+
}
|
|
759
|
+
}
|
|
760
|
+
}
|
|
761
|
+
else {
|
|
762
|
+
observer.error();
|
|
763
|
+
}
|
|
764
|
+
});
|
|
765
|
+
return wobs;
|
|
766
|
+
}
|
|
767
|
+
fetchRecordingFileIndDbAudioBuffer(aCtx, persistentAudioStorageTarget, projectName, recordingFile) {
|
|
768
|
+
let wobs = new Observable(observer => {
|
|
769
|
+
if (recordingFile.session) {
|
|
770
|
+
let baseUrl = this.audioFileUrl(projectName, recordingFile);
|
|
771
|
+
if (baseUrl) {
|
|
772
|
+
if (recordingFile.samplerate) {
|
|
773
|
+
let lengthInSeconds = RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;
|
|
774
|
+
let obs = this.chunkedInddbAudioRequest(aCtx, persistentAudioStorageTarget, baseUrl, recordingFile.samplerate, lengthInSeconds);
|
|
775
|
+
let subscr = obs.subscribe({
|
|
776
|
+
next: aab => {
|
|
777
|
+
//console.debug("fetchSprRecordingFileIndDbAudioBuffer: observer.closed: "+observer.closed);
|
|
778
|
+
if (observer.closed) {
|
|
779
|
+
subscr.unsubscribe();
|
|
780
|
+
}
|
|
781
|
+
observer.next(aab);
|
|
782
|
+
},
|
|
783
|
+
complete: () => {
|
|
784
|
+
observer.complete();
|
|
785
|
+
},
|
|
786
|
+
error: (err) => {
|
|
787
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
788
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
789
|
+
observer.next(null);
|
|
790
|
+
observer.complete();
|
|
791
|
+
}
|
|
792
|
+
else {
|
|
793
|
+
// all other errors are real errors
|
|
794
|
+
observer.error(err);
|
|
795
|
+
}
|
|
796
|
+
}
|
|
797
|
+
});
|
|
798
|
+
}
|
|
799
|
+
else {
|
|
800
|
+
observer.error(new Error('Unknown sample rate of recording file ID: ' + recordingFile.recordingFileId));
|
|
801
|
+
}
|
|
802
|
+
}
|
|
803
|
+
}
|
|
804
|
+
else {
|
|
805
|
+
observer.error();
|
|
806
|
+
}
|
|
807
|
+
});
|
|
808
|
+
return wobs;
|
|
809
|
+
}
|
|
810
|
+
fetchSprRecordingFileNetAudioBuffer(aCtx, projectName, recordingFile) {
|
|
811
|
+
let wobs = new Observable(observer => {
|
|
812
|
+
if (recordingFile.session) {
|
|
813
|
+
let baseUrl = this.sprAudioFileUrl(projectName, recordingFile);
|
|
814
|
+
if (baseUrl) {
|
|
815
|
+
let seconds = RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;
|
|
816
|
+
if (recordingFile.samplerate) {
|
|
817
|
+
let obs = this.chunkAudioRequestToNetAudioBuffer(aCtx, baseUrl, 0, recordingFile.samplerate, seconds, recordingFile.frames);
|
|
818
|
+
let subscr = obs.subscribe({
|
|
819
|
+
next: aab => {
|
|
820
|
+
//console.debug("fetchSprRecordingFileIndDbAudioBuffer: observer.closed: "+observer.closed);
|
|
821
|
+
if (observer.closed) {
|
|
822
|
+
subscr.unsubscribe();
|
|
823
|
+
}
|
|
824
|
+
else {
|
|
825
|
+
observer.next(aab);
|
|
826
|
+
}
|
|
827
|
+
},
|
|
828
|
+
complete: () => {
|
|
829
|
+
observer.complete();
|
|
830
|
+
},
|
|
831
|
+
error: (err) => {
|
|
832
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
833
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
834
|
+
observer.next(null);
|
|
835
|
+
observer.complete();
|
|
836
|
+
}
|
|
837
|
+
else {
|
|
838
|
+
// all other errors are real errors
|
|
839
|
+
observer.error(err);
|
|
840
|
+
}
|
|
841
|
+
}
|
|
842
|
+
});
|
|
843
|
+
}
|
|
844
|
+
else {
|
|
845
|
+
observer.error(new Error('Unknown sample rate of recording file ID: ' + recordingFile.recordingFileId));
|
|
846
|
+
}
|
|
847
|
+
}
|
|
848
|
+
else {
|
|
849
|
+
observer.error();
|
|
850
|
+
}
|
|
851
|
+
}
|
|
852
|
+
else {
|
|
853
|
+
observer.error();
|
|
854
|
+
}
|
|
855
|
+
});
|
|
856
|
+
return wobs;
|
|
857
|
+
}
|
|
858
|
+
fetchRecordingFileNetAudioBuffer(aCtx, projectName, recordingFile) {
|
|
859
|
+
let wobs = new Observable(observer => {
|
|
860
|
+
if (recordingFile.session) {
|
|
861
|
+
let baseUrl = this.audioFileUrl(projectName, recordingFile);
|
|
862
|
+
if (baseUrl) {
|
|
863
|
+
let seconds = RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;
|
|
864
|
+
if (recordingFile.samplerate) {
|
|
865
|
+
let obs = this.chunkAudioRequestToNetAudioBuffer(aCtx, baseUrl, 0, recordingFile.samplerate, seconds, recordingFile.frames);
|
|
866
|
+
let subscr = obs.subscribe({
|
|
867
|
+
next: aab => {
|
|
868
|
+
//console.debug("fetchSprRecordingFileIndDbAudioBuffer: observer.closed: "+observer.closed);
|
|
869
|
+
if (observer.closed) {
|
|
870
|
+
subscr.unsubscribe();
|
|
871
|
+
}
|
|
872
|
+
else {
|
|
873
|
+
observer.next(aab);
|
|
874
|
+
}
|
|
875
|
+
},
|
|
876
|
+
complete: () => {
|
|
877
|
+
observer.complete();
|
|
878
|
+
},
|
|
879
|
+
error: (err) => {
|
|
880
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
881
|
+
// Interpret not as an error, the file ist not recorded yet
|
|
882
|
+
observer.next(null);
|
|
883
|
+
observer.complete();
|
|
884
|
+
}
|
|
885
|
+
else {
|
|
886
|
+
// all other errors are real errors
|
|
887
|
+
observer.error(err);
|
|
888
|
+
}
|
|
889
|
+
}
|
|
890
|
+
});
|
|
891
|
+
}
|
|
892
|
+
else {
|
|
893
|
+
observer.error(new Error('Unknown sample rate of recording file ID: ' + recordingFile.recordingFileId));
|
|
894
|
+
}
|
|
895
|
+
}
|
|
896
|
+
else {
|
|
897
|
+
observer.error();
|
|
898
|
+
}
|
|
899
|
+
}
|
|
900
|
+
else {
|
|
901
|
+
observer.error();
|
|
902
|
+
}
|
|
903
|
+
});
|
|
904
|
+
return wobs;
|
|
905
|
+
}
|
|
241
906
|
fetchAndApplySprRecordingFile(aCtx, projectName, recordingFile) {
|
|
242
907
|
let wobs = new Observable(observer => {
|
|
243
908
|
if (recordingFile.session) {
|
|
@@ -247,7 +912,8 @@ export class RecordingService {
|
|
|
247
912
|
// Do not use Promise version, which does not work with Safari 13 (13.0.5)
|
|
248
913
|
if (resp.body) {
|
|
249
914
|
aCtx.decodeAudioData(resp.body, ab => {
|
|
250
|
-
|
|
915
|
+
let abs = new AudioBufferSource(ab);
|
|
916
|
+
RecordingFileUtils.setAudioData(recordingFile, new AudioDataHolder(abs));
|
|
251
917
|
if (this.debugDelay > 0) {
|
|
252
918
|
window.setTimeout(() => {
|
|
253
919
|
observer.next(recordingFile);
|
|
@@ -266,16 +932,15 @@ export class RecordingService {
|
|
|
266
932
|
else {
|
|
267
933
|
observer.error('Fetching audio file: response has no body');
|
|
268
934
|
}
|
|
269
|
-
}, (
|
|
270
|
-
if (
|
|
935
|
+
}, (err) => {
|
|
936
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
271
937
|
// Interpret not as an error, the file ist not recorded yet
|
|
272
938
|
observer.next(null);
|
|
273
939
|
observer.complete();
|
|
274
940
|
}
|
|
275
941
|
else {
|
|
276
942
|
// all other states are errors
|
|
277
|
-
observer.error(
|
|
278
|
-
observer.complete();
|
|
943
|
+
observer.error(err);
|
|
279
944
|
}
|
|
280
945
|
});
|
|
281
946
|
}
|
|
@@ -292,7 +957,8 @@ export class RecordingService {
|
|
|
292
957
|
// Do not use Promise version, which does not work with Safari 13
|
|
293
958
|
if (resp.body) {
|
|
294
959
|
aCtx.decodeAudioData(resp.body, ab => {
|
|
295
|
-
let
|
|
960
|
+
let abs = new AudioBufferSource(ab);
|
|
961
|
+
let adh = new AudioDataHolder(abs);
|
|
296
962
|
let rf = new SprRecordingFile(sessId, itemcode, version, adh);
|
|
297
963
|
if (this.debugDelay > 0) {
|
|
298
964
|
window.setTimeout(() => {
|
|
@@ -309,30 +975,31 @@ export class RecordingService {
|
|
|
309
975
|
else {
|
|
310
976
|
observer.error();
|
|
311
977
|
}
|
|
312
|
-
}, (
|
|
313
|
-
if (
|
|
978
|
+
}, (err) => {
|
|
979
|
+
if (err instanceof HttpErrorResponse && err.status == 404) {
|
|
314
980
|
// Interpret not as an error, the file ist not recorded yet
|
|
315
981
|
observer.next(null);
|
|
316
982
|
observer.complete();
|
|
317
983
|
}
|
|
318
984
|
else {
|
|
319
|
-
// all other
|
|
320
|
-
observer.error(
|
|
321
|
-
observer.complete();
|
|
985
|
+
// all other errors are real errors
|
|
986
|
+
observer.error(err);
|
|
322
987
|
}
|
|
323
988
|
});
|
|
324
989
|
});
|
|
325
990
|
return wobs;
|
|
326
991
|
}
|
|
327
992
|
}
|
|
993
|
+
// This seems to work
|
|
994
|
+
//public static readonly DEFAULT_CHUNKED_DOWNLOAD_SECONDS:number=4;
|
|
328
995
|
RecordingService.REC_API_CTX = 'recfile';
|
|
329
996
|
RecordingService.RECORDING_API_CTX = 'recordingfile';
|
|
330
|
-
RecordingService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "
|
|
331
|
-
RecordingService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "
|
|
332
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "
|
|
997
|
+
RecordingService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: RecordingService, deps: [{ token: i1.HttpClient }, { token: SPEECHRECORDER_CONFIG }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
998
|
+
RecordingService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: RecordingService });
|
|
999
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.2.12", ngImport: i0, type: RecordingService, decorators: [{
|
|
333
1000
|
type: Injectable
|
|
334
1001
|
}], ctorParameters: function () { return [{ type: i1.HttpClient }, { type: i2.SpeechRecorderConfig, decorators: [{
|
|
335
1002
|
type: Inject,
|
|
336
1003
|
args: [SPEECHRECORDER_CONFIG]
|
|
337
1004
|
}] }]; } });
|
|
338
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"recordings.service.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/recordings/recordings.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Created by klausj on 17.06.2017.\r\n */\r\nimport {Inject, Injectable} from '@angular/core';\r\nimport {HttpClient, HttpErrorResponse, HttpHeaders, HttpResponse} from \"@angular/common/http\";\r\nimport {ApiType, SPEECHRECORDER_CONFIG, SpeechRecorderConfig} from \"../../spr.config\";\r\n\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {SprRecordingFile, RecordingFileDescriptorImpl, RecordingFile, RecordingFileUtils} from \"../recording\";\r\nimport {ProjectService} from \"../project/project.service\";\r\nimport {SessionService} from \"../session/session.service\";\r\nimport {Observable} from \"rxjs\";\r\nimport {AudioDataHolder} from \"../../audio/audio_data_holder\";\r\n\r\n\r\nexport const REC_API_CTX='recfile'\r\n\r\n\r\n\r\n@Injectable()\r\nexport class RecordingService {\r\n\r\n  public static readonly REC_API_CTX = 'recfile'\r\n  public static readonly RECORDING_API_CTX='recordingfile'\r\n  private apiEndPoint: string;\r\n  private withCredentials: boolean = false;\r\n\r\n  //private debugDelay:number=10000;\r\n  private debugDelay:number=0;\r\n\r\n  constructor(private http: HttpClient, @Inject(SPEECHRECORDER_CONFIG) private config?: SpeechRecorderConfig) {\r\n\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  private recordingFilesUrl():string{\r\n    let recFilesUrl = this.apiEndPoint  + RecordingService.RECORDING_API_CTX;\r\n    return recFilesUrl;\r\n  }\r\n\r\n  private sessionRecordingFilesUrl(projectName: string, sessId: string | number):string{\r\n    let encPrjName=encodeURIComponent(projectName);\r\n    let encSessId=encodeURIComponent(sessId);\r\n    let recFilesUrl = this.apiEndPoint + ProjectService.PROJECT_API_CTX + '/' + encPrjName + '/' +\r\n      SessionService.SESSION_API_CTX + '/' + encSessId + '/' + RecordingService.RECORDING_API_CTX;\r\n    return recFilesUrl;\r\n  }\r\n\r\n  private sessionRecFilesUrl(projectName: string, sessId: string | number):string{\r\n    let encPrjName=encodeURIComponent(projectName);\r\n    let encSessId=encodeURIComponent(sessId);\r\n    let recFilesUrl = this.apiEndPoint + ProjectService.PROJECT_API_CTX + '/' + encPrjName + '/' +\r\n        SessionService.SESSION_API_CTX + '/' + encSessId + '/' + RecordingService.REC_API_CTX;\r\n    return recFilesUrl;\r\n  }\r\n\r\n  private sessionRecFilesReqUrl(projectName: string, sessId: string | number):string{\r\n    let recFilesUrl=this.sessionRecFilesUrl(projectName,sessId);\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      recFilesUrl = recFilesUrl + '.json?requestUUID=' + UUID.generate();\r\n    }\r\n    return recFilesUrl;\r\n  }\r\n\r\n  recordingFileDescrList(projectName: string, sessId: string | number):Observable<Array<RecordingFileDescriptorImpl>> {\r\n    let recFilesReqUrl = this.sessionRecFilesReqUrl(projectName,sessId);\r\n    let wobs = this.http.get<Array<RecordingFileDescriptorImpl>>(recFilesReqUrl,{withCredentials:this.withCredentials});\r\n    return wobs;\r\n  }\r\n\r\n  recordingFileList(projectName: string, sessId: string | number):Observable<Array<RecordingFile>> {\r\n    let recFilesReqUrl = this.sessionRecFilesReqUrl(projectName,sessId);\r\n    let wobs = this.http.get<Array<RecordingFile>>(recFilesReqUrl,{withCredentials:this.withCredentials});\r\n    return wobs;\r\n  }\r\n\r\n  sprRecordingFileList(projectName: string, sessId: string | number):Observable<Array<SprRecordingFile>> {\r\n    let recFilesReqUrl = this.sessionRecFilesReqUrl(projectName,sessId);\r\n    let wobs = this.http.get<Array<SprRecordingFile>>(recFilesReqUrl,{withCredentials:this.withCredentials});\r\n    return wobs;\r\n  }\r\n\r\n  private audioRequest(audioUrl:string): Observable<HttpResponse<ArrayBuffer>> {\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    }\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  private fetchAudiofile(projectName: string, sessId: string | number, recFileId: string|number): Observable<HttpResponse<ArrayBuffer>> {\r\n    let recFilesUrl=this.sessionRecordingFilesUrl(projectName,sessId);\r\n    let encRecFileId=encodeURIComponent(recFileId);\r\n    let recUrl = recFilesUrl + '/' + encRecFileId;\r\n    return this.audioRequest(recUrl);\r\n  }\r\n\r\n  private fetchSprAudiofile(projectName: string, sessId: string | number, itemcode: string,version:number): Observable<HttpResponse<ArrayBuffer>> {\r\n    let recFilesUrl=this.sessionRecFilesUrl(projectName,sessId);\r\n    let encItemcode=encodeURIComponent(itemcode);\r\n    let recUrl = recFilesUrl + '/' + encItemcode +'/'+version;\r\n    return this.audioRequest(recUrl);\r\n  }\r\n\r\n  fetchRecordingFileAudioBuffer(aCtx: AudioContext, projectName: string, recordingFile:RecordingFile):Observable<AudioBuffer|null> {\r\n\r\n    let wobs = new Observable<AudioBuffer|null>(observer=>{\r\n      let recFileId=recordingFile.recordingFileId;\r\n      if(!recFileId){\r\n        recFileId=recordingFile.uuid;\r\n      }\r\n      if(recordingFile.session && recFileId) {\r\n\r\n        let obs = this.fetchAudiofile(projectName, recordingFile.session, recFileId);\r\n        obs.subscribe({\r\n          next: resp => {\r\n\r\n            // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n            if (resp.body) {\r\n              aCtx.decodeAudioData(resp.body, ab => {\r\n                  observer.next(ab);\r\n                  observer.complete();\r\n\r\n              }, error => {\r\n                observer.error(error);\r\n                observer.complete();\r\n              })\r\n            } else {\r\n              observer.error('Fetching audio file: response has no body');\r\n            }\r\n          },\r\n          error:(error: 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              observer.error(error);\r\n              observer.complete();\r\n            }\r\n          }});\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchAndApplyRecordingFile(aCtx: AudioContext, projectName: string, recordingFile:RecordingFile):Observable<RecordingFile|null> {\r\n\r\n    let wobs = new Observable<RecordingFile|null>(observer=>{\r\n      let recFileId=recordingFile.recordingFileId;\r\n      if(!recFileId){\r\n        recFileId=recordingFile.uuid;\r\n      }\r\n      if(recordingFile.session && recFileId) {\r\n\r\n        let obs = this.fetchAudiofile(projectName, recordingFile.session, recFileId);\r\n        obs.subscribe(resp => {\r\n            //console.log(\"Fetched audio file. HTTP response status: \"+resp.status+\", type: \"+resp.type+\", byte length: \"+ resp.body.byteLength);\r\n\r\n            // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n            if (resp.body) {\r\n              aCtx.decodeAudioData(resp.body, ab => {\r\n                let adh=new AudioDataHolder(ab,null);\r\n                RecordingFileUtils.setAudioData(recordingFile,adh);\r\n                if (this.debugDelay > 0) {\r\n                  window.setTimeout(() => {\r\n\r\n                    observer.next(recordingFile);\r\n                    observer.complete();\r\n                  }, this.debugDelay);\r\n                } else {\r\n                  observer.next(recordingFile);\r\n                  observer.complete();\r\n                }\r\n              }, error => {\r\n                observer.error(error);\r\n                observer.complete();\r\n              })\r\n            } else {\r\n              observer.error('Fetching audio file: response has no body');\r\n            }\r\n          },\r\n          (error: 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              observer.error(error);\r\n              observer.complete();\r\n            }\r\n          });\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchSprRecordingFileAudioBuffer(aCtx: AudioContext, projectName: string, recordingFile:SprRecordingFile):Observable<AudioBuffer|null> {\r\n\r\n    let wobs = new Observable<AudioBuffer|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let obs = this.fetchSprAudiofile(projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);\r\n        obs.subscribe({\r\n          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              aCtx.decodeAudioData(resp.body, ab => {\r\n                //RecordingFileUtils.setAudioData(recordingFile,new AudioDataHolder(ab,null));\r\n                observer.next(ab);\r\n                observer.complete();\r\n\r\n              }, error => {\r\n                observer.error(error);\r\n                observer.complete();\r\n              })\r\n            } else {\r\n              observer.error('Fetching audio file: response has no body');\r\n            }\r\n          },\r\n          error: (error: 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              observer.error(error);\r\n              observer.complete();\r\n            }\r\n          }});\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n\r\n  fetchAndApplySprRecordingFile(aCtx: AudioContext, projectName: string, recordingFile:SprRecordingFile):Observable<SprRecordingFile|null> {\r\n\r\n    let wobs = new Observable<SprRecordingFile|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let obs = this.fetchSprAudiofile(projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);\r\n        obs.subscribe(resp => {\r\n              //console.log(\"Fetched audio file. HTTP response status: \"+resp.status+\", type: \"+resp.type+\", byte length: \"+ resp.body.byteLength);\r\n\r\n              // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n              if (resp.body) {\r\n                aCtx.decodeAudioData(resp.body, ab => {\r\n                  RecordingFileUtils.setAudioData(recordingFile,new AudioDataHolder(ab,null));\r\n                  if (this.debugDelay > 0) {\r\n                    window.setTimeout(() => {\r\n\r\n                      observer.next(recordingFile);\r\n                      observer.complete();\r\n                    }, this.debugDelay);\r\n                  } else {\r\n                    observer.next(recordingFile);\r\n                    observer.complete();\r\n                  }\r\n                }, error => {\r\n                  observer.error(error);\r\n                  observer.complete();\r\n                })\r\n              } else {\r\n                observer.error('Fetching audio file: response has no body');\r\n              }\r\n            },\r\n            (error: 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                observer.error(error);\r\n                observer.complete();\r\n              }\r\n            });\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchRecordingFile(aCtx: AudioContext, projectName: string, sessId: string | number, itemcode: string,version:number):Observable<SprRecordingFile|null> {\r\n\r\n    let wobs = new Observable<SprRecordingFile | null>(observer=>{\r\n      let obs = this.fetchSprAudiofile(projectName, sessId, itemcode,version);\r\n\r\n\r\n      obs.subscribe(resp => {\r\n            // Do not use Promise version, which does not work with Safari 13\r\n          if(resp.body) {\r\n            aCtx.decodeAudioData(resp.body, ab => {\r\n              let adh=new AudioDataHolder(ab,null);\r\n              let rf = new SprRecordingFile(sessId, itemcode, version, adh);\r\n              if (this.debugDelay > 0) {\r\n                window.setTimeout(() => {\r\n\r\n                  observer.next(rf);\r\n                  observer.complete();\r\n                }, this.debugDelay);\r\n              } else {\r\n                observer.next(rf);\r\n                observer.complete();\r\n              }\r\n            });\r\n          }else{\r\n            observer.error();\r\n          }\r\n        },\r\n        (error: 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            observer.error(error);\r\n            observer.complete();\r\n          }\r\n        });\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n}\r\n\r\n\r\n\r\n"]}
|
|
1005
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"recordings.service.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/recordings/recordings.service.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Created by klausj on 17.06.2017.\r\n */\r\nimport {Inject, Injectable} from '@angular/core';\r\nimport {HttpClient, HttpErrorResponse, HttpHeaders, HttpResponse} from \"@angular/common/http\";\r\nimport {ApiType, SPEECHRECORDER_CONFIG, SpeechRecorderConfig} from \"../../spr.config\";\r\n\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {SprRecordingFile, RecordingFileDescriptorImpl, RecordingFile, RecordingFileUtils} from \"../recording\";\r\nimport {ProjectService} from \"../project/project.service\";\r\nimport {SessionService} from \"../session/session.service\";\r\nimport {EMPTY, expand, map, Observable} from \"rxjs\";\r\nimport {AudioBufferSource, AudioDataHolder} from \"../../audio/audio_data_holder\";\r\nimport {ArrayAudioBuffer} from \"../../audio/array_audio_buffer\";\r\nimport {IndexedDbAudioBuffer, PersistentAudioStorageTarget} from \"../../audio/inddb_audio_buffer\";\r\nimport {NetAudioBuffer} from \"../../audio/net_audio_buffer\";\r\nimport {BasicRecordingService} from \"./basic_recording.service\";\r\n\r\n@Injectable()\r\nexport class RecordingService extends BasicRecordingService{\r\n\r\n  // This seems to work\r\n  //public static readonly DEFAULT_CHUNKED_DOWNLOAD_SECONDS:number=4;\r\n\r\n  public static readonly REC_API_CTX = 'recfile'\r\n  public static readonly RECORDING_API_CTX='recordingfile'\r\n\r\n\r\n  //private debugDelay:number=10000;\r\n  private debugDelay:number=0;\r\n\r\n  constructor(protected http: HttpClient, @Inject(SPEECHRECORDER_CONFIG) protected config?: SpeechRecorderConfig) {\r\n    super(http, config);\r\n  }\r\n\r\n  private recordingFilesUrl():string{\r\n    let recFilesUrl = this.apiEndPoint  + RecordingService.RECORDING_API_CTX;\r\n    return recFilesUrl;\r\n  }\r\n\r\n  private sessionRecordingFilesUrl(projectName: string, sessId: string | number):string{\r\n    let encPrjName=encodeURIComponent(projectName);\r\n    let encSessId=encodeURIComponent(sessId);\r\n    let recFilesUrl = this.apiEndPoint + ProjectService.PROJECT_API_CTX + '/' + encPrjName + '/' +\r\n      SessionService.SESSION_API_CTX + '/' + encSessId + '/' + RecordingService.RECORDING_API_CTX;\r\n    return recFilesUrl;\r\n  }\r\n\r\n  // private sessionRecordingFilesUrl(projectName: string, sessId: string | number):URL{\r\n  //   let encPrjName=encodeURIComponent(projectName);\r\n  //   let encSessId=encodeURIComponent(sessId);\r\n  //\r\n  //   let recFilesUrlStr = this.apiEndPoint + ProjectService.PROJECT_API_CTX + '/' + encPrjName + '/' +\r\n  //     SessionService.SESSION_API_CTX + '/' + encSessId + '/' + RecordingService.RECORDING_API_CTX;\r\n  //   let recFilesUrl=new URL(recFilesUrlStr);\r\n  //   return recFilesUrl;\r\n  // }\r\n\r\n  private sessionRecFilesUrl(projectName: string, sessId: string | number):string{\r\n    let encPrjName=encodeURIComponent(projectName);\r\n    let encSessId=encodeURIComponent(sessId);\r\n    let recFilesUrl = this.apiEndPoint + ProjectService.PROJECT_API_CTX + '/' + encPrjName + '/' +\r\n        SessionService.SESSION_API_CTX + '/' + encSessId + '/' + RecordingService.REC_API_CTX;\r\n    return recFilesUrl;\r\n  }\r\n\r\n  private sessionRecFilesReqUrl(projectName: string, sessId: string | number):string{\r\n    let recFilesUrl=this.sessionRecFilesUrl(projectName,sessId);\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      recFilesUrl = recFilesUrl + '.json?requestUUID=' + UUID.generate();\r\n    }\r\n    return recFilesUrl;\r\n  }\r\n\r\n  recordingFileDescrList(projectName: string, sessId: string | number):Observable<Array<RecordingFileDescriptorImpl>> {\r\n    let recFilesReqUrl = this.sessionRecFilesReqUrl(projectName,sessId);\r\n    let wobs = this.http.get<Array<RecordingFileDescriptorImpl>>(recFilesReqUrl,{withCredentials:this.withCredentials});\r\n    return wobs;\r\n  }\r\n\r\n  recordingFileList(projectName: string, sessId: string | number):Observable<Array<RecordingFile>> {\r\n    let recFilesReqUrl = this.sessionRecFilesReqUrl(projectName,sessId);\r\n    let wobs = this.http.get<Array<RecordingFile>>(recFilesReqUrl,{withCredentials:this.withCredentials});\r\n    return wobs;\r\n  }\r\n\r\n  sprRecordingFileList(projectName: string, sessId: string | number):Observable<Array<SprRecordingFile>> {\r\n    let recFilesReqUrl = this.sessionRecFilesReqUrl(projectName,sessId);\r\n    let wobs = this.http.get<Array<SprRecordingFile>>(recFilesReqUrl,{withCredentials:this.withCredentials});\r\n    return wobs;\r\n  }\r\n\r\n  private audioRequest(audioUrl:string): Observable<HttpResponse<ArrayBuffer>> {\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    }\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\r\n\r\n\r\n\r\n  private chunkAudioRequestToIndDb(aCtx:AudioContext,persistentAudioStorageTarget:PersistentAudioStorageTarget,inddbAudioBuffer:IndexedDbAudioBuffer|null,baseAudioUrl:string,startFrame:number=0,orgSampleRate: number,seconds:number): Observable<IndexedDbAudioBuffer|null> {\r\n    //let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;\r\n    //let audioUrl=new URL(baseAudioUrl);\r\n    let frameLength=orgSampleRate * Math.round(seconds); // Important: multiple of original sample rate to prevent numeric rounding errors on resampling. 10 seconds is a good value for ind db storage.\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<IndexedDbAudioBuffer|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(\"chunkAudioRequestToIndDb: subscriber.closed: \"+subscriber.closed);\r\n            //console.debug(\"chunkAudioRequestToIndDb: Audio file bytes: \"+resp.body.byteLength);\r\n            aCtx.decodeAudioData(resp.body, ab => {\r\n                //console.debug(\"chunkAudioRequestToIndDb: Decoded audio chunk frames for inddb: \"+ab.length);\r\n\r\n                if(!inddbAudioBuffer){\r\n                  //console.debug(\"chunkAudioRequestToIndDb: Create inddb ab from chunk ab...\");\r\n                 IndexedDbAudioBuffer.fromChunkAudioBuffer(persistentAudioStorageTarget,ab).subscribe({\r\n                    next:(iab)=>{\r\n                      //console.debug(\"chunkAudioRequestToIndDb: Built inddb ab from chunk ab: \"+iab);\r\n                      inddbAudioBuffer=iab;\r\n                      if(inddbAudioBuffer.frameLen<frameLength){\r\n                        //console.debug(\"chunkAudioRequestToIndDb: Built inddb ab from chunk ab: First chunk shorter tha frameLength (\"+inddbAudioBuffer.frameLen+\"<\"+frameLength+\"), assuming end of data, sealing inddb ab.\");\r\n                        inddbAudioBuffer.seal();\r\n                      }\r\n                      subscriber.next(iab);\r\n                    },\r\n                   complete:()=>{\r\n                     //console.debug(\"chunkAudioRequestToIndDb: Built inddb ab from chunk ab complete.\");\r\n                     subscriber.complete();\r\n                   },\r\n                   error:(err)=>{\r\n                      // iPad may throw QuotaExceededError here\r\n                     console.error(\"chunkAudioRequestToIndDb: Built inddb ab from chunk ab error: \"+err);\r\n                      subscriber.error(err);\r\n                   }\r\n                  })\r\n                }else {\r\n                  //console.debug(\"chunkAudioRequestToIndDb: Append audio chunk to inddb ab...\");\r\n                  inddbAudioBuffer.appendChunkAudioBuffer(ab).subscribe({\r\n                    next:(iab)=>{\r\n                      // if(inddbAudioBuffer) {\r\n                      //   console.debug(\"chunkAudioRequestToIndDb: Appended audio chunk to inddb audio buffer: \"+inddbAudioBuffer);\r\n                      // }else{\r\n                      //   console.debug(\"chunkAudioRequestToIndDb: Append audio chunk returned null\");\r\n                      // }\r\n                      subscriber.next(inddbAudioBuffer);\r\n                    },\r\n                    complete:()=>{\r\n                      subscriber.complete();\r\n                    },\r\n                    error:(err)=>{\r\n                      subscriber.error(err);\r\n                    }\r\n                  })\r\n                }\r\n              }\r\n              , error => {\r\n                console.error('chunkAudioRequestToIndDb: error: '+error);\r\n                //if(error instanceof HttpErrorResponse) {\r\n                  subscriber.error(error);\r\n                //}\r\n              })\r\n          } else {\r\n            console.error('chunkAudioRequestToIndDb: Fetching audio file: response has no body');\r\n            subscriber.error('chunkAudioRequestToIndDb: Fetching audio file: response has no body');\r\n          }\r\n        },\r\n        error:(error) => {\r\n          console.error('chunkAudioRequestToIndDb: 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\r\n\r\n  private chunkedAudioRequestToArrayBuffer(aCtx:AudioContext,baseAudioUrl:string,orgSampleRate:number,seconds:number): Observable<ArrayAudioBuffer|null> {\r\n    let obs=new Observable<ArrayAudioBuffer|null>(subscriber => {\r\n      let arrayAudioBuffer: ArrayAudioBuffer | null = null;\r\n      let startFrame=0;\r\n      let frameLength=orgSampleRate * Math.round(seconds); // Important: multiple of original sample rate to prevent numeric rounding errors on resampling.\r\n      //console.debug(\"Chunk audio request startFrame 0\");\r\n      let subscr=this.chunkAudioRequest(aCtx, baseAudioUrl, startFrame, frameLength).pipe(\r\n\r\n        expand(value => {\r\n          if(subscriber.closed){\r\n            subscr.unsubscribe();\r\n          }\r\n            if (value) {\r\n              if (arrayAudioBuffer) {\r\n                if (arrayAudioBuffer?.sealed()) {\r\n                  return EMPTY;\r\n                } else {\r\n                  //let nextStartFrame=arrayAudioBuffer.frameLen + 1;\r\n                  // We cannot use the frame length of the array audio buffer to determine the next start frame because\r\n                  // AudioContext.decodeAudioData might change the sample rate of the original file. (E.g. Recordings from Mac/iOS with SR 44100 and listen to the on Windows platform will resmaple the audio data to 48000.)\r\n                  // Frame count then differs on client and server\r\n\r\n                  // Simply proceed in steps of frameLength\r\n                  // More advanced method would be to parse the WAV header to find out the rela frame length of the chunk audio file\r\n                  startFrame+=frameLength;\r\n                  //console.debug(\"Next start frame: \"+startFrame);\r\n                  //console.debug(\"chunkedAudioRequest: expand() subscriber.closed: \"+subscriber.closed);\r\n                  return this.chunkAudioRequest(aCtx, baseAudioUrl, startFrame, frameLength);\r\n                }\r\n              } else {\r\n                return EMPTY;\r\n              }\r\n            }else{\r\n              return EMPTY;\r\n            }\r\n          }\r\n        ),\r\n      map(chDl => {\r\n        let ab:AudioBuffer|null=null;\r\n        if(chDl===null){\r\n          return null;\r\n        }else{\r\n          ab=chDl.decodedAudioBuffer;\r\n        }\r\n        if(ab) {\r\n          if (arrayAudioBuffer) {\r\n            arrayAudioBuffer.appendAudioBuffer(ab);\r\n          } else {\r\n            arrayAudioBuffer = ArrayAudioBuffer.fromAudioBuffer(ab, ab.length);\r\n          }\r\n          // Following code is wrong if original samplerate is less-then platform samplerate\r\n          // E.g. Loading 44100Hz file on Safari\r\n          // console.debug(\"Array ab seal?: \"+ab.length+\" < \"+frameLength);\r\n          // if (ab.length < frameLength) {\r\n          //   arrayAudioBuffer.seal();\r\n          //   console.debug(\"Array ab sealed.\");\r\n          // }\r\n        }\r\n        return arrayAudioBuffer;\r\n      })\r\n      ).subscribe({\r\n        next: (aab)=>{\r\n          //console.debug(\"chunkedAudioRequest: subscriber.closed: \"+subscriber.closed);\r\n          subscriber.next(aab)\r\n        },\r\n        complete: ()=>{\r\n          subscriber.complete()\r\n        },\r\n        error: (err)=>{\r\n          if(err instanceof HttpErrorResponse){\r\n            if (err.status == 404) {\r\n              if(arrayAudioBuffer){\r\n                arrayAudioBuffer.seal();\r\n                subscriber.next(arrayAudioBuffer);\r\n                subscriber.complete();\r\n              }else {\r\n                // Interpret not as an error, the file ist not recorded yet\r\n                subscriber.next(null);\r\n                subscriber.complete();\r\n              }\r\n            //   observer.complete()\r\n             } else {\r\n              // all other states are errors\r\n              subscriber.error(err);\r\n             }\r\n          }else{\r\n            subscriber.error(err);\r\n          }\r\n        }\r\n      });\r\n    });\r\n    return obs;\r\n  }\r\n\r\n  private chunkedInddbAudioRequest(aCtx:AudioContext,persistentAudioStorageTarget:PersistentAudioStorageTarget,baseAudioUrl:string,orgSampleRate:number,seconds:number): Observable<IndexedDbAudioBuffer|null> {\r\n    let obs=new Observable<IndexedDbAudioBuffer|null>(subscriber => {\r\n      let inddbAudioBuffer: IndexedDbAudioBuffer | null = null;\r\n      let startFrame=0;\r\n      //let frameLength = DEFAULT_CHUNKED_DOWNLOAD_FRAMELENGTH;\r\n      let frameLength=orgSampleRate*Math.round(seconds);\r\n      //console.debug(\"chunkedInddbAudioRequest: Chunk audio request for inddb. startFrame: \"+startFrame);\r\n      let subscr=this.chunkAudioRequestToIndDb(aCtx,persistentAudioStorageTarget,null, baseAudioUrl, startFrame,orgSampleRate,seconds).pipe(\r\n\r\n        expand(iab => {\r\n         // console.debug(\"chunkedInddbAudioRequest (pipe/expand): Got inddb ab: \"+iab);\r\n            if(subscriber.closed){\r\n              subscr.unsubscribe();\r\n            }\r\n            if (iab) {\r\n              inddbAudioBuffer=iab;\r\n              //if (inddbAudioBuffer) {\r\n                if (inddbAudioBuffer?.sealed()) {\r\n                  return EMPTY;\r\n                } else {\r\n                  //let nextStartFrame=inddbAudioBuffer.frameLen + 1;\r\n                  // We cannot use the frame length of the array audio buffer to determine the next start frame because\r\n                  // AudioContext.decodeAudioData might change the sample rate of the original file. (E.g. Recordings from Mac/iOS with SR 44100 and listen to the on Windows platform will resmaple the audio data to 48000.)\r\n                  // Frame count then differs on client and server\r\n\r\n                  // Simply proceed in steps of frameLength\r\n                  // More advanced method would be to parse the WAV header to find out the real frame length of the chunk audio file\r\n                  startFrame+=frameLength;\r\n                  //console.debug(\"Next start frame: \"+startFrame);\r\n                  //console.debug(\"chunkedInddbAudioRequest: expand() subscriber.closed: \"+subscriber.closed);\r\n                  return this.chunkAudioRequestToIndDb(aCtx,persistentAudioStorageTarget,inddbAudioBuffer, baseAudioUrl, startFrame, orgSampleRate,seconds);\r\n                }\r\n              // } else {\r\n              //   return EMPTY;\r\n              // }\r\n            }else{\r\n              return EMPTY;\r\n            }\r\n          }\r\n        )\r\n      ).subscribe({\r\n        next: (aab)=>{\r\n          //console.debug(\"chunkedInddbAudioRequest: subscriber.closed: \"+subscriber.closed);\r\n          subscriber.next(aab)\r\n        },\r\n        complete: ()=>{\r\n          subscriber.complete()\r\n        },\r\n        error: (err)=>{\r\n          if(err instanceof HttpErrorResponse){\r\n            if (err.status == 404) {\r\n              //console.debug(\"chunkedInddbAudioRequest: Received HTTP 404 not found.\");\r\n              if(inddbAudioBuffer){\r\n                inddbAudioBuffer.seal();\r\n                //console.debug(\"chunkedInddbAudioRequest: Sealed inddb audio buffer.\");\r\n                subscriber.next(inddbAudioBuffer);\r\n                subscriber.complete();\r\n              }else {\r\n                // Interpret not as an error, the file ist not recorded yet\r\n                //console.debug(\"chunkedInddbAudioRequest: Interpret HTTP 404 as not recorded yet.\");\r\n                subscriber.next(null);\r\n                subscriber.complete();\r\n              }\r\n              //   observer.complete()\r\n            } else {\r\n              // all other states are (real) errors\r\n              console.error(\"chunkedInddbAudioRequest: Error: \"+err.message);\r\n              subscriber.error(err);\r\n            }\r\n          }else {\r\n            // all other errors are (real) errors\r\n            console.error(\"chunkedInddbAudioRequest: Error: \"+err);\r\n            subscriber.error(err);\r\n          }\r\n        }\r\n      });\r\n    });\r\n    return obs;\r\n  }\r\n\r\n\r\n  // private chunkedNetAudioRequest(aCtx:AudioContext,baseAudioUrl:string): Observable<NetAudioBuffer|null> {\r\n  //   let obs=new Observable<NetAudioBuffer|null>(subscriber => {\r\n  //     let inddbAudioBuffer: NetAudioBuffer | null = null;\r\n  //     let startFrame=0;\r\n  //     let frameLength = DEFAULT_CHUNKED_DOWNLOAD_FRAMELENGTH;\r\n  //     //console.debug(\"chunkedInddbAudioRequest: Chunk audio request for inddb. startFrame: \"+startFrame);\r\n  //     let subscr=this.chunkAudioRequestToIndDb(aCtx,persistentAudioStorageTarget,null, baseAudioUrl, startFrame, frameLength).pipe(\r\n  //\r\n  //         expand(iab => {\r\n  //               // console.debug(\"chunkedInddbAudioRequest (pipe/expand): Got inddb ab: \"+iab);\r\n  //               if(subscriber.closed){\r\n  //                 subscr.unsubscribe();\r\n  //               }\r\n  //               if (iab) {\r\n  //                 inddbAudioBuffer=iab;\r\n  //                 //if (inddbAudioBuffer) {\r\n  //                 if (inddbAudioBuffer?.sealed()) {\r\n  //                   return EMPTY;\r\n  //                 } else {\r\n  //                   //let nextStartFrame=inddbAudioBuffer.frameLen + 1;\r\n  //                   // We cannot use the frame length of the array audio buffer to determine the next start frame because\r\n  //                   // AudioContext.decodeAudioData might change the sample rate of the original file. (E.g. Recordings from Mac/iOS with SR 44100 and listen to the on Windows platform will resmaple the audio data to 48000.)\r\n  //                   // Frame count then differs on client and server\r\n  //\r\n  //                   // Simply proceed in steps of frameLength\r\n  //                   // More advanced method would be to parse the WAV header to find out the rela frame length of the chunk audio file\r\n  //                   startFrame+=frameLength;\r\n  //                   //console.debug(\"Next start frame: \"+startFrame);\r\n  //                   //console.debug(\"chunkedInddbAudioRequest: expand() subscriber.closed: \"+subscriber.closed);\r\n  //                   return this.chunkAudioRequestToIndDb(aCtx,persistentAudioStorageTarget,inddbAudioBuffer, baseAudioUrl, startFrame, frameLength);\r\n  //                 }\r\n  //                 // } else {\r\n  //                 //   return EMPTY;\r\n  //                 // }\r\n  //               }else{\r\n  //                 return EMPTY;\r\n  //               }\r\n  //             }\r\n  //         )\r\n  //     ).subscribe({\r\n  //       next: (aab)=>{\r\n  //         //console.debug(\"chunkedInddbAudioRequest: subscriber.closed: \"+subscriber.closed);\r\n  //         subscriber.next(aab)\r\n  //       },\r\n  //       complete: ()=>{\r\n  //         subscriber.complete()\r\n  //       },\r\n  //       error: (err)=>{\r\n  //         if(err instanceof HttpErrorResponse){\r\n  //           if (err.status == 404) {\r\n  //             //console.debug(\"chunkedInddbAudioRequest: Received HTTP 404 not found.\");\r\n  //             if(inddbAudioBuffer){\r\n  //               inddbAudioBuffer.seal();\r\n  //               //console.debug(\"chunkedInddbAudioRequest: Sealed inddb audio buffer.\");\r\n  //               subscriber.next(inddbAudioBuffer);\r\n  //               subscriber.complete();\r\n  //             }else {\r\n  //               // Interpret not as an error, the file ist not recorded yet\r\n  //               //console.debug(\"chunkedInddbAudioRequest: Interpret HTTP 404 as not recorded yet.\");\r\n  //               subscriber.next(null);\r\n  //               subscriber.complete();\r\n  //             }\r\n  //             //   observer.complete()\r\n  //           } else {\r\n  //             // all other states are (real) errors\r\n  //             console.error(\"chunkedInddbAudioRequest: Error: \"+err.message);\r\n  //             subscriber.error(err);\r\n  //           }\r\n  //         }else {\r\n  //           // all other errors are (real) errors\r\n  //           console.error(\"chunkedInddbAudioRequest: Error: \"+err);\r\n  //           subscriber.error(err);\r\n  //         }\r\n  //       }\r\n  //     });\r\n  //   });\r\n  //   return obs;\r\n  // }\r\n\r\n  audioFileUrlById(projectName: string, sessId: string | number, recFileId: string|number){\r\n    let recFilesUrl=this.sessionRecordingFilesUrl(projectName,sessId);\r\n    let encRecFileId=encodeURIComponent(recFileId);\r\n    return recFilesUrl + '/' + encRecFileId;\r\n  }\r\n\r\n  private fetchAudiofile(projectName: string, sessId: string | number, recFileId: string|number): Observable<HttpResponse<ArrayBuffer>> {\r\n    let recUrl = this.audioFileUrlById(projectName,sessId,recFileId);\r\n    return this.audioRequest(recUrl);\r\n  }\r\n\r\n  audioFileUrl(projectName: string, recordingFile:RecordingFile):string|null {\r\n    let url:string|null = null;\r\n    let recFileId = recordingFile.recordingFileId;\r\n    if (!recFileId) {\r\n      recFileId = recordingFile.uuid;\r\n    }\r\n    if (recordingFile.session && recFileId) {\r\n      url = this.audioFileUrlById(projectName, recordingFile.session, recFileId);\r\n    }\r\n    return url;\r\n  }\r\n\r\n  audioFileUrlByUUID(projectName: string, session:string|number,uuid:string):string|null {\r\n    let url = this.audioFileUrlById(projectName, session, uuid);\r\n    return url;\r\n  }\r\n\r\n  private fetchSprAudiofile(projectName: string, sessId: string | number, itemcode: string,version:number): Observable<HttpResponse<ArrayBuffer>> {\r\n    let recFilesUrl=this.sessionRecFilesUrl(projectName,sessId);\r\n    let encItemcode=encodeURIComponent(itemcode);\r\n    let recUrl = recFilesUrl + '/' + encItemcode +'/'+version;\r\n    return this.audioRequest(recUrl);\r\n  }\r\n\r\n  public sprAudioFileUrlByItemcode(projectName: string, sessId: string | number, itemcode: string,version:number){\r\n    let recFilesUrl=this.sessionRecFilesUrl(projectName,sessId);\r\n    let encItemcode=encodeURIComponent(itemcode);\r\n    let recUrl = recFilesUrl + '/' + encItemcode +'/'+version;\r\n    return recUrl;\r\n  }\r\n\r\n  sprAudioFileUrl(projectName: string, recordingFile:SprRecordingFile):string|null{\r\n    let url=null;\r\n    if(recordingFile.session) {\r\n      url = this.sprAudioFileUrlByItemcode(projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);\r\n    }\r\n    return url;\r\n  }\r\n\r\n\r\n\r\n  // private fetchSprAudiofileArrayBuffer(aCtx:AudioContext,projectName: string, sessId: string | number, itemcode: string,version:number): Observable<ArrayAudioBuffer|null>{\r\n  //   let recFilesUrl=this.sessionRecFilesUrl(projectName,sessId);\r\n  //   let encItemcode=encodeURIComponent(itemcode);\r\n  //   let recUrl = recFilesUrl + '/' + encItemcode +'/'+version;\r\n  //   return this.chunkedAudioRequest(aCtx,recUrl);\r\n  // }\r\n\r\n  // private fetchSprAudiofileInddbBuffer(aCtx:AudioContext, persistentAudioStorageTarget:PersistentAudioStorageTarget,projectName: string, sessId: string | number, itemcode: string,version:number): Observable<IndexedDbAudioBuffer|null>{\r\n  //   let recFilesUrlStr=this.sessionRecFilesUrl(projectName,sessId);\r\n  //   let encItemcode=encodeURIComponent(itemcode);\r\n  //   let recUrlStr = recFilesUrlStr + '/' + encItemcode +'/'+version;\r\n  //   //let recUrl=new URL(recUrlStr);\r\n  //   return this.chunkedInddbAudioRequest(aCtx,persistentAudioStorageTarget,recUrlStr);\r\n  // }\r\n\r\n  // private sprNetAudiofileBuffer(aCtx:AudioContext,projectName: string, sessId: string | number, itemcode: string,version:number): Observable<IndexedDbAudioBuffer|null>{\r\n  //   let recFilesUrlStr=this.sessionRecFilesUrl(projectName,sessId);\r\n  //   let encItemcode=encodeURIComponent(itemcode);\r\n  //   let recUrlStr = recFilesUrlStr + '/' + encItemcode +'/'+version;\r\n  //   //let recUrl=new URL(recUrlStr);\r\n  //   return this.chunkedInddbAudioRequest(aCtx,recUrlStr);\r\n  // }\r\n\r\n  fetchRecordingFileAudioBuffer(aCtx: AudioContext, projectName: string, recordingFile:RecordingFile):Observable<AudioBuffer|null> {\r\n\r\n    let wobs = new Observable<AudioBuffer|null>(observer=>{\r\n      let recFileId=recordingFile.recordingFileId;\r\n      if(!recFileId){\r\n        recFileId=recordingFile.uuid;\r\n      }\r\n      if(recordingFile.session && recFileId) {\r\n\r\n        let obs = this.fetchAudiofile(projectName, recordingFile.session, recFileId);\r\n        obs.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              aCtx.decodeAudioData(resp.body, ab => {\r\n                  observer.next(ab);\r\n                  observer.complete();\r\n              }, error => {\r\n                observer.error(error);\r\n                observer.complete();\r\n              })\r\n            } else {\r\n              observer.error('Fetching audio file: response has no body');\r\n            }\r\n          }, (err) => {\r\n          if(err instanceof HttpErrorResponse) {\r\n            if (err.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              observer.error(err);\r\n              observer.complete();\r\n            }\r\n          }else{\r\n            observer.error(err);\r\n          }\r\n          });\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchAndApplyRecordingFile(aCtx: AudioContext, projectName: string, recordingFile:RecordingFile):Observable<RecordingFile|null> {\r\n\r\n    let wobs = new Observable<RecordingFile|null>(observer=>{\r\n      let recFileId=recordingFile.recordingFileId;\r\n      if(!recFileId){\r\n        recFileId=recordingFile.uuid;\r\n      }\r\n      if(recordingFile.session && recFileId) {\r\n\r\n        let obs = this.fetchAudiofile(projectName, recordingFile.session, recFileId);\r\n        obs.subscribe(resp => {\r\n            //console.log(\"Fetched audio file. HTTP response status: \"+resp.status+\", type: \"+resp.type+\", byte length: \"+ resp.body.byteLength);\r\n\r\n            // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n            if (resp.body) {\r\n              aCtx.decodeAudioData(resp.body, ab => {\r\n                let abs=new AudioBufferSource(ab);\r\n                let adh=new AudioDataHolder(abs);\r\n                RecordingFileUtils.setAudioData(recordingFile,adh);\r\n                if (this.debugDelay > 0) {\r\n                  window.setTimeout(() => {\r\n\r\n                    observer.next(recordingFile);\r\n                    observer.complete();\r\n                  }, this.debugDelay);\r\n                } else {\r\n                  observer.next(recordingFile);\r\n                  observer.complete();\r\n                }\r\n              }, error => {\r\n                observer.error(error);\r\n                observer.complete();\r\n              })\r\n            } else {\r\n              observer.error('Fetching audio file: response has no body');\r\n            }\r\n          },\r\n          (err) => {\r\n          if(err instanceof HttpErrorResponse) {\r\n            if (err.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              observer.error(err);\r\n              observer.complete();\r\n            }\r\n          }else{\r\n            observer.error(err);\r\n          }\r\n          });\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchSprRecordingFileAudioBuffer(aCtx: AudioContext, projectName: string, recordingFile:SprRecordingFile):Observable<AudioBuffer|null> {\r\n\r\n    let wobs = new Observable<AudioBuffer|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let obs = this.fetchSprAudiofile(projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);\r\n        obs.subscribe({\r\n          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              aCtx.decodeAudioData(resp.body, ab => {\r\n                //RecordingFileUtils.setAudioData(recordingFile,new AudioDataHolder(ab,null));\r\n                observer.next(ab);\r\n                observer.complete();\r\n\r\n              }, error => {\r\n                observer.error(error);\r\n                observer.complete();\r\n              })\r\n            } else {\r\n              observer.error('Fetching audio file: response has no body');\r\n            }\r\n          },\r\n          error: (err) => {\r\n            if (err instanceof HttpErrorResponse && err.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              observer.error(err);\r\n            }\r\n          }});\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n\r\n  fetchSprRecordingFileArrayAudioBuffer(aCtx: AudioContext, projectName: string, recordingFile:SprRecordingFile):Observable<ArrayAudioBuffer|null> {\r\n    let wobs = new Observable<ArrayAudioBuffer|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let baseUrl=this.sprAudioFileUrl(projectName,recordingFile);\r\n        if(baseUrl) {\r\n          if(recordingFile.samplerate) {\r\n            let lengthInSeconds = RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;\r\n            let obs = this.chunkedAudioRequestToArrayBuffer(aCtx, baseUrl, recordingFile.samplerate, lengthInSeconds);\r\n\r\n            //let obs = this.fetchSprAudiofileArrayBuffer(aCtx,projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);\r\n            let subscr = obs.subscribe({\r\n              next: aab => {\r\n                //console.debug(\"fetchSprRecordingFileArrayAudioBuffer: observer.closed: \"+observer.closed);\r\n                if (observer.closed) {\r\n                  subscr.unsubscribe()\r\n                }\r\n                observer.next(aab)\r\n              },\r\n              complete: () => {\r\n                observer.complete();\r\n              },\r\n              error: (err) => {\r\n                if (err instanceof HttpErrorResponse && err.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                  observer.error(err);\r\n                }\r\n              }\r\n            });\r\n          }else{\r\n            observer.error(new Error('Unknown sample rate of recording file ID: '+recordingFile.recordingFileId));\r\n          }\r\n        }\r\n      }else{\r\n        observer.error(new Error('Could not get session ID of recording file'));\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchRecordingFileArrayAudioBuffer(aCtx: AudioContext, projectName: string, recordingFile:RecordingFile):Observable<ArrayAudioBuffer|null> {\r\n    let wobs = new Observable<ArrayAudioBuffer|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let baseUrl=this.audioFileUrl(projectName,recordingFile);\r\n        if(baseUrl) {\r\n          if(recordingFile.samplerate) {\r\n            let lengthInSeconds = RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;\r\n            let obs = this.chunkedAudioRequestToArrayBuffer(aCtx, baseUrl, recordingFile.samplerate, lengthInSeconds);\r\n\r\n            //let obs = this.fetchSprAudiofileArrayBuffer(aCtx,projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);\r\n            let subscr = obs.subscribe({\r\n              next: aab => {\r\n                //console.debug(\"fetchSprRecordingFileArrayAudioBuffer: observer.closed: \"+observer.closed);\r\n                if (observer.closed) {\r\n                  subscr.unsubscribe()\r\n                }\r\n                observer.next(aab)\r\n              },\r\n              complete: () => {\r\n                observer.complete();\r\n              },\r\n              error: (err) => {\r\n                if (err instanceof HttpErrorResponse && err.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                  observer.error(err);\r\n                }\r\n              }\r\n            });\r\n          }else{\r\n            observer.error(new Error('Unknown sample rate of recording file ID: '+recordingFile.recordingFileId));\r\n          }\r\n        }\r\n      }else{\r\n        observer.error(new Error('Could not get session ID of recording file'));\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchSprRecordingFileIndDbAudioBuffer(aCtx: AudioContext,persistentAudioStorageTarget:PersistentAudioStorageTarget, projectName: string, recordingFile:SprRecordingFile):Observable<IndexedDbAudioBuffer|null> {\r\n    let wobs = new Observable<IndexedDbAudioBuffer|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let baseUrl=this.sprAudioFileUrl(projectName,recordingFile);\r\n        if(baseUrl) {\r\n          if(recordingFile.samplerate) {\r\n            let lengthInSeconds= RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;\r\n            let obs = this.chunkedInddbAudioRequest(aCtx, persistentAudioStorageTarget,baseUrl,recordingFile.samplerate,lengthInSeconds);\r\n            let subscr = obs.subscribe({\r\n              next: aab => {\r\n                //console.debug(\"fetchSprRecordingFileIndDbAudioBuffer: observer.closed: \"+observer.closed);\r\n                if (observer.closed) {\r\n                  subscr.unsubscribe()\r\n                }\r\n                observer.next(aab)\r\n              },\r\n              complete: () => {\r\n                observer.complete();\r\n              },\r\n              error: (err) => {\r\n                if (err instanceof HttpErrorResponse && err.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 errors are real errors\r\n                  observer.error(err);\r\n                }\r\n              }\r\n            });\r\n          }else{\r\n            observer.error(new Error('Unknown sample rate of recording file ID: '+recordingFile.recordingFileId));\r\n          }\r\n        }\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchRecordingFileIndDbAudioBuffer(aCtx: AudioContext,persistentAudioStorageTarget:PersistentAudioStorageTarget, projectName: string, recordingFile:RecordingFile):Observable<IndexedDbAudioBuffer|null> {\r\n    let wobs = new Observable<IndexedDbAudioBuffer|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let baseUrl=this.audioFileUrl(projectName,recordingFile);\r\n        if(baseUrl) {\r\n          if(recordingFile.samplerate) {\r\n            let lengthInSeconds= RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;\r\n            let obs = this.chunkedInddbAudioRequest(aCtx, persistentAudioStorageTarget,baseUrl,recordingFile.samplerate,lengthInSeconds);\r\n            let subscr = obs.subscribe({\r\n              next: aab => {\r\n                //console.debug(\"fetchSprRecordingFileIndDbAudioBuffer: observer.closed: \"+observer.closed);\r\n                if (observer.closed) {\r\n                  subscr.unsubscribe()\r\n                }\r\n                observer.next(aab)\r\n              },\r\n              complete: () => {\r\n                observer.complete();\r\n              },\r\n              error: (err) => {\r\n                if (err instanceof HttpErrorResponse && err.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 errors are real errors\r\n                  observer.error(err);\r\n                }\r\n              }\r\n            });\r\n          }else{\r\n            observer.error(new Error('Unknown sample rate of recording file ID: '+recordingFile.recordingFileId));\r\n          }\r\n        }\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n\r\n  fetchSprRecordingFileNetAudioBuffer(aCtx: AudioContext,projectName: string, recordingFile:SprRecordingFile):Observable<NetAudioBuffer|null> {\r\n    let wobs = new Observable<NetAudioBuffer|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let baseUrl=this.sprAudioFileUrl(projectName,recordingFile);\r\n        if(baseUrl) {\r\n          let seconds= RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;\r\n          if(recordingFile.samplerate) {\r\n            let obs = this.chunkAudioRequestToNetAudioBuffer(aCtx, baseUrl, 0, recordingFile.samplerate,seconds, recordingFile.frames);\r\n            let subscr = obs.subscribe({\r\n              next: aab => {\r\n                //console.debug(\"fetchSprRecordingFileIndDbAudioBuffer: observer.closed: \"+observer.closed);\r\n                if (observer.closed) {\r\n                  subscr.unsubscribe();\r\n                }else {\r\n                  observer.next(aab)\r\n                }\r\n              },\r\n              complete: () => {\r\n                observer.complete();\r\n              },\r\n              error: (err) => {\r\n                if (err instanceof HttpErrorResponse && err.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 errors are real errors\r\n                  observer.error(err);\r\n                }\r\n              }\r\n            });\r\n          }else{\r\n            observer.error(new Error('Unknown sample rate of recording file ID: '+recordingFile.recordingFileId));\r\n          }\r\n        }else{\r\n          observer.error();\r\n        }\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchRecordingFileNetAudioBuffer(aCtx: AudioContext,projectName: string, recordingFile:RecordingFile):Observable<NetAudioBuffer|null> {\r\n    let wobs = new Observable<NetAudioBuffer|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let baseUrl=this.audioFileUrl(projectName,recordingFile);\r\n        if(baseUrl) {\r\n          let seconds= RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;\r\n          if(recordingFile.samplerate) {\r\n            let obs = this.chunkAudioRequestToNetAudioBuffer(aCtx, baseUrl, 0, recordingFile.samplerate,seconds, recordingFile.frames);\r\n            let subscr = obs.subscribe({\r\n              next: aab => {\r\n                //console.debug(\"fetchSprRecordingFileIndDbAudioBuffer: observer.closed: \"+observer.closed);\r\n                if (observer.closed) {\r\n                  subscr.unsubscribe();\r\n                }else {\r\n                  observer.next(aab)\r\n                }\r\n              },\r\n              complete: () => {\r\n                observer.complete();\r\n              },\r\n              error: (err) => {\r\n                if (err instanceof HttpErrorResponse && err.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 errors are real errors\r\n                  observer.error(err);\r\n                }\r\n              }\r\n            });\r\n          }else{\r\n            observer.error(new Error('Unknown sample rate of recording file ID: '+recordingFile.recordingFileId));\r\n          }\r\n        }else{\r\n          observer.error();\r\n        }\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n\r\n  fetchAndApplySprRecordingFile(aCtx: AudioContext, projectName: string, recordingFile:SprRecordingFile):Observable<SprRecordingFile|null> {\r\n\r\n    let wobs = new Observable<SprRecordingFile|null>(observer=>{\r\n      if(recordingFile.session) {\r\n        let obs = this.fetchSprAudiofile(projectName, recordingFile.session, recordingFile.itemCode, recordingFile.version);\r\n        obs.subscribe(resp => {\r\n              //console.log(\"Fetched audio file. HTTP response status: \"+resp.status+\", type: \"+resp.type+\", byte length: \"+ resp.body.byteLength);\r\n\r\n              // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n              if (resp.body) {\r\n                aCtx.decodeAudioData(resp.body, ab => {\r\n                  let abs=new AudioBufferSource(ab);\r\n                  RecordingFileUtils.setAudioData(recordingFile,new AudioDataHolder(abs));\r\n                  if (this.debugDelay > 0) {\r\n                    window.setTimeout(() => {\r\n\r\n                      observer.next(recordingFile);\r\n                      observer.complete();\r\n                    }, this.debugDelay);\r\n                  } else {\r\n                    observer.next(recordingFile);\r\n                    observer.complete();\r\n                  }\r\n                }, error => {\r\n                  observer.error(error);\r\n                  observer.complete();\r\n                })\r\n              } else {\r\n                observer.error('Fetching audio file: response has no body');\r\n              }\r\n            },\r\n            (err) => {\r\n              if (err instanceof HttpErrorResponse && err.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                observer.error(err);\r\n              }\r\n            });\r\n      }else{\r\n        observer.error();\r\n      }\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n\r\n  fetchRecordingFile(aCtx: AudioContext, projectName: string, sessId: string | number, itemcode: string,version:number):Observable<SprRecordingFile|null> {\r\n\r\n    let wobs = new Observable<SprRecordingFile | null>(observer=>{\r\n      let obs = this.fetchSprAudiofile(projectName, sessId, itemcode,version);\r\n\r\n\r\n      obs.subscribe(resp => {\r\n            // Do not use Promise version, which does not work with Safari 13\r\n          if(resp.body) {\r\n            aCtx.decodeAudioData(resp.body, ab => {\r\n              let abs=new AudioBufferSource(ab);\r\n              let adh=new AudioDataHolder(abs);\r\n              let rf = new SprRecordingFile(sessId, itemcode, version, adh);\r\n              if (this.debugDelay > 0) {\r\n                window.setTimeout(() => {\r\n\r\n                  observer.next(rf);\r\n                  observer.complete();\r\n                }, this.debugDelay);\r\n              } else {\r\n                observer.next(rf);\r\n                observer.complete();\r\n              }\r\n            });\r\n          }else{\r\n            observer.error();\r\n          }\r\n        },\r\n        (err) => {\r\n          if (err instanceof HttpErrorResponse && err.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 errors are real errors\r\n            observer.error(err);\r\n          }\r\n        });\r\n    });\r\n\r\n    return wobs;\r\n  }\r\n}\r\n\r\n\r\n\r\n"]}
|