2017-06-30 15:45:54 +07:00
import React from "react" ;
import lbry from "lbry" ;
import lbryuri from "lbryuri" ;
import { FormField , FormRow } from "component/form.js" ;
import Link from "component/link" ;
import Modal from "component/modal" ;
import { BusyMessage } from "component/common" ;
2017-07-13 10:19:27 -04:00
import ChannelSection from "./internal/channelSection" ;
2017-06-30 15:45:54 +07:00
class PublishForm extends React . PureComponent {
constructor ( props ) {
super ( props ) ;
this . _requiredFields = [ "name" , "bid" , "meta_title" , "tosAgree" ] ;
this . _defaultCopyrightNotice = "All rights reserved." ;
this . state = {
rawName : "" ,
name : "" ,
bid : 10 ,
hasFile : false ,
feeAmount : "" ,
feeCurrency : "USD" ,
channel : "anonymous" ,
newChannelName : "@" ,
newChannelBid : 10 ,
meta _title : "" ,
meta _thumbnail : "" ,
meta _description : "" ,
meta _language : "en" ,
meta _nsfw : "0" ,
licenseType : "" ,
copyrightNotice : this . _defaultCopyrightNotice ,
otherLicenseDescription : "" ,
otherLicenseUrl : "" ,
tosAgree : false ,
prefillDone : false ,
uploadProgress : 0.0 ,
uploaded : false ,
errorMessage : null ,
submitting : false ,
creatingChannel : false ,
modal : null ,
} ;
}
_updateChannelList ( channel ) {
const { fetchingChannels , fetchChannelListMine } = this . props ;
if ( ! fetchingChannels ) fetchChannelListMine ( ) ;
}
handleSubmit ( event ) {
if ( typeof event !== "undefined" ) {
event . preventDefault ( ) ;
}
this . setState ( {
submitting : true ,
} ) ;
let checkFields = this . _requiredFields ;
if ( ! this . myClaimExists ( ) ) {
checkFields . unshift ( "file" ) ;
}
let missingFieldFound = false ;
for ( let fieldName of checkFields ) {
const field = this . refs [ fieldName ] ;
if ( field ) {
if ( field . getValue ( ) === "" || field . getValue ( ) === false ) {
field . showRequiredError ( ) ;
if ( ! missingFieldFound ) {
field . focus ( ) ;
missingFieldFound = true ;
}
} else {
field . clearError ( ) ;
}
}
}
if ( missingFieldFound ) {
this . setState ( {
submitting : false ,
} ) ;
return ;
}
let metadata = { } ;
for ( let metaField of [ "title" , "description" , "thumbnail" , "language" ] ) {
const value = this . state [ "meta_" + metaField ] ;
if ( value ) {
metadata [ metaField ] = value ;
}
}
metadata . license = this . getLicense ( ) ;
metadata . licenseUrl = this . getLicenseUrl ( ) ;
metadata . nsfw = ! ! parseInt ( this . state . meta _nsfw ) ;
var doPublish = ( ) => {
var publishArgs = {
name : this . state . name ,
bid : parseFloat ( this . state . bid ) ,
metadata : metadata ,
... ( this . state . channel != "new" && this . state . channel != "anonymous"
? { channel _name : this . state . channel }
: { } ) ,
} ;
if ( this . refs . file . getValue ( ) !== "" ) {
publishArgs . file _path = this . refs . file . getValue ( ) ;
}
const success = claim => { } ;
const failure = error => this . handlePublishError ( error ) ;
this . handlePublishStarted ( ) ;
this . props . publish ( publishArgs ) . then ( success , failure ) ;
} ;
if ( this . state . isFee ) {
lbry . wallet _unused _address ( ) . then ( address => {
metadata . fee = {
currency : this . state . feeCurrency ,
amount : parseFloat ( this . state . feeAmount ) ,
address : address ,
} ;
doPublish ( ) ;
} ) ;
} else {
doPublish ( ) ;
}
}
handlePublishStarted ( ) {
this . setState ( {
modal : "publishStarted" ,
} ) ;
}
handlePublishStartedConfirmed ( ) {
this . props . navigate ( "/published" ) ;
}
handlePublishError ( error ) {
this . setState ( {
submitting : false ,
modal : "error" ,
errorMessage : error . message ,
} ) ;
}
claim ( ) {
const { claimsByUri } = this . props ;
const { uri } = this . state ;
return claimsByUri [ uri ] ;
}
topClaimValue ( ) {
if ( ! this . claim ( ) ) return null ;
return parseFloat ( this . claim ( ) . amount ) ;
}
myClaimExists ( ) {
const { myClaims } = this . props ;
const { name } = this . state ;
if ( ! name ) return false ;
return ! ! myClaims . find ( claim => claim . name === name ) ;
}
topClaimIsMine ( ) {
const myClaimInfo = this . myClaimInfo ( ) ;
const { claimsByUri } = this . props ;
const { uri } = this . state ;
if ( ! uri ) return null ;
const claim = claimsByUri [ uri ] ;
if ( ! claim ) return true ;
if ( ! myClaimInfo ) return false ;
2017-07-12 14:05:10 +07:00
return myClaimInfo . amount >= claim . amount ;
2017-06-30 15:45:54 +07:00
}
myClaimInfo ( ) {
const { name } = this . state ;
return Object . values ( this . props . myClaims ) . find (
claim => claim . name === name
) ;
}
handleNameChange ( event ) {
var rawName = event . target . value ;
this . nameChanged ( rawName ) ;
}
nameChanged ( rawName ) {
if ( ! rawName ) {
this . setState ( {
rawName : "" ,
name : "" ,
uri : "" ,
prefillDone : false ,
} ) ;
return ;
}
if ( ! lbryuri . isValidName ( rawName , false ) ) {
this . refs . name . showError (
_ _ ( "LBRY names must contain only letters, numbers and dashes." )
) ;
return ;
}
let channel = "" ;
if ( this . state . channel !== "anonymous" ) channel = this . state . channel ;
const name = rawName . toLowerCase ( ) ;
const uri = lbryuri . build ( { contentName : name , channelName : channel } ) ;
this . setState ( {
rawName : rawName ,
name : name ,
prefillDone : false ,
uri ,
} ) ;
if ( this . resolveUriTimeout ) {
clearTimeout ( this . resolveUriTimeout ) ;
this . resolveUriTimeout = undefined ;
}
const resolve = ( ) => this . props . resolveUri ( uri ) ;
this . resolveUriTimeout = setTimeout ( resolve . bind ( this ) , 500 , {
once : true ,
} ) ;
}
handlePrefillClicked ( ) {
const claimInfo = this . myClaimInfo ( ) ;
const {
license ,
licenseUrl ,
title ,
thumbnail ,
description ,
language ,
nsfw ,
} = claimInfo . value . stream . metadata ;
let newState = {
meta _title : title ,
meta _thumbnail : thumbnail ,
meta _description : description ,
meta _language : language ,
meta _nsfw : nsfw ,
prefillDone : true ,
bid : claimInfo . amount ,
} ;
if ( license == this . _defaultCopyrightNotice ) {
newState . licenseType = "copyright" ;
newState . copyrightNotice = this . _defaultCopyrightNotice ;
} else {
// If the license URL or description matches one of the drop-down options, use that
let licenseType = "other" ; // Will be overridden if we find a match
for ( let option of this . _meta _license . getOptions ( ) ) {
if (
option . getAttribute ( "data-url" ) === licenseUrl ||
option . text === license
) {
licenseType = option . value ;
}
}
if ( licenseType == "other" ) {
newState . otherLicenseDescription = license ;
newState . otherLicenseUrl = licenseUrl ;
}
newState . licenseType = licenseType ;
}
this . setState ( newState ) ;
}
handleBidChange ( event ) {
this . setState ( {
bid : event . target . value ,
} ) ;
}
handleFeeAmountChange ( event ) {
this . setState ( {
feeAmount : event . target . value ,
} ) ;
}
handleFeeCurrencyChange ( event ) {
this . setState ( {
feeCurrency : event . target . value ,
} ) ;
}
handleFeePrefChange ( feeEnabled ) {
this . setState ( {
isFee : feeEnabled ,
} ) ;
}
handleMetadataChange ( event ) {
/ * *
* This function is used for all metadata inputs that store the final value directly into state .
* The only exceptions are inputs related to license description and license URL , which require
* more complex logic and the final value is determined at submit time .
* /
this . setState ( {
[ "meta_" + event . target . name ] : event . target . value ,
} ) ;
}
handleDescriptionChanged ( text ) {
this . setState ( {
meta _description : text ,
} ) ;
}
handleLicenseTypeChange ( event ) {
this . setState ( {
licenseType : event . target . value ,
} ) ;
}
handleCopyrightNoticeChange ( event ) {
this . setState ( {
copyrightNotice : event . target . value ,
} ) ;
}
handleOtherLicenseDescriptionChange ( event ) {
this . setState ( {
otherLicenseDescription : event . target . value ,
} ) ;
}
handleOtherLicenseUrlChange ( event ) {
this . setState ( {
otherLicenseUrl : event . target . value ,
} ) ;
}
handleChannelChange ( channelName ) {
this . setState ( {
channel : channelName ,
} ) ;
const nameChanged = ( ) => this . nameChanged ( this . state . rawName ) ;
setTimeout ( nameChanged . bind ( this ) , 500 , { once : true } ) ;
}
handleTOSChange ( event ) {
this . setState ( {
tosAgree : event . target . checked ,
} ) ;
}
handleCreateChannelClick ( event ) {
if ( this . state . newChannelName . length < 5 ) {
this . refs . newChannelName . showError (
_ _ ( "LBRY channel names must be at least 4 characters in length." )
) ;
return ;
}
this . setState ( {
creatingChannel : true ,
} ) ;
const newChannelName = this . state . newChannelName ;
lbry
. channel _new ( {
channel _name : newChannelName ,
amount : parseFloat ( this . state . newChannelBid ) ,
} )
. then (
( ) => {
setTimeout ( ( ) => {
this . setState ( {
creatingChannel : false ,
} ) ;
this . _updateChannelList ( newChannelName ) ;
} , 10000 ) ;
} ,
error => {
// TODO: better error handling
this . refs . newChannelName . showError (
_ _ ( "Unable to create channel due to an internal error." )
) ;
this . setState ( {
creatingChannel : false ,
} ) ;
}
) ;
}
getLicense ( ) {
switch ( this . state . licenseType ) {
case "copyright" :
return this . state . copyrightNotice ;
case "other" :
return this . state . otherLicenseDescription ;
default :
return this . _meta _license . getSelectedElement ( ) . text ;
}
}
getLicenseUrl ( ) {
switch ( this . state . licenseType ) {
case "copyright" :
return "" ;
case "other" :
return this . state . otherLicenseUrl ;
default :
return this . _meta _license . getSelectedElement ( ) . getAttribute ( "data-url" ) ;
}
}
componentWillMount ( ) {
this . props . fetchClaimListMine ( ) ;
this . _updateChannelList ( ) ;
}
onFileChange ( ) {
if ( this . refs . file . getValue ( ) ) {
this . setState ( { hasFile : true } ) ;
} else {
this . setState ( { hasFile : false } ) ;
}
}
getNameBidHelpText ( ) {
if ( this . state . prefillDone ) {
2017-07-12 14:05:10 +07:00
return _ _ ( "Existing claim data was prefilled" ) ;
2017-06-30 15:45:54 +07:00
}
if (
this . state . uri &&
this . props . resolvingUris . indexOf ( this . state . uri ) !== - 1 &&
this . claim ( ) === undefined
) {
return _ _ ( "Checking..." ) ;
} else if ( ! this . state . name ) {
return _ _ ( "Select a URL for this publish." ) ;
} else if ( ! this . claim ( ) ) {
return _ _ ( "This URL is unused." ) ;
} else if ( this . myClaimExists ( ) && ! this . state . prefillDone ) {
return (
2017-07-12 14:05:10 +07:00
< span >
2017-06-30 15:45:54 +07:00
{ _ _ ( "You already have a claim with this name." ) } { " " }
< Link
label = { _ _ ( "Use data from my existing claim" ) }
onClick = { ( ) => this . handlePrefillClicked ( ) }
/ >
2017-07-12 14:05:10 +07:00
< / span >
2017-06-30 15:45:54 +07:00
) ;
} else if ( this . claim ( ) ) {
if ( this . topClaimValue ( ) === 1 ) {
return (
< span >
{ _ _ (
'A deposit of at least one credit is required to win "%s". However, you can still get a permanent URL for any amount.' ,
this . state . name
) }
< / span >
) ;
} else {
return (
< span >
{ _ _ (
'A deposit of at least "%s" credits is required to win "%s". However, you can still get a permanent URL for any amount.' ,
this . topClaimValue ( ) ,
this . state . name
) }
< / span >
) ;
}
} else {
return "" ;
}
}
closeModal ( ) {
this . setState ( {
modal : null ,
} ) ;
}
render ( ) {
const lbcInputHelp = _ _ (
"This LBC remains yours and the deposit can be undone at any time."
) ;
return (
< main className = "main--single-column" >
< form
onSubmit = { event => {
this . handleSubmit ( event ) ;
} }
>
< section className = "card" >
< div className = "card__title-primary" >
< h4 > { _ _ ( "Content" ) } < / h4 >
< div className = "card__subtitle" >
{ _ _ ( "What are you publishing?" ) }
< / div >
< / div >
< div className = "card__content" >
< FormRow
name = "file"
label = "File"
ref = "file"
type = "file"
onChange = { event => {
this . onFileChange ( event ) ;
} }
helper = {
this . myClaimExists ( )
? _ _ (
"If you don't choose a file, the file from your existing claim will be used."
)
: null
}
/ >
< / div >
{ ! this . state . hasFile && ! this . myClaimExists ( )
? null
: < div >
< div className = "card__content" >
< FormRow
label = { _ _ ( "Title" ) }
type = "text"
name = "title"
value = { this . state . meta _title }
placeholder = "Titular Title"
onChange = { event => {
this . handleMetadataChange ( event ) ;
} }
/ >
< / div >
< div className = "card__content" >
< FormRow
type = "text"
label = { _ _ ( "Thumbnail URL" ) }
name = "thumbnail"
value = { this . state . meta _thumbnail }
placeholder = "http://spee.ch/mylogo"
onChange = { event => {
this . handleMetadataChange ( event ) ;
} }
/ >
< / div >
< div className = "card__content" >
< FormRow
label = { _ _ ( "Description" ) }
type = "SimpleMDE"
ref = "meta_description"
name = "description"
value = { this . state . meta _description }
placeholder = { _ _ ( "Description of your content" ) }
onChange = { text => {
this . handleDescriptionChanged ( text ) ;
} }
/ >
< / div >
< div className = "card__content" >
< FormRow
label = { _ _ ( "Language" ) }
type = "select"
value = { this . state . meta _language }
name = "language"
onChange = { event => {
this . handleMetadataChange ( event ) ;
} }
>
< option value = "en" > { _ _ ( "English" ) } < / option >
< option value = "zh" > { _ _ ( "Chinese" ) } < / option >
< option value = "fr" > { _ _ ( "French" ) } < / option >
< option value = "de" > { _ _ ( "German" ) } < / option >
< option value = "jp" > { _ _ ( "Japanese" ) } < / option >
< option value = "ru" > { _ _ ( "Russian" ) } < / option >
< option value = "es" > { _ _ ( "Spanish" ) } < / option >
< / FormRow >
< / div >
< div className = "card__content" >
< FormRow
type = "select"
label = { _ _ ( "Maturity" ) }
value = { this . state . meta _nsfw }
name = "nsfw"
onChange = { event => {
this . handleMetadataChange ( event ) ;
} }
>
{ /* <option value=""></option> */ }
< option value = "0" > { _ _ ( "All Ages" ) } < / option >
< option value = "1" > { _ _ ( "Adults Only" ) } < / option >
< / FormRow >
< / div >
< / div > }
< / section >
< section className = "card" >
< div className = "card__title-primary" >
< h4 > { _ _ ( "Access" ) } < / h4 >
< div className = "card__subtitle" >
{ _ _ ( "How much does this content cost?" ) }
< / div >
< / div >
< div className = "card__content" >
< div className = "form-row__label-row" >
< label className = "form-row__label" > { _ _ ( "Price" ) } < / label >
< / div >
< FormRow
label = { _ _ ( "Free" ) }
type = "radio"
name = "isFree"
value = "1"
onChange = { ( ) => {
this . handleFeePrefChange ( false ) ;
} }
defaultChecked = { ! this . state . isFee }
/ >
< FormField
type = "radio"
name = "isFree"
label = { ! this . state . isFee ? _ _ ( "Choose price..." ) : _ _ ( "Price " ) }
onChange = { ( ) => {
this . handleFeePrefChange ( true ) ;
} }
defaultChecked = { this . state . isFee }
/ >
< span className = { ! this . state . isFee ? "hidden" : "" } >
< FormField
type = "number"
className = "form-field__input--inline"
step = "0.01"
placeholder = "1.00"
min = "0.01"
onChange = { event => this . handleFeeAmountChange ( event ) }
/ > { " " }
< FormField
type = "select"
onChange = { event => {
this . handleFeeCurrencyChange ( event ) ;
} }
>
< option value = "USD" > { _ _ ( "US Dollars" ) } < / option >
< option value = "LBC" > { _ _ ( "LBRY credits" ) } < / option >
< / FormField >
< / span >
{ this . state . isFee
? < div className = "form-field__helper" >
{ _ _ (
"If you choose to price this content in dollars, the number of credits charged will be adjusted based on the value of LBRY credits at the time of purchase."
) }
< / div >
: "" }
< FormRow
label = "License"
type = "select"
value = { this . state . licenseType }
ref = { row => {
this . _meta _license = row ;
} }
onChange = { event => {
this . handleLicenseTypeChange ( event ) ;
} }
>
< option / >
< option value = "publicDomain" > { _ _ ( "Public Domain" ) } < / option >
< option
value = "cc-by"
data - url = "https://creativecommons.org/licenses/by/4.0/legalcode"
>
{ _ _ ( "Creative Commons Attribution 4.0 International" ) }
< / option >
< option
value = "cc-by-sa"
data - url = "https://creativecommons.org/licenses/by-sa/4.0/legalcode"
>
{ _ _ (
"Creative Commons Attribution-ShareAlike 4.0 International"
) }
< / option >
< option
value = "cc-by-nd"
data - url = "https://creativecommons.org/licenses/by-nd/4.0/legalcode"
>
{ _ _ (
"Creative Commons Attribution-NoDerivatives 4.0 International"
) }
< / option >
< option
value = "cc-by-nc"
data - url = "https://creativecommons.org/licenses/by-nc/4.0/legalcode"
>
{ _ _ (
"Creative Commons Attribution-NonCommercial 4.0 International"
) }
< / option >
< option
value = "cc-by-nc-sa"
data - url = "https://creativecommons.org/licenses/by-nc-sa/4.0/legalcode"
>
{ _ _ (
"Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International"
) }
< / option >
< option
value = "cc-by-nc-nd"
data - url = "https://creativecommons.org/licenses/by-nc-nd/4.0/legalcode"
>
{ _ _ (
"Creative Commons Attribution-NonCommercial-NoDerivatives 4.0 International"
) }
< / option >
< option value = "copyright" >
{ _ _ ( "Copyrighted..." ) }
< / option >
< option value = "other" >
{ _ _ ( "Other..." ) }
< / option >
< / FormRow >
{ this . state . licenseType == "copyright"
? < FormRow
label = { _ _ ( "Copyright notice" ) }
type = "text"
name = "copyright-notice"
value = { this . state . copyrightNotice }
onChange = { event => {
this . handleCopyrightNoticeChange ( event ) ;
} }
/ >
: null }
{ this . state . licenseType == "other"
? < FormRow
label = { _ _ ( "License description" ) }
type = "text"
name = "other-license-description"
value = { this . state . otherLicenseDescription }
onChange = { event => {
this . handleOtherLicenseDescriptionChange ( event ) ;
} }
/ >
: null }
{ this . state . licenseType == "other"
? < FormRow
label = { _ _ ( "License URL" ) }
type = "text"
name = "other-license-url"
value = { this . state . otherLicenseUrl }
onChange = { event => {
this . handleOtherLicenseUrlChange ( event ) ;
} }
/ >
: null }
< / div >
< / section >
< ChannelSection
{ ... this . props }
handleChannelChange = { this . handleChannelChange . bind ( this ) }
channel = { this . state . channel }
/ >
< section className = "card" >
< div className = "card__title-primary" >
< h4 > { _ _ ( "Address" ) } < / h4 >
< div className = "card__subtitle" >
{ _ _ ( "Where should this content permanently reside?" ) }
{ " " }
< Link
label = { _ _ ( "Read more" ) }
href = "https://lbry.io/faq/naming"
/ > .
< / div >
< / div >
< div className = "card__content" >
< FormRow
prefix = { ` lbry:// ${ this . state . channel === "anonymous"
? ""
: ` ${ this . state . channel } / ` } ` }
type = "text"
ref = "name"
placeholder = "myname"
value = { this . state . rawName }
onChange = { event => {
this . handleNameChange ( event ) ;
} }
helper = { this . getNameBidHelpText ( ) }
/ >
< / div >
{ this . state . rawName
? < div className = "card__content" >
< FormRow
ref = "bid"
type = "number"
step = "0.01"
label = { _ _ ( "Deposit" ) }
postfix = "LBC"
onChange = { event => {
this . handleBidChange ( event ) ;
} }
value = { this . state . bid }
placeholder = { this . claim ( ) ? this . topClaimValue ( ) + 10 : 100 }
helper = { lbcInputHelp }
/ >
< / div >
: "" }
< / section >
< section className = "card" >
< div className = "card__title-primary" >
< h4 > { _ _ ( "Terms of Service" ) } < / h4 >
< / div >
< div className = "card__content" >
< FormRow
label = {
< span >
{ _ _ ( "I agree to the" ) }
{ " " }
< Link
href = "https://www.lbry.io/termsofservice"
label = { _ _ ( "LBRY terms of service" ) }
/ >
< / span >
}
type = "checkbox"
checked = { this . state . tosAgree }
onChange = { event => {
this . handleTOSChange ( event ) ;
} }
/ >
< / div >
< / section >
< div className = "card-series-submit" >
< Link
button = "primary"
label = {
! this . state . submitting ? _ _ ( "Publish" ) : _ _ ( "Publishing..." )
}
onClick = { event => {
this . handleSubmit ( event ) ;
} }
2017-07-08 15:03:12 +07:00
disabled = {
this . state . submitting ||
( this . state . uri &&
2017-07-12 13:36:08 +07:00
this . props . resolvingUris . indexOf ( this . state . uri ) !== - 1 ) ||
( this . claim ( ) &&
! this . topClaimIsMine ( ) &&
this . state . bid <= this . topClaimValue ( ) )
2017-07-08 15:03:12 +07:00
}
2017-06-30 15:45:54 +07:00
/ >
< Link
button = "cancel"
onClick = { this . props . back }
label = { _ _ ( "Cancel" ) }
/ >
< input type = "submit" className = "hidden" / >
< / div >
< / form >
< Modal
isOpen = { this . state . modal == "publishStarted" }
contentLabel = { _ _ ( "File published" ) }
onConfirmed = { event => {
this . handlePublishStartedConfirmed ( event ) ;
} }
>
< p >
{ _ _ ( "Your file has been published to LBRY at the address" ) }
{ " " } < code > { this . state . uri } < / code > !
< / p >
< p >
{ _ _ (
'The file will take a few minutes to appear for other LBRY users. Until then it will be listed as "pending" under your published files.'
) }
< / p >
< / Modal >
< Modal
isOpen = { this . state . modal == "error" }
contentLabel = { _ _ ( "Error publishing file" ) }
onConfirmed = { event => {
this . closeModal ( event ) ;
} }
>
{ _ _ (
"The following error occurred when attempting to publish your file"
) } : { this . state . errorMessage }
< / Modal >
< / main >
) ;
}
}
export default PublishForm ;