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.
session-desktop/ts/components/SessionSearchInput.tsx

121 lines
3.2 KiB
TypeScript

1 year ago
import { Dispatch } from '@reduxjs/toolkit';
import { debounce } from 'lodash';
import { useRef, useState } from 'react';
import { useDispatch, useSelector } from 'react-redux';
import styled from 'styled-components';
import { clearSearch, search, updateSearchTerm } from '../state/ducks/search';
import { getConversationsCount } from '../state/selectors/conversations';
import { getLeftOverlayMode } from '../state/selectors/section';
import { SessionIconButton } from './icon';
import { useHotkey } from '../hooks/useHotkey';
3 years ago
const StyledSearchInput = styled.div`
height: var(--search-input-height);
width: 100%;
margin-inline-end: 1px;
margin-bottom: 10px;
display: inline-flex;
flex-shrink: 0;
.session-icon-button {
margin: auto 10px;
&:hover svg path {
fill: var(--search-bar-icon-hover-color);
}
}
&:hover {
svg path:first-child {
fill: var(--search-bar-icon-hover-color);
}
3 years ago
}
`;
const StyledInput = styled.input`
width: inherit;
height: inherit;
border: none;
flex-grow: 1;
font-size: var(--font-size-sm);
font-family: var(--font-default);
text-overflow: ellipsis;
background: none;
color: var(--search-bar-text-control-color);
3 years ago
&:focus {
color: var(--search-bar-text-user-color);
3 years ago
outline: none !important;
}
`;
const doTheSearch = (dispatch: Dispatch<any>, cleanedTerm: string) => {
dispatch(search(cleanedTerm));
};
const debouncedSearch = debounce(doTheSearch, 50);
function updateSearch(dispatch: Dispatch<any>, searchTerm: string) {
if (!searchTerm) {
dispatch(clearSearch());
return;
}
// this updates our current state and text field.
dispatch(updateSearchTerm(searchTerm));
debouncedSearch(dispatch, searchTerm);
}
export const SessionSearchInput = () => {
const [currentSearchTerm, setCurrentSearchTerm] = useState('');
const dispatch = useDispatch();
const isGroupCreationSearch = useSelector(getLeftOverlayMode) === 'closed-group';
const convoCount = useSelector(getConversationsCount);
const inputRef = useRef<HTMLInputElement>(null);
useHotkey('Escape', () => {
if (inputRef.current !== null && inputRef.current === document.activeElement) {
setCurrentSearchTerm('');
dispatch(clearSearch());
}
});
// just after onboard we only have a conversation with ourself
if (convoCount <= 1) {
return null;
}
const placeholder = isGroupCreationSearch ? window.i18n('searchContacts') : window.i18n('search');
return (
<StyledSearchInput>
<SessionIconButton
iconColor="var(--search-bar-icon-color)"
iconSize="medium"
iconType="search"
/>
<StyledInput
ref={inputRef}
value={currentSearchTerm}
onChange={e => {
const inputValue = e.target.value;
setCurrentSearchTerm(inputValue);
updateSearch(dispatch, inputValue);
}}
placeholder={placeholder}
/>
{Boolean(currentSearchTerm.length) && (
<SessionIconButton
iconColor="var(--search-bar-icon-color)"
iconSize="tiny"
iconType="exit"
onClick={() => {
setCurrentSearchTerm('');
dispatch(clearSearch());
}}
/>
)}
</StyledSearchInput>
);
};