2023-02-06 10:01:03 -08:00
|
|
|
import clsx from 'clsx';
|
2022-03-21 11:09:01 -07:00
|
|
|
import debounce from 'lodash/debounce';
|
2023-05-04 06:24:37 -07:00
|
|
|
import React, { useCallback, useEffect } from 'react';
|
2022-03-21 11:09:01 -07:00
|
|
|
import { defineMessages, useIntl } from 'react-intl';
|
2022-03-22 05:42:26 -07:00
|
|
|
import { useHistory } from 'react-router-dom';
|
2022-03-21 11:09:01 -07:00
|
|
|
|
|
|
|
import {
|
|
|
|
changeSearch,
|
|
|
|
clearSearch,
|
2022-09-09 14:20:26 -07:00
|
|
|
clearSearchResults,
|
2022-08-08 08:36:11 -07:00
|
|
|
setSearchAccount,
|
2022-03-21 11:09:01 -07:00
|
|
|
showSearch,
|
2022-08-08 08:36:11 -07:00
|
|
|
submitSearch,
|
2022-05-24 03:24:26 -07:00
|
|
|
} from 'soapbox/actions/search';
|
2022-11-15 06:10:14 -08:00
|
|
|
import AutosuggestAccountInput from 'soapbox/components/autosuggest-account-input';
|
2022-10-04 14:00:14 -07:00
|
|
|
import { Input } from 'soapbox/components/ui';
|
2022-05-24 03:24:26 -07:00
|
|
|
import SvgIcon from 'soapbox/components/ui/icon/svg-icon';
|
2023-01-09 14:13:12 -08:00
|
|
|
import { useAppDispatch, useAppSelector } from 'soapbox/hooks';
|
|
|
|
import { AppDispatch, RootState } from 'soapbox/store';
|
2022-03-21 11:09:01 -07:00
|
|
|
|
|
|
|
const messages = defineMessages({
|
|
|
|
placeholder: { id: 'search.placeholder', defaultMessage: 'Search' },
|
|
|
|
action: { id: 'search.action', defaultMessage: 'Search for “{query}”' },
|
|
|
|
});
|
|
|
|
|
2022-06-04 15:47:40 -07:00
|
|
|
function redirectToAccount(accountId: string, routerHistory: any) {
|
2023-01-09 14:13:12 -08:00
|
|
|
return (_dispatch: AppDispatch, getState: () => RootState) => {
|
2022-03-21 11:09:01 -07:00
|
|
|
const acct = getState().getIn(['accounts', accountId, 'acct']);
|
|
|
|
|
|
|
|
if (acct && routerHistory) {
|
|
|
|
routerHistory.push(`/@${acct}`);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2022-03-22 05:42:26 -07:00
|
|
|
interface ISearch {
|
2023-02-15 13:26:27 -08:00
|
|
|
autoFocus?: boolean
|
|
|
|
autoSubmit?: boolean
|
|
|
|
autosuggest?: boolean
|
2022-03-21 11:09:01 -07:00
|
|
|
openInRoute?: boolean
|
|
|
|
}
|
|
|
|
|
|
|
|
const Search = (props: ISearch) => {
|
|
|
|
const {
|
|
|
|
autoFocus = false,
|
|
|
|
autoSubmit = false,
|
|
|
|
autosuggest = false,
|
|
|
|
openInRoute = false,
|
|
|
|
} = props;
|
|
|
|
|
2023-01-09 14:13:12 -08:00
|
|
|
const dispatch = useAppDispatch();
|
2022-03-22 05:42:26 -07:00
|
|
|
const history = useHistory();
|
2022-03-21 11:09:01 -07:00
|
|
|
const intl = useIntl();
|
|
|
|
|
2022-06-07 09:25:53 -07:00
|
|
|
const value = useAppSelector((state) => state.search.value);
|
|
|
|
const submitted = useAppSelector((state) => state.search.submitted);
|
2022-03-21 11:09:01 -07:00
|
|
|
|
2022-08-08 08:36:11 -07:00
|
|
|
const debouncedSubmit = useCallback(debounce(() => {
|
2022-03-21 11:09:01 -07:00
|
|
|
dispatch(submitSearch());
|
2022-08-08 08:36:11 -07:00
|
|
|
}, 900), []);
|
2022-03-21 11:09:01 -07:00
|
|
|
|
|
|
|
const handleChange = (event: React.ChangeEvent<HTMLInputElement>) => {
|
|
|
|
const { value } = event.target;
|
|
|
|
|
|
|
|
dispatch(changeSearch(value));
|
|
|
|
|
|
|
|
if (autoSubmit) {
|
|
|
|
debouncedSubmit();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleClear = (event: React.MouseEvent<HTMLDivElement>) => {
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
if (value.length > 0 || submitted) {
|
2022-09-09 14:20:26 -07:00
|
|
|
dispatch(clearSearchResults());
|
2022-03-21 11:09:01 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleSubmit = () => {
|
|
|
|
if (openInRoute) {
|
2022-08-08 08:36:11 -07:00
|
|
|
dispatch(setSearchAccount(null));
|
|
|
|
dispatch(submitSearch());
|
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
history.push('/search');
|
2022-08-08 08:36:11 -07:00
|
|
|
} else {
|
|
|
|
dispatch(submitSearch());
|
2022-03-21 11:09:01 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleKeyDown = (event: React.KeyboardEvent<HTMLInputElement>) => {
|
|
|
|
if (event.key === 'Enter') {
|
|
|
|
event.preventDefault();
|
|
|
|
|
|
|
|
handleSubmit();
|
|
|
|
} else if (event.key === 'Escape') {
|
2022-03-24 12:27:27 -07:00
|
|
|
document.querySelector('.ui')?.parentElement?.focus();
|
2022-03-21 11:09:01 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
const handleFocus = () => {
|
|
|
|
dispatch(showSearch());
|
|
|
|
};
|
|
|
|
|
2022-06-04 15:47:40 -07:00
|
|
|
const handleSelected = (accountId: string) => {
|
2022-03-21 11:09:01 -07:00
|
|
|
dispatch(clearSearch());
|
|
|
|
dispatch(redirectToAccount(accountId, history));
|
|
|
|
};
|
|
|
|
|
|
|
|
const makeMenu = () => [
|
|
|
|
{
|
|
|
|
text: intl.formatMessage(messages.action, { query: value }),
|
2022-07-09 09:20:02 -07:00
|
|
|
icon: require('@tabler/icons/search.svg'),
|
2022-03-21 11:09:01 -07:00
|
|
|
action: handleSubmit,
|
|
|
|
},
|
|
|
|
];
|
|
|
|
|
|
|
|
const hasValue = value.length > 0 || submitted;
|
2022-08-31 11:31:59 -07:00
|
|
|
const componentProps: any = {
|
|
|
|
type: 'text',
|
|
|
|
id: 'search',
|
|
|
|
placeholder: intl.formatMessage(messages.placeholder),
|
|
|
|
value,
|
|
|
|
onChange: handleChange,
|
|
|
|
onKeyDown: handleKeyDown,
|
|
|
|
onFocus: handleFocus,
|
|
|
|
autoFocus: autoFocus,
|
2022-10-04 14:00:14 -07:00
|
|
|
theme: 'search',
|
2022-12-23 14:24:04 -08:00
|
|
|
className: 'pr-10 rtl:pl-10 rtl:pr-3',
|
2022-08-31 11:31:59 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
if (autosuggest) {
|
|
|
|
componentProps.onSelected = handleSelected;
|
|
|
|
componentProps.menu = makeMenu();
|
|
|
|
componentProps.autoSelect = false;
|
|
|
|
}
|
2022-03-21 11:09:01 -07:00
|
|
|
|
2023-05-04 06:24:37 -07:00
|
|
|
useEffect(() => {
|
|
|
|
return () => {
|
|
|
|
const newPath = history.location.pathname;
|
2023-05-31 11:34:08 -07:00
|
|
|
const shouldPersistSearch = !!newPath.match(/@.+\/posts\/[a-zA-Z0-9]+/g)
|
2023-05-04 06:24:37 -07:00
|
|
|
|| !!newPath.match(/\/tags\/.+/g);
|
|
|
|
|
|
|
|
if (!shouldPersistSearch) {
|
|
|
|
dispatch(changeSearch(''));
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}, []);
|
|
|
|
|
2022-03-21 11:09:01 -07:00
|
|
|
return (
|
|
|
|
<div className='w-full'>
|
|
|
|
<label htmlFor='search' className='sr-only'>{intl.formatMessage(messages.placeholder)}</label>
|
|
|
|
|
|
|
|
<div className='relative'>
|
2022-08-31 11:31:59 -07:00
|
|
|
{autosuggest ? (
|
|
|
|
<AutosuggestAccountInput {...componentProps} />
|
|
|
|
) : (
|
2022-10-04 14:00:14 -07:00
|
|
|
<Input {...componentProps} />
|
2022-08-31 11:31:59 -07:00
|
|
|
)}
|
2022-03-21 11:09:01 -07:00
|
|
|
|
|
|
|
<div
|
|
|
|
role='button'
|
|
|
|
tabIndex={0}
|
2023-02-01 14:13:42 -08:00
|
|
|
className='absolute inset-y-0 right-0 flex cursor-pointer items-center px-3 rtl:left-0 rtl:right-auto'
|
2022-03-21 11:09:01 -07:00
|
|
|
onClick={handleClear}
|
|
|
|
>
|
2022-04-07 11:47:06 -07:00
|
|
|
<SvgIcon
|
2022-07-09 09:20:02 -07:00
|
|
|
src={require('@tabler/icons/search.svg')}
|
2023-02-06 10:01:03 -08:00
|
|
|
className={clsx('h-4 w-4 text-gray-600', { hidden: hasValue })}
|
2022-03-21 11:09:01 -07:00
|
|
|
/>
|
|
|
|
|
2022-04-07 11:47:06 -07:00
|
|
|
<SvgIcon
|
2022-07-09 09:20:02 -07:00
|
|
|
src={require('@tabler/icons/x.svg')}
|
2023-02-06 10:01:03 -08:00
|
|
|
className={clsx('h-4 w-4 text-gray-600', { hidden: !hasValue })}
|
2022-03-21 11:09:01 -07:00
|
|
|
aria-label={intl.formatMessage(messages.placeholder)}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|
|
|
|
|
2022-03-22 05:42:26 -07:00
|
|
|
export default Search;
|