2017-12-28 11:51:03 -08:00
|
|
|
import React from 'react';
|
2018-02-02 16:06:21 -08:00
|
|
|
import { withRouter } from 'react-router-dom';
|
2018-01-17 15:00:03 -08:00
|
|
|
import Dropzone from 'containers/Dropzone';
|
|
|
|
import PublishTitleInput from 'containers/PublishTitleInput';
|
|
|
|
import PublishUrlInput from 'containers/PublishUrlInput';
|
|
|
|
import PublishThumbnailInput from 'containers/PublishThumbnailInput';
|
|
|
|
import PublishMetadataInputs from 'containers/PublishMetadataInputs';
|
|
|
|
import ChannelSelect from 'containers/ChannelSelect';
|
2018-01-24 11:24:49 -08:00
|
|
|
import * as publishStates from 'constants/publish_claim_states';
|
2018-01-11 15:37:32 -08:00
|
|
|
|
2018-01-04 16:10:25 -08:00
|
|
|
class PublishForm extends React.Component {
|
|
|
|
constructor (props) {
|
|
|
|
super(props);
|
2018-02-06 15:05:31 -08:00
|
|
|
// this.makePublishRequest = this.makePublishRequest.bind(this);
|
2018-01-04 14:00:02 -08:00
|
|
|
this.publish = this.publish.bind(this);
|
2018-01-02 17:12:57 -08:00
|
|
|
}
|
2018-01-24 12:32:24 -08:00
|
|
|
validateChannelSelection () {
|
2018-01-25 13:37:59 -08:00
|
|
|
console.log('validating channel selection');
|
2018-01-24 12:32:24 -08:00
|
|
|
// make sure all required data is provided
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
// if publishInChannel is true, is a channel selected & logged in?
|
|
|
|
if (this.props.publishInChannel && (this.props.selectedChannel !== this.props.loggedInChannel.name)) {
|
|
|
|
// update state with error
|
2018-01-25 13:37:59 -08:00
|
|
|
this.props.onChannelSelectionError('Log in to a channel or select Anonymous"');
|
2018-01-24 12:32:24 -08:00
|
|
|
// reject this promise
|
|
|
|
return reject(new Error('Fix the channel'));
|
|
|
|
}
|
2018-01-25 13:37:59 -08:00
|
|
|
resolve();
|
2018-01-24 12:32:24 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
validatePublishParams () {
|
2018-01-25 13:37:59 -08:00
|
|
|
console.log('validating publish params');
|
2018-01-11 12:51:38 -08:00
|
|
|
// make sure all required data is provided
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
// is there a file?
|
|
|
|
if (!this.props.file) {
|
|
|
|
return reject(new Error('Please choose a file'));
|
|
|
|
}
|
|
|
|
// is there a claim chosen?
|
|
|
|
if (!this.props.claim) {
|
2018-01-12 10:21:40 -08:00
|
|
|
return reject(new Error('Please enter a URL'));
|
2018-01-11 12:51:38 -08:00
|
|
|
}
|
2018-01-12 14:27:34 -08:00
|
|
|
if (this.props.urlError) {
|
|
|
|
return reject(new Error('Fix the url'));
|
|
|
|
}
|
2018-01-11 12:51:38 -08:00
|
|
|
resolve();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
makePublishRequest (file, metadata) {
|
2018-01-25 13:37:59 -08:00
|
|
|
console.log('making publish request');
|
2018-02-06 21:55:04 -08:00
|
|
|
const uri = '/api/claim/publish';
|
2018-01-11 12:51:38 -08:00
|
|
|
const xhr = new XMLHttpRequest();
|
|
|
|
const fd = this.appendDataToFormData(file, metadata);
|
2018-02-06 15:05:31 -08:00
|
|
|
xhr.upload.addEventListener('loadstart', () => {
|
|
|
|
this.props.onPublishStatusChange(publishStates.LOAD_START, 'upload started');
|
2018-01-11 12:51:38 -08:00
|
|
|
});
|
2018-02-06 15:05:31 -08:00
|
|
|
xhr.upload.addEventListener('progress', (e) => {
|
2018-01-11 12:51:38 -08:00
|
|
|
if (e.lengthComputable) {
|
|
|
|
const percentage = Math.round((e.loaded * 100) / e.total);
|
|
|
|
console.log('progress:', percentage);
|
2018-02-06 15:05:31 -08:00
|
|
|
this.props.onPublishStatusChange(publishStates.LOADING, `${percentage}%`);
|
2018-01-11 12:51:38 -08:00
|
|
|
}
|
|
|
|
}, false);
|
2018-02-06 15:05:31 -08:00
|
|
|
xhr.upload.addEventListener('load', () => {
|
2018-01-11 12:51:38 -08:00
|
|
|
console.log('loaded 100%');
|
2018-02-06 15:05:31 -08:00
|
|
|
this.props.onPublishStatusChange(publishStates.PUBLISHING, null);
|
2018-01-11 12:51:38 -08:00
|
|
|
}, false);
|
|
|
|
xhr.open('POST', uri, true);
|
2018-02-06 15:05:31 -08:00
|
|
|
xhr.onreadystatechange = () => {
|
2018-01-11 12:51:38 -08:00
|
|
|
if (xhr.readyState === 4) {
|
2018-02-02 16:06:21 -08:00
|
|
|
const response = JSON.parse(xhr.response);
|
|
|
|
console.log('publish response:', response);
|
|
|
|
if ((xhr.status === 200) && response.success) {
|
2018-02-06 15:05:31 -08:00
|
|
|
this.props.onPublishStatusChange(publishStates.SUCCESS, response.data.url);
|
|
|
|
this.props.history.push(`/${response.data.claimId}/${response.data.name}`);
|
2018-01-11 12:51:38 -08:00
|
|
|
} else {
|
2018-02-06 15:05:31 -08:00
|
|
|
this.props.onPublishStatusChange(publishStates.FAILED, response.message);
|
2018-01-11 12:51:38 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
// Initiate a multipart/form-data upload
|
|
|
|
xhr.send(fd);
|
|
|
|
}
|
|
|
|
createMetadata () {
|
2018-01-25 13:37:59 -08:00
|
|
|
console.log('creating metadata');
|
2018-01-11 12:51:38 -08:00
|
|
|
let metadata = {
|
|
|
|
name : this.props.claim,
|
|
|
|
title : this.props.title,
|
|
|
|
description: this.props.description,
|
|
|
|
license : this.props.license,
|
|
|
|
nsfw : this.props.nsfw,
|
|
|
|
type : this.props.file.type,
|
|
|
|
thumbnail : this.props.thumbnail,
|
|
|
|
};
|
|
|
|
if (this.props.publishInChannel) {
|
2018-01-24 12:32:24 -08:00
|
|
|
metadata['channelName'] = this.props.selectedChannel;
|
2018-01-11 12:51:38 -08:00
|
|
|
}
|
|
|
|
return metadata;
|
|
|
|
}
|
|
|
|
appendDataToFormData (file, metadata) {
|
|
|
|
var fd = new FormData();
|
|
|
|
fd.append('file', file);
|
|
|
|
for (var key in metadata) {
|
|
|
|
if (metadata.hasOwnProperty(key)) {
|
|
|
|
fd.append(key, metadata[key]);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return fd;
|
|
|
|
}
|
2018-01-02 17:12:57 -08:00
|
|
|
publish () {
|
2018-01-25 13:37:59 -08:00
|
|
|
console.log('publishing file');
|
2018-01-02 17:12:57 -08:00
|
|
|
// publish the asset
|
2018-01-24 12:32:24 -08:00
|
|
|
this.validateChannelSelection()
|
|
|
|
.then(() => {
|
2018-02-06 15:05:31 -08:00
|
|
|
return this.validatePublishParams();
|
2018-01-24 12:32:24 -08:00
|
|
|
})
|
2018-01-11 12:51:38 -08:00
|
|
|
.then(() => {
|
2018-02-06 15:05:31 -08:00
|
|
|
const metadata = this.createMetadata();
|
2018-01-11 12:51:38 -08:00
|
|
|
// publish the claim
|
2018-02-06 15:05:31 -08:00
|
|
|
return this.makePublishRequest(this.props.file, metadata);
|
2018-01-11 12:51:38 -08:00
|
|
|
})
|
|
|
|
.then(() => {
|
2018-02-06 15:05:31 -08:00
|
|
|
this.props.onPublishStatusChange('publish request made');
|
2018-01-11 12:51:38 -08:00
|
|
|
})
|
|
|
|
.catch((error) => {
|
2018-02-06 15:05:31 -08:00
|
|
|
this.props.onPublishSubmitError(error.message);
|
2018-01-11 12:51:38 -08:00
|
|
|
});
|
2018-01-02 17:12:57 -08:00
|
|
|
}
|
2017-12-28 11:51:03 -08:00
|
|
|
render () {
|
|
|
|
return (
|
2018-01-04 11:05:16 -08:00
|
|
|
<div className="row row--no-bottom">
|
2018-01-02 17:12:57 -08:00
|
|
|
<div className="column column--10">
|
2018-01-08 17:46:17 -08:00
|
|
|
<PublishTitleInput />
|
2018-01-02 17:12:57 -08:00
|
|
|
</div>
|
|
|
|
<div className="column column--5 column--sml-10" >
|
2018-01-10 17:41:17 -08:00
|
|
|
<div className="row row--padded">
|
2018-01-17 09:24:17 -08:00
|
|
|
<Dropzone />
|
2017-12-28 11:51:03 -08:00
|
|
|
</div>
|
2018-01-02 17:12:57 -08:00
|
|
|
</div>
|
|
|
|
<div className="column column--5 column--sml-10 align-content-top">
|
|
|
|
<div id="publish-active-area" className="row row--padded">
|
2018-01-10 11:26:01 -08:00
|
|
|
<div className="row row--padded row--no-top row--wide">
|
|
|
|
<PublishUrlInput />
|
|
|
|
</div>
|
|
|
|
<div className="row row--padded row--no-top row--wide">
|
2018-01-17 09:46:16 -08:00
|
|
|
<ChannelSelect />
|
2018-01-10 11:26:01 -08:00
|
|
|
</div>
|
2018-01-11 12:51:38 -08:00
|
|
|
{ (this.props.file.type === 'video/mp4') && (
|
2018-01-17 09:46:16 -08:00
|
|
|
<div className="row row--padded row--no-top row--wide ">
|
2018-01-11 12:51:38 -08:00
|
|
|
<PublishThumbnailInput />
|
|
|
|
</div>
|
|
|
|
)}
|
2018-01-10 13:10:08 -08:00
|
|
|
<div className="row row--padded row--no-top row--no-bottom row--wide">
|
|
|
|
<PublishMetadataInputs />
|
|
|
|
</div>
|
2018-01-17 09:46:16 -08:00
|
|
|
<div className="row row--wide align-content-center">
|
2018-01-05 16:47:55 -08:00
|
|
|
<button id="publish-submit" className="button--primary button--large" onClick={this.publish}>Publish</button>
|
2017-12-28 11:51:03 -08:00
|
|
|
</div>
|
2018-01-17 09:46:16 -08:00
|
|
|
<div className="row row--padded row--no-bottom align-content-center">
|
2018-01-08 17:06:31 -08:00
|
|
|
<button className="button--cancel" onClick={this.props.onFileClear}>Cancel</button>
|
2018-01-02 17:12:57 -08:00
|
|
|
</div>
|
|
|
|
<div className="row row--short align-content-center">
|
2018-01-19 14:53:31 -08:00
|
|
|
<p className="fine-print">By clicking 'Publish', you affirm that you have the rights to publish this content to the LBRY network, and that you understand the properties of publishing it to a decentralized, user-controlled network. <a className="link--primary" target="_blank" href="https://lbry.io/learn">Read more.</a></p>
|
2017-12-28 11:51:03 -08:00
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2018-02-02 16:06:21 -08:00
|
|
|
export default withRouter(PublishForm);
|