mirror of
https://github.com/grafana/grafana.git
synced 2024-12-02 13:39:19 -06:00
590b07539f
* Update core border radii * Add non-default border-radius on some grafana-ui items
135 lines
3.9 KiB
TypeScript
135 lines
3.9 KiB
TypeScript
import { css, cx } from '@emotion/css';
|
|
import { ActionId, ActionImpl } from 'kbar';
|
|
import React from 'react';
|
|
|
|
import { GrafanaTheme2 } from '@grafana/data';
|
|
import { useStyles2 } from '@grafana/ui';
|
|
|
|
export const ResultItem = React.forwardRef(
|
|
(
|
|
{
|
|
action,
|
|
active,
|
|
currentRootActionId,
|
|
}: {
|
|
action: ActionImpl;
|
|
active: boolean;
|
|
currentRootActionId: ActionId;
|
|
},
|
|
ref: React.Ref<HTMLDivElement>
|
|
) => {
|
|
const ancestors = React.useMemo(() => {
|
|
if (!currentRootActionId) {
|
|
return action.ancestors;
|
|
}
|
|
|
|
const index = action.ancestors.findIndex((ancestor) => ancestor.id === currentRootActionId);
|
|
// +1 removes the currentRootAction; e.g.
|
|
// if we are on the "Set theme" parent action,
|
|
// the UI should not display "Set theme… > Dark"
|
|
// but rather just "Dark"
|
|
return action.ancestors.slice(index + 1);
|
|
}, [action.ancestors, currentRootActionId]);
|
|
|
|
const styles = useStyles2(getResultItemStyles);
|
|
|
|
let name = action.name;
|
|
|
|
const hasCommandOrLink = (action: ActionImpl) =>
|
|
// eslint-disable-next-line @typescript-eslint/consistent-type-assertions
|
|
Boolean(action.command?.perform || (action as ActionImpl & { url?: string }).url);
|
|
|
|
// TODO: does this needs adjusting for i18n?
|
|
if (action.children.length && !hasCommandOrLink(action) && !name.endsWith('...')) {
|
|
name += '...';
|
|
}
|
|
|
|
return (
|
|
<div ref={ref} className={cx(styles.row, active && styles.activeRow)}>
|
|
<div className={styles.actionContainer}>
|
|
{action.icon}
|
|
<div className={styles.textContainer}>
|
|
{ancestors.map((ancestor) => (
|
|
<React.Fragment key={ancestor.id}>
|
|
{!hasCommandOrLink(ancestor) && (
|
|
<>
|
|
<span className={styles.breadcrumbAncestor}>{ancestor.name}</span>
|
|
<span className={styles.breadcrumbSeparator}>›</span>
|
|
</>
|
|
)}
|
|
</React.Fragment>
|
|
))}
|
|
<span>{name}</span>
|
|
</div>
|
|
{action.subtitle && <span className={styles.subtitleText}>{action.subtitle}</span>}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|
|
);
|
|
|
|
ResultItem.displayName = 'ResultItem';
|
|
|
|
const getResultItemStyles = (theme: GrafanaTheme2) => {
|
|
return {
|
|
row: css({
|
|
padding: theme.spacing(1, 2),
|
|
display: 'flex',
|
|
alightItems: 'center',
|
|
justifyContent: 'space-between',
|
|
cursor: 'pointer',
|
|
position: 'relative',
|
|
borderRadius: theme.shape.radius.default,
|
|
margin: theme.spacing(0, 1),
|
|
}),
|
|
activeRow: css({
|
|
color: theme.colors.text.maxContrast,
|
|
background: theme.colors.emphasize(theme.colors.background.primary, 0.03),
|
|
'&:before': {
|
|
display: 'block',
|
|
content: '" "',
|
|
position: 'absolute',
|
|
left: 0,
|
|
top: 0,
|
|
bottom: 0,
|
|
width: theme.spacing(0.5),
|
|
borderRadius: theme.shape.radius.default,
|
|
backgroundImage: theme.colors.gradients.brandVertical,
|
|
},
|
|
}),
|
|
actionContainer: css({
|
|
display: 'flex',
|
|
gap: theme.spacing(1),
|
|
alignItems: 'baseline',
|
|
fontSize: theme.typography.fontSize,
|
|
width: '100%',
|
|
}),
|
|
textContainer: css({
|
|
display: 'block',
|
|
overflow: 'hidden',
|
|
textOverflow: 'ellipsis',
|
|
whiteSpace: 'nowrap',
|
|
}),
|
|
breadcrumbAncestor: css({
|
|
color: theme.colors.text.secondary,
|
|
}),
|
|
breadcrumbSeparator: css({
|
|
color: theme.colors.text.secondary,
|
|
marginLeft: theme.spacing(1),
|
|
marginRight: theme.spacing(1),
|
|
}),
|
|
subtitleText: css({
|
|
...theme.typography.bodySmall,
|
|
color: theme.colors.text.secondary,
|
|
display: 'block',
|
|
flexBasis: '20%',
|
|
flexGrow: 1,
|
|
flexShrink: 0,
|
|
maxWidth: 'fit-content',
|
|
overflow: 'hidden',
|
|
textOverflow: 'ellipsis',
|
|
whiteSpace: 'nowrap',
|
|
}),
|
|
};
|
|
};
|