import React, { useState } from 'react'; import { defineMessages, useIntl, FormattedMessage } from 'react-intl'; import { useHistory } from 'react-router-dom'; import { remoteInteraction } from 'soapbox/actions/interactions'; import { Button, Form, Input, Modal, Stack, Text } from 'soapbox/components/ui'; import { useAppSelector, useAppDispatch, useFeatures, useInstance, useRegistrationStatus } from 'soapbox/hooks'; import { selectAccount } from 'soapbox/selectors'; import toast from 'soapbox/toast'; const messages = defineMessages({ close: { id: 'lightbox.close', defaultMessage: 'Close' }, accountPlaceholder: { id: 'remote_interaction.account_placeholder', defaultMessage: 'Enter your username@domain you want to act from' }, userNotFoundError: { id: 'remote_interaction.user_not_found_error', defaultMessage: 'Couldn\'t find given user' }, }); interface IUnauthorizedModal { /** Unauthorized action type. */ action: 'FOLLOW' | 'REPLY' | 'REBLOG' | 'FAVOURITE' | 'DISLIKE' | 'POLL_VOTE' | 'JOIN' /** Close event handler. */ onClose: (modalType: string) => void /** ActivityPub ID of the account OR status being acted upon. */ ap_id?: string /** Account ID of the account being acted upon. */ account?: string } /** Modal to display when a logged-out user tries to do something that requires login. */ const UnauthorizedModal: React.FC<IUnauthorizedModal> = ({ action, onClose, account: accountId, ap_id: apId }) => { const intl = useIntl(); const history = useHistory(); const dispatch = useAppDispatch(); const instance = useInstance(); const { isOpen } = useRegistrationStatus(); const username = useAppSelector(state => selectAccount(state, accountId!)?.display_name); const features = useFeatures(); const [account, setAccount] = useState(''); const onAccountChange: React.ChangeEventHandler<HTMLInputElement> = e => { setAccount(e.target.value); }; const onClickClose = () => { onClose('UNAUTHORIZED'); }; const onSubmit: React.FormEventHandler = e => { e.preventDefault(); dispatch(remoteInteraction(apId!, account)) .then(url => { window.open(url, '_new', 'noopener,noreferrer'); onClose('UNAUTHORIZED'); }) .catch(error => { if (error.message === 'Couldn\'t find user') { toast.error(intl.formatMessage(messages.userNotFoundError)); } }); }; const onLogin = () => { history.push('/login'); onClickClose(); }; const onRegister = () => { history.push('/signup'); onClickClose(); }; const renderRemoteInteractions = () => { let header; let button; if (action === 'FOLLOW') { header = <FormattedMessage id='remote_interaction.follow_title' defaultMessage='Follow {user} remotely' values={{ user: username }} />; button = <FormattedMessage id='remote_interaction.follow' defaultMessage='Proceed to follow' />; } else if (action === 'REPLY') { header = <FormattedMessage id='remote_interaction.reply_title' defaultMessage='Reply to a post remotely' />; button = <FormattedMessage id='remote_interaction.reply' defaultMessage='Proceed to reply' />; } else if (action === 'REBLOG') { header = <FormattedMessage id='remote_interaction.reblog_title' defaultMessage='Reblog a post remotely' />; button = <FormattedMessage id='remote_interaction.reblog' defaultMessage='Proceed to repost' />; } else if (action === 'FAVOURITE') { header = <FormattedMessage id='remote_interaction.favourite_title' defaultMessage='Like a post remotely' />; button = <FormattedMessage id='remote_interaction.favourite' defaultMessage='Proceed to like' />; } else if (action === 'DISLIKE') { header = <FormattedMessage id='remote_interaction.dislike_title' defaultMessage='Dislike a post remotely' />; button = <FormattedMessage id='remote_interaction.dislike' defaultMessage='Proceed to dislike' />; } else if (action === 'POLL_VOTE') { header = <FormattedMessage id='remote_interaction.poll_vote_title' defaultMessage='Vote in a poll remotely' />; button = <FormattedMessage id='remote_interaction.poll_vote' defaultMessage='Proceed to vote' />; } else if (action === 'JOIN') { header = <FormattedMessage id='remote_interaction.event_join_title' defaultMessage='Join an event remotely' />; button = <FormattedMessage id='remote_interaction.event_join' defaultMessage='Proceed to join' />; } return ( <Modal title={header} onClose={onClickClose} confirmationAction={onLogin} confirmationText={<FormattedMessage id='account.login' defaultMessage='Log in' />} secondaryAction={isOpen ? onRegister : undefined} secondaryText={isOpen ? <FormattedMessage id='account.register' defaultMessage='Sign up' /> : undefined} > <div className='remote-interaction-modal__content'> <Form className='remote-interaction-modal__fields' onSubmit={onSubmit}> <Input placeholder={intl.formatMessage(messages.accountPlaceholder)} name='remote_follow[acct]' value={account} autoCorrect='off' autoCapitalize='off' onChange={onAccountChange} required /> <Button type='submit' theme='primary'>{button}</Button> </Form> <div className='remote-interaction-modal__divider'> <Text align='center'> <FormattedMessage id='remote_interaction.divider' defaultMessage='or' /> </Text> </div> {isOpen && ( <Text size='lg' weight='medium'> <FormattedMessage id='unauthorized_modal.title' defaultMessage='Sign up for {site_title}' values={{ site_title: instance.title }} /> </Text> )} </div> </Modal> ); }; if (action && features.remoteInteractions && features.federating) { return renderRemoteInteractions(); } return ( <Modal title={<FormattedMessage id='unauthorized_modal.title' defaultMessage='Sign up for {site_title}' values={{ site_title: instance.title }} />} onClose={onClickClose} confirmationAction={onLogin} confirmationText={<FormattedMessage id='account.login' defaultMessage='Log in' />} secondaryAction={isOpen ? onRegister : undefined} secondaryText={isOpen ? <FormattedMessage id='account.register' defaultMessage='Sign up' /> : undefined} > <Stack> <Text> <FormattedMessage id='unauthorized_modal.text' defaultMessage='You need to be logged in to do that.' /> </Text> </Stack> </Modal> ); }; export default UnauthorizedModal;