1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
|
import React, { useState, useEffect } from 'react';
import SearchIcon from '@material-ui/icons/Search';
import {
InputBase,
List,
ListItem,
Paper,
Divider,
ClickAwayListener
} from '@material-ui/core';
import { makeStyles } from '@material-ui/core/styles';
import { User } from 'which-types';
import { get } from '../../requests';
import UserStrip from '../UserStrip/UserStrip';
import { useNavigate } from '../../hooks/useNavigate';
const INTERVAL = 300;
const LIMIT = 7;
const useStyles = makeStyles(theme => ({
root: {
position: 'relative',
background: 'rgba(255, 255, 255, 0.5)',
borderRadius: '2px',
padding: theme.spacing(0.5),
display: 'flex',
alignItems: 'center'
},
results: {
position: 'absolute',
width: '100%',
top: theme.spacing(5)
},
listItem: {
padding: 0
}
}));
const SearchBar: React.FC = () => {
const [results, setResults] = useState<User[]>([]);
const [query, setQuery] = useState<string>('');
const [debouncedQuery, setDebouncedQuery] = useState<string>(query);
const { navigate } = useNavigate();
const classes = useStyles();
useEffect(() => {
const handler = setTimeout(() => setDebouncedQuery(query), INTERVAL);
return () => clearTimeout(handler);
}, [query]);
useEffect(() => {
const fetchPolls = () => {
get(`/users?username[$regex]=${debouncedQuery}&$limit=${LIMIT}`)
.then(response => setResults(response.data))
.catch(() => setResults([]));
};
if (debouncedQuery) fetchPolls();
else setResults([]);
}, [debouncedQuery]);
const handleChange = (event: React.ChangeEvent<HTMLInputElement>): void => {
setQuery(event.target.value.trim());
};
const handleClose = () => {
setDebouncedQuery('');
setQuery('');
};
const handleNavigate = (index: number) => () => {
navigate('profile', results[index]._id);
handleClose();
};
const SearchResults = (
<ClickAwayListener onClickAway={handleClose}>
<Paper className={classes.results}>
<List>
{
results.map((result, index) => (
<div key={result._id}>
<ListItem button onClick={handleNavigate(index)} className={classes.listItem}>
<UserStrip user={result} />
</ListItem>
{(index < results.length - 1) && <Divider />}
</div>
))
}
</List>
</Paper>
</ClickAwayListener>
);
return (
<div className={classes.root}>
<SearchIcon />
<InputBase
placeholder="Search..."
value={query}
onChange={handleChange}
/>
{results.length > 0 && SearchResults}
</div>
);
};
export default SearchBar;
|