// Copyright 2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { useEffect, useMemo, useState } from 'react'; import { noop } from 'lodash'; import { SearchInput } from './SearchInput'; import { filterAndSortConversationsByRecent } from '../util/filterAndSortConversations'; import type { ConversationType } from '../state/ducks/conversations'; import type { LocalizerType } from '../types/Util'; import type { PreferredBadgeSelectorType } from '../state/selectors/badges'; import type { PropsType as StoriesSettingsModalPropsType } from './StoriesSettingsModal'; import type { StoryDistributionListWithMembersDataType } from '../types/Stories'; import type { UUIDStringType } from '../types/UUID'; import { Alert } from './Alert'; import { Avatar, AvatarSize } from './Avatar'; import { Button, ButtonVariant } from './Button'; import { Checkbox } from './Checkbox'; import { ConfirmationDialog } from './ConfirmationDialog'; import { ContextMenu } from './ContextMenu'; import { DistributionListSettingsModal, EditDistributionListModal, EditMyStoriesPrivacy, Page as StoriesSettingsPage, } from './StoriesSettingsModal'; import { MY_STORIES_ID, getStoryDistributionListName } from '../types/Stories'; import type { RenderModalPage, ModalPropsType } from './Modal'; import { PagedModal, ModalPage } from './Modal'; import { StoryDistributionListName } from './StoryDistributionListName'; import { Theme } from '../util/theme'; import { isNotNil } from '../util/isNotNil'; export type PropsType = { candidateConversations: Array; distributionLists: Array; getPreferredBadge: PreferredBadgeSelectorType; groupConversations: Array; groupStories: Array; hasFirstStoryPostExperience: boolean; i18n: LocalizerType; me: ConversationType; onClose: () => unknown; onDeleteList: (listId: string) => unknown; onDistributionListCreated: ( name: string, viewerUuids: Array ) => unknown; onSelectedStoryList: (memberUuids: Array) => unknown; onSend: ( listIds: Array, conversationIds: Array ) => unknown; signalConnections: Array; toggleGroupsForStorySend: (cids: Array) => unknown; } & Pick< StoriesSettingsModalPropsType, | 'onHideMyStoriesFrom' | 'onRemoveMember' | 'onRepliesNReactionsChanged' | 'onViewersUpdated' | 'setMyStoriesToAllSignalConnections' | 'toggleSignalConnectionsModal' >; enum SendStoryPage { ChooseGroups = 'ChooseGroups', EditingDistributionList = 'EditingDistributionList', SendStory = 'SendStory', SetMyStoriesPrivacy = 'SetMyStoriesPrivacy', } const Page = { ...SendStoryPage, ...StoriesSettingsPage, }; type PageType = SendStoryPage | StoriesSettingsPage; function getListMemberUuids( list: StoryDistributionListWithMembersDataType, signalConnections: Array ): Array { const memberUuids = list.members.map(({ uuid }) => uuid).filter(isNotNil); if (list.id === MY_STORIES_ID && list.isBlockList) { const excludeUuids = new Set(memberUuids); return signalConnections .map(conversation => conversation.uuid) .filter(isNotNil) .filter(uuid => !excludeUuids.has(uuid)); } return memberUuids; } function getListViewers( list: StoryDistributionListWithMembersDataType, i18n: LocalizerType, signalConnections: Array ): string { let memberCount = list.members.length; if (list.id === MY_STORIES_ID && list.isBlockList) { memberCount = list.isBlockList ? signalConnections.length - list.members.length : signalConnections.length; } return memberCount === 1 ? i18n('StoriesSettings__viewers--singular', ['1']) : i18n('StoriesSettings__viewers--plural', [String(memberCount)]); } export const SendStoryModal = ({ candidateConversations, distributionLists, getPreferredBadge, groupConversations, groupStories, hasFirstStoryPostExperience, i18n, me, onClose, onDeleteList, onDistributionListCreated, onHideMyStoriesFrom, onRemoveMember, onRepliesNReactionsChanged, onSelectedStoryList, onSend, onViewersUpdated, setMyStoriesToAllSignalConnections, signalConnections, toggleGroupsForStorySend, toggleSignalConnectionsModal, }: PropsType): JSX.Element => { const [page, setPage] = useState(Page.SendStory); const [selectedListIds, setSelectedListIds] = useState>( new Set() ); const [selectedGroupIds, setSelectedGroupIds] = useState>( new Set() ); const selectedStoryNames = useMemo( () => distributionLists .filter(list => selectedListIds.has(list.id)) .map(list => list.name) .concat( groupStories .filter(group => selectedGroupIds.has(group.id)) .map(group => group.title) ), [distributionLists, groupStories, selectedGroupIds, selectedListIds] ); const [searchTerm, setSearchTerm] = useState(''); const [filteredConversations, setFilteredConversations] = useState( filterAndSortConversationsByRecent( groupConversations, searchTerm, undefined ) ); const normalizedSearchTerm = searchTerm.trim(); useEffect(() => { const timeout = setTimeout(() => { setFilteredConversations( filterAndSortConversationsByRecent( groupConversations, normalizedSearchTerm, undefined ) ); }, 200); return () => { clearTimeout(timeout); }; }, [groupConversations, normalizedSearchTerm, setFilteredConversations]); const [chosenGroupIds, setChosenGroupIds] = useState>( new Set() ); const chosenGroupNames = useMemo( () => filteredConversations .filter(group => chosenGroupIds.has(group.id)) .map(group => group.title), [filteredConversations, chosenGroupIds] ); const [selectedContacts, setSelectedContacts] = useState< Array >([]); const [hasAnnouncementsOnlyAlert, setHasAnnouncementsOnlyAlert] = useState(false); const [confirmRemoveGroupId, setConfirmRemoveGroupId] = useState< string | undefined >(); const [confirmDeleteListId, setConfirmDeleteListId] = useState< string | undefined >(); const [listIdToEdit, setListIdToEdit] = useState(); useEffect(() => { if (listIdToEdit) { setPage(Page.EditingDistributionList); } else { setPage(Page.SendStory); } }, [listIdToEdit]); const listToEdit = useMemo(() => { if (!listIdToEdit) { return; } return distributionLists.find(list => list.id === listIdToEdit); }, [distributionLists, listIdToEdit]); // myStoriesPrivacy, myStoriesPrivacyUuids, and myStories are only used // during the first time posting to My Stories experience where we have // to select the privacy settings. const ogMyStories = useMemo( () => distributionLists.find(list => list.id === MY_STORIES_ID), [distributionLists] ); const initialMyStories = useMemo( () => ({ allowsReplies: true, id: MY_STORIES_ID, name: i18n('Stories__mine'), isBlockList: ogMyStories?.isBlockList ?? true, members: ogMyStories?.members || [], }), [i18n, ogMyStories] ); const initialMyStoriesMemberUuids = useMemo( () => (ogMyStories?.members || []).map(({ uuid }) => uuid).filter(isNotNil), [ogMyStories] ); const [stagedMyStories, setStagedMyStories] = useState(initialMyStories); const [stagedMyStoriesMemberUuids, setStagedMyStoriesMemberUuids] = useState< Array >(initialMyStoriesMemberUuids); let selectedNames: string | undefined; if (page === Page.ChooseGroups) { selectedNames = chosenGroupNames.join(', '); } else { selectedNames = selectedStoryNames .map(listName => getStoryDistributionListName(i18n, listName, listName)) .join(', '); } const modalCommonProps: Pick = { hasXButton: true, i18n, }; let modal: RenderModalPage; if (page === Page.SetMyStoriesPrivacy) { const footer = ( <>
); modal = handleClose => ( { let nextSelectedContacts = stagedMyStories.members; if (!stagedMyStories.isBlockList) { setStagedMyStories(myStories => ({ ...myStories, isBlockList: true, members: [], })); nextSelectedContacts = []; } setSelectedContacts(nextSelectedContacts); setPage(Page.HideStoryFrom); }} onClickOnlyShareWith={() => { if (!stagedMyStories.isBlockList) { setSelectedContacts(stagedMyStories.members); } else { setStagedMyStories(myStories => ({ ...myStories, isBlockList: false, members: [], })); } setPage(Page.AddViewer); }} setSelectedContacts={setSelectedContacts} setMyStoriesToAllSignalConnections={() => { setStagedMyStories(myStories => ({ ...myStories, isBlockList: true, members: [], })); setSelectedContacts([]); }} toggleSignalConnectionsModal={toggleSignalConnectionsModal} /> ); } else if (page === Page.EditingDistributionList && listToEdit) { modal = handleClose => ( setListIdToEdit(undefined)} onClose={handleClose} /> ); } else if ( page === Page.ChooseViewers || page === Page.NameStory || page === Page.AddViewer || page === Page.HideStoryFrom ) { modal = handleClose => ( { onDistributionListCreated(name, uuids); setPage(Page.SendStory); }} onViewersUpdated={uuids => { if (listIdToEdit && page === Page.AddViewer) { onViewersUpdated(listIdToEdit, uuids); setPage(Page.EditingDistributionList); } else if (page === Page.ChooseViewers) { setPage(Page.NameStory); } else if (listIdToEdit && page === Page.HideStoryFrom) { onHideMyStoriesFrom(uuids); setPage(Page.SendStory); } else if (page === Page.HideStoryFrom || page === Page.AddViewer) { setStagedMyStoriesMemberUuids(uuids); setPage(Page.SetMyStoriesPrivacy); } else { setPage(Page.SendStory); } }} page={page} onClose={handleClose} onBackButtonClick={() => { if (listIdToEdit) { if ( page === Page.AddViewer || page === Page.HideStoryFrom || page === Page.ChooseViewers ) { setPage(Page.EditingDistributionList); } else { setListIdToEdit(undefined); } } else if (page === Page.HideStoryFrom || page === Page.AddViewer) { setSelectedContacts([]); setStagedMyStories(initialMyStories); setStagedMyStoriesMemberUuids(initialMyStoriesMemberUuids); setPage(Page.SetMyStoriesPrivacy); } else if (page === Page.ChooseViewers) { setSelectedContacts([]); setPage(Page.SendStory); } else if (page === Page.NameStory) { setPage(Page.ChooseViewers); } }} selectedContacts={selectedContacts} setSelectedContacts={setSelectedContacts} /> ); } else if (page === Page.ChooseGroups) { const footer = ( <>
{selectedNames}