2022-03-04 19:48:44 +00:00
|
|
|
// Copyright 2021-2022 Signal Messenger, LLC
|
2021-03-11 21:29:31 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2022-04-05 00:38:22 +00:00
|
|
|
import React, { useReducer } from 'react';
|
2021-03-11 21:29:31 +00:00
|
|
|
import { without } from 'lodash';
|
|
|
|
|
2022-04-05 00:38:22 +00:00
|
|
|
import type { LocalizerType } from '../../../types/Util';
|
2021-03-11 21:29:31 +00:00
|
|
|
import {
|
|
|
|
AddGroupMemberErrorDialog,
|
|
|
|
AddGroupMemberErrorDialogMode,
|
|
|
|
} from '../../AddGroupMemberErrorDialog';
|
2022-04-05 00:38:22 +00:00
|
|
|
import type { SmartChooseGroupMembersModalPropsType } from '../../../state/smart/ChooseGroupMembersModal';
|
|
|
|
import type { SmartConfirmAdditionsModalPropsType } from '../../../state/smart/ConfirmAdditionsModal';
|
2021-03-11 21:29:31 +00:00
|
|
|
import {
|
|
|
|
toggleSelectedContactForGroupAddition,
|
|
|
|
OneTimeModalState,
|
|
|
|
} from '../../../groups/toggleSelectedContactForGroupAddition';
|
|
|
|
import { missingCaseError } from '../../../util/missingCaseError';
|
2021-10-26 19:15:33 +00:00
|
|
|
import type { RequestState } from './util';
|
2021-03-11 21:29:31 +00:00
|
|
|
|
|
|
|
type PropsType = {
|
|
|
|
clearRequestError: () => void;
|
|
|
|
conversationIdsAlreadyInGroup: Set<string>;
|
|
|
|
groupTitle: string;
|
|
|
|
i18n: LocalizerType;
|
|
|
|
makeRequest: (conversationIds: ReadonlyArray<string>) => Promise<void>;
|
|
|
|
onClose: () => void;
|
|
|
|
requestState: RequestState;
|
2022-10-24 20:46:36 +00:00
|
|
|
maxGroupSize: number;
|
|
|
|
maxRecommendedGroupSize: number;
|
2022-04-05 00:38:22 +00:00
|
|
|
|
|
|
|
renderChooseGroupMembersModal: (
|
|
|
|
props: SmartChooseGroupMembersModalPropsType
|
|
|
|
) => JSX.Element;
|
|
|
|
renderConfirmAdditionsModal: (
|
|
|
|
props: SmartConfirmAdditionsModalPropsType
|
|
|
|
) => JSX.Element;
|
2021-03-11 21:29:31 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
enum Stage {
|
|
|
|
ChoosingContacts,
|
|
|
|
ConfirmingAdds,
|
|
|
|
}
|
|
|
|
|
|
|
|
type StateType = {
|
2022-10-24 20:46:36 +00:00
|
|
|
maxGroupSize: number;
|
|
|
|
maxRecommendedGroupSize: number;
|
2021-03-11 21:29:31 +00:00
|
|
|
maximumGroupSizeModalState: OneTimeModalState;
|
|
|
|
recommendedGroupSizeModalState: OneTimeModalState;
|
|
|
|
searchTerm: string;
|
2022-12-22 00:07:02 +00:00
|
|
|
selectedConversationIds: ReadonlyArray<string>;
|
2021-03-11 21:29:31 +00:00
|
|
|
stage: Stage;
|
|
|
|
};
|
|
|
|
|
|
|
|
enum ActionType {
|
|
|
|
CloseMaximumGroupSizeModal,
|
|
|
|
CloseRecommendedMaximumGroupSizeModal,
|
|
|
|
ConfirmAdds,
|
|
|
|
RemoveSelectedContact,
|
|
|
|
ReturnToContactChooser,
|
|
|
|
ToggleSelectedContact,
|
|
|
|
UpdateSearchTerm,
|
|
|
|
}
|
|
|
|
|
|
|
|
type Action =
|
|
|
|
| { type: ActionType.CloseMaximumGroupSizeModal }
|
|
|
|
| { type: ActionType.CloseRecommendedMaximumGroupSizeModal }
|
|
|
|
| { type: ActionType.ConfirmAdds }
|
|
|
|
| { type: ActionType.ReturnToContactChooser }
|
|
|
|
| { type: ActionType.RemoveSelectedContact; conversationId: string }
|
|
|
|
| {
|
|
|
|
type: ActionType.ToggleSelectedContact;
|
|
|
|
conversationId: string;
|
|
|
|
numberOfContactsAlreadyInGroup: number;
|
|
|
|
}
|
|
|
|
| { type: ActionType.UpdateSearchTerm; searchTerm: string };
|
|
|
|
|
|
|
|
// `<ConversationDetails>` isn't currently hooked up to Redux, but that's not desirable in
|
|
|
|
// the long term (see DESKTOP-1260). For now, this component has internal state with a
|
|
|
|
// reducer. Hopefully, this will make things easier to port to Redux in the future.
|
|
|
|
function reducer(
|
|
|
|
state: Readonly<StateType>,
|
|
|
|
action: Readonly<Action>
|
|
|
|
): StateType {
|
|
|
|
switch (action.type) {
|
|
|
|
case ActionType.CloseMaximumGroupSizeModal:
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
maximumGroupSizeModalState: OneTimeModalState.Shown,
|
|
|
|
};
|
|
|
|
case ActionType.CloseRecommendedMaximumGroupSizeModal:
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
recommendedGroupSizeModalState: OneTimeModalState.Shown,
|
|
|
|
};
|
|
|
|
case ActionType.ConfirmAdds:
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
stage: Stage.ConfirmingAdds,
|
|
|
|
};
|
|
|
|
case ActionType.ReturnToContactChooser:
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
stage: Stage.ChoosingContacts,
|
|
|
|
};
|
|
|
|
case ActionType.RemoveSelectedContact:
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
selectedConversationIds: without(
|
|
|
|
state.selectedConversationIds,
|
|
|
|
action.conversationId
|
|
|
|
),
|
|
|
|
};
|
|
|
|
case ActionType.ToggleSelectedContact:
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
...toggleSelectedContactForGroupAddition(action.conversationId, {
|
2022-10-24 20:46:36 +00:00
|
|
|
maxGroupSize: state.maxGroupSize,
|
|
|
|
maxRecommendedGroupSize: state.maxRecommendedGroupSize,
|
2021-03-11 21:29:31 +00:00
|
|
|
maximumGroupSizeModalState: state.maximumGroupSizeModalState,
|
|
|
|
numberOfContactsAlreadyInGroup: action.numberOfContactsAlreadyInGroup,
|
|
|
|
recommendedGroupSizeModalState: state.recommendedGroupSizeModalState,
|
|
|
|
selectedConversationIds: state.selectedConversationIds,
|
|
|
|
}),
|
|
|
|
};
|
|
|
|
case ActionType.UpdateSearchTerm:
|
|
|
|
return {
|
|
|
|
...state,
|
|
|
|
searchTerm: action.searchTerm,
|
|
|
|
};
|
|
|
|
default:
|
|
|
|
throw missingCaseError(action);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export function AddGroupMembersModal({
|
2021-03-11 21:29:31 +00:00
|
|
|
clearRequestError,
|
|
|
|
conversationIdsAlreadyInGroup,
|
|
|
|
groupTitle,
|
|
|
|
i18n,
|
|
|
|
onClose,
|
|
|
|
makeRequest,
|
2022-10-24 20:46:36 +00:00
|
|
|
maxGroupSize,
|
|
|
|
maxRecommendedGroupSize,
|
2021-03-11 21:29:31 +00:00
|
|
|
requestState,
|
2022-04-05 00:38:22 +00:00
|
|
|
renderChooseGroupMembersModal,
|
|
|
|
renderConfirmAdditionsModal,
|
2022-11-18 00:45:19 +00:00
|
|
|
}: PropsType): JSX.Element {
|
2021-03-11 21:29:31 +00:00
|
|
|
const numberOfContactsAlreadyInGroup = conversationIdsAlreadyInGroup.size;
|
|
|
|
const isGroupAlreadyFull = numberOfContactsAlreadyInGroup >= maxGroupSize;
|
|
|
|
const isGroupAlreadyOverRecommendedMaximum =
|
|
|
|
numberOfContactsAlreadyInGroup >= maxRecommendedGroupSize;
|
|
|
|
|
|
|
|
const [
|
|
|
|
{
|
|
|
|
maximumGroupSizeModalState,
|
|
|
|
recommendedGroupSizeModalState,
|
|
|
|
searchTerm,
|
|
|
|
selectedConversationIds,
|
|
|
|
stage,
|
|
|
|
},
|
|
|
|
dispatch,
|
|
|
|
] = useReducer(reducer, {
|
2022-10-24 20:46:36 +00:00
|
|
|
maxGroupSize,
|
|
|
|
maxRecommendedGroupSize,
|
2021-03-11 21:29:31 +00:00
|
|
|
maximumGroupSizeModalState: isGroupAlreadyFull
|
|
|
|
? OneTimeModalState.Showing
|
|
|
|
: OneTimeModalState.NeverShown,
|
|
|
|
recommendedGroupSizeModalState: isGroupAlreadyOverRecommendedMaximum
|
|
|
|
? OneTimeModalState.Shown
|
|
|
|
: OneTimeModalState.NeverShown,
|
|
|
|
searchTerm: '',
|
|
|
|
selectedConversationIds: [],
|
|
|
|
stage: Stage.ChoosingContacts,
|
|
|
|
});
|
|
|
|
|
|
|
|
if (maximumGroupSizeModalState === OneTimeModalState.Showing) {
|
|
|
|
return (
|
|
|
|
<AddGroupMemberErrorDialog
|
|
|
|
i18n={i18n}
|
|
|
|
maximumNumberOfContacts={maxGroupSize}
|
|
|
|
mode={AddGroupMemberErrorDialogMode.MaximumGroupSize}
|
|
|
|
onClose={() => {
|
|
|
|
dispatch({ type: ActionType.CloseMaximumGroupSizeModal });
|
|
|
|
}}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (recommendedGroupSizeModalState === OneTimeModalState.Showing) {
|
|
|
|
return (
|
|
|
|
<AddGroupMemberErrorDialog
|
|
|
|
i18n={i18n}
|
|
|
|
mode={AddGroupMemberErrorDialogMode.RecommendedMaximumGroupSize}
|
|
|
|
onClose={() => {
|
|
|
|
dispatch({
|
|
|
|
type: ActionType.CloseRecommendedMaximumGroupSizeModal,
|
|
|
|
});
|
|
|
|
}}
|
|
|
|
recommendedMaximumNumberOfContacts={maxRecommendedGroupSize}
|
|
|
|
/>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
switch (stage) {
|
|
|
|
case Stage.ChoosingContacts: {
|
|
|
|
// See note above: these will soon become Redux actions.
|
|
|
|
const confirmAdds = () => {
|
|
|
|
dispatch({ type: ActionType.ConfirmAdds });
|
|
|
|
};
|
|
|
|
const removeSelectedContact = (conversationId: string) => {
|
|
|
|
dispatch({
|
|
|
|
type: ActionType.RemoveSelectedContact,
|
|
|
|
conversationId,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
const setSearchTerm = (term: string) => {
|
|
|
|
dispatch({
|
|
|
|
type: ActionType.UpdateSearchTerm,
|
|
|
|
searchTerm: term,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
const toggleSelectedContact = (conversationId: string) => {
|
|
|
|
dispatch({
|
|
|
|
type: ActionType.ToggleSelectedContact,
|
|
|
|
conversationId,
|
|
|
|
numberOfContactsAlreadyInGroup,
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
2022-04-05 00:38:22 +00:00
|
|
|
return renderChooseGroupMembersModal({
|
|
|
|
confirmAdds,
|
|
|
|
selectedConversationIds,
|
|
|
|
conversationIdsAlreadyInGroup,
|
|
|
|
maxGroupSize,
|
|
|
|
onClose,
|
|
|
|
removeSelectedContact,
|
|
|
|
searchTerm,
|
|
|
|
setSearchTerm,
|
|
|
|
toggleSelectedContact,
|
|
|
|
});
|
2021-03-11 21:29:31 +00:00
|
|
|
}
|
|
|
|
case Stage.ConfirmingAdds: {
|
|
|
|
const onCloseConfirmationDialog = () => {
|
|
|
|
dispatch({ type: ActionType.ReturnToContactChooser });
|
|
|
|
clearRequestError();
|
|
|
|
};
|
|
|
|
|
2022-04-05 00:38:22 +00:00
|
|
|
return renderConfirmAdditionsModal({
|
|
|
|
groupTitle,
|
|
|
|
makeRequest: () => {
|
2022-12-21 18:41:48 +00:00
|
|
|
void makeRequest(selectedConversationIds);
|
2022-04-05 00:38:22 +00:00
|
|
|
},
|
|
|
|
onClose: onCloseConfirmationDialog,
|
|
|
|
requestState,
|
|
|
|
selectedConversationIds,
|
|
|
|
});
|
2021-03-11 21:29:31 +00:00
|
|
|
}
|
|
|
|
default:
|
|
|
|
throw missingCaseError(stage);
|
|
|
|
}
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|