pleroma/app/soapbox/features/ui/components/modals/account-moderation-modal/account-moderation-modal.tsx

190 lines
6.8 KiB
TypeScript
Raw Normal View History

import React, { ChangeEventHandler, useState } from 'react';
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
2022-09-11 09:25:48 -07:00
import {
verifyUser,
unverifyUser,
suggestUsers,
unsuggestUsers,
setBadges as saveBadges,
} from 'soapbox/actions/admin';
import { deactivateUserModal, deleteUserModal } from 'soapbox/actions/moderation';
import { useAccount } from 'soapbox/api/hooks';
import Account from 'soapbox/components/account';
import List, { ListItem } from 'soapbox/components/list';
2022-11-15 08:00:49 -08:00
import MissingIndicator from 'soapbox/components/missing-indicator';
import OutlineBox from 'soapbox/components/outline-box';
2022-09-11 12:20:05 -07:00
import { Button, Text, HStack, Modal, Stack, Toggle } from 'soapbox/components/ui';
import { useAppDispatch, useFeatures, useOwnAccount } from 'soapbox/hooks';
2022-12-20 07:47:46 -08:00
import toast from 'soapbox/toast';
import { isLocal } from 'soapbox/utils/accounts';
import { getBadges } from 'soapbox/utils/badges';
2022-09-11 09:25:48 -07:00
import BadgeInput from './badge-input';
import StaffRolePicker from './staff-role-picker';
const messages = defineMessages({
userVerified: { id: 'admin.users.user_verified_message', defaultMessage: '@{acct} was verified' },
userUnverified: { id: 'admin.users.user_unverified_message', defaultMessage: '@{acct} was unverified' },
setDonorSuccess: { id: 'admin.users.set_donor_message', defaultMessage: '@{acct} was set as a donor' },
removeDonorSuccess: { id: 'admin.users.remove_donor_message', defaultMessage: '@{acct} was removed as a donor' },
userSuggested: { id: 'admin.users.user_suggested_message', defaultMessage: '@{acct} was suggested' },
userUnsuggested: { id: 'admin.users.user_unsuggested_message', defaultMessage: '@{acct} was unsuggested' },
badgesSaved: { id: 'admin.users.badges_saved_message', defaultMessage: 'Custom badges updated.' },
});
2022-09-11 09:25:48 -07:00
interface IAccountModerationModal {
/** Action to close the modal. */
onClose: (type: string) => void
2022-09-11 09:25:48 -07:00
/** ID of the account to moderate. */
accountId: string
2022-09-11 09:25:48 -07:00
}
/** Moderator actions against accounts. */
const AccountModerationModal: React.FC<IAccountModerationModal> = ({ onClose, accountId }) => {
const intl = useIntl();
const dispatch = useAppDispatch();
2023-06-25 10:35:09 -07:00
const { account: ownAccount } = useOwnAccount();
const features = useFeatures();
const { account } = useAccount(accountId);
2022-09-11 09:25:48 -07:00
const accountBadges = account ? getBadges(account) : [];
const [badges, setBadges] = useState<string[]>(accountBadges);
2022-09-11 09:25:48 -07:00
const handleClose = () => onClose('ACCOUNT_MODERATION');
2022-09-11 12:46:01 -07:00
if (!account || !ownAccount) {
2022-09-11 09:25:48 -07:00
return (
<Modal onClose={handleClose}>
<MissingIndicator />
</Modal>
);
}
const handleAdminFE = () => {
window.open(`/pleroma/admin/#/users/${account.id}/`, '_blank');
};
const handleVerifiedChange: ChangeEventHandler<HTMLInputElement> = (e) => {
const { checked } = e.target;
const message = checked ? messages.userVerified : messages.userUnverified;
const action = checked ? verifyUser : unverifyUser;
dispatch(action(account.id))
2022-12-20 07:47:46 -08:00
.then(() => toast.success(intl.formatMessage(message, { acct: account.acct })))
.catch(() => {});
};
const handleSuggestedChange: ChangeEventHandler<HTMLInputElement> = (e) => {
const { checked } = e.target;
const message = checked ? messages.userSuggested : messages.userUnsuggested;
const action = checked ? suggestUsers : unsuggestUsers;
dispatch(action([account.id]))
2022-12-20 07:47:46 -08:00
.then(() => toast.success(intl.formatMessage(message, { acct: account.acct })))
.catch(() => {});
};
const handleDeactivate = () => {
dispatch(deactivateUserModal(intl, account.id));
};
const handleDelete = () => {
dispatch(deleteUserModal(intl, account.id));
};
const handleSaveBadges = () => {
dispatch(saveBadges(account.id, accountBadges, badges))
2022-12-20 07:47:46 -08:00
.then(() => toast.success(intl.formatMessage(messages.badgesSaved)))
.catch(() => {});
};
2022-09-11 09:25:48 -07:00
return (
<Modal
title={<FormattedMessage id='account_moderation_modal.title' defaultMessage='Moderate @{acct}' values={{ acct: account.acct }} />}
2022-09-11 09:25:48 -07:00
onClose={handleClose}
>
<Stack space={4}>
<OutlineBox>
<Account
account={account}
showProfileHoverCard={false}
withLinkToProfile={false}
hideActions
/>
</OutlineBox>
<List>
2022-09-11 12:46:01 -07:00
{(ownAccount.admin && isLocal(account)) && (
<ListItem label={<FormattedMessage id='account_moderation_modal.fields.account_role' defaultMessage='Staff level' />}>
<div className='w-auto'>
<StaffRolePicker account={account} />
</div>
</ListItem>
)}
<ListItem label={<FormattedMessage id='account_moderation_modal.fields.verified' defaultMessage='Verified account' />}>
<Toggle
checked={account.verified}
onChange={handleVerifiedChange}
/>
</ListItem>
{features.suggestionsV2 && (
<ListItem label={<FormattedMessage id='account_moderation_modal.fields.suggested' defaultMessage='Suggested in people to follow' />}>
<Toggle
checked={account.pleroma?.is_suggested === true}
onChange={handleSuggestedChange}
/>
</ListItem>
)}
<ListItem label={<FormattedMessage id='account_moderation_modal.fields.badges' defaultMessage='Custom badges' />}>
2023-02-01 14:13:42 -08:00
<div className='grow'>
<HStack className='w-full' alignItems='center' space={2}>
<BadgeInput badges={badges} onChange={setBadges} />
<Button onClick={handleSaveBadges}>
<FormattedMessage id='save' defaultMessage='Save' />
</Button>
</HStack>
</div>
</ListItem>
</List>
<List>
<ListItem
label={<FormattedMessage id='account_moderation_modal.fields.deactivate' defaultMessage='Deactivate account' />}
onClick={handleDeactivate}
/>
<ListItem
label={<FormattedMessage id='account_moderation_modal.fields.delete' defaultMessage='Delete account' />}
onClick={handleDelete}
/>
</List>
2022-09-11 12:20:05 -07:00
<Text theme='subtle' size='xs'>
<FormattedMessage
id='account_moderation_modal.info.id'
defaultMessage='ID: {id}'
values={{ id: account.id }}
/>
</Text>
{features.adminFE && (
<HStack justifyContent='center'>
<Button icon={require('@tabler/icons/external-link.svg')} size='sm' theme='secondary' onClick={handleAdminFE}>
<FormattedMessage id='account_moderation_modal.admin_fe' defaultMessage='Open in AdminFE' />
</Button>
</HStack>
)}
</Stack>
2022-09-11 09:25:48 -07:00
</Modal>
);
};
export default AccountModerationModal;