2017-05-02 20:04:16 -04:00
|
|
|
import React from 'react';
|
2016-09-17 12:05:02 -04:00
|
|
|
import ImmutablePropTypes from 'react-immutable-proptypes';
|
2017-04-21 14:05:35 -04:00
|
|
|
import PropTypes from 'prop-types';
|
2016-11-16 11:20:52 -05:00
|
|
|
import IconButton from './icon_button';
|
2016-11-23 05:23:32 -05:00
|
|
|
import { defineMessages, injectIntl, FormattedMessage } from 'react-intl';
|
2017-03-07 03:54:57 -05:00
|
|
|
import { isIOS } from '../is_mobile';
|
2016-11-18 09:36:16 -05:00
|
|
|
|
|
|
|
const messages = defineMessages({
|
2017-01-19 21:24:30 -05:00
|
|
|
toggle_sound: { id: 'video_player.toggle_sound', defaultMessage: 'Toggle sound' },
|
2017-04-13 09:04:18 -04:00
|
|
|
toggle_visible: { id: 'video_player.toggle_visible', defaultMessage: 'Toggle visibility' },
|
2017-04-15 19:12:47 -04:00
|
|
|
expand_video: { id: 'video_player.expand', defaultMessage: 'Expand video' },
|
|
|
|
expand_video: { id: 'video_player.video_error', defaultMessage: 'Video could not be played' }
|
2016-11-18 09:36:16 -05:00
|
|
|
});
|
2016-09-17 12:05:02 -04:00
|
|
|
|
2017-04-21 14:05:35 -04:00
|
|
|
class VideoPlayer extends React.PureComponent {
|
2016-09-17 12:05:02 -04:00
|
|
|
|
2017-04-21 14:05:35 -04:00
|
|
|
constructor (props, context) {
|
|
|
|
super(props, context);
|
|
|
|
this.state = {
|
2017-01-19 21:24:30 -05:00
|
|
|
visible: !this.props.sensitive,
|
|
|
|
preview: true,
|
2017-02-28 17:48:41 -05:00
|
|
|
muted: true,
|
2017-04-15 19:12:47 -04:00
|
|
|
hasAudio: true,
|
|
|
|
videoError: false
|
2016-09-18 06:39:00 -04:00
|
|
|
};
|
2017-04-22 22:26:55 -04:00
|
|
|
|
2017-04-21 14:05:35 -04:00
|
|
|
this.handleClick = this.handleClick.bind(this);
|
|
|
|
this.handleVideoClick = this.handleVideoClick.bind(this);
|
|
|
|
this.handleOpen = this.handleOpen.bind(this);
|
|
|
|
this.handleVisibility = this.handleVisibility.bind(this);
|
|
|
|
this.handleExpand = this.handleExpand.bind(this);
|
|
|
|
this.setRef = this.setRef.bind(this);
|
|
|
|
this.handleLoadedData = this.handleLoadedData.bind(this);
|
|
|
|
this.handleVideoError = this.handleVideoError.bind(this);
|
|
|
|
}
|
2016-09-17 12:05:02 -04:00
|
|
|
|
2016-09-18 06:39:00 -04:00
|
|
|
handleClick () {
|
|
|
|
this.setState({ muted: !this.state.muted });
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2016-09-18 06:39:00 -04:00
|
|
|
|
2016-11-07 13:05:32 -05:00
|
|
|
handleVideoClick (e) {
|
|
|
|
e.stopPropagation();
|
|
|
|
|
2017-05-02 20:04:16 -04:00
|
|
|
const node = this.video;
|
2016-11-07 13:05:32 -05:00
|
|
|
|
|
|
|
if (node.paused) {
|
|
|
|
node.play();
|
|
|
|
} else {
|
|
|
|
node.pause();
|
|
|
|
}
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2016-11-07 13:05:32 -05:00
|
|
|
|
2016-11-23 05:23:32 -05:00
|
|
|
handleOpen () {
|
2017-01-19 21:24:30 -05:00
|
|
|
this.setState({ preview: !this.state.preview });
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2017-01-19 21:24:30 -05:00
|
|
|
|
|
|
|
handleVisibility () {
|
|
|
|
this.setState({
|
|
|
|
visible: !this.state.visible,
|
|
|
|
preview: true
|
|
|
|
});
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2016-11-23 05:23:32 -05:00
|
|
|
|
2017-04-13 09:04:18 -04:00
|
|
|
handleExpand () {
|
2017-04-13 11:01:09 -04:00
|
|
|
this.video.pause();
|
|
|
|
this.props.onOpenVideo(this.props.media, this.video.currentTime);
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2017-04-13 09:04:18 -04:00
|
|
|
|
2017-02-28 17:48:41 -05:00
|
|
|
setRef (c) {
|
|
|
|
this.video = c;
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2017-02-28 17:48:41 -05:00
|
|
|
|
|
|
|
handleLoadedData () {
|
|
|
|
if (('WebkitAppearance' in document.documentElement.style && this.video.audioTracks.length === 0) || this.video.mozHasAudio === false) {
|
|
|
|
this.setState({ hasAudio: false });
|
|
|
|
}
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2017-02-28 17:48:41 -05:00
|
|
|
|
2017-04-15 19:12:47 -04:00
|
|
|
handleVideoError () {
|
|
|
|
this.setState({ videoError: true });
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2017-04-15 19:12:47 -04:00
|
|
|
|
2017-02-28 17:48:41 -05:00
|
|
|
componentDidMount () {
|
|
|
|
if (!this.video) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.video.addEventListener('loadeddata', this.handleLoadedData);
|
2017-04-15 19:12:47 -04:00
|
|
|
this.video.addEventListener('error', this.handleVideoError);
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2017-02-28 17:48:41 -05:00
|
|
|
|
|
|
|
componentDidUpdate () {
|
|
|
|
if (!this.video) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.video.addEventListener('loadeddata', this.handleLoadedData);
|
2017-04-15 19:12:47 -04:00
|
|
|
this.video.addEventListener('error', this.handleVideoError);
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2017-02-28 17:48:41 -05:00
|
|
|
|
|
|
|
componentWillUnmount () {
|
|
|
|
if (!this.video) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
this.video.removeEventListener('loadeddata', this.handleLoadedData);
|
2017-04-15 19:12:47 -04:00
|
|
|
this.video.removeEventListener('error', this.handleVideoError);
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
2017-02-28 17:48:41 -05:00
|
|
|
|
2016-09-17 12:05:02 -04:00
|
|
|
render () {
|
2017-02-28 17:48:41 -05:00
|
|
|
const { media, intl, width, height, sensitive, autoplay } = this.props;
|
2016-11-23 05:23:32 -05:00
|
|
|
|
2017-01-19 21:24:30 -05:00
|
|
|
let spoilerButton = (
|
2017-04-22 22:26:55 -04:00
|
|
|
<div className='status__video-player-spoiler' style={{ display: !this.state.visible ? 'none' : 'block' }} >
|
2017-04-13 11:01:09 -04:00
|
|
|
<IconButton overlay title={intl.formatMessage(messages.toggle_visible)} icon={this.state.visible ? 'eye' : 'eye-slash'} onClick={this.handleVisibility} />
|
2017-01-19 21:24:30 -05:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
2017-04-13 09:04:18 -04:00
|
|
|
let expandButton = (
|
2017-04-22 22:26:55 -04:00
|
|
|
<div className='status__video-player-expand'>
|
2017-04-13 11:01:09 -04:00
|
|
|
<IconButton overlay title={intl.formatMessage(messages.expand_video)} icon='expand' onClick={this.handleExpand} />
|
2017-04-13 09:04:18 -04:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
|
2017-02-28 17:48:41 -05:00
|
|
|
let muteButton = '';
|
|
|
|
|
|
|
|
if (this.state.hasAudio) {
|
|
|
|
muteButton = (
|
2017-04-22 22:26:55 -04:00
|
|
|
<div className='status__video-player-mute'>
|
2017-04-13 11:01:09 -04:00
|
|
|
<IconButton overlay title={intl.formatMessage(messages.toggle_sound)} icon={this.state.muted ? 'volume-off' : 'volume-up'} onClick={this.handleClick} />
|
2017-02-28 17:48:41 -05:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-01-19 21:24:30 -05:00
|
|
|
if (!this.state.visible) {
|
|
|
|
if (sensitive) {
|
|
|
|
return (
|
2017-04-22 22:26:55 -04:00
|
|
|
<div role='button' tabIndex='0' style={{ width: `${width}px`, height: `${height}px` }} className='media-spoiler' onClick={this.handleVisibility}>
|
2017-01-19 21:24:30 -05:00
|
|
|
{spoilerButton}
|
2017-04-22 22:26:55 -04:00
|
|
|
<span className='media-spoiler__warning'><FormattedMessage id='status.sensitive_warning' defaultMessage='Sensitive content' /></span>
|
|
|
|
<span className='media-spoiler__trigger'><FormattedMessage id='status.sensitive_toggle' defaultMessage='Click to view' /></span>
|
2017-01-19 21:24:30 -05:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return (
|
2017-04-22 22:26:55 -04:00
|
|
|
<div role='button' tabIndex='0' style={{ width: `${width}px`, height: `${height}px` }} className='media-spoiler' onClick={this.handleVisibility}>
|
2017-01-19 21:24:30 -05:00
|
|
|
{spoilerButton}
|
2017-04-22 22:26:55 -04:00
|
|
|
<span className='media-spoiler__warning'><FormattedMessage id='status.media_hidden' defaultMessage='Media hidden' /></span>
|
|
|
|
<span className='media-spoiler__trigger'><FormattedMessage id='status.sensitive_toggle' defaultMessage='Click to view' /></span>
|
2017-01-19 21:24:30 -05:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-28 17:48:41 -05:00
|
|
|
if (this.state.preview && !autoplay) {
|
2016-12-04 06:26:12 -05:00
|
|
|
return (
|
2017-04-22 22:26:55 -04:00
|
|
|
<div role='button' tabIndex='0' className='media-spoiler-video' style={{ width: `${width}px`, height: `${height}px`, background: `url(${media.get('preview_url')}) no-repeat center` }} onClick={this.handleOpen}>
|
2017-01-19 21:24:30 -05:00
|
|
|
{spoilerButton}
|
2017-04-22 22:26:55 -04:00
|
|
|
<div className='media-spoiler-video-play-icon'><i className='fa fa-play' /></div>
|
2016-12-04 06:26:12 -05:00
|
|
|
</div>
|
|
|
|
);
|
2016-11-23 05:23:32 -05:00
|
|
|
}
|
2016-11-16 11:20:52 -05:00
|
|
|
|
2017-04-15 19:12:47 -04:00
|
|
|
if (this.state.videoError) {
|
|
|
|
return (
|
2017-04-22 22:26:55 -04:00
|
|
|
<div style={{ width: `${width}px`, height: `${height}px` }} className='video-error-cover' >
|
|
|
|
<span className='media-spoiler__warning'><FormattedMessage id='video_player.video_error' defaultMessage='Video could not be played' /></span>
|
2017-04-15 19:12:47 -04:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2016-09-17 12:05:02 -04:00
|
|
|
return (
|
2017-04-22 22:26:55 -04:00
|
|
|
<div className='status__video-player' style={{ width: `${width}px`, height: `${height}px` }}>
|
2017-01-19 21:24:30 -05:00
|
|
|
{spoilerButton}
|
2017-02-28 17:48:41 -05:00
|
|
|
{muteButton}
|
2017-04-13 09:04:18 -04:00
|
|
|
{expandButton}
|
2017-05-02 20:04:16 -04:00
|
|
|
|
|
|
|
<video
|
|
|
|
className='status__video-player-video'
|
|
|
|
role='button'
|
|
|
|
tabIndex='0'
|
|
|
|
ref={this.setRef}
|
|
|
|
src={media.get('url')}
|
|
|
|
autoPlay={!isIOS()}
|
|
|
|
loop={true}
|
|
|
|
muted={this.state.muted}
|
|
|
|
onClick={this.handleVideoClick}
|
|
|
|
/>
|
2016-09-17 12:05:02 -04:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2017-04-21 14:05:35 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
VideoPlayer.propTypes = {
|
|
|
|
media: ImmutablePropTypes.map.isRequired,
|
|
|
|
width: PropTypes.number,
|
|
|
|
height: PropTypes.number,
|
|
|
|
sensitive: PropTypes.bool,
|
|
|
|
intl: PropTypes.object.isRequired,
|
|
|
|
autoplay: PropTypes.bool,
|
|
|
|
onOpenVideo: PropTypes.func.isRequired
|
|
|
|
};
|
|
|
|
|
|
|
|
VideoPlayer.defaultProps = {
|
|
|
|
width: 239,
|
|
|
|
height: 110
|
|
|
|
};
|
2016-09-17 12:05:02 -04:00
|
|
|
|
2016-11-16 11:20:52 -05:00
|
|
|
export default injectIntl(VideoPlayer);
|