2023-03-20 14:41:12 -07:00
|
|
|
import React from 'react';
|
2022-12-18 09:03:41 -08:00
|
|
|
import { FormattedMessage, defineMessages, useIntl } from 'react-intl';
|
|
|
|
|
|
|
|
import Account from 'soapbox/components/account';
|
2023-03-20 15:46:10 -07:00
|
|
|
import { AuthorizeRejectButtons } from 'soapbox/components/authorize-reject-buttons';
|
2022-12-18 09:03:41 -08:00
|
|
|
import ScrollableList from 'soapbox/components/scrollable-list';
|
2023-03-20 15:46:10 -07:00
|
|
|
import { Column, HStack, Spinner } from 'soapbox/components/ui';
|
|
|
|
import { useGroup } from 'soapbox/hooks';
|
2023-03-20 13:54:06 -07:00
|
|
|
import { useGroupMembershipRequests } from 'soapbox/hooks/api/groups/useGroupMembershipRequests';
|
2023-01-27 14:04:42 -08:00
|
|
|
import toast from 'soapbox/toast';
|
2022-12-18 09:03:41 -08:00
|
|
|
|
|
|
|
import ColumnForbidden from '../ui/components/column-forbidden';
|
|
|
|
|
2023-03-20 14:41:12 -07:00
|
|
|
import type { Account as AccountEntity } from 'soapbox/schemas';
|
|
|
|
|
2022-12-18 09:03:41 -08:00
|
|
|
type RouteParams = { id: string };
|
|
|
|
|
|
|
|
const messages = defineMessages({
|
|
|
|
heading: { id: 'column.group_pending_requests', defaultMessage: 'Pending requests' },
|
2023-03-20 15:46:10 -07:00
|
|
|
authorizeFail: { id: 'group.group_mod_authorize.fail', defaultMessage: 'Failed to approve @{name}' },
|
|
|
|
rejectFail: { id: 'group.group_mod_reject.fail', defaultMessage: 'Failed to reject @{name}' },
|
2022-12-18 09:03:41 -08:00
|
|
|
});
|
|
|
|
|
|
|
|
interface IMembershipRequest {
|
2023-03-20 14:41:12 -07:00
|
|
|
account: AccountEntity
|
2023-03-20 17:32:24 -07:00
|
|
|
onAuthorize(account: AccountEntity): Promise<unknown>
|
|
|
|
onReject(account: AccountEntity): Promise<unknown>
|
2022-12-18 09:03:41 -08:00
|
|
|
}
|
|
|
|
|
2023-03-20 15:46:10 -07:00
|
|
|
const MembershipRequest: React.FC<IMembershipRequest> = ({ account, onAuthorize, onReject }) => {
|
2022-12-18 09:03:41 -08:00
|
|
|
if (!account) return null;
|
|
|
|
|
2023-03-20 17:32:24 -07:00
|
|
|
const handleAuthorize = () => onAuthorize(account);
|
|
|
|
const handleReject = () => onReject(account);
|
2022-12-18 09:03:41 -08:00
|
|
|
|
|
|
|
return (
|
|
|
|
<HStack space={1} alignItems='center' justifyContent='between' className='p-2.5'>
|
|
|
|
<div className='w-full'>
|
|
|
|
<Account account={account} withRelationship={false} />
|
|
|
|
</div>
|
2023-03-20 15:46:10 -07:00
|
|
|
|
|
|
|
<AuthorizeRejectButtons
|
|
|
|
onAuthorize={handleAuthorize}
|
|
|
|
onReject={handleReject}
|
|
|
|
/>
|
2022-12-18 09:03:41 -08:00
|
|
|
</HStack>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
interface IGroupMembershipRequests {
|
|
|
|
params: RouteParams
|
|
|
|
}
|
|
|
|
|
|
|
|
const GroupMembershipRequests: React.FC<IGroupMembershipRequests> = ({ params }) => {
|
2023-03-15 17:26:37 -07:00
|
|
|
const id = params?.id;
|
2023-03-20 17:32:24 -07:00
|
|
|
const intl = useIntl();
|
2022-12-18 09:03:41 -08:00
|
|
|
|
2023-03-15 17:26:37 -07:00
|
|
|
const { group } = useGroup(id);
|
2023-03-20 15:46:10 -07:00
|
|
|
const { accounts, isLoading, authorize, reject } = useGroupMembershipRequests(id);
|
2022-12-18 09:03:41 -08:00
|
|
|
|
2023-03-20 14:41:12 -07:00
|
|
|
if (!group || !group.relationship || isLoading) {
|
2022-12-18 09:03:41 -08:00
|
|
|
return (
|
|
|
|
<Column label={intl.formatMessage(messages.heading)}>
|
|
|
|
<Spinner />
|
|
|
|
</Column>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2023-03-15 15:19:42 -07:00
|
|
|
if (!group.relationship.role || !['owner', 'admin', 'moderator'].includes(group.relationship.role)) {
|
2023-03-20 17:32:24 -07:00
|
|
|
return <ColumnForbidden />;
|
2022-12-18 09:03:41 -08:00
|
|
|
}
|
|
|
|
|
2023-03-20 17:32:24 -07:00
|
|
|
async function handleAuthorize(account: AccountEntity) {
|
|
|
|
try {
|
|
|
|
await authorize(account.id);
|
|
|
|
} catch (_e) {
|
|
|
|
toast.error(intl.formatMessage(messages.authorizeFail, { name: account.username }));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
async function handleReject(account: AccountEntity) {
|
|
|
|
try {
|
|
|
|
await reject(account.id);
|
|
|
|
} catch (_e) {
|
|
|
|
toast.error(intl.formatMessage(messages.rejectFail, { name: account.username }));
|
|
|
|
}
|
|
|
|
}
|
2022-12-18 09:03:41 -08:00
|
|
|
|
|
|
|
return (
|
|
|
|
<Column label={intl.formatMessage(messages.heading)} backHref={`/groups/${id}/manage`}>
|
|
|
|
<ScrollableList
|
|
|
|
scrollKey='group_membership_requests'
|
2023-03-20 15:46:10 -07:00
|
|
|
emptyMessage={<FormattedMessage id='empty_column.group_membership_requests' defaultMessage='There are no pending membership requests for this group.' />}
|
2022-12-18 09:03:41 -08:00
|
|
|
>
|
2023-03-20 15:46:10 -07:00
|
|
|
{accounts.map((account) => (
|
|
|
|
<MembershipRequest
|
|
|
|
key={account.id}
|
|
|
|
account={account}
|
|
|
|
onAuthorize={handleAuthorize}
|
|
|
|
onReject={handleReject}
|
|
|
|
/>
|
|
|
|
))}
|
2022-12-18 09:03:41 -08:00
|
|
|
</ScrollableList>
|
|
|
|
</Column>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
|
|
|
export default GroupMembershipRequests;
|