509 lines
15 KiB
TypeScript
509 lines
15 KiB
TypeScript
// Copyright 2021 Signal Messenger, LLC
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
import React from 'react';
|
|
import classNames from 'classnames';
|
|
import _ from 'lodash';
|
|
|
|
import type { ConversationType } from '../../../state/ducks/conversations';
|
|
import type { LocalizerType, ThemeType } from '../../../types/Util';
|
|
import type { PreferredBadgeSelectorType } from '../../../state/selectors/badges';
|
|
import type { UUIDStringType } from '../../../types/UUID';
|
|
import { Avatar } from '../../Avatar';
|
|
import { ConfirmationDialog } from '../../ConfirmationDialog';
|
|
import { PanelSection } from './PanelSection';
|
|
import { PanelRow } from './PanelRow';
|
|
import { ConversationDetailsIcon, IconType } from './ConversationDetailsIcon';
|
|
import { isAccessControlEnabled } from '../../../groups/util';
|
|
import { assertDev } from '../../../util/assert';
|
|
|
|
export type PropsType = {
|
|
readonly conversation?: ConversationType;
|
|
readonly getPreferredBadge: PreferredBadgeSelectorType;
|
|
readonly i18n: LocalizerType;
|
|
readonly ourUuid: UUIDStringType;
|
|
readonly pendingApprovalMemberships: ReadonlyArray<GroupV2RequestingMembership>;
|
|
readonly pendingMemberships: ReadonlyArray<GroupV2PendingMembership>;
|
|
readonly approvePendingMembership: (conversationId: string) => void;
|
|
readonly revokePendingMemberships: (conversationIds: Array<string>) => void;
|
|
readonly theme: ThemeType;
|
|
};
|
|
|
|
export type GroupV2PendingMembership = {
|
|
metadata: {
|
|
addedByUserId?: UUIDStringType;
|
|
};
|
|
member: ConversationType;
|
|
};
|
|
|
|
export type GroupV2RequestingMembership = {
|
|
member: ConversationType;
|
|
};
|
|
|
|
enum Tab {
|
|
Requests = 'Requests',
|
|
Pending = 'Pending',
|
|
}
|
|
|
|
enum StageType {
|
|
APPROVE_REQUEST = 'APPROVE_REQUEST',
|
|
DENY_REQUEST = 'DENY_REQUEST',
|
|
REVOKE_INVITE = 'REVOKE_INVITE',
|
|
}
|
|
|
|
type StagedMembershipType = {
|
|
type: StageType;
|
|
membership: GroupV2PendingMembership | GroupV2RequestingMembership;
|
|
};
|
|
|
|
export function PendingInvites({
|
|
approvePendingMembership,
|
|
conversation,
|
|
getPreferredBadge,
|
|
i18n,
|
|
ourUuid,
|
|
pendingMemberships,
|
|
pendingApprovalMemberships,
|
|
revokePendingMemberships,
|
|
theme,
|
|
}: PropsType): JSX.Element {
|
|
if (!conversation || !ourUuid) {
|
|
throw new Error(
|
|
'PendingInvites rendered without a conversation or ourUuid'
|
|
);
|
|
}
|
|
|
|
const [selectedTab, setSelectedTab] = React.useState(Tab.Requests);
|
|
const [stagedMemberships, setStagedMemberships] =
|
|
React.useState<Array<StagedMembershipType> | null>(null);
|
|
|
|
return (
|
|
<div className="conversation-details-panel">
|
|
<div className="ConversationDetails__tabs">
|
|
<div
|
|
className={classNames({
|
|
ConversationDetails__tab: true,
|
|
'ConversationDetails__tab--selected': selectedTab === Tab.Requests,
|
|
})}
|
|
onClick={() => {
|
|
setSelectedTab(Tab.Requests);
|
|
}}
|
|
onKeyUp={(e: React.KeyboardEvent) => {
|
|
if (e.target === e.currentTarget && e.keyCode === 13) {
|
|
setSelectedTab(Tab.Requests);
|
|
}
|
|
}}
|
|
role="tab"
|
|
tabIndex={0}
|
|
>
|
|
{i18n('PendingInvites--tab-requests', {
|
|
count: String(pendingApprovalMemberships.length),
|
|
})}
|
|
</div>
|
|
|
|
<div
|
|
className={classNames({
|
|
ConversationDetails__tab: true,
|
|
'ConversationDetails__tab--selected': selectedTab === Tab.Pending,
|
|
})}
|
|
onClick={() => {
|
|
setSelectedTab(Tab.Pending);
|
|
}}
|
|
onKeyUp={(e: React.KeyboardEvent) => {
|
|
if (e.target === e.currentTarget && e.keyCode === 13) {
|
|
setSelectedTab(Tab.Pending);
|
|
}
|
|
}}
|
|
role="tab"
|
|
tabIndex={0}
|
|
>
|
|
{i18n('PendingInvites--tab-invites', {
|
|
count: String(pendingMemberships.length),
|
|
})}
|
|
</div>
|
|
</div>
|
|
|
|
{selectedTab === Tab.Requests ? (
|
|
<MembersPendingAdminApproval
|
|
conversation={conversation}
|
|
getPreferredBadge={getPreferredBadge}
|
|
i18n={i18n}
|
|
memberships={pendingApprovalMemberships}
|
|
setStagedMemberships={setStagedMemberships}
|
|
theme={theme}
|
|
/>
|
|
) : null}
|
|
{selectedTab === Tab.Pending ? (
|
|
<MembersPendingProfileKey
|
|
conversation={conversation}
|
|
getPreferredBadge={getPreferredBadge}
|
|
i18n={i18n}
|
|
members={conversation.sortedGroupMembers || []}
|
|
memberships={pendingMemberships}
|
|
ourUuid={ourUuid}
|
|
setStagedMemberships={setStagedMemberships}
|
|
theme={theme}
|
|
/>
|
|
) : null}
|
|
|
|
{stagedMemberships && stagedMemberships.length && (
|
|
<MembershipActionConfirmation
|
|
approvePendingMembership={approvePendingMembership}
|
|
conversation={conversation}
|
|
i18n={i18n}
|
|
members={conversation.sortedGroupMembers || []}
|
|
onClose={() => setStagedMemberships(null)}
|
|
ourUuid={ourUuid}
|
|
revokePendingMemberships={revokePendingMemberships}
|
|
stagedMemberships={stagedMemberships}
|
|
/>
|
|
)}
|
|
</div>
|
|
);
|
|
}
|
|
|
|
function MembershipActionConfirmation({
|
|
approvePendingMembership,
|
|
conversation,
|
|
i18n,
|
|
members,
|
|
onClose,
|
|
ourUuid,
|
|
revokePendingMemberships,
|
|
stagedMemberships,
|
|
}: {
|
|
approvePendingMembership: (conversationId: string) => void;
|
|
conversation: ConversationType;
|
|
i18n: LocalizerType;
|
|
members: Array<ConversationType>;
|
|
onClose: () => void;
|
|
ourUuid: string;
|
|
revokePendingMemberships: (conversationIds: Array<string>) => void;
|
|
stagedMemberships: Array<StagedMembershipType>;
|
|
}) {
|
|
const revokeStagedMemberships = () => {
|
|
if (!stagedMemberships) {
|
|
return;
|
|
}
|
|
revokePendingMemberships(
|
|
stagedMemberships.map(({ membership }) => membership.member.id)
|
|
);
|
|
};
|
|
|
|
const approveStagedMembership = () => {
|
|
if (!stagedMemberships) {
|
|
return;
|
|
}
|
|
approvePendingMembership(stagedMemberships[0].membership.member.id);
|
|
};
|
|
|
|
const membershipType = stagedMemberships[0].type;
|
|
|
|
const modalAction =
|
|
membershipType === StageType.APPROVE_REQUEST
|
|
? approveStagedMembership
|
|
: revokeStagedMemberships;
|
|
|
|
let modalActionText = i18n('PendingInvites--revoke');
|
|
|
|
if (membershipType === StageType.APPROVE_REQUEST) {
|
|
modalActionText = i18n('PendingRequests--approve');
|
|
} else if (membershipType === StageType.DENY_REQUEST) {
|
|
modalActionText = i18n('PendingRequests--deny');
|
|
} else if (membershipType === StageType.REVOKE_INVITE) {
|
|
modalActionText = i18n('PendingInvites--revoke');
|
|
}
|
|
|
|
return (
|
|
<ConfirmationDialog
|
|
dialogName="PendingInvites.actionConfirmation"
|
|
actions={[
|
|
{
|
|
action: modalAction,
|
|
style: 'affirmative',
|
|
text: modalActionText,
|
|
},
|
|
]}
|
|
i18n={i18n}
|
|
onClose={onClose}
|
|
>
|
|
{getConfirmationMessage({
|
|
conversation,
|
|
i18n,
|
|
members,
|
|
ourUuid,
|
|
stagedMemberships,
|
|
})}
|
|
</ConfirmationDialog>
|
|
);
|
|
}
|
|
|
|
function getConfirmationMessage({
|
|
conversation,
|
|
i18n,
|
|
members,
|
|
ourUuid,
|
|
stagedMemberships,
|
|
}: Readonly<{
|
|
conversation: ConversationType;
|
|
i18n: LocalizerType;
|
|
members: ReadonlyArray<ConversationType>;
|
|
ourUuid: string;
|
|
stagedMemberships: ReadonlyArray<StagedMembershipType>;
|
|
}>): string {
|
|
if (!stagedMemberships || !stagedMemberships.length) {
|
|
return '';
|
|
}
|
|
|
|
const membershipType = stagedMemberships[0].type;
|
|
const firstMembership = stagedMemberships[0].membership;
|
|
|
|
// Requesting a membership since they weren't added by anyone
|
|
if (membershipType === StageType.DENY_REQUEST) {
|
|
const key = isAccessControlEnabled(
|
|
conversation.accessControlAddFromInviteLink
|
|
)
|
|
? 'PendingRequests--deny-for--with-link'
|
|
: 'PendingRequests--deny-for';
|
|
return i18n(key, {
|
|
name: firstMembership.member.title,
|
|
});
|
|
}
|
|
|
|
if (membershipType === StageType.APPROVE_REQUEST) {
|
|
return i18n('PendingRequests--approve-for', {
|
|
name: firstMembership.member.title,
|
|
});
|
|
}
|
|
|
|
if (membershipType !== StageType.REVOKE_INVITE) {
|
|
throw new Error('getConfirmationMessage: Invalid staging type');
|
|
}
|
|
|
|
const firstPendingMembership = firstMembership as GroupV2PendingMembership;
|
|
|
|
// Pending invite
|
|
const invitedByUs = firstPendingMembership.metadata.addedByUserId === ourUuid;
|
|
|
|
if (invitedByUs) {
|
|
return i18n('PendingInvites--revoke-for', {
|
|
name: firstPendingMembership.member.title,
|
|
});
|
|
}
|
|
|
|
const inviter = members.find(
|
|
({ uuid }) => uuid === firstPendingMembership.metadata.addedByUserId
|
|
);
|
|
|
|
if (inviter === undefined) {
|
|
return '';
|
|
}
|
|
|
|
const name = inviter.title;
|
|
|
|
if (stagedMemberships.length === 1) {
|
|
return i18n('PendingInvites--revoke-from-singular', { name });
|
|
}
|
|
|
|
return i18n('PendingInvites--revoke-from-plural', {
|
|
number: stagedMemberships.length.toString(),
|
|
name,
|
|
});
|
|
}
|
|
|
|
function MembersPendingAdminApproval({
|
|
conversation,
|
|
getPreferredBadge,
|
|
i18n,
|
|
memberships,
|
|
setStagedMemberships,
|
|
theme,
|
|
}: Readonly<{
|
|
conversation: ConversationType;
|
|
getPreferredBadge: PreferredBadgeSelectorType;
|
|
i18n: LocalizerType;
|
|
memberships: ReadonlyArray<GroupV2RequestingMembership>;
|
|
setStagedMemberships: (stagedMembership: Array<StagedMembershipType>) => void;
|
|
theme: ThemeType;
|
|
}>) {
|
|
return (
|
|
<PanelSection>
|
|
{memberships.map(membership => (
|
|
<PanelRow
|
|
alwaysShowActions
|
|
key={membership.member.id}
|
|
icon={
|
|
<Avatar
|
|
badge={getPreferredBadge(membership.member.badges)}
|
|
conversationType="direct"
|
|
size={32}
|
|
i18n={i18n}
|
|
theme={theme}
|
|
{...membership.member}
|
|
/>
|
|
}
|
|
label={membership.member.title}
|
|
actions={
|
|
conversation.areWeAdmin ? (
|
|
<>
|
|
<button
|
|
type="button"
|
|
className="module-button__small ConversationDetails__action-button"
|
|
onClick={() => {
|
|
setStagedMemberships([
|
|
{
|
|
type: StageType.DENY_REQUEST,
|
|
membership,
|
|
},
|
|
]);
|
|
}}
|
|
>
|
|
{i18n('delete')}
|
|
</button>
|
|
<button
|
|
type="button"
|
|
className="module-button__small ConversationDetails__action-button"
|
|
onClick={() => {
|
|
setStagedMemberships([
|
|
{
|
|
type: StageType.APPROVE_REQUEST,
|
|
membership,
|
|
},
|
|
]);
|
|
}}
|
|
>
|
|
{i18n('accept')}
|
|
</button>
|
|
</>
|
|
) : null
|
|
}
|
|
/>
|
|
))}
|
|
<div className="ConversationDetails__pending--info">
|
|
{i18n('PendingRequests--info', [conversation.title])}
|
|
</div>
|
|
</PanelSection>
|
|
);
|
|
}
|
|
|
|
function MembersPendingProfileKey({
|
|
conversation,
|
|
i18n,
|
|
members,
|
|
memberships,
|
|
ourUuid,
|
|
setStagedMemberships,
|
|
getPreferredBadge,
|
|
theme,
|
|
}: Readonly<{
|
|
conversation: ConversationType;
|
|
getPreferredBadge: PreferredBadgeSelectorType;
|
|
i18n: LocalizerType;
|
|
members: Array<ConversationType>;
|
|
memberships: ReadonlyArray<GroupV2PendingMembership>;
|
|
ourUuid: string;
|
|
setStagedMemberships: (stagedMembership: Array<StagedMembershipType>) => void;
|
|
theme: ThemeType;
|
|
}>) {
|
|
const groupedPendingMemberships = _.groupBy(
|
|
memberships,
|
|
membership => membership.metadata.addedByUserId
|
|
);
|
|
|
|
const { [ourUuid]: ourPendingMemberships, ...otherPendingMembershipGroups } =
|
|
groupedPendingMemberships;
|
|
|
|
const otherPendingMemberships = Object.keys(otherPendingMembershipGroups)
|
|
.map(id => members.find(member => member.uuid === id))
|
|
.filter((member): member is ConversationType => member !== undefined)
|
|
.map(member => {
|
|
assertDev(member.uuid, 'We just verified that member has uuid above');
|
|
return {
|
|
member,
|
|
pendingMemberships: otherPendingMembershipGroups[member.uuid],
|
|
};
|
|
});
|
|
|
|
return (
|
|
<PanelSection>
|
|
{ourPendingMemberships && (
|
|
<PanelSection title={i18n('PendingInvites--invited-by-you')}>
|
|
{ourPendingMemberships.map(membership => (
|
|
<PanelRow
|
|
key={membership.member.id}
|
|
icon={
|
|
<Avatar
|
|
badge={getPreferredBadge(membership.member.badges)}
|
|
conversationType="direct"
|
|
size={32}
|
|
i18n={i18n}
|
|
theme={theme}
|
|
{...membership.member}
|
|
/>
|
|
}
|
|
label={membership.member.title}
|
|
actions={
|
|
conversation.areWeAdmin ? (
|
|
<ConversationDetailsIcon
|
|
ariaLabel={i18n('PendingInvites--revoke-for-label')}
|
|
icon={IconType.trash}
|
|
onClick={() => {
|
|
setStagedMemberships([
|
|
{
|
|
type: StageType.REVOKE_INVITE,
|
|
membership,
|
|
},
|
|
]);
|
|
}}
|
|
/>
|
|
) : null
|
|
}
|
|
/>
|
|
))}
|
|
</PanelSection>
|
|
)}
|
|
{otherPendingMemberships.length > 0 && (
|
|
<PanelSection title={i18n('PendingInvites--invited-by-others')}>
|
|
{otherPendingMemberships.map(({ member, pendingMemberships }) => (
|
|
<PanelRow
|
|
key={member.id}
|
|
icon={
|
|
<Avatar
|
|
badge={getPreferredBadge(member.badges)}
|
|
conversationType="direct"
|
|
size={32}
|
|
i18n={i18n}
|
|
theme={theme}
|
|
{...member}
|
|
/>
|
|
}
|
|
label={member.title}
|
|
right={i18n('PendingInvites--invited-count', [
|
|
pendingMemberships.length.toString(),
|
|
])}
|
|
actions={
|
|
conversation.areWeAdmin ? (
|
|
<ConversationDetailsIcon
|
|
ariaLabel={i18n('PendingInvites--revoke-for-label')}
|
|
icon={IconType.trash}
|
|
onClick={() => {
|
|
setStagedMemberships(
|
|
pendingMemberships.map(membership => ({
|
|
type: StageType.REVOKE_INVITE,
|
|
membership,
|
|
}))
|
|
);
|
|
}}
|
|
/>
|
|
) : null
|
|
}
|
|
/>
|
|
))}
|
|
</PanelSection>
|
|
)}
|
|
<div className="ConversationDetails__pending--info">
|
|
{i18n('PendingInvites--info')}
|
|
</div>
|
|
</PanelSection>
|
|
);
|
|
}
|