2023-01-03 19:55:46 +00:00
|
|
|
// Copyright 2021 Signal Messenger, LLC
|
2021-11-01 18:43:02 +00:00
|
|
|
// SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2024-11-19 17:26:51 +00:00
|
|
|
import React, { useEffect, useMemo, useRef } from 'react';
|
2024-11-13 19:33:41 +00:00
|
|
|
import classNames from 'classnames';
|
2022-04-30 05:24:20 +00:00
|
|
|
import type {
|
|
|
|
ConversationType,
|
2022-06-16 19:12:50 +00:00
|
|
|
ShowConversationType,
|
2022-04-30 05:24:20 +00:00
|
|
|
} from '../state/ducks/conversations';
|
2021-11-01 18:43:02 +00:00
|
|
|
import type { LocalizerType } from '../types/Util';
|
|
|
|
import { Avatar, AvatarSize } from './Avatar';
|
2022-02-14 17:57:11 +00:00
|
|
|
import { SearchInput } from './SearchInput';
|
|
|
|
import { usePrevious } from '../hooks/usePrevious';
|
2024-11-13 19:33:41 +00:00
|
|
|
import { Tooltip, TooltipPlacement } from './Tooltip';
|
|
|
|
import { Theme } from '../util/theme';
|
2024-11-19 17:26:51 +00:00
|
|
|
import { isProduction } from '../util/version';
|
2021-11-01 18:43:02 +00:00
|
|
|
|
2024-11-13 19:33:41 +00:00
|
|
|
type BasePropsType = {
|
2022-02-14 17:57:11 +00:00
|
|
|
clearConversationSearch: () => void;
|
2024-11-13 19:33:41 +00:00
|
|
|
clearSearchQuery: () => void;
|
2021-11-01 18:43:02 +00:00
|
|
|
disabled?: boolean;
|
2024-03-09 01:08:30 +00:00
|
|
|
endConversationSearch: () => void;
|
|
|
|
endSearch: () => void;
|
2021-11-01 18:43:02 +00:00
|
|
|
i18n: LocalizerType;
|
2024-03-09 01:08:30 +00:00
|
|
|
isSearchingGlobally: boolean;
|
2022-04-30 05:24:20 +00:00
|
|
|
onEnterKeyDown?: (
|
2024-11-13 19:33:41 +00:00
|
|
|
clearSearchQuery: () => void,
|
2022-06-16 19:12:50 +00:00
|
|
|
showConversation: ShowConversationType
|
2022-04-30 05:24:20 +00:00
|
|
|
) => void;
|
2022-12-14 19:05:32 +00:00
|
|
|
searchConversation?: ConversationType;
|
|
|
|
searchTerm: string;
|
|
|
|
showConversation: ShowConversationType;
|
|
|
|
startSearchCounter: number;
|
|
|
|
updateSearchTerm: (searchTerm: string) => void;
|
2021-11-01 18:43:02 +00:00
|
|
|
};
|
|
|
|
|
2024-11-13 19:33:41 +00:00
|
|
|
type NoFilterPropsType = BasePropsType & {
|
|
|
|
filterButtonEnabled?: false;
|
|
|
|
filterPressed?: false;
|
|
|
|
onFilterClick?: () => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
type WithFilterPropsType = BasePropsType & {
|
|
|
|
filterButtonEnabled: boolean;
|
|
|
|
filterPressed: boolean;
|
|
|
|
onFilterClick: (enabled: boolean) => void;
|
|
|
|
};
|
|
|
|
|
|
|
|
type PropsType = NoFilterPropsType | WithFilterPropsType;
|
|
|
|
|
2022-11-18 00:45:19 +00:00
|
|
|
export function LeftPaneSearchInput({
|
2022-02-14 17:57:11 +00:00
|
|
|
clearConversationSearch,
|
2024-11-13 19:33:41 +00:00
|
|
|
clearSearchQuery,
|
2022-02-14 17:57:11 +00:00
|
|
|
disabled,
|
2024-03-09 01:08:30 +00:00
|
|
|
endConversationSearch,
|
|
|
|
endSearch,
|
2022-02-14 17:57:11 +00:00
|
|
|
i18n,
|
2024-03-09 01:08:30 +00:00
|
|
|
isSearchingGlobally,
|
2022-12-14 19:05:32 +00:00
|
|
|
onEnterKeyDown,
|
2022-02-14 17:57:11 +00:00
|
|
|
searchConversation,
|
|
|
|
searchTerm,
|
2022-12-14 19:05:32 +00:00
|
|
|
showConversation,
|
2022-02-14 17:57:11 +00:00
|
|
|
startSearchCounter,
|
|
|
|
updateSearchTerm,
|
2024-11-13 19:33:41 +00:00
|
|
|
filterButtonEnabled = false,
|
|
|
|
filterPressed = false,
|
|
|
|
onFilterClick,
|
2022-11-18 00:45:19 +00:00
|
|
|
}: PropsType): JSX.Element {
|
2022-02-14 17:57:11 +00:00
|
|
|
const inputRef = useRef<null | HTMLInputElement>(null);
|
2021-11-01 18:43:02 +00:00
|
|
|
|
2022-02-14 17:57:11 +00:00
|
|
|
const prevSearchConversationId = usePrevious(
|
|
|
|
undefined,
|
|
|
|
searchConversation?.id
|
|
|
|
);
|
|
|
|
const prevSearchCounter = usePrevious(startSearchCounter, startSearchCounter);
|
2024-03-09 01:08:30 +00:00
|
|
|
const wasSearchingGlobally = usePrevious(false, isSearchingGlobally);
|
2021-11-01 18:43:02 +00:00
|
|
|
|
2022-02-14 17:57:11 +00:00
|
|
|
useEffect(() => {
|
|
|
|
// When user chooses to search in a given conversation we focus the field for them
|
|
|
|
if (
|
|
|
|
searchConversation &&
|
|
|
|
searchConversation.id !== prevSearchConversationId
|
|
|
|
) {
|
|
|
|
inputRef.current?.focus();
|
|
|
|
}
|
|
|
|
// When user chooses to start a new search, we focus the field
|
2024-03-09 01:08:30 +00:00
|
|
|
if (
|
|
|
|
(isSearchingGlobally && !wasSearchingGlobally) ||
|
|
|
|
startSearchCounter !== prevSearchCounter
|
|
|
|
) {
|
2022-02-14 17:57:11 +00:00
|
|
|
inputRef.current?.select();
|
|
|
|
}
|
|
|
|
}, [
|
|
|
|
prevSearchConversationId,
|
|
|
|
prevSearchCounter,
|
|
|
|
searchConversation,
|
|
|
|
startSearchCounter,
|
2024-03-09 01:08:30 +00:00
|
|
|
isSearchingGlobally,
|
|
|
|
wasSearchingGlobally,
|
2022-02-14 17:57:11 +00:00
|
|
|
]);
|
2021-11-01 18:43:02 +00:00
|
|
|
|
2022-02-14 17:57:11 +00:00
|
|
|
const changeValue = (nextSearchTerm: string) => {
|
|
|
|
if (!nextSearchTerm) {
|
|
|
|
if (searchConversation) {
|
|
|
|
clearConversationSearch();
|
|
|
|
} else {
|
2024-11-13 19:33:41 +00:00
|
|
|
clearSearchQuery();
|
2022-02-14 17:57:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (updateSearchTerm) {
|
|
|
|
updateSearchTerm(nextSearchTerm);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2024-11-13 19:33:41 +00:00
|
|
|
let label: string;
|
|
|
|
if (searchConversation) {
|
|
|
|
label = i18n('icu:searchIn');
|
|
|
|
} else if (filterPressed) {
|
|
|
|
label = i18n('icu:searchUnreadChats');
|
|
|
|
} else {
|
|
|
|
label = i18n('icu:search');
|
|
|
|
}
|
2022-02-14 17:57:11 +00:00
|
|
|
|
2024-11-19 17:26:51 +00:00
|
|
|
const eligibleToShowFilterByUnread = useMemo(
|
|
|
|
() => window.getVersion && !isProduction(window.getVersion()),
|
|
|
|
[]
|
|
|
|
);
|
|
|
|
|
2022-02-14 17:57:11 +00:00
|
|
|
return (
|
2024-11-13 19:33:41 +00:00
|
|
|
<>
|
|
|
|
<SearchInput
|
|
|
|
disabled={disabled}
|
|
|
|
label={label}
|
|
|
|
hasSearchIcon={!searchConversation}
|
|
|
|
i18n={i18n}
|
|
|
|
moduleClassName="LeftPaneSearchInput"
|
|
|
|
onBlur={() => {
|
|
|
|
if (!searchConversation && !searchTerm) {
|
|
|
|
endSearch();
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
onKeyDown={event => {
|
|
|
|
if (onEnterKeyDown && event.key === 'Enter') {
|
|
|
|
onEnterKeyDown(clearSearchQuery, showConversation);
|
|
|
|
event.preventDefault();
|
|
|
|
event.stopPropagation();
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
onChange={event => {
|
|
|
|
changeValue(event.currentTarget.value);
|
|
|
|
}}
|
|
|
|
onClear={() => {
|
|
|
|
if (searchTerm) {
|
|
|
|
clearSearchQuery();
|
|
|
|
inputRef.current?.focus();
|
|
|
|
} else if (searchConversation) {
|
|
|
|
endConversationSearch();
|
2022-02-14 17:57:11 +00:00
|
|
|
inputRef.current?.focus();
|
2024-11-13 19:33:41 +00:00
|
|
|
} else {
|
|
|
|
inputRef.current?.blur();
|
|
|
|
}
|
|
|
|
}}
|
|
|
|
ref={inputRef}
|
|
|
|
placeholder={label}
|
|
|
|
value={searchTerm}
|
|
|
|
>
|
|
|
|
{searchConversation && (
|
|
|
|
// Clicking the non-X part of the pill should focus the input but have a normal
|
|
|
|
// cursor. This effectively simulates `pointer-events: none` while still
|
|
|
|
// letting us change the cursor.
|
|
|
|
// eslint-disable-next-line max-len
|
|
|
|
// eslint-disable-next-line jsx-a11y/click-events-have-key-events, jsx-a11y/no-static-element-interactions
|
|
|
|
<div
|
|
|
|
className="LeftPaneSearchInput__in-conversation-pill"
|
|
|
|
onClick={() => {
|
|
|
|
inputRef.current?.focus();
|
|
|
|
}}
|
|
|
|
>
|
|
|
|
<Avatar
|
|
|
|
acceptedMessageRequest={searchConversation.acceptedMessageRequest}
|
|
|
|
avatarUrl={searchConversation.avatarUrl}
|
|
|
|
badge={undefined}
|
|
|
|
color={searchConversation.color}
|
|
|
|
conversationType={searchConversation.type}
|
|
|
|
i18n={i18n}
|
|
|
|
isMe={searchConversation.isMe}
|
|
|
|
noteToSelf={searchConversation.isMe}
|
|
|
|
sharedGroupNames={searchConversation.sharedGroupNames}
|
|
|
|
size={AvatarSize.TWENTY}
|
|
|
|
title={searchConversation.title}
|
|
|
|
unblurredAvatarUrl={searchConversation.unblurredAvatarUrl}
|
|
|
|
/>
|
|
|
|
<button
|
|
|
|
aria-label={i18n('icu:clearSearch')}
|
|
|
|
className="LeftPaneSearchInput__in-conversation-pill__x-button"
|
|
|
|
onClick={endConversationSearch}
|
|
|
|
type="button"
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
)}
|
|
|
|
</SearchInput>
|
2024-11-19 17:26:51 +00:00
|
|
|
{filterButtonEnabled && eligibleToShowFilterByUnread && (
|
2024-11-13 19:33:41 +00:00
|
|
|
<Tooltip
|
|
|
|
direction={TooltipPlacement.Bottom}
|
|
|
|
content={i18n('icu:filterByUnreadButtonLabel')}
|
|
|
|
theme={Theme.Dark}
|
|
|
|
delay={600}
|
2022-02-14 17:57:11 +00:00
|
|
|
>
|
2021-11-01 18:43:02 +00:00
|
|
|
<button
|
2024-11-13 19:33:41 +00:00
|
|
|
className={classNames('LeftPaneSearchInput__FilterButton', {
|
|
|
|
'LeftPaneSearchInput__FilterButton--pressed': filterPressed,
|
|
|
|
})}
|
2021-11-01 18:43:02 +00:00
|
|
|
type="button"
|
2024-11-13 19:33:41 +00:00
|
|
|
aria-pressed={filterPressed}
|
|
|
|
onClick={() => onFilterClick?.(!filterPressed)}
|
|
|
|
>
|
|
|
|
<span className="LeftPaneSearchInput__FilterLabel">
|
|
|
|
{i18n('icu:filterByUnreadButtonLabel')}
|
|
|
|
</span>
|
|
|
|
</button>
|
|
|
|
</Tooltip>
|
2022-02-14 17:57:11 +00:00
|
|
|
)}
|
2024-11-13 19:33:41 +00:00
|
|
|
</>
|
2022-02-14 17:57:11 +00:00
|
|
|
);
|
2022-11-18 00:45:19 +00:00
|
|
|
}
|