2022-05-03 06:27:15 -07:00
|
|
|
import React from 'react';
|
|
|
|
import { defineMessages, useIntl } from 'react-intl';
|
|
|
|
|
|
|
|
import {
|
|
|
|
blockAccount,
|
|
|
|
unblockAccount,
|
|
|
|
muteAccount,
|
|
|
|
unmuteAccount,
|
2022-07-30 14:31:38 -07:00
|
|
|
authorizeFollowRequest,
|
|
|
|
rejectFollowRequest,
|
2022-05-03 06:27:15 -07:00
|
|
|
} from 'soapbox/actions/accounts';
|
|
|
|
import { openModal } from 'soapbox/actions/modals';
|
2023-06-22 21:38:50 -07:00
|
|
|
import { useFollow } from 'soapbox/api/hooks';
|
2022-07-30 14:31:38 -07:00
|
|
|
import { Button, HStack } from 'soapbox/components/ui';
|
2023-06-22 21:38:50 -07:00
|
|
|
import { useAppDispatch, useFeatures, useLoggedIn } from 'soapbox/hooks';
|
2022-05-03 06:27:15 -07:00
|
|
|
|
2023-03-15 11:53:17 -07:00
|
|
|
import type { Account } from 'soapbox/schemas';
|
2022-05-03 06:27:15 -07:00
|
|
|
|
|
|
|
const messages = defineMessages({
|
2022-05-03 10:22:06 -07:00
|
|
|
block: { id: 'account.block', defaultMessage: 'Block @{name}' },
|
|
|
|
blocked: { id: 'account.blocked', defaultMessage: 'Blocked' },
|
|
|
|
edit_profile: { id: 'account.edit_profile', defaultMessage: 'Edit profile' },
|
2022-05-03 06:27:15 -07:00
|
|
|
follow: { id: 'account.follow', defaultMessage: 'Follow' },
|
2022-05-03 10:22:06 -07:00
|
|
|
mute: { id: 'account.mute', defaultMessage: 'Mute @{name}' },
|
2022-05-03 06:27:15 -07:00
|
|
|
remote_follow: { id: 'account.remote_follow', defaultMessage: 'Remote follow' },
|
|
|
|
requested: { id: 'account.requested', defaultMessage: 'Awaiting approval. Click to cancel follow request' },
|
|
|
|
requested_small: { id: 'account.requested_small', defaultMessage: 'Awaiting approval' },
|
|
|
|
unblock: { id: 'account.unblock', defaultMessage: 'Unblock @{name}' },
|
2022-05-03 10:22:06 -07:00
|
|
|
unfollow: { id: 'account.unfollow', defaultMessage: 'Unfollow' },
|
2022-05-03 06:27:15 -07:00
|
|
|
unmute: { id: 'account.unmute', defaultMessage: 'Unmute @{name}' },
|
2022-07-30 14:31:38 -07:00
|
|
|
authorize: { id: 'follow_request.authorize', defaultMessage: 'Authorize' },
|
|
|
|
reject: { id: 'follow_request.reject', defaultMessage: 'Reject' },
|
2022-05-03 06:27:15 -07:00
|
|
|
});
|
|
|
|
|
2022-05-14 09:44:31 -07:00
|
|
|
interface IActionButton {
|
2022-05-14 09:48:57 -07:00
|
|
|
/** Target account for the action. */
|
2023-06-23 19:41:36 -07:00
|
|
|
account: Account
|
2022-05-14 09:48:57 -07:00
|
|
|
/** Type of action to prioritize, eg on Blocks and Mutes pages. */
|
2022-07-30 14:31:38 -07:00
|
|
|
actionType?: 'muting' | 'blocking' | 'follow_request'
|
2022-05-14 09:48:57 -07:00
|
|
|
/** Displays shorter text on the "Awaiting approval" button. */
|
2022-05-03 06:27:15 -07:00
|
|
|
small?: boolean
|
|
|
|
}
|
|
|
|
|
2022-05-14 09:48:57 -07:00
|
|
|
/**
|
|
|
|
* Circumstantial action button (usually "Follow") to display on accounts.
|
|
|
|
* May say "Unblock" or something else, depending on the relationship and
|
|
|
|
* `actionType` prop.
|
|
|
|
*/
|
2022-05-14 09:44:31 -07:00
|
|
|
const ActionButton: React.FC<IActionButton> = ({ account, actionType, small }) => {
|
2023-01-09 14:13:12 -08:00
|
|
|
const dispatch = useAppDispatch();
|
2022-05-03 06:27:15 -07:00
|
|
|
const features = useFeatures();
|
|
|
|
const intl = useIntl();
|
|
|
|
|
2023-06-22 21:38:50 -07:00
|
|
|
const { isLoggedIn, me } = useLoggedIn();
|
|
|
|
const { follow, unfollow } = useFollow();
|
2022-05-03 06:27:15 -07:00
|
|
|
|
|
|
|
const handleFollow = () => {
|
2022-06-04 00:22:36 -07:00
|
|
|
if (account.relationship?.following || account.relationship?.requested) {
|
2023-06-22 21:38:50 -07:00
|
|
|
unfollow(account.id);
|
2022-05-03 06:27:15 -07:00
|
|
|
} else {
|
2023-06-22 21:38:50 -07:00
|
|
|
follow(account.id);
|
2022-05-03 06:27:15 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleBlock = () => {
|
2022-06-04 00:22:36 -07:00
|
|
|
if (account.relationship?.blocking) {
|
2022-05-14 09:44:31 -07:00
|
|
|
dispatch(unblockAccount(account.id));
|
2022-05-03 06:27:15 -07:00
|
|
|
} else {
|
2022-05-14 09:44:31 -07:00
|
|
|
dispatch(blockAccount(account.id));
|
2022-05-03 06:27:15 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleMute = () => {
|
2022-06-04 00:22:36 -07:00
|
|
|
if (account.relationship?.muting) {
|
2022-05-14 09:44:31 -07:00
|
|
|
dispatch(unmuteAccount(account.id));
|
2022-05-03 06:27:15 -07:00
|
|
|
} else {
|
2022-05-14 09:44:31 -07:00
|
|
|
dispatch(muteAccount(account.id));
|
2022-05-03 06:27:15 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-07-30 14:31:38 -07:00
|
|
|
const handleAuthorize = () => {
|
|
|
|
dispatch(authorizeFollowRequest(account.id));
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleReject = () => {
|
|
|
|
dispatch(rejectFollowRequest(account.id));
|
|
|
|
};
|
|
|
|
|
2022-05-03 06:27:15 -07:00
|
|
|
const handleRemoteFollow = () => {
|
|
|
|
dispatch(openModal('UNAUTHORIZED', {
|
|
|
|
action: 'FOLLOW',
|
2022-05-14 09:44:31 -07:00
|
|
|
account: account.id,
|
|
|
|
ap_id: account.url,
|
2022-05-03 06:27:15 -07:00
|
|
|
}));
|
|
|
|
};
|
|
|
|
|
2022-05-14 09:48:57 -07:00
|
|
|
/** Handles actionType='muting' */
|
2022-05-03 10:22:06 -07:00
|
|
|
const mutingAction = () => {
|
2022-06-04 00:22:36 -07:00
|
|
|
const isMuted = account.relationship?.muting;
|
2022-05-03 10:22:06 -07:00
|
|
|
const messageKey = isMuted ? messages.unmute : messages.mute;
|
2022-05-14 09:44:31 -07:00
|
|
|
const text = intl.formatMessage(messageKey, { name: account.username });
|
2022-05-03 10:22:06 -07:00
|
|
|
|
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
theme={isMuted ? 'danger' : 'secondary'}
|
|
|
|
size='sm'
|
|
|
|
text={text}
|
|
|
|
onClick={handleMute}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-05-14 09:48:57 -07:00
|
|
|
/** Handles actionType='blocking' */
|
2022-05-03 10:22:06 -07:00
|
|
|
const blockingAction = () => {
|
2022-06-04 00:22:36 -07:00
|
|
|
const isBlocked = account.relationship?.blocking;
|
2022-05-03 10:22:06 -07:00
|
|
|
const messageKey = isBlocked ? messages.unblock : messages.block;
|
2022-05-14 09:44:31 -07:00
|
|
|
const text = intl.formatMessage(messageKey, { name: account.username });
|
2022-05-03 10:22:06 -07:00
|
|
|
|
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
theme={isBlocked ? 'danger' : 'secondary'}
|
|
|
|
size='sm'
|
|
|
|
text={text}
|
|
|
|
onClick={handleBlock}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-07-30 14:31:38 -07:00
|
|
|
const followRequestAction = () => {
|
|
|
|
if (account.relationship?.followed_by) return null;
|
|
|
|
|
|
|
|
return (
|
|
|
|
<HStack space={2}>
|
|
|
|
<Button
|
|
|
|
theme='secondary'
|
|
|
|
size='sm'
|
|
|
|
text={intl.formatMessage(messages.authorize)}
|
|
|
|
onClick={handleAuthorize}
|
|
|
|
/>
|
|
|
|
<Button
|
|
|
|
theme='danger'
|
|
|
|
size='sm'
|
|
|
|
text={intl.formatMessage(messages.reject)}
|
|
|
|
onClick={handleReject}
|
|
|
|
/>
|
|
|
|
</HStack>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-05-14 09:21:12 -07:00
|
|
|
/** Render a remote follow button, depending on features. */
|
2022-05-14 09:48:57 -07:00
|
|
|
const renderRemoteFollow = () => {
|
2022-05-14 09:21:12 -07:00
|
|
|
// Remote follow through the API.
|
2022-06-04 23:45:51 -07:00
|
|
|
if (features.remoteInteractions) {
|
2022-05-03 06:27:15 -07:00
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
onClick={handleRemoteFollow}
|
2022-07-09 09:20:02 -07:00
|
|
|
icon={require('@tabler/icons/plus.svg')}
|
2022-05-03 06:27:15 -07:00
|
|
|
text={intl.formatMessage(messages.follow)}
|
2022-11-21 12:03:35 -08:00
|
|
|
size='sm'
|
2022-05-03 06:27:15 -07:00
|
|
|
/>
|
|
|
|
);
|
2022-07-22 10:30:16 -07:00
|
|
|
// Pleroma's classic remote follow form.
|
2022-05-14 09:21:12 -07:00
|
|
|
} else if (features.pleromaRemoteFollow) {
|
|
|
|
return (
|
|
|
|
<form method='POST' action='/main/ostatus'>
|
|
|
|
<input type='hidden' name='nickname' value={account.acct} />
|
|
|
|
<input type='hidden' name='profile' value='' />
|
2022-11-21 12:03:35 -08:00
|
|
|
<Button
|
|
|
|
text={intl.formatMessage(messages.remote_follow)}
|
|
|
|
type='submit'
|
|
|
|
size='sm'
|
|
|
|
/>
|
2022-05-14 09:21:12 -07:00
|
|
|
</form>
|
|
|
|
);
|
2022-05-03 06:27:15 -07:00
|
|
|
}
|
|
|
|
|
2022-05-14 09:21:12 -07:00
|
|
|
return null;
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Render remote follow if federating, otherwise hide the button. */
|
2022-05-14 09:48:57 -07:00
|
|
|
const renderLoggedOut = () => {
|
2022-05-14 09:21:12 -07:00
|
|
|
if (features.federating) {
|
|
|
|
return renderRemoteFollow();
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
};
|
|
|
|
|
2023-06-22 21:38:50 -07:00
|
|
|
if (!isLoggedIn) {
|
2022-05-14 09:21:12 -07:00
|
|
|
return renderLoggedOut();
|
2022-05-03 06:27:15 -07:00
|
|
|
}
|
|
|
|
|
2022-05-14 09:21:12 -07:00
|
|
|
if (me !== account.id) {
|
2022-06-04 00:22:36 -07:00
|
|
|
const isFollowing = account.relationship?.following;
|
|
|
|
const blockedBy = account.relationship?.blocked_by as boolean;
|
2022-05-03 06:27:15 -07:00
|
|
|
|
2022-05-03 10:22:06 -07:00
|
|
|
if (actionType) {
|
|
|
|
if (actionType === 'muting') {
|
|
|
|
return mutingAction();
|
|
|
|
} else if (actionType === 'blocking') {
|
|
|
|
return blockingAction();
|
2022-07-30 14:31:38 -07:00
|
|
|
} else if (actionType === 'follow_request') {
|
|
|
|
return followRequestAction();
|
2022-05-03 10:22:06 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-04 00:22:36 -07:00
|
|
|
if (!account.relationship) {
|
2022-05-03 06:27:15 -07:00
|
|
|
// Wait until the relationship is loaded
|
2022-05-14 09:49:28 -07:00
|
|
|
return null;
|
2022-06-04 00:22:36 -07:00
|
|
|
} else if (account.relationship?.requested) {
|
2022-05-03 06:27:15 -07:00
|
|
|
// Awaiting acceptance
|
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
size='sm'
|
2022-07-22 10:30:16 -07:00
|
|
|
theme='tertiary'
|
2022-05-03 06:27:15 -07:00
|
|
|
text={small ? intl.formatMessage(messages.requested_small) : intl.formatMessage(messages.requested)}
|
|
|
|
onClick={handleFollow}
|
|
|
|
/>
|
|
|
|
);
|
2022-06-04 00:22:36 -07:00
|
|
|
} else if (!account.relationship?.blocking && !account.relationship?.muting) {
|
2022-05-03 06:27:15 -07:00
|
|
|
// Follow & Unfollow
|
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
size='sm'
|
|
|
|
disabled={blockedBy}
|
|
|
|
theme={isFollowing ? 'secondary' : 'primary'}
|
2022-07-09 09:20:02 -07:00
|
|
|
icon={blockedBy ? require('@tabler/icons/ban.svg') : (!isFollowing && require('@tabler/icons/plus.svg'))}
|
2022-05-03 06:27:15 -07:00
|
|
|
onClick={handleFollow}
|
|
|
|
>
|
|
|
|
{isFollowing ? (
|
|
|
|
intl.formatMessage(messages.unfollow)
|
|
|
|
) : (
|
|
|
|
intl.formatMessage(blockedBy ? messages.blocked : messages.follow)
|
|
|
|
)}
|
|
|
|
</Button>
|
|
|
|
);
|
2022-06-04 00:22:36 -07:00
|
|
|
} else if (account.relationship?.blocking) {
|
2022-05-03 06:27:15 -07:00
|
|
|
// Unblock
|
|
|
|
return (
|
|
|
|
<Button
|
|
|
|
theme='danger'
|
|
|
|
size='sm'
|
2022-05-14 09:44:31 -07:00
|
|
|
text={intl.formatMessage(messages.unblock, { name: account.username })}
|
2022-05-03 06:27:15 -07:00
|
|
|
onClick={handleBlock}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// Edit profile
|
|
|
|
return (
|
|
|
|
<Button
|
2022-07-22 10:30:16 -07:00
|
|
|
theme='tertiary'
|
2022-05-03 06:27:15 -07:00
|
|
|
size='sm'
|
|
|
|
text={intl.formatMessage(messages.edit_profile)}
|
|
|
|
to='/settings/profile'
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-05-14 09:49:28 -07:00
|
|
|
return null;
|
2022-05-03 06:27:15 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
export default ActionButton;
|