speechrecorderng 2.22.1 → 2.22.5

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.
@@ -0,0 +1,283 @@
1
+ import { TransportActions } from "./controlpanel";
2
+ import { Browser, UserAgentBuilder } from "../../utils/ua-parser";
3
+ import { MessageDialog } from "../../ui/message_dialog";
4
+ import { LevelMeasure, StreamLevelMeasure } from "../../audio/dsp/level_measure";
5
+ import { Action } from "../../action/action";
6
+ import { MIN_DB_LEVEL } from "../../ui/recordingitem_display";
7
+ export const FORCE_REQUEST_AUDIO_PERMISSIONS = false;
8
+ export const RECFILE_API_CTX = 'recfile';
9
+ export const MAX_RECORDING_TIME_MS = 1000 * 60 * 60 * 60; // 1 hour
10
+ export const LEVEL_BAR_INTERVALL_SECONDS = 0.1; // 100ms
11
+ export class BasicRecorder {
12
+ constructor(dialog, sessionService) {
13
+ this.dialog = dialog;
14
+ this.sessionService = sessionService;
15
+ this.statusMsg = '';
16
+ this.statusWaiting = false;
17
+ this.readonly = false;
18
+ this.processingRecording = false;
19
+ this.ac = null;
20
+ this._selectedDeviceId = undefined;
21
+ this._channelCount = 2;
22
+ this._session = null;
23
+ this.uploadProgress = 100;
24
+ this.uploadStatus = 'ok';
25
+ this.audioSignalCollapsed = true;
26
+ this.peakLevelInDb = MIN_DB_LEVEL;
27
+ this.displayLevelInfos = null;
28
+ this.displayAudioClip = null;
29
+ this.audioFetchSubscription = null;
30
+ this.destroyed = false;
31
+ this.navigationDisabled = true;
32
+ this.userAgent = UserAgentBuilder.userAgent();
33
+ console.debug("Detected platform: " + this.userAgent.detectedPlatform);
34
+ console.debug("Detected browser: " + this.userAgent.detectedBrowser);
35
+ this.transportActions = new TransportActions();
36
+ this.playStartAction = new Action('Play');
37
+ this.levelMeasure = new LevelMeasure();
38
+ this.streamLevelMeasure = new StreamLevelMeasure();
39
+ this.selCaptureDeviceId = null;
40
+ }
41
+ set audioDevices(audioDevices) {
42
+ this._audioDevices = audioDevices;
43
+ }
44
+ set autoGainControlConfigs(autoGainControlConfigs) {
45
+ this._autoGainControlConfigs = autoGainControlConfigs;
46
+ }
47
+ set channelCount(channelCount) {
48
+ this._channelCount = channelCount;
49
+ }
50
+ set session(session) {
51
+ this._session = session;
52
+ }
53
+ get session() {
54
+ return this._session;
55
+ }
56
+ enableStartUserGesture() {
57
+ this.statusAlertType = 'info';
58
+ this.statusMsg = 'Ready.';
59
+ }
60
+ start() {
61
+ this.statusAlertType = 'info';
62
+ this.statusMsg = 'Starting session...';
63
+ this.statusWaiting = false;
64
+ if (this._session) {
65
+ if (this._session.sealed) {
66
+ this.readonly = true;
67
+ this.statusMsg = 'Session sealed!';
68
+ //let dialogRef = this.dialog.open(SessionSealedDialog, {});
69
+ this.dialog.open(MessageDialog, {
70
+ data: {
71
+ type: 'error',
72
+ title: 'Error',
73
+ msg: "This session is sealed. Recordings cannot be added anymore.",
74
+ advise: 'Please ask your experimenter what to do (e.g start a new session).',
75
+ }
76
+ });
77
+ }
78
+ else {
79
+ let body = {};
80
+ if (this._session.status === "CREATED") {
81
+ this._session.status = "LOADED";
82
+ body.status = this._session.status;
83
+ if (!this._session.loadedDate) {
84
+ this._session.loadedDate = new Date();
85
+ body.loadedDate = this._session.loadedDate;
86
+ }
87
+ }
88
+ else {
89
+ this._session.restartedDate = new Date();
90
+ body.restartedDate = this._session.restartedDate;
91
+ }
92
+ this.sessionService.patchSessionObserver(this._session, body).subscribe();
93
+ }
94
+ }
95
+ // Check browser compatibility
96
+ if (this.userAgent.detectedBrowser === Browser.Safari && this._channelCount > 1) {
97
+ let eMsg = "Error: Safari browser does not support stereo recordings.";
98
+ console.error(eMsg);
99
+ this.dialog.open(MessageDialog, {
100
+ data: {
101
+ type: 'error',
102
+ title: 'Browser not supported',
103
+ msg: eMsg,
104
+ advice: "Please use a supported browser, e.g. Mozilla Firefox."
105
+ }
106
+ });
107
+ }
108
+ //console.log("Session ID: "+this._session.session+ " status: "+this._session.status)
109
+ this._selectedDeviceId = undefined;
110
+ if (!this.readonly && this.ac && (FORCE_REQUEST_AUDIO_PERMISSIONS || (this._audioDevices && this._audioDevices.length > 0))) {
111
+ this.statusMsg = 'Requesting audio permissions...';
112
+ this.statusAlertType = 'info';
113
+ this.ac.deviceInfos((mdis) => {
114
+ let audioCaptureDeviceAvail = false;
115
+ let audioPlayDeviceAvail = false;
116
+ if (mdis) {
117
+ if (this.ac) {
118
+ this.ac.printDevices(mdis);
119
+ }
120
+ if (mdis.length > 0) {
121
+ for (let mdii = 0; mdii < mdis.length; mdii++) {
122
+ let mdi = mdis[mdii];
123
+ let kind = mdi.kind;
124
+ if (kind === "audioinput") {
125
+ audioCaptureDeviceAvail = true;
126
+ }
127
+ else if (kind === "audiooutput") {
128
+ audioPlayDeviceAvail = true;
129
+ }
130
+ }
131
+ }
132
+ if (this._session && this._session.type !== 'TEST_DEF_A' && this._audioDevices && this._audioDevices.length > 0) {
133
+ let fdi = null;
134
+ for (let adI = 0; adI < this._audioDevices.length; adI++) {
135
+ let ad = this._audioDevices[adI];
136
+ if (ad.playback) {
137
+ // project audio device config for playback device
138
+ // not used for now
139
+ continue;
140
+ }
141
+ for (let mdii = 0; mdii < mdis.length; mdii++) {
142
+ let mdi = mdis[mdii];
143
+ let kind = mdi.kind;
144
+ if (kind === "audioinput") {
145
+ audioCaptureDeviceAvail = true;
146
+ }
147
+ else if (kind === "audiooutput") {
148
+ audioPlayDeviceAvail = true;
149
+ }
150
+ if (ad.regex) {
151
+ //console.log("Match?: \'"+mdi.label+"\' \'"+ad.name+"\'");
152
+ if (mdi.label.match(ad.name)) {
153
+ fdi = mdi;
154
+ //console.log("Match!");
155
+ }
156
+ }
157
+ else {
158
+ if (mdi.label.trim() === ad.name.trim()) {
159
+ fdi = mdi;
160
+ }
161
+ }
162
+ if (fdi) {
163
+ break;
164
+ }
165
+ }
166
+ if (fdi) {
167
+ break;
168
+ }
169
+ }
170
+ if (fdi) {
171
+ // matching device found
172
+ // Not able to open device here since Chrome 71
173
+ // Chrome 71 requires a user gesture before the AudioContext can be resumed
174
+ // sessionmanager.ts:712 Open session with default audio device for 1 channels
175
+ // capture.ts:128 The AudioContext was not allowed to start. It must be resumed (or created) after a user gesture on the page. https://goo.gl/7K7WLu
176
+ // push../projects/speechrecorderng/src/lib/audio/capture/capture.ts.AudioCapture.open @ capture.ts:128
177
+ //this.ac.open(this._channelCount, fdi.deviceId);
178
+ console.info("Set selected audio device: \'" + fdi.label + "\' Id: \'" + fdi.deviceId + "\'");
179
+ this._selectedDeviceId = fdi.deviceId;
180
+ this.enableStartUserGesture();
181
+ }
182
+ else {
183
+ // device not found
184
+ this.statusMsg = 'ERROR: Required audio device not available!';
185
+ this.statusAlertType = 'error';
186
+ this.readonly = true;
187
+ this.dialog.open(MessageDialog, {
188
+ data: {
189
+ type: 'error',
190
+ title: 'Required audio device',
191
+ msg: "Required audio device not found",
192
+ advice: "Please connect a suitable audio device for this project and retry (press the browser reload button)."
193
+ }
194
+ });
195
+ }
196
+ }
197
+ else {
198
+ if (!audioCaptureDeviceAvail && !audioPlayDeviceAvail) {
199
+ // no device found
200
+ this.statusMsg = 'ERROR: No audio device available!';
201
+ this.statusAlertType = 'warn';
202
+ //this.readonly = true;
203
+ this.dialog.open(MessageDialog, {
204
+ data: {
205
+ type: 'warn',
206
+ title: 'No audio device',
207
+ msg: "No audio device found",
208
+ advice: "Please connect an audio device and retry (press the browser reload button) or try to continue anyway."
209
+ }
210
+ });
211
+ }
212
+ else {
213
+ if (!this.readonly && !audioCaptureDeviceAvail) {
214
+ // no device found
215
+ this.statusMsg = 'WARNING: No audio capture device available!';
216
+ this.statusAlertType = 'warning';
217
+ //this.readonly = true;
218
+ this.dialog.open(MessageDialog, {
219
+ data: {
220
+ type: 'warning',
221
+ title: 'No audio capture device',
222
+ msg: "No audio capture device found",
223
+ advice: "Please connect an audio capture device and retry (press the browser reload button) or try to continue anyway."
224
+ }
225
+ });
226
+ }
227
+ // Safari does not list playback devices
228
+ if (!audioPlayDeviceAvail) {
229
+ // Firefox does not enumerate audiooutput devices
230
+ // Do not show this warning, because it would always appear on Firefox
231
+ // When https://bugzilla.mozilla.org/show_bug.cgi?id=1498512 is fixed the warning can be enabled for Firefox as well
232
+ // It is already implemneted but kept behind a preference setting https://bugzilla.mozilla.org/show_bug.cgi?id=1152401
233
+ // Output devices are listed if about:config media.setsinkid.enabled=true
234
+ // but default setting is false
235
+ // Same problem with Safari
236
+ if (!(this.userAgent.detectedBrowser === Browser.Safari || this.userAgent.detectedBrowser === Browser.Firefox)) {
237
+ // no device found
238
+ this.statusMsg = 'WARNING: No audio playback device available!';
239
+ this.statusAlertType = 'warn';
240
+ //this.readonly = true;
241
+ this.dialog.open(MessageDialog, {
242
+ data: {
243
+ type: 'warn',
244
+ title: 'No audio playback device',
245
+ msg: "No audio playback device found",
246
+ advice: "Please connect an audio playback device and retry (press the browser reload button) or try to continue anyway."
247
+ }
248
+ });
249
+ }
250
+ }
251
+ }
252
+ // Not able to open device here since Chrome 71
253
+ // Chrome 71 requires a user gesture before the AudioContext can be resumed
254
+ // sessionmanager.ts:712 Open session with default audio device for 1 channels
255
+ // capture.ts:128 The AudioContext was not allowed to start. It must be resumed (or created) after a user gesture on the page. https://goo.gl/7K7WLu
256
+ // push../projects/speechrecorderng/src/lib/audio/capture/capture.ts.AudioCapture.open @ capture.ts:128
257
+ //console.log("Open session with default audio device for "+this._channelCount+" channels");
258
+ //this.ac.open(this._channelCount);
259
+ // enable start gesture anyway
260
+ this.enableStartUserGesture();
261
+ }
262
+ }
263
+ });
264
+ }
265
+ }
266
+ closed() {
267
+ this.statusAlertType = 'info';
268
+ this.statusMsg = 'Session closed.';
269
+ }
270
+ error(msg = 'An unknown error occured during recording.', advice = 'Please retry.') {
271
+ this.statusMsg = 'ERROR: Recording.';
272
+ this.statusAlertType = 'error';
273
+ this.dialog.open(MessageDialog, {
274
+ data: {
275
+ type: 'error',
276
+ title: 'Recording error',
277
+ msg: msg,
278
+ advice: advice
279
+ }
280
+ });
281
+ }
282
+ }
283
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"basicrecorder.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/session/basicrecorder.ts"],"names":[],"mappings":"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","sourcesContent":["import {TransportActions} from \"./controlpanel\";\r\nimport {Browser, UserAgent, UserAgentBuilder} from \"../../utils/ua-parser\";\r\nimport {MatDialog} from \"@angular/material/dialog\";\r\nimport {MessageDialog} from \"../../ui/message_dialog\";\r\nimport {Session} from \"./session\";\r\nimport {SessionService} from \"./session.service\";\r\nimport {AudioCapture} from \"../../audio/capture/capture\";\r\nimport {AudioDevice, AutoGainControlConfig} from \"../project/project\";\r\nimport {LevelInfos, LevelMeasure, StreamLevelMeasure} from \"../../audio/dsp/level_measure\";\r\nimport {AudioPlayer} from \"../../audio/playback/player\";\r\nimport {Subscription} from \"rxjs\";\r\nimport {AudioClip} from \"../../audio/persistor\";\r\nimport {Action} from \"../../action/action\";\r\nimport {MIN_DB_LEVEL} from \"../../ui/recordingitem_display\";\r\n\r\nexport const FORCE_REQUEST_AUDIO_PERMISSIONS=false;\r\nexport const RECFILE_API_CTX = 'recfile';\r\nexport const MAX_RECORDING_TIME_MS = 1000 * 60 * 60 * 60; // 1 hour\r\n\r\nexport const LEVEL_BAR_INTERVALL_SECONDS = 0.1;  // 100ms\r\n\r\nexport class BasicRecorder {\r\n  protected userAgent:UserAgent;\r\n  statusMsg: string='';\r\n  statusAlertType!: string;\r\n  statusWaiting: boolean=false;\r\n  readonly=false;\r\n  processingRecording=false;\r\n  ac: AudioCapture|null=null;\r\n  // Property audioDevices from project config: list of names of allowed audio devices.\r\n  protected _audioDevices: Array<AudioDevice> | null| undefined;\r\n  protected _selectedDeviceId:string|undefined=undefined;\r\n  protected selCaptureDeviceId: ConstrainDOMString | null;\r\n  protected _channelCount = 2;\r\n  protected _autoGainControlConfigs: Array<AutoGainControlConfig> | null| undefined;\r\n\r\n  _session: Session|null=null;\r\n\r\n  transportActions: TransportActions;\r\n  playStartAction: Action<void>;\r\n\r\n  uploadProgress: number = 100;\r\n  uploadStatus: string = 'ok'\r\n  audioSignalCollapsed = true;\r\n\r\n  protected streamLevelMeasure: StreamLevelMeasure;\r\n  protected levelMeasure: LevelMeasure;\r\n  peakLevelInDb:number=MIN_DB_LEVEL;\r\n  displayLevelInfos: LevelInfos | null=null;\r\n  protected _controlAudioPlayer!: AudioPlayer;\r\n  displayAudioClip: AudioClip | null=null;\r\n  protected audioFetchSubscription:Subscription|null=null;\r\n\r\n  protected destroyed=false;\r\n\r\n  protected navigationDisabled=true;\r\n\r\n  constructor(  public dialog: MatDialog,protected sessionService:SessionService) {\r\n    this.userAgent=UserAgentBuilder.userAgent();\r\n    console.debug(\"Detected platform: \"+this.userAgent.detectedPlatform);\r\n    console.debug(\"Detected browser: \"+this.userAgent.detectedBrowser);\r\n    this.transportActions = new TransportActions();\r\n    this.playStartAction = new Action('Play');\r\n    this.levelMeasure = new LevelMeasure();\r\n    this.streamLevelMeasure = new StreamLevelMeasure();\r\n    this.selCaptureDeviceId = null;\r\n  }\r\n\r\n  set audioDevices(audioDevices: Array<AudioDevice> | null | undefined) {\r\n    this._audioDevices = audioDevices;\r\n  }\r\n\r\n  set autoGainControlConfigs(autoGainControlConfigs: Array<AutoGainControlConfig>|null|undefined){\r\n    this._autoGainControlConfigs=autoGainControlConfigs;\r\n  }\r\n\r\n  set channelCount(channelCount: number) {\r\n    this._channelCount = channelCount;\r\n  }\r\n\r\n  set session(session: Session|null) {\r\n    this._session = session;\r\n  }\r\n\r\n  get session():Session|null{\r\n    return this._session;\r\n  }\r\n\r\n  enableStartUserGesture() {\r\n    this.statusAlertType = 'info';\r\n    this.statusMsg = 'Ready.';\r\n  }\r\n\r\n  start() {\r\n    this.statusAlertType = 'info';\r\n    this.statusMsg = 'Starting session...';\r\n    this.statusWaiting=false;\r\n    if(this._session) {\r\n      if (this._session.sealed) {\r\n        this.readonly = true\r\n        this.statusMsg = 'Session sealed!';\r\n        //let dialogRef = this.dialog.open(SessionSealedDialog, {});\r\n        this.dialog.open(MessageDialog, {\r\n          data: {\r\n            type: 'error',\r\n            title: 'Error',\r\n            msg: \"This session is sealed. Recordings cannot be added anymore.\",\r\n            advise: 'Please ask your experimenter what to do (e.g start a new session).',\r\n          }\r\n        });\r\n      } else {\r\n        let body: any = {};\r\n        if (this._session.status === \"CREATED\") {\r\n          this._session.status = \"LOADED\";\r\n          body.status = this._session.status;\r\n          if (!this._session.loadedDate) {\r\n            this._session.loadedDate = new Date();\r\n            body.loadedDate = this._session.loadedDate;\r\n          }\r\n        } else {\r\n          this._session.restartedDate = new Date();\r\n          body.restartedDate = this._session.restartedDate;\r\n        }\r\n        this.sessionService.patchSessionObserver(this._session, body).subscribe()\r\n      }\r\n    }\r\n\r\n    // Check browser compatibility\r\n    if(this.userAgent.detectedBrowser===Browser.Safari && this._channelCount>1){\r\n      let eMsg=\"Error: Safari browser does not support stereo recordings.\";\r\n      console.error(eMsg);\r\n      this.dialog.open(MessageDialog, {\r\n        data: {\r\n          type: 'error',\r\n          title: 'Browser not supported',\r\n          msg: eMsg,\r\n          advice: \"Please use a supported browser, e.g. Mozilla Firefox.\"\r\n        }\r\n      })\r\n    }\r\n\r\n    //console.log(\"Session ID: \"+this._session.session+ \" status: \"+this._session.status)\r\n    this._selectedDeviceId=undefined;\r\n\r\n    if (!this.readonly && this.ac && (FORCE_REQUEST_AUDIO_PERMISSIONS || (this._audioDevices && this._audioDevices.length > 0))) {\r\n      this.statusMsg = 'Requesting audio permissions...';\r\n      this.statusAlertType = 'info';\r\n\r\n      this.ac.deviceInfos((mdis) => {\r\n        let audioCaptureDeviceAvail: boolean = false;\r\n        let audioPlayDeviceAvail: boolean = false;\r\n        if (mdis) {\r\n          if(this.ac) {\r\n            this.ac.printDevices(mdis);\r\n          }\r\n          if (mdis.length > 0) {\r\n            for (let mdii = 0; mdii < mdis.length; mdii++) {\r\n              let mdi = mdis[mdii];\r\n              let kind=mdi.kind;\r\n              if(kind === \"audioinput\"){\r\n                audioCaptureDeviceAvail=true;\r\n              }else if(kind=== \"audiooutput\"){\r\n                audioPlayDeviceAvail=true;\r\n              }\r\n            }\r\n          }\r\n\r\n          if (this._session && this._session.type !== 'TEST_DEF_A' && this._audioDevices && this._audioDevices.length > 0) {\r\n            let fdi: MediaDeviceInfo | null = null;\r\n            for (let adI = 0; adI < this._audioDevices.length; adI++) {\r\n              let ad = this._audioDevices[adI];\r\n              if (ad.playback) {\r\n                // project audio device config for playback device\r\n                // not used for now\r\n                continue;\r\n              }\r\n              for (let mdii = 0; mdii < mdis.length; mdii++) {\r\n                let mdi = mdis[mdii];\r\n                let kind=mdi.kind;\r\n                if(kind === \"audioinput\"){\r\n                  audioCaptureDeviceAvail=true;\r\n                }else if(kind=== \"audiooutput\"){\r\n                  audioPlayDeviceAvail=true;\r\n                }\r\n                if (ad.regex) {\r\n                  //console.log(\"Match?: \\'\"+mdi.label+\"\\' \\'\"+ad.name+\"\\'\");\r\n                  if (mdi.label.match(ad.name)) {\r\n                    fdi = mdi;\r\n                    //console.log(\"Match!\");\r\n                  }\r\n                } else {\r\n                  if (mdi.label.trim() === ad.name.trim()) {\r\n                    fdi = mdi;\r\n                  }\r\n                }\r\n                if (fdi) {\r\n                  break;\r\n                }\r\n              }\r\n              if (fdi) {\r\n                break;\r\n              }\r\n            }\r\n\r\n            if (fdi) {\r\n              // matching device found\r\n\r\n              // Not able to open device here since Chrome 71\r\n              // Chrome 71 requires a user gesture before the AudioContext can be resumed\r\n              // sessionmanager.ts:712 Open session with default audio device for 1 channels\r\n              // capture.ts:128 The AudioContext was not allowed to start. It must be resumed (or created) after a user gesture on the page. https://goo.gl/7K7WLu\r\n              // push../projects/speechrecorderng/src/lib/audio/capture/capture.ts.AudioCapture.open @ capture.ts:128\r\n\r\n              //this.ac.open(this._channelCount, fdi.deviceId);\r\n              console.info(\"Set selected audio device: \\'\" + fdi.label + \"\\' Id: \\'\" + fdi.deviceId + \"\\'\");\r\n              this._selectedDeviceId = fdi.deviceId;\r\n\r\n              this.enableStartUserGesture()\r\n            } else {\r\n              // device not found\r\n              this.statusMsg = 'ERROR: Required audio device not available!';\r\n              this.statusAlertType = 'error';\r\n              this.readonly = true;\r\n\r\n              this.dialog.open(MessageDialog, {\r\n                data: {\r\n                  type: 'error',\r\n                  title: 'Required audio device',\r\n                  msg: \"Required audio device not found\",\r\n                  advice: \"Please connect a suitable audio device for this project and retry (press the browser reload button).\"\r\n                }\r\n              })\r\n            }\r\n          }else{\r\n            if(!audioCaptureDeviceAvail && !audioPlayDeviceAvail){\r\n              // no device found\r\n              this.statusMsg = 'ERROR: No audio device available!';\r\n              this.statusAlertType = 'warn';\r\n              //this.readonly = true;\r\n\r\n              this.dialog.open(MessageDialog, {\r\n                data: {\r\n                  type: 'warn',\r\n                  title: 'No audio device',\r\n                  msg: \"No audio device found\",\r\n                  advice: \"Please connect an audio device and retry (press the browser reload button) or try to continue anyway.\"\r\n                }\r\n              })\r\n            }else {\r\n              if (!this.readonly && !audioCaptureDeviceAvail) {\r\n                // no device found\r\n                this.statusMsg = 'WARNING: No audio capture device available!';\r\n                this.statusAlertType = 'warning';\r\n                //this.readonly = true;\r\n\r\n                this.dialog.open(MessageDialog, {\r\n                  data: {\r\n                    type: 'warning',\r\n                    title: 'No audio capture device',\r\n                    msg: \"No audio capture device found\",\r\n                    advice: \"Please connect an audio capture device and retry (press the browser reload button) or try to continue anyway.\"\r\n                  }\r\n                })\r\n              }\r\n              // Safari does not list playback devices\r\n              if (!audioPlayDeviceAvail) {\r\n                // Firefox does not enumerate audiooutput devices\r\n                // Do not show this warning, because it would always appear on Firefox\r\n                // When https://bugzilla.mozilla.org/show_bug.cgi?id=1498512 is fixed the warning can be enabled for Firefox as well\r\n\r\n                // It is already implemneted but kept behind a preference setting https://bugzilla.mozilla.org/show_bug.cgi?id=1152401\r\n\r\n\r\n                // Output devices are listed if about:config media.setsinkid.enabled=true\r\n                // but default setting is false\r\n\r\n\r\n                // Same problem with Safari\r\n\r\n                if (!(this.userAgent.detectedBrowser===Browser.Safari || this.userAgent.detectedBrowser===Browser.Firefox)) {\r\n                  // no device found\r\n                  this.statusMsg = 'WARNING: No audio playback device available!';\r\n                  this.statusAlertType = 'warn';\r\n                  //this.readonly = true;\r\n\r\n                  this.dialog.open(MessageDialog, {\r\n                    data: {\r\n                      type: 'warn',\r\n                      title: 'No audio playback device',\r\n                      msg: \"No audio playback device found\",\r\n                      advice: \"Please connect an audio playback device and retry (press the browser reload button) or try to continue anyway.\"\r\n                    }\r\n                  })\r\n                }\r\n              }\r\n            }\r\n\r\n            // Not able to open device here since Chrome 71\r\n            // Chrome 71 requires a user gesture before the AudioContext can be resumed\r\n            // sessionmanager.ts:712 Open session with default audio device for 1 channels\r\n            // capture.ts:128 The AudioContext was not allowed to start. It must be resumed (or created) after a user gesture on the page. https://goo.gl/7K7WLu\r\n            // push../projects/speechrecorderng/src/lib/audio/capture/capture.ts.AudioCapture.open @ capture.ts:128\r\n            //console.log(\"Open session with default audio device for \"+this._channelCount+\" channels\");\r\n            //this.ac.open(this._channelCount);\r\n\r\n            // enable start gesture anyway\r\n            this.enableStartUserGesture()\r\n          }\r\n        }\r\n\r\n      });\r\n    }\r\n  }\r\n\r\n  closed() {\r\n    this.statusAlertType = 'info';\r\n    this.statusMsg = 'Session closed.';\r\n  }\r\n\r\n  error(msg='An unknown error occured during recording.',advice:string='Please retry.') {\r\n    this.statusMsg = 'ERROR: Recording.';\r\n    this.statusAlertType = 'error';\r\n    this.dialog.open(MessageDialog, {\r\n      data: {\r\n        type: 'error',\r\n        title: 'Recording error',\r\n        msg: msg,\r\n        advice: advice\r\n      }\r\n    });\r\n  }\r\n\r\n}\r\n"]}
@@ -265,7 +265,7 @@ TransportPanel.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version
265
265
  <mat-icon>chevron_right</mat-icon>
