speaker-calibration 2.2.212 → 2.2.214

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 (142) hide show
  1. package/.eslintignore +71 -71
  2. package/.eslintrc.json +40 -40
  3. package/.github/workflows/update-phrases.yml +37 -0
  4. package/.prettierignore +69 -69
  5. package/.prettierrc +14 -14
  6. package/LICENSE +20 -20
  7. package/README.md +133 -133
  8. package/__mocks__/fileMock.js +1 -1
  9. package/__mocks__/styleMock.js +1 -1
  10. package/babel.config.js +3 -3
  11. package/coverage/clover.xml +71 -71
  12. package/coverage/coverage-final.json +224 -224
  13. package/coverage/lcov-report/PythonServerInterface.js.html +265 -265
  14. package/coverage/lcov-report/base.css +354 -354
  15. package/coverage/lcov-report/block-navigation.js +82 -82
  16. package/coverage/lcov-report/favicon.png +0 -0
  17. package/coverage/lcov-report/index.html +123 -123
  18. package/coverage/lcov-report/prettify.css +101 -101
  19. package/coverage/lcov-report/prettify.js +937 -937
  20. package/coverage/lcov-report/sort-arrow-sprite.png +0 -0
  21. package/coverage/lcov-report/sorter.js +189 -189
  22. package/coverage/lcov-report/src/index.html +121 -121
  23. package/coverage/lcov-report/src/server/PythonServerInterface.js.html +268 -268
  24. package/coverage/lcov-report/src/server/index.html +123 -123
  25. package/coverage/lcov-report/src/tasks/audioCalibrator.js.html +499 -499
  26. package/coverage/lcov-report/src/tasks/audioRecorder.js.html +412 -412
  27. package/coverage/lcov-report/src/tasks/index.html +143 -143
  28. package/coverage/lcov-report/src/tasks/volume/index.html +123 -123
  29. package/coverage/lcov-report/src/tasks/volume/volume.js.html +409 -409
  30. package/coverage/lcov-report/src/utils.js.html +172 -172
  31. package/coverage/lcov.info +91 -91
  32. package/dist/Procfile +0 -0
  33. package/dist/example/NoSleep.min.js +1 -1
  34. package/dist/example/credentials.json.gpg +0 -0
  35. package/dist/example/fetch-languages-sheets.js +77 -77
  36. package/dist/example/i18n.js +27366 -27366
  37. package/dist/example/index.html +47 -47
  38. package/dist/example/listener.html +79 -79
  39. package/dist/example/server.js +51 -51
  40. package/dist/example/speaker.html +145 -145
  41. package/dist/example/speakerUI.js +273 -273
  42. package/dist/example/styles.css +99 -99
  43. package/dist/listener.js +3459 -0
  44. package/dist/main.js +2420 -207
  45. package/dist/main.js.LICENSE.txt +0 -0
  46. package/dist/mlsGen.js +6814 -6814
  47. package/dist/mlsGen.wasm +0 -0
  48. package/dist/package-lock.json +1018 -1018
  49. package/dist/package.json +18 -18
  50. package/doc/AudioCalibrator.html +417 -417
  51. package/doc/AudioPeer.html +251 -251
  52. package/doc/AudioRecorder.html +195 -195
  53. package/doc/ImpulseResponse.html +215 -215
  54. package/doc/Listener.html +308 -308
  55. package/doc/MlsGenInterface.html +226 -226
  56. package/doc/MyEventEmitter.html +274 -274
  57. package/doc/PythonServerAPI.html +109 -109
  58. package/doc/Speaker-Calibration-UML-Diagram.png +0 -0
  59. package/doc/Speaker.html +276 -276
  60. package/doc/Takes%20a%20target%20element%20where%20html%20elements%20will%20be%20appended..html +128 -128
  61. package/doc/Takes%20the%20url%20of%20the%20current%20site%0Aand%20a%20target%20element%20where%20html%20elements%20will%20be%20appended..html +138 -138
  62. package/doc/Takes%20the%20url%20of%20the%20current%20site%20and%20a%20target%20element%20where%20html%20elements%20will%20be%20appended..html +137 -137
  63. package/doc/Volume.html +88 -88
  64. package/doc/audioCalibrator.js.html +179 -179
  65. package/doc/audioPeer.js.html +175 -175
  66. package/doc/audioRecorder.js.html +163 -163
  67. package/doc/creates%20a%20new%20AudioRecorder%20instance.%20%0ASets%20up%20the%20audio%20context%20and%20file%20reader..html +114 -114
  68. package/doc/fonts/OpenSans-Bold-webfont.eot +0 -0
  69. package/doc/fonts/OpenSans-Bold-webfont.svg +1829 -1829
  70. package/doc/fonts/OpenSans-Bold-webfont.woff +0 -0
  71. package/doc/fonts/OpenSans-BoldItalic-webfont.eot +0 -0
  72. package/doc/fonts/OpenSans-BoldItalic-webfont.svg +1829 -1829
  73. package/doc/fonts/OpenSans-BoldItalic-webfont.woff +0 -0
  74. package/doc/fonts/OpenSans-Italic-webfont.eot +0 -0
  75. package/doc/fonts/OpenSans-Italic-webfont.svg +1829 -1829
  76. package/doc/fonts/OpenSans-Italic-webfont.woff +0 -0
  77. package/doc/fonts/OpenSans-Light-webfont.eot +0 -0
  78. package/doc/fonts/OpenSans-Light-webfont.svg +1830 -1830
  79. package/doc/fonts/OpenSans-Light-webfont.woff +0 -0
  80. package/doc/fonts/OpenSans-LightItalic-webfont.eot +0 -0
  81. package/doc/fonts/OpenSans-LightItalic-webfont.svg +1834 -1834
  82. package/doc/fonts/OpenSans-LightItalic-webfont.woff +0 -0
  83. package/doc/fonts/OpenSans-Regular-webfont.eot +0 -0
  84. package/doc/fonts/OpenSans-Regular-webfont.svg +1830 -1830
  85. package/doc/fonts/OpenSans-Regular-webfont.woff +0 -0
  86. package/doc/global.html +308 -308
  87. package/doc/index.html +58 -58
  88. package/doc/listener.js.html +170 -170
  89. package/doc/mlsGen_mlsGenInterface.js.html +117 -117
  90. package/doc/myEventEmitter.js.html +124 -124
  91. package/doc/peer-connection_audioPeer.js.html +188 -188
  92. package/doc/peer-connection_listener.js.html +311 -311
  93. package/doc/peer-connection_speaker.js.html +381 -381
  94. package/doc/sc-activity-diagram.png +0 -0
  95. package/doc/scripts/linenumber.js +25 -25
  96. package/doc/scripts/prettify/Apache-License-2.0.txt +202 -202
  97. package/doc/scripts/prettify/lang-css.js +24 -24
  98. package/doc/scripts/prettify/prettify.js +640 -640
  99. package/doc/server_PythonServerAPI.js.html +160 -160
  100. package/doc/speaker.js.html +248 -248
  101. package/doc/styles/jsdoc-default.css +371 -371
  102. package/doc/styles/prettify-jsdoc.css +111 -111
  103. package/doc/styles/prettify-tomorrow.css +163 -163
  104. package/doc/tasks_audioCalibrator.js.html +207 -207
  105. package/doc/tasks_audioRecorder.js.html +190 -190
  106. package/doc/tasks_impulse-response_impulseResponse.js.html +442 -442
  107. package/doc/tasks_impulse-response_mlsGen_mlsGenInterface.js.html +175 -175
  108. package/doc/tasks_volume_volume.js.html +185 -185
  109. package/doc/utils.js.html +105 -105
  110. package/jest.config.js +173 -173
  111. package/makefile +74 -0
  112. package/netlify.toml +26 -26
  113. package/package.json +78 -73
  114. package/src/config/firebase.js +26 -26
  115. package/src/index.html +21 -21
  116. package/{dist/example → src/listener-app}/listener.js +153 -152
  117. package/src/main.js +23 -23
  118. package/src/myEventEmitter.js +83 -83
  119. package/src/peer-connection/audioPeer.js +183 -183
  120. package/src/peer-connection/listener.js +366 -367
  121. package/src/peer-connection/peerErrors.js +25 -25
  122. package/src/peer-connection/speaker.js +743 -738
  123. package/src/powerCheck.js +98 -98
  124. package/src/server/PythonServerAPI.js +869 -869
  125. package/src/tasks/audioCalibrator.js +351 -351
  126. package/src/tasks/audioRecorder.js +315 -315
  127. package/src/tasks/combination/combination.js +3160 -3031
  128. package/src/tasks/combination/mlsGen/mlsGen.cpp +98 -98
  129. package/src/tasks/combination/mlsGen/mlsGen.hpp +303 -303
  130. package/src/tasks/combination/mlsGen/mlsGenInterface.js +131 -131
  131. package/src/tasks/combination/mlsGen/mlsGenTest.cpp +180 -180
  132. package/src/tasks/impulse-response/impulseResponse.js +610 -610
  133. package/src/tasks/impulse-response/mlsGen/mlsGen.cpp +98 -98
  134. package/src/tasks/impulse-response/mlsGen/mlsGen.hpp +303 -303
  135. package/src/tasks/impulse-response/mlsGen/mlsGenInterface.js +131 -131
  136. package/src/tasks/impulse-response/mlsGen/mlsGenTest.cpp +180 -180
  137. package/src/tasks/volume/volume.cpp +2 -2
  138. package/src/tasks/volume/volume.hpp +22 -22
  139. package/src/tasks/volume/volume.js +279 -279
  140. package/src/utils.js +205 -205
  141. package/webpack.config.js +64 -37
  142. package/.gitignore +0 -81
