speechrecorderng 3.10.13 → 3.11.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/fesm2022/speechrecorderng.mjs +204 -201
- package/fesm2022/speechrecorderng.mjs.map +1 -1
- package/lib/audio/ui/audio_canvas_layer_comp.d.ts +2 -2
- package/lib/spr.module.version.d.ts +1 -1
- package/lib/ui/canvas_layer_comp.d.ts +1 -1
- package/lib/ui/responsive_component.d.ts +1 -1
- package/package.json +13 -15
- package/esm2022/lib/action/action.mjs +0 -73
- package/esm2022/lib/audio/array_audio_buffer.mjs +0 -164
- package/esm2022/lib/audio/array_audio_buffer_input_stream.mjs +0 -86
- package/esm2022/lib/audio/array_audio_buffer_random_access_stream.mjs +0 -16
- package/esm2022/lib/audio/audio_data_holder.mjs +0 -264
- package/esm2022/lib/audio/audio_display.mjs +0 -118
- package/esm2022/lib/audio/audio_player.mjs +0 -238
- package/esm2022/lib/audio/capture/capture.mjs +0 -855
- package/esm2022/lib/audio/context.mjs +0 -79
- package/esm2022/lib/audio/dsp/level_measure.mjs +0 -517
- package/esm2022/lib/audio/format.mjs +0 -20
- package/esm2022/lib/audio/impl/wavformat.mjs +0 -7
- package/esm2022/lib/audio/impl/wavreader.mjs +0 -144
- package/esm2022/lib/audio/impl/wavwriter.mjs +0 -191
- package/esm2022/lib/audio/inddb_audio_buffer.mjs +0 -508
- package/esm2022/lib/audio/io/stream.mjs +0 -59
- package/esm2022/lib/audio/net_audio_buffer.mjs +0 -293
- package/esm2022/lib/audio/persistor.mjs +0 -81
- package/esm2022/lib/audio/playback/array_audio_buffer_source_node.mjs +0 -126
- package/esm2022/lib/audio/playback/audio_source_node.mjs +0 -18
- package/esm2022/lib/audio/playback/audio_source_worklet_module_loader.mjs +0 -167
- package/esm2022/lib/audio/playback/inddb_audio_buffer_source_node.mjs +0 -167
- package/esm2022/lib/audio/playback/net_audio_buffer_source_node.mjs +0 -218
- package/esm2022/lib/audio/playback/player.mjs +0 -402
- package/esm2022/lib/audio/ui/audio_canvas_layer_comp.mjs +0 -347
- package/esm2022/lib/audio/ui/audio_display_control.mjs +0 -150
- package/esm2022/lib/audio/ui/audio_display_scroll_pane.mjs +0 -146
- package/esm2022/lib/audio/ui/audiosignal.mjs +0 -533
- package/esm2022/lib/audio/ui/common.mjs +0 -19
- package/esm2022/lib/audio/ui/container.mjs +0 -414
- package/esm2022/lib/audio/ui/livelevel.mjs +0 -361
- package/esm2022/lib/audio/ui/scroll_pane_horizontal.mjs +0 -11
- package/esm2022/lib/audio/ui/sonagram.mjs +0 -900
- package/esm2022/lib/db/inddb.mjs +0 -120
- package/esm2022/lib/dsp/utils.mjs +0 -48
- package/esm2022/lib/environment/environment.defaults.mjs +0 -9
- package/esm2022/lib/io/BinaryReader.mjs +0 -93
- package/esm2022/lib/io/BinaryWriter.mjs +0 -80
- package/esm2022/lib/io/stream.mjs +0 -259
- package/esm2022/lib/math/2d/geometry.mjs +0 -28
- package/esm2022/lib/math/complex.mjs +0 -58
- package/esm2022/lib/math/dft.mjs +0 -196
- package/esm2022/lib/media/utils.mjs +0 -14
- package/esm2022/lib/net/uploader.mjs +0 -367
- package/esm2022/lib/recorder_component.mjs +0 -65
- package/esm2022/lib/speechrecorder/project/project.mjs +0 -54
- package/esm2022/lib/speechrecorder/project/project.service.mjs +0 -64
- package/esm2022/lib/speechrecorder/recording.mjs +0 -124
- package/esm2022/lib/speechrecorder/recordings/basic_recording.service.mjs +0 -221
- package/esm2022/lib/speechrecorder/recordings/recordings.service.mjs +0 -1014
- package/esm2022/lib/speechrecorder/script/script.mjs +0 -114
- package/esm2022/lib/speechrecorder/script/script.service.mjs +0 -47
- package/esm2022/lib/speechrecorder/session/audiorecorder.mjs +0 -1179
- package/esm2022/lib/speechrecorder/session/basicrecorder.mjs +0 -676
- package/esm2022/lib/speechrecorder/session/controlpanel.mjs +0 -416
- package/esm2022/lib/speechrecorder/session/item.mjs +0 -30
- package/esm2022/lib/speechrecorder/session/progress.mjs +0 -135
- package/esm2022/lib/speechrecorder/session/prompting.mjs +0 -639
- package/esm2022/lib/speechrecorder/session/recorder_combi_pane.mjs +0 -88
- package/esm2022/lib/speechrecorder/session/recording_file_cache.mjs +0 -195
- package/esm2022/lib/speechrecorder/session/recording_list.mjs +0 -188
- package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +0 -128
- package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs +0 -114
- package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +0 -146
- package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +0 -424
- package/esm2022/lib/speechrecorder/session/recordingfile/recording-file.mjs +0 -68
- package/esm2022/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +0 -288
- package/esm2022/lib/speechrecorder/session/session.mjs +0 -2
- package/esm2022/lib/speechrecorder/session/session.service.mjs +0 -69
- package/esm2022/lib/speechrecorder/session/session_finished_dialog.mjs +0 -46
- package/esm2022/lib/speechrecorder/session/sessionmanager.mjs +0 -1385
- package/esm2022/lib/speechrecorder/session/warning_bar.mjs +0 -33
- package/esm2022/lib/speechrecorder/spruploader.mjs +0 -22
- package/esm2022/lib/speechrecorder/startstopsignal/startstopsignal.mjs +0 -2
- package/esm2022/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.mjs +0 -57
- package/esm2022/lib/speechrecorderng.component.mjs +0 -392
- package/esm2022/lib/speechrecorderng.module.mjs +0 -97
- package/esm2022/lib/spr.config.mjs +0 -27
- package/esm2022/lib/spr.module.version.mjs +0 -2
- package/esm2022/lib/ui/canvas_layer_comp.mjs +0 -38
- package/esm2022/lib/ui/intersection-observer.directive.mjs +0 -32
- package/esm2022/lib/ui/message_dialog.mjs +0 -51
- package/esm2022/lib/ui/recordingitem_display.mjs +0 -253
- package/esm2022/lib/ui/responsive_component.mjs +0 -24
- package/esm2022/lib/utils/scrollIntoViewToBottom.mjs +0 -23
- package/esm2022/lib/utils/ua-parser.mjs +0 -190
- package/esm2022/lib/utils/utils.mjs +0 -132
- package/esm2022/lib/utils/wake_lock.mjs +0 -114
- package/esm2022/lib/utils/wake_lock_media.mjs +0 -2
- package/esm2022/public-api.mjs +0 -35
- package/esm2022/speechrecorderng.mjs +0 -5
|
@@ -1,1179 +0,0 @@
|
|
|
1
|
-
import { AudioCapture } from '../../audio/capture/capture';
|
|
2
|
-
import { AudioPlayer, EventType } from '../../audio/playback/player';
|
|
3
|
-
import { WavWriter } from '../../audio/impl/wavwriter';
|
|
4
|
-
import { RecordingFile, RecordingFileUtils } from '../recording';
|
|
5
|
-
import { Component, HostListener, Inject, Input, ViewChild } from "@angular/core";
|
|
6
|
-
import { SessionService } from "./session.service";
|
|
7
|
-
import { SPEECHRECORDER_CONFIG } from "../../spr.config";
|
|
8
|
-
import { AudioStorageFormatEncoding, AudioStorageType, ProjectUtil } from "../project/project";
|
|
9
|
-
import { MessageDialog } from "../../ui/message_dialog";
|
|
10
|
-
import { RecordingService } from "../recordings/recordings.service";
|
|
11
|
-
import { AudioClip } from "../../audio/persistor";
|
|
12
|
-
import { Upload, UploaderStatus, UploadHolder } from "../../net/uploader";
|
|
13
|
-
import { LevelBar, State as LiveLevelState } from "../../audio/ui/livelevel";
|
|
14
|
-
import { RecorderCombiPane } from "./recorder_combi_pane";
|
|
15
|
-
import { BasicRecorder, MAX_RECORDING_TIME_MS, RECFILE_API_CTX } from "./basicrecorder";
|
|
16
|
-
import { RecorderComponent } from "../../recorder_component";
|
|
17
|
-
import { AudioBufferSource, AudioDataHolder } from "../../audio/audio_data_holder";
|
|
18
|
-
import { NetAudioBuffer } from "../../audio/net_audio_buffer";
|
|
19
|
-
import * as i0 from "@angular/core";
|
|
20
|
-
import * as i1 from "@angular/cdk/layout";
|
|
21
|
-
import * as i2 from "@angular/material/dialog";
|
|
22
|
-
import * as i3 from "./session.service";
|
|
23
|
-
import * as i4 from "../recordings/recordings.service";
|
|
24
|
-
import * as i5 from "../spruploader";
|
|
25
|
-
import * as i6 from "@angular/common";
|
|
26
|
-
import * as i7 from "@angular/material/icon";
|
|
27
|
-
import * as i8 from "@angular/material/button";
|
|
28
|
-
import * as i9 from "../../audio/ui/livelevel";
|
|
29
|
-
import * as i10 from "./controlpanel";
|
|
30
|
-
import * as i11 from "../../ui/recordingitem_display";
|
|
31
|
-
import * as i12 from "./warning_bar";
|
|
32
|
-
import * as i13 from "./recorder_combi_pane";
|
|
33
|
-
import * as i14 from "../../spr.config";
|
|
34
|
-
import * as i15 from "@angular/router";
|
|
35
|
-
import * as i16 from "../project/project.service";
|
|
36
|
-
export class AudioRecorder extends BasicRecorder {
|
|
37
|
-
constructor(bpo, changeDetectorRef, renderer, dialog, sessionService, recFileService, uploader, config) {
|
|
38
|
-
super(bpo, changeDetectorRef, dialog, sessionService, uploader, config);
|
|
39
|
-
this.bpo = bpo;
|
|
40
|
-
this.renderer = renderer;
|
|
41
|
-
this.recFileService = recFileService;
|
|
42
|
-
this.uploader = uploader;
|
|
43
|
-
this._project = null;
|
|
44
|
-
this.projectName = null;
|
|
45
|
-
this.enableUploadRecordings = true;
|
|
46
|
-
this.enableDownloadRecordings = false;
|
|
47
|
-
this.status = 0 /* Status.BLOCKED */;
|
|
48
|
-
this.dataSaved = true;
|
|
49
|
-
this._displayRecFile = null;
|
|
50
|
-
this.displayRecFileVersion = 0;
|
|
51
|
-
//super(injector);
|
|
52
|
-
this.status = 1 /* Status.IDLE */;
|
|
53
|
-
this.audio = document.getElementById('audio');
|
|
54
|
-
if (this.config && this.config.enableUploadRecordings != null) {
|
|
55
|
-
this.enableUploadRecordings = this.config.enableUploadRecordings;
|
|
56
|
-
}
|
|
57
|
-
if (this.config && this.config.enableDownloadRecordings != null) {
|
|
58
|
-
this.enableDownloadRecordings = this.config.enableDownloadRecordings;
|
|
59
|
-
}
|
|
60
|
-
//this.init();
|
|
61
|
-
}
|
|
62
|
-
ngAfterViewInit() {
|
|
63
|
-
this.streamLevelMeasure.levelListener = this.liveLevelDisplay;
|
|
64
|
-
this.streamLevelMeasure.peakLevelListener = (peakLvlInDb) => {
|
|
65
|
-
this.peakLevelInDb = peakLvlInDb;
|
|
66
|
-
this.changeDetectorRef.detectChanges();
|
|
67
|
-
};
|
|
68
|
-
//let wakeLockSupp=('wakeLock' in navigator);
|
|
69
|
-
//alert('Wake lock API supported: '+wakeLockSupp);
|
|
70
|
-
}
|
|
71
|
-
ready() {
|
|
72
|
-
return this.dataSaved && !this.isActive();
|
|
73
|
-
}
|
|
74
|
-
ngOnDestroy() {
|
|
75
|
-
this.disableWakeLockCond();
|
|
76
|
-
this.destroyed = true;
|
|
77
|
-
// TODO stop capture /playback
|
|
78
|
-
}
|
|
79
|
-
ngOnInit() {
|
|
80
|
-
this.transportActions.startAction.disabled = true;
|
|
81
|
-
this.transportActions.stopAction.disabled = true;
|
|
82
|
-
this.transportActions.nextAction.disabled = true;
|
|
83
|
-
this.transportActions.pauseAction.disabled = true;
|
|
84
|
-
this.playStartAction.disabled = true;
|
|
85
|
-
this.ac = new AudioCapture();
|
|
86
|
-
if (this.ac) {
|
|
87
|
-
this.transportActions.startAction.onAction = () => this.startItem();
|
|
88
|
-
this.ac.listener = this;
|
|
89
|
-
this.configureStreamCaptureStream();
|
|
90
|
-
// Don't list the devices here. If we do not have audio permissions we only get anonymized devices without labels.
|
|
91
|
-
//this.ac.listDevices();
|
|
92
|
-
}
|
|
93
|
-
else {
|
|
94
|
-
this.transportActions.startAction.disabled = true;
|
|
95
|
-
let errMsg = 'Browser does not support Media/Audio API!';
|
|
96
|
-
this.statusMsg = 'ERROR: ' + errMsg;
|
|
97
|
-
this.statusAlertType = 'error';
|
|
98
|
-
this.dialog.open(MessageDialog, {
|
|
99
|
-
data: {
|
|
100
|
-
type: 'error',
|
|
101
|
-
title: 'Error',
|
|
102
|
-
msg: errMsg,
|
|
103
|
-
advice: 'Please use a supported browser.',
|
|
104
|
-
}
|
|
105
|
-
});
|
|
106
|
-
return;
|
|
107
|
-
}
|
|
108
|
-
this.transportActions.stopAction.onAction = () => this.stopItem();
|
|
109
|
-
this.transportActions.nextAction.onAction = () => this.stopItem();
|
|
110
|
-
//this.transportActions.pauseAction.onAction = () => this.pauseItem();
|
|
111
|
-
this.playStartAction.onAction = () => this.controlAudioPlayer?.start();
|
|
112
|
-
this.uploader.listener = (ue) => {
|
|
113
|
-
this.uploadUpdate(ue);
|
|
114
|
-
};
|
|
115
|
-
}
|
|
116
|
-
onKeyPress(ke) {
|
|
117
|
-
if (ke.key == ' ') {
|
|
118
|
-
//this.transportActions.startAction.perform();
|
|
119
|
-
//this.transportActions.nextAction.perform();
|
|
120
|
-
}
|
|
121
|
-
}
|
|
122
|
-
onKeyDown(ke) {
|
|
123
|
-
if (ke.key == ' ') {
|
|
124
|
-
if (!this.transportActions.startAction.disabled) {
|
|
125
|
-
this.transportActions.startAction.perform();
|
|
126
|
-
}
|
|
127
|
-
else if (!this.transportActions.stopAction.disabled) {
|
|
128
|
-
this.transportActions.stopAction.perform();
|
|
129
|
-
}
|
|
130
|
-
}
|
|
131
|
-
if (ke.key == 'p') {
|
|
132
|
-
this.transportActions.pauseAction.perform();
|
|
133
|
-
}
|
|
134
|
-
if (ke.key == 'Escape') {
|
|
135
|
-
if (!this.audioSignalCollapsed) {
|
|
136
|
-
this.audioSignalCollapsed = true;
|
|
137
|
-
}
|
|
138
|
-
this.transportActions.stopAction.perform();
|
|
139
|
-
this.transportActions.pauseAction.perform();
|
|
140
|
-
}
|
|
141
|
-
if (ke.key == 'MediaPlayPause') {
|
|
142
|
-
this.playStartAction.perform();
|
|
143
|
-
}
|
|
144
|
-
if (ke.key === 'ArrowRight') {
|
|
145
|
-
this.transportActions.fwdAction.perform();
|
|
146
|
-
}
|
|
147
|
-
if (ke.key === 'ArrowLeft') {
|
|
148
|
-
this.transportActions.bwdAction.perform();
|
|
149
|
-
}
|
|
150
|
-
}
|
|
151
|
-
isTestSession() {
|
|
152
|
-
return ((this._session != null) && (this._session.type === 'TEST' || this._session.type === 'TEST_DEF_A' || this._session.type === 'SINUS_TEST'));
|
|
153
|
-
}
|
|
154
|
-
isDefaultAudioTestSession() {
|
|
155
|
-
return ((this._session != null) && (this._session.type === 'TEST_DEF_A'));
|
|
156
|
-
}
|
|
157
|
-
isDefaultAudioTestSessionOverwriteingProjectRequirements() {
|
|
158
|
-
return ((this._session != null) && (this._session.type === 'TEST_DEF_A') && (this.audioDevices != null) && this.audioDevices.length > 0);
|
|
159
|
-
}
|
|
160
|
-
fetchRecordings(sess) {
|
|
161
|
-
this.statusAlertType = 'info';
|
|
162
|
-
this.statusMsg = 'Fetching infos of recordings...';
|
|
163
|
-
this.statusWaiting = true;
|
|
164
|
-
//let prNm:string|null=null;
|
|
165
|
-
if (this.project) {
|
|
166
|
-
let rfsObs = this.recFileService.recordingFileList(this.project.name, sess.sessionId);
|
|
167
|
-
rfsObs.subscribe({ next: (rfs) => {
|
|
168
|
-
this.statusAlertType = 'info';
|
|
169
|
-
this.statusMsg = 'Received infos of recordings.';
|
|
170
|
-
this.statusWaiting = false;
|
|
171
|
-
if (rfs) {
|
|
172
|
-
if (rfs instanceof Array) {
|
|
173
|
-
rfs.forEach((rf) => {
|
|
174
|
-
// the list comes from the server, asssuem all recording files as server persisted
|
|
175
|
-
rf.serverPersisted = true;
|
|
176
|
-
if (rf.startedDate) {
|
|
177
|
-
rf._startedAsDateObj = new Date(rf.startedDate);
|
|
178
|
-
}
|
|
179
|
-
if (rf.date) {
|
|
180
|
-
rf._dateAsDateObj = new Date(rf.date);
|
|
181
|
-
}
|
|
182
|
-
this.recorderCombiPane.addRecFile(rf);
|
|
183
|
-
});
|
|
184
|
-
}
|
|
185
|
-
else {
|
|
186
|
-
console.error('Expected type array for list of already recorded files ');
|
|
187
|
-
}
|
|
188
|
-
}
|
|
189
|
-
else {
|
|
190
|
-
//console.debug("Recording file list: " + rfs);
|
|
191
|
-
}
|
|
192
|
-
}, error: (err) => {
|
|
193
|
-
// Failed fetching existing, but we start the session anyway
|
|
194
|
-
this.start();
|
|
195
|
-
}, complete: () => {
|
|
196
|
-
// Normal start
|
|
197
|
-
this.start();
|
|
198
|
-
} });
|
|
199
|
-
}
|
|
200
|
-
else {
|
|
201
|
-
// No project def -> error
|
|
202
|
-
this.statusAlertType = 'error';
|
|
203
|
-
this.statusMsg = 'No project definiton.';
|
|
204
|
-
this.statusWaiting = false;
|
|
205
|
-
console.error(this.statusMsg);
|
|
206
|
-
}
|
|
207
|
-
}
|
|
208
|
-
set project(project) {
|
|
209
|
-
this._project = project;
|
|
210
|
-
let chCnt = ProjectUtil.DEFAULT_AUDIO_CHANNEL_COUNT;
|
|
211
|
-
if (project) {
|
|
212
|
-
console.info("Project name: " + project.name);
|
|
213
|
-
if (project.recordingDeviceWakeLock === true) {
|
|
214
|
-
this.wakeLock = true;
|
|
215
|
-
}
|
|
216
|
-
this.audioDevices = project.audioDevices;
|
|
217
|
-
chCnt = ProjectUtil.audioChannelCount(project);
|
|
218
|
-
console.info("Project requested recording channel count: " + chCnt);
|
|
219
|
-
this.autoGainControlConfigs = project.autoGainControlConfigs;
|
|
220
|
-
if (project.allowEchoCancellation !== undefined) {
|
|
221
|
-
this.allowEchoCancellation = project.allowEchoCancellation;
|
|
222
|
-
}
|
|
223
|
-
if (project.chunkedRecording === true) {
|
|
224
|
-
this.uploadChunkSizeSeconds = BasicRecorder.DEFAULT_CHUNK_SIZE_SECONDS;
|
|
225
|
-
}
|
|
226
|
-
else {
|
|
227
|
-
this.uploadChunkSizeSeconds = null;
|
|
228
|
-
}
|
|
229
|
-
if (project.clientAudioStorageType) {
|
|
230
|
-
this.clientAudioStorageType = project.clientAudioStorageType;
|
|
231
|
-
}
|
|
232
|
-
}
|
|
233
|
-
else {
|
|
234
|
-
console.error("Empty project configuration!");
|
|
235
|
-
}
|
|
236
|
-
this.channelCount = chCnt;
|
|
237
|
-
}
|
|
238
|
-
get project() {
|
|
239
|
-
return this._project;
|
|
240
|
-
}
|
|
241
|
-
selectRecordingFile(rf) {
|
|
242
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
243
|
-
this.keepLiveLevel = false;
|
|
244
|
-
this.displayRecFile = rf;
|
|
245
|
-
}
|
|
246
|
-
uploadUpdate(ue) {
|
|
247
|
-
let upStatus = ue.status;
|
|
248
|
-
this.dataSaved = (UploaderStatus.DONE === upStatus);
|
|
249
|
-
let percentUpl = ue.percentDone();
|
|
250
|
-
if (UploaderStatus.ERR === upStatus) {
|
|
251
|
-
this.uploadStatus = 'warn';
|
|
252
|
-
}
|
|
253
|
-
else {
|
|
254
|
-
if (percentUpl < 50) {
|
|
255
|
-
this.uploadStatus = 'accent';
|
|
256
|
-
}
|
|
257
|
-
else {
|
|
258
|
-
this.uploadStatus = 'success';
|
|
259
|
-
}
|
|
260
|
-
this.uploadProgress = percentUpl;
|
|
261
|
-
}
|
|
262
|
-
this.changeDetectorRef.detectChanges();
|
|
263
|
-
}
|
|
264
|
-
set controlAudioPlayer(controlAudioPlayer) {
|
|
265
|
-
this._controlAudioPlayer = controlAudioPlayer;
|
|
266
|
-
if (this._controlAudioPlayer) {
|
|
267
|
-
this._controlAudioPlayer.listener = this;
|
|
268
|
-
}
|
|
269
|
-
}
|
|
270
|
-
get controlAudioPlayer() {
|
|
271
|
-
return this._controlAudioPlayer;
|
|
272
|
-
}
|
|
273
|
-
update(e) {
|
|
274
|
-
if (e.type == EventType.STARTED) {
|
|
275
|
-
this.playStartAction.disabled = true;
|
|
276
|
-
this.updateTimerId = window.setInterval(() => {
|
|
277
|
-
//this.audioSignal.playFramePosition = this.ap.playPositionFrames;
|
|
278
|
-
}, 50);
|
|
279
|
-
}
|
|
280
|
-
else if (e.type == EventType.STOPPED || e.type == EventType.ENDED) {
|
|
281
|
-
window.clearInterval(this.updateTimerId);
|
|
282
|
-
//console.debug("Enable play start action (by player events stopped or ended): "+(!(this.displayRecFile)));
|
|
283
|
-
this.playStartAction.disabled = (!(this.displayRecFile));
|
|
284
|
-
}
|
|
285
|
-
}
|
|
286
|
-
startDisabled() {
|
|
287
|
-
return !this.transportActions || this.readonly || this.transportActions.startAction.disabled;
|
|
288
|
-
}
|
|
289
|
-
stopDisabled() {
|
|
290
|
-
return !this.transportActions || this.transportActions.stopAction.disabled;
|
|
291
|
-
}
|
|
292
|
-
startStopNextName() {
|
|
293
|
-
if (!this.startDisabled()) {
|
|
294
|
-
this.startStopNextButtonName = "Start";
|
|
295
|
-
}
|
|
296
|
-
else if (!this.stopDisabled()) {
|
|
297
|
-
this.startStopNextButtonName = "Stop";
|
|
298
|
-
}
|
|
299
|
-
return this.startStopNextButtonName;
|
|
300
|
-
}
|
|
301
|
-
startStopNextIconName() {
|
|
302
|
-
if (!this.startDisabled()) {
|
|
303
|
-
this.startStopNextButtonIconName = "fiber_manual_record";
|
|
304
|
-
}
|
|
305
|
-
else if (!this.stopDisabled()) {
|
|
306
|
-
this.startStopNextButtonIconName = "stop";
|
|
307
|
-
}
|
|
308
|
-
return this.startStopNextButtonIconName;
|
|
309
|
-
}
|
|
310
|
-
startStopNextIconColor() {
|
|
311
|
-
if (!this.startDisabled()) {
|
|
312
|
-
return "red";
|
|
313
|
-
}
|
|
314
|
-
else if (!this.stopDisabled()) {
|
|
315
|
-
return "yellow";
|
|
316
|
-
}
|
|
317
|
-
else {
|
|
318
|
-
return "grey";
|
|
319
|
-
}
|
|
320
|
-
}
|
|
321
|
-
startStopPerform() {
|
|
322
|
-
if (!this.startDisabled()) {
|
|
323
|
-
this.transportActions.startAction.perform();
|
|
324
|
-
}
|
|
325
|
-
else if (!this.stopDisabled()) {
|
|
326
|
-
this.transportActions.stopAction.perform();
|
|
327
|
-
}
|
|
328
|
-
}
|
|
329
|
-
startItem() {
|
|
330
|
-
this.status = 2 /* Status.STARTING */;
|
|
331
|
-
super.startItem();
|
|
332
|
-
if (this.readonly) {
|
|
333
|
-
this.status = 1 /* Status.IDLE */;
|
|
334
|
-
return;
|
|
335
|
-
}
|
|
336
|
-
this.transportActions.fwdAction.disabled = true;
|
|
337
|
-
this.transportActions.fwdNextAction.disabled = true;
|
|
338
|
-
this.transportActions.bwdAction.disabled = true;
|
|
339
|
-
this.displayRecFile = null;
|
|
340
|
-
this.displayRecFileVersion = 0;
|
|
341
|
-
this.displayAudioClip = null;
|
|
342
|
-
this.liveLevelDisplay.reset(true);
|
|
343
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
344
|
-
this.showRecording();
|
|
345
|
-
this.startCapture();
|
|
346
|
-
}
|
|
347
|
-
downloadRecording() {
|
|
348
|
-
if (this.displayRecFile) {
|
|
349
|
-
let ab = this.displayRecFile.audioDataHolder;
|
|
350
|
-
const ww = new WavWriter(this.project?.mediaStorageFormat?.audioEncoding === AudioStorageFormatEncoding.PCM_FLOAT, this.project?.mediaStorageFormat?.audioPCMsampleSizeInBits);
|
|
351
|
-
let as = ab?.audioSource;
|
|
352
|
-
if (as instanceof AudioBufferSource) {
|
|
353
|
-
ww.writeAsync(as.audioBuffer, (wavFile) => {
|
|
354
|
-
let blob = new Blob([wavFile], { type: 'audio/wav' });
|
|
355
|
-
let rfUrl = URL.createObjectURL(blob);
|
|
356
|
-
let dataDnlLnk = this.renderer.createElement('a');
|
|
357
|
-
//dataDnlLnk.name = 'Recording';
|
|
358
|
-
dataDnlLnk.href = rfUrl;
|
|
359
|
-
this.renderer.appendChild(document.body, dataDnlLnk);
|
|
360
|
-
// download property not yet in TS def
|
|
361
|
-
if (this.displayRecFile) {
|
|
362
|
-
let fn = this.displayRecFile.filenameString();
|
|
363
|
-
fn += '_' + this.displayRecFileVersion;
|
|
364
|
-
fn += '.wav';
|
|
365
|
-
dataDnlLnk.download = fn;
|
|
366
|
-
dataDnlLnk.click();
|
|
367
|
-
}
|
|
368
|
-
this.renderer.removeChild(document.body, dataDnlLnk);
|
|
369
|
-
});
|
|
370
|
-
}
|
|
371
|
-
}
|
|
372
|
-
}
|
|
373
|
-
set displayRecFile(displayRecFile) {
|
|
374
|
-
//this.audioLoaded=false;
|
|
375
|
-
this._displayRecFile = displayRecFile;
|
|
376
|
-
if (this._displayRecFile) {
|
|
377
|
-
let adh = this._displayRecFile.audioDataHolder;
|
|
378
|
-
if (adh) {
|
|
379
|
-
this.displayAudioClip = new AudioClip(adh);
|
|
380
|
-
//this.audioLoaded=true;
|
|
381
|
-
//console.debug(" set recording file: display audio clip set");
|
|
382
|
-
if (this._controlAudioPlayer) {
|
|
383
|
-
this._controlAudioPlayer.audioClip = this.displayAudioClip;
|
|
384
|
-
}
|
|
385
|
-
this.showRecording();
|
|
386
|
-
}
|
|
387
|
-
else {
|
|
388
|
-
// clear for now ...
|
|
389
|
-
this.displayAudioClip = null;
|
|
390
|
-
//console.debug("set recording file: display audio clip null");
|
|
391
|
-
if (this._controlAudioPlayer) {
|
|
392
|
-
this._controlAudioPlayer.audioClip = null;
|
|
393
|
-
}
|
|
394
|
-
if (this._controlAudioPlayer && this._session) {
|
|
395
|
-
//... and try to fetch from server
|
|
396
|
-
this.liveLevelDisplayState = LiveLevelState.LOADING;
|
|
397
|
-
const rf = this._displayRecFile;
|
|
398
|
-
let audioDownloadType = this._clientAudioStorageType;
|
|
399
|
-
if (AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this._clientAudioStorageType || AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this._clientAudioStorageType) {
|
|
400
|
-
// Default is network mode
|
|
401
|
-
audioDownloadType = AudioStorageType.NET_CHUNKED;
|
|
402
|
-
if (rf.channels && rf.frames) {
|
|
403
|
-
const samples = rf.channels * rf.frames;
|
|
404
|
-
if (samples <= this._maxAutoNetMemStoreSamples) {
|
|
405
|
-
// But if audio file is small, load in continuous resp. chunked mode
|
|
406
|
-
if (AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this._clientAudioStorageType) {
|
|
407
|
-
audioDownloadType = AudioStorageType.MEM_ENTIRE;
|
|
408
|
-
}
|
|
409
|
-
else if (AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this._clientAudioStorageType) {
|
|
410
|
-
audioDownloadType = AudioStorageType.MEM_CHUNKED;
|
|
411
|
-
}
|
|
412
|
-
}
|
|
413
|
-
}
|
|
414
|
-
}
|
|
415
|
-
if (AudioStorageType.DB_CHUNKED === this._clientAudioStorageType) {
|
|
416
|
-
// Fetch chunked indexed db audio buffer
|
|
417
|
-
let nextIab = null;
|
|
418
|
-
if (!this._persistentAudioStorageTarget) {
|
|
419
|
-
throw Error('Error: Persistent storage target not set.');
|
|
420
|
-
}
|
|
421
|
-
else {
|
|
422
|
-
//console.debug("Fetch audio and store to indexed db...");
|
|
423
|
-
this.audioFetchSubscription = this.recFileService.fetchRecordingFileIndDbAudioBuffer(this._persistentAudioStorageTarget, this._session.project, rf).subscribe({
|
|
424
|
-
next: (iab) => {
|
|
425
|
-
//console.debug("Sessionmanager: Received inddb audio buffer: "+iab);
|
|
426
|
-
nextIab = iab;
|
|
427
|
-
},
|
|
428
|
-
complete: () => {
|
|
429
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
430
|
-
let fabDh = null;
|
|
431
|
-
if (nextIab) {
|
|
432
|
-
if (rf) {
|
|
433
|
-
fabDh = new AudioDataHolder(nextIab);
|
|
434
|
-
//this.audioLoaded=true;
|
|
435
|
-
this.recorderCombiPane.setRecFileAudioData(rf, fabDh);
|
|
436
|
-
}
|
|
437
|
-
}
|
|
438
|
-
else {
|
|
439
|
-
// Should actually be handled by the error resolver
|
|
440
|
-
this.statusMsg = 'Recording file could not be loaded.';
|
|
441
|
-
this.statusAlertType = 'error';
|
|
442
|
-
}
|
|
443
|
-
if (fabDh) {
|
|
444
|
-
// this.displayAudioClip could have been changed meanwhile, but the recorder unsubcribes before changing the item. Therefore, there should be no risk to set to wrong item
|
|
445
|
-
this.displayAudioClip = new AudioClip(fabDh);
|
|
446
|
-
}
|
|
447
|
-
if (this._controlAudioPlayer) {
|
|
448
|
-
this._controlAudioPlayer.audioClip = this.displayAudioClip;
|
|
449
|
-
}
|
|
450
|
-
this.showRecording();
|
|
451
|
-
},
|
|
452
|
-
error: err => {
|
|
453
|
-
console.error("Could not load recording file from server: " + err);
|
|
454
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
455
|
-
this.statusMsg = 'Recording file could not be loaded: ' + err;
|
|
456
|
-
this.statusAlertType = 'error';
|
|
457
|
-
this.changeDetectorRef.detectChanges();
|
|
458
|
-
}
|
|
459
|
-
});
|
|
460
|
-
}
|
|
461
|
-
}
|
|
462
|
-
else if (AudioStorageType.NET_CHUNKED === audioDownloadType) {
|
|
463
|
-
// Fetch chunked audio buffer from network
|
|
464
|
-
let nextNetAb = null;
|
|
465
|
-
//console.debug("Fetch chunked audio from network");
|
|
466
|
-
this.audioFetchSubscription = this.recFileService.fetchRecordingFileNetAudioBuffer(this._session.project, rf).subscribe({
|
|
467
|
-
next: (netAb) => {
|
|
468
|
-
//console.debug("Sessionmanager: Received net audio buffer: "+netAb);
|
|
469
|
-
nextNetAb = netAb;
|
|
470
|
-
},
|
|
471
|
-
complete: () => {
|
|
472
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
473
|
-
let fabDh = null;
|
|
474
|
-
if (nextNetAb) {
|
|
475
|
-
if (rf) {
|
|
476
|
-
fabDh = new AudioDataHolder(nextNetAb);
|
|
477
|
-
this.recorderCombiPane.setRecFileAudioData(rf, fabDh);
|
|
478
|
-
}
|
|
479
|
-
}
|
|
480
|
-
else {
|
|
481
|
-
// Should actually be handled by the error resolver
|
|
482
|
-
this.statusMsg = 'Recording file could not be loaded.';
|
|
483
|
-
this.statusAlertType = 'error';
|
|
484
|
-
}
|
|
485
|
-
if (fabDh) {
|
|
486
|
-
// this.displayAudioClip could have been changed meanwhile, but the recorder unsubcribes before changing the item. Therefore, there should be no risk to set to wrong item
|
|
487
|
-
//console.debug("set displayRecFile(): fetch net ab complete, set displayAudioClip.")
|
|
488
|
-
this.displayAudioClip = new AudioClip(fabDh);
|
|
489
|
-
// fabDh.onReady=()=>{
|
|
490
|
-
// this.audioLoaded=true;
|
|
491
|
-
// }
|
|
492
|
-
}
|
|
493
|
-
if (this._controlAudioPlayer) {
|
|
494
|
-
this._controlAudioPlayer.audioClip = this.displayAudioClip;
|
|
495
|
-
}
|
|
496
|
-
this.showRecording();
|
|
497
|
-
},
|
|
498
|
-
error: err => {
|
|
499
|
-
console.error("Could not load recording file from server: " + err);
|
|
500
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
501
|
-
this.statusMsg = 'Recording file could not be loaded: ' + err;
|
|
502
|
-
this.statusAlertType = 'error';
|
|
503
|
-
this.changeDetectorRef.detectChanges();
|
|
504
|
-
}
|
|
505
|
-
});
|
|
506
|
-
}
|
|
507
|
-
else if (AudioStorageType.MEM_CHUNKED === audioDownloadType) {
|
|
508
|
-
// Fetch chunked array audio buffer
|
|
509
|
-
let nextAab = null;
|
|
510
|
-
//console.debug("Fetch audio and store to (chunked) array buffer...");
|
|
511
|
-
this.audioFetchSubscription = this.recFileService.fetchRecordingFileArrayAudioBuffer(this._session.project, rf).subscribe({
|
|
512
|
-
next: (aab) => {
|
|
513
|
-
nextAab = aab;
|
|
514
|
-
},
|
|
515
|
-
complete: () => {
|
|
516
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
517
|
-
let fabDh = null;
|
|
518
|
-
if (nextAab) {
|
|
519
|
-
if (rf) {
|
|
520
|
-
fabDh = new AudioDataHolder(nextAab);
|
|
521
|
-
this.recorderCombiPane.setRecFileAudioData(rf, fabDh);
|
|
522
|
-
}
|
|
523
|
-
}
|
|
524
|
-
else {
|
|
525
|
-
// Should actually be handled by the error resolver
|
|
526
|
-
this.statusMsg = 'Recording file could not be loaded.';
|
|
527
|
-
this.statusAlertType = 'error';
|
|
528
|
-
}
|
|
529
|
-
if (fabDh) {
|
|
530
|
-
// this.displayAudioClip could have been changed meanwhile, but the recorder unsubcribes before changing the item. Therefore, there should be no risk to set to wrong item
|
|
531
|
-
this.displayAudioClip = new AudioClip(fabDh);
|
|
532
|
-
//this.audioLoaded=true;
|
|
533
|
-
}
|
|
534
|
-
if (this._controlAudioPlayer) {
|
|
535
|
-
this._controlAudioPlayer.audioClip = this.displayAudioClip;
|
|
536
|
-
}
|
|
537
|
-
this.showRecording();
|
|
538
|
-
},
|
|
539
|
-
error: err => {
|
|
540
|
-
console.error("Could not load recording file from server: " + err);
|
|
541
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
542
|
-
this.statusMsg = 'Recording file could not be loaded: ' + err;
|
|
543
|
-
this.statusAlertType = 'error';
|
|
544
|
-
}
|
|
545
|
-
});
|
|
546
|
-
}
|
|
547
|
-
else {
|
|
548
|
-
this.audioFetchSubscription = this.recFileService.fetchRecordingFileAudioBuffer(this._session.project, rf).subscribe({
|
|
549
|
-
next: ab => {
|
|
550
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
551
|
-
let fabDh = null;
|
|
552
|
-
if (ab) {
|
|
553
|
-
let abSrc = new AudioBufferSource(ab);
|
|
554
|
-
if (rf) {
|
|
555
|
-
fabDh = new AudioDataHolder(abSrc);
|
|
556
|
-
this.recorderCombiPane.setRecFileAudioData(rf, fabDh);
|
|
557
|
-
}
|
|
558
|
-
}
|
|
559
|
-
else {
|
|
560
|
-
console.error('Recording file could not be loaded.');
|
|
561
|
-
this.statusMsg = 'Recording file could not be loaded.';
|
|
562
|
-
this.statusAlertType = 'error';
|
|
563
|
-
}
|
|
564
|
-
if (fabDh) {
|
|
565
|
-
// this.displayAudioClip could have been changed meanwhile, but the recorder unsubcribes before changing the item. Therefore, there should be no risk to set to wrong item
|
|
566
|
-
this.displayAudioClip = new AudioClip(fabDh);
|
|
567
|
-
//this.audioLoaded=true;
|
|
568
|
-
//console.debug("set recording file: display audio clip from fetched audio buffer");
|
|
569
|
-
}
|
|
570
|
-
if (this._controlAudioPlayer) {
|
|
571
|
-
this._controlAudioPlayer.audioClip = this.displayAudioClip;
|
|
572
|
-
}
|
|
573
|
-
this.showRecording();
|
|
574
|
-
}, error: err => {
|
|
575
|
-
console.error("Could not load recording file from server: " + err);
|
|
576
|
-
this.liveLevelDisplayState = LiveLevelState.READY;
|
|
577
|
-
this.statusMsg = 'Recording file could not be loaded: ' + err;
|
|
578
|
-
this.statusAlertType = 'error';
|
|
579
|
-
}
|
|
580
|
-
});
|
|
581
|
-
}
|
|
582
|
-
}
|
|
583
|
-
else {
|
|
584
|
-
this.statusMsg = 'Recording file could not be decoded. Audio context unavailable.';
|
|
585
|
-
this.statusAlertType = 'error';
|
|
586
|
-
}
|
|
587
|
-
}
|
|
588
|
-
}
|
|
589
|
-
else {
|
|
590
|
-
//console.debug("recording file null");
|
|
591
|
-
this.displayAudioClip = null;
|
|
592
|
-
if (this._controlAudioPlayer) {
|
|
593
|
-
this._controlAudioPlayer.audioClip = null;
|
|
594
|
-
}
|
|
595
|
-
}
|
|
596
|
-
this.showRecording();
|
|
597
|
-
}
|
|
598
|
-
get displayRecFile() {
|
|
599
|
-
return this._displayRecFile;
|
|
600
|
-
}
|
|
601
|
-
updateStartActionDisableState() {
|
|
602
|
-
this.transportActions.startAction.disabled = !(this.ac);
|
|
603
|
-
}
|
|
604
|
-
start() {
|
|
605
|
-
super.start();
|
|
606
|
-
this.recorderCombiPane.selectTop();
|
|
607
|
-
this.enableNavigation();
|
|
608
|
-
this.updateStartActionDisableState();
|
|
609
|
-
}
|
|
610
|
-
isRecording() {
|
|
611
|
-
return (this.status === 3 /* Status.RECORDING */ || this.status === 4 /* Status.STOPPING_STOP */);
|
|
612
|
-
}
|
|
613
|
-
isActive() {
|
|
614
|
-
return (!(this.status === 0 /* Status.BLOCKED */ || this.status === 1 /* Status.IDLE */ || this.status === 5 /* Status.ERROR */) || this.processingRecording || this.sessionService.uploadCount > 0);
|
|
615
|
-
}
|
|
616
|
-
updateWakeLock(dataSaved = this.dataSaved) {
|
|
617
|
-
//console.debug("Update wake lock: dataSaved: "+dataSaved+", not active: "+! this.isActive())
|
|
618
|
-
if (dataSaved && !this.isActive()) {
|
|
619
|
-
this.disableWakeLockCond();
|
|
620
|
-
}
|
|
621
|
-
}
|
|
622
|
-
updateNavigationActions() {
|
|
623
|
-
this.transportActions.fwdAction.disabled = this.navigationDisabled;
|
|
624
|
-
this.transportActions.bwdAction.disabled = this.navigationDisabled;
|
|
625
|
-
}
|
|
626
|
-
enableStartUserGesture() {
|
|
627
|
-
super.enableStartUserGesture();
|
|
628
|
-
this.transportActions.startAction.disabled = !(this.ac);
|
|
629
|
-
}
|
|
630
|
-
enableNavigation() {
|
|
631
|
-
this.navigationDisabled = false;
|
|
632
|
-
this.updateNavigationActions();
|
|
633
|
-
}
|
|
634
|
-
started() {
|
|
635
|
-
this.status = 3 /* Status.RECORDING */;
|
|
636
|
-
super.started();
|
|
637
|
-
this.statusAlertType = 'info';
|
|
638
|
-
this.statusMsg = 'Recording...';
|
|
639
|
-
let sessId = 0;
|
|
640
|
-
if (this._session) {
|
|
641
|
-
sessId = this._session.sessionId;
|
|
642
|
-
}
|
|
643
|
-
if (this.rfUuid) {
|
|
644
|
-
let rf = new RecordingFile(this.rfUuid, sessId, null);
|
|
645
|
-
rf._startedAsDateObj = this.startedDate;
|
|
646
|
-
if (rf._startedAsDateObj) {
|
|
647
|
-
rf.startedDate = rf._startedAsDateObj.toString();
|
|
648
|
-
}
|
|
649
|
-
this._recordingFile = rf;
|
|
650
|
-
}
|
|
651
|
-
let maxRecordingTimeMs = MAX_RECORDING_TIME_MS;
|
|
652
|
-
this.maxRecTimerId = window.setTimeout(() => {
|
|
653
|
-
this.stopRecordingMaxRec();
|
|
654
|
-
}, maxRecordingTimeMs);
|
|
655
|
-
this.maxRecTimerRunning = true;
|
|
656
|
-
this.transportActions.stopAction.disabled = false;
|
|
657
|
-
}
|
|
658
|
-
stopItem() {
|
|
659
|
-
this.transportActions.stopAction.disabled = true;
|
|
660
|
-
this.transportActions.nextAction.disabled = true;
|
|
661
|
-
this.status = 4 /* Status.STOPPING_STOP */;
|
|
662
|
-
this.stopRecording();
|
|
663
|
-
}
|
|
664
|
-
stopRecording() {
|
|
665
|
-
if (this.maxRecTimerRunning && this.maxRecTimerId != null) {
|
|
666
|
-
window.clearTimeout(this.maxRecTimerId);
|
|
667
|
-
this.maxRecTimerRunning = false;
|
|
668
|
-
}
|
|
669
|
-
if (this.ac) {
|
|
670
|
-
this.ac.stop();
|
|
671
|
-
}
|
|
672
|
-
}
|
|
673
|
-
stopRecordingMaxRec() {
|
|
674
|
-
this.maxRecTimerRunning = false;
|
|
675
|
-
this.status = 4 /* Status.STOPPING_STOP */;
|
|
676
|
-
if (this.ac) {
|
|
677
|
-
this.ac.stop();
|
|
678
|
-
}
|
|
679
|
-
}
|
|
680
|
-
stopped() {
|
|
681
|
-
this.updateStartActionDisableState();
|
|
682
|
-
this.transportActions.stopAction.disabled = true;
|
|
683
|
-
this.transportActions.nextAction.disabled = true;
|
|
684
|
-
this.transportActions.pauseAction.disabled = true;
|
|
685
|
-
this.statusAlertType = 'info';
|
|
686
|
-
this.statusMsg = 'Recorded.';
|
|
687
|
-
let ab = null;
|
|
688
|
-
if (this.ac) {
|
|
689
|
-
let adh = null;
|
|
690
|
-
let as = null;
|
|
691
|
-
if (this.ac) {
|
|
692
|
-
if (AudioStorageType.NET_CHUNKED === this.ac.audioStorageType) {
|
|
693
|
-
this.playStartAction.disabled = true;
|
|
694
|
-
//this.audioLoaded=false;
|
|
695
|
-
this.keepLiveLevel = true;
|
|
696
|
-
let rUUID = null;
|
|
697
|
-
let burl = null;
|
|
698
|
-
if (this._session) {
|
|
699
|
-
if (this._recordingFile) {
|
|
700
|
-
let rf = this._recordingFile;
|
|
701
|
-
rf.frames = this.ac.framesRecorded;
|
|
702
|
-
rUUID = rf.uuid;
|
|
703
|
-
//console.debug("stopped(): Set frames: "+rf.frames+" on rfId: "+this.displayRecFile?.recordingFileId);
|
|
704
|
-
burl = this.recFileService.audioFileUrl(this._session?.project, rf);
|
|
705
|
-
}
|
|
706
|
-
else if (this.session?.project) {
|
|
707
|
-
if (this.ac.recUUID) {
|
|
708
|
-
rUUID = this.ac.recUUID;
|
|
709
|
-
burl = this.recFileService.audioFileUrlByUUID(this.session.project, this.session.sessionId, rUUID);
|
|
710
|
-
}
|
|
711
|
-
}
|
|
712
|
-
else {
|
|
713
|
-
console.error("Could not create net audio buffer.");
|
|
714
|
-
}
|
|
715
|
-
if (burl) {
|
|
716
|
-
const sr = this.ac.currentSampleRate;
|
|
717
|
-
const chFl = sr * RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;
|
|
718
|
-
//console.debug("stopped(): rfID: "+this._recordingFile?.recordingFileId+", net ab url: " + burl+", frames: "+this.ac.framesRecorded+", sample rate: "+sr);
|
|
719
|
-
let netAs = new NetAudioBuffer(this.recFileService, burl, this.ac.channelCount, sr, chFl, this.ac.framesRecorded, rUUID, chFl);
|
|
720
|
-
as = netAs;
|
|
721
|
-
if (this.uploadSet) {
|
|
722
|
-
this.uploadSet.onDone = (uploadSet) => {
|
|
723
|
-
//console.debug("upload set on done: Call ready provider.ready");
|
|
724
|
-
netAs.ready();
|
|
725
|
-
};
|
|
726
|
-
}
|
|
727
|
-
}
|
|
728
|
-
}
|
|
729
|
-
}
|
|
730
|
-
else if (AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this.ac.audioStorageType || AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this.ac.audioStorageType) {
|
|
731
|
-
if (AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this.ac.audioStorageType) {
|
|
732
|
-
const acAb = this.ac.audioBuffer();
|
|
733
|
-
if (acAb) {
|
|
734
|
-
as = new AudioBufferSource(acAb);
|
|
735
|
-
}
|
|
736
|
-
}
|
|
737
|
-
if (AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this.ac.audioStorageType) {
|
|
738
|
-
as = this.ac.audioBufferArray();
|
|
739
|
-
}
|
|
740
|
-
if (!as) {
|
|
741
|
-
this.playStartAction.disabled = true;
|
|
742
|
-
this.keepLiveLevel = true;
|
|
743
|
-
let rUUID = null;
|
|
744
|
-
let burl = null;
|
|
745
|
-
if (this._session) {
|
|
746
|
-
if (this._recordingFile) {
|
|
747
|
-
let rf = this._recordingFile;
|
|
748
|
-
rf.frames = this.ac.framesRecorded;
|
|
749
|
-
rUUID = rf.uuid;
|
|
750
|
-
//console.debug("stopped(): Set frames: "+rf.frames+" on rfId: "+this.displayRecFile?.recordingFileId);
|
|
751
|
-
burl = this.recFileService.audioFileUrl(this._session?.project, rf);
|
|
752
|
-
}
|
|
753
|
-
else if (this.session?.project) {
|
|
754
|
-
if (this.ac.recUUID) {
|
|
755
|
-
rUUID = this.ac.recUUID;
|
|
756
|
-
burl = this.recFileService.audioFileUrlByUUID(this.session.project, this.session.sessionId, rUUID);
|
|
757
|
-
}
|
|
758
|
-
}
|
|
759
|
-
else {
|
|
760
|
-
console.error("Could not create net audio buffer.");
|
|
761
|
-
}
|
|
762
|
-
if (burl) {
|
|
763
|
-
const sr = this.ac.currentSampleRate;
|
|
764
|
-
const chFl = sr * RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;
|
|
765
|
-
//console.debug("stopped(): rfID: "+this._recordingFile?.recordingFileId+", net ab url: " + burl+", frames: "+this.ac.framesRecorded+", sample rate: "+sr);
|
|
766
|
-
let netAs = new NetAudioBuffer(this.recFileService, burl, this.ac.channelCount, sr, chFl, this.ac.framesRecorded, rUUID, chFl);
|
|
767
|
-
as = netAs;
|
|
768
|
-
if (this.uploadSet) {
|
|
769
|
-
this.uploadSet.onDone = (uploadSet) => {
|
|
770
|
-
//console.debug("upload set on done: Call ready provider.ready");
|
|
771
|
-
netAs.ready();
|
|
772
|
-
};
|
|
773
|
-
}
|
|
774
|
-
}
|
|
775
|
-
}
|
|
776
|
-
}
|
|
777
|
-
}
|
|
778
|
-
else if (AudioStorageType.DB_CHUNKED === this.ac.audioStorageType) {
|
|
779
|
-
as = this.ac.inddbAudioBufferArray();
|
|
780
|
-
}
|
|
781
|
-
else if (AudioStorageType.MEM_CHUNKED === this.ac.audioStorageType) {
|
|
782
|
-
as = this.ac.audioBufferArray();
|
|
783
|
-
}
|
|
784
|
-
else {
|
|
785
|
-
ab = this.ac.audioBuffer();
|
|
786
|
-
if (ab) {
|
|
787
|
-
as = new AudioBufferSource(ab);
|
|
788
|
-
}
|
|
789
|
-
}
|
|
790
|
-
if (as) {
|
|
791
|
-
adh = new AudioDataHolder(as);
|
|
792
|
-
}
|
|
793
|
-
}
|
|
794
|
-
let sessId = 0;
|
|
795
|
-
if (this._session) {
|
|
796
|
-
sessId = this._session.sessionId;
|
|
797
|
-
}
|
|
798
|
-
if (this._recordingFile) {
|
|
799
|
-
//this._recordingFile.samplerate=this.ac.currentSampleRate;
|
|
800
|
-
// Use an own reference since the writing of the wave file is asynchronous and this._recordingFile might already contain the next recording
|
|
801
|
-
let rf = this._recordingFile;
|
|
802
|
-
RecordingFileUtils.setAudioData(rf, adh);
|
|
803
|
-
this.recorderCombiPane.addRecFile(rf);
|
|
804
|
-
// Upload if upload enabled and not in chunked upload mode
|
|
805
|
-
if (this.enableUploadRecordings && this._uploadChunkSizeSeconds === null && AudioStorageType.MEM_ENTIRE === this._clientAudioStorageType && rf != null && ab != null) {
|
|
806
|
-
let apiEndPoint = '';
|
|
807
|
-
if (this.config && this.config.apiEndPoint) {
|
|
808
|
-
apiEndPoint = this.config.apiEndPoint;
|
|
809
|
-
}
|
|
810
|
-
if (apiEndPoint !== '') {
|
|
811
|
-
apiEndPoint = apiEndPoint + '/';
|
|
812
|
-
}
|
|
813
|
-
let sessionsUrl = apiEndPoint + SessionService.SESSION_API_CTX;
|
|
814
|
-
let recUrl = sessionsUrl + '/' + rf.session + '/' + RECFILE_API_CTX + '/' + rf.uuid;
|
|
815
|
-
// convert asynchronously to 16-bit integer PCM
|
|
816
|
-
// TODO could we avoid conversion to save CPU resources and transfer float PCM directly?
|
|
817
|
-
// TODO duplicate conversion for manual download
|
|
818
|
-
this.processingRecording = true;
|
|
819
|
-
const ww = new WavWriter(this.project?.mediaStorageFormat?.audioEncoding === AudioStorageFormatEncoding.PCM_FLOAT, this.project?.mediaStorageFormat?.audioPCMsampleSizeInBits);
|
|
820
|
-
ww.writeAsync(ab, (wavFile) => {
|
|
821
|
-
this.postRecordingMultipart(wavFile, recUrl, rf);
|
|
822
|
-
this.processingRecording = false;
|
|
823
|
-
this.updateWakeLock();
|
|
824
|
-
this.changeDetectorRef.detectChanges();
|
|
825
|
-
});
|
|
826
|
-
}
|
|
827
|
-
}
|
|
828
|
-
}
|
|
829
|
-
this.displayRecFile = this._recordingFile;
|
|
830
|
-
this.status = 1 /* Status.IDLE */;
|
|
831
|
-
this.navigationDisabled = false;
|
|
832
|
-
this.updateNavigationActions();
|
|
833
|
-
this.updateWakeLock();
|
|
834
|
-
this.changeDetectorRef.detectChanges();
|
|
835
|
-
}
|
|
836
|
-
error(msg = 'An unknown error occured during recording.', advice = 'Please retry.') {
|
|
837
|
-
this.status = 5 /* Status.ERROR */;
|
|
838
|
-
super.error(msg, advice);
|
|
839
|
-
this.updateNavigationActions();
|
|
840
|
-
this.updateStartActionDisableState();
|
|
841
|
-
}
|
|
842
|
-
postRecordingMultipart(wavFile, recUrl, rf) {
|
|
843
|
-
let wavBlob = new Blob([wavFile], { type: 'audio/wav' });
|
|
844
|
-
let fd = new FormData();
|
|
845
|
-
if (rf.uuid) {
|
|
846
|
-
fd.set('uuid', rf.uuid);
|
|
847
|
-
}
|
|
848
|
-
if (rf.session !== null) {
|
|
849
|
-
fd.set('sessionId', rf.session.toString());
|
|
850
|
-
}
|
|
851
|
-
if (rf._startedAsDateObj) {
|
|
852
|
-
fd.set('startedDate', rf._startedAsDateObj.toJSON());
|
|
853
|
-
}
|
|
854
|
-
fd.set('audio', wavBlob);
|
|
855
|
-
let ul = new Upload(fd, recUrl, rf);
|
|
856
|
-
this.uploader.queueUpload(ul);
|
|
857
|
-
}
|
|
858
|
-
postChunkAudioBuffer(audioBuffer, chunkIdx) {
|
|
859
|
-
this.processingRecording = true;
|
|
860
|
-
const ww = new WavWriter(this.project?.mediaStorageFormat?.audioEncoding === AudioStorageFormatEncoding.PCM_FLOAT, this.project?.mediaStorageFormat?.audioPCMsampleSizeInBits);
|
|
861
|
-
let sessionsUrl = this.sessionsBaseUrl();
|
|
862
|
-
let recUrl = sessionsUrl + '/' + this.session?.sessionId + '/' + RECFILE_API_CTX + '/' + this.rfUuid + '/' + chunkIdx;
|
|
863
|
-
let rf = this._recordingFile;
|
|
864
|
-
// The upload holder is required to add the upload now to the upload set. The real upload is created async in postrecording and the upload set is already complete at that time.
|
|
865
|
-
let ulh = new UploadHolder();
|
|
866
|
-
if (this.uploadSet) {
|
|
867
|
-
this.uploadSet.add(ulh);
|
|
868
|
-
}
|
|
869
|
-
ww.writeAsync(audioBuffer, (wavFile) => {
|
|
870
|
-
this.postRecording(wavFile, recUrl, rf, ulh);
|
|
871
|
-
this.processingRecording = false;
|
|
872
|
-
});
|
|
873
|
-
}
|
|
874
|
-
stop() {
|
|
875
|
-
if (this.ac) {
|
|
876
|
-
this.ac.close();
|
|
877
|
-
}
|
|
878
|
-
}
|
|
879
|
-
updateControlPlaybackPosition() {
|
|
880
|
-
if (this._controlAudioPlayer) {
|
|
881
|
-
const ppFrames = this._controlAudioPlayer.playPositionFrames;
|
|
882
|
-
if (ppFrames !== null) {
|
|
883
|
-
this.recorderCombiPane.audioDisplay.playFramePosition = ppFrames;
|
|
884
|
-
this.liveLevelDisplay.playFramePosition = ppFrames;
|
|
885
|
-
}
|
|
886
|
-
}
|
|
887
|
-
}
|
|
888
|
-
audioPlayerUpdate(e) {
|
|
889
|
-
if (EventType.READY === e.type) {
|
|
890
|
-
}
|
|
891
|
-
else if (EventType.STARTED === e.type) {
|
|
892
|
-
//this.status = 'Playback...';
|
|
893
|
-
this.updateTimerId = window.setInterval(() => this.updateControlPlaybackPosition(), 50);
|
|
894
|
-
}
|
|
895
|
-
else if (EventType.ENDED === e.type) {
|
|
896
|
-
//.status='Ready.';
|
|
897
|
-
window.clearInterval(this.updateTimerId);
|
|
898
|
-
}
|
|
899
|
-
if (!this.destroyed) {
|
|
900
|
-
this.changeDetectorRef.detectChanges();
|
|
901
|
-
}
|
|
902
|
-
}
|
|
903
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: AudioRecorder, deps: [{ token: i1.BreakpointObserver }, { token: i0.ChangeDetectorRef }, { token: i0.Renderer2 }, { token: i2.MatDialog }, { token: i3.SessionService }, { token: i4.RecordingService }, { token: i5.SpeechRecorderUploader }, { token: SPEECHRECORDER_CONFIG }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
904
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.12", type: AudioRecorder, selector: "app-audiorecorder", inputs: { projectName: "projectName", dataSaved: "dataSaved" }, host: { listeners: { "window:keypress": "onKeyPress($event)", "window:keydown": "onKeyDown($event)" } }, providers: [SessionService], viewQueries: [{ propertyName: "recorderCombiPane", first: true, predicate: RecorderCombiPane, descendants: true, static: true }, { propertyName: "liveLevelDisplay", first: true, predicate: LevelBar, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: `
|
|
905
|
-
<app-warningbar [show]="isTestSession()" warningText="Test recording only!"></app-warningbar>
|
|
906
|
-
<app-warningbar [show]="isDefaultAudioTestSession()"
|
|
907
|
-
warningText="This test uses default audio device! Regular sessions may require a particular audio device (microphone)!"></app-warningbar>
|
|
908
|
-
<app-recordercombipane (selectedRecordingFileChanged)="selectRecordingFile($event)"
|
|
909
|
-
[audioSignalCollapsed]="audioSignalCollapsed"
|
|
910
|
-
[selectedRecordingFile]="displayRecFile"
|
|
911
|
-
[selectDisabled]="isActive()"
|
|
912
|
-
[displayAudioClip]="displayAudioClip"
|
|
913
|
-
[playStartAction]="controlAudioPlayer?.startAction"
|
|
914
|
-
[playStopAction]="controlAudioPlayer?.stopAction"
|
|
915
|
-
[playSelectionAction]="controlAudioPlayer?.startSelectionAction"
|
|
916
|
-
[autoPlayOnSelectToggleAction]="controlAudioPlayer?.autoPlayOnSelectToggleAction"
|
|
917
|
-
></app-recordercombipane>
|
|
918
|
-
|
|
919
|
-
<div [class]="{audioStatusDisplay:!screenXs,audioStatusDisplayXs:screenXs}">
|
|
920
|
-
<audio-levelbar style="flex:1 0 1%" [streamingMode]="isRecording() || keepLiveLevel" [state]="liveLevelDisplayState"
|
|
921
|
-
[displayLevelInfos]="displayAudioClip?.levelInfos"></audio-levelbar>
|
|
922
|
-
<div style="display:flex;flex-direction: row">
|
|
923
|
-
<spr-recordingitemcontrols style="flex:10 0 1px"
|
|
924
|
-
[disableAudioDetails]="disableAudioDetails"
|
|
925
|
-
[audioLoaded]="audioLoaded"
|
|
926
|
-
[playStartAction]="controlAudioPlayer?.startAction"
|
|
927
|
-
[playStopAction]="controlAudioPlayer?.stopAction"
|
|
928
|
-
[peakDbLvl]="peakLevelInDb"
|
|
929
|
-
[agc]="this.ac?.agcStatus"
|
|
930
|
-
(onShowRecordingDetails)="audioSignalCollapsed=!audioSignalCollapsed">
|
|
931
|
-
</spr-recordingitemcontrols>
|
|
932
|
-
|
|
933
|
-
<app-uploadstatus *ngIf="screenXs && enableUploadRecordings" class="ricontrols dark" style="flex:0 0 0"
|
|
934
|
-
[value]="uploadProgress"
|
|
935
|
-
[status]="uploadStatus" [awaitNewUpload]="processingRecording"></app-uploadstatus>
|
|
936
|
-
<app-wakelockindicator *ngIf="screenXs" class="ricontrols dark" style="flex:0 0 0" [screenLocked]="screenLocked"></app-wakelockindicator>
|
|
937
|
-
<app-readystateindicator *ngIf="screenXs" class="ricontrols dark" style="flex:0 0 0"
|
|
938
|
-
[ready]="dataSaved && !isActive()"></app-readystateindicator>
|
|
939
|
-
</div>
|
|
940
|
-
</div>
|
|
941
|
-
<div #controlpanel class="controlpanel">
|
|
942
|
-
<app-sprstatusdisplay *ngIf="!screenXs" style="flex:0 1 30%;" [statusMsg]="statusMsg" [statusAlertType]="statusAlertType"
|
|
943
|
-
[statusWaiting]="statusWaiting"
|
|
944
|
-
class="hidden-xs"></app-sprstatusdisplay>
|
|
945
|
-
<div [class.startstop]="!screenXs" [class.startstopscreenxs]="screenXs">
|
|
946
|
-
<div style="align-content: center">
|
|
947
|
-
<button (click)="startStopPerform()" [disabled]="startDisabled() && stopDisabled()" mat-raised-button class="bigbutton">
|
|
948
|
-
<mat-icon class="bigbuttonicon" [style.color]="startStopNextIconColor()">{{startStopNextIconName()}}</mat-icon>
|
|
949
|
-
<span class="bigbuttontext">{{startStopNextName()}}</span>
|
|
950
|
-
</button>
|
|
951
|
-
</div>
|
|
952
|
-
</div>
|
|
953
|
-
<div style="flex:0 1 30%;display:flex;justify-items: flex-end;justify-content:flex-end" >
|
|
954
|
-
<app-uploadstatus *ngIf="!screenXs && enableUploadRecordings" class="ricontrols"
|
|
955
|
-
[value]="uploadProgress"
|
|
956
|
-
[status]="uploadStatus" [awaitNewUpload]="processingRecording"></app-uploadstatus>
|
|
957
|
-
<app-wakelockindicator *ngIf="!screenXs" class="ricontrols" [screenLocked]="screenLocked"></app-wakelockindicator>
|
|
958
|
-
<app-readystateindicator *ngIf="!screenXs" class="ricontrols"
|
|
959
|
-
[ready]="dataSaved && !isActive()"></app-readystateindicator>
|
|
960
|
-
</div>
|
|
961
|
-
</div>
|
|
962
|
-
`, isInline: true, styles: [":host{flex:2;background:#d3d3d3;display:flex;flex-direction:column;margin:0;padding:0;height:100%;min-height:0px;overflow:hidden}\n", ".ricontrols{padding:4px;box-sizing:border-box;height:100%}\n", ".dark{background:#a9a9a9}\n", ".controlpanel{display:flex;flex-direction:row;align-content:center;align-items:center;margin:0;padding:20px;min-height:min-content}\n", ".startstop{width:100%;flex:1 0 30%;align-items:center;text-align:center;align-content:center}\n", ".startstopscreenxs{width:100%;flex:1 0 100%;align-items:center;text-align:center;align-content:center}\n", ".bigbutton{vertical-align:middle;overflow:hidden;text-overflow:clip;white-space:nowrap;letter-spacing:normal;min-width:70px;min-height:50px;border-radius:20px}\n", ".bigbuttonicon{min-width:50px;min-height:50px;font-size:50px}\n", ".bigbuttontext{font-weight:bolder;font-size:14px;vertical-align:middle}\n", ".audioStatusDisplay{display:flex;flex-direction:row;height:100px;min-height:100px}\n", ".audioStatusDisplayXs{display:flex;flex-direction:column;height:125px;min-height:125px}\n"], dependencies: [{ kind: "directive", type: i6.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i7.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i8.MatButton, selector: " button[mat-button], button[mat-raised-button], button[mat-flat-button], button[mat-stroked-button] ", exportAs: ["matButton"] }, { kind: "component", type: i9.LevelBar, selector: "audio-levelbar", inputs: ["streamingMode", "displayLevelInfos", "state"] }, { kind: "component", type: i10.StatusDisplay, selector: "app-sprstatusdisplay", inputs: ["statusAlertType", "statusMsg", "statusWaiting"] }, { kind: "component", type: i11.RecordingItemControls, selector: "spr-recordingitemcontrols", inputs: ["audioSignalCollapsed", "enableDownload", "peakDbLvl", "agc", "disableAudioDetails", "audioLoaded", "playStartAction", "playStopAction", "displayLevelInfos"], outputs: ["onShowRecordingDetails", "onDownloadRecording"] }, { kind: "component", type: i10.UploadStatus, selector: "app-uploadstatus", inputs: ["value", "awaitNewUpload", "status"] }, { kind: "component", type: i10.WakeLockIndicator, selector: "app-wakelockindicator", inputs: ["screenLocked"] }, { kind: "component", type: i10.ReadyStateIndicator, selector: "app-readystateindicator", inputs: ["ready"] }, { kind: "component", type: i12.WarningBar, selector: "app-warningbar", inputs: ["warningText", "show"] }, { kind: "component", type: i13.RecorderCombiPane, selector: "app-recordercombipane", inputs: ["selectDisabled", "selectedRecordingFile", "audioSignalCollapsed", "displayAudioClip", "playStartAction", "playSelectionAction", "autoPlayOnSelectToggleAction", "playStopAction"], outputs: ["selectedRecordingFileChanged"] }] }); }
|
|
963
|
-
}
|
|
964
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: AudioRecorder, decorators: [{
|
|
965
|
-
type: Component,
|
|
966
|
-
args: [{ selector: 'app-audiorecorder', providers: [SessionService], template: `
|
|
967
|
-
<app-warningbar [show]="isTestSession()" warningText="Test recording only!"></app-warningbar>
|
|
968
|
-
<app-warningbar [show]="isDefaultAudioTestSession()"
|
|
969
|
-
warningText="This test uses default audio device! Regular sessions may require a particular audio device (microphone)!"></app-warningbar>
|
|
970
|
-
<app-recordercombipane (selectedRecordingFileChanged)="selectRecordingFile($event)"
|
|
971
|
-
[audioSignalCollapsed]="audioSignalCollapsed"
|
|
972
|
-
[selectedRecordingFile]="displayRecFile"
|
|
973
|
-
[selectDisabled]="isActive()"
|
|
974
|
-
[displayAudioClip]="displayAudioClip"
|
|
975
|
-
[playStartAction]="controlAudioPlayer?.startAction"
|
|
976
|
-
[playStopAction]="controlAudioPlayer?.stopAction"
|
|
977
|
-
[playSelectionAction]="controlAudioPlayer?.startSelectionAction"
|
|
978
|
-
[autoPlayOnSelectToggleAction]="controlAudioPlayer?.autoPlayOnSelectToggleAction"
|
|
979
|
-
></app-recordercombipane>
|
|
980
|
-
|
|
981
|
-
<div [class]="{audioStatusDisplay:!screenXs,audioStatusDisplayXs:screenXs}">
|
|
982
|
-
<audio-levelbar style="flex:1 0 1%" [streamingMode]="isRecording() || keepLiveLevel" [state]="liveLevelDisplayState"
|
|
983
|
-
[displayLevelInfos]="displayAudioClip?.levelInfos"></audio-levelbar>
|
|
984
|
-
<div style="display:flex;flex-direction: row">
|
|
985
|
-
<spr-recordingitemcontrols style="flex:10 0 1px"
|
|
986
|
-
[disableAudioDetails]="disableAudioDetails"
|
|
987
|
-
[audioLoaded]="audioLoaded"
|
|
988
|
-
[playStartAction]="controlAudioPlayer?.startAction"
|
|
989
|
-
[playStopAction]="controlAudioPlayer?.stopAction"
|
|
990
|
-
[peakDbLvl]="peakLevelInDb"
|
|
991
|
-
[agc]="this.ac?.agcStatus"
|
|
992
|
-
(onShowRecordingDetails)="audioSignalCollapsed=!audioSignalCollapsed">
|
|
993
|
-
</spr-recordingitemcontrols>
|
|
994
|
-
|
|
995
|
-
<app-uploadstatus *ngIf="screenXs && enableUploadRecordings" class="ricontrols dark" style="flex:0 0 0"
|
|
996
|
-
[value]="uploadProgress"
|
|
997
|
-
[status]="uploadStatus" [awaitNewUpload]="processingRecording"></app-uploadstatus>
|
|
998
|
-
<app-wakelockindicator *ngIf="screenXs" class="ricontrols dark" style="flex:0 0 0" [screenLocked]="screenLocked"></app-wakelockindicator>
|
|
999
|
-
<app-readystateindicator *ngIf="screenXs" class="ricontrols dark" style="flex:0 0 0"
|
|
1000
|
-
[ready]="dataSaved && !isActive()"></app-readystateindicator>
|
|
1001
|
-
</div>
|
|
1002
|
-
</div>
|
|
1003
|
-
<div #controlpanel class="controlpanel">
|
|
1004
|
-
<app-sprstatusdisplay *ngIf="!screenXs" style="flex:0 1 30%;" [statusMsg]="statusMsg" [statusAlertType]="statusAlertType"
|
|
1005
|
-
[statusWaiting]="statusWaiting"
|
|
1006
|
-
class="hidden-xs"></app-sprstatusdisplay>
|
|
1007
|
-
<div [class.startstop]="!screenXs" [class.startstopscreenxs]="screenXs">
|
|
1008
|
-
<div style="align-content: center">
|
|
1009
|
-
<button (click)="startStopPerform()" [disabled]="startDisabled() && stopDisabled()" mat-raised-button class="bigbutton">
|
|
1010
|
-
<mat-icon class="bigbuttonicon" [style.color]="startStopNextIconColor()">{{startStopNextIconName()}}</mat-icon>
|
|
1011
|
-
<span class="bigbuttontext">{{startStopNextName()}}</span>
|
|
1012
|
-
</button>
|
|
1013
|
-
</div>
|
|
1014
|
-
</div>
|
|
1015
|
-
<div style="flex:0 1 30%;display:flex;justify-items: flex-end;justify-content:flex-end" >
|
|
1016
|
-
<app-uploadstatus *ngIf="!screenXs && enableUploadRecordings" class="ricontrols"
|
|
1017
|
-
[value]="uploadProgress"
|
|
1018
|
-
[status]="uploadStatus" [awaitNewUpload]="processingRecording"></app-uploadstatus>
|
|
1019
|
-
<app-wakelockindicator *ngIf="!screenXs" class="ricontrols" [screenLocked]="screenLocked"></app-wakelockindicator>
|
|
1020
|
-
<app-readystateindicator *ngIf="!screenXs" class="ricontrols"
|
|
1021
|
-
[ready]="dataSaved && !isActive()"></app-readystateindicator>
|
|
1022
|
-
</div>
|
|
1023
|
-
</div>
|
|
1024
|
-
`, styles: [":host{flex:2;background:#d3d3d3;display:flex;flex-direction:column;margin:0;padding:0;height:100%;min-height:0px;overflow:hidden}\n", ".ricontrols{padding:4px;box-sizing:border-box;height:100%}\n", ".dark{background:#a9a9a9}\n", ".controlpanel{display:flex;flex-direction:row;align-content:center;align-items:center;margin:0;padding:20px;min-height:min-content}\n", ".startstop{width:100%;flex:1 0 30%;align-items:center;text-align:center;align-content:center}\n", ".startstopscreenxs{width:100%;flex:1 0 100%;align-items:center;text-align:center;align-content:center}\n", ".bigbutton{vertical-align:middle;overflow:hidden;text-overflow:clip;white-space:nowrap;letter-spacing:normal;min-width:70px;min-height:50px;border-radius:20px}\n", ".bigbuttonicon{min-width:50px;min-height:50px;font-size:50px}\n", ".bigbuttontext{font-weight:bolder;font-size:14px;vertical-align:middle}\n", ".audioStatusDisplay{display:flex;flex-direction:row;height:100px;min-height:100px}\n", ".audioStatusDisplayXs{display:flex;flex-direction:column;height:125px;min-height:125px}\n"] }]
|
|
1025
|
-
}], ctorParameters: () => [{ type: i1.BreakpointObserver }, { type: i0.ChangeDetectorRef }, { type: i0.Renderer2 }, { type: i2.MatDialog }, { type: i3.SessionService }, { type: i4.RecordingService }, { type: i5.SpeechRecorderUploader }, { type: i14.SpeechRecorderConfig, decorators: [{
|
|
1026
|
-
type: Inject,
|
|
1027
|
-
args: [SPEECHRECORDER_CONFIG]
|
|
1028
|
-
}] }], propDecorators: { projectName: [{
|
|
1029
|
-
type: Input
|
|
1030
|
-
}], recorderCombiPane: [{
|
|
1031
|
-
type: ViewChild,
|
|
1032
|
-
args: [RecorderCombiPane, { static: true }]
|
|
1033
|
-
}], liveLevelDisplay: [{
|
|
1034
|
-
type: ViewChild,
|
|
1035
|
-
args: [LevelBar, { static: true }]
|
|
1036
|
-
}], dataSaved: [{
|
|
1037
|
-
type: Input
|
|
1038
|
-
}], onKeyPress: [{
|
|
1039
|
-
type: HostListener,
|
|
1040
|
-
args: ['window:keypress', ['$event']]
|
|
1041
|
-
}], onKeyDown: [{
|
|
1042
|
-
type: HostListener,
|
|
1043
|
-
args: ['window:keydown', ['$event']]
|
|
1044
|
-
}] } });
|
|
1045
|
-
export class AudioRecorderComponent extends RecorderComponent {
|
|
1046
|
-
constructor(route, router, changeDetectorRef, sessionService, projectService, uploader) {
|
|
1047
|
-
super(uploader);
|
|
1048
|
-
this.route = route;
|
|
1049
|
-
this.router = router;
|
|
1050
|
-
this.changeDetectorRef = changeDetectorRef;
|
|
1051
|
-
this.sessionService = sessionService;
|
|
1052
|
-
this.projectService = projectService;
|
|
1053
|
-
this.uploader = uploader;
|
|
1054
|
-
}
|
|
1055
|
-
ngOnInit() {
|
|
1056
|
-
this.controlAudioPlayer = new AudioPlayer(this.ar);
|
|
1057
|
-
this.ar.controlAudioPlayer = this.controlAudioPlayer;
|
|
1058
|
-
//TODO Duplicate code in SpeechRecorderComponent
|
|
1059
|
-
window.addEventListener('beforeunload', (e) => {
|
|
1060
|
-
console.debug("Before page unload event");
|
|
1061
|
-
if (this.ready()) {
|
|
1062
|
-
return;
|
|
1063
|
-
}
|
|
1064
|
-
else {
|
|
1065
|
-
// all this attempts to customize the message do not work anymore (for security reasons)!!
|
|
1066
|
-
const message = "Please do not leave the page, until all recordings are uploaded!";
|
|
1067
|
-
alert(message);
|
|
1068
|
-
e = e || window.event;
|
|
1069
|
-
if (e) {
|
|
1070
|
-
e.returnValue = message;
|
|
1071
|
-
e.cancelBubble = true;
|
|
1072
|
-
if (e.stopPropagation) {
|
|
1073
|
-
e.stopPropagation();
|
|
1074
|
-
}
|
|
1075
|
-
if (e.preventDefault) {
|
|
1076
|
-
e.preventDefault();
|
|
1077
|
-
}
|
|
1078
|
-
}
|
|
1079
|
-
return message;
|
|
1080
|
-
}
|
|
1081
|
-
});
|
|
1082
|
-
}
|
|
1083
|
-
ngAfterViewInit() {
|
|
1084
|
-
// TODO call prepare !!
|
|
1085
|
-
this.uploader.listener = (ue) => {
|
|
1086
|
-
this.uploadUpdate(ue);
|
|
1087
|
-
};
|
|
1088
|
-
this.route.queryParams.subscribe((params) => {
|
|
1089
|
-
if (params['sessionId']) {
|
|
1090
|
-
this.fetchSession(params['sessionId']);
|
|
1091
|
-
}
|
|
1092
|
-
});
|
|
1093
|
-
this.route.params.subscribe((params) => {
|
|
1094
|
-
let routeParamsId = params['id'];
|
|
1095
|
-
if (routeParamsId) {
|
|
1096
|
-
this.fetchSession(routeParamsId);
|
|
1097
|
-
}
|
|
1098
|
-
});
|
|
1099
|
-
}
|
|
1100
|
-
ngOnDestroy() {
|
|
1101
|
-
//super.ngOnDestroy();
|
|
1102
|
-
}
|
|
1103
|
-
get screenLocked() {
|
|
1104
|
-
return this.ar.screenLocked;
|
|
1105
|
-
}
|
|
1106
|
-
fetchSession(sessionId) {
|
|
1107
|
-
let sessObs = this.sessionService.sessionObserver(sessionId);
|
|
1108
|
-
if (sessObs) {
|
|
1109
|
-
sessObs.subscribe({
|
|
1110
|
-
next: (sess) => {
|
|
1111
|
-
this.ar.statusAlertType = 'info';
|
|
1112
|
-
this.ar.statusMsg = 'Received session info.';
|
|
1113
|
-
this.ar.statusWaiting = false;
|
|
1114
|
-
this.session = sess;
|
|
1115
|
-
this.ar.session = sess;
|
|
1116
|
-
if (sess.project) {
|
|
1117
|
-
//console.debug("Session associated project: "+sess.project)
|
|
1118
|
-
this.projectService.projectObservable(sess.project).subscribe({
|
|
1119
|
-
next: (project) => {
|
|
1120
|
-
this.ar.project = project;
|
|
1121
|
-
this.ar.fetchRecordings(sess);
|
|
1122
|
-
}, error: (reason) => {
|
|
1123
|
-
this.ar.statusMsg = reason;
|
|
1124
|
-
this.ar.statusAlertType = 'error';
|
|
1125
|
-
this.ar.statusWaiting = false;
|
|
1126
|
-
console.error("Error fetching project config: " + reason);
|
|
1127
|
-
}
|
|
1128
|
-
});
|
|
1129
|
-
}
|
|
1130
|
-
else {
|
|
1131
|
-
console.info("Session has no associated project. Using default configuration.");
|
|
1132
|
-
}
|
|
1133
|
-
},
|
|
1134
|
-
error: (reason) => {
|
|
1135
|
-
this.ar.statusMsg = reason;
|
|
1136
|
-
this.ar.statusAlertType = 'error';
|
|
1137
|
-
this.ar.statusWaiting = false;
|
|
1138
|
-
console.error("Error fetching session " + reason);
|
|
1139
|
-
}
|
|
1140
|
-
});
|
|
1141
|
-
}
|
|
1142
|
-
}
|
|
1143
|
-
uploadUpdate(ue) {
|
|
1144
|
-
let upStatus = ue.status;
|
|
1145
|
-
this.dataSaved = (UploaderStatus.DONE === upStatus);
|
|
1146
|
-
let percentUpl = ue.percentDone();
|
|
1147
|
-
if (UploaderStatus.ERR === upStatus) {
|
|
1148
|
-
this.ar.uploadStatus = 'warn';
|
|
1149
|
-
}
|
|
1150
|
-
else {
|
|
1151
|
-
if (percentUpl < 50) {
|
|
1152
|
-
this.ar.uploadStatus = 'accent';
|
|
1153
|
-
}
|
|
1154
|
-
else {
|
|
1155
|
-
this.ar.uploadStatus = 'success';
|
|
1156
|
-
}
|
|
1157
|
-
this.ar.uploadProgress = percentUpl;
|
|
1158
|
-
}
|
|
1159
|
-
this.ar.updateWakeLock(this.dataSaved);
|
|
1160
|
-
this.changeDetectorRef.detectChanges();
|
|
1161
|
-
}
|
|
1162
|
-
ready() {
|
|
1163
|
-
return this.dataSaved && !this.ar.isActive();
|
|
1164
|
-
}
|
|
1165
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: AudioRecorderComponent, deps: [{ token: i15.ActivatedRoute }, { token: i15.Router }, { token: i0.ChangeDetectorRef }, { token: i3.SessionService }, { token: i16.ProjectService }, { token: i5.SpeechRecorderUploader }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
1166
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.12", type: AudioRecorderComponent, selector: "app-audiorecorder-comp", providers: [SessionService], viewQueries: [{ propertyName: "ar", first: true, predicate: AudioRecorder, descendants: true, static: true }], usesInheritance: true, ngImport: i0, template: `
|
|
1167
|
-
<app-audiorecorder [projectName]="_project?.name" [dataSaved]="dataSaved"></app-audiorecorder>
|
|
1168
|
-
`, isInline: true, styles: [":host{flex:2;display:flex;height:100%;flex-direction:column;min-height:0}\n"], dependencies: [{ kind: "component", type: AudioRecorder, selector: "app-audiorecorder", inputs: ["projectName", "dataSaved"] }] }); }
|
|
1169
|
-
}
|
|
1170
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.12", ngImport: i0, type: AudioRecorderComponent, decorators: [{
|
|
1171
|
-
type: Component,
|
|
1172
|
-
args: [{ selector: 'app-audiorecorder-comp', providers: [SessionService], template: `
|
|
1173
|
-
<app-audiorecorder [projectName]="_project?.name" [dataSaved]="dataSaved"></app-audiorecorder>
|
|
1174
|
-
`, styles: [":host{flex:2;display:flex;height:100%;flex-direction:column;min-height:0}\n"] }]
|
|
1175
|
-
}], ctorParameters: () => [{ type: i15.ActivatedRoute }, { type: i15.Router }, { type: i0.ChangeDetectorRef }, { type: i3.SessionService }, { type: i16.ProjectService }, { type: i5.SpeechRecorderUploader }], propDecorators: { ar: [{
|
|
1176
|
-
type: ViewChild,
|
|
1177
|
-
args: [AudioRecorder, { static: true }]
|
|
1178
|
-
}] } });
|
|
1179
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audiorecorder.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/session/audiorecorder.ts"],"names":[],"mappings":"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","sourcesContent":["import {AudioCapture, AudioCaptureListener} from '../../audio/capture/capture';\r\nimport {AudioPlayer, AudioPlayerEvent, EventType} from '../../audio/playback/player'\r\nimport {WavWriter} from '../../audio/impl/wavwriter'\r\nimport {RecordingFile, RecordingFileUtils} from '../recording'\r\nimport {\r\n  AfterViewInit,\r\n  ChangeDetectorRef,\r\n  Component,\r\n  HostListener,\r\n  Inject,\r\n  Input,\r\n  OnDestroy,\r\n  OnInit,\r\n  Renderer2,\r\n  ViewChild\r\n} from \"@angular/core\";\r\nimport {SessionService} from \"./session.service\";\r\nimport {MatDialog} from \"@angular/material/dialog\";\r\nimport {SpeechRecorderUploader} from \"../spruploader\";\r\nimport {SPEECHRECORDER_CONFIG, SpeechRecorderConfig} from \"../../spr.config\";\r\nimport {Session} from \"./session\";\r\nimport {AudioStorageFormatEncoding, AudioStorageType, Project, ProjectUtil} from \"../project/project\";\r\nimport {MessageDialog} from \"../../ui/message_dialog\";\r\nimport {RecordingService} from \"../recordings/recordings.service\";\r\nimport {AudioClip} from \"../../audio/persistor\";\r\n\r\nimport {Upload, UploaderStatus, UploaderStatusChangeEvent, UploadHolder} from \"../../net/uploader\";\r\nimport {ActivatedRoute, Params, Router} from \"@angular/router\";\r\nimport {ProjectService} from \"../project/project.service\";\r\nimport {LevelBar, State as LiveLevelState} from \"../../audio/ui/livelevel\";\r\nimport {RecorderCombiPane} from \"./recorder_combi_pane\";\r\nimport {BasicRecorder, ChunkAudioBufferReceiver, MAX_RECORDING_TIME_MS, RECFILE_API_CTX} from \"./basicrecorder\";\r\nimport {ReadyStateProvider, RecorderComponent} from \"../../recorder_component\";\r\nimport {Mode} from \"../../speechrecorderng.component\";\r\nimport {AudioBufferSource, AudioDataHolder, AudioSource} from \"../../audio/audio_data_holder\";\r\nimport {ArrayAudioBuffer} from \"../../audio/array_audio_buffer\";\r\nimport {NetAudioBuffer} from \"../../audio/net_audio_buffer\";\r\nimport {IndexedDbAudioBuffer} from \"../../audio/inddb_audio_buffer\";\r\nimport {BreakpointObserver} from \"@angular/cdk/layout\";\r\n\r\nexport const enum Status {\r\n  BLOCKED, IDLE,STARTING, RECORDING,  STOPPING_STOP, ERROR\r\n}\r\n\r\n\r\n@Component({\r\n  selector: 'app-audiorecorder',\r\n  providers: [SessionService],\r\n  template: `\r\n    <app-warningbar [show]=\"isTestSession()\" warningText=\"Test recording only!\"></app-warningbar>\r\n    <app-warningbar [show]=\"isDefaultAudioTestSession()\"\r\n                    warningText=\"This test uses default audio device! Regular sessions may require a particular audio device (microphone)!\"></app-warningbar>\r\n    <app-recordercombipane (selectedRecordingFileChanged)=\"selectRecordingFile($event)\"\r\n                           [audioSignalCollapsed]=\"audioSignalCollapsed\"\r\n                           [selectedRecordingFile]=\"displayRecFile\"\r\n                           [selectDisabled]=\"isActive()\"\r\n                           [displayAudioClip]=\"displayAudioClip\"\r\n                           [playStartAction]=\"controlAudioPlayer?.startAction\"\r\n                           [playStopAction]=\"controlAudioPlayer?.stopAction\"\r\n                           [playSelectionAction]=\"controlAudioPlayer?.startSelectionAction\"\r\n                           [autoPlayOnSelectToggleAction]=\"controlAudioPlayer?.autoPlayOnSelectToggleAction\"\r\n    ></app-recordercombipane>\r\n\r\n    <div [class]=\"{audioStatusDisplay:!screenXs,audioStatusDisplayXs:screenXs}\">\r\n      <audio-levelbar style=\"flex:1 0 1%\" [streamingMode]=\"isRecording() || keepLiveLevel\" [state]=\"liveLevelDisplayState\"\r\n                      [displayLevelInfos]=\"displayAudioClip?.levelInfos\"></audio-levelbar>\r\n      <div style=\"display:flex;flex-direction: row\">\r\n        <spr-recordingitemcontrols style=\"flex:10 0 1px\"\r\n                                   [disableAudioDetails]=\"disableAudioDetails\"\r\n                                   [audioLoaded]=\"audioLoaded\"\r\n                                   [playStartAction]=\"controlAudioPlayer?.startAction\"\r\n                                   [playStopAction]=\"controlAudioPlayer?.stopAction\"\r\n                                   [peakDbLvl]=\"peakLevelInDb\"\r\n                                   [agc]=\"this.ac?.agcStatus\"\r\n                                   (onShowRecordingDetails)=\"audioSignalCollapsed=!audioSignalCollapsed\">\r\n        </spr-recordingitemcontrols>\r\n\r\n        <app-uploadstatus *ngIf=\"screenXs && enableUploadRecordings\" class=\"ricontrols dark\" style=\"flex:0 0 0\"\r\n                          [value]=\"uploadProgress\"\r\n                          [status]=\"uploadStatus\" [awaitNewUpload]=\"processingRecording\"></app-uploadstatus>\r\n        <app-wakelockindicator *ngIf=\"screenXs\" class=\"ricontrols dark\" style=\"flex:0 0 0\" [screenLocked]=\"screenLocked\"></app-wakelockindicator>\r\n        <app-readystateindicator *ngIf=\"screenXs\" class=\"ricontrols dark\" style=\"flex:0 0 0\"\r\n                                 [ready]=\"dataSaved && !isActive()\"></app-readystateindicator>\r\n      </div>\r\n    </div>\r\n    <div #controlpanel class=\"controlpanel\">\r\n      <app-sprstatusdisplay *ngIf=\"!screenXs\" style=\"flex:0 1 30%;\" [statusMsg]=\"statusMsg\" [statusAlertType]=\"statusAlertType\"\r\n                            [statusWaiting]=\"statusWaiting\"\r\n                            class=\"hidden-xs\"></app-sprstatusdisplay>\r\n      <div [class.startstop]=\"!screenXs\" [class.startstopscreenxs]=\"screenXs\">\r\n        <div style=\"align-content: center\">\r\n          <button (click)=\"startStopPerform()\" [disabled]=\"startDisabled() && stopDisabled()\" mat-raised-button class=\"bigbutton\">\r\n            <mat-icon class=\"bigbuttonicon\" [style.color]=\"startStopNextIconColor()\">{{startStopNextIconName()}}</mat-icon>\r\n            <span class=\"bigbuttontext\">{{startStopNextName()}}</span>\r\n          </button>\r\n        </div>\r\n      </div>\r\n      <div style=\"flex:0 1 30%;display:flex;justify-items: flex-end;justify-content:flex-end\" >\r\n        <app-uploadstatus *ngIf=\"!screenXs && enableUploadRecordings\" class=\"ricontrols\"\r\n                          [value]=\"uploadProgress\"\r\n                          [status]=\"uploadStatus\" [awaitNewUpload]=\"processingRecording\"></app-uploadstatus>\r\n        <app-wakelockindicator  *ngIf=\"!screenXs\" class=\"ricontrols\" [screenLocked]=\"screenLocked\"></app-wakelockindicator>\r\n        <app-readystateindicator *ngIf=\"!screenXs\" class=\"ricontrols\"\r\n                                 [ready]=\"dataSaved && !isActive()\"></app-readystateindicator>\r\n      </div>\r\n    </div>\r\n  `,\r\n  styles: [`:host {\r\n    flex: 2;\r\n    background: lightgrey;\r\n    display: flex; /* Vertical flex container: Bottom transport panel, above prompting panel */\r\n    flex-direction: column;\r\n    margin: 0;\r\n    padding: 0;\r\n    height: 100%;\r\n    min-height: 0px;\r\n      /* Prevents horizontal scroll bar on swipe right */\r\n      overflow: hidden;\r\n  }`,`.ricontrols {\r\n        padding: 4px;\r\n        box-sizing: border-box;\r\n        height: 100%;\r\n    }`,`.dark {\r\n    background: darkgray;\r\n  }`,`.controlpanel {\r\n    display:flex;\r\n    flex-direction: row;\r\n    align-content: center;\r\n    align-items: center;\r\n    margin: 0;\r\n    padding: 20px;\r\n    min-height: min-content; /* important */\r\n  }`,`.startstop {\r\n    width: 100%;\r\n    flex:1 0 30%;\r\n    align-items: center;\r\n    text-align: center;\r\n    align-content: center;\r\n  }`,`.startstopscreenxs {\r\n    width: 100%;\r\n    flex:1 0 100%;\r\n    align-items: center;\r\n    text-align: center;\r\n    align-content: center;\r\n  }`,`.bigbutton {\r\n    vertical-align: middle;\r\n    overflow: hidden;\r\n    text-overflow: clip;\r\n    white-space: nowrap;\r\n    letter-spacing: normal;\r\n    min-width: 70px;\r\n    min-height: 50px;\r\n    border-radius: 20px;\r\n  }`,`.bigbuttonicon {\r\n    min-width: 50px;\r\n    min-height: 50px;\r\n    font-size: 50px;\r\n  }`,`.bigbuttontext {\r\n      font-weight: bolder;\r\n      font-size: 14px;\r\n      vertical-align: middle;\r\n  }\r\n  `,`.audioStatusDisplay{\r\n    display:flex;\r\n    flex-direction: row;\r\n    height:100px;\r\n    min-height: 100px;\r\n  }`,`.audioStatusDisplayXs{\r\n    display:flex;\r\n    flex-direction: column;\r\n    height:125px;\r\n    min-height: 125px;\r\n  }`\r\n   ]\r\n})\r\nexport class AudioRecorder extends BasicRecorder implements OnInit,AfterViewInit,OnDestroy, AudioCaptureListener,ReadyStateProvider,ChunkAudioBufferReceiver {\r\n\r\n  _project:Project|undefined| null=null;\r\n  @Input() projectName:string|undefined|null=null;\r\n  enableUploadRecordings: boolean = true;\r\n  enableDownloadRecordings: boolean = false;\r\n  status: Status = Status.BLOCKED;\r\n\r\n  @ViewChild(RecorderCombiPane, { static: true }) recorderCombiPane!: RecorderCombiPane;\r\n  @ViewChild(LevelBar, { static: true }) liveLevelDisplay!: LevelBar;\r\n\r\n  @Input() dataSaved=true\r\n\r\n\r\n\r\n\r\n  startStopNextButtonName!:string;\r\n  startStopNextButtonIconName!:string;\r\n\r\n  audio: any;\r\n\r\n  private _displayRecFile: RecordingFile | null=null;\r\n  private displayRecFileVersion: number=0;\r\n\r\n\r\n  constructor(protected bpo:BreakpointObserver,changeDetectorRef: ChangeDetectorRef,\r\n              private renderer: Renderer2,\r\n              dialog: MatDialog,\r\n              sessionService:SessionService,\r\n              private recFileService:RecordingService,\r\n              protected uploader: SpeechRecorderUploader,\r\n              @Inject(SPEECHRECORDER_CONFIG) config?: SpeechRecorderConfig) {\r\n    super(bpo,changeDetectorRef,dialog,sessionService,uploader,config);\r\n\r\n    //super(injector);\r\n    this.status = Status.IDLE;\r\n\r\n    this.audio = document.getElementById('audio');\r\n\r\n    if (this.config && this.config.enableUploadRecordings != null) {\r\n      this.enableUploadRecordings = this.config.enableUploadRecordings;\r\n    }\r\n    if (this.config && this.config.enableDownloadRecordings != null) {\r\n      this.enableDownloadRecordings = this.config.enableDownloadRecordings;\r\n    }\r\n    //this.init();\r\n  }\r\n\r\n\r\n  ngAfterViewInit() {\r\n\r\n    this.streamLevelMeasure.levelListener = this.liveLevelDisplay;\r\n    this.streamLevelMeasure.peakLevelListener=(peakLvlInDb)=>{\r\n      this.peakLevelInDb=peakLvlInDb;\r\n      this.changeDetectorRef.detectChanges();\r\n    }\r\n    //let wakeLockSupp=('wakeLock' in navigator);\r\n    //alert('Wake lock API supported: '+wakeLockSupp);\r\n  }\r\n\r\n  ready():boolean{\r\n    return this.dataSaved && !this.isActive()\r\n  }\r\n    ngOnDestroy() {\r\n      this.disableWakeLockCond();\r\n       this.destroyed=true;\r\n       // TODO stop capture /playback\r\n    }\r\n\r\n  ngOnInit() {\r\n    this.transportActions.startAction.disabled = true;\r\n    this.transportActions.stopAction.disabled = true;\r\n    this.transportActions.nextAction.disabled = true;\r\n    this.transportActions.pauseAction.disabled = true;\r\n    this.playStartAction.disabled = true;\r\n\r\n      this.ac = new AudioCapture();\r\n      if (this.ac) {\r\n        this.transportActions.startAction.onAction = () => this.startItem();\r\n        this.ac.listener = this;\r\n        this.configureStreamCaptureStream();\r\n\r\n        // Don't list the devices here. If we do not have audio permissions we only get anonymized devices without labels.\r\n        //this.ac.listDevices();\r\n      } else {\r\n        this.transportActions.startAction.disabled = true;\r\n        let errMsg = 'Browser does not support Media/Audio API!';\r\n        this.statusMsg = 'ERROR: ' + errMsg;\r\n        this.statusAlertType = 'error';\r\n        this.dialog.open(MessageDialog, {\r\n          data: {\r\n            type: 'error',\r\n            title: 'Error',\r\n            msg: errMsg,\r\n            advice: 'Please use a supported browser.',\r\n          }\r\n        });\r\n        return;\r\n      }\r\n      this.transportActions.stopAction.onAction = () => this.stopItem();\r\n      this.transportActions.nextAction.onAction = () => this.stopItem();\r\n      //this.transportActions.pauseAction.onAction = () => this.pauseItem();\r\n\r\n      this.playStartAction.onAction = () => this.controlAudioPlayer?.start();\r\n\r\n    this.uploader.listener = (ue) => {\r\n      this.uploadUpdate(ue);\r\n    }\r\n  }\r\n\r\n  @HostListener('window:keypress', ['$event'])\r\n  onKeyPress(ke: KeyboardEvent) {\r\n    if (ke.key == ' ') {\r\n      //this.transportActions.startAction.perform();\r\n      //this.transportActions.nextAction.perform();\r\n    }\r\n  }\r\n\r\n  @HostListener('window:keydown', ['$event'])\r\n  onKeyDown(ke: KeyboardEvent) {\r\n    if (ke.key == ' ') {\r\n      if(!this.transportActions.startAction.disabled){\r\n        this.transportActions.startAction.perform();\r\n      }else if(!this.transportActions.stopAction.disabled) {\r\n        this.transportActions.stopAction.perform();\r\n      }\r\n    }\r\n    if (ke.key == 'p') {\r\n      this.transportActions.pauseAction.perform();\r\n    }\r\n    if (ke.key == 'Escape') {\r\n      if (!this.audioSignalCollapsed) {\r\n        this.audioSignalCollapsed = true;\r\n      }\r\n      this.transportActions.stopAction.perform();\r\n      this.transportActions.pauseAction.perform();\r\n    }\r\n\r\n    if (ke.key == 'MediaPlayPause') {\r\n      this.playStartAction.perform();\r\n    }\r\n    if (ke.key === 'ArrowRight') {\r\n      this.transportActions.fwdAction.perform();\r\n    }\r\n    if (ke.key === 'ArrowLeft') {\r\n      this.transportActions.bwdAction.perform();\r\n    }\r\n  }\r\n\r\n  isTestSession():boolean {\r\n    return ((this._session!=null) && (this._session.type === 'TEST' || this._session.type==='TEST_DEF_A' || this._session.type === 'SINUS_TEST'));\r\n  }\r\n\r\n  isDefaultAudioTestSession():boolean {\r\n    return ((this._session!=null) && (this._session.type==='TEST_DEF_A'));\r\n  }\r\n\r\n  isDefaultAudioTestSessionOverwriteingProjectRequirements():boolean {\r\n    return ((this._session!=null) && (this._session.type==='TEST_DEF_A') && (this.audioDevices!=null) && this.audioDevices.length>0)\r\n  }\r\n\r\n\r\n\r\n  fetchRecordings(sess:Session){\r\n    this.statusAlertType='info';\r\n    this.statusMsg = 'Fetching infos of recordings...';\r\n    this.statusWaiting=true;\r\n    //let prNm:string|null=null;\r\n    if(this.project) {\r\n      let rfsObs = this.recFileService.recordingFileList(this.project.name, sess.sessionId);\r\n      rfsObs.subscribe({next:(rfs: Array<RecordingFile>) => {\r\n        this.statusAlertType = 'info';\r\n        this.statusMsg = 'Received infos of recordings.';\r\n        this.statusWaiting = false;\r\n        if (rfs) {\r\n          if (rfs instanceof Array) {\r\n            rfs.forEach((rf) => {\r\n              // the list comes from the server, asssuem all recording files as server persisted\r\n              rf.serverPersisted=true;\r\n              if(rf.startedDate){\r\n                rf._startedAsDateObj=new Date(rf.startedDate);\r\n              }\r\n              if(rf.date){\r\n                rf._dateAsDateObj=new Date(rf.date);\r\n              }\r\n              this.recorderCombiPane.addRecFile(rf);\r\n            })\r\n          } else {\r\n            console.error('Expected type array for list of already recorded files ')\r\n          }\r\n\r\n        } else {\r\n          //console.debug(\"Recording file list: \" + rfs);\r\n        }\r\n      }, error:(err) => {\r\n        // Failed fetching existing, but we start the session anyway\r\n        this.start()\r\n      }, complete:() => {\r\n        // Normal start\r\n        this.start()\r\n      }\r\n    })\r\n    }else{\r\n      // No project def -> error\r\n      this.statusAlertType = 'error';\r\n      this.statusMsg = 'No project definiton.';\r\n      this.statusWaiting = false;\r\n      console.error(this.statusMsg);\r\n    }\r\n  }\r\n\r\n\r\n  set project(project: Project|undefined|null) {\r\n    this._project = project;\r\n    let chCnt = ProjectUtil.DEFAULT_AUDIO_CHANNEL_COUNT;\r\n\r\n    if (project) {\r\n      console.info(\"Project name: \" + project.name)\r\n      if (project.recordingDeviceWakeLock === true) {\r\n        this.wakeLock = true;\r\n      }\r\n      this.audioDevices = project.audioDevices;\r\n      chCnt = ProjectUtil.audioChannelCount(project);\r\n      console.info(\"Project requested recording channel count: \" + chCnt);\r\n      this.autoGainControlConfigs = project.autoGainControlConfigs;\r\n      if(project.allowEchoCancellation!==undefined) {\r\n        this.allowEchoCancellation = project.allowEchoCancellation;\r\n      }\r\n      if (project.chunkedRecording === true) {\r\n        this.uploadChunkSizeSeconds = BasicRecorder.DEFAULT_CHUNK_SIZE_SECONDS;\r\n      } else {\r\n        this.uploadChunkSizeSeconds = null;\r\n      }\r\n      if (project.clientAudioStorageType) {\r\n        this.clientAudioStorageType = project.clientAudioStorageType;\r\n      }\r\n    } else {\r\n      console.error(\"Empty project configuration!\")\r\n    }\r\n    this.channelCount = chCnt;\r\n  }\r\n\r\n\r\n  get project():Project|undefined|null{\r\n    return this._project;\r\n  }\r\n\r\n  selectRecordingFile(rf:RecordingFile){\r\n    this.liveLevelDisplayState=LiveLevelState.READY;\r\n    this.keepLiveLevel=false;\r\n    this.displayRecFile=rf;\r\n  }\r\n\r\n  uploadUpdate(ue: UploaderStatusChangeEvent) {\r\n    let upStatus = ue.status;\r\n    this.dataSaved = (UploaderStatus.DONE === upStatus);\r\n    let percentUpl = ue.percentDone();\r\n    if (UploaderStatus.ERR === upStatus) {\r\n      this.uploadStatus = 'warn'\r\n    } else {\r\n      if (percentUpl < 50) {\r\n        this.uploadStatus = 'accent'\r\n      } else {\r\n        this.uploadStatus = 'success'\r\n      }\r\n      this.uploadProgress = percentUpl;\r\n    }\r\n\r\n    this.changeDetectorRef.detectChanges()\r\n  }\r\n\r\n  set controlAudioPlayer(controlAudioPlayer: AudioPlayer|null) {\r\n    this._controlAudioPlayer=controlAudioPlayer;\r\n    if (this._controlAudioPlayer) {\r\n      this._controlAudioPlayer.listener = this;\r\n    }\r\n  }\r\n\r\n  get controlAudioPlayer(): AudioPlayer|null {\r\n    return this._controlAudioPlayer;\r\n  }\r\n\r\n  update(e: AudioPlayerEvent) {\r\n    if (e.type == EventType.STARTED) {\r\n      this.playStartAction.disabled = true;\r\n      this.updateTimerId = window.setInterval(() => {\r\n        //this.audioSignal.playFramePosition = this.ap.playPositionFrames;\r\n      }, 50);\r\n    } else if (e.type == EventType.STOPPED || e.type == EventType.ENDED) {\r\n      window.clearInterval(this.updateTimerId);\r\n      //console.debug(\"Enable play start action (by player events stopped or ended): \"+(!(this.displayRecFile)));\r\n      this.playStartAction.disabled = (!(this.displayRecFile));\r\n\r\n    }\r\n  }\r\n\r\n  startDisabled() {\r\n    return !this.transportActions || this.readonly || this.transportActions.startAction.disabled\r\n  }\r\n\r\n  stopDisabled() {\r\n    return !this.transportActions || this.transportActions.stopAction.disabled\r\n  }\r\n\r\n  startStopNextName():string{\r\n    if(!this.startDisabled()){\r\n      this.startStopNextButtonName=\"Start\"\r\n    }else if(!this.stopDisabled()) {\r\n      this.startStopNextButtonName = \"Stop\"\r\n    }\r\n    return this.startStopNextButtonName;\r\n  }\r\n  startStopNextIconName():string{\r\n    if(!this.startDisabled()){\r\n      this.startStopNextButtonIconName=\"fiber_manual_record\"\r\n    }else if(!this.stopDisabled()){\r\n      this.startStopNextButtonIconName=\"stop\"\r\n    }\r\n    return this.startStopNextButtonIconName\r\n  }\r\n  startStopNextIconColor():string{\r\n    if(!this.startDisabled()){\r\n      return \"red\"\r\n    }else if(!this.stopDisabled()){\r\n      return \"yellow\"\r\n    }else{\r\n      return \"grey\";\r\n    }\r\n  }\r\n\r\n  startStopPerform(){\r\n    if(!this.startDisabled()){\r\n      this.transportActions.startAction.perform();\r\n    }else if(!this.stopDisabled()){\r\n      this.transportActions.stopAction.perform();\r\n    }\r\n  }\r\n\r\n  startItem() {\r\n    this.status=Status.STARTING;\r\n    super.startItem();\r\n    if (this.readonly) {\r\n      this.status=Status.IDLE;\r\n      return\r\n    }\r\n    this.transportActions.fwdAction.disabled = true\r\n    this.transportActions.fwdNextAction.disabled = true\r\n    this.transportActions.bwdAction.disabled = true\r\n    this.displayRecFile = null;\r\n    this.displayRecFileVersion = 0;\r\n    this.displayAudioClip = null;\r\n    this.liveLevelDisplay.reset(true);\r\n    this.liveLevelDisplayState=LiveLevelState.READY;\r\n    this.showRecording();\r\n\r\n    this.startCapture();\r\n  }\r\n\r\n\r\n  downloadRecording() {\r\n    if (this.displayRecFile) {\r\n      let ab: AudioDataHolder | null = this.displayRecFile.audioDataHolder;\r\n      const ww = new WavWriter(this.project?.mediaStorageFormat?.audioEncoding===AudioStorageFormatEncoding.PCM_FLOAT,this.project?.mediaStorageFormat?.audioPCMsampleSizeInBits);\r\n      let as=ab?.audioSource;\r\n      if(as instanceof AudioBufferSource) {\r\n          ww.writeAsync(as.audioBuffer, (wavFile) => {\r\n            let blob = new Blob([wavFile], {type: 'audio/wav'});\r\n            let rfUrl = URL.createObjectURL(blob);\r\n\r\n            let dataDnlLnk = this.renderer.createElement('a');\r\n            //dataDnlLnk.name = 'Recording';\r\n            dataDnlLnk.href = rfUrl;\r\n\r\n            this.renderer.appendChild(document.body, dataDnlLnk);\r\n\r\n            // download property not yet in TS def\r\n            if (this.displayRecFile) {\r\n              let fn = this.displayRecFile.filenameString();\r\n              fn += '_' + this.displayRecFileVersion;\r\n              fn += '.wav';\r\n              dataDnlLnk.download = fn;\r\n              dataDnlLnk.click();\r\n            }\r\n            this.renderer.removeChild(document.body, dataDnlLnk);\r\n          });\r\n      }\r\n    }\r\n  }\r\n\r\n  set displayRecFile(displayRecFile: RecordingFile | null) {\r\n    //this.audioLoaded=false;\r\n    this._displayRecFile = displayRecFile;\r\n    if (this._displayRecFile) {\r\n      let adh: AudioDataHolder| null = this._displayRecFile.audioDataHolder;\r\n      if(adh) {\r\n        this.displayAudioClip = new AudioClip(adh);\r\n        //this.audioLoaded=true;\r\n        //console.debug(\" set recording file: display audio clip set\");\r\n        if(this._controlAudioPlayer) {\r\n          this._controlAudioPlayer.audioClip = this.displayAudioClip;\r\n        }\r\n        this.showRecording();\r\n      }else {\r\n        // clear for now ...\r\n        this.displayAudioClip = null;\r\n        //console.debug(\"set recording file: display audio clip null\");\r\n        if(this._controlAudioPlayer) {\r\n          this._controlAudioPlayer.audioClip = null;\r\n        }\r\n\r\n        if (this._controlAudioPlayer && this._session) {\r\n          //... and try to fetch from server\r\n          this.liveLevelDisplayState=LiveLevelState.LOADING;\r\n          const rf=this._displayRecFile;\r\n\r\n          let audioDownloadType=this._clientAudioStorageType;\r\n          if(AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED===this._clientAudioStorageType || AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED===this._clientAudioStorageType) {\r\n            // Default is network mode\r\n            audioDownloadType=AudioStorageType.NET_CHUNKED;\r\n            if (rf.channels && rf.frames) {\r\n              const samples = rf.channels * rf.frames;\r\n              if (samples <= this._maxAutoNetMemStoreSamples) {\r\n                // But if audio file is small, load in continuous resp. chunked mode\r\n                if(AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED===this._clientAudioStorageType){\r\n                  audioDownloadType=AudioStorageType.MEM_ENTIRE;\r\n                }else if(AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED===this._clientAudioStorageType) {\r\n                  audioDownloadType = AudioStorageType.MEM_CHUNKED;\r\n                }\r\n              }\r\n            }\r\n          }\r\n\r\n          if(AudioStorageType.DB_CHUNKED===this._clientAudioStorageType){\r\n            // Fetch chunked indexed db audio buffer\r\n            let nextIab: IndexedDbAudioBuffer | null = null;\r\n            if(!this._persistentAudioStorageTarget){\r\n              throw Error('Error: Persistent storage target not set.');\r\n            }else {\r\n              //console.debug(\"Fetch audio and store to indexed db...\");\r\n              this.audioFetchSubscription = this.recFileService.fetchRecordingFileIndDbAudioBuffer(this._persistentAudioStorageTarget, this._session.project, rf).subscribe({\r\n                next: (iab) => {\r\n                  //console.debug(\"Sessionmanager: Received inddb audio buffer: \"+iab);\r\n                  nextIab = iab;\r\n                },\r\n                complete: () => {\r\n                  this.liveLevelDisplayState = LiveLevelState.READY;\r\n                  let fabDh = null;\r\n                  if (nextIab) {\r\n                    if (rf ) {\r\n                      fabDh = new AudioDataHolder(nextIab);\r\n                      //this.audioLoaded=true;\r\n                      this.recorderCombiPane.setRecFileAudioData(rf, fabDh);\r\n                    }\r\n                  } else {\r\n                    // Should actually be handled by the error resolver\r\n                    this.statusMsg = 'Recording file could not be loaded.'\r\n                    this.statusAlertType = 'error'\r\n                  }\r\n                  if (fabDh) {\r\n                    // this.displayAudioClip could have been changed meanwhile, but the recorder unsubcribes before changing the item. Therefore, there should be no risk to set to wrong item\r\n                    this.displayAudioClip = new AudioClip(fabDh);\r\n                  }\r\n                  if(this._controlAudioPlayer) {\r\n                    this._controlAudioPlayer.audioClip = this.displayAudioClip\r\n                  }\r\n                  this.showRecording();\r\n                },\r\n                error: err => {\r\n                  console.error(\"Could not load recording file from server: \" + err);\r\n                  this.liveLevelDisplayState = LiveLevelState.READY;\r\n                  this.statusMsg = 'Recording file could not be loaded: ' + err;\r\n                  this.statusAlertType = 'error';\r\n                  this.changeDetectorRef.detectChanges();\r\n                }\r\n              });\r\n            }\r\n          }else if(AudioStorageType.NET_CHUNKED===audioDownloadType){\r\n            // Fetch chunked audio buffer from network\r\n            let nextNetAb: NetAudioBuffer | null = null;\r\n\r\n            //console.debug(\"Fetch chunked audio from network\");\r\n            this.audioFetchSubscription = this.recFileService.fetchRecordingFileNetAudioBuffer( this._session.project, rf).subscribe({\r\n              next: (netAb) => {\r\n                //console.debug(\"Sessionmanager: Received net audio buffer: \"+netAb);\r\n                nextNetAb = netAb;\r\n              },\r\n              complete: () => {\r\n                this.liveLevelDisplayState = LiveLevelState.READY;\r\n                let fabDh = null;\r\n                if (nextNetAb) {\r\n                  if (rf) {\r\n                    fabDh = new AudioDataHolder(nextNetAb);\r\n\r\n                    this.recorderCombiPane.setRecFileAudioData(rf, fabDh);\r\n                  }\r\n                } else {\r\n                  // Should actually be handled by the error resolver\r\n                  this.statusMsg = 'Recording file could not be loaded.'\r\n                  this.statusAlertType = 'error'\r\n                }\r\n                if (fabDh) {\r\n                  // this.displayAudioClip could have been changed meanwhile, but the recorder unsubcribes before changing the item. Therefore, there should be no risk to set to wrong item\r\n                  //console.debug(\"set displayRecFile(): fetch net ab complete, set displayAudioClip.\")\r\n                  this.displayAudioClip = new AudioClip(fabDh);\r\n                  // fabDh.onReady=()=>{\r\n                  //   this.audioLoaded=true;\r\n                  // }\r\n                }\r\n                if(this._controlAudioPlayer) {\r\n                  this._controlAudioPlayer.audioClip = this.displayAudioClip;\r\n                }\r\n                this.showRecording();\r\n              },\r\n              error: err => {\r\n                console.error(\"Could not load recording file from server: \" + err);\r\n                this.liveLevelDisplayState = LiveLevelState.READY;\r\n                this.statusMsg = 'Recording file could not be loaded: ' + err;\r\n                this.statusAlertType = 'error';\r\n                this.changeDetectorRef.detectChanges();\r\n              }\r\n            });\r\n\r\n          }else if(AudioStorageType.MEM_CHUNKED===audioDownloadType){\r\n            // Fetch chunked array audio buffer\r\n            let nextAab: ArrayAudioBuffer | null = null;\r\n            //console.debug(\"Fetch audio and store to (chunked) array buffer...\");\r\n            this.audioFetchSubscription = this.recFileService.fetchRecordingFileArrayAudioBuffer( this._session.project, rf).subscribe({\r\n              next: (aab) => {\r\n                nextAab = aab;\r\n              },\r\n              complete: () => {\r\n                this.liveLevelDisplayState = LiveLevelState.READY;\r\n                let fabDh = null;\r\n                if (nextAab) {\r\n                  if (rf ) {\r\n                    fabDh = new AudioDataHolder(nextAab);\r\n                    this.recorderCombiPane.setRecFileAudioData(rf, fabDh);\r\n                  }\r\n                } else {\r\n                  // Should actually be handled by the error resolver\r\n                  this.statusMsg = 'Recording file could not be loaded.'\r\n                  this.statusAlertType = 'error'\r\n                }\r\n                if (fabDh) {\r\n                  // this.displayAudioClip could have been changed meanwhile, but the recorder unsubcribes before changing the item. Therefore, there should be no risk to set to wrong item\r\n                  this.displayAudioClip = new AudioClip(fabDh);\r\n                  //this.audioLoaded=true;\r\n                }\r\n                if(this._controlAudioPlayer) {\r\n                  this._controlAudioPlayer.audioClip = this.displayAudioClip;\r\n                }\r\n                this.showRecording();\r\n              },\r\n              error: err => {\r\n                console.error(\"Could not load recording file from server: \" + err);\r\n                this.liveLevelDisplayState = LiveLevelState.READY;\r\n                this.statusMsg = 'Recording file could not be loaded: ' + err;\r\n                this.statusAlertType = 'error';\r\n              }\r\n            });\r\n\r\n          } else {\r\n            this.audioFetchSubscription = this.recFileService.fetchRecordingFileAudioBuffer(this._session.project, rf).subscribe({\r\n              next: ab => {\r\n                this.liveLevelDisplayState = LiveLevelState.READY;\r\n                let fabDh = null;\r\n                if (ab) {\r\n                  let abSrc=new AudioBufferSource(ab);\r\n                  if (rf) {\r\n                    fabDh = new AudioDataHolder(abSrc);\r\n                    this.recorderCombiPane.setRecFileAudioData(rf, fabDh);\r\n                  }\r\n                } else {\r\n                  console.error('Recording file could not be loaded.');\r\n                  this.statusMsg = 'Recording file could not be loaded.';\r\n                  this.statusAlertType = 'error';\r\n                }\r\n                if (fabDh) {\r\n                  // this.displayAudioClip could have been changed meanwhile, but the recorder unsubcribes before changing the item. Therefore, there should be no risk to set to wrong item\r\n                  this.displayAudioClip = new AudioClip(fabDh);\r\n                  //this.audioLoaded=true;\r\n                  //console.debug(\"set recording file: display audio clip from fetched audio buffer\");\r\n                }\r\n                if(this._controlAudioPlayer) {\r\n                  this._controlAudioPlayer.audioClip = this.displayAudioClip;\r\n                }\r\n                this.showRecording();\r\n              }, error: err => {\r\n                console.error(\"Could not load recording file from server: \" + err);\r\n                this.liveLevelDisplayState = LiveLevelState.READY;\r\n                this.statusMsg = 'Recording file could not be loaded: ' + err;\r\n                this.statusAlertType = 'error';\r\n              }\r\n            })\r\n          }\r\n        }else{\r\n          this.statusMsg = 'Recording file could not be decoded. Audio context unavailable.';\r\n          this.statusAlertType = 'error';\r\n        }\r\n      }\r\n\r\n    } else {\r\n      //console.debug(\"recording file null\");\r\n      this.displayAudioClip = null;\r\n      if(this._controlAudioPlayer) {\r\n        this._controlAudioPlayer.audioClip = null;\r\n      }\r\n    }\r\n    this.showRecording();\r\n  }\r\n\r\n  get displayRecFile(): RecordingFile | null {\r\n    return this._displayRecFile;\r\n  }\r\n\r\n  updateStartActionDisableState(){\r\n    this.transportActions.startAction.disabled=!(this.ac);\r\n  }\r\n\r\n  start() {\r\n    super.start();\r\n    this.recorderCombiPane.selectTop();\r\n    this.enableNavigation();\r\n    this.updateStartActionDisableState();\r\n\r\n  }\r\n\r\n  isRecording(): boolean {\r\n    return (this.status === Status.RECORDING || this.status===Status.STOPPING_STOP);\r\n  }\r\n\r\n  isActive(): boolean{\r\n    return (!(this.status === Status.BLOCKED || this.status=== Status.IDLE || this.status===Status.ERROR) || this.processingRecording || this.sessionService.uploadCount>0)\r\n  }\r\n\r\n\r\n  updateWakeLock(dataSaved:boolean=this.dataSaved){\r\n    //console.debug(\"Update wake lock: dataSaved: \"+dataSaved+\", not active: \"+! this.isActive())\r\n    if(dataSaved && ! this.isActive()){\r\n      this.disableWakeLockCond();\r\n    }\r\n  }\r\n\r\n  private updateNavigationActions(){\r\n\r\n    this.transportActions.fwdAction.disabled = this.navigationDisabled;\r\n    this.transportActions.bwdAction.disabled = this.navigationDisabled;\r\n  }\r\n\r\n  enableStartUserGesture() {\r\n    super.enableStartUserGesture();\r\n    this.transportActions.startAction.disabled=!(this.ac);\r\n  }\r\n\r\n  enableNavigation(){\r\n    this.navigationDisabled=false;\r\n    this.updateNavigationActions();\r\n  }\r\n\r\n  started() {\r\n    this.status = Status.RECORDING;\r\n    super.started();\r\n    this.statusAlertType = 'info';\r\n    this.statusMsg = 'Recording...';\r\n\r\n    let sessId: string | number = 0;\r\n    if(this._session){\r\n      sessId=this._session.sessionId;\r\n    }\r\n\r\n    if(this.rfUuid) {\r\n      let rf = new RecordingFile(this.rfUuid, sessId, null);\r\n      rf._startedAsDateObj = this.startedDate;\r\n      if (rf._startedAsDateObj) {\r\n        rf.startedDate = rf._startedAsDateObj.toString();\r\n      }\r\n      this._recordingFile = rf;\r\n    }\r\n    let maxRecordingTimeMs = MAX_RECORDING_TIME_MS;\r\n    this.maxRecTimerId = window.setTimeout(() => {\r\n      this.stopRecordingMaxRec()\r\n    }, maxRecordingTimeMs);\r\n    this.maxRecTimerRunning = true;\r\n    this.transportActions.stopAction.disabled = false;\r\n  }\r\n\r\n  stopItem() {\r\n\r\n    this.transportActions.stopAction.disabled = true;\r\n    this.transportActions.nextAction.disabled = true;\r\n    this.status = Status.STOPPING_STOP;\r\n    this.stopRecording();\r\n\r\n  }\r\n\r\n  stopRecording() {\r\n    if (this.maxRecTimerRunning && this.maxRecTimerId!=null) {\r\n      window.clearTimeout(this.maxRecTimerId);\r\n      this.maxRecTimerRunning = false;\r\n    }\r\n    if(this.ac) {\r\n      this.ac.stop();\r\n    }\r\n  }\r\n\r\n  stopRecordingMaxRec(){\r\n\r\n    this.maxRecTimerRunning = false;\r\n    this.status = Status.STOPPING_STOP;\r\n    if(this.ac) {\r\n      this.ac.stop();\r\n    }\r\n  }\r\n\r\n  stopped() {\r\n    this.updateStartActionDisableState()\r\n    this.transportActions.stopAction.disabled = true;\r\n    this.transportActions.nextAction.disabled = true;\r\n    this.transportActions.pauseAction.disabled = true;\r\n    this.statusAlertType = 'info';\r\n    this.statusMsg = 'Recorded.';\r\n\r\n    let ab:AudioBuffer|null=null;\r\n\r\n    if(this.ac) {\r\n      let adh:AudioDataHolder|null=null;\r\n      let as:AudioSource|null=null;\r\n      if(this.ac) {\r\n        if (AudioStorageType.NET_CHUNKED === this.ac.audioStorageType) {\r\n          this.playStartAction.disabled = true;\r\n          //this.audioLoaded=false;\r\n          this.keepLiveLevel=true;\r\n          let rUUID:string|null=null;\r\n            let burl:string|null=null;\r\n            if(this._session) {\r\n              if (this._recordingFile) {\r\n\r\n                let rf = this._recordingFile;\r\n                rf.frames=this.ac.framesRecorded;\r\n                rUUID=rf.uuid;\r\n                //console.debug(\"stopped(): Set frames: \"+rf.frames+\" on rfId: \"+this.displayRecFile?.recordingFileId);\r\n                burl = this.recFileService.audioFileUrl(this._session?.project, rf);\r\n              } else if (this.session?.project) {\r\n                if(this.ac.recUUID) {\r\n                  rUUID=this.ac.recUUID;\r\n                  burl = this.recFileService.audioFileUrlByUUID(this.session.project, this.session.sessionId, rUUID);\r\n                }\r\n              }else{\r\n                console.error(\"Could not create net audio buffer.\");\r\n              }\r\n              if (burl) {\r\n                const sr = this.ac.currentSampleRate;\r\n                const chFl=sr*RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;\r\n                //console.debug(\"stopped(): rfID: \"+this._recordingFile?.recordingFileId+\", net ab url: \" + burl+\", frames: \"+this.ac.framesRecorded+\", sample rate: \"+sr);\r\n                let netAs = new NetAudioBuffer(this.recFileService, burl, this.ac.channelCount, sr, chFl, this.ac.framesRecorded, rUUID, chFl);\r\n                as=netAs;\r\n                if(this.uploadSet){\r\n                  this.uploadSet.onDone=(uploadSet)=>{\r\n                    //console.debug(\"upload set on done: Call ready provider.ready\");\r\n                    netAs.ready();\r\n                  }\r\n                }\r\n\r\n              }\r\n            }\r\n\r\n        }else if (AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this.ac.audioStorageType || AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this.ac.audioStorageType) {\r\n          if (AudioStorageType.MEM_ENTIRE_AUTO_NET_CHUNKED === this.ac.audioStorageType){\r\n            const acAb=this.ac.audioBuffer();\r\n            if(acAb) {\r\n              as = new AudioBufferSource(acAb);\r\n            }\r\n          }\r\n          if(AudioStorageType.MEM_CHUNKED_AUTO_NET_CHUNKED === this.ac.audioStorageType){\r\n            as = this.ac.audioBufferArray();\r\n          }\r\n          if(!as){\r\n            this.playStartAction.disabled = true;\r\n            this.keepLiveLevel=true;\r\n            let rUUID:string|null=null;\r\n            let burl:string|null=null;\r\n            if(this._session) {\r\n              if (this._recordingFile) {\r\n                let rf = this._recordingFile;\r\n                rf.frames=this.ac.framesRecorded;\r\n                rUUID=rf.uuid;\r\n                //console.debug(\"stopped(): Set frames: \"+rf.frames+\" on rfId: \"+this.displayRecFile?.recordingFileId);\r\n                burl = this.recFileService.audioFileUrl(this._session?.project, rf);\r\n              } else if (this.session?.project) {\r\n                if(this.ac.recUUID) {\r\n                  rUUID=this.ac.recUUID;\r\n                  burl = this.recFileService.audioFileUrlByUUID(this.session.project, this.session.sessionId, rUUID);\r\n                }\r\n              }else{\r\n                console.error(\"Could not create net audio buffer.\");\r\n              }\r\n              if (burl) {\r\n                const sr = this.ac.currentSampleRate;\r\n                const chFl=sr*RecordingService.DEFAULT_CHUNKED_DOWNLOAD_SECONDS;\r\n                //console.debug(\"stopped(): rfID: \"+this._recordingFile?.recordingFileId+\", net ab url: \" + burl+\", frames: \"+this.ac.framesRecorded+\", sample rate: \"+sr);\r\n                let netAs = new NetAudioBuffer(this.recFileService, burl, this.ac.channelCount, sr, chFl, this.ac.framesRecorded, rUUID, chFl);\r\n                as = netAs;\r\n                if (this.uploadSet) {\r\n                  this.uploadSet.onDone = (uploadSet) => {\r\n                    //console.debug(\"upload set on done: Call ready provider.ready\");\r\n                    netAs.ready();\r\n                  }\r\n                }\r\n              }\r\n            }\r\n          }\r\n        } else if (AudioStorageType.DB_CHUNKED === this.ac.audioStorageType) {\r\n          as = this.ac.inddbAudioBufferArray();\r\n        } else if (AudioStorageType.MEM_CHUNKED === this.ac.audioStorageType) {\r\n          as = this.ac.audioBufferArray();\r\n        } else {\r\n          ab = this.ac.audioBuffer();\r\n          if(ab) {\r\n            as = new AudioBufferSource(ab);\r\n          }\r\n        }\r\n        if (as) {\r\n          adh = new AudioDataHolder(as);\r\n        }\r\n      }\r\n\r\n      let sessId: string | number = 0;\r\n      if(this._session){\r\n        sessId=this._session.sessionId;\r\n      }\r\n\r\n      if(this._recordingFile) {\r\n        //this._recordingFile.samplerate=this.ac.currentSampleRate;\r\n        // Use an own reference since the writing of the wave file is asynchronous and this._recordingFile might already contain the next recording\r\n        let rf = this._recordingFile;\r\n        RecordingFileUtils.setAudioData(rf,adh);\r\n        this.recorderCombiPane.addRecFile(rf);\r\n\r\n        // Upload if upload enabled and not in chunked upload mode\r\n        if (this.enableUploadRecordings && this._uploadChunkSizeSeconds===null && AudioStorageType.MEM_ENTIRE===this._clientAudioStorageType && rf != null && ab != null) {\r\n          let apiEndPoint = '';\r\n\r\n          if (this.config && this.config.apiEndPoint) {\r\n            apiEndPoint = this.config.apiEndPoint;\r\n          }\r\n          if (apiEndPoint !== '') {\r\n            apiEndPoint = apiEndPoint + '/'\r\n          }\r\n\r\n          let sessionsUrl = apiEndPoint + SessionService.SESSION_API_CTX;\r\n          let recUrl: string = sessionsUrl + '/' + rf.session + '/' + RECFILE_API_CTX + '/' + rf.uuid;\r\n\r\n          // convert asynchronously to 16-bit integer PCM\r\n          // TODO could we avoid conversion to save CPU resources and transfer float PCM directly?\r\n          // TODO duplicate conversion for manual download\r\n\r\n          this.processingRecording = true\r\n          const ww = new WavWriter(this.project?.mediaStorageFormat?.audioEncoding===AudioStorageFormatEncoding.PCM_FLOAT,this.project?.mediaStorageFormat?.audioPCMsampleSizeInBits);\r\n          ww.writeAsync(ab, (wavFile) => {\r\n            this.postRecordingMultipart(wavFile,recUrl,rf);\r\n            this.processingRecording = false;\r\n            this.updateWakeLock();\r\n            this.changeDetectorRef.detectChanges();\r\n          });\r\n        }\r\n      }\r\n    }\r\n    this.displayRecFile = this._recordingFile;\r\n    this.status = Status.IDLE;\r\n    this.navigationDisabled = false;\r\n    this.updateNavigationActions();\r\n    this.updateWakeLock();\r\n    this.changeDetectorRef.detectChanges();\r\n  }\r\n\r\n  error(msg='An unknown error occured during recording.',advice:string='Please retry.') {\r\n    this.status=Status.ERROR;\r\n    super.error(msg,advice);\r\n    this.updateNavigationActions();\r\n    this.updateStartActionDisableState();\r\n  }\r\n\r\n  postRecordingMultipart(wavFile: Uint8Array,recUrl: string,rf:RecordingFile) {\r\n    let wavBlob = new Blob([wavFile], {type: 'audio/wav'});\r\n\r\n    let fd=new FormData();\r\n    if(rf.uuid) {\r\n      fd.set('uuid', rf.uuid);\r\n    }\r\n    if(rf.session!==null) {\r\n      fd.set('sessionId', rf.session.toString());\r\n    }\r\n    if(rf._startedAsDateObj){\r\n      fd.set('startedDate',rf._startedAsDateObj.toJSON());\r\n    }\r\n    fd.set('audio',wavBlob);\r\n    let ul = new Upload(fd, recUrl,rf);\r\n    this.uploader.queueUpload(ul);\r\n  }\r\n\r\n  postChunkAudioBuffer(audioBuffer: AudioBuffer, chunkIdx: number): void {\r\n    this.processingRecording = true;\r\n    const ww = new WavWriter(this.project?.mediaStorageFormat?.audioEncoding===AudioStorageFormatEncoding.PCM_FLOAT,this.project?.mediaStorageFormat?.audioPCMsampleSizeInBits);\r\n    let sessionsUrl = this.sessionsBaseUrl();\r\n    let recUrl: string = sessionsUrl + '/' + this.session?.sessionId + '/' + RECFILE_API_CTX + '/' + this.rfUuid+'/'+chunkIdx;\r\n    let rf=this._recordingFile;\r\n\r\n    // The upload holder is required to add the upload now to the upload set. The real upload is created async in postrecording and the upload set is already complete at that time.\r\n    let ulh=new UploadHolder();\r\n    if(this.uploadSet){\r\n      this.uploadSet.add(ulh);\r\n    }\r\n    ww.writeAsync(audioBuffer, (wavFile) => {\r\n      this.postRecording(wavFile, recUrl,rf,ulh);\r\n      this.processingRecording = false;\r\n    });\r\n  }\r\n\r\n  stop() {\r\n    if(this.ac) {\r\n      this.ac.close();\r\n    }\r\n  }\r\n\r\n  private updateControlPlaybackPosition() {\r\n    if(this._controlAudioPlayer){\r\n      const ppFrames=this._controlAudioPlayer.playPositionFrames;\r\n      if (ppFrames!==null) {\r\n        this.recorderCombiPane.audioDisplay.playFramePosition = ppFrames;\r\n        this.liveLevelDisplay.playFramePosition = ppFrames;\r\n      }\r\n    }\r\n  }\r\n\r\n  audioPlayerUpdate(e: AudioPlayerEvent) {\r\n    if (EventType.READY === e.type) {\r\n\r\n    } else if (EventType.STARTED === e.type) {\r\n      //this.status = 'Playback...';\r\n      this.updateTimerId = window.setInterval(() => this.updateControlPlaybackPosition(), 50);\r\n\r\n    } else if (EventType.ENDED === e.type) {\r\n      //.status='Ready.';\r\n      window.clearInterval(this.updateTimerId);\r\n\r\n    }\r\n    if(!this.destroyed) {\r\n        this.changeDetectorRef.detectChanges();\r\n    }\r\n  }\r\n}\r\n\r\n@Component({\r\n  selector: 'app-audiorecorder-comp',\r\n  providers: [SessionService],\r\n  template: `\r\n    <app-audiorecorder [projectName]=\"_project?.name\" [dataSaved]=\"dataSaved\"></app-audiorecorder>\r\n  `,\r\n  styles: [`:host{\r\n    flex: 2;\r\n    display: flex;\r\n      height: 100%;\r\n    flex-direction: column;\r\n    min-height:0;\r\n\r\n  }`]\r\n\r\n})\r\nexport class AudioRecorderComponent extends RecorderComponent  implements OnInit,OnDestroy,AfterViewInit,ReadyStateProvider {\r\n\r\n  mode!:Mode;\r\n  controlAudioPlayer!:AudioPlayer;\r\n  audio:any;\r\n\r\n  _project:Project|undefined;\r\n  sessionId!: string;\r\n  session!:Session;\r\n\r\n  @ViewChild(AudioRecorder, { static: true }) ar!:AudioRecorder;\r\n\r\n  constructor(private route: ActivatedRoute,\r\n              private router: Router,\r\n              private changeDetectorRef: ChangeDetectorRef,\r\n              private sessionService:SessionService,\r\n              private projectService:ProjectService,\r\n              protected uploader:SpeechRecorderUploader\r\n  ) {\r\n    super(uploader);\r\n  }\r\n\r\n  ngOnInit() {\r\n\r\n    this.controlAudioPlayer = new AudioPlayer(this.ar);\r\n\r\n    this.ar.controlAudioPlayer=this.controlAudioPlayer;\r\n\r\n    //TODO Duplicate code in SpeechRecorderComponent\r\n    window.addEventListener('beforeunload', (e) => {\r\n      console.debug(\"Before page unload event\");\r\n\r\n      if (this.ready()) {\r\n        return;\r\n      } else {\r\n        // all this attempts to customize the message do not work anymore (for security reasons)!!\r\n        const message = \"Please do not leave the page, until all recordings are uploaded!\";\r\n        alert(message);\r\n        e = e || window.event;\r\n\r\n        if (e) {\r\n          e.returnValue = message;\r\n          e.cancelBubble = true;\r\n          if (e.stopPropagation) {\r\n            e.stopPropagation();\r\n          }\r\n          if (e.preventDefault) {\r\n            e.preventDefault();\r\n          }\r\n        }\r\n\r\n        return message;\r\n      }\r\n    });\r\n  }\r\n\r\n  ngAfterViewInit() {\r\n\r\n    // TODO call prepare !!\r\n\r\n    this.uploader.listener = (ue) => {\r\n      this.uploadUpdate(ue);\r\n    }\r\n    this.route.queryParams.subscribe((params: Params) => {\r\n      if (params['sessionId']) {\r\n        this.fetchSession(params['sessionId']);\r\n      }\r\n    });\r\n    this.route.params.subscribe((params: Params) => {\r\n      let routeParamsId = params['id'];\r\n      if (routeParamsId) {\r\n        this.fetchSession(routeParamsId);\r\n      }\r\n    });\r\n  }\r\n\r\n  ngOnDestroy() {\r\n    //super.ngOnDestroy();\r\n  }\r\n\r\n  get screenLocked():boolean{\r\n    return  this.ar.screenLocked;\r\n  }\r\n\r\n  fetchSession(sessionId:string){\r\n\r\n    let sessObs= this.sessionService.sessionObserver(sessionId);\r\n\r\n    if(sessObs) {\r\n      sessObs.subscribe({\r\n        next:(sess) => {\r\n          this.ar.statusAlertType='info';\r\n          this.ar.statusMsg = 'Received session info.';\r\n          this.ar.statusWaiting=false;\r\n          this.session=sess;\r\n          this.ar.session=sess;\r\n          if (sess.project) {\r\n            //console.debug(\"Session associated project: \"+sess.project)\r\n            this.projectService.projectObservable(sess.project).subscribe({\r\n              next:(project)=>{\r\n              this.ar.project=project;\r\n              this.ar.fetchRecordings(sess);\r\n            },error:(reason) =>{\r\n              this.ar.statusMsg=reason;\r\n              this.ar.statusAlertType='error';\r\n              this.ar.statusWaiting=false;\r\n              console.error(\"Error fetching project config: \"+reason)\r\n            }});\r\n\r\n          } else {\r\n            console.info(\"Session has no associated project. Using default configuration.\")\r\n          }\r\n        },\r\n        error:(reason) => {\r\n          this.ar.statusMsg = reason;\r\n          this.ar.statusAlertType = 'error';\r\n          this.ar.statusWaiting=false;\r\n          console.error(\"Error fetching session \" + reason)\r\n        }});\r\n    }\r\n  }\r\n\r\n  uploadUpdate(ue: UploaderStatusChangeEvent) {\r\n    let upStatus = ue.status;\r\n    this.dataSaved = (UploaderStatus.DONE === upStatus);\r\n    let percentUpl = ue.percentDone();\r\n    if (UploaderStatus.ERR === upStatus) {\r\n      this.ar.uploadStatus = 'warn'\r\n    } else {\r\n      if (percentUpl < 50) {\r\n        this.ar.uploadStatus = 'accent'\r\n      } else {\r\n        this.ar.uploadStatus = 'success'\r\n      }\r\n      this.ar.uploadProgress = percentUpl;\r\n    }\r\n    this.ar.updateWakeLock(this.dataSaved);\r\n    this.changeDetectorRef.detectChanges()\r\n  }\r\n\r\n\r\n  ready():boolean{\r\n    return this.dataSaved && !this.ar.isActive()\r\n  }\r\n\r\n}\r\n"]}
|