2022-09-11 11:50:03 -07:00
|
|
|
import React, { ChangeEventHandler } from 'react';
|
|
|
|
import { defineMessages, FormattedMessage, useIntl } from 'react-intl';
|
2022-09-11 09:25:48 -07:00
|
|
|
|
2022-09-11 11:58:05 -07:00
|
|
|
import {
|
|
|
|
verifyUser,
|
|
|
|
unverifyUser,
|
|
|
|
setDonor,
|
|
|
|
removeDonor,
|
2022-09-11 12:09:09 -07:00
|
|
|
suggestUsers,
|
|
|
|
unsuggestUsers,
|
2022-09-11 11:58:05 -07:00
|
|
|
} from 'soapbox/actions/admin';
|
2022-09-11 11:50:03 -07:00
|
|
|
import snackbar from 'soapbox/actions/snackbar';
|
2022-09-11 10:33:33 -07:00
|
|
|
import Account from 'soapbox/components/account';
|
2022-09-11 11:19:08 -07:00
|
|
|
import List, { ListItem } from 'soapbox/components/list';
|
2022-09-11 09:25:48 -07:00
|
|
|
import MissingIndicator from 'soapbox/components/missing_indicator';
|
2022-09-11 11:50:03 -07:00
|
|
|
import { Button, HStack, Modal, Stack, Toggle } from 'soapbox/components/ui';
|
|
|
|
import { useAppDispatch, useAppSelector, useFeatures } from 'soapbox/hooks';
|
2022-09-11 09:25:48 -07:00
|
|
|
import { makeGetAccount } from 'soapbox/selectors';
|
2022-09-11 11:22:10 -07:00
|
|
|
import { isLocal } from 'soapbox/utils/accounts';
|
2022-09-11 09:25:48 -07:00
|
|
|
|
2022-09-11 11:36:01 -07:00
|
|
|
import StaffRolePicker from './staff-role-picker';
|
2022-09-11 11:19:08 -07:00
|
|
|
|
2022-09-11 09:25:48 -07:00
|
|
|
const getAccount = makeGetAccount();
|
|
|
|
|
2022-09-11 11:50:03 -07:00
|
|
|
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' },
|
2022-09-11 11:58:05 -07:00
|
|
|
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' },
|
2022-09-11 12:09:09 -07:00
|
|
|
userSuggested: { id: 'admin.users.user_suggested_message', defaultMessage: '@{acct} was suggested' },
|
|
|
|
userUnsuggested: { id: 'admin.users.user_unsuggested_message', defaultMessage: '@{acct} was unsuggested' },
|
2022-09-11 11:50:03 -07:00
|
|
|
});
|
|
|
|
|
2022-09-11 09:25:48 -07:00
|
|
|
interface IAccountModerationModal {
|
|
|
|
/** Action to close the modal. */
|
|
|
|
onClose: (type: string) => void,
|
|
|
|
/** ID of the account to moderate. */
|
|
|
|
accountId: string,
|
|
|
|
}
|
|
|
|
|
|
|
|
/** Moderator actions against accounts. */
|
|
|
|
const AccountModerationModal: React.FC<IAccountModerationModal> = ({ onClose, accountId }) => {
|
2022-09-11 11:50:03 -07:00
|
|
|
const intl = useIntl();
|
|
|
|
const dispatch = useAppDispatch();
|
|
|
|
|
2022-09-11 09:48:37 -07:00
|
|
|
const features = useFeatures();
|
2022-09-11 09:25:48 -07:00
|
|
|
const account = useAppSelector(state => getAccount(state, accountId));
|
|
|
|
|
|
|
|
const handleClose = () => onClose('ACCOUNT_MODERATION');
|
|
|
|
|
|
|
|
if (!account) {
|
|
|
|
return (
|
|
|
|
<Modal onClose={handleClose}>
|
|
|
|
<MissingIndicator />
|
|
|
|
</Modal>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-09-11 09:46:08 -07:00
|
|
|
const handleAdminFE = () => {
|
|
|
|
window.open(`/pleroma/admin/#/users/${account.id}/`, '_blank');
|
|
|
|
};
|
|
|
|
|
2022-09-11 11:50:03 -07:00
|
|
|
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))
|
|
|
|
.then(() => dispatch(snackbar.success(intl.formatMessage(message, { acct: account.acct }))))
|
|
|
|
.catch(() => {});
|
|
|
|
};
|
|
|
|
|
2022-09-11 11:58:05 -07:00
|
|
|
const handleDonorChange: ChangeEventHandler<HTMLInputElement> = (e) => {
|
|
|
|
const { checked } = e.target;
|
|
|
|
|
|
|
|
const message = checked ? messages.setDonorSuccess : messages.removeDonorSuccess;
|
|
|
|
const action = checked ? setDonor : removeDonor;
|
|
|
|
|
|
|
|
dispatch(action(account.id))
|
|
|
|
.then(() => dispatch(snackbar.success(intl.formatMessage(message, { acct: account.acct }))))
|
|
|
|
.catch(() => {});
|
|
|
|
};
|
|
|
|
|
2022-09-11 12:09:09 -07:00
|
|
|
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]))
|
|
|
|
.then(() => dispatch(snackbar.success(intl.formatMessage(message, { acct: account.acct }))))
|
|
|
|
.catch(() => {});
|
|
|
|
};
|
|
|
|
|
2022-09-11 09:25:48 -07:00
|
|
|
return (
|
|
|
|
<Modal
|
2022-09-11 09:46:08 -07:00
|
|
|
title={<FormattedMessage id='account_moderation_modal.title' defaultMessage='Moderate @{acct}' values={{ acct: account.acct }} />}
|
2022-09-11 09:25:48 -07:00
|
|
|
onClose={handleClose}
|
|
|
|
>
|
2022-09-11 10:33:33 -07:00
|
|
|
<Stack space={4}>
|
|
|
|
<div className='p-4 rounded-lg border border-solid border-gray-300 dark:border-gray-800'>
|
|
|
|
<Account
|
|
|
|
account={account}
|
|
|
|
showProfileHoverCard={false}
|
|
|
|
withLinkToProfile={false}
|
|
|
|
hideActions
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
|
2022-09-11 11:50:03 -07:00
|
|
|
<List>
|
|
|
|
{isLocal(account) && (
|
2022-09-11 11:22:10 -07:00
|
|
|
<ListItem label={<FormattedMessage id='account_moderation_modal.fields.account_role' defaultMessage='Staff level' />}>
|
2022-09-11 11:36:01 -07:00
|
|
|
<div className='w-auto'>
|
|
|
|
<StaffRolePicker account={account} />
|
|
|
|
</div>
|
2022-09-11 11:22:10 -07:00
|
|
|
</ListItem>
|
2022-09-11 11:50:03 -07:00
|
|
|
)}
|
|
|
|
|
|
|
|
<ListItem label={<FormattedMessage id='account_moderation_modal.fields.verified' defaultMessage='Verified account' />}>
|
|
|
|
<Toggle
|
|
|
|
checked={account.verified}
|
|
|
|
onChange={handleVerifiedChange}
|
|
|
|
/>
|
|
|
|
</ListItem>
|
2022-09-11 11:58:05 -07:00
|
|
|
|
|
|
|
<ListItem label={<FormattedMessage id='account_moderation_modal.fields.donor' defaultMessage='Donor' />}>
|
|
|
|
<Toggle
|
|
|
|
checked={account.donor}
|
|
|
|
onChange={handleDonorChange}
|
|
|
|
/>
|
|
|
|
</ListItem>
|
2022-09-11 12:09:09 -07:00
|
|
|
|
|
|
|
{features.suggestionsV2 && (
|
|
|
|
<ListItem label={<FormattedMessage id='account_moderation_modal.fields.suggested' defaultMessage='Suggested in people to follow' />}>
|
|
|
|
<Toggle
|
|
|
|
checked={account.getIn(['pleroma', 'is_suggested']) === true}
|
|
|
|
onChange={handleSuggestedChange}
|
|
|
|
/>
|
|
|
|
</ListItem>
|
|
|
|
)}
|
2022-09-11 11:50:03 -07:00
|
|
|
</List>
|
2022-09-11 11:19:08 -07:00
|
|
|
|
2022-09-11 10:33:33 -07:00
|
|
|
{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;
|