spee.ch/react/containers/PublishForm/view.jsx

172 lines
6.3 KiB
React
Raw Normal View History

import React from 'react';
import { withRouter } from 'react-router-dom';
2018-01-18 00:00:03 +01: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';
import * as publishStates from 'constants/publish_claim_states';
2018-01-12 00:37:32 +01:00
class PublishForm extends React.Component {
constructor (props) {
super(props);
2018-02-07 00:05:31 +01:00
// this.makePublishRequest = this.makePublishRequest.bind(this);
this.publish = this.publish.bind(this);
}
validateChannelSelection () {
console.log('validating channel selection');
// 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
this.props.onChannelSelectionError('Log in to a channel or select Anonymous"');
// reject this promise
return reject(new Error('Fix the channel'));
}
resolve();
});
}
validatePublishParams () {
console.log('validating publish params');
2018-01-11 21:51:38 +01: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) {
return reject(new Error('Please enter a URL'));
2018-01-11 21:51:38 +01:00
}
if (this.props.urlError) {
return reject(new Error('Fix the url'));
}
2018-01-11 21:51:38 +01:00
resolve();
});
}
makePublishRequest (file, metadata) {
console.log('making publish request');
2018-02-07 06:55:04 +01:00
const uri = '/api/claim/publish';
2018-01-11 21:51:38 +01:00
const xhr = new XMLHttpRequest();
const fd = this.appendDataToFormData(file, metadata);
2018-02-07 00:05:31 +01:00
xhr.upload.addEventListener('loadstart', () => {
this.props.onPublishStatusChange(publishStates.LOAD_START, 'upload started');
2018-01-11 21:51:38 +01:00
});
2018-02-07 00:05:31 +01:00
xhr.upload.addEventListener('progress', (e) => {
2018-01-11 21:51:38 +01:00
if (e.lengthComputable) {
const percentage = Math.round((e.loaded * 100) / e.total);
console.log('progress:', percentage);
2018-02-07 00:05:31 +01:00
this.props.onPublishStatusChange(publishStates.LOADING, `${percentage}%`);
2018-01-11 21:51:38 +01:00
}
}, false);
2018-02-07 00:05:31 +01:00
xhr.upload.addEventListener('load', () => {
2018-01-11 21:51:38 +01:00
console.log('loaded 100%');
2018-02-07 00:05:31 +01:00
this.props.onPublishStatusChange(publishStates.PUBLISHING, null);
2018-01-11 21:51:38 +01:00
}, false);
xhr.open('POST', uri, true);
2018-02-07 00:05:31 +01:00
xhr.onreadystatechange = () => {
2018-01-11 21:51:38 +01:00
if (xhr.readyState === 4) {
const response = JSON.parse(xhr.response);
console.log('publish response:', response);
if ((xhr.status === 200) && response.success) {
2018-02-07 00:05:31 +01:00
this.props.history.push(`/${response.data.claimId}/${response.data.name}`);
this.props.onFileClear();
2018-01-11 21:51:38 +01:00
} else {
2018-02-07 00:05:31 +01:00
this.props.onPublishStatusChange(publishStates.FAILED, response.message);
2018-01-11 21:51:38 +01:00
}
}
};
// Initiate a multipart/form-data upload
xhr.send(fd);
}
createMetadata () {
console.log('creating metadata');
2018-01-11 21:51:38 +01: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) {
metadata['channelName'] = this.props.selectedChannel;
2018-01-11 21:51:38 +01: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;
}
publish () {
console.log('publishing file');
// publish the asset
this.validateChannelSelection()
.then(() => {
2018-02-07 00:05:31 +01:00
return this.validatePublishParams();
})
2018-01-11 21:51:38 +01:00
.then(() => {
2018-02-07 00:05:31 +01:00
const metadata = this.createMetadata();
2018-01-11 21:51:38 +01:00
// publish the claim
2018-02-07 00:05:31 +01:00
return this.makePublishRequest(this.props.file, metadata);
2018-01-11 21:51:38 +01:00
})
.catch((error) => {
2018-02-07 00:05:31 +01:00
this.props.onPublishSubmitError(error.message);
2018-01-11 21:51:38 +01:00
});
}
render () {
return (
<div className='row row--no-bottom'>
<div className='column column--10'>
<PublishTitleInput />
</div>
2018-02-06 03:14:12 +01:00
{/* left column */}
<div className='column column--5 column--sml-10' >
<div className='row row--padded'>
2018-01-17 18:24:17 +01:00
<Dropzone />
</div>
</div>
2018-02-06 03:14:12 +01:00
{/* right column */}
<div className='column column--5 column--sml-10 align-content-top'>
<div id='publish-active-area' className='row row--padded'>
<div className='row row--padded row--no-top row--wide'>
2018-01-10 20:26:01 +01:00
<PublishUrlInput />
</div>
<div className='row row--padded row--no-top row--wide'>
2018-01-17 18:46:16 +01:00
<ChannelSelect />
2018-01-10 20:26:01 +01:00
</div>
2018-01-11 21:51:38 +01:00
{ (this.props.file.type === 'video/mp4') && (
<div className='row row--padded row--no-top row--wide '>
2018-01-11 21:51:38 +01:00
<PublishThumbnailInput />
</div>
)}
<div className='row row--padded row--no-top row--no-bottom row--wide'>
2018-01-10 22:10:08 +01:00
<PublishMetadataInputs />
</div>
<div className='row row--wide align-content-center'>
<button id='publish-submit' className='button--primary button--large' onClick={this.publish}>Publish</button>
</div>
<div className='row row--padded row--no-bottom align-content-center'>
<button className='button--cancel' onClick={this.props.onFileClear}>Cancel</button>
</div>
<div className='row row--short align-content-center'>
<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>
</div>
</div>
</div>
</div>
);
}
};
export default withRouter(PublishForm);