2015-06-05 14:14:59 +02:00
|
|
|
'use strict';
|
|
|
|
|
2015-05-27 17:34:15 +02:00
|
|
|
import React from 'react';
|
2015-07-24 13:44:28 +02:00
|
|
|
|
2015-06-04 17:37:46 +02:00
|
|
|
import Panel from 'react-bootstrap/lib/Panel';
|
2016-03-09 16:54:11 +01:00
|
|
|
|
2016-06-02 15:35:31 +02:00
|
|
|
import audiojs from '../../third_party/imports/audiojs';
|
|
|
|
import shmui from '../../third_party/imports/shmui';
|
|
|
|
import videojs from '../../third_party/imports/videojs';
|
|
|
|
|
2016-06-13 17:33:59 +02:00
|
|
|
import { escapeHTML } from '../../utils/dom';
|
2016-06-13 14:35:02 +02:00
|
|
|
import { extractFileExtensionFromUrl } from '../../utils/file';
|
2015-05-27 17:34:15 +02:00
|
|
|
|
2016-06-02 15:35:31 +02:00
|
|
|
|
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() {
|
2015-09-03 16:11:41 +02:00
|
|
|
let filename = this.props.url.split('/').pop();
|
|
|
|
let tokens = filename.split('.');
|
2016-03-09 16:54:11 +01:00
|
|
|
let thumbnail;
|
2015-09-03 16:11:41 +02:00
|
|
|
|
|
|
|
if (tokens.length > 1) {
|
2016-03-09 16:54:11 +01:00
|
|
|
thumbnail = '.' + tokens.pop();
|
2015-09-03 16:11:41 +02:00
|
|
|
} else {
|
2016-03-09 16:54:11 +01:00
|
|
|
thumbnail = 'file';
|
2015-09-03 16:11:41 +02:00
|
|
|
}
|
2015-06-04 17:37:46 +02:00
|
|
|
|
|
|
|
return (
|
|
|
|
<Panel className="media-other">
|
|
|
|
<p className="text-center">
|
2016-03-09 16:54:11 +01:00
|
|
|
{thumbnail}
|
2015-06-04 17:37:46 +02:00
|
|
|
</p>
|
|
|
|
</Panel>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2015-06-03 16:53:27 +02:00
|
|
|
let Image = React.createClass({
|
2015-06-05 14:14:59 +02:00
|
|
|
propTypes: {
|
2015-11-11 10:51:45 +01:00
|
|
|
url: React.PropTypes.string,
|
2016-03-09 16:54:11 +01:00
|
|
|
thumbnail: React.PropTypes.string.isRequired
|
2015-06-05 14:14:59 +02:00
|
|
|
},
|
|
|
|
|
2015-06-05 13:15:31 +02:00
|
|
|
componentDidMount() {
|
2016-03-09 16:54:11 +01:00
|
|
|
if (this.props.url) {
|
2016-06-02 15:35:31 +02:00
|
|
|
shmui
|
|
|
|
.importLib()
|
|
|
|
.then(() => window.jQuery('.shmui-ascribe').shmui());
|
2015-11-11 10:51:45 +01:00
|
|
|
}
|
2015-06-05 13:15:31 +02:00
|
|
|
},
|
|
|
|
|
2015-06-03 16:53:27 +02:00
|
|
|
render() {
|
2016-03-09 16:54:11 +01:00
|
|
|
const { url, thumbnail } = this.props;
|
2015-11-11 10:51:45 +01:00
|
|
|
|
2016-03-10 15:09:22 +01:00
|
|
|
// TIFFs can not be displayed by the browser, so we just display their thumbnail
|
2016-03-10 17:07:11 +01:00
|
|
|
// url is not necessarily defined, which would cause this function to fail
|
|
|
|
if (url && extractFileExtensionFromUrl(url) !== 'tif' && extractFileExtensionFromUrl(url) !== 'tiff') {
|
2015-11-11 10:51:45 +01:00
|
|
|
return (
|
2016-03-09 16:54:11 +01:00
|
|
|
<img className="shmui-ascribe" src={thumbnail} data-large-src={url} />
|
2015-11-11 10:51:45 +01:00
|
|
|
);
|
|
|
|
} else {
|
|
|
|
return (
|
2016-03-09 16:54:11 +01:00
|
|
|
<img src={thumbnail} />
|
2015-11-11 10:51:45 +01: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
|
|
|
|
},
|
|
|
|
|
|
|
|
componentDidMount() {
|
2016-06-02 15:35:31 +02:00
|
|
|
audiojs
|
|
|
|
.importLib()
|
|
|
|
.then(() => window.audiojs.events.ready(() => window.audiojs.createAll()));
|
2015-07-02 18:27:09 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
render() {
|
|
|
|
return (
|
|
|
|
<audio className="ascribe-audio" src={this.props.url} preload="auto" />
|
|
|
|
);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2015-07-29 18:19:04 +02:00
|
|
|
|
2015-06-03 16:53:27 +02:00
|
|
|
let Video = React.createClass({
|
2015-07-29 18:19:04 +02:00
|
|
|
/**
|
|
|
|
* The solution here is a bit convoluted.
|
|
|
|
* ReactJS is responsible for DOM manipulation but VideoJS updates the DOM
|
2015-07-29 18:24:54 +02:00
|
|
|
* to install itself to display the video, therefore ReactJS complains that we are
|
|
|
|
* changing the DOM under its feet.
|
2015-10-20 14:16:49 +02:00
|
|
|
* The component supports a fall-back to HTML5 video tag.
|
2015-07-29 18:19:04 +02:00
|
|
|
*
|
|
|
|
* What we do is the following:
|
2015-10-20 14:16:49 +02:00
|
|
|
* 1) set `state.libraryLoaded = null` (state.libraryLoaded can be in three states: `null`
|
|
|
|
* if we don't know anything about it, `true` if the external library has been loaded,
|
|
|
|
* `false` if we failed to load the external library)
|
|
|
|
* 2) render the cover using the `<Image />` component (because libraryLoaded is null)
|
2015-07-29 18:19:04 +02:00
|
|
|
* 3) on `componentDidMount`, we load the external `css` and `js` resources using
|
2015-11-09 14:32:14 +01:00
|
|
|
* the `InjectInHeadUtils`, attaching a function to `Promise.then` to change
|
2015-10-20 14:16:49 +02:00
|
|
|
* `state.libraryLoaded` to true
|
|
|
|
* 4) when the promise is succesfully resolved, we change `state.libraryLoaded` triggering
|
2015-07-29 18:19:04 +02:00
|
|
|
* a re-render
|
|
|
|
* 5) the new render calls `prepareVideoHTML` to get the raw HTML of the video tag
|
|
|
|
* (that will be later processed and expanded by VideoJS)
|
|
|
|
* 6) `componentDidUpdate` is called after `render`, setting `state.videoMounted` to true,
|
|
|
|
* to avoid re-installing the VideoJS library
|
|
|
|
* 7) to close the lifecycle, `componentWillUnmount` is called removing VideoJS from the DOM.
|
|
|
|
*/
|
|
|
|
|
2015-05-27 17:34:15 +02:00
|
|
|
propTypes: {
|
2016-03-09 16:54:11 +01:00
|
|
|
thumbnail: React.PropTypes.string.isRequired,
|
2015-06-03 16:53:27 +02:00
|
|
|
url: React.PropTypes.string.isRequired,
|
2016-03-10 14:33:54 +01:00
|
|
|
extraData: React.PropTypes.array.isRequired
|
2015-05-27 17:34:15 +02:00
|
|
|
},
|
|
|
|
|
2015-06-03 16:53:27 +02:00
|
|
|
getInitialState() {
|
2015-10-20 14:16:49 +02:00
|
|
|
return { libraryLoaded: null, videoMounted: false };
|
2015-06-03 16:53:27 +02:00
|
|
|
},
|
|
|
|
|
2015-05-27 17:34:15 +02:00
|
|
|
componentDidMount() {
|
2016-06-02 15:35:31 +02:00
|
|
|
videojs
|
|
|
|
.importLib()
|
|
|
|
.then(() => this.setState({ libraryLoaded: true }))
|
|
|
|
.catch(() => this.setState({ libraryLoaded: false }));
|
2015-07-02 17:10:32 +02:00
|
|
|
},
|
|
|
|
|
2015-11-16 15:54:49 +01:00
|
|
|
shouldComponentUpdate(nextProps, nextState) {
|
|
|
|
return nextState.videoMounted === false;
|
|
|
|
},
|
|
|
|
|
2015-07-02 17:10:32 +02:00
|
|
|
componentDidUpdate() {
|
2015-10-20 14:16:49 +02:00
|
|
|
if (this.state.libraryLoaded && !this.state.videoMounted) {
|
2015-07-29 18:00:49 +02:00
|
|
|
window.videojs('#mainvideo');
|
2015-07-29 18:19:04 +02:00
|
|
|
/* eslint-disable */
|
2015-07-29 18:00:49 +02:00
|
|
|
this.setState({videoMounted: true});
|
2015-07-29 18:19:04 +02:00
|
|
|
/* eslint-enable*/
|
2015-07-02 17:10:32 +02:00
|
|
|
}
|
2015-06-05 14:14:59 +02:00
|
|
|
},
|
|
|
|
|
2015-07-29 18:00:49 +02:00
|
|
|
componentWillUnmount() {
|
2015-10-20 14:16:49 +02:00
|
|
|
if (this.state.videoMounted) {
|
|
|
|
window.videojs('#mainvideo').dispose();
|
|
|
|
}
|
2015-07-29 18:00:49 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
prepareVideoHTML() {
|
|
|
|
let sources = this.props.extraData.map((data) => '<source type="video/' + data.type + '" src="' + escapeHTML(data.url) + '" />');
|
|
|
|
let html = [
|
2016-03-09 16:54:11 +01:00
|
|
|
'<video id="mainvideo" class="video-js vjs-default-skin" poster="' + escapeHTML(this.props.thumbnail) + '"',
|
2015-07-29 18:00:49 +02:00
|
|
|
'controls preload="none" width="auto" height="auto">',
|
|
|
|
sources.join('\n'),
|
|
|
|
'</video>'];
|
|
|
|
return html.join('\n');
|
|
|
|
},
|
|
|
|
|
2015-06-03 16:53:27 +02:00
|
|
|
render() {
|
2015-10-20 14:16:49 +02:00
|
|
|
if (this.state.libraryLoaded !== null) {
|
2015-06-03 16:53:27 +02:00
|
|
|
return (
|
2015-07-29 18:00:49 +02:00
|
|
|
<div dangerouslySetInnerHTML={{__html: this.prepareVideoHTML() }}/>
|
2015-06-03 16:53:27 +02:00
|
|
|
);
|
|
|
|
} else {
|
2015-06-05 14:14:59 +02:00
|
|
|
return (
|
2016-03-09 16:54:11 +01:00
|
|
|
<Image thumbnail={this.props.thumbnail} />
|
2015-06-03 16:53:27 +02:00
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
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,
|
2016-03-09 16:54:11 +01:00
|
|
|
thumbnail: 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,
|
2016-03-10 14:33:54 +01:00
|
|
|
encodingMessage: React.PropTypes.node
|
2015-05-27 17:34:15 +02:00
|
|
|
},
|
|
|
|
|
|
|
|
render() {
|
2016-03-10 14:33:54 +01:00
|
|
|
const {
|
|
|
|
mimetype,
|
|
|
|
thumbnail,
|
|
|
|
url,
|
|
|
|
extraData,
|
|
|
|
encodingMessage
|
|
|
|
} = this.props;
|
|
|
|
|
|
|
|
if (encodingMessage) {
|
|
|
|
return encodingMessage;
|
2015-07-03 15:05:14 +02:00
|
|
|
} else {
|
2015-11-11 10:51:45 +01:00
|
|
|
let Component = resourceMap[mimetype] || Other;
|
|
|
|
let componentProps = {
|
2016-03-09 16:54:11 +01:00
|
|
|
thumbnail,
|
2015-11-11 10:51:45 +01:00
|
|
|
url,
|
|
|
|
extraData,
|
|
|
|
};
|
|
|
|
|
|
|
|
// Since the launch of the portfolio whitelabel submission,
|
|
|
|
// we allow the user to specify a thumbnail upon piece-registration.
|
|
|
|
// As the `Component` is chosen according to its filetype but could potentially
|
|
|
|
// have a manually submitted thumbnail, we match if the to `Mediaplayer` submitted thumbnail
|
|
|
|
// is not the generally used fallback `url` (ascribe_spiral.png).
|
|
|
|
//
|
|
|
|
// If this is the case, we disable shmui by deleting the original `url` prop and replace
|
|
|
|
// the assigned component to `Image`.
|
2016-03-09 16:54:11 +01:00
|
|
|
if (!decodeURIComponent(thumbnail).match(/https:\/\/.*\/media\/thumbnails\/ascribe_spiral.png/) &&
|
2015-11-11 10:51:45 +01:00
|
|
|
Component === Other) {
|
|
|
|
Component = resourceMap.image;
|
|
|
|
delete componentProps.url;
|
|
|
|
}
|
|
|
|
|
2015-07-03 15:05:14 +02:00
|
|
|
return (
|
|
|
|
<div className="ascribe-media-player">
|
2015-11-11 10:51:45 +01:00
|
|
|
<Component {...componentProps}/>
|
2015-07-03 15:05:14 +02:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
2015-05-27 17:34:15 +02:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2015-06-04 16:15:59 +02:00
|
|
|
export default MediaPlayer;
|