source: src/layouts/_common/searchbar/searchbar.tsx@ 5d6f37a

main
Last change on this file since 5d6f37a was 5d6f37a, checked in by Naum Shapkarovski <naumshapkarovski@…>, 7 weeks ago

add customer

  • Property mode set to 100644
File size: 4.7 KB
Line 
1'use client';
2
3import { useState, memo, useCallback } from 'react';
4import parse from 'autosuggest-highlight/parse';
5import match from 'autosuggest-highlight/match';
6// @mui
7import { useTheme } from '@mui/material/styles';
8import Box from '@mui/material/Box';
9import List from '@mui/material/List';
10import Stack from '@mui/material/Stack';
11import InputBase from '@mui/material/InputBase';
12import IconButton from '@mui/material/IconButton';
13import InputAdornment from '@mui/material/InputAdornment';
14import Dialog, { dialogClasses } from '@mui/material/Dialog';
15// hooks
16import { useBoolean } from 'src/hooks/use-boolean';
17import { useResponsive } from 'src/hooks/use-responsive';
18import { useEventListener } from 'src/hooks/use-event-listener';
19// components
20import Label from 'src/components/label';
21import Iconify from 'src/components/iconify';
22import Scrollbar from 'src/components/scrollbar';
23import { useRouter } from 'src/routes/hooks';
24import SearchNotFound from 'src/components/search-not-found';
25//
26import ResultItem from './result-item';
27import { useNavData } from '../../dashboard/config-navigation';
28import { applyFilter, groupedData, getAllItems } from './utils';
29
30// ----------------------------------------------------------------------
31
32function Searchbar() {
33 const theme = useTheme();
34
35 const router = useRouter();
36
37 const search = useBoolean();
38
39 const lgUp = useResponsive('up', 'lg');
40
41 const [searchQuery, setSearchQuery] = useState('');
42
43 const navData = useNavData();
44
45 const handleClose = useCallback(() => {
46 search.onFalse();
47 setSearchQuery('');
48 }, [search]);
49
50 const handleKeyDown = (event: KeyboardEvent) => {
51 if (event.key === 'k' && event.metaKey) {
52 search.onToggle();
53 setSearchQuery('');
54 }
55 };
56
57 useEventListener('keydown', handleKeyDown);
58
59 const handleClick = useCallback(
60 (path: string) => {
61 if (path.includes('http')) {
62 window.open(path);
63 } else {
64 router.push(path);
65 }
66 handleClose();
67 },
68 [handleClose, router]
69 );
70
71 const handleSearch = useCallback((event: React.ChangeEvent<HTMLTextAreaElement>) => {
72 setSearchQuery(event.target.value);
73 }, []);
74
75 const dataFiltered = applyFilter({
76 inputData: getAllItems({ data: navData }),
77 query: searchQuery,
78 });
79
80 const notFound = searchQuery && !dataFiltered.length;
81
82 const renderItems = () => {
83 const data = groupedData(dataFiltered);
84
85 return Object.keys(data)
86 .sort((a, b) => -b.localeCompare(a))
87 .map((group, index) => (
88 <List key={group || index} disablePadding>
89 {data[group].map((item) => {
90 const { title, path } = item;
91
92 const partsTitle = parse(title, match(title, searchQuery));
93
94 const partsPath = parse(path, match(path, searchQuery));
95
96 return (
97 <ResultItem
98 path={partsPath}
99 title={partsTitle}
100 key={`${title}${path}`}
101 groupLabel={searchQuery && group}
102 onClickItem={() => handleClick(path)}
103 />
104 );
105 })}
106 </List>
107 ));
108 };
109
110 const renderButton = (
111 <Stack direction="row" alignItems="center">
112 <IconButton onClick={search.onTrue}>
113 <Iconify icon="eva:search-fill" />
114 </IconButton>
115
116 {lgUp && <Label sx={{ px: 0.75, fontSize: 12, color: 'text.secondary' }}>⌘K</Label>}
117 </Stack>
118 );
119
120 return (
121 <>
122 {renderButton}
123
124 <Dialog
125 fullWidth
126 maxWidth="sm"
127 open={search.value}
128 onClose={handleClose}
129 transitionDuration={{
130 enter: theme.transitions.duration.shortest,
131 exit: 0,
132 }}
133 PaperProps={{
134 sx: {
135 mt: 15,
136 overflow: 'unset',
137 },
138 }}
139 sx={{
140 [`& .${dialogClasses.container}`]: {
141 alignItems: 'flex-start',
142 },
143 }}
144 >
145 <Box sx={{ p: 3, borderBottom: `solid 1px ${theme.palette.divider}` }}>
146 <InputBase
147 fullWidth
148 autoFocus
149 placeholder="Search..."
150 value={searchQuery}
151 onChange={handleSearch}
152 startAdornment={
153 <InputAdornment position="start">
154 <Iconify icon="eva:search-fill" width={24} sx={{ color: 'text.disabled' }} />
155 </InputAdornment>
156 }
157 endAdornment={<Label sx={{ letterSpacing: 1, color: 'text.secondary' }}>esc</Label>}
158 inputProps={{
159 sx: { typography: 'h6' },
160 }}
161 />
162 </Box>
163
164 <Scrollbar sx={{ p: 3, pt: 2, height: 400 }}>
165 {notFound ? <SearchNotFound query={searchQuery} sx={{ py: 10 }} /> : renderItems()}
166 </Scrollbar>
167 </Dialog>
168 </>
169 );
170}
171
172export default memo(Searchbar);
Note: See TracBrowser for help on using the repository browser.