2019-03-19 15:42:25 +00:00
|
|
|
// @flow
|
2018-10-30 05:02:23 +00:00
|
|
|
|
2018-03-07 00:28:19 +00:00
|
|
|
import InlineMessage from '@atlaskit/inline-message';
|
|
|
|
import React, { Component } from 'react';
|
2019-03-19 15:42:25 +00:00
|
|
|
import type { Dispatch } from 'redux';
|
2018-03-07 00:28:19 +00:00
|
|
|
|
2019-07-31 12:47:52 +00:00
|
|
|
import { createToolbarEvent, sendAnalytics } from '../../analytics';
|
|
|
|
import { setAudioOnly } from '../../base/audio-only';
|
|
|
|
import { VIDEO_QUALITY_LEVELS, setPreferredReceiverVideoQuality } from '../../base/conference';
|
2018-03-07 00:28:19 +00:00
|
|
|
import { translate } from '../../base/i18n';
|
|
|
|
import JitsiMeetJS from '../../base/lib-jitsi-meet';
|
2019-03-21 16:38:29 +00:00
|
|
|
import { connect } from '../../base/redux';
|
2018-03-07 00:28:19 +00:00
|
|
|
|
2019-08-21 14:50:00 +00:00
|
|
|
import logger from '../logger';
|
2018-03-07 00:28:19 +00:00
|
|
|
|
|
|
|
const {
|
|
|
|
HIGH,
|
|
|
|
STANDARD,
|
|
|
|
LOW
|
|
|
|
} = VIDEO_QUALITY_LEVELS;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates an analytics event for a press of one of the buttons in the video
|
|
|
|
* quality dialog.
|
|
|
|
*
|
|
|
|
* @param {string} quality - The quality which was selected.
|
|
|
|
* @returns {Object} The event in a format suitable for sending via
|
|
|
|
* sendAnalytics.
|
|
|
|
*/
|
|
|
|
const createEvent = function(quality) {
|
|
|
|
return createToolbarEvent(
|
|
|
|
'video.quality',
|
|
|
|
{
|
|
|
|
quality
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2018-10-30 05:02:23 +00:00
|
|
|
* The type of the React {@code Component} props of {@link VideoQualitySlider}.
|
2018-03-07 00:28:19 +00:00
|
|
|
*/
|
2018-10-30 05:02:23 +00:00
|
|
|
type Props = {
|
|
|
|
|
2018-03-07 00:28:19 +00:00
|
|
|
/**
|
2018-10-30 05:02:23 +00:00
|
|
|
* Whether or not the conference is in audio only mode.
|
2018-03-07 00:28:19 +00:00
|
|
|
*/
|
2018-10-30 05:02:23 +00:00
|
|
|
_audioOnly: Boolean,
|
2018-03-07 00:28:19 +00:00
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
/**
|
|
|
|
* Whether or not the conference is in peer to peer mode.
|
|
|
|
*/
|
|
|
|
_p2p: Boolean,
|
2018-03-07 00:28:19 +00:00
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
/**
|
|
|
|
* The currently configured maximum quality resolution to be received
|
|
|
|
* from remote participants.
|
|
|
|
*/
|
|
|
|
_receiverVideoQuality: Number,
|
2018-03-07 00:28:19 +00:00
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
/**
|
|
|
|
* Whether or not displaying video is supported in the current
|
|
|
|
* environment. If false, the slider will be disabled.
|
|
|
|
*/
|
|
|
|
_videoSupported: Boolean,
|
2018-03-07 00:28:19 +00:00
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
/**
|
|
|
|
* Invoked to request toggling of audio only mode.
|
|
|
|
*/
|
2019-03-19 15:42:25 +00:00
|
|
|
dispatch: Dispatch<any>,
|
2018-03-07 00:28:19 +00:00
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
/**
|
|
|
|
* Invoked to obtain translated strings.
|
|
|
|
*/
|
|
|
|
t: Function
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Implements a React {@link Component} which displays a slider for selecting a
|
|
|
|
* new receive video quality.
|
|
|
|
*
|
|
|
|
* @extends Component
|
|
|
|
*/
|
|
|
|
class VideoQualitySlider extends Component<Props> {
|
|
|
|
_sliderOptions: Array<Object>;
|
2018-03-07 00:28:19 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Initializes a new {@code VideoQualitySlider} 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._enableAudioOnly = this._enableAudioOnly.bind(this);
|
|
|
|
this._enableHighDefinition = this._enableHighDefinition.bind(this);
|
|
|
|
this._enableLowDefinition = this._enableLowDefinition.bind(this);
|
|
|
|
this._enableStandardDefinition
|
|
|
|
= this._enableStandardDefinition.bind(this);
|
|
|
|
this._onSliderChange = this._onSliderChange.bind(this);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* An array of configuration options for displaying a choice in the
|
|
|
|
* input. The onSelect callback will be invoked when the option is
|
|
|
|
* selected and videoQuality helps determine which choice matches with
|
|
|
|
* the currently active quality level.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @type {Object[]}
|
|
|
|
*/
|
|
|
|
this._sliderOptions = [
|
|
|
|
{
|
|
|
|
audioOnly: true,
|
|
|
|
onSelect: this._enableAudioOnly,
|
|
|
|
textKey: 'audioOnly.audioOnly'
|
|
|
|
},
|
|
|
|
{
|
|
|
|
onSelect: this._enableLowDefinition,
|
|
|
|
textKey: 'videoStatus.lowDefinition',
|
|
|
|
videoQuality: LOW
|
|
|
|
},
|
|
|
|
{
|
|
|
|
onSelect: this._enableStandardDefinition,
|
|
|
|
textKey: 'videoStatus.standardDefinition',
|
|
|
|
videoQuality: STANDARD
|
|
|
|
},
|
|
|
|
{
|
|
|
|
onSelect: this._enableHighDefinition,
|
|
|
|
textKey: 'videoStatus.highDefinition',
|
|
|
|
videoQuality: HIGH
|
|
|
|
}
|
|
|
|
];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Implements React's {@link Component#render()}.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
render() {
|
|
|
|
const { _audioOnly, _p2p, _videoSupported, t } = this.props;
|
|
|
|
const activeSliderOption = this._mapCurrentQualityToSliderValue();
|
|
|
|
|
|
|
|
let classNames = 'video-quality-dialog';
|
|
|
|
let warning = null;
|
|
|
|
|
|
|
|
if (!_videoSupported) {
|
|
|
|
classNames += ' video-not-supported';
|
|
|
|
warning = this._renderAudioOnlyLockedMessage();
|
|
|
|
} else if (_p2p && !_audioOnly) {
|
|
|
|
warning = this._renderP2PMessage();
|
|
|
|
}
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div className = { classNames }>
|
|
|
|
<h3 className = 'video-quality-dialog-title'>
|
|
|
|
{ t('videoStatus.callQuality') }
|
|
|
|
</h3>
|
|
|
|
<div className = { warning ? '' : 'hide-warning' }>
|
|
|
|
{ warning }
|
|
|
|
</div>
|
|
|
|
<div className = 'video-quality-dialog-contents'>
|
|
|
|
<div className = 'video-quality-dialog-slider-container'>
|
|
|
|
{ /* FIXME: onChange and onMouseUp are both used for
|
|
|
|
* compatibility with IE11. This workaround can be
|
|
|
|
* removed after upgrading to React 16.
|
|
|
|
*/ }
|
|
|
|
<input
|
|
|
|
className = 'video-quality-dialog-slider'
|
|
|
|
disabled = { !_videoSupported }
|
|
|
|
max = { this._sliderOptions.length - 1 }
|
|
|
|
min = '0'
|
|
|
|
onChange = { this._onSliderChange }
|
|
|
|
onMouseUp = { this._onSliderChange }
|
|
|
|
step = '1'
|
|
|
|
type = 'range'
|
|
|
|
value
|
|
|
|
= { activeSliderOption } />
|
|
|
|
|
|
|
|
</div>
|
|
|
|
<div className = 'video-quality-dialog-labels'>
|
|
|
|
{ this._createLabels(activeSliderOption) }
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates a React Element for notifying that the browser is in audio only
|
|
|
|
* and cannot be changed.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
_renderAudioOnlyLockedMessage() {
|
|
|
|
const { t } = this.props;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<InlineMessage
|
|
|
|
title = { t('videoStatus.onlyAudioAvailable') }>
|
|
|
|
{ t('videoStatus.onlyAudioSupported') }
|
|
|
|
</InlineMessage>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates React Elements for notifying that peer to peer is enabled.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {ReactElement}
|
|
|
|
*/
|
|
|
|
_renderP2PMessage() {
|
|
|
|
const { t } = this.props;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<InlineMessage
|
|
|
|
secondaryText = { t('videoStatus.recHighDefinitionOnly') }
|
|
|
|
title = { t('videoStatus.p2pEnabled') }>
|
|
|
|
{ t('videoStatus.p2pVideoQualityDescription') }
|
|
|
|
</InlineMessage>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Creates React Elements to display mock tick marks with associated labels.
|
|
|
|
*
|
|
|
|
* @param {number} activeLabelIndex - Which of the sliderOptions should
|
|
|
|
* display as currently active.
|
|
|
|
* @private
|
|
|
|
* @returns {ReactElement[]}
|
|
|
|
*/
|
|
|
|
_createLabels(activeLabelIndex) {
|
|
|
|
const labelsCount = this._sliderOptions.length;
|
|
|
|
const maxWidthOfLabel = `${100 / labelsCount}%`;
|
|
|
|
|
|
|
|
return this._sliderOptions.map((sliderOption, index) => {
|
|
|
|
const style = {
|
|
|
|
maxWidth: maxWidthOfLabel,
|
|
|
|
left: `${(index * 100) / (labelsCount - 1)}%`
|
|
|
|
};
|
|
|
|
|
|
|
|
const isActiveClass = activeLabelIndex === index ? 'active' : '';
|
|
|
|
const className
|
|
|
|
= `video-quality-dialog-label-container ${isActiveClass}`;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<div
|
|
|
|
className = { className }
|
|
|
|
key = { index }
|
|
|
|
style = { style }>
|
|
|
|
<div className = 'video-quality-dialog-label'>
|
|
|
|
{ this.props.t(sliderOption.textKey) }
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
_enableAudioOnly: () => void;
|
|
|
|
|
2018-03-07 00:28:19 +00:00
|
|
|
/**
|
|
|
|
* Dispatches an action to enable audio only mode.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_enableAudioOnly() {
|
|
|
|
sendAnalytics(createEvent('audio.only'));
|
|
|
|
logger.log('Video quality: audio only enabled');
|
|
|
|
this.props.dispatch(setAudioOnly(true));
|
|
|
|
}
|
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
_enableHighDefinition: () => void;
|
|
|
|
|
2018-03-07 00:28:19 +00:00
|
|
|
/**
|
|
|
|
* Handles the action of the high definition video being selected.
|
|
|
|
* Dispatches an action to receive high quality video from remote
|
|
|
|
* participants.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_enableHighDefinition() {
|
|
|
|
sendAnalytics(createEvent('high'));
|
|
|
|
logger.log('Video quality: high enabled');
|
2018-07-24 20:43:09 +00:00
|
|
|
this._setPreferredVideoQuality(HIGH);
|
2018-03-07 00:28:19 +00:00
|
|
|
}
|
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
_enableLowDefinition: () => void;
|
|
|
|
|
2018-03-07 00:28:19 +00:00
|
|
|
/**
|
|
|
|
* Dispatches an action to receive low quality video from remote
|
|
|
|
* participants.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_enableLowDefinition() {
|
|
|
|
sendAnalytics(createEvent('low'));
|
|
|
|
logger.log('Video quality: low enabled');
|
2018-07-24 20:43:09 +00:00
|
|
|
this._setPreferredVideoQuality(LOW);
|
2018-03-07 00:28:19 +00:00
|
|
|
}
|
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
_enableStandardDefinition: () => void;
|
|
|
|
|
2018-03-07 00:28:19 +00:00
|
|
|
/**
|
|
|
|
* Dispatches an action to receive standard quality video from remote
|
|
|
|
* participants.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_enableStandardDefinition() {
|
|
|
|
sendAnalytics(createEvent('standard'));
|
|
|
|
logger.log('Video quality: standard enabled');
|
2018-07-24 20:43:09 +00:00
|
|
|
this._setPreferredVideoQuality(STANDARD);
|
2018-03-07 00:28:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Matches the current video quality state with corresponding index of the
|
|
|
|
* component's slider options.
|
|
|
|
*
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_mapCurrentQualityToSliderValue() {
|
2018-07-23 22:42:57 +00:00
|
|
|
const { _audioOnly, _receiverVideoQuality } = this.props;
|
2018-03-07 00:28:19 +00:00
|
|
|
const { _sliderOptions } = this;
|
|
|
|
|
|
|
|
if (_audioOnly) {
|
|
|
|
const audioOnlyOption = _sliderOptions.find(
|
|
|
|
({ audioOnly }) => audioOnly);
|
|
|
|
|
|
|
|
return _sliderOptions.indexOf(audioOnlyOption);
|
|
|
|
}
|
|
|
|
|
|
|
|
const matchingOption = _sliderOptions.find(
|
2018-07-23 22:42:57 +00:00
|
|
|
({ videoQuality }) => videoQuality === _receiverVideoQuality);
|
2018-03-07 00:28:19 +00:00
|
|
|
|
|
|
|
return _sliderOptions.indexOf(matchingOption);
|
|
|
|
}
|
|
|
|
|
2018-10-30 05:02:23 +00:00
|
|
|
_onSliderChange: () => void;
|
|
|
|
|
2018-03-07 00:28:19 +00:00
|
|
|
/**
|
|
|
|
* Invokes a callback when the selected video quality changes.
|
|
|
|
*
|
|
|
|
* @param {Object} event - The slider's change event.
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onSliderChange(event) {
|
2018-07-23 22:42:57 +00:00
|
|
|
const { _audioOnly, _receiverVideoQuality } = this.props;
|
2018-03-07 00:28:19 +00:00
|
|
|
const {
|
|
|
|
audioOnly,
|
|
|
|
onSelect,
|
|
|
|
videoQuality
|
|
|
|
} = this._sliderOptions[event.target.value];
|
|
|
|
|
|
|
|
// Take no action if the newly chosen option does not change audio only
|
|
|
|
// or video quality state.
|
|
|
|
if ((_audioOnly && audioOnly)
|
2018-07-23 22:42:57 +00:00
|
|
|
|| (!_audioOnly && videoQuality === _receiverVideoQuality)) {
|
2018-03-07 00:28:19 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
onSelect();
|
|
|
|
}
|
2018-07-24 20:43:09 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Helper for changing the preferred maximum video quality to receive and
|
|
|
|
* disable audio only.
|
|
|
|
*
|
|
|
|
* @param {number} qualityLevel - The new maximum video quality. Should be
|
|
|
|
* a value enumerated in {@code VIDEO_QUALITY_LEVELS}.
|
|
|
|
* @private
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_setPreferredVideoQuality(qualityLevel) {
|
|
|
|
this.props.dispatch(setPreferredReceiverVideoQuality(qualityLevel));
|
|
|
|
|
|
|
|
if (this.props._audioOnly) {
|
|
|
|
this.props.dispatch(setAudioOnly(false));
|
|
|
|
}
|
|
|
|
}
|
2018-03-07 00:28:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Maps (parts of) the Redux state to the associated props for the
|
|
|
|
* {@code VideoQualitySlider} component.
|
|
|
|
*
|
|
|
|
* @param {Object} state - The Redux state.
|
|
|
|
* @private
|
|
|
|
* @returns {{
|
|
|
|
* _audioOnly: boolean,
|
|
|
|
* _p2p: boolean,
|
2018-07-23 22:42:57 +00:00
|
|
|
* _receiverVideoQuality: boolean
|
2018-03-07 00:28:19 +00:00
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
function _mapStateToProps(state) {
|
2019-07-31 12:47:52 +00:00
|
|
|
const { enabled: audioOnly } = state['features/base/audio-only'];
|
|
|
|
const { p2p, preferredReceiverVideoQuality } = state['features/base/conference'];
|
2018-03-07 00:28:19 +00:00
|
|
|
|
|
|
|
return {
|
|
|
|
_audioOnly: audioOnly,
|
|
|
|
_p2p: p2p,
|
2018-07-23 22:42:57 +00:00
|
|
|
_receiverVideoQuality: preferredReceiverVideoQuality,
|
2018-03-07 00:28:19 +00:00
|
|
|
_videoSupported: JitsiMeetJS.mediaDevices.supportsVideo()
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
export default translate(connect(_mapStateToProps)(VideoQualitySlider));
|