-
-
Notifications
You must be signed in to change notification settings - Fork 135
/
Copy pathSearchInput.tsx
132 lines (110 loc) · 3.27 KB
/
SearchInput.tsx
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
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
import { useCallback, useEffect, useRef } from "react";
import { useSearchParams } from "react-router-dom";
import { useAppContext } from "@contexts/AppContext";
import { QueryParams } from "@utils/enums";
import { SearchIcon } from "./Icons";
const SearchInput = () => {
const [searchParams, setSearchParams] = useSearchParams();
const { searchText, setSearchText } = useAppContext();
const inputRef = useRef<HTMLInputElement | null>(null);
const handleSearchFieldClick = () => {
inputRef.current?.focus();
};
const clearSearch = useCallback(() => {
setSearchText("");
searchParams.delete(QueryParams.SEARCH);
setSearchParams(searchParams);
}, [searchParams, setSearchParams, setSearchText]);
const performSearch = useCallback(() => {
// Check if the input element is focused.
if (document.activeElement !== inputRef.current) {
return;
}
const formattedVal = searchText.toLowerCase();
setSearchText(formattedVal);
if (!formattedVal) {
searchParams.delete(QueryParams.SEARCH);
setSearchParams(searchParams);
} else {
searchParams.set(QueryParams.SEARCH, formattedVal);
setSearchParams(searchParams);
}
}, [searchParams, searchText, setSearchParams, setSearchText]);
/**
* Focus the search input when the user presses the `/` key.
*/
const handleSearchKeyPress = (e: KeyboardEvent) => {
if (e.key === "/") {
e.preventDefault();
inputRef.current?.focus();
}
};
/**
* Clear search text and blur the input when the `Escape` key is pressed.
*/
const handleEscapeKeyPress = useCallback(
(e: KeyboardEvent) => {
if (e.key !== "Escape") {
return;
}
// Check if the input element is focused.
if (document.activeElement !== inputRef.current) {
return;
}
inputRef.current?.blur();
clearSearch();
},
[clearSearch]
);
useEffect(() => {
window.addEventListener("keydown", handleSearchKeyPress);
window.addEventListener("keyup", handleEscapeKeyPress);
return () => {
window.removeEventListener("keydown", handleSearchKeyPress);
window.removeEventListener("keyup", handleEscapeKeyPress);
};
}, [handleEscapeKeyPress]);
/**
* Update the search query in the URL when the search text changes.
*/
useEffect(() => {
performSearch();
}, [searchText, performSearch]);
/**
* Set the search text to the search query from the URL on mount.
*/
useEffect(() => {
const searchQuery = searchParams.get(QueryParams.SEARCH);
if (!searchQuery) {
return;
}
setSearchText(searchQuery);
// eslint-disable-next-line react-hooks/exhaustive-deps
}, []);
return (
<div className="search-field" onClick={handleSearchFieldClick}>
<SearchIcon />
<input
ref={inputRef}
value={searchText}
type="search"
id="search"
autoComplete="off"
onChange={(e) => {
const newValue = e.target.value;
if (!newValue) {
clearSearch();
return;
}
setSearchText(newValue);
}}
/>
{!searchText && (
<label htmlFor="search">
Type <kbd>/</kbd> to search
</label>
)}
</div>
);
};
export default SearchInput;