2018-09-13 15:20:22 +00:00
|
|
|
// @flow
|
|
|
|
|
2019-09-13 10:06:02 +00:00
|
|
|
import React from 'react';
|
2019-03-29 13:26:49 +00:00
|
|
|
import { View } from 'react-native';
|
2019-03-19 15:42:25 +00:00
|
|
|
import type { Dispatch } from 'redux';
|
2016-10-05 14:36:59 +00:00
|
|
|
|
2019-01-22 10:35:28 +00:00
|
|
|
import { ColorSchemeRegistry } from '../../../base/color-scheme';
|
2018-12-19 18:40:17 +00:00
|
|
|
import { openDialog } from '../../../base/dialog';
|
2019-09-13 10:00:08 +00:00
|
|
|
import { MEDIA_TYPE, VIDEO_TYPE } from '../../../base/media';
|
2017-02-27 20:37:53 +00:00
|
|
|
import {
|
|
|
|
PARTICIPANT_ROLE,
|
|
|
|
ParticipantView,
|
2019-09-12 13:03:20 +00:00
|
|
|
getParticipantCount,
|
2019-05-22 10:00:17 +00:00
|
|
|
isEveryoneModerator,
|
2017-02-27 20:37:53 +00:00
|
|
|
pinParticipant
|
2018-05-10 23:01:55 +00:00
|
|
|
} from '../../../base/participants';
|
|
|
|
import { Container } from '../../../base/react';
|
2019-03-21 16:38:29 +00:00
|
|
|
import { connect } from '../../../base/redux';
|
2019-01-22 10:35:28 +00:00
|
|
|
import { StyleType } from '../../../base/styles';
|
2018-05-10 23:01:55 +00:00
|
|
|
import { getTrackByMediaTypeAndParticipant } from '../../../base/tracks';
|
2019-04-15 16:23:28 +00:00
|
|
|
import { ConnectionIndicator } from '../../../connection-indicator';
|
2019-04-11 10:04:50 +00:00
|
|
|
import { DisplayNameLabel } from '../../../display-name';
|
2018-12-19 18:40:17 +00:00
|
|
|
import { RemoteVideoMenu } from '../../../remote-video-menu';
|
2019-06-19 12:44:39 +00:00
|
|
|
import { toggleToolboxVisible } from '../../../toolbox';
|
2018-12-19 18:40:17 +00:00
|
|
|
|
2018-05-10 23:01:55 +00:00
|
|
|
import AudioMutedIndicator from './AudioMutedIndicator';
|
|
|
|
import DominantSpeakerIndicator from './DominantSpeakerIndicator';
|
|
|
|
import ModeratorIndicator from './ModeratorIndicator';
|
2019-03-27 10:23:41 +00:00
|
|
|
import RaisedHandIndicator from './RaisedHandIndicator';
|
2018-05-10 23:01:55 +00:00
|
|
|
import VideoMutedIndicator from './VideoMutedIndicator';
|
2020-05-20 10:57:03 +00:00
|
|
|
import styles, { AVATAR_SIZE } from './styles';
|
2018-02-05 10:57:40 +00:00
|
|
|
|
2016-10-05 14:36:59 +00:00
|
|
|
/**
|
2018-09-13 15:20:22 +00:00
|
|
|
* Thumbnail component's property types.
|
2016-10-05 14:36:59 +00:00
|
|
|
*/
|
2018-09-13 15:20:22 +00:00
|
|
|
type Props = {
|
|
|
|
|
2016-12-01 01:52:39 +00:00
|
|
|
/**
|
2019-09-13 10:00:08 +00:00
|
|
|
* Whether local audio (microphone) is muted or not.
|
2018-09-13 15:20:22 +00:00
|
|
|
*/
|
2019-09-13 10:00:08 +00:00
|
|
|
_audioMuted: boolean,
|
2018-09-13 15:20:22 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The Redux representation of the state "features/large-video".
|
|
|
|
*/
|
|
|
|
_largeVideo: Object,
|
|
|
|
|
2019-01-05 16:49:21 +00:00
|
|
|
/**
|
|
|
|
* Handles click/tap event on the thumbnail.
|
|
|
|
*/
|
|
|
|
_onClick: ?Function,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handles long press on the thumbnail.
|
|
|
|
*/
|
|
|
|
_onShowRemoteVideoMenu: ?Function,
|
|
|
|
|
2019-09-12 13:03:20 +00:00
|
|
|
/**
|
|
|
|
* Whether to show the dominant speaker indicator or not.
|
|
|
|
*/
|
2019-09-13 09:51:49 +00:00
|
|
|
_renderDominantSpeakerIndicator: boolean,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Whether to show the moderator indicator or not.
|
|
|
|
*/
|
|
|
|
_renderModeratorIndicator: boolean,
|
2019-09-12 13:03:20 +00:00
|
|
|
|
2019-01-22 10:35:28 +00:00
|
|
|
/**
|
|
|
|
* The color-schemed stylesheet of the feature.
|
|
|
|
*/
|
|
|
|
_styles: StyleType,
|
|
|
|
|
2018-09-13 15:20:22 +00:00
|
|
|
/**
|
|
|
|
* The Redux representation of the participant's video track.
|
|
|
|
*/
|
|
|
|
_videoTrack: Object,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If true, there will be no color overlay (tint) on the thumbnail
|
|
|
|
* indicating the participant associated with the thumbnail is displayed on
|
|
|
|
* large video. By default there will be a tint.
|
2016-12-01 01:52:39 +00:00
|
|
|
*/
|
2018-09-13 15:20:22 +00:00
|
|
|
disableTint?: boolean,
|
2016-12-01 01:52:39 +00:00
|
|
|
|
2018-09-13 15:20:22 +00:00
|
|
|
/**
|
|
|
|
* Invoked to trigger state changes in Redux.
|
|
|
|
*/
|
2019-03-19 15:42:25 +00:00
|
|
|
dispatch: Dispatch<any>,
|
2018-09-13 15:20:22 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The Redux representation of the participant to display.
|
|
|
|
*/
|
|
|
|
participant: Object,
|
|
|
|
|
2019-04-11 10:04:50 +00:00
|
|
|
/**
|
|
|
|
* Whether to display or hide the display name of the participant in the thumbnail.
|
|
|
|
*/
|
|
|
|
renderDisplayName: ?boolean,
|
|
|
|
|
2018-09-13 15:20:22 +00:00
|
|
|
/**
|
|
|
|
* Optional styling to add or override on the Thumbnail component root.
|
|
|
|
*/
|
2019-06-19 12:44:39 +00:00
|
|
|
styleOverrides?: Object,
|
|
|
|
|
|
|
|
/**
|
|
|
|
* If true, it tells the thumbnail that it needs to behave differently. E.g. react differently to a single tap.
|
|
|
|
*/
|
|
|
|
tileView?: boolean
|
2018-09-13 15:20:22 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* React component for video thumbnail.
|
|
|
|
*
|
2019-09-13 10:06:02 +00:00
|
|
|
* @param {Props} props - Properties passed to this functional component.
|
|
|
|
* @returns {Component} - A React component.
|
2018-09-13 15:20:22 +00:00
|
|
|
*/
|
2019-09-13 10:06:02 +00:00
|
|
|
function Thumbnail(props: Props) {
|
|
|
|
const {
|
|
|
|
_audioMuted: audioMuted,
|
|
|
|
_largeVideo: largeVideo,
|
|
|
|
_onClick,
|
|
|
|
_onShowRemoteVideoMenu,
|
|
|
|
_renderDominantSpeakerIndicator: renderDominantSpeakerIndicator,
|
|
|
|
_renderModeratorIndicator: renderModeratorIndicator,
|
|
|
|
_styles,
|
|
|
|
_videoTrack: videoTrack,
|
|
|
|
disableTint,
|
|
|
|
participant,
|
|
|
|
renderDisplayName,
|
|
|
|
tileView
|
|
|
|
} = props;
|
|
|
|
|
|
|
|
const participantId = participant.id;
|
|
|
|
const participantInLargeVideo
|
|
|
|
= participantId === largeVideo.participantId;
|
|
|
|
const videoMuted = !videoTrack || videoTrack.muted;
|
|
|
|
const isScreenShare = videoTrack && videoTrack.videoType === VIDEO_TYPE.DESKTOP;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<Container
|
|
|
|
onClick = { _onClick }
|
|
|
|
onLongPress = { participant.local ? undefined : _onShowRemoteVideoMenu }
|
|
|
|
style = { [
|
|
|
|
styles.thumbnail,
|
|
|
|
participant.pinned && !tileView
|
|
|
|
? _styles.thumbnailPinned : null,
|
2019-09-16 11:15:17 +00:00
|
|
|
props.styleOverrides || null
|
2019-09-13 10:06:02 +00:00
|
|
|
] }
|
|
|
|
touchFeedback = { false }>
|
|
|
|
|
|
|
|
<ParticipantView
|
|
|
|
avatarSize = { AVATAR_SIZE }
|
2020-06-12 10:15:16 +00:00
|
|
|
disableVideo = { isScreenShare || participant.isFakeParticipant }
|
2019-09-13 10:06:02 +00:00
|
|
|
participantId = { participantId }
|
|
|
|
style = { _styles.participantViewStyle }
|
|
|
|
tintEnabled = { participantInLargeVideo && !disableTint }
|
|
|
|
tintStyle = { _styles.activeThumbnailTint }
|
|
|
|
zOrder = { 1 } />
|
|
|
|
|
|
|
|
{ renderDisplayName && <DisplayNameLabel participantId = { participantId } /> }
|
|
|
|
|
|
|
|
{ renderModeratorIndicator
|
|
|
|
&& <View style = { styles.moderatorIndicatorContainer }>
|
|
|
|
<ModeratorIndicator />
|
2020-06-12 10:15:16 +00:00
|
|
|
</View>}
|
2019-09-13 10:06:02 +00:00
|
|
|
|
2020-06-12 10:15:16 +00:00
|
|
|
{ !participant.isFakeParticipant && <View
|
2018-09-13 15:20:22 +00:00
|
|
|
style = { [
|
2019-09-13 10:06:02 +00:00
|
|
|
styles.thumbnailTopIndicatorContainer,
|
|
|
|
styles.thumbnailTopLeftIndicatorContainer
|
|
|
|
] }>
|
|
|
|
<RaisedHandIndicator participantId = { participant.id } />
|
|
|
|
{ renderDominantSpeakerIndicator && <DominantSpeakerIndicator /> }
|
2020-06-12 10:15:16 +00:00
|
|
|
</View> }
|
2019-09-13 10:06:02 +00:00
|
|
|
|
2020-06-12 10:15:16 +00:00
|
|
|
{ !participant.isFakeParticipant && <View
|
2019-09-13 10:06:02 +00:00
|
|
|
style = { [
|
|
|
|
styles.thumbnailTopIndicatorContainer,
|
|
|
|
styles.thumbnailTopRightIndicatorContainer
|
|
|
|
] }>
|
|
|
|
<ConnectionIndicator participantId = { participant.id } />
|
2020-06-12 10:15:16 +00:00
|
|
|
</View> }
|
2019-09-13 10:06:02 +00:00
|
|
|
|
2020-06-12 10:15:16 +00:00
|
|
|
{ !participant.isFakeParticipant && <Container style = { styles.thumbnailIndicatorContainer }>
|
2019-09-13 10:06:02 +00:00
|
|
|
{ audioMuted
|
|
|
|
&& <AudioMutedIndicator /> }
|
|
|
|
|
|
|
|
{ videoMuted
|
|
|
|
&& <VideoMutedIndicator /> }
|
2020-06-12 10:15:16 +00:00
|
|
|
</Container> }
|
2019-09-13 10:06:02 +00:00
|
|
|
|
|
|
|
</Container>
|
|
|
|
);
|
2019-01-05 16:49:21 +00:00
|
|
|
}
|
2017-01-28 23:28:13 +00:00
|
|
|
|
2019-01-05 16:49:21 +00:00
|
|
|
/**
|
|
|
|
* Maps part of redux actions to component's props.
|
|
|
|
*
|
|
|
|
* @param {Function} dispatch - Redux's {@code dispatch} function.
|
|
|
|
* @param {Props} ownProps - The own props of the component.
|
|
|
|
* @returns {{
|
|
|
|
* _onClick: Function,
|
|
|
|
* _onShowRemoteVideoMenu: Function
|
|
|
|
* }}
|
|
|
|
*/
|
|
|
|
function _mapDispatchToProps(dispatch: Function, ownProps): Object {
|
|
|
|
return {
|
|
|
|
/**
|
|
|
|
* Handles click/tap event on the thumbnail.
|
|
|
|
*
|
|
|
|
* @protected
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onClick() {
|
2019-06-19 12:44:39 +00:00
|
|
|
const { participant, tileView } = ownProps;
|
2019-01-05 16:49:21 +00:00
|
|
|
|
2019-06-19 12:44:39 +00:00
|
|
|
if (tileView) {
|
|
|
|
dispatch(toggleToolboxVisible());
|
|
|
|
} else {
|
|
|
|
dispatch(pinParticipant(participant.pinned ? null : participant.id));
|
|
|
|
}
|
2019-01-05 16:49:21 +00:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Handles long press on the thumbnail.
|
|
|
|
*
|
|
|
|
* @returns {void}
|
|
|
|
*/
|
|
|
|
_onShowRemoteVideoMenu() {
|
|
|
|
const { participant } = ownProps;
|
|
|
|
|
|
|
|
dispatch(openDialog(RemoteVideoMenu, {
|
|
|
|
participant
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
};
|
2016-10-05 14:36:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Function that maps parts of Redux state tree into component props.
|
|
|
|
*
|
|
|
|
* @param {Object} state - Redux state.
|
2019-01-05 16:49:21 +00:00
|
|
|
* @param {Props} ownProps - Properties of component.
|
2019-09-12 13:03:20 +00:00
|
|
|
* @returns {Object}
|
2016-10-05 14:36:59 +00:00
|
|
|
*/
|
2017-01-28 23:34:57 +00:00
|
|
|
function _mapStateToProps(state, ownProps) {
|
2017-01-17 14:44:50 +00:00
|
|
|
// We need read-only access to the state of features/large-video so that the
|
2017-04-10 17:59:44 +00:00
|
|
|
// filmstrip doesn't render the video of the participant who is rendered on
|
2016-10-05 14:36:59 +00:00
|
|
|
// the stage i.e. as a large video.
|
2017-01-17 14:44:50 +00:00
|
|
|
const largeVideo = state['features/large-video'];
|
2016-10-05 14:36:59 +00:00
|
|
|
const tracks = state['features/base/tracks'];
|
2019-09-13 09:51:49 +00:00
|
|
|
const { participant } = ownProps;
|
|
|
|
const id = participant.id;
|
2016-10-05 14:36:59 +00:00
|
|
|
const audioTrack
|
|
|
|
= getTrackByMediaTypeAndParticipant(tracks, MEDIA_TYPE.AUDIO, id);
|
|
|
|
const videoTrack
|
|
|
|
= getTrackByMediaTypeAndParticipant(tracks, MEDIA_TYPE.VIDEO, id);
|
2019-09-12 13:03:20 +00:00
|
|
|
const participantCount = getParticipantCount(state);
|
2019-09-13 09:51:49 +00:00
|
|
|
const renderDominantSpeakerIndicator = participant.dominantSpeaker && participantCount > 2;
|
|
|
|
const _isEveryoneModerator = isEveryoneModerator(state);
|
|
|
|
const renderModeratorIndicator = !_isEveryoneModerator && participant.role === PARTICIPANT_ROLE.MODERATOR;
|
2016-10-05 14:36:59 +00:00
|
|
|
|
|
|
|
return {
|
2019-09-13 10:00:08 +00:00
|
|
|
_audioMuted: audioTrack?.muted ?? true,
|
2017-01-28 03:36:20 +00:00
|
|
|
_largeVideo: largeVideo,
|
2019-09-13 09:51:49 +00:00
|
|
|
_renderDominantSpeakerIndicator: renderDominantSpeakerIndicator,
|
|
|
|
_renderModeratorIndicator: renderModeratorIndicator,
|
2019-01-22 10:35:28 +00:00
|
|
|
_styles: ColorSchemeRegistry.get(state, 'Thumbnail'),
|
2017-01-28 03:36:20 +00:00
|
|
|
_videoTrack: videoTrack
|
2016-10-05 14:36:59 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2019-01-05 16:49:21 +00:00
|
|
|
export default connect(_mapStateToProps, _mapDispatchToProps)(Thumbnail);
|