Probably I have a config issue I just can't find out where exactly. The error message is when I run my test:
TypeError: Cannot read properties of undefined (reading 'Brand')
14 | className='bg-body-tertiary shadow-lg'>
15 | <Container>
> 16 | <Navbar.Brand href='#'>
| ^
17 | Skill Hunter
18 | </Navbar.Brand>
19 | <Navbar.Toggle aria-controls='basic-navbar-nav' />
My package.json
:
{
"name": "skillhunter-reactjs",
"version": "0.1.0",
"private": true,
"dependencies": {
"axios": "^1.6.5",
"bootstrap": "^5.3.2",
"react": "^18.2.0",
"react-bootstrap": "^2.10.0",
"react-dom": "^18.2.0",
"react-icons": "^5.0.1",
"react-router-dom": "^6.21.2"
},
"scripts": {
"dev": "vite",
"test": "jest"
},
"eslintConfig": {
"extends": [
"react-app",
"react-app/jest"
]
},
"browserslist": {
"production": [
">0.2%",
"not dead",
"not op_mini all"
],
"development": [
"last 1 chrome version",
"last 1 firefox version",
"last 1 safari version"
]
},
"devDependencies": {
"@testing-library/cypress": "^10.0.1",
"@testing-library/dom": "^9.3.4",
"@testing-library/jest-dom": "^6.4.2",
"@testing-library/react": "^14.2.2",
"@testing-library/user-event": "^14.5.2",
"@types/jest": "^29.5.12",
"@types/react": "^18.2.66",
"@types/react-dom": "^18.2.22",
"@typescript-eslint/eslint-plugin": "^7.2.0",
"@typescript-eslint/parser": "^7.2.0",
"@vitejs/plugin-react": "^4.2.1",
"cypress": "^13.6.4",
"eslint": "^8.57.0",
"eslint-plugin-react-hooks": "^4.6.0",
"eslint-plugin-react-refresh": "^0.4.6",
"identity-obj-proxy": "^3.0.0",
"jest": "^29.1.2",
"jest-environment-jsdom": "^29.1.2",
"ts-jest": "^29.1.2",
"ts-node": "^10.9.2",
"typescript": "^5.2.2",
"vite": "^5.2.0"
}
}
My jest.config.ts
import type { Config } from '@jest/types';
const config: Config.InitialOptions = {
verbose: true,
preset: 'ts-jest',
testEnvironment: 'jest-environment-jsdom',
"moduleNameMapper": {
"axios": require.resolve("axios"),
"\\.(css|less|scss)$": "identity-obj-proxy",
},
setupFilesAfterEnv: ["<rootDir>/jest.setup.ts"],
};
export default config;
The test is
import { render, screen, within } from '@testing-library/react';
import userEvent from '@testing-library/user-event';
import { BrowserRouter } from 'react-router-dom';
import App from './App';
import '@testing-library/jest-dom';
// https://testing-library.com/docs/example-react-router/
test('navigating to admin / prog langs', async () => {
const adminMenu = await clickOnAdminMenu();
await userEvent.click(adminMenu.getByText(/Programming Languages/i));
expect(screen.getByText(/List of Programming Languages/i)).toBeInTheDocument();
});
async function clickOnAdminMenu() {
render(<App />, { wrapper: BrowserRouter });
// first simulating to click on the 'Administration'
const tMenuAdmin: HTMLElement = screen.getByTestId('t-menu-admin');
await userEvent.click(within(tMenuAdmin).getByRole('button'));
screen.debug();
return within(tMenuAdmin);
}
The App.tsx
is
import { Routes, Route } from 'react-router-dom';
import Layout from './components/Layout';
import ProgLangList from './components/admin/prog-lang/ProgLangList';
import Missing from './components/Missing';
import RepositoryList from './components/admin/repository/RepositoryList';
import Dashboard from './components/Dashboard';
import 'bootstrap/dist/css/bootstrap.min.css';
import { ProgLangProvider } from './context/ProgLangProvider';
import { RepositoryProvider } from './context/RepositoryProvider';
function App() {
return (
<Routes>
<Route path='/' element={<Layout />}>``
<Route index element={<Dashboard />} />
<Route path='admin'>
<Route path='prog-langs'>
<Route index element={<ProgLangProvider><ProgLangList /></ProgLangProvider>} />
</Route>
<Route path='repositories'>
<Route index element={<RepositoryProvider><RepositoryList /></RepositoryProvider>} />
</Route>
</Route>
<Route path="*" element={<Missing />} />
</Route>
</Routes>
)
}
export default App
The Layout.tsx
import Header from './Header';
import Footer from './Footer';
import { Outlet } from 'react-router-dom';
import { ReactElement } from 'react';
const Layout = (): ReactElement => {
return (
<div className='body'>
<Header />
<div className='main'>
<Outlet />
</div>
<Footer />
</div>
)
}
export default Layout;
And the Header.tsx
is what it complains about
import Nav from 'react-bootstrap/Nav'
import Navbar from 'react-bootstrap/Navbar'
import Container from 'react-bootstrap/Container'
import NavDropdown from 'react-bootstrap/NavDropdown'
import { FaUserNinja } from 'react-icons/fa6'
import { Link } from 'react-router-dom'
import { ReactElement } from 'react'
const Header = (): ReactElement => {
return (
<header>
<Navbar
expand='lg'
className='bg-body-tertiary shadow-lg'>
<Container>
<Navbar.Brand href='#'>
Skill Hunter
</Navbar.Brand>
<Navbar.Toggle aria-controls='basic-navbar-nav' />
It seems it doesn't like the <Navbar.Brand>
element but not sure why.
Any ideas?
I made a silly mistake in tsconfig.json
, I forgot to include all the relevant .ts
files, now it looks like
"include": ["**/*.ts", "**/*.tsx", "./testSetup.ts"],
and now it works.