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.
Files changed (99) hide show
  1. package/README.md +1 -1
  2. package/fesm2022/speechrecorderng.mjs +204 -201
  3. package/fesm2022/speechrecorderng.mjs.map +1 -1
  4. package/lib/audio/ui/audio_canvas_layer_comp.d.ts +2 -2
  5. package/lib/spr.module.version.d.ts +1 -1
  6. package/lib/ui/canvas_layer_comp.d.ts +1 -1
  7. package/lib/ui/responsive_component.d.ts +1 -1
  8. package/package.json +13 -15
  9. package/esm2022/lib/action/action.mjs +0 -73
  10. package/esm2022/lib/audio/array_audio_buffer.mjs +0 -164
  11. package/esm2022/lib/audio/array_audio_buffer_input_stream.mjs +0 -86
  12. package/esm2022/lib/audio/array_audio_buffer_random_access_stream.mjs +0 -16
  13. package/esm2022/lib/audio/audio_data_holder.mjs +0 -264
  14. package/esm2022/lib/audio/audio_display.mjs +0 -118
  15. package/esm2022/lib/audio/audio_player.mjs +0 -238
  16. package/esm2022/lib/audio/capture/capture.mjs +0 -855
  17. package/esm2022/lib/audio/context.mjs +0 -79
  18. package/esm2022/lib/audio/dsp/level_measure.mjs +0 -517
  19. package/esm2022/lib/audio/format.mjs +0 -20
  20. package/esm2022/lib/audio/impl/wavformat.mjs +0 -7
  21. package/esm2022/lib/audio/impl/wavreader.mjs +0 -144
  22. package/esm2022/lib/audio/impl/wavwriter.mjs +0 -191
  23. package/esm2022/lib/audio/inddb_audio_buffer.mjs +0 -508
  24. package/esm2022/lib/audio/io/stream.mjs +0 -59
  25. package/esm2022/lib/audio/net_audio_buffer.mjs +0 -293
  26. package/esm2022/lib/audio/persistor.mjs +0 -81
  27. package/esm2022/lib/audio/playback/array_audio_buffer_source_node.mjs +0 -126
  28. package/esm2022/lib/audio/playback/audio_source_node.mjs +0 -18
  29. package/esm2022/lib/audio/playback/audio_source_worklet_module_loader.mjs +0 -167
  30. package/esm2022/lib/audio/playback/inddb_audio_buffer_source_node.mjs +0 -167
  31. package/esm2022/lib/audio/playback/net_audio_buffer_source_node.mjs +0 -218
  32. package/esm2022/lib/audio/playback/player.mjs +0 -402
  33. package/esm2022/lib/audio/ui/audio_canvas_layer_comp.mjs +0 -347
  34. package/esm2022/lib/audio/ui/audio_display_control.mjs +0 -150
  35. package/esm2022/lib/audio/ui/audio_display_scroll_pane.mjs +0 -146
  36. package/esm2022/lib/audio/ui/audiosignal.mjs +0 -533
  37. package/esm2022/lib/audio/ui/common.mjs +0 -19
  38. package/esm2022/lib/audio/ui/container.mjs +0 -414
  39. package/esm2022/lib/audio/ui/livelevel.mjs +0 -361
  40. package/esm2022/lib/audio/ui/scroll_pane_horizontal.mjs +0 -11
  41. package/esm2022/lib/audio/ui/sonagram.mjs +0 -900
  42. package/esm2022/lib/db/inddb.mjs +0 -120
  43. package/esm2022/lib/dsp/utils.mjs +0 -48
  44. package/esm2022/lib/environment/environment.defaults.mjs +0 -9
  45. package/esm2022/lib/io/BinaryReader.mjs +0 -93
  46. package/esm2022/lib/io/BinaryWriter.mjs +0 -80
  47. package/esm2022/lib/io/stream.mjs +0 -259
  48. package/esm2022/lib/math/2d/geometry.mjs +0 -28
  49. package/esm2022/lib/math/complex.mjs +0 -58
  50. package/esm2022/lib/math/dft.mjs +0 -196
  51. package/esm2022/lib/media/utils.mjs +0 -14
  52. package/esm2022/lib/net/uploader.mjs +0 -367
  53. package/esm2022/lib/recorder_component.mjs +0 -65
  54. package/esm2022/lib/speechrecorder/project/project.mjs +0 -54
  55. package/esm2022/lib/speechrecorder/project/project.service.mjs +0 -64
  56. package/esm2022/lib/speechrecorder/recording.mjs +0 -124
  57. package/esm2022/lib/speechrecorder/recordings/basic_recording.service.mjs +0 -221
  58. package/esm2022/lib/speechrecorder/recordings/recordings.service.mjs +0 -1014
  59. package/esm2022/lib/speechrecorder/script/script.mjs +0 -114
  60. package/esm2022/lib/speechrecorder/script/script.service.mjs +0 -47
  61. package/esm2022/lib/speechrecorder/session/audiorecorder.mjs +0 -1179
  62. package/esm2022/lib/speechrecorder/session/basicrecorder.mjs +0 -676
  63. package/esm2022/lib/speechrecorder/session/controlpanel.mjs +0 -416
  64. package/esm2022/lib/speechrecorder/session/item.mjs +0 -30
  65. package/esm2022/lib/speechrecorder/session/progress.mjs +0 -135
  66. package/esm2022/lib/speechrecorder/session/prompting.mjs +0 -639
  67. package/esm2022/lib/speechrecorder/session/recorder_combi_pane.mjs +0 -88
  68. package/esm2022/lib/speechrecorder/session/recording_file_cache.mjs +0 -195
  69. package/esm2022/lib/speechrecorder/session/recording_list.mjs +0 -188
  70. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-meta.component.mjs +0 -128
  71. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-navi.component.mjs +0 -114
  72. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-u-i.component.mjs +0 -146
  73. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file-view.component.mjs +0 -424
  74. package/esm2022/lib/speechrecorder/session/recordingfile/recording-file.mjs +0 -68
  75. package/esm2022/lib/speechrecorder/session/recordingfile/recordingfile-service.mjs +0 -288
  76. package/esm2022/lib/speechrecorder/session/session.mjs +0 -2
  77. package/esm2022/lib/speechrecorder/session/session.service.mjs +0 -69
  78. package/esm2022/lib/speechrecorder/session/session_finished_dialog.mjs +0 -46
  79. package/esm2022/lib/speechrecorder/session/sessionmanager.mjs +0 -1385
  80. package/esm2022/lib/speechrecorder/session/warning_bar.mjs +0 -33
  81. package/esm2022/lib/speechrecorder/spruploader.mjs +0 -22
  82. package/esm2022/lib/speechrecorder/startstopsignal/startstopsignal.mjs +0 -2
  83. package/esm2022/lib/speechrecorder/startstopsignal/ui/simpletrafficlight.mjs +0 -57
  84. package/esm2022/lib/speechrecorderng.component.mjs +0 -392
  85. package/esm2022/lib/speechrecorderng.module.mjs +0 -97
  86. package/esm2022/lib/spr.config.mjs +0 -27
  87. package/esm2022/lib/spr.module.version.mjs +0 -2
  88. package/esm2022/lib/ui/canvas_layer_comp.mjs +0 -38
  89. package/esm2022/lib/ui/intersection-observer.directive.mjs +0 -32
  90. package/esm2022/lib/ui/message_dialog.mjs +0 -51
  91. package/esm2022/lib/ui/recordingitem_display.mjs +0 -253
  92. package/esm2022/lib/ui/responsive_component.mjs +0 -24
  93. package/esm2022/lib/utils/scrollIntoViewToBottom.mjs +0 -23
  94. package/esm2022/lib/utils/ua-parser.mjs +0 -190
  95. package/esm2022/lib/utils/utils.mjs +0 -132
  96. package/esm2022/lib/utils/wake_lock.mjs +0 -114
  97. package/esm2022/lib/utils/wake_lock_media.mjs +0 -2
  98. package/esm2022/public-api.mjs +0 -35
  99. 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":"AAAA,OAAO,EAAC,YAAY,EAAuB,MAAM,6BAA6B,CAAC;AAC/E,OAAO,EAAC,WAAW,EAAoB,SAAS,EAAC,MAAM,6BAA6B,CAAA;AACpF,OAAO,EAAC,SAAS,EAAC,MAAM,4BAA4B,CAAA;AACpD,OAAO,EAAC,aAAa,EAAE,kBAAkB,EAAC,MAAM,cAAc,CAAA;AAC9D,OAAO,EAGL,SAAS,EACT,YAAY,EACZ,MAAM,EACN,KAAK,EAIL,SAAS,EACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAC,cAAc,EAAC,MAAM,mBAAmB,CAAC;AAGjD,OAAO,EAAC,qBAAqB,EAAuB,MAAM,kBAAkB,CAAC;AAE7E,OAAO,EAAC,0BAA0B,EAAE,gBAAgB,EAAW,WAAW,EAAC,MAAM,oBAAoB,CAAC;AACtG,OAAO,EAAC,aAAa,EAAC,MAAM,yBAAyB,CAAC;AACtD,OAAO,EAAC,gBAAgB,EAAC,MAAM,kCAAkC,CAAC;AAClE,OAAO,EAAC,SAAS,EAAC,MAAM,uBAAuB,CAAC;AAEhD,OAAO,EAAC,MAAM,EAAE,cAAc,EAA6B,YAAY,EAAC,MAAM,oBAAoB,CAAC;AAGnG,OAAO,EAAC,QAAQ,EAAE,KAAK,IAAI,cAAc,EAAC,MAAM,0BAA0B,CAAC;AAC3E,OAAO,EAAC,iBAAiB,EAAC,MAAM,uBAAuB,CAAC;AACxD,OAAO,EAAC,aAAa,EAA4B,qBAAqB,EAAE,eAAe,EAAC,MAAM,iBAAiB,CAAC;AAChH,OAAO,EAAqB,iBAAiB,EAAC,MAAM,0BAA0B,CAAC;AAE/E,OAAO,EAAC,iBAAiB,EAAE,eAAe,EAAc,MAAM,+BAA+B,CAAC;AAE9F,OAAO,EAAC,cAAc,EAAC,MAAM,8BAA8B,CAAC;;;;;;;;;;;;;;;;;;AA2I5D,MAAM,OAAO,aAAc,SAAQ,aAAa;IAyB9C,YAAsB,GAAsB,EAAC,iBAAoC,EAC7D,QAAmB,EAC3B,MAAiB,EACjB,cAA6B,EACrB,cAA+B,EAC7B,QAAgC,EACX,MAA6B;QACtE,KAAK,CAAC,GAAG,EAAC,iBAAiB,EAAC,MAAM,EAAC,cAAc,EAAC,QAAQ,EAAC,MAAM,CAAC,CAAC;QAP/C,QAAG,GAAH,GAAG,CAAmB;QACxB,aAAQ,GAAR,QAAQ,CAAW;QAGnB,mBAAc,GAAd,cAAc,CAAiB;QAC7B,aAAQ,GAAR,QAAQ,CAAwB;QA5BtD,aAAQ,GAAyB,IAAI,CAAC;QAC7B,gBAAW,GAAuB,IAAI,CAAC;QAChD,2BAAsB,GAAY,IAAI,CAAC;QACvC,6BAAwB,GAAY,KAAK,CAAC;QAC1C,WAAM,0BAA0B;QAKvB,cAAS,GAAC,IAAI,CAAA;QAUf,oBAAe,GAAuB,IAAI,CAAC;QAC3C,0BAAqB,GAAS,CAAC,CAAC;QAYtC,kBAAkB;QAClB,IAAI,CAAC,MAAM,sBAAc,CAAC;QAE1B,IAAI,CAAC,KAAK,GAAG,QAAQ,CAAC,cAAc,CAAC,OAAO,CAAC,CAAC;QAE9C,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,CAAC,sBAAsB,IAAI,IAAI,EAAE,CAAC;YAC9D,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,MAAM,CAAC,sBAAsB,CAAC;QACnE,CAAC;QACD,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,CAAC,wBAAwB,IAAI,IAAI,EAAE,CAAC;YAChE,IAAI,CAAC,wBAAwB,GAAG,IAAI,CAAC,MAAM,CAAC,wBAAwB,CAAC;QACvE,CAAC;QACD,cAAc;IAChB,CAAC;IAGD,eAAe;QAEb,IAAI,CAAC,kBAAkB,CAAC,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC;QAC9D,IAAI,CAAC,kBAAkB,CAAC,iBAAiB,GAAC,CAAC,WAAW,EAAC,EAAE;YACvD,IAAI,CAAC,aAAa,GAAC,WAAW,CAAC;YAC/B,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;QACzC,CAAC,CAAA;QACD,6CAA6C;QAC7C,kDAAkD;IACpD,CAAC;IAED,KAAK;QACH,OAAO,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAA;IAC3C,CAAC;IACC,WAAW;QACT,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC1B,IAAI,CAAC,SAAS,GAAC,IAAI,CAAC;QACpB,8BAA8B;IACjC,CAAC;IAEH,QAAQ;QACN,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,GAAG,IAAI,CAAC;QAClD,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,IAAI,CAAC;QACjD,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,IAAI,CAAC;QACjD,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,GAAG,IAAI,CAAC;QAClD,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,IAAI,CAAC;QAEnC,IAAI,CAAC,EAAE,GAAG,IAAI,YAAY,EAAE,CAAC;QAC7B,IAAI,IAAI,CAAC,EAAE,EAAE,CAAC;YACZ,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC;YACpE,IAAI,CAAC,EAAE,CAAC,QAAQ,GAAG,IAAI,CAAC;YACxB,IAAI,CAAC,4BAA4B,EAAE,CAAC;YAEpC,kHAAkH;YAClH,wBAAwB;QAC1B,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,GAAG,IAAI,CAAC;YAClD,IAAI,MAAM,GAAG,2CAA2C,CAAC;YACzD,IAAI,CAAC,SAAS,GAAG,SAAS,GAAG,MAAM,CAAC;YACpC,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;YAC/B,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,EAAE;gBAC9B,IAAI,EAAE;oBACJ,IAAI,EAAE,OAAO;oBACb,KAAK,EAAE,OAAO;oBACd,GAAG,EAAE,MAAM;oBACX,MAAM,EAAE,iCAAiC;iBAC1C;aACF,CAAC,CAAC;YACH,OAAO;QACT,CAAC;QACD,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;QAClE,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE,CAAC;QAClE,sEAAsE;QAEtE,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,EAAE,KAAK,EAAE,CAAC;QAEzE,IAAI,CAAC,QAAQ,CAAC,QAAQ,GAAG,CAAC,EAAE,EAAE,EAAE;YAC9B,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC;QACxB,CAAC,CAAA;IACH,CAAC;IAGD,UAAU,CAAC,EAAiB;QAC1B,IAAI,EAAE,CAAC,GAAG,IAAI,GAAG,EAAE,CAAC;YAClB,8CAA8C;YAC9C,6CAA6C;QAC/C,CAAC;IACH,CAAC;IAGD,SAAS,CAAC,EAAiB;QACzB,IAAI,EAAE,CAAC,GAAG,IAAI,GAAG,EAAE,CAAC;YAClB,IAAG,CAAC,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,EAAC,CAAC;gBAC9C,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;YAC9C,CAAC;iBAAK,IAAG,CAAC,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;gBACpD,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;YAC7C,CAAC;QACH,CAAC;QACD,IAAI,EAAE,CAAC,GAAG,IAAI,GAAG,EAAE,CAAC;YAClB,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;QAC9C,CAAC;QACD,IAAI,EAAE,CAAC,GAAG,IAAI,QAAQ,EAAE,CAAC;YACvB,IAAI,CAAC,IAAI,CAAC,oBAAoB,EAAE,CAAC;gBAC/B,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;YACnC,CAAC;YACD,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;YAC3C,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;QAC9C,CAAC;QAED,IAAI,EAAE,CAAC,GAAG,IAAI,gBAAgB,EAAE,CAAC;YAC/B,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC;QACjC,CAAC;QACD,IAAI,EAAE,CAAC,GAAG,KAAK,YAAY,EAAE,CAAC;YAC5B,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC;QAC5C,CAAC;QACD,IAAI,EAAE,CAAC,GAAG,KAAK,WAAW,EAAE,CAAC;YAC3B,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC;QAC5C,CAAC;IACH,CAAC;IAED,aAAa;QACX,OAAO,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAE,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,KAAK,MAAM,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,KAAG,YAAY,IAAI,IAAI,CAAC,QAAQ,CAAC,IAAI,KAAK,YAAY,CAAC,CAAC,CAAC;IAChJ,CAAC;IAED,yBAAyB;QACvB,OAAO,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAE,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,KAAG,YAAY,CAAC,CAAC,CAAC;IACxE,CAAC;IAED,wDAAwD;QACtD,OAAO,CAAC,CAAC,IAAI,CAAC,QAAQ,IAAE,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,KAAG,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,YAAY,IAAE,IAAI,CAAC,IAAI,IAAI,CAAC,YAAY,CAAC,MAAM,GAAC,CAAC,CAAC,CAAA;IAClI,CAAC;IAID,eAAe,CAAC,IAAY;QAC1B,IAAI,CAAC,eAAe,GAAC,MAAM,CAAC;QAC5B,IAAI,CAAC,SAAS,GAAG,iCAAiC,CAAC;QACnD,IAAI,CAAC,aAAa,GAAC,IAAI,CAAC;QACxB,4BAA4B;QAC5B,IAAG,IAAI,CAAC,OAAO,EAAE,CAAC;YAChB,IAAI,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;YACtF,MAAM,CAAC,SAAS,CAAC,EAAC,IAAI,EAAC,CAAC,GAAyB,EAAE,EAAE;oBACnD,IAAI,CAAC,eAAe,GAAG,MAAM,CAAC;oBAC9B,IAAI,CAAC,SAAS,GAAG,+BAA+B,CAAC;oBACjD,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;oBAC3B,IAAI,GAAG,EAAE,CAAC;wBACR,IAAI,GAAG,YAAY,KAAK,EAAE,CAAC;4BACzB,GAAG,CAAC,OAAO,CAAC,CAAC,EAAE,EAAE,EAAE;gCACjB,kFAAkF;gCAClF,EAAE,CAAC,eAAe,GAAC,IAAI,CAAC;gCACxB,IAAG,EAAE,CAAC,WAAW,EAAC,CAAC;oCACjB,EAAE,CAAC,iBAAiB,GAAC,IAAI,IAAI,CAAC,EAAE,CAAC,WAAW,CAAC,CAAC;gCAChD,CAAC;gCACD,IAAG,EAAE,CAAC,IAAI,EAAC,CAAC;oCACV,EAAE,CAAC,cAAc,GAAC,IAAI,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC;gCACtC,CAAC;gCACD,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;4BACxC,CAAC,CAAC,CAAA;wBACJ,CAAC;6BAAM,CAAC;4BACN,OAAO,CAAC,KAAK,CAAC,yDAAyD,CAAC,CAAA;wBAC1E,CAAC;oBAEH,CAAC;yBAAM,CAAC;wBACN,+CAA+C;oBACjD,CAAC;gBACH,CAAC,EAAE,KAAK,EAAC,CAAC,GAAG,EAAE,EAAE;oBACf,4DAA4D;oBAC5D,IAAI,CAAC,KAAK,EAAE,CAAA;gBACd,CAAC,EAAE,QAAQ,EAAC,GAAG,EAAE;oBACf,eAAe;oBACf,IAAI,CAAC,KAAK,EAAE,CAAA;gBACd,CAAC,EACF,CAAC,CAAA;QACF,CAAC;aAAI,CAAC;YACJ,0BAA0B;YAC1B,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;YAC/B,IAAI,CAAC,SAAS,GAAG,uBAAuB,CAAC;YACzC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;YAC3B,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAChC,CAAC;IACH,CAAC;IAGD,IAAI,OAAO,CAAC,OAA+B;QACzC,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC;QACxB,IAAI,KAAK,GAAG,WAAW,CAAC,2BAA2B,CAAC;QAEpD,IAAI,OAAO,EAAE,CAAC;YACZ,OAAO,CAAC,IAAI,CAAC,gBAAgB,GAAG,OAAO,CAAC,IAAI,CAAC,CAAA;YAC7C,IAAI,OAAO,CAAC,uBAAuB,KAAK,IAAI,EAAE,CAAC;gBAC7C,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;YACvB,CAAC;YACD,IAAI,CAAC,YAAY,GAAG,OAAO,CAAC,YAAY,CAAC;YACzC,KAAK,GAAG,WAAW,CAAC,iBAAiB,CAAC,OAAO,CAAC,CAAC;YAC/C,OAAO,CAAC,IAAI,CAAC,6CAA6C,GAAG,KAAK,CAAC,CAAC;YACpE,IAAI,CAAC,sBAAsB,GAAG,OAAO,CAAC,sBAAsB,CAAC;YAC7D,IAAG,OAAO,CAAC,qBAAqB,KAAG,SAAS,EAAE,CAAC;gBAC7C,IAAI,CAAC,qBAAqB,GAAG,OAAO,CAAC,qBAAqB,CAAC;YAC7D,CAAC;YACD,IAAI,OAAO,CAAC,gBAAgB,KAAK,IAAI,EAAE,CAAC;gBACtC,IAAI,CAAC,sBAAsB,GAAG,aAAa,CAAC,0BAA0B,CAAC;YACzE,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC;YACrC,CAAC;YACD,IAAI,OAAO,CAAC,sBAAsB,EAAE,CAAC;gBACnC,IAAI,CAAC,sBAAsB,GAAG,OAAO,CAAC,sBAAsB,CAAC;YAC/D,CAAC;QACH,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,KAAK,CAAC,8BAA8B,CAAC,CAAA;QAC/C,CAAC;QACD,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;IAC5B,CAAC;IAGD,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,QAAQ,CAAC;IACvB,CAAC;IAED,mBAAmB,CAAC,EAAgB;QAClC,IAAI,CAAC,qBAAqB,GAAC,cAAc,CAAC,KAAK,CAAC;QAChD,IAAI,CAAC,aAAa,GAAC,KAAK,CAAC;QACzB,IAAI,CAAC,cAAc,GAAC,EAAE,CAAC;IACzB,CAAC;IAED,YAAY,CAAC,EAA6B;QACxC,IAAI,QAAQ,GAAG,EAAE,CAAC,MAAM,CAAC;QACzB,IAAI,CAAC,SAAS,GAAG,CAAC,cAAc,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC;QACpD,IAAI,UAAU,GAAG,EAAE,CAAC,WAAW,EAAE,CAAC;QAClC,IAAI,cAAc,CAAC,GAAG,KAAK,QAAQ,EAAE,CAAC;YACpC,IAAI,CAAC,YAAY,GAAG,MAAM,CAAA;QAC5B,CAAC;aAAM,CAAC;YACN,IAAI,UAAU,GAAG,EAAE,EAAE,CAAC;gBACpB,IAAI,CAAC,YAAY,GAAG,QAAQ,CAAA;YAC9B,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,YAAY,GAAG,SAAS,CAAA;YAC/B,CAAC;YACD,IAAI,CAAC,cAAc,GAAG,UAAU,CAAC;QACnC,CAAC;QAED,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAA;IACxC,CAAC;IAED,IAAI,kBAAkB,CAAC,kBAAoC;QACzD,IAAI,CAAC,mBAAmB,GAAC,kBAAkB,CAAC;QAC5C,IAAI,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC7B,IAAI,CAAC,mBAAmB,CAAC,QAAQ,GAAG,IAAI,CAAC;QAC3C,CAAC;IACH,CAAC;IAED,IAAI,kBAAkB;QACpB,OAAO,IAAI,CAAC,mBAAmB,CAAC;IAClC,CAAC;IAED,MAAM,CAAC,CAAmB;QACxB,IAAI,CAAC,CAAC,IAAI,IAAI,SAAS,CAAC,OAAO,EAAE,CAAC;YAChC,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,IAAI,CAAC;YACrC,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC,WAAW,CAAC,GAAG,EAAE;gBAC3C,kEAAkE;YACpE,CAAC,EAAE,EAAE,CAAC,CAAC;QACT,CAAC;aAAM,IAAI,CAAC,CAAC,IAAI,IAAI,SAAS,CAAC,OAAO,IAAI,CAAC,CAAC,IAAI,IAAI,SAAS,CAAC,KAAK,EAAE,CAAC;YACpE,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YACzC,2GAA2G;YAC3G,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC;QAE3D,CAAC;IACH,CAAC;IAED,aAAa;QACX,OAAO,CAAC,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,CAAA;IAC9F,CAAC;IAED,YAAY;QACV,OAAO,CAAC,IAAI,CAAC,gBAAgB,IAAI,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,CAAA;IAC5E,CAAC;IAED,iBAAiB;QACf,IAAG,CAAC,IAAI,CAAC,aAAa,EAAE,EAAC,CAAC;YACxB,IAAI,CAAC,uBAAuB,GAAC,OAAO,CAAA;QACtC,CAAC;aAAK,IAAG,CAAC,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;YAC9B,IAAI,CAAC,uBAAuB,GAAG,MAAM,CAAA;QACvC,CAAC;QACD,OAAO,IAAI,CAAC,uBAAuB,CAAC;IACtC,CAAC;IACD,qBAAqB;QACnB,IAAG,CAAC,IAAI,CAAC,aAAa,EAAE,EAAC,CAAC;YACxB,IAAI,CAAC,2BAA2B,GAAC,qBAAqB,CAAA;QACxD,CAAC;aAAK,IAAG,CAAC,IAAI,CAAC,YAAY,EAAE,EAAC,CAAC;YAC7B,IAAI,CAAC,2BAA2B,GAAC,MAAM,CAAA;QACzC,CAAC;QACD,OAAO,IAAI,CAAC,2BAA2B,CAAA;IACzC,CAAC;IACD,sBAAsB;QACpB,IAAG,CAAC,IAAI,CAAC,aAAa,EAAE,EAAC,CAAC;YACxB,OAAO,KAAK,CAAA;QACd,CAAC;aAAK,IAAG,CAAC,IAAI,CAAC,YAAY,EAAE,EAAC,CAAC;YAC7B,OAAO,QAAQ,CAAA;QACjB,CAAC;aAAI,CAAC;YACJ,OAAO,MAAM,CAAC;QAChB,CAAC;IACH,CAAC;IAED,gBAAgB;QACd,IAAG,CAAC,IAAI,CAAC,aAAa,EAAE,EAAC,CAAC;YACxB,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;QAC9C,CAAC;aAAK,IAAG,CAAC,IAAI,CAAC,YAAY,EAAE,EAAC,CAAC;YAC7B,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,OAAO,EAAE,CAAC;QAC7C,CAAC;IACH,CAAC;IAED,SAAS;QACP,IAAI,CAAC,MAAM,0BAAgB,CAAC;QAC5B,KAAK,CAAC,SAAS,EAAE,CAAC;QAClB,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YAClB,IAAI,CAAC,MAAM,sBAAY,CAAC;YACxB,OAAM;QACR,CAAC;QACD,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,QAAQ,GAAG,IAAI,CAAA;QAC/C,IAAI,CAAC,gBAAgB,CAAC,aAAa,CAAC,QAAQ,GAAG,IAAI,CAAA;QACnD,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,QAAQ,GAAG,IAAI,CAAA;QAC/C,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;QAC3B,IAAI,CAAC,qBAAqB,GAAG,CAAC,CAAC;QAC/B,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAC7B,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;QAClC,IAAI,CAAC,qBAAqB,GAAC,cAAc,CAAC,KAAK,CAAC;QAChD,IAAI,CAAC,aAAa,EAAE,CAAC;QAErB,IAAI,CAAC,YAAY,EAAE,CAAC;IACtB,CAAC;IAGD,iBAAiB;QACf,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YACxB,IAAI,EAAE,GAA2B,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC;YACrE,MAAM,EAAE,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,OAAO,EAAE,kBAAkB,EAAE,aAAa,KAAG,0BAA0B,CAAC,SAAS,EAAC,IAAI,CAAC,OAAO,EAAE,kBAAkB,EAAE,wBAAwB,CAAC,CAAC;YAC5K,IAAI,EAAE,GAAC,EAAE,EAAE,WAAW,CAAC;YACvB,IAAG,EAAE,YAAY,iBAAiB,EAAE,CAAC;gBACjC,EAAE,CAAC,UAAU,CAAC,EAAE,CAAC,WAAW,EAAE,CAAC,OAAO,EAAE,EAAE;oBACxC,IAAI,IAAI,GAAG,IAAI,IAAI,CAAC,CAAC,OAAO,CAAC,EAAE,EAAC,IAAI,EAAE,WAAW,EAAC,CAAC,CAAC;oBACpD,IAAI,KAAK,GAAG,GAAG,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;oBAEtC,IAAI,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;oBAClD,gCAAgC;oBAChC,UAAU,CAAC,IAAI,GAAG,KAAK,CAAC;oBAExB,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,QAAQ,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;oBAErD,sCAAsC;oBACtC,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;wBACxB,IAAI,EAAE,GAAG,IAAI,CAAC,cAAc,CAAC,cAAc,EAAE,CAAC;wBAC9C,EAAE,IAAI,GAAG,GAAG,IAAI,CAAC,qBAAqB,CAAC;wBACvC,EAAE,IAAI,MAAM,CAAC;wBACb,UAAU,CAAC,QAAQ,GAAG,EAAE,CAAC;wBACzB,UAAU,CAAC,KAAK,EAAE,CAAC;oBACrB,CAAC;oBACD,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,QAAQ,CAAC,IAAI,EAAE,UAAU,CAAC,CAAC;gBACvD,CAAC,CAAC,CAAC;YACP,CAAC;QACH,CAAC;IACH,CAAC;IAED,IAAI,cAAc,CAAC,cAAoC;QACrD,yBAAyB;QACzB,IAAI,CAAC,eAAe,GAAG,cAAc,CAAC;QACtC,IAAI,IAAI,CAAC,eAAe,EAAE,CAAC;YACzB,IAAI,GAAG,GAA0B,IAAI,CAAC,eAAe,CAAC,eAAe,CAAC;YACtE,IAAG,GAAG,EAAE,CAAC;gBACP,IAAI,CAAC,gBAAgB,GAAG,IAAI,SAAS,CAAC,GAAG,CAAC,CAAC;gBAC3C,wBAAwB;gBACxB,+DAA+D;gBAC/D,IAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;oBAC5B,IAAI,CAAC,mBAAmB,CAAC,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC;gBAC7D,CAAC;gBACD,IAAI,CAAC,aAAa,EAAE,CAAC;YACvB,CAAC;iBAAK,CAAC;gBACL,oBAAoB;gBACpB,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;gBAC7B,+DAA+D;gBAC/D,IAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;oBAC5B,IAAI,CAAC,mBAAmB,CAAC,SAAS,GAAG,IAAI,CAAC;gBAC5C,CAAC;gBAED,IAAI,IAAI,CAAC,mBAAmB,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;oBAC9C,kCAAkC;oBAClC,IAAI,CAAC,qBAAqB,GAAC,cAAc,CAAC,OAAO,CAAC;oBAClD,MAAM,EAAE,GAAC,IAAI,CAAC,eAAe,CAAC;oBAE9B,IAAI,iBAAiB,GAAC,IAAI,CAAC,uBAAuB,CAAC;oBACnD,IAAG,gBAAgB,CAAC,2BAA2B,KAAG,IAAI,CAAC,uBAAuB,IAAI,gBAAgB,CAAC,4BAA4B,KAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC;wBAC/J,0BAA0B;wBAC1B,iBAAiB,GAAC,gBAAgB,CAAC,WAAW,CAAC;wBAC/C,IAAI,EAAE,CAAC,QAAQ,IAAI,EAAE,CAAC,MAAM,EAAE,CAAC;4BAC7B,MAAM,OAAO,GAAG,EAAE,CAAC,QAAQ,GAAG,EAAE,CAAC,MAAM,CAAC;4BACxC,IAAI,OAAO,IAAI,IAAI,CAAC,0BAA0B,EAAE,CAAC;gCAC/C,oEAAoE;gCACpE,IAAG,gBAAgB,CAAC,2BAA2B,KAAG,IAAI,CAAC,uBAAuB,EAAC,CAAC;oCAC9E,iBAAiB,GAAC,gBAAgB,CAAC,UAAU,CAAC;gCAChD,CAAC;qCAAK,IAAG,gBAAgB,CAAC,4BAA4B,KAAG,IAAI,CAAC,uBAAuB,EAAE,CAAC;oCACtF,iBAAiB,GAAG,gBAAgB,CAAC,WAAW,CAAC;gCACnD,CAAC;4BACH,CAAC;wBACH,CAAC;oBACH,CAAC;oBAED,IAAG,gBAAgB,CAAC,UAAU,KAAG,IAAI,CAAC,uBAAuB,EAAC,CAAC;wBAC7D,wCAAwC;wBACxC,IAAI,OAAO,GAAgC,IAAI,CAAC;wBAChD,IAAG,CAAC,IAAI,CAAC,6BAA6B,EAAC,CAAC;4BACtC,MAAM,KAAK,CAAC,2CAA2C,CAAC,CAAC;wBAC3D,CAAC;6BAAK,CAAC;4BACL,0DAA0D;4BAC1D,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,cAAc,CAAC,kCAAkC,CAAC,IAAI,CAAC,6BAA6B,EAAE,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,SAAS,CAAC;gCAC5J,IAAI,EAAE,CAAC,GAAG,EAAE,EAAE;oCACZ,qEAAqE;oCACrE,OAAO,GAAG,GAAG,CAAC;gCAChB,CAAC;gCACD,QAAQ,EAAE,GAAG,EAAE;oCACb,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC,KAAK,CAAC;oCAClD,IAAI,KAAK,GAAG,IAAI,CAAC;oCACjB,IAAI,OAAO,EAAE,CAAC;wCACZ,IAAI,EAAE,EAAG,CAAC;4CACR,KAAK,GAAG,IAAI,eAAe,CAAC,OAAO,CAAC,CAAC;4CACrC,wBAAwB;4CACxB,IAAI,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;wCACxD,CAAC;oCACH,CAAC;yCAAM,CAAC;wCACN,mDAAmD;wCACnD,IAAI,CAAC,SAAS,GAAG,qCAAqC,CAAA;wCACtD,IAAI,CAAC,eAAe,GAAG,OAAO,CAAA;oCAChC,CAAC;oCACD,IAAI,KAAK,EAAE,CAAC;wCACV,0KAA0K;wCAC1K,IAAI,CAAC,gBAAgB,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC;oCAC/C,CAAC;oCACD,IAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;wCAC5B,IAAI,CAAC,mBAAmB,CAAC,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAA;oCAC5D,CAAC;oCACD,IAAI,CAAC,aAAa,EAAE,CAAC;gCACvB,CAAC;gCACD,KAAK,EAAE,GAAG,CAAC,EAAE;oCACX,OAAO,CAAC,KAAK,CAAC,6CAA6C,GAAG,GAAG,CAAC,CAAC;oCACnE,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC,KAAK,CAAC;oCAClD,IAAI,CAAC,SAAS,GAAG,sCAAsC,GAAG,GAAG,CAAC;oCAC9D,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;oCAC/B,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;gCACzC,CAAC;6BACF,CAAC,CAAC;wBACL,CAAC;oBACH,CAAC;yBAAK,IAAG,gBAAgB,CAAC,WAAW,KAAG,iBAAiB,EAAC,CAAC;wBACzD,0CAA0C;wBAC1C,IAAI,SAAS,GAA0B,IAAI,CAAC;wBAE5C,oDAAoD;wBACpD,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,cAAc,CAAC,gCAAgC,CAAE,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,SAAS,CAAC;4BACvH,IAAI,EAAE,CAAC,KAAK,EAAE,EAAE;gCACd,qEAAqE;gCACrE,SAAS,GAAG,KAAK,CAAC;4BACpB,CAAC;4BACD,QAAQ,EAAE,GAAG,EAAE;gCACb,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC,KAAK,CAAC;gCAClD,IAAI,KAAK,GAAG,IAAI,CAAC;gCACjB,IAAI,SAAS,EAAE,CAAC;oCACd,IAAI,EAAE,EAAE,CAAC;wCACP,KAAK,GAAG,IAAI,eAAe,CAAC,SAAS,CAAC,CAAC;wCAEvC,IAAI,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;oCACxD,CAAC;gCACH,CAAC;qCAAM,CAAC;oCACN,mDAAmD;oCACnD,IAAI,CAAC,SAAS,GAAG,qCAAqC,CAAA;oCACtD,IAAI,CAAC,eAAe,GAAG,OAAO,CAAA;gCAChC,CAAC;gCACD,IAAI,KAAK,EAAE,CAAC;oCACV,0KAA0K;oCAC1K,qFAAqF;oCACrF,IAAI,CAAC,gBAAgB,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC;oCAC7C,sBAAsB;oCACtB,2BAA2B;oCAC3B,IAAI;gCACN,CAAC;gCACD,IAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;oCAC5B,IAAI,CAAC,mBAAmB,CAAC,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC;gCAC7D,CAAC;gCACD,IAAI,CAAC,aAAa,EAAE,CAAC;4BACvB,CAAC;4BACD,KAAK,EAAE,GAAG,CAAC,EAAE;gCACX,OAAO,CAAC,KAAK,CAAC,6CAA6C,GAAG,GAAG,CAAC,CAAC;gCACnE,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC,KAAK,CAAC;gCAClD,IAAI,CAAC,SAAS,GAAG,sCAAsC,GAAG,GAAG,CAAC;gCAC9D,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;gCAC/B,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;4BACzC,CAAC;yBACF,CAAC,CAAC;oBAEL,CAAC;yBAAK,IAAG,gBAAgB,CAAC,WAAW,KAAG,iBAAiB,EAAC,CAAC;wBACzD,mCAAmC;wBACnC,IAAI,OAAO,GAA4B,IAAI,CAAC;wBAC5C,sEAAsE;wBACtE,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,cAAc,CAAC,kCAAkC,CAAE,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,SAAS,CAAC;4BACzH,IAAI,EAAE,CAAC,GAAG,EAAE,EAAE;gCACZ,OAAO,GAAG,GAAG,CAAC;4BAChB,CAAC;4BACD,QAAQ,EAAE,GAAG,EAAE;gCACb,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC,KAAK,CAAC;gCAClD,IAAI,KAAK,GAAG,IAAI,CAAC;gCACjB,IAAI,OAAO,EAAE,CAAC;oCACZ,IAAI,EAAE,EAAG,CAAC;wCACR,KAAK,GAAG,IAAI,eAAe,CAAC,OAAO,CAAC,CAAC;wCACrC,IAAI,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;oCACxD,CAAC;gCACH,CAAC;qCAAM,CAAC;oCACN,mDAAmD;oCACnD,IAAI,CAAC,SAAS,GAAG,qCAAqC,CAAA;oCACtD,IAAI,CAAC,eAAe,GAAG,OAAO,CAAA;gCAChC,CAAC;gCACD,IAAI,KAAK,EAAE,CAAC;oCACV,0KAA0K;oCAC1K,IAAI,CAAC,gBAAgB,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC;oCAC7C,wBAAwB;gCAC1B,CAAC;gCACD,IAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;oCAC5B,IAAI,CAAC,mBAAmB,CAAC,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC;gCAC7D,CAAC;gCACD,IAAI,CAAC,aAAa,EAAE,CAAC;4BACvB,CAAC;4BACD,KAAK,EAAE,GAAG,CAAC,EAAE;gCACX,OAAO,CAAC,KAAK,CAAC,6CAA6C,GAAG,GAAG,CAAC,CAAC;gCACnE,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC,KAAK,CAAC;gCAClD,IAAI,CAAC,SAAS,GAAG,sCAAsC,GAAG,GAAG,CAAC;gCAC9D,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;4BACjC,CAAC;yBACF,CAAC,CAAC;oBAEL,CAAC;yBAAM,CAAC;wBACN,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,cAAc,CAAC,6BAA6B,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,EAAE,EAAE,CAAC,CAAC,SAAS,CAAC;4BACnH,IAAI,EAAE,EAAE,CAAC,EAAE;gCACT,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC,KAAK,CAAC;gCAClD,IAAI,KAAK,GAAG,IAAI,CAAC;gCACjB,IAAI,EAAE,EAAE,CAAC;oCACP,IAAI,KAAK,GAAC,IAAI,iBAAiB,CAAC,EAAE,CAAC,CAAC;oCACpC,IAAI,EAAE,EAAE,CAAC;wCACP,KAAK,GAAG,IAAI,eAAe,CAAC,KAAK,CAAC,CAAC;wCACnC,IAAI,CAAC,iBAAiB,CAAC,mBAAmB,CAAC,EAAE,EAAE,KAAK,CAAC,CAAC;oCACxD,CAAC;gCACH,CAAC;qCAAM,CAAC;oCACN,OAAO,CAAC,KAAK,CAAC,qCAAqC,CAAC,CAAC;oCACrD,IAAI,CAAC,SAAS,GAAG,qCAAqC,CAAC;oCACvD,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;gCACjC,CAAC;gCACD,IAAI,KAAK,EAAE,CAAC;oCACV,0KAA0K;oCAC1K,IAAI,CAAC,gBAAgB,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,CAAC;oCAC7C,wBAAwB;oCACxB,oFAAoF;gCACtF,CAAC;gCACD,IAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;oCAC5B,IAAI,CAAC,mBAAmB,CAAC,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC;gCAC7D,CAAC;gCACD,IAAI,CAAC,aAAa,EAAE,CAAC;4BACvB,CAAC,EAAE,KAAK,EAAE,GAAG,CAAC,EAAE;gCACd,OAAO,CAAC,KAAK,CAAC,6CAA6C,GAAG,GAAG,CAAC,CAAC;gCACnE,IAAI,CAAC,qBAAqB,GAAG,cAAc,CAAC,KAAK,CAAC;gCAClD,IAAI,CAAC,SAAS,GAAG,sCAAsC,GAAG,GAAG,CAAC;gCAC9D,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;4BACjC,CAAC;yBACF,CAAC,CAAA;oBACJ,CAAC;gBACH,CAAC;qBAAI,CAAC;oBACJ,IAAI,CAAC,SAAS,GAAG,iEAAiE,CAAC;oBACnF,IAAI,CAAC,eAAe,GAAG,OAAO,CAAC;gBACjC,CAAC;YACH,CAAC;QAEH,CAAC;aAAM,CAAC;YACN,uCAAuC;YACvC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;YAC7B,IAAG,IAAI,CAAC,mBAAmB,EAAE,CAAC;gBAC5B,IAAI,CAAC,mBAAmB,CAAC,SAAS,GAAG,IAAI,CAAC;YAC5C,CAAC;QACH,CAAC;QACD,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAED,IAAI,cAAc;QAChB,OAAO,IAAI,CAAC,eAAe,CAAC;IAC9B,CAAC;IAED,6BAA6B;QAC3B,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,GAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACxD,CAAC;IAED,KAAK;QACH,KAAK,CAAC,KAAK,EAAE,CAAC;QACd,IAAI,CAAC,iBAAiB,CAAC,SAAS,EAAE,CAAC;QACnC,IAAI,CAAC,gBAAgB,EAAE,CAAC;QACxB,IAAI,CAAC,6BAA6B,EAAE,CAAC;IAEvC,CAAC;IAED,WAAW;QACT,OAAO,CAAC,IAAI,CAAC,MAAM,6BAAqB,IAAI,IAAI,CAAC,MAAM,iCAAuB,CAAC,CAAC;IAClF,CAAC;IAED,QAAQ;QACN,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,MAAM,2BAAmB,IAAI,IAAI,CAAC,MAAM,wBAAe,IAAI,IAAI,CAAC,MAAM,yBAAe,CAAC,IAAI,IAAI,CAAC,mBAAmB,IAAI,IAAI,CAAC,cAAc,CAAC,WAAW,GAAC,CAAC,CAAC,CAAA;IACzK,CAAC;IAGD,cAAc,CAAC,YAAkB,IAAI,CAAC,SAAS;QAC7C,6FAA6F;QAC7F,IAAG,SAAS,IAAI,CAAE,IAAI,CAAC,QAAQ,EAAE,EAAC,CAAC;YACjC,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;IAEO,uBAAuB;QAE7B,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,QAAQ,GAAG,IAAI,CAAC,kBAAkB,CAAC;QACnE,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,QAAQ,GAAG,IAAI,CAAC,kBAAkB,CAAC;IACrE,CAAC;IAED,sBAAsB;QACpB,KAAK,CAAC,sBAAsB,EAAE,CAAC;QAC/B,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,GAAC,CAAC,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IACxD,CAAC;IAED,gBAAgB;QACd,IAAI,CAAC,kBAAkB,GAAC,KAAK,CAAC;QAC9B,IAAI,CAAC,uBAAuB,EAAE,CAAC;IACjC,CAAC;IAED,OAAO;QACL,IAAI,CAAC,MAAM,2BAAmB,CAAC;QAC/B,KAAK,CAAC,OAAO,EAAE,CAAC;QAChB,IAAI,CAAC,eAAe,GAAG,MAAM,CAAC;QAC9B,IAAI,CAAC,SAAS,GAAG,cAAc,CAAC;QAEhC,IAAI,MAAM,GAAoB,CAAC,CAAC;QAChC,IAAG,IAAI,CAAC,QAAQ,EAAC,CAAC;YAChB,MAAM,GAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC;QACjC,CAAC;QAED,IAAG,IAAI,CAAC,MAAM,EAAE,CAAC;YACf,IAAI,EAAE,GAAG,IAAI,aAAa,CAAC,IAAI,CAAC,MAAM,EAAE,MAAM,EAAE,IAAI,CAAC,CAAC;YACtD,EAAE,CAAC,iBAAiB,GAAG,IAAI,CAAC,WAAW,CAAC;YACxC,IAAI,EAAE,CAAC,iBAAiB,EAAE,CAAC;gBACzB,EAAE,CAAC,WAAW,GAAG,EAAE,CAAC,iBAAiB,CAAC,QAAQ,EAAE,CAAC;YACnD,CAAC;YACD,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;QAC3B,CAAC;QACD,IAAI,kBAAkB,GAAG,qBAAqB,CAAC;QAC/C,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC,UAAU,CAAC,GAAG,EAAE;YAC1C,IAAI,CAAC,mBAAmB,EAAE,CAAA;QAC5B,CAAC,EAAE,kBAAkB,CAAC,CAAC;QACvB,IAAI,CAAC,kBAAkB,GAAG,IAAI,CAAC;QAC/B,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,KAAK,CAAC;IACpD,CAAC;IAED,QAAQ;QAEN,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,IAAI,CAAC;QACjD,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,IAAI,CAAC;QACjD,IAAI,CAAC,MAAM,+BAAuB,CAAC;QACnC,IAAI,CAAC,aAAa,EAAE,CAAC;IAEvB,CAAC;IAED,aAAa;QACX,IAAI,IAAI,CAAC,kBAAkB,IAAI,IAAI,CAAC,aAAa,IAAE,IAAI,EAAE,CAAC;YACxD,MAAM,CAAC,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YACxC,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;QAClC,CAAC;QACD,IAAG,IAAI,CAAC,EAAE,EAAE,CAAC;YACX,IAAI,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QACjB,CAAC;IACH,CAAC;IAED,mBAAmB;QAEjB,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;QAChC,IAAI,CAAC,MAAM,+BAAuB,CAAC;QACnC,IAAG,IAAI,CAAC,EAAE,EAAE,CAAC;YACX,IAAI,CAAC,EAAE,CAAC,IAAI,EAAE,CAAC;QACjB,CAAC;IACH,CAAC;IAED,OAAO;QACL,IAAI,CAAC,6BAA6B,EAAE,CAAA;QACpC,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,IAAI,CAAC;QACjD,IAAI,CAAC,gBAAgB,CAAC,UAAU,CAAC,QAAQ,GAAG,IAAI,CAAC;QACjD,IAAI,CAAC,gBAAgB,CAAC,WAAW,CAAC,QAAQ,GAAG,IAAI,CAAC;QAClD,IAAI,CAAC,eAAe,GAAG,MAAM,CAAC;QAC9B,IAAI,CAAC,SAAS,GAAG,WAAW,CAAC;QAE7B,IAAI,EAAE,GAAkB,IAAI,CAAC;QAE7B,IAAG,IAAI,CAAC,EAAE,EAAE,CAAC;YACX,IAAI,GAAG,GAAsB,IAAI,CAAC;YAClC,IAAI,EAAE,GAAkB,IAAI,CAAC;YAC7B,IAAG,IAAI,CAAC,EAAE,EAAE,CAAC;gBACX,IAAI,gBAAgB,CAAC,WAAW,KAAK,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAE,CAAC;oBAC9D,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,IAAI,CAAC;oBACrC,yBAAyB;oBACzB,IAAI,CAAC,aAAa,GAAC,IAAI,CAAC;oBACxB,IAAI,KAAK,GAAa,IAAI,CAAC;oBACzB,IAAI,IAAI,GAAa,IAAI,CAAC;oBAC1B,IAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;wBACjB,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;4BAExB,IAAI,EAAE,GAAG,IAAI,CAAC,cAAc,CAAC;4BAC7B,EAAE,CAAC,MAAM,GAAC,IAAI,CAAC,EAAE,CAAC,cAAc,CAAC;4BACjC,KAAK,GAAC,EAAE,CAAC,IAAI,CAAC;4BACd,uGAAuG;4BACvG,IAAI,GAAG,IAAI,CAAC,cAAc,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,EAAE,EAAE,CAAC,CAAC;wBACtE,CAAC;6BAAM,IAAI,IAAI,CAAC,OAAO,EAAE,OAAO,EAAE,CAAC;4BACjC,IAAG,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC;gCACnB,KAAK,GAAC,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC;gCACtB,IAAI,GAAG,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;4BACrG,CAAC;wBACH,CAAC;6BAAI,CAAC;4BACJ,OAAO,CAAC,KAAK,CAAC,oCAAoC,CAAC,CAAC;wBACtD,CAAC;wBACD,IAAI,IAAI,EAAE,CAAC;4BACT,MAAM,EAAE,GAAG,IAAI,CAAC,EAAE,CAAC,iBAAiB,CAAC;4BACrC,MAAM,IAAI,GAAC,EAAE,GAAC,gBAAgB,CAAC,gCAAgC,CAAC;4BAChE,2JAA2J;4BAC3J,IAAI,KAAK,GAAG,IAAI,cAAc,CAAC,IAAI,CAAC,cAAc,EAAE,IAAI,EAAE,IAAI,CAAC,EAAE,CAAC,YAAY,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,EAAE,CAAC,cAAc,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;4BAC/H,EAAE,GAAC,KAAK,CAAC;4BACT,IAAG,IAAI,CAAC,SAAS,EAAC,CAAC;gCACjB,IAAI,CAAC,SAAS,CAAC,MAAM,GAAC,CAAC,SAAS,EAAC,EAAE;oCACjC,iEAAiE;oCACjE,KAAK,CAAC,KAAK,EAAE,CAAC;gCAChB,CAAC,CAAA;4BACH,CAAC;wBAEH,CAAC;oBACH,CAAC;gBAEL,CAAC;qBAAK,IAAI,gBAAgB,CAAC,2BAA2B,KAAK,IAAI,CAAC,EAAE,CAAC,gBAAgB,IAAI,gBAAgB,CAAC,4BAA4B,KAAK,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAE,CAAC;oBAClK,IAAI,gBAAgB,CAAC,2BAA2B,KAAK,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAC,CAAC;wBAC7E,MAAM,IAAI,GAAC,IAAI,CAAC,EAAE,CAAC,WAAW,EAAE,CAAC;wBACjC,IAAG,IAAI,EAAE,CAAC;4BACR,EAAE,GAAG,IAAI,iBAAiB,CAAC,IAAI,CAAC,CAAC;wBACnC,CAAC;oBACH,CAAC;oBACD,IAAG,gBAAgB,CAAC,4BAA4B,KAAK,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAC,CAAC;wBAC7E,EAAE,GAAG,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAE,CAAC;oBAClC,CAAC;oBACD,IAAG,CAAC,EAAE,EAAC,CAAC;wBACN,IAAI,CAAC,eAAe,CAAC,QAAQ,GAAG,IAAI,CAAC;wBACrC,IAAI,CAAC,aAAa,GAAC,IAAI,CAAC;wBACxB,IAAI,KAAK,GAAa,IAAI,CAAC;wBAC3B,IAAI,IAAI,GAAa,IAAI,CAAC;wBAC1B,IAAG,IAAI,CAAC,QAAQ,EAAE,CAAC;4BACjB,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;gCACxB,IAAI,EAAE,GAAG,IAAI,CAAC,cAAc,CAAC;gCAC7B,EAAE,CAAC,MAAM,GAAC,IAAI,CAAC,EAAE,CAAC,cAAc,CAAC;gCACjC,KAAK,GAAC,EAAE,CAAC,IAAI,CAAC;gCACd,uGAAuG;gCACvG,IAAI,GAAG,IAAI,CAAC,cAAc,CAAC,YAAY,CAAC,IAAI,CAAC,QAAQ,EAAE,OAAO,EAAE,EAAE,CAAC,CAAC;4BACtE,CAAC;iCAAM,IAAI,IAAI,CAAC,OAAO,EAAE,OAAO,EAAE,CAAC;gCACjC,IAAG,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,CAAC;oCACnB,KAAK,GAAC,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC;oCACtB,IAAI,GAAG,IAAI,CAAC,cAAc,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC,OAAO,EAAE,IAAI,CAAC,OAAO,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;gCACrG,CAAC;4BACH,CAAC;iCAAI,CAAC;gCACJ,OAAO,CAAC,KAAK,CAAC,oCAAoC,CAAC,CAAC;4BACtD,CAAC;4BACD,IAAI,IAAI,EAAE,CAAC;gCACT,MAAM,EAAE,GAAG,IAAI,CAAC,EAAE,CAAC,iBAAiB,CAAC;gCACrC,MAAM,IAAI,GAAC,EAAE,GAAC,gBAAgB,CAAC,gCAAgC,CAAC;gCAChE,2JAA2J;gCAC3J,IAAI,KAAK,GAAG,IAAI,cAAc,CAAC,IAAI,CAAC,cAAc,EAAE,IAAI,EAAE,IAAI,CAAC,EAAE,CAAC,YAAY,EAAE,EAAE,EAAE,IAAI,EAAE,IAAI,CAAC,EAAE,CAAC,cAAc,EAAE,KAAK,EAAE,IAAI,CAAC,CAAC;gCAC/H,EAAE,GAAG,KAAK,CAAC;gCACX,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;oCACnB,IAAI,CAAC,SAAS,CAAC,MAAM,GAAG,CAAC,SAAS,EAAE,EAAE;wCACpC,iEAAiE;wCACjE,KAAK,CAAC,KAAK,EAAE,CAAC;oCAChB,CAAC,CAAA;gCACH,CAAC;4BACH,CAAC;wBACH,CAAC;oBACH,CAAC;gBACH,CAAC;qBAAM,IAAI,gBAAgB,CAAC,UAAU,KAAK,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAE,CAAC;oBACpE,EAAE,GAAG,IAAI,CAAC,EAAE,CAAC,qBAAqB,EAAE,CAAC;gBACvC,CAAC;qBAAM,IAAI,gBAAgB,CAAC,WAAW,KAAK,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAE,CAAC;oBACrE,EAAE,GAAG,IAAI,CAAC,EAAE,CAAC,gBAAgB,EAAE,CAAC;gBAClC,CAAC;qBAAM,CAAC;oBACN,EAAE,GAAG,IAAI,CAAC,EAAE,CAAC,WAAW,EAAE,CAAC;oBAC3B,IAAG,EAAE,EAAE,CAAC;wBACN,EAAE,GAAG,IAAI,iBAAiB,CAAC,EAAE,CAAC,CAAC;oBACjC,CAAC;gBACH,CAAC;gBACD,IAAI,EAAE,EAAE,CAAC;oBACP,GAAG,GAAG,IAAI,eAAe,CAAC,EAAE,CAAC,CAAC;gBAChC,CAAC;YACH,CAAC;YAED,IAAI,MAAM,GAAoB,CAAC,CAAC;YAChC,IAAG,IAAI,CAAC,QAAQ,EAAC,CAAC;gBAChB,MAAM,GAAC,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC;YACjC,CAAC;YAED,IAAG,IAAI,CAAC,cAAc,EAAE,CAAC;gBACvB,2DAA2D;gBAC3D,2IAA2I;gBAC3I,IAAI,EAAE,GAAG,IAAI,CAAC,cAAc,CAAC;gBAC7B,kBAAkB,CAAC,YAAY,CAAC,EAAE,EAAC,GAAG,CAAC,CAAC;gBACxC,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;gBAEtC,0DAA0D;gBAC1D,IAAI,IAAI,CAAC,sBAAsB,IAAI,IAAI,CAAC,uBAAuB,KAAG,IAAI,IAAI,gBAAgB,CAAC,UAAU,KAAG,IAAI,CAAC,uBAAuB,IAAI,EAAE,IAAI,IAAI,IAAI,EAAE,IAAI,IAAI,EAAE,CAAC;oBACjK,IAAI,WAAW,GAAG,EAAE,CAAC;oBAErB,IAAI,IAAI,CAAC,MAAM,IAAI,IAAI,CAAC,MAAM,CAAC,WAAW,EAAE,CAAC;wBAC3C,WAAW,GAAG,IAAI,CAAC,MAAM,CAAC,WAAW,CAAC;oBACxC,CAAC;oBACD,IAAI,WAAW,KAAK,EAAE,EAAE,CAAC;wBACvB,WAAW,GAAG,WAAW,GAAG,GAAG,CAAA;oBACjC,CAAC;oBAED,IAAI,WAAW,GAAG,WAAW,GAAG,cAAc,CAAC,eAAe,CAAC;oBAC/D,IAAI,MAAM,GAAW,WAAW,GAAG,GAAG,GAAG,EAAE,CAAC,OAAO,GAAG,GAAG,GAAG,eAAe,GAAG,GAAG,GAAG,EAAE,CAAC,IAAI,CAAC;oBAE5F,+CAA+C;oBAC/C,wFAAwF;oBACxF,gDAAgD;oBAEhD,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAA;oBAC/B,MAAM,EAAE,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,OAAO,EAAE,kBAAkB,EAAE,aAAa,KAAG,0BAA0B,CAAC,SAAS,EAAC,IAAI,CAAC,OAAO,EAAE,kBAAkB,EAAE,wBAAwB,CAAC,CAAC;oBAC5K,EAAE,CAAC,UAAU,CAAC,EAAE,EAAE,CAAC,OAAO,EAAE,EAAE;wBAC5B,IAAI,CAAC,sBAAsB,CAAC,OAAO,EAAC,MAAM,EAAC,EAAE,CAAC,CAAC;wBAC/C,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;wBACjC,IAAI,CAAC,cAAc,EAAE,CAAC;wBACtB,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;oBACzC,CAAC,CAAC,CAAC;gBACL,CAAC;YACH,CAAC;QACH,CAAC;QACD,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC;QAC1C,IAAI,CAAC,MAAM,sBAAc,CAAC;QAC1B,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;QAChC,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAC/B,IAAI,CAAC,cAAc,EAAE,CAAC;QACtB,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;IACzC,CAAC;IAED,KAAK,CAAC,GAAG,GAAC,4CAA4C,EAAC,SAAc,eAAe;QAClF,IAAI,CAAC,MAAM,uBAAa,CAAC;QACzB,KAAK,CAAC,KAAK,CAAC,GAAG,EAAC,MAAM,CAAC,CAAC;QACxB,IAAI,CAAC,uBAAuB,EAAE,CAAC;QAC/B,IAAI,CAAC,6BAA6B,EAAE,CAAC;IACvC,CAAC;IAED,sBAAsB,CAAC,OAAmB,EAAC,MAAc,EAAC,EAAgB;QACxE,IAAI,OAAO,GAAG,IAAI,IAAI,CAAC,CAAC,OAAO,CAAC,EAAE,EAAC,IAAI,EAAE,WAAW,EAAC,CAAC,CAAC;QAEvD,IAAI,EAAE,GAAC,IAAI,QAAQ,EAAE,CAAC;QACtB,IAAG,EAAE,CAAC,IAAI,EAAE,CAAC;YACX,EAAE,CAAC,GAAG,CAAC,MAAM,EAAE,EAAE,CAAC,IAAI,CAAC,CAAC;QAC1B,CAAC;QACD,IAAG,EAAE,CAAC,OAAO,KAAG,IAAI,EAAE,CAAC;YACrB,EAAE,CAAC,GAAG,CAAC,WAAW,EAAE,EAAE,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC,CAAC;QAC7C,CAAC;QACD,IAAG,EAAE,CAAC,iBAAiB,EAAC,CAAC;YACvB,EAAE,CAAC,GAAG,CAAC,aAAa,EAAC,EAAE,CAAC,iBAAiB,CAAC,MAAM,EAAE,CAAC,CAAC;QACtD,CAAC;QACD,EAAE,CAAC,GAAG,CAAC,OAAO,EAAC,OAAO,CAAC,CAAC;QACxB,IAAI,EAAE,GAAG,IAAI,MAAM,CAAC,EAAE,EAAE,MAAM,EAAC,EAAE,CAAC,CAAC;QACnC,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC,CAAC;IAChC,CAAC;IAED,oBAAoB,CAAC,WAAwB,EAAE,QAAgB;QAC7D,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;QAChC,MAAM,EAAE,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,OAAO,EAAE,kBAAkB,EAAE,aAAa,KAAG,0BAA0B,CAAC,SAAS,EAAC,IAAI,CAAC,OAAO,EAAE,kBAAkB,EAAE,wBAAwB,CAAC,CAAC;QAC5K,IAAI,WAAW,GAAG,IAAI,CAAC,eAAe,EAAE,CAAC;QACzC,IAAI,MAAM,GAAW,WAAW,GAAG,GAAG,GAAG,IAAI,CAAC,OAAO,EAAE,SAAS,GAAG,GAAG,GAAG,eAAe,GAAG,GAAG,GAAG,IAAI,CAAC,MAAM,GAAC,GAAG,GAAC,QAAQ,CAAC;QAC1H,IAAI,EAAE,GAAC,IAAI,CAAC,cAAc,CAAC;QAE3B,gLAAgL;QAChL,IAAI,GAAG,GAAC,IAAI,YAAY,EAAE,CAAC;QAC3B,IAAG,IAAI,CAAC,SAAS,EAAC,CAAC;YACjB,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;QAC1B,CAAC;QACD,EAAE,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,OAAO,EAAE,EAAE;YACrC,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,MAAM,EAAC,EAAE,EAAC,GAAG,CAAC,CAAC;YAC3C,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;QACnC,CAAC,CAAC,CAAC;IACL,CAAC;IAED,IAAI;QACF,IAAG,IAAI,CAAC,EAAE,EAAE,CAAC;YACX,IAAI,CAAC,EAAE,CAAC,KAAK,EAAE,CAAC;QAClB,CAAC;IACH,CAAC;IAEO,6BAA6B;QACnC,IAAG,IAAI,CAAC,mBAAmB,EAAC,CAAC;YAC3B,MAAM,QAAQ,GAAC,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,CAAC;YAC3D,IAAI,QAAQ,KAAG,IAAI,EAAE,CAAC;gBACpB,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,iBAAiB,GAAG,QAAQ,CAAC;gBACjE,IAAI,CAAC,gBAAgB,CAAC,iBAAiB,GAAG,QAAQ,CAAC;YACrD,CAAC;QACH,CAAC;IACH,CAAC;IAED,iBAAiB,CAAC,CAAmB;QACnC,IAAI,SAAS,CAAC,KAAK,KAAK,CAAC,CAAC,IAAI,EAAE,CAAC;QAEjC,CAAC;aAAM,IAAI,SAAS,CAAC,OAAO,KAAK,CAAC,CAAC,IAAI,EAAE,CAAC;YACxC,8BAA8B;YAC9B,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC,WAAW,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,6BAA6B,EAAE,EAAE,EAAE,CAAC,CAAC;QAE1F,CAAC;aAAM,IAAI,SAAS,CAAC,KAAK,KAAK,CAAC,CAAC,IAAI,EAAE,CAAC;YACtC,mBAAmB;YACnB,MAAM,CAAC,aAAa,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QAE3C,CAAC;QACD,IAAG,CAAC,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAC;QAC3C,CAAC;IACH,CAAC;+GAr7BU,aAAa,2OA+BJ,qBAAqB;mGA/B9B,aAAa,qNAhIb,CAAC,cAAc,CAAC,6EAwIhB,iBAAiB,iGACjB,QAAQ,qFAxIT;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0DT;;4FAqEU,aAAa;kBAlIzB,SAAS;+BACE,mBAAmB,aAClB,CAAC,cAAc,CAAC,YACjB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA0DT;;0BAoGY,MAAM;2BAAC,qBAAqB;yCA5BhC,WAAW;sBAAnB,KAAK;gBAK0C,iBAAiB;sBAAhE,SAAS;uBAAC,iBAAiB,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBACP,gBAAgB;sBAAtD,SAAS;uBAAC,QAAQ,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBAE5B,SAAS;sBAAjB,KAAK;gBAoGN,UAAU;sBADT,YAAY;uBAAC,iBAAiB,EAAE,CAAC,QAAQ,CAAC;gBAS3C,SAAS;sBADR,YAAY;uBAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC;;AAk1B5C,MAAM,OAAO,sBAAuB,SAAQ,iBAAiB;IAY3D,YAAoB,KAAqB,EACrB,MAAc,EACd,iBAAoC,EACpC,cAA6B,EAC7B,cAA6B,EAC3B,QAA+B;QAEnD,KAAK,CAAC,QAAQ,CAAC,CAAC;QAPE,UAAK,GAAL,KAAK,CAAgB;QACrB,WAAM,GAAN,MAAM,CAAQ;QACd,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,mBAAc,GAAd,cAAc,CAAe;QAC7B,mBAAc,GAAd,cAAc,CAAe;QAC3B,aAAQ,GAAR,QAAQ,CAAuB;IAGrD,CAAC;IAED,QAAQ;QAEN,IAAI,CAAC,kBAAkB,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;QAEnD,IAAI,CAAC,EAAE,CAAC,kBAAkB,GAAC,IAAI,CAAC,kBAAkB,CAAC;QAEnD,gDAAgD;QAChD,MAAM,CAAC,gBAAgB,CAAC,cAAc,EAAE,CAAC,CAAC,EAAE,EAAE;YAC5C,OAAO,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAC;YAE1C,IAAI,IAAI,CAAC,KAAK,EAAE,EAAE,CAAC;gBACjB,OAAO;YACT,CAAC;iBAAM,CAAC;gBACN,0FAA0F;gBAC1F,MAAM,OAAO,GAAG,kEAAkE,CAAC;gBACnF,KAAK,CAAC,OAAO,CAAC,CAAC;gBACf,CAAC,GAAG,CAAC,IAAI,MAAM,CAAC,KAAK,CAAC;gBAEtB,IAAI,CAAC,EAAE,CAAC;oBACN,CAAC,CAAC,WAAW,GAAG,OAAO,CAAC;oBACxB,CAAC,CAAC,YAAY,GAAG,IAAI,CAAC;oBACtB,IAAI,CAAC,CAAC,eAAe,EAAE,CAAC;wBACtB,CAAC,CAAC,eAAe,EAAE,CAAC;oBACtB,CAAC;oBACD,IAAI,CAAC,CAAC,cAAc,EAAE,CAAC;wBACrB,CAAC,CAAC,cAAc,EAAE,CAAC;oBACrB,CAAC;gBACH,CAAC;gBAED,OAAO,OAAO,CAAC;YACjB,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,eAAe;QAEb,uBAAuB;QAEvB,IAAI,CAAC,QAAQ,CAAC,QAAQ,GAAG,CAAC,EAAE,EAAE,EAAE;YAC9B,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,CAAC;QACxB,CAAC,CAAA;QACD,IAAI,CAAC,KAAK,CAAC,WAAW,CAAC,SAAS,CAAC,CAAC,MAAc,EAAE,EAAE;YAClD,IAAI,MAAM,CAAC,WAAW,CAAC,EAAE,CAAC;gBACxB,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC,CAAC;YACzC,CAAC;QACH,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC,MAAc,EAAE,EAAE;YAC7C,IAAI,aAAa,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC;YACjC,IAAI,aAAa,EAAE,CAAC;gBAClB,IAAI,CAAC,YAAY,CAAC,aAAa,CAAC,CAAC;YACnC,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,WAAW;QACT,sBAAsB;IACxB,CAAC;IAED,IAAI,YAAY;QACd,OAAQ,IAAI,CAAC,EAAE,CAAC,YAAY,CAAC;IAC/B,CAAC;IAED,YAAY,CAAC,SAAgB;QAE3B,IAAI,OAAO,GAAE,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC;QAE5D,IAAG,OAAO,EAAE,CAAC;YACX,OAAO,CAAC,SAAS,CAAC;gBAChB,IAAI,EAAC,CAAC,IAAI,EAAE,EAAE;oBACZ,IAAI,CAAC,EAAE,CAAC,eAAe,GAAC,MAAM,CAAC;oBAC/B,IAAI,CAAC,EAAE,CAAC,SAAS,GAAG,wBAAwB,CAAC;oBAC7C,IAAI,CAAC,EAAE,CAAC,aAAa,GAAC,KAAK,CAAC;oBAC5B,IAAI,CAAC,OAAO,GAAC,IAAI,CAAC;oBAClB,IAAI,CAAC,EAAE,CAAC,OAAO,GAAC,IAAI,CAAC;oBACrB,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;wBACjB,4DAA4D;wBAC5D,IAAI,CAAC,cAAc,CAAC,iBAAiB,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,SAAS,CAAC;4BAC5D,IAAI,EAAC,CAAC,OAAO,EAAC,EAAE;gCAChB,IAAI,CAAC,EAAE,CAAC,OAAO,GAAC,OAAO,CAAC;gCACxB,IAAI,CAAC,EAAE,CAAC,eAAe,CAAC,IAAI,CAAC,CAAC;4BAChC,CAAC,EAAC,KAAK,EAAC,CAAC,MAAM,EAAE,EAAE;gCACjB,IAAI,CAAC,EAAE,CAAC,SAAS,GAAC,MAAM,CAAC;gCACzB,IAAI,CAAC,EAAE,CAAC,eAAe,GAAC,OAAO,CAAC;gCAChC,IAAI,CAAC,EAAE,CAAC,aAAa,GAAC,KAAK,CAAC;gCAC5B,OAAO,CAAC,KAAK,CAAC,iCAAiC,GAAC,MAAM,CAAC,CAAA;4BACzD,CAAC;yBAAC,CAAC,CAAC;oBAEN,CAAC;yBAAM,CAAC;wBACN,OAAO,CAAC,IAAI,CAAC,iEAAiE,CAAC,CAAA;oBACjF,CAAC;gBACH,CAAC;gBACD,KAAK,EAAC,CAAC,MAAM,EAAE,EAAE;oBACf,IAAI,CAAC,EAAE,CAAC,SAAS,GAAG,MAAM,CAAC;oBAC3B,IAAI,CAAC,EAAE,CAAC,eAAe,GAAG,OAAO,CAAC;oBAClC,IAAI,CAAC,EAAE,CAAC,aAAa,GAAC,KAAK,CAAC;oBAC5B,OAAO,CAAC,KAAK,CAAC,yBAAyB,GAAG,MAAM,CAAC,CAAA;gBACnD,CAAC;aAAC,CAAC,CAAC;QACR,CAAC;IACH,CAAC;IAED,YAAY,CAAC,EAA6B;QACxC,IAAI,QAAQ,GAAG,EAAE,CAAC,MAAM,CAAC;QACzB,IAAI,CAAC,SAAS,GAAG,CAAC,cAAc,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC;QACpD,IAAI,UAAU,GAAG,EAAE,CAAC,WAAW,EAAE,CAAC;QAClC,IAAI,cAAc,CAAC,GAAG,KAAK,QAAQ,EAAE,CAAC;YACpC,IAAI,CAAC,EAAE,CAAC,YAAY,GAAG,MAAM,CAAA;QAC/B,CAAC;aAAM,CAAC;YACN,IAAI,UAAU,GAAG,EAAE,EAAE,CAAC;gBACpB,IAAI,CAAC,EAAE,CAAC,YAAY,GAAG,QAAQ,CAAA;YACjC,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,EAAE,CAAC,YAAY,GAAG,SAAS,CAAA;YAClC,CAAC;YACD,IAAI,CAAC,EAAE,CAAC,cAAc,GAAG,UAAU,CAAC;QACtC,CAAC;QACD,IAAI,CAAC,EAAE,CAAC,cAAc,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QACvC,IAAI,CAAC,iBAAiB,CAAC,aAAa,EAAE,CAAA;IACxC,CAAC;IAGD,KAAK;QACH,OAAO,IAAI,CAAC,SAAS,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,QAAQ,EAAE,CAAA;IAC9C,CAAC;+GA/IU,sBAAsB;mGAAtB,sBAAsB,iDAdtB,CAAC,cAAc,CAAC,8DAwBhB,aAAa,qFAvBd;;GAET,qJA77BU,aAAa;;4FAw8Bb,sBAAsB;kBAhBlC,SAAS;+BACE,wBAAwB,aACvB,CAAC,cAAc,CAAC,YACjB;;GAET;0OAqB2C,EAAE;sBAA7C,SAAS;uBAAC,aAAa,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE","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"]}