1
0
mirror of https://github.com/ascribe/onion.git synced 2025-01-15 07:17:45 +01:00
onion/js/components/ascribe_media/media_player.js

176 lines
4.8 KiB
JavaScript
Raw Normal View History

2015-06-05 14:14:59 +02:00
'use strict';
2015-05-27 17:34:15 +02:00
import React from 'react';
import InjectInHeadMixin from '../../mixins/inject_in_head_mixin';
2015-06-04 17:37:46 +02:00
import Panel from 'react-bootstrap/lib/Panel';
2015-07-03 15:05:14 +02:00
import ProgressBar from 'react-bootstrap/lib/ProgressBar';
2015-06-16 14:42:36 +02:00
import AppConstants from '../../constants/application_constants.js';
2015-05-27 17:34:15 +02:00
/**
* This is the component that implements display-specific functionality.
*
* ResourceViewer handles the following mimetypes:
* - image
* - video
* - audio
* - pdf
* - other
*/
2015-06-04 17:37:46 +02:00
let Other = React.createClass({
2015-06-05 14:14:59 +02:00
propTypes: {
url: React.PropTypes.string.isRequired
},
2015-06-04 17:37:46 +02:00
render() {
let ext = this.props.url.split('.').pop();
return (
<Panel className="media-other">
<p className="text-center">
.{ext}
</p>
</Panel>
);
}
});
2015-06-03 16:53:27 +02:00
let Image = React.createClass({
2015-06-05 14:14:59 +02:00
propTypes: {
url: React.PropTypes.string.isRequired,
preview: React.PropTypes.string.isRequired
},
2015-06-05 13:15:31 +02:00
mixins: [InjectInHeadMixin],
componentDidMount() {
2015-06-17 10:16:28 +02:00
this.inject('https://code.jquery.com/jquery-2.1.4.min.js')
2015-06-05 13:15:31 +02:00
.then(() =>
Promise.all([
2015-06-16 14:42:36 +02:00
this.inject(AppConstants.baseUrl + 'static/thirdparty/shmui/shmui.css'),
this.inject(AppConstants.baseUrl + 'static/thirdparty/shmui/jquery.shmui.js')
2015-06-05 14:14:59 +02:00
]).then(() => { window.jQuery('.shmui-ascribe').shmui(); }));
2015-06-05 13:15:31 +02:00
},
2015-06-03 16:53:27 +02:00
render() {
2015-06-04 17:17:39 +02:00
return (
2015-06-05 13:15:31 +02:00
<img className="shmui-ascribe" src={this.props.preview} data-large-src={this.props.url}/>
2015-06-04 17:37:46 +02:00
);
2015-06-03 16:53:27 +02:00
}
});
2015-07-02 18:27:09 +02:00
let Audio = React.createClass({
propTypes: {
url: React.PropTypes.string.isRequired
},
mixins: [InjectInHeadMixin],
componentDidMount() {
this.inject(AppConstants.baseUrl + 'static/thirdparty/audiojs/audiojs/audio.min.js').then(this.ready);
},
ready() {
window.audiojs.events.ready(function() {
2015-07-03 15:05:14 +02:00
window.audiojs.createAll();
2015-07-02 18:27:09 +02:00
});
},
render() {
return (
<audio className="ascribe-audio" src={this.props.url} preload="auto" />
);
}
});
2015-06-03 16:53:27 +02:00
let Video = React.createClass({
2015-05-27 17:34:15 +02:00
propTypes: {
2015-06-03 16:53:27 +02:00
preview: React.PropTypes.string.isRequired,
url: React.PropTypes.string.isRequired,
2015-07-03 15:05:14 +02:00
extraData: React.PropTypes.array.isRequired,
encodingStatus: React.PropTypes.number
2015-05-27 17:34:15 +02:00
},
mixins: [InjectInHeadMixin],
2015-06-03 16:53:27 +02:00
getInitialState() {
return { ready: false };
},
2015-05-27 17:34:15 +02:00
componentDidMount() {
2015-07-02 17:10:32 +02:00
Promise.all([
this.inject('//vjs.zencdn.net/4.12/video-js.css'),
this.inject('//vjs.zencdn.net/4.12/video.js')
]).then(this.ready);
},
componentDidUpdate() {
if (this.state.ready && !this.state.videojs) {
window.videojs(React.findDOMNode(this.refs.video));
}
2015-06-05 14:14:59 +02:00
},
ready() {
2015-07-02 17:10:32 +02:00
this.setState({ready: true, videojs: false});
2015-06-03 16:53:27 +02:00
},
render() {
if (this.state.ready) {
return (
2015-07-02 17:10:32 +02:00
<video ref="video" className="video-js vjs-default-skin" poster={this.props.preview}
controls preload="none" width="auto" height="auto">
2015-06-03 16:53:27 +02:00
{this.props.extraData.map((data, i) =>
<source key={i} type={'video/' + data.type} src={data.url} />
)}
</video>
);
} else {
2015-06-05 14:14:59 +02:00
return (
2015-06-03 16:53:27 +02:00
<Image src={this.props.preview} />
);
}
}
});
let resourceMap = {
'image': Image,
2015-06-04 17:37:46 +02:00
'video': Video,
2015-07-02 18:27:09 +02:00
'audio': Audio,
2015-06-04 17:37:46 +02:00
'other': Other
2015-06-05 14:14:59 +02:00
};
2015-06-03 16:53:27 +02:00
2015-06-04 16:15:59 +02:00
let MediaPlayer = React.createClass({
2015-06-03 16:53:27 +02:00
propTypes: {
mimetype: React.PropTypes.oneOf(['image', 'video', 'audio', 'pdf', 'other']).isRequired,
preview: React.PropTypes.string.isRequired,
2015-06-05 14:14:59 +02:00
url: React.PropTypes.string.isRequired,
2015-07-03 15:05:14 +02:00
extraData: React.PropTypes.array,
encodingStatus: React.PropTypes.number
2015-05-27 17:34:15 +02:00
},
render() {
2015-07-08 14:37:20 +02:00
if (this.props.mimetype === 'video' && this.props.encodingStatus !== undefined && this.props.encodingStatus !== 100) {
2015-07-03 15:05:14 +02:00
return (
<div className="ascribe-detail-header ascribe-media-player">
<p><em>Please be patient, the video is been encoded</em></p>
<ProgressBar now={this.props.encodingStatus}
label='%(percent)s%' />
</div>
);
} else {
let Component = resourceMap[this.props.mimetype] || Other;
return (
<div className="ascribe-media-player">
<Component preview={this.props.preview}
url={this.props.url}
extraData={this.props.extraData}
encodingStatus={this.props.encodingStatus} />
</div>
);
}
2015-05-27 17:34:15 +02:00
}
});
2015-06-04 16:15:59 +02:00
export default MediaPlayer;