@@ -1,351 +1,351 @@
1
- /* eslint-disable no-await-in-loop */
2
- import AudioRecorder from './audioRecorder';
3
- import PythonServerAPI from '../server/PythonServerAPI';
4
- import {sleep, saveToCSV} from '../utils';
5
-
6
- /**
7
- * .
8
- * .
9
- * .
10
- * Provides methods for calibrating the user's speakers
11
- *
12
- * @extends AudioRecorder
13
- */
14
- class AudioCalibrator extends AudioRecorder {
15
- /**
16
- *
17
- * @param numCaptures
18
- * @param numMLSPerCapture
19
- * @example
20
- */
21
- constructor(numCaptures = 1, numMLSPerCapture = 1) {
22
- super();
23
- this.numCaptures = numCaptures;
24
- this.numMLSPerCapture = numMLSPerCapture;
25
- this.pyServerAPI = new PythonServerAPI();
26
- this.currentTime = 0;
27
- }
28
-
29
- /** @private */
30
- isCalibrating = false;
31
-
32
- /** @private */
33
- sourceAudioContext;
34
-
35
- /** @private */
36
- sourceAudioContextConvolved;
37
-
38
- /** @protected */
39
- numCalibratingRounds = 1;
40
-
41
- /** @protected */
42
- numSuccessfulCaptured = 0;
43
-
44
- /** @private */
45
- sourceSamplingRate;
46
-
47
- /** @protected */
48
- calibrationNodes = [];
49
-
50
- /** @protected */
51
- calibrationNodesConvolved = [];
52
-
53
- /** @protected */
54
- localAudio;
55
-
56
- /** @private */
57
- startTime;
58
-
59
- numCalibratingRoundsCompleted=0;
60
- /**
61
- * Called when a call is received.
62
- * Creates a local audio DOM element and attaches it to the page.
63
- *
64
- * @param targetElement
65
- * @example
66
- */
67
- createLocalAudio = targetElement => {
68
- this.localAudio = document.createElement('audio');
69
- this.localAudio.setAttribute('id', 'localAudio');
70
- targetElement.appendChild(this.localAudio);
71
- };
72
-
73
- addTimeStamp = taskName => {
74
- const currentTime = new Date().getTime(); // Current time in milliseconds
75
- const elapsedTime = (currentTime - this.startTime) / 1000; // Total time since start in seconds
76
- const stepDuration = elapsedTime - this.currentTime; // Time taken for the current step
77
-
78
- this.currentTime = elapsedTime; // Update for the next step
79
-
80
- // Round to 1 decimal place for consistent formatting
81
-
82
- ;
83
-
84
- // Log the timestamp with aligned bars
85
- this.timeStamp.push(
86
- `SOUND Total: ${elapsedTime.toFixed(1)} s Step: ${stepDuration.toFixed(1)} s ${taskName}`
87
- );
88
- };
89
-
90
-
91
- recordBackground = async (
92
- stream,
93
- loopCondition = () => false,
94
- duringRecord = async () => {},
95
- afterRecord = async () => {},
96
- mode,
97
- checkRec
98
- ) => {
99
- console.warn('before recording background noise');
100
- // calibration loop
101
- while (loopCondition()) {
102
- // start recording
103
- console.warn('startRecording');
104
- await this.startRecording(stream);
105
-
106
- // do something during the recording such as sleep n amount of time
107
- console.warn('duringRecord');
108
- await duringRecord();
109
-
110
- // when done, stop recording
111
- console.warn('stopRecording');
112
- await this.stopRecording(mode, checkRec);
113
-
114
- // do something after recording such as start processing values
115
- console.warn('afterRecord');
116
- await afterRecord();
117
-
118
- // eslint-disable-next-line no-await-in-loop
119
- await sleep(1);
120
- }
121
- };
122
-
123
- /**
124
- *
125
- * @param {MediaStream} stream
126
- * @param {Function} playCalibrationAudio - (async) function that plays the calibration audio
127
- * @param {*} beforePlay - (async) function that is called before playing the audio
128
- * @param {*} beforeRecord - (async) function that is called before recording
129
- * @param {*} duringRecord - (async) function that is called while recording
130
- * @param {*} afterRecord - (async) function that is called after recording
131
- * @example
132
- */
133
- calibrationSteps = async (
134
- stream,
135
- playCalibrationAudio,
136
- beforePlay = async () => {},
137
- beforeRecord = async () => {},
138
- loopCondition = () => false,
139
- duringRecord = async () => {},
140
- afterRecord = async () => {},
141
- mode,
142
- checkRec
143
- ) => {
144
- // if it finished 2 attempts, it move to next iteration so reset numSuccessfulCaptured
145
- if (this.numSuccessfulCaptured >=2) {
146
- this.numSuccessfulCaptured = 0;
147
- }
148
-
149
- // do something before playing such as using the MLS to fill the buffers
150
- console.warn('beforePlay');
151
- await beforePlay();
152
-
153
- // play calibration audio
154
- console.warn('playCalibrationAudio');
155
- playCalibrationAudio();
156
-
157
- // do something before recording such as awaiting a certain amount of time
158
- console.warn('beforeRecord');
159
- await beforeRecord();
160
- const totalSec = this._calibrateSoundBurstPreSec + (this.numMLSPerCapture - this.num_mls_to_skip) * this._calibrateSoundBurstSec + this._calibrateSoundBurstPostSec;
161
-
162
-
163
- // calibration loop
164
- while (loopCondition()) {
165
- if (this.isCalibrating) break;
166
- // start recording
167
- console.warn('startRecording');
168
- await this.startRecording(stream);
169
-
170
- if (this.isCalibrating) break;
171
- // do something during the recording such as sleep n amount of time
172
- console.warn('duringRecord');
173
- await duringRecord();
174
-
175
- if (this.isCalibrating) break;
176
- // when done, stop recording
177
- console.warn('stopRecording');
178
- await this.stopRecording(mode, checkRec);
179
-
180
- if (this.isCalibrating) break;
181
- // do something after recording such as start processing values
182
- console.warn('afterRecord');
183
- await afterRecord();
184
-
185
- // eslint-disable-next-line no-await-in-loop
186
- await sleep(1);
187
- }
188
- };
189
-
190
- /**
191
- *
192
- * @param {MediaStream} stream
193
- * @param {Function} playCalibrationAudio - (async) function that plays the calibration audio
194
- * @param {*} beforeRecord - (async) function that is called before recording
195
- * @param {*} afterRecord - (async) function that is called after recording
196
- * @param {Number} gainValue - the gain value to set the gain node to
197
- */
198
- volumeCalibrationSteps = async (
199
- stream,
200
- playCalibrationAudio,
201
- beforeRecord = () => {},
202
- afterRecord = () => {},
203
- gainValue,
204
- lCalib = 104.92978421490648,
205
- checkRec,
206
- checkSD,
207
- maxSD
208
- ) => {
209
- this.numCalibratingRoundsCompleted = 0;
210
- this.numCalibratingRounds = 2;
211
- console.log("maxSD in VolumeCaibrationSteps: ", maxSD, '0' >= maxSD);
212
- // calibration loop
213
- while (!this.isCalibrating && this.numCalibratingRoundsCompleted < this.numCalibratingRounds) {
214
- if (this.isCalibrating) break;
215
- // before recording
216
- await beforeRecord(gainValue);
217
- if (this.isCalibrating) break;
218
- // start recording
219
- await this.startRecording(stream);
220
- if (this.isCalibrating) break;
221
- // play calibration audio
222
- console.log(`Calibration Round ${this.numCalibratingRoundsCompleted}`);
223
- await playCalibrationAudio();
224
- if (this.isCalibrating) break;
225
- // when done, stop recording
226
- console.log('Calibration Round Complete');
227
- await this.stopRecording('volume', checkRec);
228
- if (this.isCalibrating) break;
229
- // after recording
230
- await afterRecord(lCalib);
231
- const sd = await checkSD();
232
- if (sd <= maxSD) {
233
- console.log(`SD =${sd}, less than calibrateSound1000HzMaxSD_dB=${maxSD}`);
234
- this.numCalibratingRoundsCompleted += 2;
235
- } else {
236
- // if exist the maxSD do it one more time and only one more time
237
- console.log(`SD =${sd}, greater than calibrateSound1000HzMaxSD_dB=${maxSD}`);
238
- this.numCalibratingRoundsCompleted += 1;
239
- }
240
- this.calibrationNodes = [];
241
-
242
- // eslint-disable-next-line no-await-in-loop
243
- await sleep(2);
244
- }
245
- };
246
-
247
- /**
248
- * Getter for the isCalibrating property.
249
- *
250
- * @public
251
- * @returns - True if the audio is being calibrated, false otherwise.
252
- * @example
253
- */
254
- getCalibrationStatus = () => this.isCalibrating;
255
-
256
- /** .
257
- * .
258
- * .
259
- * Set the sampling rate to the value received from the listener
260
- *
261
- * @param {*} sinkSamplingRate
262
- * @param samplingRate
263
- * @example
264
- */
265
- setSamplingRates = samplingRate => {
266
- this.sinkSamplingRate = samplingRate;
267
- this.sourceSamplingRate = samplingRate;
268
-
269
- // this.emit('update', {message: `sampling at ${samplingRate}Hz...`});
270
- };
271
-
272
- setSampleSize = sampleSize => {
273
- this.sampleSize = sampleSize;
274
- };
275
-
276
- setFlags = flags => {
277
- this.flags = flags;
278
- }
279
-
280
- sampleRatesSet = () => this.sourceSamplingRate && this.sinkSamplingRate;
281
-
282
- addCalibrationNode = node => {
283
- this.calibrationNodes.push(node);
284
- };
285
-
286
- addCalibrationNodeConvolved = node => {
287
- this.calibrationNodesConvolved.push(node);
288
- };
289
-
290
- makeNewSourceAudioContext = () => {
291
- const options = {
292
- sampleRate: this.sourceSamplingRate,
293
- };
294
-
295
- this.sourceAudioContext = new (window.AudioContext ||
296
- window.webkitAudioContext ||
297
- window.audioContext)(options);
298
-
299
- return this.sourceAudioContext;
300
- };
301
-
302
- makeNewSourceAudioContextConvolved = () => {
303
- const options = {
304
- sampleRate: this.sourceSamplingRate,
305
- };
306
-
307
- this.sourceAudioContextConvolved = new (window.AudioContext ||
308
- window.webkitAudioContext ||
309
- window.audioContext)(options);
310
-
311
- return this.sourceAudioContextConvolved;
312
- };
313
-
314
- /** .
315
- * .
316
- * .
317
- * Download the result of the calibration roudns
318
- *
319
- * @example
320
- */
321
- downloadData = () => {
322
- const recordings = this.getAllRecordedSignals();
323
- const i = recordings.length - 1;
324
- saveToCSV(recordings[i], `recordedMLSignal_${i}_unconvolved.csv`);
325
- };
326
- downloadSingleUnfilteredRecording = () => {
327
- const recordings = this.getAllUnfilteredRecordedSignals();
328
- saveToCSV(recordings[recordings.length - 1], `recordedMLSignal_unconvolved.csv`);
329
- };
330
- downloadSingleFilteredRecording = () => {
331
- const recordings = this.getAllFilteredRecordedSignals();
332
- console.log('Single filtered recording should be of length: ' + recordings[0].length);
333
- saveToCSV(recordings[0], `recordedMLSignal_convolved.csv`);
334
- };
335
- downloadUnfilteredRecordings = () => {
336
- const recordings = this.getAllRecordedSignals();
337
- console.log('unfilterd download?');
338
- for (let i = 0; i < recordings.length; i++) {
339
- console.log(i);
340
- saveToCSV(recordings[i], `recordedMLSignal_${i}_unconvolved.csv`);
341
- }
342
- };
343
- downloadFilteredRecordings = () => {
344
- const recordings = this.getAllFilteredRecordedSignals();
345
- for (let i = 0; i < recordings.length; i++) {
346
- saveToCSV(recordings[i], `recordedMLSignal_${i}_convolved.csv`);
347
- }
348
- };
349
- }
350
-
351
- export default AudioCalibrator;
1
+ /* eslint-disable no-await-in-loop */
2
+ import AudioRecorder from './audioRecorder';
3
+ import PythonServerAPI from '../server/PythonServerAPI';
4
+ import {sleep, saveToCSV} from '../utils';
5
+
6
+ /**
7
+ * .
8
+ * .
9
+ * .
10
+ * Provides methods for calibrating the user's speakers
11
+ *
12
+ * @extends AudioRecorder
13
+ */
14
+ class AudioCalibrator extends AudioRecorder {
15
+ /**
16
+ *
17
+ * @param numCaptures
18
+ * @param numMLSPerCapture
19
+ * @example
20
+ */
21
+ constructor(numCaptures = 1, numMLSPerCapture = 1) {
22
+ super();
23
+ this.numCaptures = numCaptures;
24
+ this.numMLSPerCapture = numMLSPerCapture;
25
+ this.pyServerAPI = new PythonServerAPI();
26
+ this.currentTime = 0;
27
+ }
28
+
29
+ /** @private */
30
+ isCalibrating = false;
31
+
32
+ /** @private */
33
+ sourceAudioContext;
34
+
35
+ /** @private */
36
+ sourceAudioContextConvolved;
37
+
38
+ /** @protected */
39
+ numCalibratingRounds = 1;
40
+
41
+ /** @protected */
42
+ numSuccessfulCaptured = 0;
43
+
44
+ /** @private */
45
+ sourceSamplingRate;
46
+
47
+ /** @protected */
48
+ calibrationNodes = [];
49
+
50
+ /** @protected */
51
+ calibrationNodesConvolved = [];
52
+
53
+ /** @protected */
54
+ localAudio;
55
+
56
+ /** @private */
57
+ startTime;
58
+
59
+ numCalibratingRoundsCompleted=0;
60
+ /**
61
+ * Called when a call is received.
62
+ * Creates a local audio DOM element and attaches it to the page.
63
+ *
64
+ * @param targetElement
65
+ * @example
66
+ */
67
+ createLocalAudio = targetElement => {
68
+ this.localAudio = document.createElement('audio');
69
+ this.localAudio.setAttribute('id', 'localAudio');
70
+ targetElement.appendChild(this.localAudio);
71
+ };
72
+
73
+ addTimeStamp = taskName => {
74
+ const currentTime = new Date().getTime(); // Current time in milliseconds
75
+ const elapsedTime = (currentTime - this.startTime) / 1000; // Total time since start in seconds
76
+ const stepDuration = elapsedTime - this.currentTime; // Time taken for the current step
77
+
78
+ this.currentTime = elapsedTime; // Update for the next step
79
+
80
+ // Round to 1 decimal place for consistent formatting
81
+
82
+ ;
83
+
84
+ // Log the timestamp with aligned bars
85
+ this.timeStamp.push(
86
+ `SOUND Total: ${elapsedTime.toFixed(1)} s Step: ${stepDuration.toFixed(1)} s ${taskName}`
87
+ );
88
+ };
89
+
90
+
91
+ recordBackground = async (
92
+ stream,
93
+ loopCondition = () => false,
94
+ duringRecord = async () => {},
95
+ afterRecord = async () => {},
96
+ mode,
97
+ checkRec
98
+ ) => {
99
+ console.warn('before recording background noise');
100
+ // calibration loop
101
+ while (loopCondition()) {
102
+ // start recording
103
+ console.warn('startRecording');
104
+ await this.startRecording(stream);
105
+
106
+ // do something during the recording such as sleep n amount of time
107
+ console.warn('duringRecord');
108
+ await duringRecord();
109
+
110
+ // when done, stop recording
111
+ console.warn('stopRecording');
112
+ await this.stopRecording(mode, checkRec);
113
+
114
+ // do something after recording such as start processing values
115
+ console.warn('afterRecord');
116
+ await afterRecord();
117
+
118
+ // eslint-disable-next-line no-await-in-loop
119
+ await sleep(1);
120
+ }
121
+ };
122
+
123
+ /**
124
+ *
125
+ * @param {MediaStream} stream
126
+ * @param {Function} playCalibrationAudio - (async) function that plays the calibration audio
127
+ * @param {*} beforePlay - (async) function that is called before playing the audio
128
+ * @param {*} beforeRecord - (async) function that is called before recording
129
+ * @param {*} duringRecord - (async) function that is called while recording
130
+ * @param {*} afterRecord - (async) function that is called after recording
131
+ * @example
132
+ */
133
+ calibrationSteps = async (
134
+ stream,
135
+ playCalibrationAudio,
136
+ beforePlay = async () => {},
137
+ beforeRecord = async () => {},
138
+ loopCondition = () => false,
139
+ duringRecord = async () => {},
140
+ afterRecord = async () => {},
141
+ mode,
142
+ checkRec
143
+ ) => {
144
+ // if it finished 2 attempts, it move to next iteration so reset numSuccessfulCaptured
145
+ if (this.numSuccessfulCaptured >=2) {
146
+ this.numSuccessfulCaptured = 0;
147
+ }
148
+
149
+ // do something before playing such as using the MLS to fill the buffers
150
+ console.warn('beforePlay');
151
+ await beforePlay();
152
+
153
+ // play calibration audio
154
+ console.warn('playCalibrationAudio');
155
+ playCalibrationAudio();
156
+
157
+ // do something before recording such as awaiting a certain amount of time
158
+ console.warn('beforeRecord');
159
+ await beforeRecord();
160
+ const totalSec = this._calibrateSoundBurstPreSec + (this.numMLSPerCapture - this.num_mls_to_skip) * this._calibrateSoundBurstSec + this._calibrateSoundBurstPostSec;
161
+
162
+
163
+ // calibration loop
164
+ while (loopCondition()) {
165
+ if (this.isCalibrating) break;
166
+ // start recording
167
+ console.warn('startRecording');
168
+ await this.startRecording(stream);
169
+
170
+ if (this.isCalibrating) break;
171
+ // do something during the recording such as sleep n amount of time
172
+ console.warn('duringRecord');
173
+ await duringRecord();
174
+
175
+ if (this.isCalibrating) break;
176
+ // when done, stop recording
177
+ console.warn('stopRecording');
178
+ await this.stopRecording(mode, checkRec);
179
+
180
+ if (this.isCalibrating) break;
181
+ // do something after recording such as start processing values
182
+ console.warn('afterRecord');
183
+ await afterRecord();
184
+
185
+ // eslint-disable-next-line no-await-in-loop
186
+ await sleep(1);
187
+ }
188
+ };
189
+
190
+ /**
191
+ *
192
+ * @param {MediaStream} stream
193
+ * @param {Function} playCalibrationAudio - (async) function that plays the calibration audio
194
+ * @param {*} beforeRecord - (async) function that is called before recording
195
+ * @param {*} afterRecord - (async) function that is called after recording
196
+ * @param {Number} gainValue - the gain value to set the gain node to
197
+ */
198
+ volumeCalibrationSteps = async (
199
+ stream,
200
+ playCalibrationAudio,
201
+ beforeRecord = () => {},
202
+ afterRecord = () => {},
203
+ gainValue,
204
+ lCalib = 104.92978421490648,
205
+ checkRec,
206
+ checkSD,
207
+ maxSD
208
+ ) => {
209
+ this.numCalibratingRoundsCompleted = 0;
210
+ this.numCalibratingRounds = 2;
211
+ console.log("maxSD in VolumeCaibrationSteps: ", maxSD, '0' >= maxSD);
212
+ // calibration loop
213
+ while (!this.isCalibrating && this.numCalibratingRoundsCompleted < this.numCalibratingRounds) {
214
+ if (this.isCalibrating) break;
215
+ // before recording
216
+ await beforeRecord(gainValue);
217
+ if (this.isCalibrating) break;
218
+ // start recording
219
+ await this.startRecording(stream);
220
+ if (this.isCalibrating) break;
221
+ // play calibration audio
222
+ console.log(`Calibration Round ${this.numCalibratingRoundsCompleted}`);
223
+ await playCalibrationAudio();
224
+ if (this.isCalibrating) break;
225
+ // when done, stop recording
226
+ console.log('Calibration Round Complete');
227
+ await this.stopRecording('volume', checkRec);
228
+ if (this.isCalibrating) break;
229
+ // after recording
230
+ await afterRecord(lCalib);
231
+ const sd = await checkSD();
232
+ if (sd <= maxSD) {
233
+ console.log(`SD =${sd}, less than calibrateSound1000HzMaxSD_dB=${maxSD}`);
234
+ this.numCalibratingRoundsCompleted += 2;
235
+ } else {
236
+ // if exist the maxSD do it one more time and only one more time
237
+ console.log(`SD =${sd}, greater than calibrateSound1000HzMaxSD_dB=${maxSD}`);
238
+ this.numCalibratingRoundsCompleted += 1;
239
+ }
240
+ this.calibrationNodes = [];
241
+
242
+ // eslint-disable-next-line no-await-in-loop
243
+ await sleep(2);
244
+ }
245
+ };
246
+
247
+ /**
248
+ * Getter for the isCalibrating property.
249
+ *
250
+ * @public
251
+ * @returns - True if the audio is being calibrated, false otherwise.
252
+ * @example
253
+ */
254
+ getCalibrationStatus = () => this.isCalibrating;
255
+
256
+ /** .
257
+ * .
258
+ * .
259
+ * Set the sampling rate to the value received from the listener
260
+ *
261
+ * @param {*} sinkSamplingRate
262
+ * @param samplingRate
263
+ * @example
264
+ */
265
+ setSamplingRates = samplingRate => {
266
+ this.sinkSamplingRate = samplingRate;
267
+ this.sourceSamplingRate = samplingRate;
268
+
269
+ // this.emit('update', {message: `sampling at ${samplingRate}Hz...`});
270
+ };
271
+
272
+ setSampleSize = sampleSize => {
273
+ this.sampleSize = sampleSize;
274
+ };
275
+
276
+ setFlags = flags => {
277
+ this.flags = flags;
278
+ }
279
+
280
+ sampleRatesSet = () => this.sourceSamplingRate && this.sinkSamplingRate;
281
+
282
+ addCalibrationNode = node => {
283
+ this.calibrationNodes.push(node);
284
+ };
285
+
286
+ addCalibrationNodeConvolved = node => {
287
+ this.calibrationNodesConvolved.push(node);
288
+ };
289
+
290
+ makeNewSourceAudioContext = () => {
291
+ const options = {
292
+ sampleRate: this.sourceSamplingRate,
293
+ };
294
+
295
+ this.sourceAudioContext = new (window.AudioContext ||
296
+ window.webkitAudioContext ||
297
+ window.audioContext)(options);
298
+
299
+ return this.sourceAudioContext;
300
+ };
301
+
302
+ makeNewSourceAudioContextConvolved = () => {
303
+ const options = {
304
+ sampleRate: this.sourceSamplingRate,
305
+ };
306
+
307
+ this.sourceAudioContextConvolved = new (window.AudioContext ||
308
+ window.webkitAudioContext ||
309
+ window.audioContext)(options);
310
+
311
+ return this.sourceAudioContextConvolved;
312
+ };
313
+
314
+ /** .
315
+ * .
316
+ * .
317
+ * Download the result of the calibration roudns
318
+ *
319
+ * @example
320
+ */
321
+ downloadData = () => {
322
+ const recordings = this.getAllRecordedSignals();
323
+ const i = recordings.length - 1;
324
+ saveToCSV(recordings[i], `recordedMLSignal_${i}_unconvolved.csv`);
325
+ };
326
+ downloadSingleUnfilteredRecording = () => {
327
+ const recordings = this.getAllUnfilteredRecordedSignals();
328
+ saveToCSV(recordings[recordings.length - 1], `recordedMLSignal_unconvolved.csv`);
329
+ };
330
+ downloadSingleFilteredRecording = () => {
331
+ const recordings = this.getAllFilteredRecordedSignals();
332
+ console.log('Single filtered recording should be of length: ' + recordings[0].length);
333
+ saveToCSV(recordings[0], `recordedMLSignal_convolved.csv`);
334
+ };
335
+ downloadUnfilteredRecordings = () => {
336
+ const recordings = this.getAllRecordedSignals();
337
+ console.log('unfilterd download?');
338
+ for (let i = 0; i < recordings.length; i++) {
339
+ console.log(i);
340
+ saveToCSV(recordings[i], `recordedMLSignal_${i}_unconvolved.csv`);
341
+ }
342
+ };
343
+ downloadFilteredRecordings = () => {
344
+ const recordings = this.getAllFilteredRecordedSignals();
345
+ for (let i = 0; i < recordings.length; i++) {
346
+ saveToCSV(recordings[i], `recordedMLSignal_${i}_convolved.csv`);
347
+ }
348
+ };
349
+ }
350
+
351
+ export default AudioCalibrator;