2018-05-22 19:47:10 +00:00
|
|
|
// @flow
|
|
|
|
|
2017-08-29 20:55:35 +00:00
|
|
|
import Tooltip from '@atlaskit/tooltip';
|
2018-05-22 19:47:10 +00:00
|
|
|
import React from 'react';
|
2017-04-21 21:14:14 +00:00
|
|
|
|
2017-05-15 18:55:08 +00:00
|
|
|
import { translate } from '../../base/i18n';
|
2018-05-16 14:00:16 +00:00
|
|
|
import { CircularLabel } from '../../base/label';
|
2018-04-03 16:42:59 +00:00
|
|
|
import { MEDIA_TYPE } from '../../base/media';
|
2019-03-21 16:38:29 +00:00
|
|
|
import { connect } from '../../base/redux';
|
2018-04-03 16:42:59 +00:00
|
|
|
import { getTrackByMediaTypeAndParticipant } from '../../base/tracks';
|
2017-08-29 15:08:16 +00:00
|
|
|
|
2018-05-22 19:47:10 +00:00
|
|
|
import AbstractVideoQualityLabel, {
|
|
|
|
_abstractMapStateToProps,
|
|
|
|
type Props as AbstractProps
|
|
|
|
} from './AbstractVideoQualityLabel';
|
|
|
|
|
|
|
|
type Props = AbstractProps & {
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The message to show within the label.
|
|
|
|
*/
|
|
|
|
_labelKey: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The message to show within the label's tooltip.
|
|
|
|
*/
|
|
|
|
_tooltipKey: string,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The redux representation of the JitsiTrack displayed on large video.
|
|
|
|
*/
|
|
|
|
_videoTrack: Object
|
|
|
|
};
|
|
|
|
|
2017-11-04 00:21:33 +00:00
|
|
|
/**
|
|
|
|
* A map of video resolution (number) to translation key.
|
|
|
|
*
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
const RESOLUTION_TO_TRANSLATION_KEY = {
|
2018-05-22 19:47:10 +00:00
|
|
|
'720': 'videoStatus.hd',
|
|
|
|
'360': 'videoStatus.sd',
|
|
|
|
'180': 'videoStatus.ld'
|
2017-11-04 00:21:33 +00:00
|
|
|
};
|
2017-08-09 19:40:03 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Expected video resolutions placed into an array, sorted from lowest to
|
|
|
|
* highest resolution.
|
|
|
|
*
|
|
|
|
* @type {number[]}
|
|
|
|
*/
|
|
|
|
const RESOLUTIONS
|
2017-11-04 00:21:33 +00:00
|
|
|
= Object.keys(RESOLUTION_TO_TRANSLATION_KEY)
|
|
|
|
.map(resolution => parseInt(resolution, 10))
|
|
|
|
.sort((a, b) => a - b);
|
2017-08-09 19:40:03 +00:00
|
|
|
|
2017-04-21 21:14:14 +00:00
|
|
|
/**
|
|
|
|
* React {@code Component} responsible for displaying a label that indicates
|
|
|
|
* the displayed video state of the current conference. {@code AudioOnlyLabel}
|
|
|
|
* will display when the conference is in audio only mode. {@code HDVideoLabel}
|
|
|
|
* will display if not in audio only mode and a high-definition large video is
|
|
|
|
* being displayed.
|
|
|
|
*/
|
2018-05-22 19:47:10 +00:00
|
|
|
export class VideoQualityLabel extends AbstractVideoQualityLabel<Props> {
|
2017-05-15 18:55:08 +00:00
|
|
|
|
2017-04-21 21:14:14 +00:00
|
|
|
/**
|
|
|
|
* Implements React's {@link Component#render()}.
|
|
|
|
*
|
|
|
|
* @inheritdoc
|
2017-05-15 18:55:08 +00:00
|
|
|
* @returns {ReactElement}
|
2017-04-21 21:14:14 +00:00
|
|
|
*/
|
|
|
|
render() {
|
2017-05-23 18:58:07 +00:00
|
|
|
const {
|
|
|
|
_audioOnly,
|
2018-05-16 14:00:16 +00:00
|
|
|
_labelKey,
|
|
|
|
_tooltipKey,
|
2018-04-03 16:42:59 +00:00
|
|
|
_videoTrack,
|
2017-08-29 20:55:35 +00:00
|
|
|
t
|
2017-05-23 18:58:07 +00:00
|
|
|
} = this.props;
|
2017-05-15 18:55:08 +00:00
|
|
|
|
2018-04-03 16:42:59 +00:00
|
|
|
|
2018-05-16 14:00:16 +00:00
|
|
|
let className, labelContent, tooltipKey;
|
2018-04-03 16:42:59 +00:00
|
|
|
|
|
|
|
if (_audioOnly) {
|
2018-05-16 14:00:16 +00:00
|
|
|
className = 'audio-only';
|
|
|
|
labelContent = t('videoStatus.audioOnly');
|
2018-04-03 16:42:59 +00:00
|
|
|
tooltipKey = 'videoStatus.labelTooltipAudioOnly';
|
|
|
|
} else if (!_videoTrack || _videoTrack.muted) {
|
2018-05-16 14:00:16 +00:00
|
|
|
className = 'no-video';
|
|
|
|
labelContent = t('videoStatus.audioOnly');
|
2018-04-03 16:42:59 +00:00
|
|
|
tooltipKey = 'videoStatus.labelTooiltipNoVideo';
|
|
|
|
} else {
|
2018-05-16 14:00:16 +00:00
|
|
|
className = 'current-video-quality';
|
|
|
|
labelContent = t(_labelKey);
|
|
|
|
tooltipKey = _tooltipKey;
|
2018-04-03 16:42:59 +00:00
|
|
|
}
|
|
|
|
|
2017-05-23 18:58:07 +00:00
|
|
|
|
2017-05-15 18:55:08 +00:00
|
|
|
return (
|
2018-05-16 14:00:16 +00:00
|
|
|
<Tooltip
|
|
|
|
content = { t(tooltipKey) }
|
|
|
|
position = { 'left' }>
|
|
|
|
<CircularLabel
|
|
|
|
className = { className }
|
2018-05-22 19:44:40 +00:00
|
|
|
id = 'videoResolutionLabel'
|
|
|
|
label = { labelContent } />
|
2018-05-16 14:00:16 +00:00
|
|
|
</Tooltip>
|
2017-05-15 18:55:08 +00:00
|
|
|
);
|
|
|
|
}
|
2018-05-16 14:00:16 +00:00
|
|
|
}
|
2017-05-15 18:55:08 +00:00
|
|
|
|
2018-05-16 14:00:16 +00:00
|
|
|
/**
|
|
|
|
* Matches the passed in resolution with a translation keys for describing
|
|
|
|
* the resolution. The passed in resolution will be matched with a known
|
|
|
|
* resolution that it is at least greater than or equal to.
|
|
|
|
*
|
|
|
|
* @param {number} resolution - The video height to match with a
|
|
|
|
* translation.
|
|
|
|
* @private
|
|
|
|
* @returns {Object}
|
|
|
|
*/
|
|
|
|
function _mapResolutionToTranslationsKeys(resolution) {
|
|
|
|
// Set the default matching resolution of the lowest just in case a match is
|
|
|
|
// not found.
|
|
|
|
let highestMatchingResolution = RESOLUTIONS[0];
|
2017-08-09 19:40:03 +00:00
|
|
|
|
2018-05-16 14:00:16 +00:00
|
|
|
for (let i = 0; i < RESOLUTIONS.length; i++) {
|
|
|
|
const knownResolution = RESOLUTIONS[i];
|
2018-04-03 17:07:57 +00:00
|
|
|
|
2018-05-16 14:00:16 +00:00
|
|
|
if (resolution >= knownResolution) {
|
|
|
|
highestMatchingResolution = knownResolution;
|
|
|
|
} else {
|
|
|
|
break;
|
|
|
|
}
|
2017-05-15 18:55:08 +00:00
|
|
|
}
|
2018-05-16 14:00:16 +00:00
|
|
|
|
|
|
|
const labelKey
|
|
|
|
= RESOLUTION_TO_TRANSLATION_KEY[highestMatchingResolution];
|
|
|
|
|
|
|
|
return {
|
|
|
|
labelKey,
|
|
|
|
tooltipKey: `${labelKey}Tooltip`
|
|
|
|
};
|
2017-04-21 21:14:14 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2017-08-09 19:40:03 +00:00
|
|
|
* Maps (parts of) the Redux state to the associated {@code VideoQualityLabel}'s
|
2017-04-21 21:14:14 +00:00
|
|
|
* props.
|
|
|
|
*
|
|
|
|
* @param {Object} state - The Redux state.
|
|
|
|
* @private
|
|
|
|
* @returns {{
|
2018-05-16 14:00:16 +00:00
|
|
|
* _labelKey: string,
|
|
|
|
* _tooltipKey: string,
|
2018-04-03 16:42:59 +00:00
|
|
|
* _videoTrack: Object
|
2017-04-21 21:14:14 +00:00
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
function _mapStateToProps(state) {
|
2018-05-16 14:00:16 +00:00
|
|
|
const { audioOnly } = state['features/base/conference'];
|
2018-04-03 16:42:59 +00:00
|
|
|
const { resolution, participantId } = state['features/large-video'];
|
|
|
|
const videoTrackOnLargeVideo = getTrackByMediaTypeAndParticipant(
|
|
|
|
state['features/base/tracks'],
|
|
|
|
MEDIA_TYPE.VIDEO,
|
|
|
|
participantId
|
|
|
|
);
|
2017-04-21 21:14:14 +00:00
|
|
|
|
2018-05-16 14:00:16 +00:00
|
|
|
const translationKeys
|
|
|
|
= audioOnly ? {} : _mapResolutionToTranslationsKeys(resolution);
|
|
|
|
|
2017-04-21 21:14:14 +00:00
|
|
|
return {
|
2018-05-22 19:47:10 +00:00
|
|
|
..._abstractMapStateToProps(state),
|
2018-05-16 14:00:16 +00:00
|
|
|
_labelKey: translationKeys.labelKey,
|
|
|
|
_tooltipKey: translationKeys.tooltipKey,
|
2018-04-03 16:42:59 +00:00
|
|
|
_videoTrack: videoTrackOnLargeVideo
|
2017-04-21 21:14:14 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2017-08-09 19:40:03 +00:00
|
|
|
export default translate(connect(_mapStateToProps)(VideoQualityLabel));
|