2018-01-04 14:14:03 -08:00
|
|
|
import React from 'react';
|
|
|
|
|
2018-03-01 20:09:22 -08:00
|
|
|
function dataURItoBlob(dataURI) {
|
|
|
|
// convert base64/URLEncoded data component to raw binary data held in a string
|
|
|
|
let byteString = atob(dataURI.split(',')[1]);
|
|
|
|
// separate out the mime component
|
|
|
|
let mimeString = dataURI.split(',')[0].split(':')[1].split(';')[0];
|
|
|
|
// write the bytes of the string to a typed array
|
|
|
|
let ia = new Uint8Array(byteString.length);
|
2018-03-02 14:57:25 -08:00
|
|
|
for (let i = 0; i < byteString.length; i++) {
|
2018-03-01 20:09:22 -08:00
|
|
|
ia[i] = byteString.charCodeAt(i);
|
|
|
|
}
|
|
|
|
return new Blob([ia], {type: mimeString});
|
|
|
|
}
|
|
|
|
|
2018-01-17 10:49:57 -08:00
|
|
|
class PublishThumbnailInput extends React.Component {
|
2018-01-10 17:41:17 -08:00
|
|
|
constructor (props) {
|
|
|
|
super(props);
|
|
|
|
this.state = {
|
2018-02-28 21:25:51 -08:00
|
|
|
videoSource : null,
|
|
|
|
error : null,
|
|
|
|
sliderMinRange: 1,
|
|
|
|
sliderMaxRange: null,
|
|
|
|
sliderValue : null,
|
2018-02-28 17:28:47 -08:00
|
|
|
};
|
2018-02-28 21:25:51 -08:00
|
|
|
this.handleVideoLoadedData = this.handleVideoLoadedData.bind(this);
|
2018-03-02 14:57:25 -08:00
|
|
|
this.handleSliderChange = this.handleSliderChange.bind(this);
|
|
|
|
this.createThumbnail = this.createThumbnail.bind(this);
|
2018-01-10 17:41:17 -08:00
|
|
|
}
|
2018-02-05 18:14:12 -08:00
|
|
|
componentDidMount () {
|
2018-03-02 14:57:25 -08:00
|
|
|
const { file } = this.props;
|
2018-03-01 19:38:02 -08:00
|
|
|
this.setVideoSource(file);
|
2018-02-05 18:14:12 -08:00
|
|
|
}
|
2018-03-01 19:38:02 -08:00
|
|
|
componentWillReceiveProps (nextProps) {
|
|
|
|
// if file changes
|
2018-03-02 14:57:25 -08:00
|
|
|
if (nextProps.file && nextProps.file !== this.props.file) {
|
2018-03-01 19:38:02 -08:00
|
|
|
const { file } = nextProps;
|
|
|
|
this.setVideoSource(file);
|
|
|
|
};
|
2018-01-17 12:54:26 -08:00
|
|
|
}
|
2018-03-01 19:38:02 -08:00
|
|
|
setVideoSource (file) {
|
2018-02-28 21:25:51 -08:00
|
|
|
const previewReader = new FileReader();
|
|
|
|
previewReader.readAsDataURL(file);
|
|
|
|
previewReader.onloadend = () => {
|
2018-03-02 15:14:02 -08:00
|
|
|
const dataUri = previewReader.result;
|
|
|
|
const blob = dataURItoBlob(dataUri);
|
|
|
|
const videoSource = URL.createObjectURL(blob);
|
|
|
|
this.setState({ videoSource });
|
2018-02-28 21:25:51 -08:00
|
|
|
};
|
2018-01-10 17:41:17 -08:00
|
|
|
}
|
2018-02-28 21:25:51 -08:00
|
|
|
handleVideoLoadedData (event) {
|
|
|
|
const duration = event.target.duration;
|
2018-03-05 22:45:39 -08:00
|
|
|
const totalMinutes = Math.floor(duration / 60);
|
|
|
|
const totalSeconds = Math.floor(duration % 60);
|
2018-02-28 21:25:51 -08:00
|
|
|
// set the slider
|
|
|
|
this.setState({
|
|
|
|
sliderMaxRange: duration * 100,
|
|
|
|
sliderValue : duration * 100 / 2,
|
2018-03-05 22:45:39 -08:00
|
|
|
totalMinutes,
|
|
|
|
totalSeconds,
|
2018-01-10 17:41:17 -08:00
|
|
|
});
|
2018-02-28 21:25:51 -08:00
|
|
|
// update the current time of the video
|
|
|
|
let video = document.getElementById('video-thumb-player');
|
|
|
|
video.currentTime = duration / 2;
|
2018-01-10 17:41:17 -08:00
|
|
|
}
|
2018-02-28 21:25:51 -08:00
|
|
|
handleSliderChange (event) {
|
|
|
|
const value = parseInt(event.target.value);
|
|
|
|
// update the slider value
|
|
|
|
this.setState({
|
|
|
|
sliderValue: value,
|
|
|
|
});
|
|
|
|
// update the current time of the video
|
|
|
|
let video = document.getElementById('video-thumb-player');
|
|
|
|
video.currentTime = value / 100;
|
2018-02-05 18:14:12 -08:00
|
|
|
}
|
2018-03-02 14:57:25 -08:00
|
|
|
createThumbnail () {
|
2018-02-28 21:25:51 -08:00
|
|
|
// take a snapshot
|
|
|
|
let video = document.getElementById('video-thumb-player');
|
|
|
|
let canvas = document.createElement('canvas');
|
|
|
|
canvas.width = video.videoWidth;
|
|
|
|
canvas.height = video.videoHeight;
|
|
|
|
canvas.getContext('2d').drawImage(video, 0, 0, canvas.width, canvas.height);
|
2018-03-02 14:57:25 -08:00
|
|
|
const dataUrl = canvas.toDataURL();
|
2018-03-02 17:02:07 -08:00
|
|
|
const blob = dataURItoBlob(dataUrl);
|
|
|
|
const snapshot = new File([blob], `thumbnail.png`, {
|
|
|
|
type: 'image/png',
|
|
|
|
});
|
2018-03-02 14:57:25 -08:00
|
|
|
// set the thumbnail in redux store
|
|
|
|
if (snapshot) {
|
|
|
|
this.props.onNewThumbnail(snapshot);
|
|
|
|
}
|
2018-01-10 17:41:17 -08:00
|
|
|
}
|
2018-01-04 14:14:03 -08:00
|
|
|
render () {
|
2018-03-05 22:45:39 -08:00
|
|
|
const { error, videoSource, sliderMinRange, sliderMaxRange, sliderValue, totalMinutes, totalSeconds } = this.state;
|
2018-01-04 14:14:03 -08:00
|
|
|
return (
|
2018-01-11 12:51:38 -08:00
|
|
|
<div>
|
2018-03-05 22:45:39 -08:00
|
|
|
<label className='label'>Thumbnail:</label>
|
|
|
|
<video
|
|
|
|
id='video-thumb-player'
|
|
|
|
preload='metadata'
|
|
|
|
muted
|
|
|
|
style={{display: 'none'}}
|
|
|
|
playsInline
|
|
|
|
onLoadedData={this.handleVideoLoadedData}
|
|
|
|
src={videoSource}
|
|
|
|
onSeeked={this.createThumbnail}
|
|
|
|
/>
|
|
|
|
{
|
|
|
|
sliderValue ? (
|
|
|
|
<div>
|
|
|
|
<div className='flex-container--row flex-container--space-between-center' style={{width: '100%'}}>
|
|
|
|
<span className='info-message'>0'00"</span>
|
|
|
|
<span className='info-message'>{totalMinutes}'{totalSeconds}"</span>
|
|
|
|
</div>
|
|
|
|
<div>
|
2018-02-28 21:25:51 -08:00
|
|
|
<input
|
|
|
|
type='range'
|
|
|
|
min={sliderMinRange}
|
|
|
|
max={sliderMaxRange}
|
|
|
|
value={sliderValue}
|
|
|
|
className='slider'
|
|
|
|
onChange={this.handleSliderChange}
|
|
|
|
/>
|
|
|
|
</div>
|
2018-03-05 22:45:39 -08:00
|
|
|
</div>
|
|
|
|
) : (
|
|
|
|
<p className='info-message' >loading... </p>
|
|
|
|
)
|
|
|
|
}
|
|
|
|
{ error ? (
|
|
|
|
<p className='info-message--failure'>{error}</p>
|
|
|
|
) : (
|
|
|
|
<p className='info-message'>Use slider to set thumbnail</p>
|
|
|
|
)}
|
2018-01-04 14:14:03 -08:00
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-01-17 10:49:57 -08:00
|
|
|
export default PublishThumbnailInput;
|