speechrecorderng 3.4.4 → 3.6.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/README.md +1 -1
- package/esm2020/lib/action/action.mjs +3 -3
- package/esm2020/lib/audio/audio_display.mjs +7 -9
- package/esm2020/lib/audio/audio_player.mjs +13 -26
- package/esm2020/lib/audio/capture/capture.mjs +244 -207
- package/esm2020/lib/audio/context.mjs +64 -2
- package/esm2020/lib/audio/io/stream.mjs +1 -1
- package/esm2020/lib/audio/net_audio_buffer.mjs +5 -9
- package/esm2020/lib/audio/playback/array_audio_buffer_source_node.mjs +2 -2
- package/esm2020/lib/audio/playback/audio_source_worklet_module_loader.mjs +2 -2
- package/esm2020/lib/audio/playback/inddb_audio_buffer_source_node.mjs +1 -1
- package/esm2020/lib/audio/playback/player.mjs +137 -96
- package/esm2020/lib/audio/ui/audio_display_control.mjs +1 -1
- package/esm2020/lib/audio/ui/container.mjs +3 -3
- package/esm2020/lib/db/inddb.mjs +1 -1
- package/esm2020/lib/net/uploader.mjs +31 -28
- package/esm2020/lib/speechrecorder/project/project.mjs +1 -1
- package/esm2020/lib/speechrecorder/project/project.service.mjs +4 -4
- package/esm2020/lib/speechrecorder/recordings/basic_recording.service.mjs +59 -56
- package/esm2020/lib/speechrecorder/recordings/recordings.service.mjs +151 -142
- package/esm2020/lib/speechrecorder/script/script.service.mjs +1 -1
- package/esm2020/lib/speechrecorder/session/audiorecorder.mjs +49 -99
- package/esm2020/lib/speechrecorder/session/basicrecorder.mjs +9 -2
- package/esm2020/lib/speechrecorder/session/controlpanel.mjs +13 -11
- package/esm2020/lib/speechrecorder/session/progress.mjs +1 -1
- package/esm2020/lib/speechrecorder/session/prompting.mjs +1 -1
- package/esm2020/lib/speechrecorder/session/recorder_combi_pane.mjs +1 -1
- package/esm2020/lib/speechrecorder/session/recording_list.mjs +1 -1
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +3 -3
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs +3 -3
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +17 -16
- package/esm2020/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +13 -14
- package/esm2020/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +145 -135
- package/esm2020/lib/speechrecorder/session/session.service.mjs +12 -9
- package/esm2020/lib/speechrecorder/session/sessionmanager.mjs +66 -70
- package/esm2020/lib/speechrecorder/startstopsignal/startstopsignal.mjs +1 -2
- package/esm2020/lib/speechrecorderng.component.mjs +10 -16
- package/esm2020/lib/speechrecorderng.module.mjs +1 -1
- package/esm2020/lib/spr.module.version.mjs +2 -2
- package/esm2020/lib/ui/recordingitem_display.mjs +2 -2
- package/fesm2015/speechrecorderng.mjs +1077 -976
- package/fesm2015/speechrecorderng.mjs.map +1 -1
- package/fesm2020/speechrecorderng.mjs +1074 -976
- package/fesm2020/speechrecorderng.mjs.map +1 -1
- package/lib/action/action.d.ts +1 -1
- package/lib/audio/audio_display.d.ts +1 -3
- package/lib/audio/audio_player.d.ts +0 -1
- package/lib/audio/capture/capture.d.ts +5 -4
- package/lib/audio/context.d.ts +2 -0
- package/lib/audio/io/stream.d.ts +1 -1
- package/lib/audio/net_audio_buffer.d.ts +2 -4
- package/lib/audio/playback/player.d.ts +3 -2
- package/lib/net/uploader.d.ts +6 -6
- package/lib/speechrecorder/project/project.d.ts +1 -0
- package/lib/speechrecorder/project/project.service.d.ts +4 -4
- package/lib/speechrecorder/recordings/basic_recording.service.d.ts +2 -2
- package/lib/speechrecorder/recordings/recordings.service.d.ts +8 -8
- package/lib/speechrecorder/script/script.service.d.ts +2 -2
- package/lib/speechrecorder/session/audiorecorder.d.ts +1 -2
- package/lib/speechrecorder/session/basicrecorder.d.ts +4 -1
- package/lib/speechrecorder/session/controlpanel.d.ts +1 -1
- package/lib/speechrecorder/session/recordingfile/recording-file-meta.component.d.ts +1 -0
- package/lib/speechrecorder/session/recordingfile/recordingfile-service.d.ts +2 -2
- package/lib/speechrecorder/session/session.service.d.ts +2 -2
- package/lib/speechrecorder/session/sessionmanager.d.ts +1 -0
- package/lib/spr.module.version.d.ts +1 -1
- package/package.json +1 -1
|
@@ -6,6 +6,7 @@ import { Observable } from "rxjs";
|
|
|
6
6
|
import { NetAudioBuffer } from "../../audio/net_audio_buffer";
|
|
7
7
|
import { UUID } from "../../utils/utils";
|
|
8
8
|
import { WavReader } from "../../audio/impl/wavreader";
|
|
9
|
+
import { AudioContextProvider } from "../../audio/context";
|
|
9
10
|
export class ChunkDownload {
|
|
10
11
|
constructor(_orgPCMAudioFormat, _orgFrameLength, _decodedAudioBuffer) {
|
|
11
12
|
this._orgPCMAudioFormat = _orgPCMAudioFormat;
|
|
@@ -58,7 +59,7 @@ let BasicRecordingService = class BasicRecordingService {
|
|
|
58
59
|
withCredentials: this.withCredentials
|
|
59
60
|
});
|
|
60
61
|
}
|
|
61
|
-
chunkAudioRequest(
|
|
62
|
+
chunkAudioRequest(baseAudioUrl, startFrame = 0, frameLength) {
|
|
62
63
|
let ausps = new URLSearchParams();
|
|
63
64
|
ausps.set('startFrame', startFrame.toString());
|
|
64
65
|
ausps.set('frameLength', frameLength.toString());
|
|
@@ -69,67 +70,69 @@ let BasicRecordingService = class BasicRecordingService {
|
|
|
69
70
|
ausps.set('requestUUID', UUID.generate());
|
|
70
71
|
}
|
|
71
72
|
let obs = new Observable(observer => {
|
|
72
|
-
this.audioRequestByURL(baseAudioUrl, ausps).subscribe(
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
73
|
+
this.audioRequestByURL(baseAudioUrl, ausps).subscribe({
|
|
74
|
+
next: resp => {
|
|
75
|
+
// Do not use Promise version, which does not work with Safari 13 (13.0.5)
|
|
76
|
+
if (resp.body) {
|
|
77
|
+
//console.debug("chunkAudioRequest: observer.closed: "+observer.closed);
|
|
78
|
+
//console.debug("Audio file bytes: "+resp.body.byteLength);
|
|
79
|
+
// Check original audio format
|
|
80
|
+
let wr = new WavReader(resp.body);
|
|
81
|
+
const pcmFmt = wr.readFormat();
|
|
82
|
+
const orgFl = wr.frameLength();
|
|
83
|
+
// if(pcmFmt){
|
|
84
|
+
// console.debug("Original WAVE format of download chunk: "+pcmFmt);
|
|
85
|
+
// }else{
|
|
86
|
+
// console.error("Could not read WAVE format of original download chunk!");
|
|
87
|
+
// }
|
|
88
|
+
// if(orgFl){
|
|
89
|
+
// console.debug("Original frame length of download chunk: "+orgFl);
|
|
90
|
+
// }else{
|
|
91
|
+
// console.error("Could not read WAVE format of original download chunk!");
|
|
92
|
+
// }
|
|
93
|
+
if (pcmFmt && orgFl) {
|
|
94
|
+
AudioContextProvider.decodeAudioData(resp.body).then((ab) => {
|
|
95
|
+
//console.debug("Decoded audio chunk frames: "+ab.length);
|
|
96
|
+
let chDl = new ChunkDownload(pcmFmt, orgFl, ab);
|
|
97
|
+
observer.next(chDl);
|
|
98
|
+
observer.complete();
|
|
99
|
+
}).catch(error => {
|
|
100
|
+
//if(error instanceof HttpErrorResponse) {
|
|
101
|
+
// if (error.status == 404) {
|
|
102
|
+
// // Interpret not as an error, the file ist not recorded yet
|
|
103
|
+
// observer.next(null);
|
|
104
|
+
// observer.complete()
|
|
105
|
+
// } else {
|
|
106
|
+
// // all other states are errors
|
|
107
|
+
console.error("Recordings service chunkAudioRequest error decoding audio data: " + error.name + ": " + error.message);
|
|
108
|
+
observer.error(error);
|
|
109
|
+
// }
|
|
110
|
+
// }
|
|
111
|
+
});
|
|
112
|
+
}
|
|
113
|
+
else {
|
|
114
|
+
const errMsg = 'Could not parse audio header for format and/or frame length of download.';
|
|
115
|
+
console.error(errMsg);
|
|
116
|
+
observer.error(errMsg);
|
|
117
|
+
}
|
|
110
118
|
}
|
|
111
119
|
else {
|
|
112
|
-
const errMsg = '
|
|
120
|
+
const errMsg = 'Fetching audio file: response has no body';
|
|
113
121
|
console.error(errMsg);
|
|
114
122
|
observer.error(errMsg);
|
|
115
123
|
}
|
|
124
|
+
}, error: (error) => {
|
|
125
|
+
// all other states are errors
|
|
126
|
+
//const errMsg='Fetching audio file HTTP error: '+error;
|
|
127
|
+
//console.error(errMsg);
|
|
128
|
+
observer.error(error);
|
|
129
|
+
//observer.complete();
|
|
116
130
|
}
|
|
117
|
-
else {
|
|
118
|
-
const errMsg = 'Fetching audio file: response has no body';
|
|
119
|
-
console.error(errMsg);
|
|
120
|
-
observer.error(errMsg);
|
|
121
|
-
}
|
|
122
|
-
}, (error) => {
|
|
123
|
-
// all other states are errors
|
|
124
|
-
//const errMsg='Fetching audio file HTTP error: '+error;
|
|
125
|
-
//console.error(errMsg);
|
|
126
|
-
observer.error(error);
|
|
127
|
-
//observer.complete();
|
|
128
131
|
});
|
|
129
132
|
});
|
|
130
133
|
return obs;
|
|
131
134
|
}
|
|
132
|
-
chunkAudioRequestToNetAudioBuffer(
|
|
135
|
+
chunkAudioRequestToNetAudioBuffer(baseAudioUrl, startFrame = 0, orgSampleRate, seconds, frames) {
|
|
133
136
|
//let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;
|
|
134
137
|
//let audioUrl=new URL(baseAudioUrl);
|
|
135
138
|
// if(orgSampleRate!=null && frameLength%orgSampleRate>0){
|
|
@@ -153,7 +156,7 @@ let BasicRecordingService = class BasicRecordingService {
|
|
|
153
156
|
if (resp.body) {
|
|
154
157
|
//console.debug("chunkAudioRequestTonetAb: subscriber.closed: "+subscriber.closed);
|
|
155
158
|
//console.debug("chunkAudioRequestTonetAb: Audio file bytes: "+resp.body.byteLength);
|
|
156
|
-
|
|
159
|
+
AudioContextProvider.decodeAudioData(resp.body).then(ab => {
|
|
157
160
|
//console.debug("chunkAudioRequestTonetAb: Decoded audio chunk frames for netAb: "+ab.length);
|
|
158
161
|
//console.debug("chunkAudioRequestTonetAb: Create netAb ab from chunk ab...");
|
|
159
162
|
if (frames === null) {
|
|
@@ -167,7 +170,7 @@ let BasicRecordingService = class BasicRecordingService {
|
|
|
167
170
|
//console.debug("Platform sr: "+ab.sampleRate+", file sr: "+orgSampleRate+", decoded/org frame length: "+fl+"/"+frames+", ab.length: "+ab.length);
|
|
168
171
|
}
|
|
169
172
|
}
|
|
170
|
-
let nab = NetAudioBuffer.fromChunkAudioBuffer(
|
|
173
|
+
let nab = NetAudioBuffer.fromChunkAudioBuffer(this, baseAudioUrl, ab, fl, frameLength);
|
|
171
174
|
//let rp=new ReadyProvider();
|
|
172
175
|
//nab.readyProvider=rp;
|
|
173
176
|
//rp.ready();
|
|
@@ -179,7 +182,7 @@ let BasicRecordingService = class BasicRecordingService {
|
|
|
179
182
|
subscriber.next(nab);
|
|
180
183
|
subscriber.complete();
|
|
181
184
|
}
|
|
182
|
-
}
|
|
185
|
+
}).catch(error => {
|
|
183
186
|
console.error('chunkAudioRequestToNetAb: error: ' + error);
|
|
184
187
|
//if(error instanceof HttpErrorResponse) {
|
|
185
188
|
subscriber.error(error);
|
|
@@ -210,4 +213,4 @@ BasicRecordingService = __decorate([
|
|
|
210
213
|
__param(1, Inject(SPEECHRECORDER_CONFIG))
|
|
211
214
|
], BasicRecordingService);
|
|
212
215
|
export { BasicRecordingService };
|
|
213
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"basic_recording.service.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/recordings/basic_recording.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {HttpClient, HttpHeaders, HttpResponse} from \"@angular/common/http\";\r\nimport {Inject} from \"@angular/core\";\r\nimport {ApiType, SPEECHRECORDER_CONFIG, SpeechRecorderConfig} from \"../../spr.config\";\r\nimport {Observable} from \"rxjs\";\r\nimport {NetAudioBuffer} from \"../../audio/net_audio_buffer\";\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {WavReader} from \"../../audio/impl/wavreader\";\r\nimport {PCMAudioFormat} from \"../../audio/format\";\r\n\r\nexport class ChunkDownload{\r\n  get orgPCMAudioFormat(): PCMAudioFormat {\r\n    return this._orgPCMAudioFormat;\r\n  }\r\n\r\n  get orgFrameLength(): number {\r\n    return this._orgFrameLength;\r\n  }\r\n\r\n  get decodedAudioBuffer(): AudioBuffer {\r\n    return this._decodedAudioBuffer;\r\n  }\r\n  constructor(private _orgPCMAudioFormat:PCMAudioFormat,private _orgFrameLength:number,private _decodedAudioBuffer:AudioBuffer){}\r\n}\r\n\r\nexport class BasicRecordingService{\r\n  // iPad 9th generation, iOS 15.7.1, sometimes:\r\n  // Failed to load resource: WebKit hat einen internen Fehler festgestellt\r\n  // Firefox on Windows crashes\r\n  public static readonly DEFAULT_CHUNKED_DOWNLOAD_SECONDS:number=10;\r\n  public static readonly DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES:number=2880000*5; // Default 5 minutes one channel at 48kHz\r\n  protected _maxAutoNetMemStoreSamples:number=BasicRecordingService.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;\r\n    protected apiEndPoint: string;\r\n    protected withCredentials: boolean = false;\r\n    constructor(protected http: HttpClient, @Inject(SPEECHRECORDER_CONFIG) protected config?: SpeechRecorderConfig) {\r\n        this.apiEndPoint = ''\r\n\r\n        if (config && config.apiEndPoint) {\r\n            this.apiEndPoint = config.apiEndPoint;\r\n        }\r\n        if (this.apiEndPoint !== '') {\r\n            this.apiEndPoint = this.apiEndPoint + '/'\r\n        }\r\n        if (config != null && config.withCredentials != null) {\r\n            this.withCredentials = config.withCredentials;\r\n        }\r\n    }\r\n\r\n    protected audioRequestByURL(audioBaseUrl:string,audioURLSearchParams:URLSearchParams): Observable<HttpResponse<ArrayBuffer>> {\r\n        let audioUrl=audioBaseUrl;\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            audioUrl=audioUrl+'.wav';\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            audioURLSearchParams.set('requestUUID',UUID.generate());\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n        }\r\n\r\n        audioUrl=audioUrl+'?'+audioURLSearchParams.toString();\r\n\r\n        let headers = new HttpHeaders();\r\n        headers = headers.set('Accept', 'audio/wav');\r\n        return this.http.get(audioUrl, {\r\n            headers: headers,\r\n            observe: 'response',\r\n            responseType: 'arraybuffer',\r\n            withCredentials: this.withCredentials\r\n        });\r\n\r\n    }\r\n\r\n    public chunkAudioRequest(aCtx:AudioContext,baseAudioUrl:string,startFrame:number=0,frameLength:number): Observable<ChunkDownload|null> {\r\n\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n\r\n        let obs=new Observable<ChunkDownload|null>(observer=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe(resp => {\r\n                    // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                    if (resp.body) {\r\n                        //console.debug(\"chunkAudioRequest: observer.closed: \"+observer.closed);\r\n                        //console.debug(\"Audio file bytes: \"+resp.body.byteLength);\r\n\r\n                        // Check original audio format\r\n                        let wr=new WavReader(resp.body);\r\n                        const pcmFmt=wr.readFormat();\r\n                        const orgFl=wr.frameLength();\r\n                        // if(pcmFmt){\r\n                        //   console.debug(\"Original WAVE format of download chunk: \"+pcmFmt);\r\n                        // }else{\r\n                        //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                        // }\r\n                        // if(orgFl){\r\n                        //   console.debug(\"Original frame length of download chunk: \"+orgFl);\r\n                        // }else{\r\n                        //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                        // }\r\n\r\n                        if(pcmFmt && orgFl) {\r\n                            aCtx.decodeAudioData(resp.body, ab => {\r\n                                    //console.debug(\"Decoded audio chunk frames: \"+ab.length);\r\n                                    let chDl = new ChunkDownload(pcmFmt, orgFl, ab);\r\n                                    observer.next(chDl);\r\n                                    observer.complete();\r\n                                }\r\n                                , error => {\r\n                                    //if(error instanceof HttpErrorResponse) {\r\n                                    // if (error.status == 404) {\r\n                                    //   // Interpret not as an error, the file ist not recorded yet\r\n                                    //   observer.next(null);\r\n                                    //   observer.complete()\r\n                                    // } else {\r\n                                    //   // all other states are errors\r\n                                    console.error(\"Recordings service chunkAudioRequest error decoding audio data: \"+error.name+\": \"+error.message);\r\n                                    observer.error(error);\r\n                                    // }\r\n                                    // }\r\n                                });\r\n                        }else{\r\n                            const errMsg='Could not parse audio header for format and/or frame length of download.';\r\n                            console.error(errMsg);\r\n                            observer.error(errMsg);\r\n                        }\r\n                    } else {\r\n                        const errMsg='Fetching audio file: response has no body';\r\n                        console.error(errMsg);\r\n                        observer.error(errMsg);\r\n                    }\r\n                },\r\n                (error) => {\r\n                    // all other states are errors\r\n                    //const errMsg='Fetching audio file HTTP error: '+error;\r\n                    //console.error(errMsg);\r\n                    observer.error(error);\r\n                    //observer.complete();\r\n\r\n                });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n\r\n    protected chunkAudioRequestToNetAudioBuffer(aCtx: AudioContext, baseAudioUrl: string, startFrame: number = 0, orgSampleRate: number, seconds:number,frames: number | null): Observable<NetAudioBuffer | null> {\r\n        //let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;\r\n        //let audioUrl=new URL(baseAudioUrl);\r\n        // if(orgSampleRate!=null && frameLength%orgSampleRate>0){\r\n        //   const errMsg='frameLength must be equal or multiple of original samplerate.';\r\n        //   console.error(errMsg+' ('+frameLength+'%'+orgSampleRate+'=='+(frameLength%orgSampleRate)+')');\r\n        //   throw Error(errMsg)\r\n        // }\r\n        let frameLength:number=orgSampleRate*Math.round(seconds);\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n        let obs=new Observable<NetAudioBuffer|null>(subscriber=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe({next:(resp) => {\r\n                    // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                    if (resp.body) {\r\n                        //console.debug(\"chunkAudioRequestTonetAb: subscriber.closed: \"+subscriber.closed);\r\n                        //console.debug(\"chunkAudioRequestTonetAb: Audio file bytes: \"+resp.body.byteLength);\r\n                        aCtx.decodeAudioData(resp.body, ab => {\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Decoded audio chunk frames for netAb: \"+ab.length);\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Create netAb ab from chunk ab...\");\r\n                                if(frames===null){\r\n                                    subscriber.error(new Error('Could not get frame length from recording file object'));\r\n                                }else {\r\n                                    let fl = frames;\r\n                                    if (ab.sampleRate !== orgSampleRate) {\r\n                                        if (orgSampleRate && frames) {\r\n                                            fl = Math.round(ab.sampleRate * frames / orgSampleRate);\r\n                                            //console.debug(\"Platform sr: \"+ab.sampleRate+\", file sr: \"+orgSampleRate+\", decoded/org frame length: \"+fl+\"/\"+frames+\", ab.length: \"+ab.length);\r\n                                        }\r\n                                    }\r\n\r\n                                    let nab = NetAudioBuffer.fromChunkAudioBuffer(aCtx, this,baseAudioUrl, ab, fl,frameLength);\r\n                                    //let rp=new ReadyProvider();\r\n                                    //nab.readyProvider=rp;\r\n                                    //rp.ready();\r\n                                    nab.ready();\r\n                                    if (nab.frameLen < frameLength) {\r\n                                        //console.debug(\"chunkAudioRequestTonetAb: Built netAb ab from chunk ab: First chunk shorter tha frameLength (\"+netAbAudioBuffer.frameLen+\"<\"+frameLength+\"), assuming end of data, sealing netAb ab.\");\r\n                                        nab.seal();\r\n                                    }\r\n                                    subscriber.next(nab);\r\n\r\n                                    subscriber.complete();\r\n                                }\r\n                            }\r\n                            , error => {\r\n                                console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                                //if(error instanceof HttpErrorResponse) {\r\n                                subscriber.error(error);\r\n                                //}\r\n                            })\r\n                    } else {\r\n                        console.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                        subscriber.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                    }\r\n                },\r\n                error:(error) => {\r\n                    console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                    subscriber.error(error);\r\n                    //subscriber.complete();\r\n                }\r\n            });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n}\r\n"]}
|
|
216
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"basic_recording.service.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/recordings/basic_recording.service.ts"],"names":[],"mappings":"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","sourcesContent":["import {HttpClient, HttpHeaders, HttpResponse} from \"@angular/common/http\";\r\nimport {Inject} from \"@angular/core\";\r\nimport {ApiType, SPEECHRECORDER_CONFIG, SpeechRecorderConfig} from \"../../spr.config\";\r\nimport {Observable} from \"rxjs\";\r\nimport {NetAudioBuffer} from \"../../audio/net_audio_buffer\";\r\nimport {UUID} from \"../../utils/utils\";\r\nimport {WavReader} from \"../../audio/impl/wavreader\";\r\nimport {PCMAudioFormat} from \"../../audio/format\";\r\nimport {AudioContextProvider} from \"../../audio/context\";\r\n\r\nexport class ChunkDownload{\r\n  get orgPCMAudioFormat(): PCMAudioFormat {\r\n    return this._orgPCMAudioFormat;\r\n  }\r\n\r\n  get orgFrameLength(): number {\r\n    return this._orgFrameLength;\r\n  }\r\n\r\n  get decodedAudioBuffer(): AudioBuffer {\r\n    return this._decodedAudioBuffer;\r\n  }\r\n  constructor(private _orgPCMAudioFormat:PCMAudioFormat,private _orgFrameLength:number,private _decodedAudioBuffer:AudioBuffer){}\r\n}\r\n\r\nexport class BasicRecordingService{\r\n  // iPad 9th generation, iOS 15.7.1, sometimes:\r\n  // Failed to load resource: WebKit hat einen internen Fehler festgestellt\r\n  // Firefox on Windows crashes\r\n  public static readonly DEFAULT_CHUNKED_DOWNLOAD_SECONDS:number=10;\r\n  public static readonly DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES:number=2880000*5; // Default 5 minutes one channel at 48kHz\r\n  protected _maxAutoNetMemStoreSamples:number=BasicRecordingService.DEFAULT_MAX_NET_AUTO_MEM_STORE_SAMPLES;\r\n    protected apiEndPoint: string;\r\n    protected withCredentials: boolean = false;\r\n\r\n    constructor(protected http: HttpClient, @Inject(SPEECHRECORDER_CONFIG) protected config?: SpeechRecorderConfig) {\r\n        this.apiEndPoint = ''\r\n\r\n        if (config && config.apiEndPoint) {\r\n            this.apiEndPoint = config.apiEndPoint;\r\n        }\r\n        if (this.apiEndPoint !== '') {\r\n            this.apiEndPoint = this.apiEndPoint + '/'\r\n        }\r\n        if (config != null && config.withCredentials != null) {\r\n            this.withCredentials = config.withCredentials;\r\n        }\r\n    }\r\n\r\n    protected audioRequestByURL(audioBaseUrl:string,audioURLSearchParams:URLSearchParams): Observable<HttpResponse<ArrayBuffer>> {\r\n        let audioUrl=audioBaseUrl;\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            audioUrl=audioUrl+'.wav';\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            audioURLSearchParams.set('requestUUID',UUID.generate());\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n        }\r\n\r\n        audioUrl=audioUrl+'?'+audioURLSearchParams.toString();\r\n\r\n        let headers = new HttpHeaders();\r\n        headers = headers.set('Accept', 'audio/wav');\r\n        return this.http.get(audioUrl, {\r\n            headers: headers,\r\n            observe: 'response',\r\n            responseType: 'arraybuffer',\r\n            withCredentials: this.withCredentials\r\n        });\r\n\r\n    }\r\n\r\n    public chunkAudioRequest(baseAudioUrl:string,startFrame:number=0,frameLength:number): Observable<ChunkDownload|null> {\r\n\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n\r\n        let obs=new Observable<ChunkDownload|null>(observer=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe(\r\n                {\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                            //console.debug(\"chunkAudioRequest: observer.closed: \"+observer.closed);\r\n                            //console.debug(\"Audio file bytes: \"+resp.body.byteLength);\r\n\r\n                            // Check original audio format\r\n                            let wr = new WavReader(resp.body);\r\n                            const pcmFmt = wr.readFormat();\r\n                            const orgFl = wr.frameLength();\r\n                            // if(pcmFmt){\r\n                            //   console.debug(\"Original WAVE format of download chunk: \"+pcmFmt);\r\n                            // }else{\r\n                            //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                            // }\r\n                            // if(orgFl){\r\n                            //   console.debug(\"Original frame length of download chunk: \"+orgFl);\r\n                            // }else{\r\n                            //   console.error(\"Could not read WAVE format of original download chunk!\");\r\n                            // }\r\n\r\n                            if (pcmFmt && orgFl) {\r\n                              AudioContextProvider.decodeAudioData(resp.body).then((ab)=>{\r\n                                      //console.debug(\"Decoded audio chunk frames: \"+ab.length);\r\n                                      let chDl = new ChunkDownload(pcmFmt, orgFl, ab);\r\n                                      observer.next(chDl);\r\n                                      observer.complete();\r\n                                    }).catch(error => {\r\n                                      //if(error instanceof HttpErrorResponse) {\r\n                                      // if (error.status == 404) {\r\n                                      //   // Interpret not as an error, the file ist not recorded yet\r\n                                      //   observer.next(null);\r\n                                      //   observer.complete()\r\n                                      // } else {\r\n                                      //   // all other states are errors\r\n                                      console.error(\"Recordings service chunkAudioRequest error decoding audio data: \" + error.name + \": \" + error.message);\r\n                                      observer.error(error);\r\n                                      // }\r\n                                      // }\r\n                                    });\r\n                            } else {\r\n                                const errMsg = 'Could not parse audio header for format and/or frame length of download.';\r\n                                console.error(errMsg);\r\n                                observer.error(errMsg);\r\n                            }\r\n                        } else {\r\n                            const errMsg = 'Fetching audio file: response has no body';\r\n                            console.error(errMsg);\r\n                            observer.error(errMsg);\r\n                        }\r\n                    }, error:\r\n                        (error) => {\r\n                            // all other states are errors\r\n                            //const errMsg='Fetching audio file HTTP error: '+error;\r\n                            //console.error(errMsg);\r\n                            observer.error(error);\r\n                            //observer.complete();\r\n\r\n                        }\r\n                });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n\r\n    protected chunkAudioRequestToNetAudioBuffer(baseAudioUrl: string, startFrame: number = 0, orgSampleRate: number, seconds:number,frames: number | null): Observable<NetAudioBuffer | null> {\r\n        //let audioUrl=baseAudioUrl+'?startFrame='+startFrame+'&frameLength='+frameLength;\r\n        //let audioUrl=new URL(baseAudioUrl);\r\n        // if(orgSampleRate!=null && frameLength%orgSampleRate>0){\r\n        //   const errMsg='frameLength must be equal or multiple of original samplerate.';\r\n        //   console.error(errMsg+' ('+frameLength+'%'+orgSampleRate+'=='+(frameLength%orgSampleRate)+')');\r\n        //   throw Error(errMsg)\r\n        // }\r\n        let frameLength:number=orgSampleRate*Math.round(seconds);\r\n        let ausps=new URLSearchParams();\r\n        ausps.set('startFrame',startFrame.toString());\r\n        ausps.set('frameLength',frameLength.toString());\r\n        if (this.config && this.config.apiType === ApiType.FILES) {\r\n            // for development and demo\r\n            // append UUID to make request URL unique to avoid localhost server caching\r\n            //audioUrl = audioUrl + '.wav?requestUUID=' + UUID.generate();\r\n            ausps.set('requestUUID',UUID.generate());\r\n        }\r\n        let obs=new Observable<NetAudioBuffer|null>(subscriber=> {\r\n            this.audioRequestByURL(baseAudioUrl,ausps).subscribe({next:(resp) => {\r\n                    // Do not use Promise version, which does not work with Safari 13 (13.0.5)\r\n                    if (resp.body) {\r\n                        //console.debug(\"chunkAudioRequestTonetAb: subscriber.closed: \"+subscriber.closed);\r\n                        //console.debug(\"chunkAudioRequestTonetAb: Audio file bytes: \"+resp.body.byteLength);\r\n                        AudioContextProvider.decodeAudioData(resp.body).then(ab => {\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Decoded audio chunk frames for netAb: \"+ab.length);\r\n                                //console.debug(\"chunkAudioRequestTonetAb: Create netAb ab from chunk ab...\");\r\n                                if(frames===null){\r\n                                    subscriber.error(new Error('Could not get frame length from recording file object'));\r\n                                }else {\r\n                                    let fl = frames;\r\n                                    if (ab.sampleRate !== orgSampleRate) {\r\n                                        if (orgSampleRate && frames) {\r\n                                            fl = Math.round(ab.sampleRate * frames / orgSampleRate);\r\n                                            //console.debug(\"Platform sr: \"+ab.sampleRate+\", file sr: \"+orgSampleRate+\", decoded/org frame length: \"+fl+\"/\"+frames+\", ab.length: \"+ab.length);\r\n                                        }\r\n                                    }\r\n\r\n                                    let nab = NetAudioBuffer.fromChunkAudioBuffer(this,baseAudioUrl, ab, fl,frameLength);\r\n                                    //let rp=new ReadyProvider();\r\n                                    //nab.readyProvider=rp;\r\n                                    //rp.ready();\r\n                                    nab.ready();\r\n                                    if (nab.frameLen < frameLength) {\r\n                                        //console.debug(\"chunkAudioRequestTonetAb: Built netAb ab from chunk ab: First chunk shorter tha frameLength (\"+netAbAudioBuffer.frameLen+\"<\"+frameLength+\"), assuming end of data, sealing netAb ab.\");\r\n                                        nab.seal();\r\n                                    }\r\n                                    subscriber.next(nab);\r\n\r\n                                    subscriber.complete();\r\n                                }\r\n                            }).catch(error => {\r\n                                console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                                //if(error instanceof HttpErrorResponse) {\r\n                                subscriber.error(error);\r\n                                //}\r\n                            })\r\n                    } else {\r\n                        console.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                        subscriber.error('chunkAudioRequestToNetAb: Fetching audio file: response has no body');\r\n                    }\r\n                },\r\n                error:(error) => {\r\n                    console.error('chunkAudioRequestToNetAb: error: '+error);\r\n                    subscriber.error(error);\r\n                    //subscriber.complete();\r\n                }\r\n            });\r\n        });\r\n        return obs;\r\n    }\r\n\r\n}\r\n"]}
|