You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
205 lines
6.6 KiB
TypeScript
205 lines
6.6 KiB
TypeScript
import useKey from 'react-use/lib/useKey';
|
|
|
|
import _ from 'lodash';
|
|
import { useDispatch, useSelector } from 'react-redux';
|
|
import { VALIDATION } from '../../session/constants';
|
|
import { getConversationController } from '../../session/conversations';
|
|
import { ToastUtils, UserUtils } from '../../session/utils';
|
|
import { updateInviteContactModal } from '../../state/ducks/modalDialog';
|
|
import { SpacerLG } from '../basic/Text';
|
|
|
|
import { useConversationPropsById } from '../../hooks/useParamSelector';
|
|
import { useSet } from '../../hooks/useSet';
|
|
import { initiateClosedGroupUpdate } from '../../session/group/closed-group';
|
|
import { SessionUtilUserGroups } from '../../session/utils/libsession/libsession_utils_user_groups';
|
|
import { getPrivateContactsPubkeys } from '../../state/selectors/conversations';
|
|
import { MemberListItem } from '../MemberListItem';
|
|
import { SessionWrapperModal } from '../SessionWrapperModal';
|
|
import { SessionButton, SessionButtonColor, SessionButtonType } from '../basic/SessionButton';
|
|
import { ConversationTypeEnum } from '../../models/types';
|
|
import { Localizer } from '../basic/Localizer';
|
|
|
|
type Props = {
|
|
conversationId: string;
|
|
};
|
|
|
|
async function submitForOpenGroup(convoId: string, pubkeys: Array<string>) {
|
|
const convo = getConversationController().get(convoId);
|
|
if (!convo || !convo.isPublic()) {
|
|
throw new Error('submitForOpenGroup group not found');
|
|
}
|
|
try {
|
|
const roomDetails = await SessionUtilUserGroups.getCommunityByConvoIdNotCached(convo.id);
|
|
if (!roomDetails) {
|
|
throw new Error(`getCommunityByFullUrl returned no result for ${convo.id}`);
|
|
}
|
|
const groupInvitation = {
|
|
url: roomDetails?.fullUrlWithPubkey,
|
|
name: convo.getNicknameOrRealUsernameOrPlaceholder(),
|
|
};
|
|
// eslint-disable-next-line @typescript-eslint/no-misused-promises
|
|
pubkeys.forEach(async pubkeyStr => {
|
|
const privateConvo = await getConversationController().getOrCreateAndWait(
|
|
pubkeyStr,
|
|
ConversationTypeEnum.PRIVATE
|
|
);
|
|
|
|
if (privateConvo) {
|
|
void privateConvo.sendMessage({
|
|
body: '',
|
|
attachments: undefined,
|
|
groupInvitation,
|
|
preview: undefined,
|
|
quote: undefined,
|
|
});
|
|
}
|
|
});
|
|
} catch (e) {
|
|
window.log.warn('submitForOpenGroup failed with:', e.message);
|
|
}
|
|
}
|
|
|
|
const submitForClosedGroup = async (convoId: string, pubkeys: Array<string>) => {
|
|
const convo = getConversationController().get(convoId);
|
|
if (!convo || !convo.isGroup()) {
|
|
throw new Error('submitForClosedGroup group not found');
|
|
}
|
|
// closed group chats
|
|
const ourPK = UserUtils.getOurPubKeyStrFromCache();
|
|
// we only care about real members. If a member is currently a zombie we have to be able to add him back
|
|
let existingMembers = convo.get('members') || [];
|
|
// at least make sure it's an array
|
|
if (!Array.isArray(existingMembers)) {
|
|
existingMembers = [];
|
|
}
|
|
existingMembers = _.compact(existingMembers);
|
|
const existingZombies = convo.get('zombies') || [];
|
|
const newMembers = pubkeys.filter(d => !existingMembers.includes(d));
|
|
|
|
if (newMembers.length > 0) {
|
|
// Do not trigger an update if there is too many members
|
|
// be sure to include current zombies in this count
|
|
if (
|
|
newMembers.length + existingMembers.length + existingZombies.length >
|
|
VALIDATION.CLOSED_GROUP_SIZE_LIMIT
|
|
) {
|
|
ToastUtils.pushTooManyMembers();
|
|
return;
|
|
}
|
|
|
|
const allMembers = _.concat(existingMembers, newMembers, [ourPK]);
|
|
const uniqMembers = _.uniq(allMembers);
|
|
|
|
const groupId = convo.get('id');
|
|
const groupName = convo.getNicknameOrRealUsernameOrPlaceholder();
|
|
|
|
await initiateClosedGroupUpdate(groupId, groupName, uniqMembers);
|
|
}
|
|
};
|
|
|
|
const InviteContactsDialogInner = (props: Props) => {
|
|
const { conversationId } = props;
|
|
const dispatch = useDispatch();
|
|
|
|
const privateContactPubkeys = useSelector(getPrivateContactsPubkeys);
|
|
let validContactsForInvite = _.clone(privateContactPubkeys);
|
|
|
|
const convoProps = useConversationPropsById(conversationId);
|
|
|
|
const { uniqueValues: selectedContacts, addTo, removeFrom } = useSet<string>();
|
|
|
|
if (!convoProps) {
|
|
throw new Error('InviteContactsDialogInner not a valid convoId given');
|
|
}
|
|
if (convoProps.isPrivate) {
|
|
throw new Error('InviteContactsDialogInner must be a group');
|
|
}
|
|
if (!convoProps.isPublic) {
|
|
// filter our zombies and current members from the list of contact we can add
|
|
const members = convoProps.members || [];
|
|
const zombies = convoProps.zombies || [];
|
|
validContactsForInvite = validContactsForInvite.filter(
|
|
d => !members.includes(d) && !zombies.includes(d)
|
|
);
|
|
}
|
|
|
|
const isPublicConvo = convoProps.isPublic;
|
|
|
|
const closeDialog = () => {
|
|
dispatch(updateInviteContactModal(null));
|
|
};
|
|
|
|
const onClickOK = () => {
|
|
if (selectedContacts.length > 0) {
|
|
if (isPublicConvo) {
|
|
void submitForOpenGroup(conversationId, selectedContacts);
|
|
} else {
|
|
void submitForClosedGroup(conversationId, selectedContacts);
|
|
}
|
|
}
|
|
|
|
closeDialog();
|
|
};
|
|
|
|
useKey((event: KeyboardEvent) => {
|
|
return event.key === 'Enter';
|
|
}, onClickOK);
|
|
|
|
useKey((event: KeyboardEvent) => {
|
|
return event.key === 'Esc' || event.key === 'Escape';
|
|
}, closeDialog);
|
|
|
|
const titleText = window.i18n('membersInvite');
|
|
const cancelText = window.i18n('cancel');
|
|
const okText = window.i18n('okay');
|
|
|
|
const hasContacts = validContactsForInvite.length > 0;
|
|
|
|
return (
|
|
<SessionWrapperModal title={titleText} onClose={closeDialog}>
|
|
<SpacerLG />
|
|
|
|
<div className="contact-selection-list">
|
|
{hasContacts ? (
|
|
validContactsForInvite.map((member: string) => (
|
|
<MemberListItem
|
|
key={`contacts-list-${member}`}
|
|
pubkey={member}
|
|
isSelected={selectedContacts.includes(member)}
|
|
onSelect={addTo}
|
|
onUnselect={removeFrom}
|
|
disableBg={true}
|
|
/>
|
|
))
|
|
) : (
|
|
<>
|
|
<SpacerLG />
|
|
<p className="no-contacts">
|
|
<Localizer token="contactNone" />
|
|
</p>
|
|
<SpacerLG />
|
|
</>
|
|
)}
|
|
</div>
|
|
<SpacerLG />
|
|
|
|
<div className="session-modal__button-group">
|
|
<SessionButton
|
|
text={okText}
|
|
buttonType={SessionButtonType.Simple}
|
|
disabled={!hasContacts}
|
|
onClick={onClickOK}
|
|
/>
|
|
<SessionButton
|
|
text={cancelText}
|
|
buttonColor={SessionButtonColor.Danger}
|
|
buttonType={SessionButtonType.Simple}
|
|
onClick={closeDialog}
|
|
/>
|
|
</div>
|
|
</SessionWrapperModal>
|
|
);
|
|
};
|
|
|
|
export const InviteContactsDialog = InviteContactsDialogInner;
|