mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
* Navigation: Implement active state for items in the Sidemenu * Navigation: Improve logic for when link is active and extract isSearchActive into a util function * Navigation: Implement custom rule for dashboards under /d/ and fix minor bugs * Navigation: only show first matching active state + strip query params from link urls
65 lines
2.0 KiB
TypeScript
65 lines
2.0 KiB
TypeScript
import React from 'react';
|
|
import { useLocation } from 'react-router-dom';
|
|
import { cloneDeep } from 'lodash';
|
|
import { css } from '@emotion/css';
|
|
import { GrafanaTheme2, NavModelItem } from '@grafana/data';
|
|
import { locationService } from '@grafana/runtime';
|
|
import { Icon, IconName, styleMixins, useTheme2 } from '@grafana/ui';
|
|
import config from '../../config';
|
|
import { isLinkActive, isSearchActive } from './utils';
|
|
import SideMenuItem from './SideMenuItem';
|
|
|
|
const TopSection = () => {
|
|
const location = useLocation();
|
|
const theme = useTheme2();
|
|
const styles = getStyles(theme);
|
|
const navTree: NavModelItem[] = cloneDeep(config.bootData.navTree);
|
|
const mainLinks = navTree.filter((item) => !item.hideFromMenu);
|
|
const activeItemId = mainLinks.find((item) => isLinkActive(location.pathname, item))?.id;
|
|
|
|
const onOpenSearch = () => {
|
|
locationService.partial({ search: 'open' });
|
|
};
|
|
|
|
return (
|
|
<div data-testid="top-section-items" className={styles.container}>
|
|
<SideMenuItem isActive={isSearchActive(location)} label="Search dashboards" onClick={onOpenSearch}>
|
|
<Icon name="search" size="xl" />
|
|
</SideMenuItem>
|
|
{mainLinks.map((link, index) => {
|
|
return (
|
|
<SideMenuItem
|
|
key={`${link.id}-${index}`}
|
|
isActive={!isSearchActive(location) && activeItemId === link.id}
|
|
label={link.text}
|
|
menuItems={link.children}
|
|
target={link.target}
|
|
url={link.url}
|
|
>
|
|
{link.icon && <Icon name={link.icon as IconName} size="xl" />}
|
|
{link.img && <img src={link.img} alt={`${link.text} logo`} />}
|
|
</SideMenuItem>
|
|
);
|
|
})}
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export default TopSection;
|
|
|
|
const getStyles = (theme: GrafanaTheme2) => ({
|
|
container: css`
|
|
display: none;
|
|
flex-grow: 1;
|
|
|
|
@media ${styleMixins.mediaUp(`${theme.breakpoints.values.md}px`)} {
|
|
display: block;
|
|
margin-top: ${theme.spacing(5)};
|
|
}
|
|
|
|
.sidemenu-open--xs & {
|
|
display: block;
|
|
}
|
|
`,
|
|
});
|