// Copyright 2019-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import React, { useEffect, useCallback, useMemo, useState } from 'react'; import type { MeasuredComponentProps } from 'react-measure'; import Measure from 'react-measure'; import classNames from 'classnames'; import { clamp, isNumber, noop } from 'lodash'; import type { LeftPaneHelper, ToFindType } from './leftPane/LeftPaneHelper'; import { FindDirection } from './leftPane/LeftPaneHelper'; import type { LeftPaneInboxPropsType } from './leftPane/LeftPaneInboxHelper'; import { LeftPaneInboxHelper } from './leftPane/LeftPaneInboxHelper'; import type { LeftPaneSearchPropsType } from './leftPane/LeftPaneSearchHelper'; import { LeftPaneSearchHelper } from './leftPane/LeftPaneSearchHelper'; import type { LeftPaneArchivePropsType } from './leftPane/LeftPaneArchiveHelper'; import { LeftPaneArchiveHelper } from './leftPane/LeftPaneArchiveHelper'; import type { LeftPaneComposePropsType } from './leftPane/LeftPaneComposeHelper'; import { LeftPaneComposeHelper } from './leftPane/LeftPaneComposeHelper'; import type { LeftPaneChooseGroupMembersPropsType } from './leftPane/LeftPaneChooseGroupMembersHelper'; import { LeftPaneChooseGroupMembersHelper } from './leftPane/LeftPaneChooseGroupMembersHelper'; import type { LeftPaneSetGroupMetadataPropsType } from './leftPane/LeftPaneSetGroupMetadataHelper'; import { LeftPaneSetGroupMetadataHelper } from './leftPane/LeftPaneSetGroupMetadataHelper'; import * as OS from '../OS'; import type { LocalizerType, ThemeType } from '../types/Util'; import { ScrollBehavior } from '../types/Util'; import type { PreferredBadgeSelectorType } from '../state/selectors/badges'; import { usePrevious } from '../hooks/usePrevious'; import { missingCaseError } from '../util/missingCaseError'; import { strictAssert } from '../util/assert'; import { isSorted } from '../util/isSorted'; import type { WidthBreakpoint } from './_util'; import { getConversationListWidthBreakpoint } from './_util'; import { ConversationList } from './ConversationList'; import { ContactCheckboxDisabledReason } from './conversationList/ContactCheckbox'; import type { DeleteAvatarFromDiskActionType, ReplaceAvatarActionType, SaveAvatarToDiskActionType, } from '../types/Avatar'; const MIN_WIDTH = 97; const SNAP_WIDTH = 200; const MIN_FULL_WIDTH = 280; const MAX_WIDTH = 380; strictAssert( isSorted([MIN_WIDTH, SNAP_WIDTH, MIN_FULL_WIDTH, MAX_WIDTH]), 'Expected widths to be in the right order' ); export enum LeftPaneMode { Inbox, Search, Archive, Compose, ChooseGroupMembers, SetGroupMetadata, } export type PropsType = { // These help prevent invalid states. For example, we don't need the list of pinned // conversations if we're trying to start a new conversation. Ideally these would be // at the top level, but this is not supported by react-redux + TypeScript. modeSpecificProps: | ({ mode: LeftPaneMode.Inbox; } & LeftPaneInboxPropsType) | ({ mode: LeftPaneMode.Search; } & LeftPaneSearchPropsType) | ({ mode: LeftPaneMode.Archive; } & LeftPaneArchivePropsType) | ({ mode: LeftPaneMode.Compose; } & LeftPaneComposePropsType) | ({ mode: LeftPaneMode.ChooseGroupMembers; } & LeftPaneChooseGroupMembersPropsType) | ({ mode: LeftPaneMode.SetGroupMetadata; } & LeftPaneSetGroupMetadataPropsType); getPreferredBadge: PreferredBadgeSelectorType; i18n: LocalizerType; preferredWidthFromStorage: number; selectedConversationId: undefined | string; selectedMessageId: undefined | string; regionCode: string; challengeStatus: 'idle' | 'required' | 'pending'; setChallengeStatus: (status: 'idle') => void; crashReportCount: number; theme: ThemeType; // Action Creators cantAddContactToGroup: (conversationId: string) => void; clearConversationSearch: () => void; clearGroupCreationError: () => void; clearSearch: () => void; closeCantAddContactToGroupModal: () => void; closeMaximumGroupSizeModal: () => void; closeRecommendedGroupSizeModal: () => void; createGroup: () => void; startNewConversationFromPhoneNumber: (e164: string) => void; startNewConversationFromUsername: (username: string) => void; openConversationInternal: (_: { conversationId: string; messageId?: string; switchToAssociatedView?: boolean; }) => void; savePreferredLeftPaneWidth: (_: number) => void; searchInConversation: (conversationId: string) => unknown; setComposeSearchTerm: (composeSearchTerm: string) => void; setComposeGroupAvatar: (_: undefined | Uint8Array) => void; setComposeGroupName: (_: string) => void; setComposeGroupExpireTimer: (_: number) => void; showArchivedConversations: () => void; showInbox: () => void; startComposing: () => void; startSearch: () => unknown; showChooseGroupMembers: () => void; startSettingGroupMetadata: () => void; toggleConversationInChooseMembers: (conversationId: string) => void; composeDeleteAvatarFromDisk: DeleteAvatarFromDiskActionType; composeReplaceAvatar: ReplaceAvatarActionType; composeSaveAvatarToDisk: SaveAvatarToDiskActionType; toggleComposeEditingAvatar: () => unknown; updateSearchTerm: (_: string) => void; // Render Props renderExpiredBuildDialog: ( _: Readonly<{ containerWidthBreakpoint: WidthBreakpoint }> ) => JSX.Element; renderMainHeader: () => JSX.Element; renderMessageSearchResult: (id: string) => JSX.Element; renderNetworkStatus: ( _: Readonly<{ containerWidthBreakpoint: WidthBreakpoint }> ) => JSX.Element; renderRelinkDialog: ( _: Readonly<{ containerWidthBreakpoint: WidthBreakpoint }> ) => JSX.Element; renderUpdateDialog: ( _: Readonly<{ containerWidthBreakpoint: WidthBreakpoint }> ) => JSX.Element; renderCaptchaDialog: (props: { onSkip(): void }) => JSX.Element; renderCrashReportDialog: () => JSX.Element; }; export const LeftPane: React.FC = ({ cantAddContactToGroup, challengeStatus, crashReportCount, clearConversationSearch, clearGroupCreationError, clearSearch, closeCantAddContactToGroupModal, closeMaximumGroupSizeModal, closeRecommendedGroupSizeModal, composeDeleteAvatarFromDisk, composeReplaceAvatar, composeSaveAvatarToDisk, createGroup, getPreferredBadge, i18n, modeSpecificProps, openConversationInternal, preferredWidthFromStorage, renderCaptchaDialog, renderCrashReportDialog, renderExpiredBuildDialog, renderMainHeader, renderMessageSearchResult, renderNetworkStatus, renderRelinkDialog, renderUpdateDialog, savePreferredLeftPaneWidth, searchInConversation, selectedConversationId, selectedMessageId, setChallengeStatus, setComposeGroupAvatar, setComposeGroupExpireTimer, setComposeGroupName, setComposeSearchTerm, showArchivedConversations, showChooseGroupMembers, showInbox, startComposing, startSearch, startNewConversationFromPhoneNumber, startNewConversationFromUsername, startSettingGroupMetadata, theme, toggleComposeEditingAvatar, toggleConversationInChooseMembers, updateSearchTerm, }) => { const [preferredWidth, setPreferredWidth] = useState( // This clamp is present just in case we get a bogus value from storage. clamp(preferredWidthFromStorage, MIN_WIDTH, MAX_WIDTH) ); const [isResizing, setIsResizing] = useState(false); const previousModeSpecificProps = usePrevious( modeSpecificProps, modeSpecificProps ); // The left pane can be in various modes: the inbox, the archive, the composer, etc. // Ideally, this would render subcomponents such as `` or // `` (and if there's a way to do that cleanly, we should refactor // this). // // But doing that presents two problems: // // 1. Different components render the same logical inputs (the main header's search), // but React doesn't know that they're the same, so you can lose focus as you change // modes. // 2. These components render virtualized lists, which are somewhat slow to initialize. // Switching between modes can cause noticable hiccups. // // To get around those problems, we use "helpers" which all correspond to the same // interface. // // Unfortunately, there's a little bit of repetition here because TypeScript isn't quite // smart enough. let helper: LeftPaneHelper; let shouldRecomputeRowHeights: boolean; switch (modeSpecificProps.mode) { case LeftPaneMode.Inbox: { const inboxHelper = new LeftPaneInboxHelper(modeSpecificProps); shouldRecomputeRowHeights = previousModeSpecificProps.mode === modeSpecificProps.mode ? inboxHelper.shouldRecomputeRowHeights(previousModeSpecificProps) : true; helper = inboxHelper; break; } case LeftPaneMode.Search: { const searchHelper = new LeftPaneSearchHelper(modeSpecificProps); shouldRecomputeRowHeights = previousModeSpecificProps.mode === modeSpecificProps.mode ? searchHelper.shouldRecomputeRowHeights(previousModeSpecificProps) : true; helper = searchHelper; break; } case LeftPaneMode.Archive: { const archiveHelper = new LeftPaneArchiveHelper(modeSpecificProps); shouldRecomputeRowHeights = previousModeSpecificProps.mode === modeSpecificProps.mode ? archiveHelper.shouldRecomputeRowHeights(previousModeSpecificProps) : true; helper = archiveHelper; break; } case LeftPaneMode.Compose: { const composeHelper = new LeftPaneComposeHelper(modeSpecificProps); shouldRecomputeRowHeights = previousModeSpecificProps.mode === modeSpecificProps.mode ? composeHelper.shouldRecomputeRowHeights(previousModeSpecificProps) : true; helper = composeHelper; break; } case LeftPaneMode.ChooseGroupMembers: { const chooseGroupMembersHelper = new LeftPaneChooseGroupMembersHelper( modeSpecificProps ); shouldRecomputeRowHeights = previousModeSpecificProps.mode === modeSpecificProps.mode ? chooseGroupMembersHelper.shouldRecomputeRowHeights( previousModeSpecificProps ) : true; helper = chooseGroupMembersHelper; break; } case LeftPaneMode.SetGroupMetadata: { const setGroupMetadataHelper = new LeftPaneSetGroupMetadataHelper( modeSpecificProps ); shouldRecomputeRowHeights = previousModeSpecificProps.mode === modeSpecificProps.mode ? setGroupMetadataHelper.shouldRecomputeRowHeights( previousModeSpecificProps ) : true; helper = setGroupMetadataHelper; break; } default: throw missingCaseError(modeSpecificProps); } useEffect(() => { const onKeyDown = (event: KeyboardEvent) => { const { ctrlKey, shiftKey, altKey, metaKey, key } = event; const commandOrCtrl = OS.isMacOS() ? metaKey : ctrlKey; if (event.key === 'Escape') { const backAction = helper.getBackAction({ showInbox, startComposing, showChooseGroupMembers, }); if (backAction) { event.preventDefault(); event.stopPropagation(); backAction(); return; } } if ( commandOrCtrl && !shiftKey && !altKey && (key === 'n' || key === 'N') ) { startComposing(); event.preventDefault(); event.stopPropagation(); return; } let conversationToOpen: | undefined | { conversationId: string; messageId?: string; }; const numericIndex = keyboardKeyToNumericIndex(event.key); if (commandOrCtrl && isNumber(numericIndex)) { conversationToOpen = helper.getConversationAndMessageAtIndex(numericIndex); } else { let toFind: undefined | ToFindType; if ( (altKey && !shiftKey && key === 'ArrowUp') || (commandOrCtrl && shiftKey && key === '[') || (ctrlKey && shiftKey && key === 'Tab') ) { toFind = { direction: FindDirection.Up, unreadOnly: false }; } else if ( (altKey && !shiftKey && key === 'ArrowDown') || (commandOrCtrl && shiftKey && key === ']') || (ctrlKey && key === 'Tab') ) { toFind = { direction: FindDirection.Down, unreadOnly: false }; } else if (altKey && shiftKey && key === 'ArrowUp') { toFind = { direction: FindDirection.Up, unreadOnly: true }; } else if (altKey && shiftKey && key === 'ArrowDown') { toFind = { direction: FindDirection.Down, unreadOnly: true }; } if (toFind) { conversationToOpen = helper.getConversationAndMessageInDirection( toFind, selectedConversationId, selectedMessageId ); } } if (conversationToOpen) { const { conversationId, messageId } = conversationToOpen; openConversationInternal({ conversationId, messageId }); event.preventDefault(); event.stopPropagation(); } helper.onKeyDown(event, { searchInConversation, selectedConversationId, startSearch, }); }; document.addEventListener('keydown', onKeyDown); return () => { document.removeEventListener('keydown', onKeyDown); }; }, [ helper, openConversationInternal, searchInConversation, selectedConversationId, selectedMessageId, showChooseGroupMembers, showInbox, startComposing, startSearch, ]); const requiresFullWidth = helper.requiresFullWidth(); useEffect(() => { if (!isResizing) { return noop; } const onMouseMove = (event: MouseEvent) => { let width: number; if (requiresFullWidth) { width = Math.max(event.clientX, MIN_FULL_WIDTH); } else if (event.clientX < SNAP_WIDTH) { width = MIN_WIDTH; } else { width = clamp(event.clientX, MIN_FULL_WIDTH, MAX_WIDTH); } setPreferredWidth(Math.min(width, MAX_WIDTH)); event.preventDefault(); }; const stopResizing = () => { setIsResizing(false); }; document.body.addEventListener('mousemove', onMouseMove); document.body.addEventListener('mouseup', stopResizing); document.body.addEventListener('mouseleave', stopResizing); return () => { document.body.removeEventListener('mousemove', onMouseMove); document.body.removeEventListener('mouseup', stopResizing); document.body.removeEventListener('mouseleave', stopResizing); }; }, [isResizing, requiresFullWidth]); useEffect(() => { if (!isResizing) { return noop; } document.body.classList.add('is-resizing-left-pane'); return () => { document.body.classList.remove('is-resizing-left-pane'); }; }, [isResizing]); useEffect(() => { if (isResizing || preferredWidth === preferredWidthFromStorage) { return; } const timeout = setTimeout(() => { savePreferredLeftPaneWidth(preferredWidth); }, 1000); return () => { clearTimeout(timeout); }; }, [ isResizing, preferredWidth, preferredWidthFromStorage, savePreferredLeftPaneWidth, ]); const preRowsNode = helper.getPreRowsNode({ clearConversationSearch, clearGroupCreationError, clearSearch, closeCantAddContactToGroupModal, closeMaximumGroupSizeModal, closeRecommendedGroupSizeModal, composeDeleteAvatarFromDisk, composeReplaceAvatar, composeSaveAvatarToDisk, createGroup, i18n, removeSelectedContact: toggleConversationInChooseMembers, setComposeGroupAvatar, setComposeGroupExpireTimer, setComposeGroupName, toggleComposeEditingAvatar, }); const footerContents = helper.getFooterContents({ createGroup, i18n, startSettingGroupMetadata, }); const getRow = useMemo(() => helper.getRow.bind(helper), [helper]); const onSelectConversation = useCallback( (conversationId: string, messageId?: string) => { openConversationInternal({ conversationId, messageId, switchToAssociatedView: true, }); }, [openConversationInternal] ); const previousSelectedConversationId = usePrevious( selectedConversationId, selectedConversationId ); let width: number; if (requiresFullWidth || preferredWidth >= SNAP_WIDTH) { width = Math.max(preferredWidth, MIN_FULL_WIDTH); } else { width = MIN_WIDTH; } const isScrollable = helper.isScrollable(); let rowIndexToScrollTo: undefined | number; let scrollBehavior: ScrollBehavior; if (isScrollable) { rowIndexToScrollTo = previousSelectedConversationId === selectedConversationId ? undefined : helper.getRowIndexToScrollTo(selectedConversationId); scrollBehavior = ScrollBehavior.Default; } else { rowIndexToScrollTo = 0; scrollBehavior = ScrollBehavior.Hard; } // We ensure that the listKey differs between some modes (e.g. inbox/archived), ensuring // that AutoSizer properly detects the new size of its slot in the flexbox. The // archive explainer text at the top of the archive view causes problems otherwise. // It also ensures that we scroll to the top when switching views. const listKey = preRowsNode ? 1 : 0; const widthBreakpoint = getConversationListWidthBreakpoint(width); // We disable this lint rule because we're trying to capture bubbled events. See [the // lint rule's docs][0]. // // [0]: https://github.com/jsx-eslint/eslint-plugin-jsx-a11y/blob/645900a0e296ca7053dbf6cd9e12cc85849de2d5/docs/rules/no-static-element-interactions.md#case-the-event-handler-is-only-being-used-to-capture-bubbled-events /* eslint-disable jsx-a11y/no-static-element-interactions */ return (
{/* eslint-enable jsx-a11y/no-static-element-interactions */}
{helper.getHeaderContents({ i18n, showInbox, startComposing, showChooseGroupMembers, }) || renderMainHeader()}
{helper.getSearchInput({ clearConversationSearch, clearSearch, i18n, onChangeComposeSearchTerm: event => { setComposeSearchTerm(event.target.value); }, updateSearchTerm, })}
{renderExpiredBuildDialog({ containerWidthBreakpoint: widthBreakpoint, })} {renderRelinkDialog({ containerWidthBreakpoint: widthBreakpoint })} {renderNetworkStatus({ containerWidthBreakpoint: widthBreakpoint })} {renderUpdateDialog({ containerWidthBreakpoint: widthBreakpoint })}
{preRowsNode && {preRowsNode}} {({ contentRect, measureRef }: MeasuredComponentProps) => (
{ switch (disabledReason) { case undefined: toggleConversationInChooseMembers(conversationId); break; case ContactCheckboxDisabledReason.AlreadyAdded: case ContactCheckboxDisabledReason.MaximumContactsSelected: // These are no-ops. break; case ContactCheckboxDisabledReason.NotCapable: cantAddContactToGroup(conversationId); break; default: throw missingCaseError(disabledReason); } }} onSelectConversation={onSelectConversation} renderMessageSearchResult={renderMessageSearchResult} rowCount={helper.getRowCount()} scrollBehavior={scrollBehavior} scrollToRowIndex={rowIndexToScrollTo} scrollable={isScrollable} shouldRecomputeRowHeights={shouldRecomputeRowHeights} showChooseGroupMembers={showChooseGroupMembers} startNewConversationFromPhoneNumber={ startNewConversationFromPhoneNumber } startNewConversationFromUsername={ startNewConversationFromUsername } theme={theme} />
)}
{footerContents && (
{footerContents}
)} <> {/* eslint-disable-next-line jsx-a11y/no-static-element-interactions */}
{ setIsResizing(true); }} /> {challengeStatus !== 'idle' && renderCaptchaDialog({ onSkip() { setChallengeStatus('idle'); }, })} {crashReportCount > 0 && renderCrashReportDialog()}
); }; function keyboardKeyToNumericIndex(key: string): undefined | number { if (key.length !== 1) { return undefined; } const result = parseInt(key, 10) - 1; const isValidIndex = Number.isInteger(result) && result >= 0 && result <= 8; return isValidIndex ? result : undefined; }