grafana/public/app/core/components/AppChrome/AppChrome.test.tsx
Ben Sully f040a8321b
Plugins: Add Command Palette extension point (#78098)
Co-authored-by: Marcus Andersson <marcus.andersson@grafana.com>
2023-12-12 08:52:09 +00:00

143 lines
4.8 KiB
TypeScript

import { render, screen, waitFor } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { KBarProvider } from 'kbar';
import React, { ReactNode } from 'react';
import { TestProvider } from 'test/helpers/TestProvider';
import { getGrafanaContextMock } from 'test/mocks/getGrafanaContextMock';
import { DataFrame, DataFrameView, FieldType, NavModelItem } from '@grafana/data';
import { config } from '@grafana/runtime';
import { HOME_NAV_ID } from 'app/core/reducers/navModel';
import { DashboardQueryResult, getGrafanaSearcher, QueryResponse } from 'app/features/search/service';
import { Page } from '../Page/Page';
import { AppChrome } from './AppChrome';
jest.mock('@grafana/runtime', () => ({
...jest.requireActual('@grafana/runtime'),
getPluginLinkExtensions: jest.fn().mockReturnValue({ extensions: [] }),
}));
const pageNav: NavModelItem = {
text: 'pageNav title',
children: [
{ text: 'pageNav child1', url: '1', active: true },
{ text: 'pageNav child2', url: '2' },
],
};
const searchData: DataFrame = {
fields: [
{ name: 'kind', type: FieldType.string, config: {}, values: [] },
{ name: 'name', type: FieldType.string, config: {}, values: [] },
{ name: 'uid', type: FieldType.string, config: {}, values: [] },
{ name: 'url', type: FieldType.string, config: {}, values: [] },
{ name: 'tags', type: FieldType.other, config: {}, values: [] },
{ name: 'location', type: FieldType.string, config: {}, values: [] },
],
length: 0,
};
const mockSearchResult: QueryResponse = {
isItemLoaded: jest.fn(),
loadMoreItems: jest.fn(),
totalRows: searchData.length,
view: new DataFrameView<DashboardQueryResult>(searchData),
};
const setup = (children: ReactNode) => {
config.bootData.navTree = [
{
id: HOME_NAV_ID,
text: 'Home',
},
{
text: 'Section name',
id: 'section',
url: 'section',
children: [
{ text: 'Child1', id: 'child1', url: 'section/child1' },
{ text: 'Child2', id: 'child2', url: 'section/child2' },
],
},
{
text: 'Help',
id: 'help',
},
];
const context = getGrafanaContextMock();
const renderResult = render(
<KBarProvider>
<TestProvider grafanaContext={context}>
<AppChrome>
<div data-testid="page-children">{children}</div>
</AppChrome>
</TestProvider>
</KBarProvider>
);
return { renderResult, context };
};
describe('AppChrome', () => {
beforeAll(() => {
// need to mock out the search service since kbar calls it to fetch recent dashboards
jest.spyOn(getGrafanaSearcher(), 'search').mockResolvedValue(mockSearchResult);
});
afterEach(() => {
jest.clearAllMocks();
});
it('should render section nav model based on navId', async () => {
setup(<Page navId="child1">Children</Page>);
expect(await screen.findByTestId('page-children')).toBeInTheDocument();
expect(screen.getByRole('tab', { name: 'Tab Section name' })).toBeInTheDocument();
expect(screen.getByRole('tab', { name: 'Tab Child1' })).toBeInTheDocument();
expect(screen.getByRole('tab', { name: 'Tab Child1' })).toBeInTheDocument();
expect(screen.getAllByRole('tab').length).toBe(3);
});
it('should render section nav model based on navId and item page nav', async () => {
setup(
<Page navId="child1" pageNav={pageNav}>
Children
</Page>
);
expect(await screen.findByTestId('page-children')).toBeInTheDocument();
expect(screen.getByRole('tab', { name: 'Tab Section name' })).toBeInTheDocument();
expect(screen.getByRole('heading', { name: 'pageNav title' })).toBeInTheDocument();
expect(screen.getByRole('tab', { name: 'Tab Child1' })).toBeInTheDocument();
expect(screen.getByRole('tab', { name: 'Tab pageNav child1' })).toBeInTheDocument();
});
it('should create a skip link to skip to main content', async () => {
setup(<Page navId="child1">Children</Page>);
expect(await screen.findByRole('link', { name: 'Skip to main content' })).toBeInTheDocument();
});
it('should focus the skip link on initial tab before carrying on with normal tab order', async () => {
setup(<Page navId="child1">Children</Page>);
await userEvent.keyboard('{tab}');
const skipLink = await screen.findByRole('link', { name: 'Skip to main content' });
expect(skipLink).toHaveFocus();
await userEvent.keyboard('{tab}');
expect(await screen.findByRole('link', { name: 'Go to home' })).toHaveFocus();
});
it('should not render a skip link if the page is chromeless', async () => {
const { context } = setup(<Page navId="child1">Children</Page>);
context.chrome.update({
chromeless: true,
});
waitFor(() => {
expect(screen.queryByRole('link', { name: 'Skip to main content' })).not.toBeInTheDocument();
});
});
});