2017-10-24 08:40:39 +00:00
|
|
|
// @flow
|
|
|
|
|
|
|
|
import _ from 'lodash';
|
|
|
|
import React, { Component } from 'react';
|
|
|
|
import { NativeModules } from 'react-native';
|
|
|
|
import { connect } from 'react-redux';
|
|
|
|
|
|
|
|
import { hideDialog, SimpleBottomSheet } from '../../../base/dialog';
|
2017-11-20 11:13:42 +00:00
|
|
|
import { translate } from '../../../base/i18n';
|
|
|
|
|
2017-10-24 08:40:39 +00:00
|
|
|
|
2017-11-14 20:18:16 +00:00
|
|
|
/**
|
|
|
|
* {@code PasswordRequiredPrompt}'s React {@code Component} prop types.
|
|
|
|
*/
|
|
|
|
type Props = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Used for hiding the dialog when the selection was completed.
|
|
|
|
*/
|
2017-11-20 11:13:42 +00:00
|
|
|
dispatch: Function,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Invoked to obtain translated strings.
|
|
|
|
*/
|
|
|
|
t: Function
|
2017-11-14 20:18:16 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
type State = {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Array of available devices.
|
|
|
|
*/
|
|
|
|
devices: Array<string>
|
|
|
|
};
|
|
|
|
|
|
|
|
const { AudioMode } = NativeModules;
|
2017-10-24 08:40:39 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps each device type to a display name and icon.
|
|
|
|
*/
|
|
|
|
const deviceInfoMap = {
|
|
|
|
BLUETOOTH: {
|
|
|
|
iconName: 'bluetooth',
|
2017-11-20 11:13:42 +00:00
|
|
|
text: 'audioDevices.bluetooth',
|
2017-10-24 08:40:39 +00:00
|
|
|
type: 'BLUETOOTH'
|
|
|
|
},
|
|
|
|
EARPIECE: {
|
|
|
|
iconName: 'phone-talk',
|
2017-11-20 11:13:42 +00:00
|
|
|
text: 'audioDevices.phone',
|
2017-10-24 08:40:39 +00:00
|
|
|
type: 'EARPIECE'
|
|
|
|
},
|
|
|
|
HEADPHONES: {
|
|
|
|
iconName: 'headset',
|
2017-11-20 11:13:42 +00:00
|
|
|
text: 'audioDevices.headphones',
|
2017-10-24 08:40:39 +00:00
|
|
|
type: 'HEADPHONES'
|
|
|
|
},
|
|
|
|
SPEAKER: {
|
|
|
|
iconName: 'volume',
|
2017-11-20 11:13:42 +00:00
|
|
|
text: 'audioDevices.speaker',
|
2017-10-24 08:40:39 +00:00
|
|
|
type: 'SPEAKER'
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2017-11-14 20:18:16 +00:00
|
|
|
* The exported React {@code Component}. {@code AudioRoutePickerDialog} is
|
|
|
|
* exported only if the {@code AudioMode} module has the capability to get / set
|
2017-10-24 08:40:39 +00:00
|
|
|
* audio devices.
|
|
|
|
*/
|
2017-11-14 20:18:16 +00:00
|
|
|
let AudioRoutePickerDialog_;
|
2017-10-24 08:40:39 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Implements a React {@code Component} which prompts the user when a password
|
|
|
|
* is required to join a conference.
|
|
|
|
*/
|
|
|
|
class AudioRoutePickerDialog extends Component<Props, State> {
|
|
|
|
state = {
|
2017-11-14 20:18:16 +00:00
|
|
|
/**
|
|
|
|
* Available audio devices, it will be set in
|
|
|
|
* {@link #componentWillMount()}.
|
|
|
|
*/
|
2017-10-24 08:40:39 +00:00
|
|
|
devices: []
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Initializes a new {@code PasswordRequiredPrompt} instance.
|
|
|
|
*
|
|
|
|
* @param {Props} props - The read-only React {@code Component} props with
|
|
|
|
* which the new instance is to be initialized.
|
|
|
|
*/
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
// Bind event handlers so they are only bound once per instance.
|
|
|
|
this._onCancel = this._onCancel.bind(this);
|
|
|
|
this._onSubmit = this._onSubmit.bind(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-11-14 20:18:16 +00:00
|
|
|
* Initializes the device list by querying {@code AudioMode}.
|
2017-10-24 08:40:39 +00:00
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
*/
|
|
|
|
componentWillMount() {
|
|
|
|
AudioMode.getAudioDevices().then(({ devices, selected }) => {
|
|
|
|
const audioDevices = [];
|
|
|
|
|
|
|
|
if (devices) {
|
|
|
|
for (const device of devices) {
|
2017-11-20 11:13:42 +00:00
|
|
|
if (deviceInfoMap[device]) {
|
|
|
|
const info = Object.assign({}, deviceInfoMap[device]);
|
2017-10-24 08:40:39 +00:00
|
|
|
|
|
|
|
info.selected = device === selected;
|
2017-11-20 11:13:42 +00:00
|
|
|
info.text = this.props.t(info.text);
|
2017-10-24 08:40:39 +00:00
|
|
|
audioDevices.push(info);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (audioDevices) {
|
2017-11-14 20:18:16 +00:00
|
|
|
// Make sure devices is alphabetically sorted.
|
|
|
|
this.setState({
|
|
|
|
devices: _.sortBy(audioDevices, 'text')
|
|
|
|
});
|
2017-10-24 08:40:39 +00:00
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dispatches a redux action to hide this sheet.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_hide() {
|
2017-11-14 20:18:16 +00:00
|
|
|
this.props.dispatch(hideDialog(AudioRoutePickerDialog_));
|
2017-10-24 08:40:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
_onCancel: () => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Cancels the dialog by hiding it.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onCancel() {
|
|
|
|
this._hide();
|
|
|
|
}
|
|
|
|
|
|
|
|
_onSubmit: (?Object) => void;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handles the selection of a device on the sheet. The selected device will
|
|
|
|
* be used by {@code AudioMode}.
|
|
|
|
*
|
|
|
|
* @param {Object} device - Object representing the selected device.
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onSubmit(device) {
|
|
|
|
this._hide();
|
|
|
|
AudioMode.setAudioDevice(device.type);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Implements React's {@link Component#render()}.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
render() {
|
2017-11-14 20:18:16 +00:00
|
|
|
const { devices } = this.state;
|
|
|
|
|
|
|
|
if (!devices.length) {
|
2017-10-24 08:40:39 +00:00
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<SimpleBottomSheet
|
|
|
|
onCancel = { this._onCancel }
|
|
|
|
onSubmit = { this._onSubmit }
|
2017-11-14 20:18:16 +00:00
|
|
|
options = { devices } />
|
2017-10-24 08:40:39 +00:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only export the dialog if we have support for getting / setting audio devices
|
|
|
|
// in AudioMode.
|
|
|
|
if (AudioMode.getAudioDevices && AudioMode.setAudioDevice) {
|
2017-11-20 11:13:42 +00:00
|
|
|
AudioRoutePickerDialog_ = translate(connect()(AudioRoutePickerDialog));
|
2017-10-24 08:40:39 +00:00
|
|
|
}
|
|
|
|
|
2017-11-14 20:18:16 +00:00
|
|
|
export default AudioRoutePickerDialog_;
|