1
0
mirror of https://github.com/Unleash/unleash.git synced 2025-04-01 01:18:10 +02:00
unleash.unleash/frontend/src/component/commandBar/CommandBar.tsx
2025-03-14 11:46:44 +01:00

422 lines
14 KiB
TypeScript

import { useEffect, useRef, useState } from 'react';
import {
Box,
IconButton,
InputBase,
Paper,
styled,
Tooltip,
} from '@mui/material';
import Close from '@mui/icons-material/Close';
import SearchIcon from '@mui/icons-material/Search';
import { ConditionallyRender } from 'component/common/ConditionallyRender/ConditionallyRender';
import { useKeyboardShortcut } from 'hooks/useKeyboardShortcut';
import { SEARCH_INPUT } from 'utils/testIds';
import { useOnClickOutside } from 'hooks/useOnClickOutside';
import {
CommandResultGroup,
type CommandResultGroupItem,
} from './RecentlyVisited/CommandResultGroup';
import { CommandPageSuggestions } from './CommandPageSuggestions';
import { useRoutes } from 'component/layout/MainLayout/NavigationSidebar/useRoutes';
import { useAsyncDebounce } from 'react-table';
import useProjects from 'hooks/api/getters/useProjects/useProjects';
import {
type CommandQueryCounter,
CommandSearchFeatures,
} from './CommandSearchFeatures';
import { usePlausibleTracker } from 'hooks/usePlausibleTracker';
import { CommandQuickSuggestions } from './CommandQuickSuggestions';
import { CommandSearchPages } from './CommandSearchPages';
import { CommandBarFeedback } from './CommandBarFeedback';
import { RecentlyVisitedRecorder } from './RecentlyVisitedRecorder';
import { ScreenReaderOnly } from 'component/common/ScreenReaderOnly/ScreenReaderOnly';
export const CommandResultsPaper = styled(Paper)(({ theme }) => ({
position: 'absolute',
width: '100%',
left: 0,
top: '39px',
zIndex: theme.zIndex.drawer,
borderTop: theme.spacing(0),
padding: theme.spacing(1.5, 0, 1.5),
borderRadius: 0,
borderBottomLeftRadius: theme.spacing(1),
borderBottomRightRadius: theme.spacing(1),
boxShadow: theme.shadows[2],
fontSize: theme.fontSizes.smallBody,
color: theme.palette.text.secondary,
wordBreak: 'break-word',
border: `1px solid ${theme.palette.neutral.border}`,
}));
const StyledContainer = styled('div', {
shouldForwardProp: (prop) => prop !== 'active',
})<{
active: boolean | undefined;
}>(({ theme, active }) => ({
border: `1px solid transparent`,
display: 'flex',
flexGrow: 1,
alignItems: 'center',
position: 'relative',
backgroundColor: theme.palette.background.application,
maxWidth: active ? '100%' : '400px',
[theme.breakpoints.down('md')]: {
marginTop: theme.spacing(1),
maxWidth: '100%',
},
}));
const StyledSearch = styled('div')<{ isOpen?: boolean }>(
({ theme, isOpen }) => ({
display: 'flex',
alignItems: 'center',
backgroundColor: theme.palette.background.paper,
border: `1px solid ${theme.palette.neutral.border}`,
borderRadius: theme.shape.borderRadiusExtraLarge,
padding: '3px 5px 3px 12px',
width: '100%',
zIndex: 3,
...(isOpen
? {
borderBottomLeftRadius: 0,
borderBottomRightRadius: 0,
borderBottom: '0px',
paddingTop: theme.spacing(0.5),
paddingBottom: theme.spacing(0.5),
}
: {}),
}),
);
const StyledInputBase = styled(InputBase)(({ theme }) => ({
width: '100%',
minWidth: '300px',
backgroundColor: theme.palette.background.paper,
}));
const StyledClose = styled(Close)(({ theme }) => ({
color: theme.palette.neutral.main,
fontSize: theme.typography.body1.fontSize,
}));
interface IPageRouteInfo {
path: string;
route: string;
title: string;
}
export const CommandBar = () => {
const { trackEvent } = usePlausibleTracker();
const searchInputRef = useRef<HTMLInputElement>(null);
const searchContainerRef = useRef<HTMLInputElement>(null);
const [showSuggestions, setShowSuggestions] = useState(false);
const [searchLoading, setSearchLoading] = useState(false);
const [searchString, setSearchString] = useState(undefined);
const [searchedProjects, setSearchedProjects] = useState<
CommandResultGroupItem[]
>([]);
const [searchedPages, setSearchedPages] = useState<
CommandResultGroupItem[]
>([]);
const [searchedFlagCount, setSearchedFlagCount] =
useState<CommandQueryCounter>({ query: '', count: 0 });
const [hasNoResults, setHasNoResults] = useState(false);
const [value, setValue] = useState<string>('');
const { routes } = useRoutes();
const allRoutes: Record<string, IPageRouteInfo> = {};
for (const route of [...routes.mainNavRoutes, ...routes.adminRoutes]) {
allRoutes[route.path] = {
path: route.path,
route: route.route,
title: route.title,
};
}
const hideSuggestions = () => {
setShowSuggestions(false);
};
const { projects } = useProjects();
const debouncedSetSearchState = useAsyncDebounce((query) => {
setSearchString(query);
const filteredProjects = projects.filter((project) =>
project.name.toLowerCase().includes(query.toLowerCase()),
);
const mappedProjects = filteredProjects.map((project) => ({
name: project.name,
link: `/projects/${project.id}`,
}));
setSearchedProjects(mappedProjects);
const filteredPages = Object.values(allRoutes).filter((route) =>
route.title.toLowerCase().includes(query.toLowerCase()),
);
const mappedPages = filteredPages.map((page) => ({
name: page.title,
link: page.path,
}));
setSearchedPages(mappedPages);
const noResultsFound =
query.length !== 0 &&
mappedProjects.length === 0 &&
mappedPages.length === 0 &&
searchedFlagCount.count === 0 &&
searchedFlagCount.query === query;
if (noResultsFound) {
trackEvent('command-bar', {
props: {
eventType: 'no search results found',
query: query,
},
});
}
setHasNoResults(noResultsFound);
}, 200);
useEffect(() => {
debouncedSetSearchState(value);
}, [JSON.stringify(searchedFlagCount)]);
const onSearchChange = (value: string) => {
debouncedSetSearchState(value);
setValue(value);
};
const clearSearchValue = () => {
onSearchChange('');
setShowSuggestions(false);
};
const hotkey = useKeyboardShortcut(
{
modifiers: ['ctrl'],
key: 'k',
preventDefault: true,
},
() => {
if (document.activeElement === searchInputRef.current) {
searchInputRef.current?.blur();
} else {
searchInputRef.current?.focus();
}
},
);
useKeyboardShortcut({ key: 'Escape' }, () => {
setShowSuggestions(false);
if (searchContainerRef.current?.contains(document.activeElement)) {
searchInputRef.current?.blur();
}
});
const placeholder = `Command menu (${hotkey})`;
const findCommandBarLinksAndSelectedIndex = () => {
const allCommandBarLinks =
searchContainerRef.current?.querySelectorAll('ul > a');
if (!allCommandBarLinks || allCommandBarLinks.length === 0) return;
let selectedIndex = -1;
allCommandBarLinks.forEach((link, index) => {
if (link === document.activeElement) {
selectedIndex = index;
}
});
return {
allCommandBarLinks,
selectedIndex,
};
};
useKeyboardShortcut(
{
key: 'ArrowDown',
preventDefault: true,
},
() => {
const itemsAndIndex = findCommandBarLinksAndSelectedIndex();
if (!itemsAndIndex) return;
const { allCommandBarLinks, selectedIndex } = itemsAndIndex;
const newIndex = selectedIndex + 1;
if (newIndex >= allCommandBarLinks.length) {
const element = searchInputRef.current;
if (element) {
element.focus();
element.setSelectionRange(
element.value.length,
element.value.length,
);
}
} else {
(allCommandBarLinks[newIndex] as HTMLElement).focus();
}
},
);
useKeyboardShortcut(
{
key: 'ArrowUp',
preventDefault: true,
},
() => {
const itemsAndIndex = findCommandBarLinksAndSelectedIndex();
if (!itemsAndIndex) return;
const { allCommandBarLinks, selectedIndex } = itemsAndIndex;
const newIndex = selectedIndex - 1;
if (newIndex >= 0) {
(allCommandBarLinks[newIndex] as HTMLElement).focus();
} else if (newIndex === -1) {
const element = searchInputRef.current;
if (element) {
element.focus();
element.setSelectionRange(
element.value.length,
element.value.length,
);
}
} else if (newIndex === -2) {
(
allCommandBarLinks[
allCommandBarLinks.length - 1
] as HTMLElement
).focus();
}
},
);
useKeyboardShortcut({ key: 'Tab' }, () => {
setShowSuggestions(false);
});
useOnClickOutside([searchContainerRef], hideSuggestions);
const onKeyDown = (event: React.KeyboardEvent) => {
if (event.key === 'Escape') {
setShowSuggestions(false);
} else if (
event.keyCode >= 48 &&
event.keyCode <= 110 &&
!hasNoResults
) {
searchInputRef.current?.focus();
}
};
return (
<StyledContainer ref={searchContainerRef} active={showSuggestions}>
<RecentlyVisitedRecorder />
<StyledSearch isOpen={showSuggestions}>
<SearchIcon
sx={{
mr: 1,
color: (theme) => theme.palette.action.disabled,
}}
/>
<ScreenReaderOnly>
<label htmlFor={'command-bar-input'}>{placeholder}</label>
</ScreenReaderOnly>
<StyledInputBase
id='command-bar-input'
inputRef={searchInputRef}
placeholder={placeholder}
inputProps={{
'data-testid': SEARCH_INPUT,
}}
value={value}
onChange={(e) => onSearchChange(e.target.value)}
onFocus={() => {
setShowSuggestions(true);
}}
/>
<Box sx={{ width: (theme) => theme.spacing(4) }}>
<ConditionallyRender
condition={Boolean(value)}
show={
<Tooltip title='Clear search query' arrow>
<IconButton
size='small'
onClick={(e) => {
e.stopPropagation();
onSearchChange('');
searchInputRef.current?.focus();
}}
sx={{
padding: (theme) => theme.spacing(1),
}}
>
<StyledClose />
</IconButton>
</Tooltip>
}
/>
</Box>
</StyledSearch>
<ConditionallyRender
condition={Boolean(value) && showSuggestions}
show={
<CommandResultsPaper onKeyDownCapture={onKeyDown}>
{searchString !== undefined && (
<CommandSearchFeatures
searchString={searchString}
setSearchedFlagCount={setSearchedFlagCount}
onClick={clearSearchValue}
setSearchLoading={setSearchLoading}
/>
)}
<ConditionallyRender
condition={!searchLoading}
show={
<>
<CommandResultGroup
groupName={'Projects'}
icon={'flag'}
onClick={clearSearchValue}
items={searchedProjects}
/>
<CommandSearchPages
items={searchedPages}
onClick={clearSearchValue}
/>
<ConditionallyRender
condition={hasNoResults}
show={
<CommandBarFeedback
onSubmit={hideSuggestions}
/>
}
/>
</>
}
/>
</CommandResultsPaper>
}
elseShow={
showSuggestions && (
<CommandResultsPaper onKeyDownCapture={onKeyDown}>
<CommandQuickSuggestions
routes={allRoutes}
onClick={clearSearchValue}
/>
<CommandPageSuggestions
routes={allRoutes}
onClick={clearSearchValue}
/>
</CommandResultsPaper>
)
}
/>
</StyledContainer>
);
};