// @flow import * as ICONS from 'constants/icons'; import * as PAGES from 'constants/pages'; import { MINIMUM_PUBLISH_BID, INVALID_NAME_ERROR } from 'constants/claim'; import React from 'react'; import Card from 'component/common/card'; import Button from 'component/button'; import ChannelSelector from 'component/channelSelector'; import { FormField } from 'component/common/form'; import { parseURI, isNameValid, isURIValid, normalizeURI } from 'util/lbryURI'; import { creditsToString } from 'util/format-credits'; import analytics from 'analytics'; import LbcSymbol from 'component/common/lbc-symbol'; import ClaimPreview from 'component/claimPreview'; import { URL as SITE_URL, URL_LOCAL, URL_DEV } from 'config'; import HelpLink from 'component/common/help-link'; import WalletSpendableBalanceHelp from 'component/walletSpendableBalanceHelp'; import BidHelpText from 'component/publish/shared/publishBid/bid-help-text'; import Spinner from 'component/spinner'; import { REPOST_PARAMS } from 'page/repost/view'; type Props = { doToast: ({ message: string }) => void, doClearRepostError: () => void, doRepost: (StreamRepostOptions) => Promise<*>, doHideModal: () => void, title: string, claim?: StreamClaim, enteredContentClaim?: StreamClaim, balance: number, doCheckPublishNameAvailability: (string) => Promise<*>, error: ?string, reposting: boolean, uri: string, name: string, contentUri: string, setRepostUri: (string) => void, setContentUri: (string) => void, doCheckPendingClaims: () => void, passedRepostAmount: number, enteredRepostAmount: number, isResolvingPassedRepost: boolean, isResolvingEnteredRepost: boolean, activeChannelClaim: ?ChannelClaim, fetchingMyChannels: boolean, incognito: boolean, isRepostPage?: boolean, }; function RepostCreate(props: Props) { const { doToast, doClearRepostError, doRepost, doHideModal, claim, enteredContentClaim, balance, reposting, doCheckPublishNameAvailability, uri, // ?from name, // ?to contentUri, setRepostUri, setContentUri, doCheckPendingClaims, enteredRepostAmount, passedRepostAmount, isResolvingPassedRepost, isResolvingEnteredRepost, activeChannelClaim, fetchingMyChannels, incognito, isRepostPage, } = props; const defaultName = name || (claim && claim.name) || ''; const contentClaimId = claim && claim.claim_id; const enteredClaimId = enteredContentClaim && enteredContentClaim.claim_id; const [repostBid, setRepostBid] = React.useState(0.01); const [repostBidError, setRepostBidError] = React.useState(undefined); const [enteredRepostName, setEnteredRepostName] = React.useState(defaultName); const [available, setAvailable] = React.useState(true); const [enteredContent, setEnteredContentUri] = React.useState(undefined); const [contentError, setContentError] = React.useState(''); const resolvingRepost = isResolvingEnteredRepost || isResolvingPassedRepost; const repostUrlName = `lbry://${incognito || !activeChannelClaim ? '' : `${activeChannelClaim.name}/`}`; const contentFirstRender = React.useRef(true); const setAutoRepostBid = (amount) => { if (balance && balance > 0.02) { if (uri) { setRepostBid(0.01); } else if (balance > amount) { setRepostBid(Number(amount.toFixed(2))); } else { setRepostBid(0.01); } } }; function getSearchUri(value) { const WEB_DEV_PREFIX = `${URL_DEV}/`; const WEB_LOCAL_PREFIX = `${URL_LOCAL}/`; const WEB_PROD_PREFIX = `${SITE_URL}/`; const ODYSEE_PREFIX = `https://odysee.com/`; const includesLbryTvProd = value.includes(WEB_PROD_PREFIX); const includesOdysee = value.includes(ODYSEE_PREFIX); const includesLbryTvLocal = value.includes(WEB_LOCAL_PREFIX); const includesLbryTvDev = value.includes(WEB_DEV_PREFIX); const wasCopiedFromWeb = includesLbryTvDev || includesLbryTvLocal || includesLbryTvProd || includesOdysee; const isLbryUrl = value.startsWith('lbry://') && value !== 'lbry://'; const error = ''; const addLbryIfNot = (term) => { return term.startsWith('lbry://') ? term : `lbry://${term}`; }; if (wasCopiedFromWeb) { let prefix = WEB_PROD_PREFIX; if (includesLbryTvLocal) prefix = WEB_LOCAL_PREFIX; if (includesLbryTvDev) prefix = WEB_DEV_PREFIX; if (includesOdysee) prefix = ODYSEE_PREFIX; let query = (value && value.slice(prefix.length).replace(/:/g, '#')) || ''; try { const lbryUrl = `lbry://${query}`; parseURI(lbryUrl); return [lbryUrl, null]; } catch (e) { return [query, 'error']; } } if (!isLbryUrl) { if (value.startsWith('@')) { if (isNameValid(value.slice(1))) { return [value, null]; } else { return [value, error]; } } return [addLbryIfNot(value), null]; } else { try { const isValid = isURIValid(value); if (isValid) { let uri; try { uri = normalizeURI(value); } catch (e) { return [value, null]; } return [uri, null]; } else { return [value, null]; } } catch (e) { return [value, 'error']; } } } // repostName let repostNameError; if (!enteredRepostName) { repostNameError = __('A name is required'); } else if (!isNameValid(enteredRepostName)) { repostNameError = INVALID_NAME_ERROR; } else if (!available) { repostNameError = __('You already have a claim with this name.'); } // contentName let contentNameError; if (!enteredContent && enteredContent !== undefined) { contentNameError = __('A name is required'); } React.useEffect(() => { if (enteredRepostName && isNameValid(enteredRepostName)) { doCheckPublishNameAvailability(enteredRepostName).then((r) => setAvailable(r)); } }, [enteredRepostName, doCheckPublishNameAvailability]); // takeover amount, bid suggestion React.useEffect(() => { const repostTakeoverAmount = Number(enteredRepostAmount) ? Number(enteredRepostAmount) + 0.01 : Number(passedRepostAmount) + 0.01; if (repostTakeoverAmount) { setAutoRepostBid(repostTakeoverAmount); } }, [enteredRepostAmount, passedRepostAmount]); // repost bid error React.useEffect(() => { let rBidError; if (repostBid === 0) { rBidError = __('Deposit cannot be 0'); } else if (balance === repostBid) { rBidError = __('Please decrease your deposit to account for transaction fees'); } else if (balance < repostBid) { rBidError = __('Deposit cannot be higher than your available balance'); } else if (repostBid < MINIMUM_PUBLISH_BID) { rBidError = __('Your deposit must be higher'); } setRepostBidError(rBidError); }, [setRepostBidError, repostBidError, repostBid]); // setContentUri given enteredUri React.useEffect(() => { if (!enteredContent && !contentFirstRender.current) { setContentError(__('A name is required')); } if (enteredContent) { contentFirstRender.current = false; const [searchContent, error] = getSearchUri(enteredContent); if (error) { setContentError(__('Something not quite right..')); } else { setContentError(''); } try { const { streamName, channelName, isChannel } = parseURI(searchContent); if (!isChannel && streamName && isNameValid(streamName)) { // contentNameValid = true; setContentUri(searchContent); } else if (isChannel && channelName && isNameValid(channelName)) { // contentNameValid = true; setContentUri(searchContent); } } catch (e) { if (enteredContent !== '@') setContentError(''); setContentUri(``); } } }, [enteredContent, setContentUri, setContentError, parseURI, isNameValid]); // setRepostName React.useEffect(() => { if (enteredRepostName) { let isValid = false; try { parseURI(enteredRepostName); isValid = true; } catch (e) { isValid = false; } if (isValid) { setRepostUri(enteredRepostName); } } }, [enteredRepostName, setRepostUri, parseURI]); const repostClaimId = contentClaimId || enteredClaimId; function handleSubmit() { if (enteredRepostName && repostBid && repostClaimId) { doRepost({ name: enteredRepostName, bid: creditsToString(repostBid), channel_id: activeChannelClaim && !incognito ? activeChannelClaim.claim_id : undefined, claim_id: repostClaimId, }).then((repostClaim: StreamClaim) => { doCheckPendingClaims(); analytics.apiLogPublish(repostClaim); doToast({ message: __('Woohoo! Successfully reposted this claim.'), linkText: __('Uploads'), linkTarget: '/uploads', }); doHideModal(); }); } } function cancelIt() { doClearRepostError(); doHideModal(); } if (fetchingMyChannels) { return (