Switching microphone on the fly: flac and wav support

This commit is contained in:
Radium Zheng 2018-07-31 19:53:22 +10:00
parent e0ac3efb5c
commit b6e1a49d33
7 changed files with 150 additions and 37 deletions

View File

@ -72,7 +72,12 @@ const ControllerState = Object.freeze({
/** /**
* Stopping. * Stopping.
*/ */
STOPPING: Symbol('STOPPING') STOPPING: Symbol('STOPPING'),
/**
* Failed, due to error during starting / stopping process.
*/
FAILED: Symbol('FAILED')
}); });
/** /**
@ -147,6 +152,13 @@ class RecordingController {
*/ */
_isMuted = false; _isMuted = false;
/**
* The ID of the active microphone.
*
* @private
*/
_micDeviceId = 'default';
/** /**
* Current recording format. This will be in effect from the next * Current recording format. This will be in effect from the next
* recording session, i.e., if this value is changed during an on-going * recording session, i.e., if this value is changed during an on-going
@ -190,14 +202,15 @@ class RecordingController {
* @returns {void} * @returns {void}
*/ */
constructor() { constructor() {
this._updateStats = this._updateStats.bind(this); this.registerEvents = this.registerEvents.bind(this);
this.getParticipantsStats = this.getParticipantsStats.bind(this);
this._onStartCommand = this._onStartCommand.bind(this); this._onStartCommand = this._onStartCommand.bind(this);
this._onStopCommand = this._onStopCommand.bind(this); this._onStopCommand = this._onStopCommand.bind(this);
this._onPingCommand = this._onPingCommand.bind(this); this._onPingCommand = this._onPingCommand.bind(this);
this._doStartRecording = this._doStartRecording.bind(this); this._doStartRecording = this._doStartRecording.bind(this);
this._doStopRecording = this._doStopRecording.bind(this); this._doStopRecording = this._doStopRecording.bind(this);
this.registerEvents = this.registerEvents.bind(this); this._updateStats = this._updateStats.bind(this);
this.getParticipantsStats = this.getParticipantsStats.bind(this); this._switchToNewSession = this._switchToNewSession.bind(this);
} }
registerEvents: () => void; registerEvents: () => void;
@ -311,6 +324,34 @@ class RecordingController {
} }
} }
/**
* Changes the current microphone.
*
* @param {string} micDeviceId - The new microphone device ID.
* @returns {void}
*/
setMicDevice(micDeviceId: string) {
if (micDeviceId !== this._micDeviceId) {
this._micDeviceId = String(micDeviceId);
if (this._state === ControllerState.RECORDING) {
// sessionManager.endSegment(this._currentSessionToken);
logger.log('Before switching microphone...');
this._adapters[this._currentSessionToken]
.setMicDevice(this._micDeviceId)
.then(() => {
logger.log('Finished switching microphone.');
// sessionManager.beginSegment(this._currentSesoken);
})
.catch(() => {
logger.error('Failed to switch microphone');
});
}
logger.log(`Switch microphone to ${this._micDeviceId}`);
}
}
/** /**
* Mute or unmute audio. When muted, the ongoing local recording should * Mute or unmute audio. When muted, the ongoing local recording should
* produce silence. * produce silence.
@ -322,7 +363,7 @@ class RecordingController {
this._isMuted = Boolean(muted); this._isMuted = Boolean(muted);
if (this._state === ControllerState.RECORDING) { if (this._state === ControllerState.RECORDING) {
this._adapters[this._currentSessionToken].setMuted(muted); this._adapters[this._currentSessionToken].setMuted(this._isMuted);
} }
} }
@ -442,11 +483,7 @@ class RecordingController {
if (this._state === ControllerState.IDLE) { if (this._state === ControllerState.IDLE) {
this._changeState(ControllerState.STARTING); this._changeState(ControllerState.STARTING);
this._format = format; this._switchToNewSession(sessionToken, format);
this._currentSessionToken = sessionToken;
logger.log(this._currentSessionToken);
this._adapters[sessionToken]
= this._createRecordingAdapter();
this._doStartRecording(); this._doStartRecording();
} else if (this._state === ControllerState.RECORDING } else if (this._state === ControllerState.RECORDING
&& this._currentSessionToken !== sessionToken) { && this._currentSessionToken !== sessionToken) {
@ -455,10 +492,8 @@ class RecordingController {
// moderator's, so we need to restart the recording. // moderator's, so we need to restart the recording.
this._changeState(ControllerState.STOPPING); this._changeState(ControllerState.STOPPING);
this._doStopRecording().then(() => { this._doStopRecording().then(() => {
this._format = format; this._changeState(ControllerState.STARTING);
this._currentSessionToken = sessionToken; this._switchToNewSession(sessionToken, format);
this._adapters[sessionToken]
= this._createRecordingAdapter();
this._doStartRecording(); this._doStartRecording();
}); });
} }
@ -518,7 +553,7 @@ class RecordingController {
if (this._state === ControllerState.STARTING) { if (this._state === ControllerState.STARTING) {
const delegate = this._adapters[this._currentSessionToken]; const delegate = this._adapters[this._currentSessionToken];
delegate.start() delegate.start(this._micDeviceId)
.then(() => { .then(() => {
this._changeState(ControllerState.RECORDING); this._changeState(ControllerState.RECORDING);
logger.log('Local recording engaged.'); logger.log('Local recording engaged.');
@ -587,6 +622,25 @@ class RecordingController {
} }
_switchToNewSession: (string, string) => void;
/**
* Switches to a new local recording session.
*
* @param {string} sessionToken - The session Token.
* @param {string} format - The recording format for the session.
* @returns {void}
*/
_switchToNewSession(sessionToken, format) {
this._format = format;
this._currentSessionToken = sessionToken;
logger.log(`New session: ${this._currentSessionToken}, `
+ `format: ${this._format}`);
this._adapters[sessionToken]
= this._createRecordingAdapter();
sessionManager.createSession(sessionToken, this._format);
}
/** /**
* Creates a recording adapter according to the current recording format. * Creates a recording adapter according to the current recording format.
* *

View File

@ -7,6 +7,7 @@ import { toggleDialog } from '../base/dialog';
import { i18next } from '../base/i18n'; import { i18next } from '../base/i18n';
import { SET_AUDIO_MUTED } from '../base/media'; import { SET_AUDIO_MUTED } from '../base/media';
import { MiddlewareRegistry } from '../base/redux'; import { MiddlewareRegistry } from '../base/redux';
import { SETTINGS_UPDATED } from '../base/settings/actionTypes';
import { showNotification } from '../notifications'; import { showNotification } from '../notifications';
import { localRecordingEngaged, localRecordingUnengaged } from './actions'; import { localRecordingEngaged, localRecordingUnengaged } from './actions';
@ -76,10 +77,15 @@ isFeatureEnabled
case SET_AUDIO_MUTED: case SET_AUDIO_MUTED:
recordingController.setMuted(action.muted); recordingController.setMuted(action.muted);
break; break;
} case SETTINGS_UPDATED: {
const { micDeviceId } = getState()['features/base/settings'];
// @todo: detect change in features/base/settings micDeviceID if (micDeviceId) {
// @todo: SET_AUDIO_MUTED, when audio is muted recordingController.setMicDevice(micDeviceId);
}
break;
}
}
return result; return result;
}); });

View File

@ -24,9 +24,9 @@ export class OggAdapter extends RecordingAdapter {
* *
* @inheritdoc * @inheritdoc
*/ */
start() { start(micDeviceId) {
if (!this._initPromise) { if (!this._initPromise) {
this._initPromise = this._initialize(); this._initPromise = this._initialize(micDeviceId);
} }
return this._initPromise.then(() => return this._initPromise.then(() =>
@ -96,15 +96,16 @@ export class OggAdapter extends RecordingAdapter {
* Initialize the adapter. * Initialize the adapter.
* *
* @private * @private
* @param {string} micDeviceId - The current microphone device ID.
* @returns {Promise} * @returns {Promise}
*/ */
_initialize() { _initialize(micDeviceId) {
if (this._mediaRecorder) { if (this._mediaRecorder) {
return Promise.resolve(); return Promise.resolve();
} }
return new Promise((resolve, error) => { return new Promise((resolve, error) => {
this._getAudioStream(0) this._getAudioStream(micDeviceId)
.then(stream => { .then(stream => {
this._stream = stream; this._stream = stream;
this._mediaRecorder = new MediaRecorder(stream); this._mediaRecorder = new MediaRecorder(stream);

View File

@ -8,9 +8,11 @@ export class RecordingAdapter {
/** /**
* Starts recording. * Starts recording.
* *
* @param {string} micDeviceId - The microphone to record on.
* @returns {Promise} * @returns {Promise}
*/ */
start() { start(/* eslint-disable no-unused-vars */
micDeviceId/* eslint-enable no-unused-vars */) {
throw new Error('Not implemented'); throw new Error('Not implemented');
} }
@ -43,6 +45,17 @@ export class RecordingAdapter {
throw new Error('Not implemented'); throw new Error('Not implemented');
} }
/**
* Changes the current microphone.
*
* @param {string} micDeviceId - The new microphone device ID.
* @returns {Promise}
*/
setMicDevice(/* eslint-disable no-unused-vars */
micDeviceId/* eslint-enable no-unused-vars */) {
throw new Error('Not implemented');
}
/** /**
* Helper method for getting an audio {@code MediaStream}. Use this instead * Helper method for getting an audio {@code MediaStream}. Use this instead
* of calling browser APIs directly. * of calling browser APIs directly.

View File

@ -65,9 +65,9 @@ export class WavAdapter extends RecordingAdapter {
* *
* @inheritdoc * @inheritdoc
*/ */
start() { start(micDeviceId) {
if (!this._initPromise) { if (!this._initPromise) {
this._initPromise = this._initialize(); this._initPromise = this._initialize(micDeviceId);
} }
return this._initPromise.then(() => { return this._initPromise.then(() => {
@ -197,15 +197,16 @@ export class WavAdapter extends RecordingAdapter {
* Initialize the adapter. * Initialize the adapter.
* *
* @private * @private
* @param {string} micDeviceId - The current microphone device ID.
* @returns {Promise} * @returns {Promise}
*/ */
_initialize() { _initialize(micDeviceId) {
if (this._isInitialized) { if (this._isInitialized) {
return Promise.resolve(); return Promise.resolve();
} }
const p = new Promise((resolve, reject) => { const p = new Promise((resolve, reject) => {
this._getAudioStream(0) this._getAudioStream(micDeviceId)
.then(stream => { .then(stream => {
this._stream = stream; this._stream = stream;
this._audioContext = new AudioContext(); this._audioContext = new AudioContext();
@ -216,9 +217,9 @@ export class WavAdapter extends RecordingAdapter {
this._audioProcessingNode.onaudioprocess = e => { this._audioProcessingNode.onaudioprocess = e => {
const channelLeft = e.inputBuffer.getChannelData(0); const channelLeft = e.inputBuffer.getChannelData(0);
// https://developer.mozilla.org/en-US/docs/ // See: https://developer.mozilla.org/en-US/docs/Web/API/
// Web/API/AudioBuffer/getChannelData // AudioBuffer/getChannelData
// the returned value is an Float32Array // The returned value is an Float32Array.
this._saveWavPCM(channelLeft); this._saveWavPCM(channelLeft);
}; };
this._isInitialized = true; this._isInitialized = true;

View File

@ -38,9 +38,9 @@ export class FlacAdapter extends RecordingAdapter {
* *
* @inheritdoc * @inheritdoc
*/ */
start() { start(micDeviceId) {
if (!this._initPromise) { if (!this._initPromise) {
this._initPromise = this._initialize(); this._initPromise = this._initialize(micDeviceId);
} }
return this._initPromise.then(() => { return this._initPromise.then(() => {
@ -114,13 +114,51 @@ export class FlacAdapter extends RecordingAdapter {
return Promise.resolve(); return Promise.resolve();
} }
/**
* Implements {@link RecordingAdapter#setMicDevice()}.
*
* @inheritdoc
*/
setMicDevice(micDeviceId) {
return this._replaceMic(micDeviceId);
}
/**
* Replaces the current microphone MediaStream.
*
* @param {*} micDeviceId - New microphone ID.
* @returns {Promise}
*/
_replaceMic(micDeviceId) {
if (this._audioContext && this._audioProcessingNode) {
return new Promise((resolve, reject) => {
this._getAudioStream(micDeviceId).then(newStream => {
const newSource = this._audioContext
.createMediaStreamSource(newStream);
this._audioSource.disconnect();
newSource.connect(this._audioProcessingNode);
this._stream = newStream;
this._audioSource = newSource;
resolve();
})
.catch(() => {
reject();
});
});
}
return Promise.resolve();
}
/** /**
* Initialize the adapter. * Initialize the adapter.
* *
* @private * @private
* @param {string} micDeviceId - The current microphone device ID.
* @returns {Promise} * @returns {Promise}
*/ */
_initialize() { _initialize(micDeviceId) {
if (this._encoder !== null) { if (this._encoder !== null) {
return Promise.resolve(); return Promise.resolve();
} }
@ -146,7 +184,7 @@ export class FlacAdapter extends RecordingAdapter {
} else if (e.data.command === DEBUG) { } else if (e.data.command === DEBUG) {
logger.log(e.data); logger.log(e.data);
} else if (e.data.command === WORKER_LIBFLAC_READY) { } else if (e.data.command === WORKER_LIBFLAC_READY) {
logger.debug('libflac is ready.'); logger.log('libflac is ready.');
resolve(); resolve();
} else { } else {
logger.error( logger.error(
@ -165,7 +203,7 @@ export class FlacAdapter extends RecordingAdapter {
}); });
const callbackInitAudioContext = (resolve, reject) => { const callbackInitAudioContext = (resolve, reject) => {
this._getAudioStream(0) this._getAudioStream(micDeviceId)
.then(stream => { .then(stream => {
this._stream = stream; this._stream = stream;
this._audioContext = new AudioContext(); this._audioContext = new AudioContext();
@ -205,7 +243,7 @@ export class FlacAdapter extends RecordingAdapter {
* @returns {void} * @returns {void}
*/ */
_loadWebWorker() { _loadWebWorker() {
// FIXME: workaround for different file names in development/ // FIXME: Workaround for different file names in development/
// production environments. // production environments.
// We cannot import flacEncodeWorker as a webpack module, // We cannot import flacEncodeWorker as a webpack module,
// because it is in a different bundle and should be lazy-loaded // because it is in a different bundle and should be lazy-loaded

View File

@ -81,7 +81,7 @@ const EncoderState = Object.freeze({
}); });
/** /**
* Default compression level. * Default FLAC compression level.
*/ */
const FLAC_COMPRESSION_LEVEL = 5; const FLAC_COMPRESSION_LEVEL = 5;