110 lines
3.4 KiB
TypeScript
110 lines
3.4 KiB
TypeScript
import classNames from 'clsx';
|
|
import { OrderedSet as ImmutableOrderedSet } from 'immutable';
|
|
import throttle from 'lodash/throttle';
|
|
import React, { useCallback, useEffect, useRef, useState } from 'react';
|
|
import { defineMessages, useIntl } from 'react-intl';
|
|
|
|
import { locationSearch } from 'soapbox/actions/events';
|
|
import AutosuggestInput, { AutoSuggestion } from 'soapbox/components/autosuggest-input';
|
|
import Icon from 'soapbox/components/icon';
|
|
import { useAppDispatch } from 'soapbox/hooks';
|
|
|
|
import AutosuggestLocation from './autosuggest-location';
|
|
|
|
const noOp = () => {};
|
|
|
|
const messages = defineMessages({
|
|
placeholder: { id: 'location_search.placeholder', defaultMessage: 'Find an address' },
|
|
});
|
|
|
|
interface ILocationSearch {
|
|
onSelected: (locationId: string) => void,
|
|
}
|
|
|
|
const LocationSearch: React.FC<ILocationSearch> = ({ onSelected }) => {
|
|
const intl = useIntl();
|
|
const dispatch = useAppDispatch();
|
|
const [locationIds, setLocationIds] = useState(ImmutableOrderedSet<string>());
|
|
const controller = useRef(new AbortController());
|
|
|
|
const [value, setValue] = useState('');
|
|
|
|
const isEmpty = (): boolean => {
|
|
return !(value.length > 0);
|
|
};
|
|
|
|
const handleChange: React.ChangeEventHandler<HTMLInputElement> = ({ target }) => {
|
|
refreshCancelToken();
|
|
handleLocationSearch(target.value);
|
|
setValue(target.value);
|
|
};
|
|
|
|
const handleSelected = (_tokenStart: number, _lastToken: string | null, suggestion: AutoSuggestion) => {
|
|
if (typeof suggestion === 'string') {
|
|
onSelected(suggestion);
|
|
}
|
|
};
|
|
|
|
const handleClear: React.MouseEventHandler = e => {
|
|
e.preventDefault();
|
|
|
|
if (!isEmpty()) {
|
|
setValue('');
|
|
}
|
|
};
|
|
|
|
const handleKeyDown: React.KeyboardEventHandler = e => {
|
|
if (e.key === 'Escape') {
|
|
document.querySelector('.ui')?.parentElement?.focus();
|
|
}
|
|
};
|
|
|
|
const refreshCancelToken = () => {
|
|
controller.current.abort();
|
|
controller.current = new AbortController();
|
|
};
|
|
|
|
const clearResults = () => {
|
|
setLocationIds(ImmutableOrderedSet());
|
|
};
|
|
|
|
const handleLocationSearch = useCallback(throttle(q => {
|
|
dispatch(locationSearch(q, controller.current.signal))
|
|
.then((locations: { origin_id: string }[]) => {
|
|
const locationIds = locations.map(location => location.origin_id);
|
|
setLocationIds(ImmutableOrderedSet(locationIds));
|
|
})
|
|
.catch(noOp);
|
|
|
|
}, 900, { leading: true, trailing: true }), []);
|
|
|
|
useEffect(() => {
|
|
if (value === '') {
|
|
clearResults();
|
|
}
|
|
}, [value]);
|
|
|
|
return (
|
|
<div className='search'>
|
|
<AutosuggestInput
|
|
className='rounded-full'
|
|
placeholder={intl.formatMessage(messages.placeholder)}
|
|
value={value}
|
|
onChange={handleChange}
|
|
suggestions={locationIds.toList()}
|
|
onSuggestionsFetchRequested={noOp}
|
|
onSuggestionsClearRequested={noOp}
|
|
onSuggestionSelected={handleSelected}
|
|
searchTokens={[]}
|
|
onKeyDown={handleKeyDown}
|
|
renderSuggestion={AutosuggestLocation}
|
|
/>
|
|
<div role='button' tabIndex={0} className='search__icon' onClick={handleClear}>
|
|
<Icon src={require('@tabler/icons/search.svg')} className={classNames('svg-icon--search', { active: isEmpty() })} />
|
|
<Icon src={require('@tabler/icons/backspace.svg')} className={classNames('svg-icon--backspace', { active: !isEmpty() })} aria-label={intl.formatMessage(messages.placeholder)} />
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default LocationSearch;
|