266
266
  </button>
267
267
 
268
- `, isInline: true, styles: [":host{flex:20;align-self:center;width:100%;text-align:center;align-content:center;margin:0}\n", "div{display:inline;flex:0}\n"], components: [{ type: i5.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.DefaultShowHideDirective, selector: " [fxShow], [fxShow.print], [fxShow.xs], [fxShow.sm], [fxShow.md], [fxShow.lg], [fxShow.xl], [fxShow.lt-sm], [fxShow.lt-md], [fxShow.lt-lg], [fxShow.lt-xl], [fxShow.gt-xs], [fxShow.gt-sm], [fxShow.gt-md], [fxShow.gt-lg], [fxHide], [fxHide.print], [fxHide.xs], [fxHide.sm], [fxHide.md], [fxHide.lg], [fxHide.xl], [fxHide.lt-sm], [fxHide.lt-md], [fxHide.lt-lg], [fxHide.lt-xl], [fxHide.gt-xs], [fxHide.gt-sm], [fxHide.gt-md], [fxHide.gt-lg]", inputs: ["fxShow", "fxShow.print", "fxShow.xs", "fxShow.sm", "fxShow.md", "fxShow.lg", "fxShow.xl", "fxShow.lt-sm", "fxShow.lt-md", "fxShow.lt-lg", "fxShow.lt-xl", "fxShow.gt-xs", "fxShow.gt-sm", "fxShow.gt-md", "fxShow.gt-lg", "fxHide", "fxHide.print", "fxHide.xs", "fxHide.sm", "fxHide.md", "fxHide.lg", "fxHide.xl", "fxHide.lt-sm", "fxHide.lt-md", "fxHide.lt-lg", "fxHide.lt-xl", "fxHide.gt-xs", "fxHide.gt-sm", "fxHide.gt-md", "fxHide.gt-lg"] }] });
268
+ `, isInline: true, styles: [":host{flex:20;align-self:center;width:100%;text-align:center;align-content:center;margin:0}\n", "div{display:inline;flex:0}\n", "button{touch-action:manipulation}\n"], components: [{ type: i5.MatButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { type: i2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }], directives: [{ type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i6.DefaultShowHideDirective, selector: " [fxShow], [fxShow.print], [fxShow.xs], [fxShow.sm], [fxShow.md], [fxShow.lg], [fxShow.xl], [fxShow.lt-sm], [fxShow.lt-md], [fxShow.lt-lg], [fxShow.lt-xl], [fxShow.gt-xs], [fxShow.gt-sm], [fxShow.gt-md], [fxShow.gt-lg], [fxHide], [fxHide.print], [fxHide.xs], [fxHide.sm], [fxHide.md], [fxHide.lg], [fxHide.xl], [fxHide.lt-sm], [fxHide.lt-md], [fxHide.lt-lg], [fxHide.lt-xl], [fxHide.gt-xs], [fxHide.gt-sm], [fxHide.gt-md], [fxHide.gt-lg]", inputs: ["fxShow", "fxShow.print", "fxShow.xs", "fxShow.sm", "fxShow.md", "fxShow.lg", "fxShow.xl", "fxShow.lt-sm", "fxShow.lt-md", "fxShow.lt-lg", "fxShow.lt-xl", "fxShow.gt-xs", "fxShow.gt-sm", "fxShow.gt-md", "fxShow.gt-lg", "fxHide", "fxHide.print", "fxHide.xs", "fxHide.sm", "fxHide.md", "fxHide.lg", "fxHide.xl", "fxHide.lt-sm", "fxHide.lt-md", "fxHide.lt-lg", "fxHide.lt-xl", "fxHide.gt-xs", "fxHide.gt-sm", "fxHide.gt-md", "fxHide.gt-lg"] }] });
269
269
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.1", ngImport: i0, type: TransportPanel, decorators: [{
270
270
  type: Component,
