// @flow import 'scss/component/_comment-create.scss'; import { buildValidSticker } from 'util/comments'; import { FF_MAX_CHARS_IN_COMMENT } from 'constants/form-field'; import { FormField, Form } from 'component/common/form'; import { getChannelIdFromClaim } from 'util/claim'; import { useHistory } from 'react-router'; import * as ICONS from 'constants/icons'; import * as KEYCODES from 'constants/keycodes'; import * as PAGES from 'constants/pages'; import Button from 'component/button'; import ChannelMentionSuggestions from 'component/channelMentionSuggestions'; import ChannelThumbnail from 'component/channelThumbnail'; import classnames from 'classnames'; import CreditAmount from 'component/common/credit-amount'; import EmoteSelector from './emote-selector'; import Empty from 'component/common/empty'; import FilePrice from 'component/filePrice'; import I18nMessage from 'component/i18nMessage'; import Icon from 'component/common/icon'; import OptimizedImage from 'component/optimizedImage'; import React from 'react'; import SelectChannel from 'component/selectChannel'; import StickerSelector from './sticker-selector'; import type { ElementRef } from 'react'; import UriIndicator from 'component/uriIndicator'; import usePersistedState from 'effects/use-persisted-state'; import WalletTipAmountSelector from 'component/walletTipAmountSelector'; import { Lbryio } from 'lbryinc'; import { getStripeEnvironment } from 'util/stripe'; const stripeEnvironment = getStripeEnvironment(); const TAB_FIAT = 'TabFiat'; const TAB_LBC = 'TabLBC'; const MENTION_DEBOUNCE_MS = 100; // for sendCashTip REMOVE // type TipParams = { tipAmount: number, tipChannelName: string, channelClaimId: string }; // type UserParams = { activeChannelName: ?string, activeChannelId: ?string }; type Props = { activeChannel: string, activeChannelClaim: ?ChannelClaim, bottom: boolean, hasChannels: boolean, // claim: StreamClaim, claimIsMine: boolean, isFetchingChannels: boolean, isNested: boolean, isReply: boolean, parentId: string, settingsByChannelId: { [channelId: string]: PerChannelSettings }, shouldFetchComment: boolean, supportDisabled: boolean, uri: string, createComment: (string, string, string, ?string, ?string, ?string, ?boolean) => Promise, doFetchCreatorSettings: (channelId: string) => Promise, doToast: ({ message: string }) => void, fetchComment: (commentId: string) => Promise, onCancelReplying?: () => void, onDoneReplying?: () => void, sendTip: ({}, (any) => void, (any) => void) => void, setQuickReply: (any) => void, toast: (string) => void, }; export function CommentCreate(props: Props) { const { activeChannelClaim, bottom, hasChannels, claim, claimIsMine, isFetchingChannels, isNested, isReply, parentId, settingsByChannelId, shouldFetchComment, supportDisabled, uri, createComment, doFetchCreatorSettings, doToast, fetchComment, onCancelReplying, onDoneReplying, sendTip, setQuickReply, } = props; const formFieldRef: ElementRef = React.useRef(); const formFieldInputRef = formFieldRef && formFieldRef.current && formFieldRef.current.input; const selectionIndex = formFieldInputRef && formFieldInputRef.current && formFieldInputRef.current.selectionStart; const buttonRef: ElementRef = React.useRef(); const { push, location: { pathname }, } = useHistory(); const [isSubmitting, setSubmitting] = React.useState(false); const [commentFailure, setCommentFailure] = React.useState(false); const [successTip, setSuccessTip] = React.useState({ txid: undefined, tipAmount: undefined }); const [isSupportComment, setIsSupportComment] = React.useState(); const [isReviewingSupportComment, setReviewingSupportComment] = React.useState(); const [isReviewingStickerComment, setReviewingStickerComment] = React.useState(); const [selectedSticker, setSelectedSticker] = React.useState(); const [tipAmount, setTipAmount] = React.useState(1); const [convertedAmount, setConvertedAmount] = React.useState(); const [commentValue, setCommentValue] = React.useState(''); const [advancedEditor, setAdvancedEditor] = usePersistedState('comment-editor-mode', false); const [stickerSelector, setStickerSelector] = React.useState(); const [activeTab, setActiveTab] = React.useState(); const [tipError, setTipError] = React.useState(); const [deletedComment, setDeletedComment] = React.useState(false); const [pauseQuickSend, setPauseQuickSend] = React.useState(false); const [showEmotes, setShowEmotes] = React.useState(false); const [disableReviewButton, setDisableReviewButton] = React.useState(); const [exchangeRate, setExchangeRate] = React.useState(); const [canReceiveFiatTip, setCanReceiveFiatTip] = React.useState(undefined); const selectedMentionIndex = commentValue.indexOf('@', selectionIndex) === selectionIndex ? commentValue.indexOf('@', selectionIndex) : commentValue.lastIndexOf('@', selectionIndex); const modifierIndex = commentValue.indexOf(':', selectedMentionIndex); const spaceIndex = commentValue.indexOf(' ', selectedMentionIndex); const mentionLengthIndex = modifierIndex >= 0 && (spaceIndex === -1 || modifierIndex < spaceIndex) ? modifierIndex : spaceIndex >= 0 && (modifierIndex === -1 || spaceIndex < modifierIndex) ? spaceIndex : commentValue.length; const channelMention = selectedMentionIndex >= 0 && selectionIndex <= mentionLengthIndex ? commentValue.substring(selectedMentionIndex, mentionLengthIndex) : ''; const claimId = claim && claim.claim_id; const channelUri = claim && (claim.signing_channel ? claim.signing_channel.permanent_url : claim.permanent_url); const charCount = commentValue ? commentValue.length : 0; const disabled = deletedComment || isSubmitting || isFetchingChannels || !commentValue.length || pauseQuickSend; const channelId = getChannelIdFromClaim(claim); const channelSettings = channelId ? settingsByChannelId[channelId] : undefined; const minSuper = (channelSettings && channelSettings.min_tip_amount_super_chat) || 0; const minTip = (channelSettings && channelSettings.min_tip_amount_comment) || 0; const minAmount = minTip || minSuper || 0; const minAmountMet = minAmount === 0 || tipAmount >= minAmount; const stickerPrice = selectedSticker && selectedSticker.price; const minAmountRef = React.useRef(minAmount); minAmountRef.current = minAmount; // ************************************************************************** // Functions // ************************************************************************** function handleSelectSticker(sticker: any) { // $FlowFixMe setSelectedSticker(sticker); setReviewingStickerComment(true); setTipAmount(sticker.price || 0); setStickerSelector(false); if (sticker.price && sticker.price > 0) { setActiveTab(TAB_FIAT); setIsSupportComment(true); } } function handleCommentChange(event) { let commentValue; if (isReply) { commentValue = advancedEditor ? event : event.target.value; } else { commentValue = advancedEditor ? event : event.target.value; } setCommentValue(commentValue); } function handleSelectMention(mentionValue, key) { let newMentionValue = mentionValue.replace('lbry://', ''); if (newMentionValue.includes('#')) newMentionValue = newMentionValue.replace('#', ':'); setCommentValue( commentValue.substring(0, selectedMentionIndex) + `${newMentionValue}` + (commentValue.length > mentionLengthIndex + 1 ? commentValue.substring(mentionLengthIndex, commentValue.length) : ' ') ); } function altEnterListener(e: SyntheticKeyboardEvent<*>) { if ((e.ctrlKey || e.metaKey) && e.keyCode === KEYCODES.ENTER) { e.preventDefault(); buttonRef.current.click(); } } function onTextareaFocus() { window.addEventListener('keydown', altEnterListener); } function onTextareaBlur() { window.removeEventListener('keydown', altEnterListener); } function handleSupportComment() { if (!activeChannelClaim) return; if (!channelId) { doToast({ message: __('Unable to verify channel settings. Try refreshing the page.'), isError: true, }); return; } // if comment post didn't work, but tip was already made, try again to create comment if (commentFailure && tipAmount === successTip.tipAmount) { handleCreateComment(successTip.txid); return; } else { setSuccessTip({ txid: undefined, tipAmount: undefined }); } // !! Beware of stale closure when editing the then-block, including doSubmitTip(). doFetchCreatorSettings(channelId).then(() => { const lockedMinAmount = minAmount; // value during closure. const currentMinAmount = minAmountRef.current; // value from latest doFetchCreatorSettings(). if (lockedMinAmount !== currentMinAmount) { doToast({ message: __('The creator just updated the minimum setting. Please revise or double-check your tip amount.'), isError: true, }); setReviewingSupportComment(false); return; } doSubmitTip(); }); } function doSubmitTip() { if (!activeChannelClaim) return; const params = { amount: tipAmount, claim_id: claimId, channel_id: activeChannelClaim.claim_id }; // FIAT ONLY - REMOVE // const activeChannelName = activeChannelClaim && activeChannelClaim.name; // const activeChannelId = activeChannelClaim && activeChannelClaim.claim_id; // setup variables for tip API // const channelClaimId = claim.signing_channel ? claim.signing_channel.claim_id : claim.claim_id; const tipChannelName = claim.signing_channel ? claim.signing_channel.name : claim.name; setSubmitting(true); if (activeTab === TAB_LBC) { // call sendTip and then run the callback from the response // second parameter is callback sendTip( params, (response) => { const { txid } = response; // todo: why the setTimeout? setTimeout(() => { handleCreateComment(txid); }, 1500); doToast({ message: __( "You sent %tipAmount% LBRY Credits as a tip to %tipChannelName%, I'm sure they appreciate it!", { tipAmount: tipAmount, // force show decimal places tipChannelName, } ), }); setSuccessTip({ txid, tipAmount }); }, () => { // reset the frontend so people can send a new comment setSubmitting(false); } ); } else { // No cash tips // const tipParams: TipParams = { tipAmount: Math.round(tipAmount * 100) / 100, tipChannelName, channelClaimId }; // const userParams: UserParams = { activeChannelName, activeChannelId }; // sendCashTip(tipParams, userParams, claim.claim_id, stripeEnvironment, (customerTipResponse) => { // const { payment_intent_id } = customerTipResponse; // // handleCreateComment(null, payment_intent_id, stripeEnvironment); // // setCommentValue(''); // setReviewingSupportComment(false); // setIsSupportComment(false); // setCommentFailure(false); // setSubmitting(false); // }); } } /** * * @param {string} [txid] Optional transaction id generated by * @param {string} [payment_intent_id] Optional payment_intent_id from Stripe payment * @param {string} [environment] Optional environment for Stripe (test|live) */ function handleCreateComment(txid, payment_intent_id, environment) { setShowEmotes(false); setSubmitting(true); const stickerValue = selectedSticker && buildValidSticker(selectedSticker.name); createComment(stickerValue || commentValue, claimId, parentId, txid, payment_intent_id, environment, !!stickerValue) .then((res) => { setSubmitting(false); if (setQuickReply) setQuickReply(res); if (res && res.signature) { if (!stickerValue) setCommentValue(''); setReviewingSupportComment(false); setIsSupportComment(false); setCommentFailure(false); if (onDoneReplying) { onDoneReplying(); } } }) .catch(() => { setSubmitting(false); setCommentFailure(true); if (channelId) { // It could be that the creator added a minimum tip setting. // Manually update for now until a websocket msg is available. doFetchCreatorSettings(channelId); } }); } // ************************************************************************** // Effects // ************************************************************************** // Fetch channel constraints if not already. React.useEffect(() => { if (!channelSettings && channelId) { doFetchCreatorSettings(channelId); } }, []); // eslint-disable-line react-hooks/exhaustive-deps // Notifications: Fetch top-level comments to identify if it has been deleted and can reply to it React.useEffect(() => { if (shouldFetchComment && fetchComment) { fetchComment(parentId).then((result) => { setDeletedComment(String(result).includes('Error')); }); } }, [fetchComment, shouldFetchComment, parentId]); // Debounce for disabling the submit button when mentioning a user with Enter // so that the comment isn't sent at the same time React.useEffect(() => { const timer = setTimeout(() => { if (pauseQuickSend) { setPauseQuickSend(false); } }, MENTION_DEBOUNCE_MS); return () => clearTimeout(timer); }, [pauseQuickSend]); // Stickers: Get LBC-USD exchange rate if hasn't yet and selected a paid sticker React.useEffect(() => { if (stickerPrice && !exchangeRate) Lbryio.getExchangeRates().then(({ LBC_USD }) => setExchangeRate(LBC_USD)); }, [exchangeRate, stickerPrice]); // Stickers: Check if creator has a tip account saved (on selector so that if a paid sticker is selected, // it defaults to LBC tip instead of USD) React.useEffect(() => { if (!stripeEnvironment || !stickerSelector || canReceiveFiatTip !== undefined) return; const channelClaimId = claim.signing_channel ? claim.signing_channel.claim_id : claim.claim_id; const tipChannelName = claim.signing_channel ? claim.signing_channel.name : claim.name; Lbryio.call( 'account', 'check', { channel_claim_id: channelClaimId, channel_name: tipChannelName, environment: stripeEnvironment, }, 'post' ) .then((accountCheckResponse) => { if (accountCheckResponse === true && canReceiveFiatTip !== true) { setCanReceiveFiatTip(true); } else { setCanReceiveFiatTip(false); } }) .catch(() => {}); }, [canReceiveFiatTip, claim.claim_id, claim.name, claim.signing_channel, stickerSelector]); // ************************************************************************** // Render // ************************************************************************** const getActionButton = (title: string, label?: string, icon: string, handleClick: () => void) => (