2016-10-05 14:36:59 +00:00
|
|
|
import React, { Component } from 'react';
|
|
|
|
|
|
|
|
import { appNavigate } from '../../app';
|
2016-12-12 19:49:23 +00:00
|
|
|
import { toggleAudioMuted, toggleVideoMuted } from '../../base/media';
|
2016-10-05 14:36:59 +00:00
|
|
|
import { ColorPalette } from '../../base/styles';
|
2016-12-13 09:14:04 +00:00
|
|
|
import { beginRoomLockRequest } from '../../room-lock';
|
2016-10-05 14:36:59 +00:00
|
|
|
|
|
|
|
import { styles } from './styles';
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Abstract (base) class for the conference toolbar.
|
|
|
|
*
|
|
|
|
* @abstract
|
|
|
|
*/
|
|
|
|
export class AbstractToolbar extends Component {
|
2016-12-01 01:52:39 +00:00
|
|
|
/**
|
|
|
|
* AbstractToolbar component's property types.
|
|
|
|
*
|
|
|
|
* @static
|
|
|
|
*/
|
|
|
|
static propTypes = {
|
2017-01-28 03:36:20 +00:00
|
|
|
_audioMuted: React.PropTypes.bool,
|
2016-12-12 19:49:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The indicator which determines whether the conference is
|
|
|
|
* locked/password-protected.
|
2017-01-28 03:36:20 +00:00
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @type {boolean}
|
2016-12-12 19:49:23 +00:00
|
|
|
*/
|
2017-01-28 03:36:20 +00:00
|
|
|
_locked: React.PropTypes.bool,
|
|
|
|
_videoMuted: React.PropTypes.bool,
|
|
|
|
dispatch: React.PropTypes.func,
|
2016-12-01 01:52:39 +00:00
|
|
|
visible: React.PropTypes.bool.isRequired
|
|
|
|
}
|
|
|
|
|
2016-10-05 14:36:59 +00:00
|
|
|
/**
|
|
|
|
* Initializes a new AbstractToolbar instance.
|
|
|
|
*
|
|
|
|
* @param {Object} props - The read-only React Component props with which
|
|
|
|
* the new instance is to be initialized.
|
|
|
|
*/
|
|
|
|
constructor(props) {
|
|
|
|
super(props);
|
|
|
|
|
|
|
|
// Bind event handlers so they are only bound once for every instance.
|
|
|
|
this._onHangup = this._onHangup.bind(this);
|
2016-12-13 09:14:04 +00:00
|
|
|
this._onRoomLock = this._onRoomLock.bind(this);
|
2016-10-05 14:36:59 +00:00
|
|
|
this._toggleAudio = this._toggleAudio.bind(this);
|
|
|
|
this._toggleVideo = this._toggleVideo.bind(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Gets the styles for a button that toggles the mute state of a specific
|
|
|
|
* media type.
|
|
|
|
*
|
|
|
|
* @param {string} mediaType - The {@link MEDIA_TYPE} associated with the
|
|
|
|
* button to get styles for.
|
|
|
|
* @protected
|
|
|
|
* @returns {{
|
|
|
|
* iconName: string,
|
2016-12-12 19:49:23 +00:00
|
|
|
* iconStyle: Object,
|
|
|
|
* style: Object
|
2016-10-05 14:36:59 +00:00
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
_getMuteButtonStyles(mediaType) {
|
|
|
|
let iconName;
|
|
|
|
let iconStyle;
|
2016-12-12 19:49:23 +00:00
|
|
|
let style = styles.primaryToolbarButton;
|
2016-10-05 14:36:59 +00:00
|
|
|
|
2017-01-28 03:36:20 +00:00
|
|
|
if (this.props[`_${mediaType}Muted`]) {
|
2016-10-05 14:36:59 +00:00
|
|
|
iconName = this[`${mediaType}MutedIcon`];
|
|
|
|
iconStyle = styles.whiteIcon;
|
2016-12-12 19:49:23 +00:00
|
|
|
style = {
|
|
|
|
...style,
|
|
|
|
backgroundColor: ColorPalette.buttonUnderlay
|
|
|
|
};
|
2016-10-05 14:36:59 +00:00
|
|
|
} else {
|
|
|
|
iconName = this[`${mediaType}Icon`];
|
|
|
|
iconStyle = styles.icon;
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
iconName,
|
2016-12-12 19:49:23 +00:00
|
|
|
iconStyle,
|
|
|
|
style
|
2016-10-05 14:36:59 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dispatches action to leave the current conference.
|
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onHangup() {
|
|
|
|
// XXX We don't know here which value is effectively/internally used
|
|
|
|
// when there's no valid room name to join. It isn't our business to
|
|
|
|
// know that anyway. The undefined value is our expression of (1) the
|
|
|
|
// lack of knowledge & (2) the desire to no longer have a valid room
|
|
|
|
// name to join.
|
|
|
|
this.props.dispatch(appNavigate(undefined));
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-12-13 09:14:04 +00:00
|
|
|
* Dispatches an action to set the lock i.e. password protection of the
|
|
|
|
* conference/room.
|
2016-10-05 14:36:59 +00:00
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
2016-12-13 09:14:04 +00:00
|
|
|
_onRoomLock() {
|
|
|
|
this.props.dispatch(beginRoomLockRequest());
|
2016-10-05 14:36:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2016-12-13 09:14:04 +00:00
|
|
|
* Dispatches an action to toggle the mute state of the audio/microphone.
|
2016-12-12 19:49:23 +00:00
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
2016-12-13 09:14:04 +00:00
|
|
|
_toggleAudio() {
|
|
|
|
this.props.dispatch(toggleAudioMuted());
|
2016-12-12 19:49:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Dispatches an action to toggle the mute state of the video/camera.
|
2016-10-05 14:36:59 +00:00
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_toggleVideo() {
|
|
|
|
this.props.dispatch(toggleVideoMuted());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps parts of media state to component props.
|
|
|
|
*
|
|
|
|
* @param {Object} state - Redux state.
|
2017-01-28 23:34:57 +00:00
|
|
|
* @protected
|
2017-01-28 03:36:20 +00:00
|
|
|
* @returns {{
|
|
|
|
* _audioMuted: boolean,
|
|
|
|
* _locked: boolean,
|
|
|
|
* _videoMuted: boolean
|
|
|
|
* }}
|
2016-10-05 14:36:59 +00:00
|
|
|
*/
|
2017-01-28 23:34:57 +00:00
|
|
|
export function _mapStateToProps(state) {
|
2016-12-12 19:49:23 +00:00
|
|
|
const conference = state['features/base/conference'];
|
2016-10-05 14:36:59 +00:00
|
|
|
const media = state['features/base/media'];
|
|
|
|
|
|
|
|
return {
|
2017-01-28 03:36:20 +00:00
|
|
|
_audioMuted: media.audio.muted,
|
2016-12-12 19:49:23 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The indicator which determines whether the conference is
|
|
|
|
* locked/password-protected.
|
|
|
|
*
|
2017-01-28 03:36:20 +00:00
|
|
|
* @protected
|
2016-12-12 19:49:23 +00:00
|
|
|
* @type {boolean}
|
|
|
|
*/
|
2017-01-28 03:36:20 +00:00
|
|
|
_locked: conference.locked,
|
|
|
|
_videoMuted: media.video.muted
|
2016-10-05 14:36:59 +00:00
|
|
|
};
|
|
|
|
}
|