// @flow import React, { Component } from 'react'; import { View } from 'react-native'; import type { Dispatch } from 'redux'; import { ColorSchemeRegistry } from '../../../base/color-scheme'; import { openDialog } from '../../../base/dialog'; import { MEDIA_TYPE, VIDEO_TYPE, Audio } from '../../../base/media'; import { PARTICIPANT_ROLE, ParticipantView, isEveryoneModerator, pinParticipant } from '../../../base/participants'; import { Container } from '../../../base/react'; import { connect } from '../../../base/redux'; import { StyleType } from '../../../base/styles'; import { getTrackByMediaTypeAndParticipant } from '../../../base/tracks'; import { ConnectionIndicator } from '../../../connection-indicator'; import { DisplayNameLabel } from '../../../display-name'; import { RemoteVideoMenu } from '../../../remote-video-menu'; import { toggleToolboxVisible } from '../../../toolbox'; import AudioMutedIndicator from './AudioMutedIndicator'; import DominantSpeakerIndicator from './DominantSpeakerIndicator'; import ModeratorIndicator from './ModeratorIndicator'; import RaisedHandIndicator from './RaisedHandIndicator'; import styles, { AVATAR_SIZE } from './styles'; import VideoMutedIndicator from './VideoMutedIndicator'; /** * Thumbnail component's property types. */ type Props = { /** * The Redux representation of the participant's audio track. */ _audioTrack: Object, /** * True if everone in the meeting is moderator. */ _isEveryoneModerator: boolean, /** * The Redux representation of the state "features/large-video". */ _largeVideo: Object, /** * Handles click/tap event on the thumbnail. */ _onClick: ?Function, /** * Handles long press on the thumbnail. */ _onShowRemoteVideoMenu: ?Function, /** * The color-schemed stylesheet of the feature. */ _styles: StyleType, /** * 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. */ disableTint?: boolean, /** * Invoked to trigger state changes in Redux. */ dispatch: Dispatch, /** * The Redux representation of the participant to display. */ participant: Object, /** * Whether to display or hide the display name of the participant in the thumbnail. */ renderDisplayName: ?boolean, /** * Optional styling to add or override on the Thumbnail component root. */ styleOverrides?: Object, /** * If true, it tells the thumbnail that it needs to behave differently. E.g. react differently to a single tap. */ tileView?: boolean }; /** * React component for video thumbnail. * * @extends Component */ class Thumbnail extends Component { /** * Implements React's {@link Component#render()}. * * @inheritdoc * @returns {ReactElement} */ render() { const { _audioTrack: audioTrack, _isEveryoneModerator, _largeVideo: largeVideo, _onClick, _onShowRemoteVideoMenu, _styles, _videoTrack: videoTrack, disableTint, participant, renderDisplayName, tileView } = this.props; // We don't render audio in any of the following: // 1. The audio (source) is muted. There's no practical reason (that we // know of, anyway) why we'd want to render it given that it's // silence (& not even comfort noise). // 2. The audio is local. If we were to render local audio, the local // participants would be hearing themselves. const audioMuted = !audioTrack || audioTrack.muted; const renderAudio = !audioMuted && !audioTrack.local; const participantId = participant.id; const participantInLargeVideo = participantId === largeVideo.participantId; const videoMuted = !videoTrack || videoTrack.muted; const isScreenShare = videoTrack && videoTrack.videoType === VIDEO_TYPE.DESKTOP; return ( { renderAudio && ); } } /** * 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() { const { participant, tileView } = ownProps; if (tileView) { dispatch(toggleToolboxVisible()); } else { dispatch(pinParticipant(participant.pinned ? null : participant.id)); } }, /** * Handles long press on the thumbnail. * * @returns {void} */ _onShowRemoteVideoMenu() { const { participant } = ownProps; dispatch(openDialog(RemoteVideoMenu, { participant })); } }; } /** * Function that maps parts of Redux state tree into component props. * * @param {Object} state - Redux state. * @param {Props} ownProps - Properties of component. * @returns {{ * _audioTrack: Track, * _largeVideo: Object, * _styles: StyleType, * _videoTrack: Track * }} */ function _mapStateToProps(state, ownProps) { // We need read-only access to the state of features/large-video so that the // filmstrip doesn't render the video of the participant who is rendered on // the stage i.e. as a large video. const largeVideo = state['features/large-video']; const tracks = state['features/base/tracks']; const id = ownProps.participant.id; const audioTrack = getTrackByMediaTypeAndParticipant(tracks, MEDIA_TYPE.AUDIO, id); const videoTrack = getTrackByMediaTypeAndParticipant(tracks, MEDIA_TYPE.VIDEO, id); return { _audioTrack: audioTrack, _isEveryoneModerator: isEveryoneModerator(state), _largeVideo: largeVideo, _styles: ColorSchemeRegistry.get(state, 'Thumbnail'), _videoTrack: videoTrack }; } export default connect(_mapStateToProps, _mapDispatchToProps)(Thumbnail);