UNPKG

@twilio/voice-sdk

Version:
530 lines 40.5 kB
/** * @packageDocumentation * @module Voice */ import { EventEmitter } from 'events'; import Device from './device'; import { InvalidArgumentError, NotSupportedError } from './errors'; import Log from './log'; import OutputDeviceCollection from './outputdevicecollection'; import MediaDeviceInfoShim from './shims/mediadeviceinfo'; import getMediaDevicesInstance from './shims/mediadevices'; import { average, difference, isFirefox } from './util'; /** * Aliases for audio kinds, used for labelling. * @private */ const kindAliases = { audioinput: 'Audio Input', audiooutput: 'Audio Output', }; /** * Provides input and output audio-based functionality in one convenient class. * @publicapi */ class AudioHelper extends EventEmitter { /** * @constructor * @private * @param onActiveOutputsChanged - A callback to be called when the user changes the active output devices. * @param onActiveInputChanged - A callback to be called when the user changes the active input device. * @param getUserMedia - The getUserMedia method to use. * @param [options] */ constructor(onActiveOutputsChanged, onActiveInputChanged, getUserMedia, options) { super(); /** * A Map of all audio input devices currently available to the browser by their device ID. */ this.availableInputDevices = new Map(); /** * A Map of all audio output devices currently available to the browser by their device ID. */ this.availableOutputDevices = new Map(); /** * The currently set audio constraints set by setAudioConstraints(). */ this._audioConstraints = null; /** * Whether each sound is enabled. */ this._enabledSounds = { [Device.SoundName.Disconnect]: true, [Device.SoundName.Incoming]: true, [Device.SoundName.Outgoing]: true, }; /** * The current input device. */ this._inputDevice = null; /** * The current input stream. */ this._inputStream = null; /** * Whether the {@link AudioHelper} is currently polling the input stream's volume. */ this._isPollingInputVolume = false; /** * An instance of Logger to use. */ this._log = Log.getInstance(); /** * A record of unknown devices (Devices without labels) */ this._unknownDeviceIndexes = { audioinput: {}, audiooutput: {}, }; /** * Remove an input device from inputs * @param lostDevice * @returns Whether the device was active */ this._removeLostInput = (lostDevice) => { if (!this.inputDevice || this.inputDevice.deviceId !== lostDevice.deviceId) { return false; } this._replaceStream(null); this._inputDevice = null; this._maybeStopPollingVolume(); const defaultDevice = this.availableInputDevices.get('default') || Array.from(this.availableInputDevices.values())[0]; if (defaultDevice) { this.setInputDevice(defaultDevice.deviceId); } return true; }; /** * Remove an input device from outputs * @param lostDevice * @returns Whether the device was active */ this._removeLostOutput = (lostDevice) => { const wasSpeakerLost = this.speakerDevices.delete(lostDevice); const wasRingtoneLost = this.ringtoneDevices.delete(lostDevice); return wasSpeakerLost || wasRingtoneLost; }; /** * Update the available input and output devices */ this._updateAvailableDevices = () => { if (!this._mediaDevices || !this._enumerateDevices) { return Promise.reject('Enumeration not supported'); } return this._enumerateDevices().then((devices) => { this._updateDevices(devices.filter((d) => d.kind === 'audiooutput'), this.availableOutputDevices, this._removeLostOutput); this._updateDevices(devices.filter((d) => d.kind === 'audioinput'), this.availableInputDevices, this._removeLostInput); const defaultDevice = this.availableOutputDevices.get('default') || Array.from(this.availableOutputDevices.values())[0]; [this.speakerDevices, this.ringtoneDevices].forEach(outputDevices => { if (!outputDevices.get().size && this.availableOutputDevices.size && this.isOutputSelectionSupported) { outputDevices.set(defaultDevice.deviceId) .catch((reason) => { this._log.warn(`Unable to set audio output devices. ${reason}`); }); } }); }); }; options = Object.assign({ AudioContext: typeof AudioContext !== 'undefined' && AudioContext, setSinkId: typeof HTMLAudioElement !== 'undefined' && HTMLAudioElement.prototype.setSinkId, }, options); this._getUserMedia = getUserMedia; this._mediaDevices = options.mediaDevices || getMediaDevicesInstance(); this._onActiveInputChanged = onActiveInputChanged; this._enumerateDevices = typeof options.enumerateDevices === 'function' ? options.enumerateDevices : this._mediaDevices && this._mediaDevices.enumerateDevices; const isAudioContextSupported = !!(options.AudioContext || options.audioContext); const isEnumerationSupported = !!this._enumerateDevices; if (options.enabledSounds) { this._enabledSounds = options.enabledSounds; } const isSetSinkSupported = typeof options.setSinkId === 'function'; this.isOutputSelectionSupported = isEnumerationSupported && isSetSinkSupported; this.isVolumeSupported = isAudioContextSupported; if (this.isVolumeSupported) { this._audioContext = options.audioContext || options.AudioContext && new options.AudioContext(); if (this._audioContext) { this._inputVolumeAnalyser = this._audioContext.createAnalyser(); this._inputVolumeAnalyser.fftSize = 32; this._inputVolumeAnalyser.smoothingTimeConstant = 0.3; } } this.ringtoneDevices = new OutputDeviceCollection('ringtone', this.availableOutputDevices, onActiveOutputsChanged, this.isOutputSelectionSupported); this.speakerDevices = new OutputDeviceCollection('speaker', this.availableOutputDevices, onActiveOutputsChanged, this.isOutputSelectionSupported); this.addListener('newListener', (eventName) => { if (eventName === 'inputVolume') { this._maybeStartPollingVolume(); } }); this.addListener('removeListener', (eventName) => { if (eventName === 'inputVolume') { this._maybeStopPollingVolume(); } }); this.once('newListener', () => { // NOTE (rrowland): Ideally we would only check isEnumerationSupported here, but // in at least one browser version (Tested in FF48) enumerateDevices actually // returns bad data for the listed devices. Instead, we check for // isOutputSelectionSupported to avoid these quirks that may negatively affect customers. if (!this.isOutputSelectionSupported) { this._log.warn('Warning: This browser does not support audio output selection.'); } if (!this.isVolumeSupported) { this._log.warn(`Warning: This browser does not support Twilio's volume indicator feature.`); } }); if (isEnumerationSupported) { this._initializeEnumeration(); } } /** * The currently set audio constraints set by setAudioConstraints(). Starts as null. */ get audioConstraints() { return this._audioConstraints; } /** * The active input device. Having no inputDevice specified by `setInputDevice()` * will disable input selection related functionality. */ get inputDevice() { return this._inputDevice; } /** * The current input stream. */ get inputStream() { return this._inputStream; } /** * Current state of the enabled sounds * @private */ _getEnabledSounds() { return this._enabledSounds; } /** * Start polling volume if it's supported and there's an input stream to poll. * @private */ _maybeStartPollingVolume() { if (!this.isVolumeSupported || !this._inputStream) { return; } this._updateVolumeSource(); if (this._isPollingInputVolume || !this._inputVolumeAnalyser) { return; } const bufferLength = this._inputVolumeAnalyser.frequencyBinCount; const buffer = new Uint8Array(bufferLength); this._isPollingInputVolume = true; const emitVolume = () => { if (!this._isPollingInputVolume) { return; } if (this._inputVolumeAnalyser) { this._inputVolumeAnalyser.getByteFrequencyData(buffer); const inputVolume = average(buffer); this.emit('inputVolume', inputVolume / 255); } requestAnimationFrame(emitVolume); }; requestAnimationFrame(emitVolume); } /** * Stop polling volume if it's currently polling and there are no listeners. * @private */ _maybeStopPollingVolume() { if (!this.isVolumeSupported) { return; } if (!this._isPollingInputVolume || (this._inputStream && this.listenerCount('inputVolume'))) { return; } if (this._inputVolumeSource) { this._inputVolumeSource.disconnect(); delete this._inputVolumeSource; } this._isPollingInputVolume = false; } /** * Unbind the listeners from mediaDevices. * @private */ _unbind() { if (!this._mediaDevices || !this._enumerateDevices) { throw new NotSupportedError('Enumeration is not supported'); } if (this._mediaDevices.removeEventListener) { this._mediaDevices.removeEventListener('devicechange', this._updateAvailableDevices); this._mediaDevices.removeEventListener('deviceinfochange', this._updateAvailableDevices); } } /** * Enable or disable the disconnect sound. * @param doEnable Passing `true` will enable the sound and `false` will disable the sound. * Not passing this parameter will not alter the enable-status of the sound. * @returns The enable-status of the sound. */ disconnect(doEnable) { return this._maybeEnableSound(Device.SoundName.Disconnect, doEnable); } /** * Enable or disable the incoming sound. * @param doEnable Passing `true` will enable the sound and `false` will disable the sound. * Not passing this parameter will not alter the enable-status of the sound. * @returns The enable-status of the sound. */ incoming(doEnable) { return this._maybeEnableSound(Device.SoundName.Incoming, doEnable); } /** * Enable or disable the outgoing sound. * @param doEnable Passing `true` will enable the sound and `false` will disable the sound. * Not passing this parameter will not alter the enable-status of the sound. * @returns The enable-status of the sound. */ outgoing(doEnable) { return this._maybeEnableSound(Device.SoundName.Outgoing, doEnable); } /** * Set the MediaTrackConstraints to be applied on every getUserMedia call for new input * device audio. Any deviceId specified here will be ignored. Instead, device IDs should * be specified using {@link AudioHelper#setInputDevice}. The returned Promise resolves * when the media is successfully reacquired, or immediately if no input device is set. * @param audioConstraints - The MediaTrackConstraints to apply. */ setAudioConstraints(audioConstraints) { this._audioConstraints = Object.assign({}, audioConstraints); delete this._audioConstraints.deviceId; return this.inputDevice ? this._setInputDevice(this.inputDevice.deviceId, true) : Promise.resolve(); } /** * Replace the current input device with a new device by ID. * @param deviceId - An ID of a device to replace the existing * input device with. */ setInputDevice(deviceId) { return !isFirefox() ? this._setInputDevice(deviceId, false) : Promise.reject(new NotSupportedError('Firefox does not currently support opening multiple ' + 'audio input tracks simultaneously, even across different tabs. As a result, ' + 'Device.audio.setInputDevice is disabled on Firefox until support is added.\n' + 'Related BugZilla thread: https://bugzilla.mozilla.org/show_bug.cgi?id=1299324')); } /** * Unset the MediaTrackConstraints to be applied on every getUserMedia call for new input * device audio. The returned Promise resolves when the media is successfully reacquired, * or immediately if no input device is set. */ unsetAudioConstraints() { this._audioConstraints = null; return this.inputDevice ? this._setInputDevice(this.inputDevice.deviceId, true) : Promise.resolve(); } /** * Unset the input device, stopping the tracks. This should only be called when not in a connection, and * will not allow removal of the input device during a live call. */ unsetInputDevice() { if (!this.inputDevice) { return Promise.resolve(); } return this._onActiveInputChanged(null).then(() => { this._replaceStream(null); this._inputDevice = null; this._maybeStopPollingVolume(); }); } /** * Get the index of an un-labeled Device. * @param mediaDeviceInfo * @returns The index of the passed MediaDeviceInfo */ _getUnknownDeviceIndex(mediaDeviceInfo) { const id = mediaDeviceInfo.deviceId; const kind = mediaDeviceInfo.kind; let index = this._unknownDeviceIndexes[kind][id]; if (!index) { index = Object.keys(this._unknownDeviceIndexes[kind]).length + 1; this._unknownDeviceIndexes[kind][id] = index; } return index; } /** * Initialize output device enumeration. */ _initializeEnumeration() { if (!this._mediaDevices || !this._enumerateDevices) { throw new NotSupportedError('Enumeration is not supported'); } if (this._mediaDevices.addEventListener) { this._mediaDevices.addEventListener('devicechange', this._updateAvailableDevices); this._mediaDevices.addEventListener('deviceinfochange', this._updateAvailableDevices); } this._updateAvailableDevices().then(() => { if (!this.isOutputSelectionSupported) { return; } Promise.all([ this.speakerDevices.set('default'), this.ringtoneDevices.set('default'), ]).catch(reason => { this._log.warn(`Warning: Unable to set audio output devices. ${reason}`); }); }); } /** * Set whether the sound is enabled or not * @param soundName * @param doEnable * @returns Whether the sound is enabled or not */ _maybeEnableSound(soundName, doEnable) { if (typeof doEnable !== 'undefined') { this._enabledSounds[soundName] = doEnable; } return this._enabledSounds[soundName]; } /** * Stop the tracks on the current input stream before replacing it with the passed stream. * @param stream - The new stream */ _replaceStream(stream) { if (this._inputStream) { this._inputStream.getTracks().forEach(track => { track.stop(); }); } this._inputStream = stream; } /** * Replace the current input device with a new device by ID. * @param deviceId - An ID of a device to replace the existing * input device with. * @param forceGetUserMedia - If true, getUserMedia will be called even if * the specified device is already active. */ _setInputDevice(deviceId, forceGetUserMedia) { if (typeof deviceId !== 'string') { return Promise.reject(new InvalidArgumentError('Must specify the device to set')); } const device = this.availableInputDevices.get(deviceId); if (!device) { return Promise.reject(new InvalidArgumentError(`Device not found: ${deviceId}`)); } if (this._inputDevice && this._inputDevice.deviceId === deviceId && this._inputStream) { if (!forceGetUserMedia) { return Promise.resolve(); } // If the currently active track is still in readyState `live`, gUM may return the same track // rather than returning a fresh track. this._inputStream.getTracks().forEach(track => { track.stop(); }); } const constraints = { audio: Object.assign({ deviceId: { exact: deviceId } }, this.audioConstraints) }; return this._getUserMedia(constraints).then((stream) => { return this._onActiveInputChanged(stream).then(() => { this._replaceStream(stream); this._inputDevice = device; this._maybeStartPollingVolume(); }); }); } /** * Update a set of devices. * @param updatedDevices - An updated list of available Devices * @param availableDevices - The previous list of available Devices * @param removeLostDevice - The method to call if a previously available Device is * no longer available. */ _updateDevices(updatedDevices, availableDevices, removeLostDevice) { const updatedDeviceIds = updatedDevices.map(d => d.deviceId); const knownDeviceIds = Array.from(availableDevices.values()).map(d => d.deviceId); const lostActiveDevices = []; // Remove lost devices const lostDeviceIds = difference(knownDeviceIds, updatedDeviceIds); lostDeviceIds.forEach((lostDeviceId) => { const lostDevice = availableDevices.get(lostDeviceId); if (lostDevice) { availableDevices.delete(lostDeviceId); if (removeLostDevice(lostDevice)) { lostActiveDevices.push(lostDevice); } } }); // Add any new devices, or devices with updated labels let deviceChanged = false; updatedDevices.forEach(newDevice => { const existingDevice = availableDevices.get(newDevice.deviceId); const newMediaDeviceInfo = this._wrapMediaDeviceInfo(newDevice); if (!existingDevice || existingDevice.label !== newMediaDeviceInfo.label) { availableDevices.set(newDevice.deviceId, newMediaDeviceInfo); deviceChanged = true; } }); if (deviceChanged || lostDeviceIds.length) { // Force a new gUM in case the underlying tracks of the active stream have changed. One // reason this might happen is when `default` is selected and set to a USB device, // then that device is unplugged or plugged back in. We can't check for the 'ended' // event or readyState because it is asynchronous and may take upwards of 5 seconds, // in my testing. (rrowland) if (this.inputDevice !== null && this.inputDevice.deviceId === 'default') { this._log.warn(`Calling getUserMedia after device change to ensure that the \ tracks of the active device (default) have not gone stale.`); this._setInputDevice(this.inputDevice.deviceId, true); } this.emit('deviceChange', lostActiveDevices); } } /** * Disconnect the old input volume source, and create and connect a new one with the current * input stream. */ _updateVolumeSource() { if (!this._inputStream || !this._audioContext || !this._inputVolumeAnalyser) { return; } if (this._inputVolumeSource) { this._inputVolumeSource.disconnect(); } try { this._inputVolumeSource = this._audioContext.createMediaStreamSource(this._inputStream); this._inputVolumeSource.connect(this._inputVolumeAnalyser); } catch (ex) { this._log.warn('Unable to update volume source', ex); delete this._inputVolumeSource; } } /** * Convert a MediaDeviceInfo to a IMediaDeviceInfoShim. * @param mediaDeviceInfo - The info to convert * @returns The converted shim */ _wrapMediaDeviceInfo(mediaDeviceInfo) { const options = { deviceId: mediaDeviceInfo.deviceId, groupId: mediaDeviceInfo.groupId, kind: mediaDeviceInfo.kind, label: mediaDeviceInfo.label, }; if (!options.label) { if (options.deviceId === 'default') { options.label = 'Default'; } else { const index = this._getUnknownDeviceIndex(mediaDeviceInfo); options.label = `Unknown ${kindAliases[options.kind]} Device ${index}`; } } return new MediaDeviceInfoShim(options); } } (function (AudioHelper) { })(AudioHelper || (AudioHelper = {})); export default AudioHelper; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"audiohelper.js","sourceRoot":"","sources":["../../lib/twilio/audiohelper.ts"],"names":[],"mappings":"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"}