271
271
  args: [{
@@ -303,7 +303,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.1", ngImpor
303
303
  div {
304
304
  display: inline;
305
305
  flex: 0;
306
- }`
306
+ }`, `
307
+ button {
308
+ touch-action: manipulation;
309
+ }`
307
310
  ]
308
311
  }]
309
312
  }], propDecorators: { readonly: [{
@@ -446,4 +449,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.2.1", ngImpor
446
449
  }], ready: [{
447
450
  type: Input
448
451
  }] } });
449
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"controlpanel.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/session/controlpanel.ts"],"names":[],"mappings":"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","sourcesContent":["import {Action} from '../../action/action'\r\nimport {\r\n  Component, ViewChild, Input\r\n} from \"@angular/core\";\r\n\r\nimport { MatDialog} from \"@angular/material/dialog\";\r\nimport {ProgressSpinnerMode} from \"@angular/material/progress-spinner\";\r\n\r\n\r\n@Component({\r\n\r\n  selector: 'app-sprstatusdisplay',\r\n\r\n  template: `\r\n    <p matTooltip=\"Status\">\r\n      <mat-progress-spinner *ngIf=\"statusWaiting\" color=\"black\"  mode=\"indeterminate\" [diameter]=\"30\" [strokeWidth]=\"5\"></mat-progress-spinner><mat-icon *ngIf=\"statusAlertType==='error'\" style=\"color:red\">report_problem</mat-icon>\r\n      {{statusMsg}}\r\n    </p>\r\n  `,\r\n  styles: [`:host {\r\n    display: inline;\r\n    text-align: left;\r\n    font-size: smaller;\r\n  }`, `\r\n    p {\r\n      padding: 4px;\r\n      white-space:nowrap;\r\n      display: inline-block;\r\n    }\r\n  `, `\r\n    mat-progress-spinner {\r\n      color: black;\r\n      display: inline-block;\r\n    }\r\n  `, `\r\n    span {\r\n      color: red;\r\n    }\r\n  `]\r\n\r\n})\r\n\r\nexport class StatusDisplay {\r\n  @Input() statusAlertType = 'info';\r\n  @Input() statusMsg = 'Initialize...';\r\n  @Input() statusWaiting =false;\r\n}\r\n\r\n\r\n@Component({\r\n  selector: 'app-uploadstatus',\r\n  template: `\r\n    <mat-progress-spinner [mode]=\"spinnerMode\" [color]=\"status\" [diameter]=\"30\" [strokeWidth]=\"5\" [value]=\"_value\" [matTooltip]=\"toolTipText\"></mat-progress-spinner>\r\n  `,\r\n  styles: [`:host {\r\n    flex: 1;\r\n  /* align-self: flex-start; */\r\n    /*display: inline; */\r\n    text-align: left;\r\n  }`,`mat-progress-spinner{\r\n      display: inline-block;\r\n  }`]\r\n})\r\nexport class UploadStatus {\r\n  private _awaitNewUpload=false;\r\n  spinnerMode:ProgressSpinnerMode = 'determinate';\r\n  _status!:string\r\n  _value = 100\r\n  displayValue:string|null=null\r\n  toolTipText:string=''\r\n\r\n  private _updateSpinner(){\r\n\r\n    let uplMsg;\r\n    if (this._awaitNewUpload || this._value === 0) {\r\n      this.spinnerMode = 'indeterminate'\r\n      this.displayValue='&nbsp;&nbsp;&nbsp;&nbsp;'\r\n      uplMsg='Preparing upload.'\r\n    } else {\r\n      this.spinnerMode = 'determinate'\r\n      this.displayValue=this._value+'%'\r\n      if(this._value===100){\r\n        uplMsg = 'Upload complete'\r\n      }else {\r\n        uplMsg = 'Upload progress: ' + this.displayValue\r\n      }\r\n    }\r\n    if(this.status==='warn'){\r\n      uplMsg='Upload error occurred. Please check your network connection. '+uplMsg\r\n    }\r\n    this.toolTipText=uplMsg\r\n  }\r\n\r\n  @Input()\r\n  set value(value: number) {\r\n    this._value = value;\r\n    this._updateSpinner()\r\n  };\r\n\r\n  @Input() set awaitNewUpload(awaitNewUpload:boolean){\r\n    this._awaitNewUpload=awaitNewUpload\r\n    this._updateSpinner()\r\n  }\r\n\r\n  @Input() set status(status:string){\r\n    this._status=status\r\n    this._updateSpinner()\r\n  }\r\n\r\n  get status():string{\r\n    return this._status\r\n  }\r\n\r\n}\r\n\r\n\r\n@Component({\r\n  selector: 'app-sprprogressdisplay',\r\n  template: `\r\n    <p>{{progressMsg}}</p>\r\n  `,\r\n  styles: [`:host {\r\n    flex: 1;\r\n  /* align-self: flex-start; */\r\n    /*display: inline; */\r\n      width: 100%;\r\n    text-align: left;\r\n  }`]\r\n})\r\nexport class ProgressDisplay {\r\n  progressMsg = '[itemcode]';\r\n}\r\n\r\n\r\n\r\n\r\nexport class TransportActions {\r\n  startAction: Action<void>;\r\n  stopAction: Action<void>;\r\n  nextAction: Action<void>;\r\n  fwdNextAction: Action<void>;\r\n  pauseAction: Action<void>;\r\n  fwdAction: Action<void>;\r\n  bwdAction: Action<void>;\r\n\r\n  constructor() {\r\n    this.startAction = new Action('Start');\r\n    this.stopAction = new Action('Stop');\r\n    this.nextAction = new Action('Next');\r\n    this.pauseAction = new Action('Pause');\r\n    this.fwdNextAction = new Action('Next recording');\r\n    this.fwdAction = new Action('Forward');\r\n    this.bwdAction = new Action('Backward');\r\n\r\n  }\r\n}\r\n\r\n@Component({\r\n\r\n  selector: 'app-sprtransport',\r\n\r\n  template: `\r\n    <button id=\"bwdBtn\" *ngIf=\"navigationEnabled\"  (click)=\"actions.bwdAction.perform()\" [disabled]=\"bwdDisabled()\"\r\n            mat-raised-button>\r\n      <mat-icon>chevron_left</mat-icon>\r\n    </button>\r\n    <button (click)=\"startStopNextPerform()\" [disabled]=\"startDisabled() && stopDisabled() && nextDisabled()\"  mat-raised-button>\r\n      <mat-icon [style.color]=\"startStopNextIconColor()\">{{startStopNextIconName()}}</mat-icon><mat-icon *ngIf=\"!nextDisabled()\" [style.color]=\"nextDisabled() ? 'grey' : 'black'\">chevron_right</mat-icon>\r\n      <span fxShow.xs=\"false\">{{startStopNextName()}}</span>\r\n    </button>\r\n    <button *ngIf=\"pausingEnabled\" (click)=\"actions.pauseAction.perform()\" [disabled]=\"pauseDisabled()\" mat-raised-button>\r\n      <mat-icon>pause</mat-icon>\r\n      <span fxShow.xs=\"false\">Pause</span>\r\n    </button>\r\n    <button id=\"fwdNextBtn\" *ngIf=\"navigationEnabled\" fxHide.xs (click)=\"actions.fwdNextAction.perform()\" [disabled]=\"fwdNextDisabled()\" mat-raised-button>\r\n      <mat-icon>redo</mat-icon>\r\n    </button>\r\n    <button id=\"fwdBtn\" *ngIf=\"navigationEnabled\"  (click)=\"actions.fwdAction.perform()\" [disabled]=\"fwdDisabled()\" mat-raised-button>\r\n      <mat-icon>chevron_right</mat-icon>\r\n    </button>\r\n\r\n  `,\r\n  styles: [`:host {\r\n    flex: 20;\r\n    align-self: center;\r\n    width: 100%;\r\n    text-align: center;\r\n    align-content: center;\r\n    margin: 0;\r\n\r\n  }`, `\r\n    div {\r\n      display: inline;\r\n      flex: 0;\r\n    }`\r\n  ]\r\n\r\n})\r\nexport class TransportPanel {\r\n\r\n  @Input() readonly!:boolean;\r\n  @Input() actions!: TransportActions;\r\n  @Input() navigationEnabled=true;\r\n  @Input() pausingEnabled=true;\r\n\r\n  startStopNextButtonName!:string;\r\n  startStopNextButtonIconName!:string;\r\n\r\n  startDisabled() {\r\n    return !this.actions || this.readonly || this.actions.startAction.disabled\r\n  }\r\n\r\n  stopDisabled() {\r\n    return !this.actions || this.actions.stopAction.disabled\r\n  }\r\n\r\n  nextDisabled() {\r\n    return !this.actions || this.actions.nextAction.disabled || !this.navigationEnabled;\r\n  }\r\n\r\n  pauseDisabled() {\r\n    return !this.actions || this.actions.pauseAction.disabled || !this.pausingEnabled;\r\n  }\r\n\r\n  fwdDisabled() {\r\n    return !this.actions || this.actions.fwdAction.disabled || !this.navigationEnabled;\r\n  }\r\n\r\n  fwdNextDisabled() {\r\n    return !this.actions || this.actions.fwdNextAction.disabled || !this.navigationEnabled;\r\n  }\r\n\r\n  bwdDisabled() {\r\n    return !this.actions || this.actions.bwdAction.disabled || !this.navigationEnabled;\r\n  }\r\n\r\n    startStopNextName():string{\r\n        if(!this.nextDisabled()){\r\n            this.startStopNextButtonName= \"Next\"\r\n        }else 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      }else if(!this.nextDisabled()){\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() || !this.nextDisabled()){\r\n            return \"yellow\"\r\n        }else{\r\n            return \"grey\";\r\n        }\r\n    }\r\n\r\n  startStopNextPerform(){\r\n    if(!this.startDisabled()){\r\n      this.actions.startAction.perform();\r\n    }else if(!this.stopDisabled()){\r\n      this.actions.stopAction.perform();\r\n    }else if(!this.nextDisabled()){\r\n      this.actions.nextAction.perform();\r\n    }\r\n  }\r\n\r\n}\r\n\r\n@Component({\r\n\r\n  selector: 'app-readystateindicator',\r\n\r\n  template: `\r\n        <mat-icon [matTooltip]=\"readyStateToolTip\">{{hourGlassIconName}}</mat-icon>\r\n  `,\r\n  styles: []\r\n})\r\nexport class ReadyStateIndicator {\r\n  _ready=true\r\n  hourGlassIconName='hourglass_empty'\r\n  readyStateToolTip:string=''\r\n\r\n  constructor() {}\r\n\r\n  @Input() set ready(ready:boolean){\r\n    this._ready=ready\r\n    this.hourGlassIconName=this._ready?'hourglass_empty':'hourglass_full'\r\n    this.readyStateToolTip=this._ready?'Audio processing and upload done. You can leave the page without data loss.':'Please wait until audio processing and upload have finished. Please do not leave the page.'\r\n  }\r\n\r\n  get ready():boolean{\r\n    return this._ready\r\n  }\r\n\r\n}\r\n\r\n@Component({\r\n\r\n  selector: 'app-sprcontrolpanel',\r\n\r\n  template: `\r\n    <div fxHide.xs  fxLayout=\"row\" >\r\n     <app-sprstatusdisplay fxFlex=\"0 0 0\" [statusMsg]=\"statusMsg\" [statusAlertType]=\"statusAlertType\" [statusWaiting]=\"statusWaiting\"\r\n                          class=\"hidden-xs\"></app-sprstatusdisplay>\r\n      <app-sprtransport fxFlex=\"10 0 0\" [readonly]=\"readonly\" [actions]=\"transportActions\" [navigationEnabled]=\"navigationEnabled\"></app-sprtransport>\r\n      <app-uploadstatus fxFlex=\"0 0 0\" *ngIf=\"enableUploadRecordings\" [value]=\"uploadProgress\"\r\n                      [status]=\"uploadStatus\" [awaitNewUpload]=\"processing\"></app-uploadstatus>\r\n      <app-readystateindicator [ready]=\"_ready\"></app-readystateindicator>\r\n    </div>\r\n    <div fxShow.xs fxHide fxLayout=\"column\">\r\n      <div fxLayout=\"row\" fxFlexFill>\r\n       <app-sprstatusdisplay fxFlex=\"10 0 0\" fxFlexAlign=\"left\" [statusMsg]=\"statusMsg\" [statusAlertType]=\"statusAlertType\" [statusWaiting]=\"statusWaiting\"\r\n                            class=\"hidden-xs\"></app-sprstatusdisplay>\r\n       <app-uploadstatus fxFlex=\"0 0 0\" *ngIf=\"enableUploadRecordings\" [value]=\"uploadProgress\"\r\n                        [status]=\"uploadStatus\" [awaitNewUpload]=\"processing\"></app-uploadstatus>\r\n        <app-readystateindicator [ready]=\"_ready\"></app-readystateindicator>\r\n      </div>\r\n      <app-sprtransport [readonly]=\"readonly\" [actions]=\"transportActions\" [navigationEnabled]=\"navigationEnabled\"></app-sprtransport>\r\n\r\n    </div>\r\n  `,\r\n  styles: [`div {\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  }`]\r\n})\r\nexport class ControlPanel {\r\n  @ViewChild(StatusDisplay, { static: true }) statusDisplay!: StatusDisplay;\r\n  @ViewChild(TransportPanel, { static: true }) transportPanel!: TransportPanel;\r\n\r\n  @Input() readonly!:boolean\r\n  @Input() transportActions!: TransportActions\r\n  @Input() processing=false\r\n  @Input() statusMsg!: string;\r\n  @Input() statusAlertType!: string;\r\n  @Input() statusWaiting!: boolean;\r\n  @Input() uploadStatus!: string;\r\n  @Input() uploadProgress!: number;\r\n  @Input() currentRecording: AudioBuffer| null| undefined;\r\n  @Input() enableUploadRecordings!: boolean;\r\n  @Input() navigationEnabled=true;\r\n\r\n  _ready=true\r\n\r\n  constructor(public dialog: MatDialog) {\r\n\r\n  }\r\n\r\n  @Input() set ready(ready:boolean){\r\n    this._ready=ready\r\n  }\r\n\r\n  get ready():boolean{\r\n    return this._ready\r\n  }\r\n\r\n}\r\n\r\n\r\n"]}
452
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"controlpanel.js","sourceRoot":"","sources":["../../../../../../projects/speechrecorderng/src/lib/speechrecorder/session/controlpanel.ts"],"names":[],"mappings":"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","sourcesContent":["import {Action} from '../../action/action'\r\nimport {\r\n  Component, ViewChild, Input\r\n} from \"@angular/core\";\r\n\r\nimport { MatDialog} from \"@angular/material/dialog\";\r\nimport {ProgressSpinnerMode} from \"@angular/material/progress-spinner\";\r\n\r\n\r\n@Component({\r\n\r\n  selector: 'app-sprstatusdisplay',\r\n\r\n  template: `\r\n    <p matTooltip=\"Status\">\r\n      <mat-progress-spinner *ngIf=\"statusWaiting\" color=\"black\"  mode=\"indeterminate\" [diameter]=\"30\" [strokeWidth]=\"5\"></mat-progress-spinner><mat-icon *ngIf=\"statusAlertType==='error'\" style=\"color:red\">report_problem</mat-icon>\r\n      {{statusMsg}}\r\n    </p>\r\n  `,\r\n  styles: [`:host {\r\n    display: inline;\r\n    text-align: left;\r\n    font-size: smaller;\r\n  }`, `\r\n    p {\r\n      padding: 4px;\r\n      white-space:nowrap;\r\n      display: inline-block;\r\n    }\r\n  `, `\r\n    mat-progress-spinner {\r\n      color: black;\r\n      display: inline-block;\r\n    }\r\n  `, `\r\n    span {\r\n      color: red;\r\n    }\r\n  `]\r\n\r\n})\r\n\r\nexport class StatusDisplay {\r\n  @Input() statusAlertType = 'info';\r\n  @Input() statusMsg = 'Initialize...';\r\n  @Input() statusWaiting =false;\r\n}\r\n\r\n\r\n@Component({\r\n  selector: 'app-uploadstatus',\r\n  template: `\r\n    <mat-progress-spinner [mode]=\"spinnerMode\" [color]=\"status\" [diameter]=\"30\" [strokeWidth]=\"5\" [value]=\"_value\" [matTooltip]=\"toolTipText\"></mat-progress-spinner>\r\n  `,\r\n  styles: [`:host {\r\n    flex: 1;\r\n  /* align-self: flex-start; */\r\n    /*display: inline; */\r\n    text-align: left;\r\n  }`,`mat-progress-spinner{\r\n      display: inline-block;\r\n  }`]\r\n})\r\nexport class UploadStatus {\r\n  private _awaitNewUpload=false;\r\n  spinnerMode:ProgressSpinnerMode = 'determinate';\r\n  _status!:string\r\n  _value = 100\r\n  displayValue:string|null=null\r\n  toolTipText:string=''\r\n\r\n  private _updateSpinner(){\r\n\r\n    let uplMsg;\r\n    if (this._awaitNewUpload || this._value === 0) {\r\n      this.spinnerMode = 'indeterminate'\r\n      this.displayValue='&nbsp;&nbsp;&nbsp;&nbsp;'\r\n      uplMsg='Preparing upload.'\r\n    } else {\r\n      this.spinnerMode = 'determinate'\r\n      this.displayValue=this._value+'%'\r\n      if(this._value===100){\r\n        uplMsg = 'Upload complete'\r\n      }else {\r\n        uplMsg = 'Upload progress: ' + this.displayValue\r\n      }\r\n    }\r\n    if(this.status==='warn'){\r\n      uplMsg='Upload error occurred. Please check your network connection. '+uplMsg\r\n    }\r\n    this.toolTipText=uplMsg\r\n  }\r\n\r\n  @Input()\r\n  set value(value: number) {\r\n    this._value = value;\r\n    this._updateSpinner()\r\n  };\r\n\r\n  @Input() set awaitNewUpload(awaitNewUpload:boolean){\r\n    this._awaitNewUpload=awaitNewUpload\r\n    this._updateSpinner()\r\n  }\r\n\r\n  @Input() set status(status:string){\r\n    this._status=status\r\n    this._updateSpinner()\r\n  }\r\n\r\n  get status():string{\r\n    return this._status\r\n  }\r\n\r\n}\r\n\r\n\r\n@Component({\r\n  selector: 'app-sprprogressdisplay',\r\n  template: `\r\n    <p>{{progressMsg}}</p>\r\n  `,\r\n  styles: [`:host {\r\n    flex: 1;\r\n  /* align-self: flex-start; */\r\n    /*display: inline; */\r\n      width: 100%;\r\n    text-align: left;\r\n  }`]\r\n})\r\nexport class ProgressDisplay {\r\n  progressMsg = '[itemcode]';\r\n}\r\n\r\n\r\n\r\n\r\nexport class TransportActions {\r\n  startAction: Action<void>;\r\n  stopAction: Action<void>;\r\n  nextAction: Action<void>;\r\n  fwdNextAction: Action<void>;\r\n  pauseAction: Action<void>;\r\n  fwdAction: Action<void>;\r\n  bwdAction: Action<void>;\r\n\r\n  constructor() {\r\n    this.startAction = new Action('Start');\r\n    this.stopAction = new Action('Stop');\r\n    this.nextAction = new Action('Next');\r\n    this.pauseAction = new Action('Pause');\r\n    this.fwdNextAction = new Action('Next recording');\r\n    this.fwdAction = new Action('Forward');\r\n    this.bwdAction = new Action('Backward');\r\n\r\n  }\r\n}\r\n\r\n@Component({\r\n\r\n  selector: 'app-sprtransport',\r\n\r\n  template: `\r\n    <button id=\"bwdBtn\" *ngIf=\"navigationEnabled\"  (click)=\"actions.bwdAction.perform()\" [disabled]=\"bwdDisabled()\"\r\n            mat-raised-button>\r\n      <mat-icon>chevron_left</mat-icon>\r\n    </button>\r\n    <button (click)=\"startStopNextPerform()\" [disabled]=\"startDisabled() && stopDisabled() && nextDisabled()\"  mat-raised-button>\r\n      <mat-icon [style.color]=\"startStopNextIconColor()\">{{startStopNextIconName()}}</mat-icon><mat-icon *ngIf=\"!nextDisabled()\" [style.color]=\"nextDisabled() ? 'grey' : 'black'\">chevron_right</mat-icon>\r\n      <span fxShow.xs=\"false\">{{startStopNextName()}}</span>\r\n    </button>\r\n    <button *ngIf=\"pausingEnabled\" (click)=\"actions.pauseAction.perform()\" [disabled]=\"pauseDisabled()\" mat-raised-button>\r\n      <mat-icon>pause</mat-icon>\r\n      <span fxShow.xs=\"false\">Pause</span>\r\n    </button>\r\n    <button id=\"fwdNextBtn\" *ngIf=\"navigationEnabled\" fxHide.xs (click)=\"actions.fwdNextAction.perform()\" [disabled]=\"fwdNextDisabled()\" mat-raised-button>\r\n      <mat-icon>redo</mat-icon>\r\n    </button>\r\n    <button id=\"fwdBtn\" *ngIf=\"navigationEnabled\"  (click)=\"actions.fwdAction.perform()\" [disabled]=\"fwdDisabled()\" mat-raised-button>\r\n      <mat-icon>chevron_right</mat-icon>\r\n    </button>\r\n\r\n  `,\r\n  styles: [`:host {\r\n    flex: 20;\r\n    align-self: center;\r\n    width: 100%;\r\n    text-align: center;\r\n    align-content: center;\r\n    margin: 0;\r\n\r\n  }`, `\r\n    div {\r\n      display: inline;\r\n      flex: 0;\r\n    }`,`\r\n     button {\r\n       touch-action: manipulation;\r\n     }`\r\n  ]\r\n\r\n})\r\nexport class TransportPanel {\r\n\r\n  @Input() readonly!:boolean;\r\n  @Input() actions!: TransportActions;\r\n  @Input() navigationEnabled=true;\r\n  @Input() pausingEnabled=true;\r\n\r\n  startStopNextButtonName!:string;\r\n  startStopNextButtonIconName!:string;\r\n\r\n  startDisabled() {\r\n    return !this.actions || this.readonly || this.actions.startAction.disabled\r\n  }\r\n\r\n  stopDisabled() {\r\n    return !this.actions || this.actions.stopAction.disabled\r\n  }\r\n\r\n  nextDisabled() {\r\n    return !this.actions || this.actions.nextAction.disabled || !this.navigationEnabled;\r\n  }\r\n\r\n  pauseDisabled() {\r\n    return !this.actions || this.actions.pauseAction.disabled || !this.pausingEnabled;\r\n  }\r\n\r\n  fwdDisabled() {\r\n    return !this.actions || this.actions.fwdAction.disabled || !this.navigationEnabled;\r\n  }\r\n\r\n  fwdNextDisabled() {\r\n    return !this.actions || this.actions.fwdNextAction.disabled || !this.navigationEnabled;\r\n  }\r\n\r\n  bwdDisabled() {\r\n    return !this.actions || this.actions.bwdAction.disabled || !this.navigationEnabled;\r\n  }\r\n\r\n    startStopNextName():string{\r\n        if(!this.nextDisabled()){\r\n            this.startStopNextButtonName= \"Next\"\r\n        }else 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      }else if(!this.nextDisabled()){\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() || !this.nextDisabled()){\r\n            return \"yellow\"\r\n        }else{\r\n            return \"grey\";\r\n        }\r\n    }\r\n\r\n  startStopNextPerform(){\r\n    if(!this.startDisabled()){\r\n      this.actions.startAction.perform();\r\n    }else if(!this.stopDisabled()){\r\n      this.actions.stopAction.perform();\r\n    }else if(!this.nextDisabled()){\r\n      this.actions.nextAction.perform();\r\n    }\r\n  }\r\n\r\n}\r\n\r\n@Component({\r\n\r\n  selector: 'app-readystateindicator',\r\n\r\n  template: `\r\n        <mat-icon [matTooltip]=\"readyStateToolTip\">{{hourGlassIconName}}</mat-icon>\r\n  `,\r\n  styles: []\r\n})\r\nexport class ReadyStateIndicator {\r\n  _ready=true\r\n  hourGlassIconName='hourglass_empty'\r\n  readyStateToolTip:string=''\r\n\r\n  constructor() {}\r\n\r\n  @Input() set ready(ready:boolean){\r\n    this._ready=ready\r\n    this.hourGlassIconName=this._ready?'hourglass_empty':'hourglass_full'\r\n    this.readyStateToolTip=this._ready?'Audio processing and upload done. You can leave the page without data loss.':'Please wait until audio processing and upload have finished. Please do not leave the page.'\r\n  }\r\n\r\n  get ready():boolean{\r\n    return this._ready\r\n  }\r\n\r\n}\r\n\r\n@Component({\r\n\r\n  selector: 'app-sprcontrolpanel',\r\n\r\n  template: `\r\n    <div fxHide.xs  fxLayout=\"row\" >\r\n     <app-sprstatusdisplay fxFlex=\"0 0 0\" [statusMsg]=\"statusMsg\" [statusAlertType]=\"statusAlertType\" [statusWaiting]=\"statusWaiting\"\r\n                          class=\"hidden-xs\"></app-sprstatusdisplay>\r\n      <app-sprtransport fxFlex=\"10 0 0\" [readonly]=\"readonly\" [actions]=\"transportActions\" [navigationEnabled]=\"navigationEnabled\"></app-sprtransport>\r\n      <app-uploadstatus fxFlex=\"0 0 0\" *ngIf=\"enableUploadRecordings\" [value]=\"uploadProgress\"\r\n                      [status]=\"uploadStatus\" [awaitNewUpload]=\"processing\"></app-uploadstatus>\r\n      <app-readystateindicator [ready]=\"_ready\"></app-readystateindicator>\r\n    </div>\r\n    <div fxShow.xs fxHide fxLayout=\"column\">\r\n      <div fxLayout=\"row\" fxFlexFill>\r\n       <app-sprstatusdisplay fxFlex=\"10 0 0\" fxFlexAlign=\"left\" [statusMsg]=\"statusMsg\" [statusAlertType]=\"statusAlertType\" [statusWaiting]=\"statusWaiting\"\r\n                            class=\"hidden-xs\"></app-sprstatusdisplay>\r\n       <app-uploadstatus fxFlex=\"0 0 0\" *ngIf=\"enableUploadRecordings\" [value]=\"uploadProgress\"\r\n                        [status]=\"uploadStatus\" [awaitNewUpload]=\"processing\"></app-uploadstatus>\r\n        <app-readystateindicator [ready]=\"_ready\"></app-readystateindicator>\r\n      </div>\r\n      <app-sprtransport [readonly]=\"readonly\" [actions]=\"transportActions\" [navigationEnabled]=\"navigationEnabled\"></app-sprtransport>\r\n\r\n    </div>\r\n  `,\r\n  styles: [`div {\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  }`]\r\n})\r\nexport class ControlPanel {\r\n  @ViewChild(StatusDisplay, { static: true }) statusDisplay!: StatusDisplay;\r\n  @ViewChild(TransportPanel, { static: true }) transportPanel!: TransportPanel;\r\n\r\n  @Input() readonly!:boolean\r\n  @Input() transportActions!: TransportActions\r\n  @Input() processing=false\r\n  @Input() statusMsg!: string;\r\n  @Input() statusAlertType!: string;\r\n  @Input() statusWaiting!: boolean;\r\n  @Input() uploadStatus!: string;\r\n  @Input() uploadProgress!: number;\r\n  @Input() currentRecording: AudioBuffer| null| undefined;\r\n  @Input() enableUploadRecordings!: boolean;\r\n  @Input() navigationEnabled=true;\r\n\r\n  _ready=true\r\n\r\n  constructor(public dialog: MatDialog) {\r\n\r\n  }\r\n\r\n  @Input() set ready(ready:boolean){\r\n    this._ready=ready\r\n  }\r\n\r\n  get ready():boolean{\r\n    return this._ready\r\n  }\r\n\r\n}\r\n\r\n\r\n"]}