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.
		
		
		
		
		
			
		
			
				
	
	
		
			206 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			TypeScript
		
	
			
		
		
	
	
			206 lines
		
	
	
		
			6.7 KiB
		
	
	
	
		
			TypeScript
		
	
import React from 'react';
 | 
						|
import useKey from 'react-use/lib/useKey';
 | 
						|
 | 
						|
import _ from 'lodash';
 | 
						|
import { useDispatch, useSelector } from 'react-redux';
 | 
						|
import { ConversationTypeEnum } from '../../models/conversationAttributes';
 | 
						|
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';
 | 
						|
 | 
						|
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 chatName = convoProps.displayNameInProfile || window.i18n('unknown');
 | 
						|
  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 unknown = window.i18n('unknown');
 | 
						|
 | 
						|
  const titleText = `${window.i18n('addingContacts', [chatName || unknown])}`;
 | 
						|
  const cancelText = window.i18n('cancel');
 | 
						|
  const okText = window.i18n('ok');
 | 
						|
 | 
						|
  const hasContacts = validContactsForInvite.length > 0;
 | 
						|
 | 
						|
  return (
 | 
						|
    <SessionWrapperModal title={titleText} onClose={closeDialog}>
 | 
						|
      <SpacerLG />
 | 
						|
 | 
						|
      <div className="contact-selection-list">
 | 
						|
        {hasContacts ? (
 | 
						|
          validContactsForInvite.map((member: string) => (
 | 
						|
            <MemberListItem
 | 
						|
              key={member}
 | 
						|
              pubkey={member}
 | 
						|
              isSelected={selectedContacts.includes(member)}
 | 
						|
              onSelect={addTo}
 | 
						|
              onUnselect={removeFrom}
 | 
						|
              disableBg={true}
 | 
						|
            />
 | 
						|
          ))
 | 
						|
        ) : (
 | 
						|
          <>
 | 
						|
            <SpacerLG />
 | 
						|
            <p className="no-contacts">{window.i18n('noContactsToAdd')}</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;
 |