grafana/public/app/features/commandPalette/CommandPalette.tsx
Josh Hunt 9b10f6c7dd
CommandPalette: Fix long dashboard names freezing the browser (#61278)
* CommandPalette: Use custom useMatches to avoid perf issues with long names

* wip start to multi search

* use search permutations, and do substring match if search is over 25 chars

* update ufuzzy

* don't permute query if greater than 5 words
2023-01-13 14:16:16 +00:00

149 lines
4.5 KiB
TypeScript

import { css } from '@emotion/css';
import { useDialog } from '@react-aria/dialog';
import { FocusScope } from '@react-aria/focus';
import { useOverlay } from '@react-aria/overlays';
import {
KBarAnimator,
KBarPortal,
KBarPositioner,
KBarResults,
KBarSearch,
VisualState,
useRegisterActions,
useKBar,
ActionImpl,
} from 'kbar';
import React, { useEffect, useMemo, useRef } from 'react';
import { GrafanaTheme2 } from '@grafana/data';
import { reportInteraction } from '@grafana/runtime';
import { useStyles2 } from '@grafana/ui';
import { t } from 'app/core/internationalization';
import { ResultItem } from './ResultItem';
import { useDashboardResults } from './actions/dashboardActions';
import useActions from './actions/useActions';
import { CommandPaletteAction } from './types';
import { useMatches } from './useMatches';
export const CommandPalette = () => {
const styles = useStyles2(getSearchStyles);
const { query, showing, searchQuery } = useKBar((state) => ({
showing: state.visualState === VisualState.showing,
searchQuery: state.searchQuery,
}));
const actions = useActions();
useRegisterActions(actions, [actions]);
const dashboardResults = useDashboardResults(searchQuery, showing);
const ref = useRef<HTMLDivElement>(null);
const { overlayProps } = useOverlay(
{ isOpen: showing, onClose: () => query.setVisualState(VisualState.animatingOut) },
ref
);
const { dialogProps } = useDialog({}, ref);
// Report interaction when opened
useEffect(() => {
showing && reportInteraction('command_palette_opened');
}, [showing]);
return actions.length > 0 ? (
<KBarPortal>
<KBarPositioner className={styles.positioner}>
<KBarAnimator className={styles.animator}>
<FocusScope contain autoFocus restoreFocus>
<div {...overlayProps} {...dialogProps}>
<KBarSearch className={styles.search} />
<RenderResults dashboardResults={dashboardResults} />
</div>
</FocusScope>
</KBarAnimator>
</KBarPositioner>
</KBarPortal>
) : null;
};
interface RenderResultsProps {
dashboardResults: CommandPaletteAction[];
}
const RenderResults = ({ dashboardResults }: RenderResultsProps) => {
const { results, rootActionId } = useMatches();
const styles = useStyles2(getSearchStyles);
const dashboardsSectionTitle = t('command-palette.section.dashboard-search-results', 'Dashboards');
// because dashboard search results aren't registered as actions, we need to manually
// convert them to ActionImpls before passing them as items to KBarResults
const dashboardResultItems = useMemo(
() => dashboardResults.map((dashboard) => new ActionImpl(dashboard, { store: {} })),
[dashboardResults]
);
const items = useMemo(
() => (dashboardResultItems.length > 0 ? [...results, dashboardsSectionTitle, ...dashboardResultItems] : results),
[results, dashboardsSectionTitle, dashboardResultItems]
);
return (
<div className={styles.resultsContainer}>
<KBarResults
items={items}
onRender={({ item, active }) =>
typeof item === 'string' ? (
<div className={styles.sectionHeader}>{item}</div>
) : (
<ResultItem action={item} active={active} currentRootActionId={rootActionId!} />
)
}
/>
</div>
);
};
const getSearchStyles = (theme: GrafanaTheme2) => ({
positioner: css({
zIndex: theme.zIndex.portal,
marginTop: '0px',
'&::before': {
content: '""',
position: 'fixed',
top: 0,
right: 0,
bottom: 0,
left: 0,
background: theme.components.overlay.background,
backdropFilter: 'blur(1px)',
},
}),
animator: css({
maxWidth: theme.breakpoints.values.sm, // supposed to be 600...
width: '100%',
background: theme.colors.background.canvas,
color: theme.colors.text.primary,
borderRadius: theme.shape.borderRadius(4),
overflow: 'hidden',
boxShadow: theme.shadows.z3,
}),
search: css({
padding: theme.spacing(2, 3),
fontSize: theme.typography.fontSize,
width: '100%',
boxSizing: 'border-box',
outline: 'none',
border: 'none',
background: theme.colors.background.canvas,
color: theme.colors.text.primary,
borderBottom: `1px solid ${theme.colors.border.weak}`,
}),
sectionHeader: css({
padding: theme.spacing(1, 2),
fontSize: theme.typography.h6.fontSize,
fontWeight: theme.typography.body.fontWeight,
color: theme.colors.text.secondary,
}),
resultsContainer: css({
padding: theme.spacing(2, 0),
}),
});