speaker-calibration 2.2.242 → 2.2.244

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.
@@ -1,323 +1,351 @@
1
- import MyEventEmitter from '../myEventEmitter';
2
-
3
- /**
4
- * @class provides a simple interface for recording audio from a microphone
5
- * using the Media Recorder API.
6
- */
7
- class AudioRecorder extends MyEventEmitter {
8
- /** @private */
9
- #mediaRecorder;
10
-
11
- /** @private */
12
- #recordedChunks = [];
13
-
14
- /** @private */
15
- #audioBlob;
16
-
17
- /** @private */
18
- #audioContext;
19
-
20
- /** @private */
21
- #recordedSignals = [];
22
-
23
- /**@private */
24
- #allHzUnfilteredRecordings = [];
25
-
26
- /**@private */
27
- #allBackgroundRecordings = [];
28
-
29
- /** @private */
30
- #allHzFilteredRecordings = [];
31
-
32
- /** @private */
33
- sinkSamplingRate;
34
-
35
- /** @private */
36
- sampleSize;
37
-
38
- /** @private */
39
- #allVolumeRecordings = [];
40
-
41
- /** @private */
42
- flags = {};
43
-
44
- /**
45
- * Decode the audio data from the recorded audio blob.
46
- *
47
- * @private
48
- * @example
49
- */
50
- #saveRecording = async (mode, checkRec) => {
51
- const arrayBuffer = await this.#audioBlob.arrayBuffer();
52
- const audioBuffer = await this.#audioContext.decodeAudioData(arrayBuffer);
53
- console.log(audioBuffer);
54
- const data = audioBuffer.getChannelData(0);
55
- const dataArray = Array.from(data);
56
-
57
- console.log(`Decoded audio buffer with ${data.length} samples`);
58
- console.log(`Unfiltered recording should be of length: ${data.length}`);
59
- if (checkRec == 'loudest') {
60
- const uniqueSet = new Set(dataArray);
61
- const numberOfUniqueValues = uniqueSet.size;
62
- const squaredValues = dataArray.map(value => value * value);
63
- const sum_of_squares = squaredValues.reduce((total, value) => total + value, 0);
64
- const squared_mean = sum_of_squares / dataArray.length;
65
- const dbLevel = 20 * Math.log10(Math.sqrt(squared_mean));
66
- const roundedDbLevel = Math.round(dbLevel * 10) / 10;
67
- console.log(
68
- 'Loudest 1000-Hz recording: ' +
69
- roundedDbLevel +
70
- ' dB with ' +
71
- numberOfUniqueValues +
72
- ' unique values.'
73
- );
74
- } else if (checkRec == 'allhz') {
75
- const uniqueSet = new Set(dataArray);
76
- const numberOfUniqueValues = uniqueSet.size;
77
- const squaredValues = dataArray.map(value => value * value);
78
- const sum_of_squares = squaredValues.reduce((total, value) => total + value, 0);
79
- const squared_mean = sum_of_squares / dataArray.length;
80
- const dbLevel = 20 * Math.log10(Math.sqrt(squared_mean));
81
- const roundedDbLevel = Math.round(dbLevel * 10) / 10;
82
- console.log(
83
- 'All Hz Recording: ' +
84
- roundedDbLevel +
85
- ' dB with ' +
86
- numberOfUniqueValues +
87
- ' unique values.'
88
- );
89
- }
90
- if (mode === 'volume') {
91
- console.log('Saving 1000 Hz Recording to #allVolumeRecordings');
92
- this.#allVolumeRecordings.push(dataArray);
93
- } else if (mode === 'unfiltered') {
94
- console.log('Saving unfiltered all Hz recording to #allHzUnfilteredRecordings');
95
- this.#allHzUnfilteredRecordings.push(dataArray);
96
- } else if (mode === 'filtered') {
97
- console.log('Saving filtered all hz recording to #allHzFilteredRecordings');
98
- this.#allHzFilteredRecordings.push(dataArray);
99
- } else if (mode === 'background') {
100
- console.log('Saving background recording to #allBackgroundRecordings');
101
- this.#allBackgroundRecordings.push(dataArray);
102
- }
103
- };
104
-
105
- #saveFilteredRecording = async () => {
106
- const arrayBuffer = await this.#audioBlob.arrayBuffer();
107
- const audioBuffer = await this.#audioContext.decodeAudioData(arrayBuffer);
108
- const data = audioBuffer.getChannelData(0);
109
-
110
- console.log(`Decoded audio buffer with ${data.length} samples`);
111
- console.log(`Filtered recording should be of length: ${data.length}`);
112
- this.#allHzFilteredRecordings.push(Array.from(data));
113
- };
114
-
115
- /**
116
- * Event listener triggered when data is available in the media recorder.
117
- *
118
- * @private
119
- * @param e - The event object.
120
- * @example
121
- */
122
- #onRecorderDataAvailable = e => {
123
- if (e.data && e.data.size > 0) this.#recordedChunks.push(e.data);
124
- };
125
-
126
- /**
127
- * Method to create a media recorder object and set up event listeners.
128
- *
129
- * @private
130
- * @param stream - The stream of audio from the Listener.
131
- * @example
132
- */
133
- #setMediaRecorder = stream => {
134
- // Create a new MediaRecorder object
135
- this.#mediaRecorder = new MediaRecorder(stream);
136
-
137
- // Add event listeners
138
- this.#mediaRecorder.ondataavailable = e => this.#onRecorderDataAvailable(e);
139
- };
140
-
141
- #removeMediaRecorder = () => {
142
- this.#mediaRecorder.ondataavailable = null;
143
- this.#mediaRecorder = null;
144
- };
145
-
146
- #setAudioContext = () => {
147
- this.#audioContext = new (window.AudioContext ||
148
- window.webkitAudioContext ||
149
- window.audioContext)({
150
- sampleRate: this.sinkSamplingRate,
151
- sampleSize: this.sampleSize,
152
- //sampleRate: 96000
153
- });
154
- };
155
-
156
- /**
157
- * Public method to start the recording process.
158
- *
159
- * @param stream - The stream of audio from the Listener.
160
- * @example
161
- */
162
- startRecording = async stream => {
163
- try {
164
- // Create a fresh audio context
165
- this.#setAudioContext();
166
- // Set up media recorder if needed
167
- if (!this.#mediaRecorder) this.#setMediaRecorder(stream);
168
- // clear recorded chunks
169
- this.#recordedChunks = [];
170
- if (this.#mediaRecorder.state === 'recording') {
171
- this.#mediaRecorder.stop();
172
- }
173
- // start recording
174
- this.#mediaRecorder.start();
175
- } catch (error) {
176
- console.error('Error in startRecording:', error);
177
- // Handle the error as needed, e.g., throw it or perform error-specific actions
178
- }
179
- };
180
-
181
- /**
182
- * Method to stop the recording process.
183
- *
184
- * @public
185
- * @example
186
- */
187
- stopRecording = async (mode, checkRec) => {
188
- try {
189
- // Stop the media recorder, and wait for the data to be available
190
- await new Promise(resolve => {
191
- this.#mediaRecorder.onstop = () => {
192
- // when the stop event is triggered, resolve the promise
193
- this.#audioBlob = new Blob(this.#recordedChunks, {
194
- type: 'audio/wav; codecs=opus',
195
- });
196
- resolve(this.#audioBlob);
197
- };
198
- // call stop
199
- this.#mediaRecorder.stop();
200
- });
201
- // Now that we have data, save it
202
- await this.#saveRecording(mode, checkRec);
203
- } catch (error) {
204
- console.error('Error in stopRecording:', error);
205
- // Handle the error as needed, e.g., throw it or perform error-specific actions
206
- }
207
- };
208
-
209
- /** .
210
- * .
211
- * .
212
- * Public method to get the last recorded audio signal
213
- *
214
- * @returns
215
- * @example
216
- */
217
- getLastRecordedSignal = () => this.#recordedSignals[this.#recordedSignals.length - 1];
218
-
219
- /** .
220
- * .
221
- * .
222
- * Public method to get the last 1000hz recorded audio signal
223
- *
224
- * @returns
225
- * @example
226
- */
227
- getLastVolumeRecordedSignal = () =>
228
- Array.from(this.#allVolumeRecordings[this.#allVolumeRecordings.length - 1]);
229
-
230
- /** .
231
- * .
232
- * .
233
- * Public method to get all the recorded audio signals
234
- *
235
- * @returns
236
- * @example
237
- */
238
- getAllRecordedSignals = () => this.#recordedSignals;
239
-
240
- /** .
241
- * .
242
- * .
243
- * Public method to get all the recorded audio signals
244
- *
245
- * @returns
246
- * @example
247
- */
248
- getAllVolumeRecordedSignals = () => this.#allVolumeRecordings;
249
-
250
- /** .
251
- * .
252
- * .
253
- * Public method to get all the recorded audio signals
254
- *
255
- * @returns
256
- * @example
257
- */
258
- getAllFilteredRecordedSignals = () => this.#allHzFilteredRecordings;
259
-
260
- /** .
261
- * .
262
- * .
263
- * Public method to get all the recorded audio signals
264
- *
265
- * @returns
266
- * @example
267
- */
268
- clearAllFilteredRecordedSignals = () => (this.#allHzFilteredRecordings = []);
269
-
270
- /** .
271
- * .
272
- * .
273
- * Public method to clear last the recorded audio signals
274
- *
275
- * @returns
276
- * @example
277
- */
278
- clearLastFilteredRecordedSignals = () => this.#allHzFilteredRecordings.pop();
279
-
280
- /** .
281
- * .
282
- * .
283
- * Public method to get all the recorded audio signals for psd
284
- *
285
- * @returns
286
- * @example
287
- */
288
- getAllUnfilteredRecordedSignals = () => this.#allHzUnfilteredRecordings;
289
-
290
- /** .
291
- * .
292
- * .
293
- * Public method to get all the recorded audio signals
294
- *
295
- * @returns
296
- * @example
297
- */
298
- clearLastUnfilteredRecordedSignals = () => this.#allHzUnfilteredRecordings.pop();
299
-
300
- /** .
301
- * .
302
- * .
303
- * Public method to get all the recorded audio signals for psd
304
- *
305
- * @returns
306
- * @example
307
- */
308
- getAllBackgroundRecordings = () => this.#allBackgroundRecordings;
309
-
310
- /** .
311
- * .
312
- * .
313
- * Public method to set the sampling rate used by the capture device
314
- *
315
- * @param {Number} sinkSamplingRate - The sampling rate of the capture device
316
- * @example
317
- */
318
- setSinkSamplingRate = sinkSamplingRate => {
319
- this.sinkSamplingRate = sinkSamplingRate;
320
- };
321
- }
322
-
323
- export default AudioRecorder;
1
+ import MyEventEmitter from '../myEventEmitter';
2
+
3
+ /**
4
+ * @class provides a simple interface for recording audio from a microphone
5
+ * using the Media Recorder API.
6
+ */
7
+ class AudioRecorder extends MyEventEmitter {
8
+ /** @private */
9
+ #mediaRecorder;
10
+
11
+ /** @private */
12
+ #recordedChunks = [];
13
+
14
+ /** @private */
15
+ #audioBlob;
16
+
17
+ /** @private */
18
+ #audioContext;
19
+
20
+ /** @private */
21
+ #recordedSignals = [];
22
+
23
+ /**@private */
24
+ #allHzUnfilteredRecordings = [];
25
+
26
+ /**@private */
27
+ #allBackgroundRecordings = [];
28
+
29
+ /** @private */
30
+ #allHzFilteredRecordings = [];
31
+
32
+ /** @private */
33
+ sinkSamplingRate;
34
+
35
+ /** @private */
36
+ sampleSize;
37
+
38
+ /** @private */
39
+ #allVolumeRecordings = [];
40
+
41
+ /** @private */
42
+ flags = {};
43
+
44
+ /**
45
+ * Decode the audio data from the recorded audio blob.
46
+ *
47
+ * @private
48
+ * @example
49
+ */
50
+ #saveRecording = async (mode, checkRec) => {
51
+ const arrayBuffer = await this.#audioBlob.arrayBuffer();
52
+ const audioBuffer = await this.#audioContext.decodeAudioData(arrayBuffer);
53
+ console.log(audioBuffer);
54
+ const data = audioBuffer.getChannelData(0);
55
+ const dataArray = Array.from(data);
56
+
57
+ console.log(`Decoded audio buffer with ${data.length} samples`);
58
+ console.log(`Unfiltered recording should be of length: ${data.length}`);
59
+ if (checkRec == 'loudest') {
60
+ const uniqueSet = new Set(dataArray);
61
+ const numberOfUniqueValues = uniqueSet.size;
62
+ const squaredValues = dataArray.map(value => value * value);
63
+ const sum_of_squares = squaredValues.reduce((total, value) => total + value, 0);
64
+ const squared_mean = sum_of_squares / dataArray.length;
65
+ const dbLevel = 20 * Math.log10(Math.sqrt(squared_mean));
66
+ const roundedDbLevel = Math.round(dbLevel * 10) / 10;
67
+ console.log(
68
+ 'Loudest 1000-Hz recording: ' +
69
+ roundedDbLevel +
70
+ ' dB with ' +
71
+ numberOfUniqueValues +
72
+ ' unique values.'
73
+ );
74
+ } else if (checkRec == 'allhz') {
75
+ const uniqueSet = new Set(dataArray);
76
+ const numberOfUniqueValues = uniqueSet.size;
77
+ const squaredValues = dataArray.map(value => value * value);
78
+ const sum_of_squares = squaredValues.reduce((total, value) => total + value, 0);
79
+ const squared_mean = sum_of_squares / dataArray.length;
80
+ const dbLevel = 20 * Math.log10(Math.sqrt(squared_mean));
81
+ const roundedDbLevel = Math.round(dbLevel * 10) / 10;
82
+ console.log(
83
+ 'All Hz Recording: ' +
84
+ roundedDbLevel +
85
+ ' dB with ' +
86
+ numberOfUniqueValues +
87
+ ' unique values.'
88
+ );
89
+ }
90
+ if (mode === 'volume') {
91
+ console.log('Saving 1000 Hz Recording to #allVolumeRecordings');
92
+ this.#allVolumeRecordings.push(dataArray);
93
+ } else if (mode === 'unfiltered') {
94
+ console.log('Saving unfiltered all Hz recording to #allHzUnfilteredRecordings');
95
+ this.#allHzUnfilteredRecordings.push(dataArray);
96
+ } else if (mode === 'filtered') {
97
+ console.log('Saving filtered all hz recording to #allHzFilteredRecordings');
98
+ this.#allHzFilteredRecordings.push(dataArray);
99
+ } else if (mode === 'background') {
100
+ console.log('Saving background recording to #allBackgroundRecordings');
101
+ this.#allBackgroundRecordings.push(dataArray);
102
+ }
103
+ };
104
+
105
+ saveVolumeRecording = async dataArray => {
106
+ this.#allVolumeRecordings.push(dataArray);
107
+ };
108
+
109
+ #saveFilteredRecording = async () => {
110
+ const arrayBuffer = await this.#audioBlob.arrayBuffer();
111
+ const audioBuffer = await this.#audioContext.decodeAudioData(arrayBuffer);
112
+ const data = audioBuffer.getChannelData(0);
113
+
114
+ console.log(`Decoded audio buffer with ${data.length} samples`);
115
+ console.log(`Filtered recording should be of length: ${data.length}`);
116
+ this.#allHzFilteredRecordings.push(Array.from(data));
117
+ };
118
+
119
+ /**
120
+ * Event listener triggered when data is available in the media recorder.
121
+ *
122
+ * @private
123
+ * @param e - The event object.
124
+ * @example
125
+ */
126
+ #onRecorderDataAvailable = e => {
127
+ if (e.data && e.data.size > 0) this.#recordedChunks.push(e.data);
128
+ };
129
+
130
+ /**
131
+ * Method to create a media recorder object and set up event listeners.
132
+ *
133
+ * @private
134
+ * @param stream - The stream of audio from the Listener.
135
+ * @example
136
+ */
137
+ #setMediaRecorder = stream => {
138
+ // Create a new MediaRecorder object
139
+ this.#mediaRecorder = new MediaRecorder(stream);
140
+
141
+ // Add event listeners
142
+ this.#mediaRecorder.ondataavailable = e => this.#onRecorderDataAvailable(e);
143
+ };
144
+
145
+ #removeMediaRecorder = () => {
146
+ this.#mediaRecorder.ondataavailable = null;
147
+ this.#mediaRecorder = null;
148
+ };
149
+
150
+ #setAudioContext = () => {
151
+ this.#audioContext = new (window.AudioContext ||
152
+ window.webkitAudioContext ||
153
+ window.audioContext)({
154
+ sampleRate: this.sinkSamplingRate,
155
+ sampleSize: this.sampleSize,
156
+ //sampleRate: 96000
157
+ });
158
+ };
159
+
160
+ /**
161
+ * Public method to start the recording process.
162
+ *
163
+ * @param stream - The stream of audio from the Listener.
164
+ * @example
165
+ */
166
+ startRecording = async stream => {
167
+ try {
168
+ // Create a fresh audio context
169
+ this.#setAudioContext();
170
+ // Set up media recorder if needed
171
+ if (!this.#mediaRecorder) this.#setMediaRecorder(stream);
172
+ // clear recorded chunks
173
+ this.#recordedChunks = [];
174
+ if (this.#mediaRecorder.state === 'recording') {
175
+ this.#mediaRecorder.stop();
176
+ }
177
+ // start recording
178
+ this.#mediaRecorder.start();
179
+ } catch (error) {
180
+ console.error('Error in startRecording:', error);
181
+ // Handle the error as needed, e.g., throw it or perform error-specific actions
182
+ }
183
+ };
184
+
185
+ /**
186
+ * Method to stop the recording process.
187
+ *
188
+ * @public
189
+ * @example
190
+ */
191
+ stopRecording = async (mode, checkRec) => {
192
+ try {
193
+ // Stop the media recorder, and wait for the data to be available
194
+ await new Promise(resolve => {
195
+ this.#mediaRecorder.onstop = () => {
196
+ // when the stop event is triggered, resolve the promise
197
+ this.#audioBlob = new Blob(this.#recordedChunks, {
198
+ type: 'audio/wav; codecs=opus',
199
+ });
200
+ resolve(this.#audioBlob);
201
+ };
202
+ // call stop
203
+ this.#mediaRecorder.stop();
204
+ });
205
+ // Now that we have data, save it
206
+ await this.#saveRecording(mode, checkRec);
207
+ } catch (error) {
208
+ console.error('Error in stopRecording:', error);
209
+ // Handle the error as needed, e.g., throw it or perform error-specific actions
210
+ }
211
+ };
212
+
213
+ /** .
214
+ * .
215
+ * .
216
+ * Public method to get the last recorded audio signal
217
+ *
218
+ * @returns
219
+ * @example
220
+ */
221
+ getLastRecordedSignal = () => this.#recordedSignals[this.#recordedSignals.length - 1];
222
+
223
+ /** .
224
+ * .
225
+ * .
226
+ * Public method to get the last 1000hz recorded audio signal
227
+ *
228
+ * @returns
229
+ * @example
230
+ */
231
+ getLastVolumeRecordedSignal = () =>
232
+ Array.from(this.#allVolumeRecordings[this.#allVolumeRecordings.length - 1]);
233
+
234
+ /** .
235
+ * .
236
+ * .
237
+ * Public method to get all the recorded audio signals
238
+ *
239
+ * @returns
240
+ * @example
241
+ */
242
+ getAllRecordedSignals = () => this.#recordedSignals;
243
+
244
+ /** .
245
+ * .
246
+ * .
247
+ * Public method to get all the recorded audio signals
248
+ *
249
+ * @returns
250
+ * @example
251
+ */
252
+ getAllVolumeRecordedSignals = () => this.#allVolumeRecordings;
253
+
254
+ /** .
255
+ * .
256
+ * .
257
+ * Public method to get all the recorded audio signals
258
+ *
259
+ * @returns
260
+ * @example
261
+ */
262
+ getAllFilteredRecordedSignals = () => this.#allHzFilteredRecordings;
263
+
264
+ /** .
265
+ * .
266
+ * .
267
+ * Public method to get all the recorded audio signals
268
+ *
269
+ * @returns
270
+ * @example
271
+ */
272
+ clearAllFilteredRecordedSignals = () => (this.#allHzFilteredRecordings = []);
273
+
274
+ /** .
275
+ * .
276
+ * .
277
+ * Public method to clear last the recorded audio signals
278
+ *
279
+ * @returns
280
+ * @example
281
+ */
282
+ clearLastFilteredRecordedSignals = () => this.#allHzFilteredRecordings.pop();
283
+
284
+ /** .
285
+ * .
286
+ * .
287
+ * Public method to get all the recorded audio signals for psd
288
+ *
289
+ * @returns
290
+ * @example
291
+ */
292
+ getAllUnfilteredRecordedSignals = () => this.#allHzUnfilteredRecordings;
293
+
294
+ /** .
295
+ * .
296
+ * .
297
+ * Public method to get all the recorded audio signals
298
+ *
299
+ * @returns
300
+ * @example
301
+ */
302
+ clearLastUnfilteredRecordedSignals = () => this.#allHzUnfilteredRecordings.pop();
303
+
304
+ /** .
305
+ * .
306
+ * .
307
+ * Public method to save a simulated unfiltered recording directly
308
+ *
309
+ * @param {Array<number>} dataArray - The simulated recording data
310
+ * @example
311
+ */
312
+ saveUnfilteredRecording = async dataArray => {
313
+ this.#allHzUnfilteredRecordings.push(dataArray);
314
+ };
315
+
316
+ /** .
317
+ * .
318
+ * .
319
+ * Public method to save a simulated filtered recording directly
320
+ *
321
+ * @param {Array<number>} dataArray - The simulated filtered recording data
322
+ * @example
323
+ */
324
+ saveFilteredRecording = async dataArray => {
325
+ this.#allHzFilteredRecordings.push(dataArray);
326
+ };
327
+
328
+ /** .
329
+ * .
330
+ * .
331
+ * Public method to get all the recorded audio signals for psd
332
+ *
333
+ * @returns
334
+ * @example
335
+ */
336
+ getAllBackgroundRecordings = () => this.#allBackgroundRecordings;
337
+
338
+ /** .
339
+ * .
340
+ * .
341
+ * Public method to set the sampling rate used by the capture device
342
+ *
343
+ * @param {Number} sinkSamplingRate - The sampling rate of the capture device
344
+ * @example
345
+ */
346
+ setSinkSamplingRate = sinkSamplingRate => {
347
+ this.sinkSamplingRate = sinkSamplingRate;
348
+ };
349
+ }
350
+
351
+ export default AudioRecorder;