diff --git a/frontend/src/__test__/components/PageTitle.test.tsx b/frontend/src/__test__/components/PageTitle.test.tsx new file mode 100644 index 00000000..4d1e5895 --- /dev/null +++ b/frontend/src/__test__/components/PageTitle.test.tsx @@ -0,0 +1,114 @@ +import React from 'react'; +import { render, screen } from '@testing-library/react'; +import PageTitle from '../../components/PageTitle'; +import { describe, it, expect, vi } from 'vitest'; +import { MemoryRouter, useLocation } from 'react-router-dom'; + +vi.mock('react-router-dom', async () => { + const actual = await vi.importActual("react-router-dom"); + return { + ...actual, + useLocation: vi.fn(), + }; +}); + +vi.mock('../../components/BCHeaderwSide/constants', async () => { + const actual = await vi.importActual('../../components/BCHeaderwSide/constants'); + return { + ...actual, + leftMenu: [ + { + name: 'Main', + items: [ + { + name: 'Submain', + link: '/submain', + breadcrumb: true, + subItems: [ + { + name: 'Test Page', + link: '/test', + breadcrumb: true, + } + ] + }, + { + name: 'Intramain', + link: '/intramain', + breadcrumb: false, + subItems: [ + { + name: 'Internal Affairs', + link: '/intramain', + breadcrumb: false, + } + ] + } + ] + } + ] + }; +}); + +describe('PageTitle Component', () => { + + it('renders the title correctly', () => { + (useLocation as vi.Mock).mockReturnValue({ pathname: '/test' }); + render( + + + + ); + const titleElement = screen.getByText(/Test Title/i); + expect(titleElement).toBeInTheDocument(); + }); + + it('renders the breadcrumb correctly', () => { + (useLocation as vi.Mock).mockReturnValue({ pathname: '/test' }); + render( + + + + ); + + const titleElement = screen.getByText(/Submain/i); + expect(titleElement).toBeInTheDocument(); + }); + + it('renders empty breadcrumb if no path found on list', () => { + (useLocation as vi.Mock).mockReturnValue({ pathname: '/homer' }); + render( + + + + ); + + const olElement = screen.getByRole('list'); + const listItems = olElement.querySelectorAll('li'); + expect(olElement).toBeInTheDocument(); + expect(listItems.length).toBe(0); + }); + + it('should not render breadcrumb if breadcrumb is false', () => { + (useLocation as vi.Mock).mockReturnValue({ pathname: '/intramain' }); + render( + + + + ); + + const olElement = screen.getByRole('list'); + const listItems = olElement.querySelectorAll('li'); + expect(olElement).toBeInTheDocument(); + expect(listItems.length).toBe(0); + }); + +}); \ No newline at end of file diff --git a/frontend/src/components/BCHeaderwSide/constants.ts b/frontend/src/components/BCHeaderwSide/constants.ts index 074342bc..c642efc6 100644 --- a/frontend/src/components/BCHeaderwSide/constants.ts +++ b/frontend/src/components/BCHeaderwSide/constants.ts @@ -5,6 +5,7 @@ export type LeftMenuItem = { icon?: keyof typeof Icons; link: string; disabled: boolean; + breadcrumb: boolean; subItems?: LeftMenuItem[]; } @@ -22,16 +23,19 @@ const mainActivitiesItems: LeftMenu[] = [ icon: 'MapBoundaryVegetation', link: '/opening', disabled: false, + breadcrumb: false, subItems: [ { name: 'Home page', link: '/opening', - disabled: false + disabled: false, + breadcrumb: false }, { name: 'Silviculture search', link: '/silviculture-search', - disabled: false + disabled: false, + breadcrumb: true } ] } @@ -47,13 +51,15 @@ const managementItems: LeftMenu[] = [ name: 'Settings', icon: 'Settings', link: '#', - disabled: true + disabled: true, + breadcrumb: false }, { name: 'Notifications', icon: 'Notification', link: '#', - disabled: true + disabled: true, + breadcrumb: false } ] } diff --git a/frontend/src/components/BCHeaderwSide/index.tsx b/frontend/src/components/BCHeaderwSide/index.tsx index 0295c581..f88efece 100644 --- a/frontend/src/components/BCHeaderwSide/index.tsx +++ b/frontend/src/components/BCHeaderwSide/index.tsx @@ -101,6 +101,8 @@ function BCHeaderwSide(): JSX.Element { title={subItem.name} renderIcon={IconComponent} isActive={isActive} + isSideNavExpanded={isActive} + defaultExpanded={isActive} > {subItem.subItems.map(subSubItem => ( = ({ title, subtitle }: PageTitleProps) => { - + + const currentLocation = useLocation().pathname; + + // This will return up to the second level, even if we use just the first one + const extractCurrentItems = (): LeftMenuItem[] => { + for (const item of leftMenu) { + if (item.items) { + for (const subItem of item.items) { + if (subItem.link === currentLocation && subItem.breadcrumb) { + return [subItem]; + } + + if (subItem.subItems) { + for (const subSubItem of subItem.subItems) { + if (subSubItem.link === currentLocation && subSubItem.breadcrumb) { + return [subItem]; + } + } + } + } + } + } + return []; + } + return ( + + {extractCurrentItems().map(item => ( + {item.name} + ))} +

{title}