// Copyright 2021 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { ThunkAction } from 'redux-thunk'; import type { ExplodePromiseResultType } from '../../util/explodePromise'; import type { GroupV2PendingMemberType } from '../../model-types.d'; import type { PropsForMessage } from '../selectors/message'; import type { RecipientsByConversation } from './stories'; import type { SafetyNumberChangeSource } from '../../components/SafetyNumberChangeDialog'; import type { StateType as RootStateType } from '../reducer'; import type { UUIDStringType } from '../../types/UUID'; import * as SingleServePromise from '../../services/singleServePromise'; import * as Stickers from '../../types/Stickers'; import { getMessageById } from '../../messages/getMessageById'; import { getMessagePropsSelector } from '../selectors/message'; import type { BoundActionCreatorsMapObject } from '../../hooks/useBoundActions'; import { longRunningTaskWrapper } from '../../util/longRunningTaskWrapper'; import { useBoundActions } from '../../hooks/useBoundActions'; import { isGroupV1 } from '../../util/whatTypeOfConversation'; import { getGroupMigrationMembers } from '../../groups'; // State export type ForwardMessagePropsType = Omit< PropsForMessage, 'renderingContext' | 'menu' | 'contextMenu' >; export type SafetyNumberChangedBlockingDataType = Readonly<{ promiseUuid: UUIDStringType; source?: SafetyNumberChangeSource; }>; type MigrateToGV2PropsType = { areWeInvited: boolean; conversationId: string; droppedMemberIds: ReadonlyArray; hasMigrated: boolean; invitedMemberIds: ReadonlyArray; }; export type GlobalModalsStateType = Readonly<{ addUserToAnotherGroupModalContactId?: string; contactModalState?: ContactModalStateType; errorModalProps?: { description?: string; title?: string; }; forwardMessageProps?: ForwardMessagePropsType; gv2MigrationProps?: MigrateToGV2PropsType; isProfileEditorVisible: boolean; isSignalConnectionsVisible: boolean; isShortcutGuideModalVisible: boolean; isStoriesSettingsVisible: boolean; isWhatsNewVisible: boolean; profileEditorHasError: boolean; safetyNumberChangedBlockingData?: SafetyNumberChangedBlockingDataType; safetyNumberModalContactId?: string; stickerPackPreviewId?: string; userNotFoundModalState?: UserNotFoundModalStateType; }>; // Actions const HIDE_CONTACT_MODAL = 'globalModals/HIDE_CONTACT_MODAL'; const SHOW_CONTACT_MODAL = 'globalModals/SHOW_CONTACT_MODAL'; const HIDE_WHATS_NEW_MODAL = 'globalModals/HIDE_WHATS_NEW_MODAL_MODAL'; const SHOW_WHATS_NEW_MODAL = 'globalModals/SHOW_WHATS_NEW_MODAL_MODAL'; const HIDE_UUID_NOT_FOUND_MODAL = 'globalModals/HIDE_UUID_NOT_FOUND_MODAL'; const SHOW_UUID_NOT_FOUND_MODAL = 'globalModals/SHOW_UUID_NOT_FOUND_MODAL'; const SHOW_STORIES_SETTINGS = 'globalModals/SHOW_STORIES_SETTINGS'; const HIDE_STORIES_SETTINGS = 'globalModals/HIDE_STORIES_SETTINGS'; const TOGGLE_FORWARD_MESSAGE_MODAL = 'globalModals/TOGGLE_FORWARD_MESSAGE_MODAL'; const TOGGLE_PROFILE_EDITOR = 'globalModals/TOGGLE_PROFILE_EDITOR'; export const TOGGLE_PROFILE_EDITOR_ERROR = 'globalModals/TOGGLE_PROFILE_EDITOR_ERROR'; const TOGGLE_SAFETY_NUMBER_MODAL = 'globalModals/TOGGLE_SAFETY_NUMBER_MODAL'; const TOGGLE_ADD_USER_TO_ANOTHER_GROUP_MODAL = 'globalModals/TOGGLE_ADD_USER_TO_ANOTHER_GROUP_MODAL'; const TOGGLE_SIGNAL_CONNECTIONS_MODAL = 'globalModals/TOGGLE_SIGNAL_CONNECTIONS_MODAL'; export const SHOW_SEND_ANYWAY_DIALOG = 'globalModals/SHOW_SEND_ANYWAY_DIALOG'; const HIDE_SEND_ANYWAY_DIALOG = 'globalModals/HIDE_SEND_ANYWAY_DIALOG'; const SHOW_GV2_MIGRATION_DIALOG = 'globalModals/SHOW_GV2_MIGRATION_DIALOG'; const CLOSE_GV2_MIGRATION_DIALOG = 'globalModals/CLOSE_GV2_MIGRATION_DIALOG'; const SHOW_STICKER_PACK_PREVIEW = 'globalModals/SHOW_STICKER_PACK_PREVIEW'; const CLOSE_STICKER_PACK_PREVIEW = 'globalModals/CLOSE_STICKER_PACK_PREVIEW'; const CLOSE_ERROR_MODAL = 'globalModals/CLOSE_ERROR_MODAL'; const SHOW_ERROR_MODAL = 'globalModals/SHOW_ERROR_MODAL'; const CLOSE_SHORTCUT_GUIDE_MODAL = 'globalModals/CLOSE_SHORTCUT_GUIDE_MODAL'; const SHOW_SHORTCUT_GUIDE_MODAL = 'globalModals/SHOW_SHORTCUT_GUIDE_MODAL'; export type ContactModalStateType = { contactId: string; conversationId?: string; }; export type UserNotFoundModalStateType = | { type: 'phoneNumber'; phoneNumber: string; } | { type: 'username'; username: string; }; type HideContactModalActionType = { type: typeof HIDE_CONTACT_MODAL; }; type ShowContactModalActionType = { type: typeof SHOW_CONTACT_MODAL; payload: ContactModalStateType; }; type HideWhatsNewModalActionType = { type: typeof HIDE_WHATS_NEW_MODAL; }; type ShowWhatsNewModalActionType = { type: typeof SHOW_WHATS_NEW_MODAL; }; type HideUserNotFoundModalActionType = { type: typeof HIDE_UUID_NOT_FOUND_MODAL; }; export type ShowUserNotFoundModalActionType = { type: typeof SHOW_UUID_NOT_FOUND_MODAL; payload: UserNotFoundModalStateType; }; type ToggleForwardMessageModalActionType = { type: typeof TOGGLE_FORWARD_MESSAGE_MODAL; payload: ForwardMessagePropsType | undefined; }; type ToggleProfileEditorActionType = { type: typeof TOGGLE_PROFILE_EDITOR; }; export type ToggleProfileEditorErrorActionType = { type: typeof TOGGLE_PROFILE_EDITOR_ERROR; }; type ToggleSafetyNumberModalActionType = { type: typeof TOGGLE_SAFETY_NUMBER_MODAL; payload: string | undefined; }; type ToggleAddUserToAnotherGroupModalActionType = { type: typeof TOGGLE_ADD_USER_TO_ANOTHER_GROUP_MODAL; payload: string | undefined; }; type ToggleSignalConnectionsModalActionType = { type: typeof TOGGLE_SIGNAL_CONNECTIONS_MODAL; }; type ShowStoriesSettingsActionType = { type: typeof SHOW_STORIES_SETTINGS; }; type HideStoriesSettingsActionType = { type: typeof HIDE_STORIES_SETTINGS; }; type StartMigrationToGV2ActionType = { type: typeof SHOW_GV2_MIGRATION_DIALOG; payload: MigrateToGV2PropsType; }; type CloseGV2MigrationDialogActionType = { type: typeof CLOSE_GV2_MIGRATION_DIALOG; }; export type ShowSendAnywayDialogActionType = { type: typeof SHOW_SEND_ANYWAY_DIALOG; payload: SafetyNumberChangedBlockingDataType & { untrustedByConversation: RecipientsByConversation; }; }; type HideSendAnywayDialogActiontype = { type: typeof HIDE_SEND_ANYWAY_DIALOG; }; export type ShowStickerPackPreviewActionType = { type: typeof SHOW_STICKER_PACK_PREVIEW; payload: string; }; type CloseStickerPackPreviewActionType = { type: typeof CLOSE_STICKER_PACK_PREVIEW; }; type CloseErrorModalActionType = { type: typeof CLOSE_ERROR_MODAL; }; type ShowErrorModalActionType = { type: typeof SHOW_ERROR_MODAL; payload: { description?: string; title?: string; }; }; type CloseShortcutGuideModalActionType = { type: typeof CLOSE_SHORTCUT_GUIDE_MODAL; }; type ShowShortcutGuideModalActionType = { type: typeof SHOW_SHORTCUT_GUIDE_MODAL; }; export type GlobalModalsActionType = | StartMigrationToGV2ActionType | CloseGV2MigrationDialogActionType | HideContactModalActionType | ShowContactModalActionType | HideWhatsNewModalActionType | ShowWhatsNewModalActionType | HideUserNotFoundModalActionType | ShowUserNotFoundModalActionType | HideStoriesSettingsActionType | ShowStoriesSettingsActionType | HideSendAnywayDialogActiontype | ShowSendAnywayDialogActionType | CloseStickerPackPreviewActionType | ShowStickerPackPreviewActionType | CloseErrorModalActionType | ShowErrorModalActionType | CloseShortcutGuideModalActionType | ShowShortcutGuideModalActionType | ToggleForwardMessageModalActionType | ToggleProfileEditorActionType | ToggleProfileEditorErrorActionType | ToggleSafetyNumberModalActionType | ToggleAddUserToAnotherGroupModalActionType | ToggleSignalConnectionsModalActionType; // Action Creators export const actions = { hideContactModal, showContactModal, hideWhatsNewModal, showWhatsNewModal, hideUserNotFoundModal, showUserNotFoundModal, hideStoriesSettings, showStoriesSettings, hideBlockingSafetyNumberChangeDialog, showBlockingSafetyNumberChangeDialog, toggleForwardMessageModal, toggleProfileEditor, toggleProfileEditorHasError, toggleSafetyNumberModal, toggleAddUserToAnotherGroupModal, toggleSignalConnectionsModal, showGV2MigrationDialog, closeGV2MigrationDialog, showStickerPackPreview, closeStickerPackPreview, closeErrorModal, showErrorModal, closeShortcutGuideModal, showShortcutGuideModal, }; export const useGlobalModalActions = (): BoundActionCreatorsMapObject< typeof actions > => useBoundActions(actions); function hideContactModal(): HideContactModalActionType { return { type: HIDE_CONTACT_MODAL, }; } function showContactModal( contactId: string, conversationId?: string ): ShowContactModalActionType { return { type: SHOW_CONTACT_MODAL, payload: { contactId, conversationId, }, }; } function hideWhatsNewModal(): HideWhatsNewModalActionType { return { type: HIDE_WHATS_NEW_MODAL, }; } function showWhatsNewModal(): ShowWhatsNewModalActionType { return { type: SHOW_WHATS_NEW_MODAL, }; } function hideUserNotFoundModal(): HideUserNotFoundModalActionType { return { type: HIDE_UUID_NOT_FOUND_MODAL, }; } function showUserNotFoundModal( payload: UserNotFoundModalStateType ): ShowUserNotFoundModalActionType { return { type: SHOW_UUID_NOT_FOUND_MODAL, payload, }; } function hideStoriesSettings(): HideStoriesSettingsActionType { return { type: HIDE_STORIES_SETTINGS }; } function showStoriesSettings(): ShowStoriesSettingsActionType { return { type: SHOW_STORIES_SETTINGS }; } function showGV2MigrationDialog( conversationId: string ): ThunkAction { return async dispatch => { const conversation = window.ConversationController.get(conversationId); if (!conversation) { throw new Error( 'showGV2MigrationDialog: Expected a conversation to be found. Doing nothing' ); } const idForLogging = conversation.idForLogging(); if (!isGroupV1(conversation.attributes)) { throw new Error( `showGV2MigrationDialog/${idForLogging}: Cannot start, not a GroupV1 group` ); } // Note: this call will throw if, after generating member lists, we are no longer a // member or are in the pending member list. const { droppedGV2MemberIds, pendingMembersV2 } = await longRunningTaskWrapper({ idForLogging, name: 'getGroupMigrationMembers', task: () => getGroupMigrationMembers(conversation), }); const invitedMemberIds = pendingMembersV2.map( (item: GroupV2PendingMemberType) => item.uuid ); dispatch({ type: SHOW_GV2_MIGRATION_DIALOG, payload: { areWeInvited: false, conversationId, droppedMemberIds: droppedGV2MemberIds, hasMigrated: false, invitedMemberIds, }, }); }; } function closeGV2MigrationDialog(): CloseGV2MigrationDialogActionType { return { type: CLOSE_GV2_MIGRATION_DIALOG, }; } function toggleForwardMessageModal( messageId?: string ): ThunkAction< void, RootStateType, unknown, ToggleForwardMessageModalActionType > { return async (dispatch, getState) => { if (!messageId) { dispatch({ type: TOGGLE_FORWARD_MESSAGE_MODAL, payload: undefined, }); return; } const message = await getMessageById(messageId); if (!message) { throw new Error( `toggleForwardMessageModal: no message found for ${messageId}` ); } const messagePropsSelector = getMessagePropsSelector(getState()); const messageProps = messagePropsSelector(message.attributes); dispatch({ type: TOGGLE_FORWARD_MESSAGE_MODAL, payload: messageProps, }); }; } function toggleProfileEditor(): ToggleProfileEditorActionType { return { type: TOGGLE_PROFILE_EDITOR }; } function toggleProfileEditorHasError(): ToggleProfileEditorErrorActionType { return { type: TOGGLE_PROFILE_EDITOR_ERROR }; } function toggleSafetyNumberModal( safetyNumberModalContactId?: string ): ToggleSafetyNumberModalActionType { return { type: TOGGLE_SAFETY_NUMBER_MODAL, payload: safetyNumberModalContactId, }; } function toggleAddUserToAnotherGroupModal( contactId?: string ): ToggleAddUserToAnotherGroupModalActionType { return { type: TOGGLE_ADD_USER_TO_ANOTHER_GROUP_MODAL, payload: contactId, }; } function toggleSignalConnectionsModal(): ToggleSignalConnectionsModalActionType { return { type: TOGGLE_SIGNAL_CONNECTIONS_MODAL, }; } function showBlockingSafetyNumberChangeDialog( untrustedByConversation: RecipientsByConversation, explodedPromise: ExplodePromiseResultType, source?: SafetyNumberChangeSource ): ThunkAction { const promiseUuid = SingleServePromise.set(explodedPromise); return dispatch => { dispatch({ type: SHOW_SEND_ANYWAY_DIALOG, payload: { untrustedByConversation, promiseUuid, source, }, }); }; } function hideBlockingSafetyNumberChangeDialog(): HideSendAnywayDialogActiontype { return { type: HIDE_SEND_ANYWAY_DIALOG, }; } function closeStickerPackPreview(): ThunkAction< void, RootStateType, unknown, CloseStickerPackPreviewActionType > { return async (dispatch, getState) => { const packId = getState().globalModals.stickerPackPreviewId; if (packId && Stickers.getStickerPack(packId) !== undefined) { await Stickers.removeEphemeralPack(packId); } dispatch({ type: CLOSE_STICKER_PACK_PREVIEW, }); }; } export function showStickerPackPreview( packId: string, packKey: string ): ShowStickerPackPreviewActionType { // Intentionally not awaiting this so that we can show the modal right away. // The modal has a loading spinner on it. void Stickers.downloadEphemeralPack(packId, packKey); return { type: SHOW_STICKER_PACK_PREVIEW, payload: packId, }; } function closeErrorModal(): CloseErrorModalActionType { return { type: CLOSE_ERROR_MODAL, }; } function showErrorModal({ description, title, }: { title?: string; description?: string; }): ShowErrorModalActionType { return { type: SHOW_ERROR_MODAL, payload: { description, title, }, }; } function closeShortcutGuideModal(): CloseShortcutGuideModalActionType { return { type: CLOSE_SHORTCUT_GUIDE_MODAL, }; } function showShortcutGuideModal(): ShowShortcutGuideModalActionType { return { type: SHOW_SHORTCUT_GUIDE_MODAL, }; } // Reducer export function getEmptyState(): GlobalModalsStateType { return { isProfileEditorVisible: false, isShortcutGuideModalVisible: false, isSignalConnectionsVisible: false, isStoriesSettingsVisible: false, isWhatsNewVisible: false, profileEditorHasError: false, }; } export function reducer( state: Readonly = getEmptyState(), action: Readonly ): GlobalModalsStateType { if (action.type === TOGGLE_PROFILE_EDITOR) { return { ...state, isProfileEditorVisible: !state.isProfileEditorVisible, }; } if (action.type === TOGGLE_PROFILE_EDITOR_ERROR) { return { ...state, profileEditorHasError: !state.profileEditorHasError, }; } if (action.type === SHOW_WHATS_NEW_MODAL) { return { ...state, isWhatsNewVisible: true, }; } if (action.type === HIDE_WHATS_NEW_MODAL) { return { ...state, isWhatsNewVisible: false, }; } if (action.type === HIDE_UUID_NOT_FOUND_MODAL) { return { ...state, userNotFoundModalState: undefined, }; } if (action.type === SHOW_UUID_NOT_FOUND_MODAL) { return { ...state, userNotFoundModalState: { ...action.payload, }, }; } if (action.type === SHOW_CONTACT_MODAL) { return { ...state, contactModalState: action.payload, }; } if (action.type === HIDE_CONTACT_MODAL) { return { ...state, contactModalState: undefined, }; } if (action.type === TOGGLE_SAFETY_NUMBER_MODAL) { return { ...state, safetyNumberModalContactId: action.payload, }; } if (action.type === TOGGLE_ADD_USER_TO_ANOTHER_GROUP_MODAL) { return { ...state, addUserToAnotherGroupModalContactId: action.payload, }; } if (action.type === TOGGLE_FORWARD_MESSAGE_MODAL) { return { ...state, forwardMessageProps: action.payload, }; } if (action.type === HIDE_STORIES_SETTINGS) { return { ...state, isStoriesSettingsVisible: false, }; } if (action.type === SHOW_STORIES_SETTINGS) { return { ...state, isStoriesSettingsVisible: true, }; } if (action.type === TOGGLE_SIGNAL_CONNECTIONS_MODAL) { return { ...state, isSignalConnectionsVisible: !state.isSignalConnectionsVisible, }; } if (action.type === SHOW_SEND_ANYWAY_DIALOG) { const { promiseUuid, source } = action.payload; return { ...state, safetyNumberChangedBlockingData: { promiseUuid, source, }, }; } if (action.type === HIDE_SEND_ANYWAY_DIALOG) { return { ...state, safetyNumberChangedBlockingData: undefined, }; } if (action.type === CLOSE_STICKER_PACK_PREVIEW) { return { ...state, stickerPackPreviewId: undefined, }; } if (action.type === SHOW_STICKER_PACK_PREVIEW) { return { ...state, stickerPackPreviewId: action.payload, }; } if (action.type === CLOSE_ERROR_MODAL) { return { ...state, errorModalProps: undefined, }; } if (action.type === SHOW_ERROR_MODAL) { return { ...state, errorModalProps: action.payload, }; } if (action.type === CLOSE_SHORTCUT_GUIDE_MODAL) { return { ...state, isShortcutGuideModalVisible: false, }; } if (action.type === SHOW_SHORTCUT_GUIDE_MODAL) { return { ...state, isShortcutGuideModalVisible: true, }; } return state; }