Skip to content

Commit 31b7e9c

Browse files
authoredDec 13, 2024··
Merge pull request #49 from SpaceyaTech/feat/enhance-vitest-setup
Feat: Enhance vitest setup
2 parents 4beac6b + 1698da5 commit 31b7e9c

9 files changed

+600
-22
lines changed
 

‎package.json

+5
Original file line numberDiff line numberDiff line change
@@ -76,6 +76,10 @@
7676
"@tanstack/react-query-devtools": "^5.59.19",
7777
"@tanstack/router-devtools": "^1.79.0",
7878
"@tanstack/router-plugin": "^1.79.0",
79+
"@testing-library/dom": "^10.4.0",
80+
"@testing-library/jest-dom": "^6.6.3",
81+
"@testing-library/react": "^16.1.0",
82+
"@testing-library/user-event": "^14.5.2",
7983
"@types/node": "^22.10.1",
8084
"@types/react": "^18.3.11",
8185
"@types/react-dom": "^18.3.1",
@@ -89,6 +93,7 @@
8993
"eslint-plugin-react-hooks": "^5.0.0",
9094
"eslint-plugin-react-refresh": "^0.4.13",
9195
"globals": "^15.11.0",
96+
"jsdom": "^25.0.1",
9297
"postcss": "^8.4.47",
9398
"prettier": "^3.3.2",
9499
"prettier-plugin-tailwindcss": "^0.6.5",

‎pnpm-lock.yaml

+478-6
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
import { render, screen } from "@testing-library/react";
2+
import RepositoriesSection, { ALL_LANGS } from "./RepositoriesSection";
3+
import { projects } from "@/data/projects";
4+
import { renderWithRouter } from "../../../tests/utils";
5+
6+
describe("Repositories Section", () => {
7+
it("should render the section heading", () => {
8+
render(<RepositoriesSection projects={[]} />);
9+
expect(screen.getByRole("heading", { level: 2 })).toBeInTheDocument();
10+
});
11+
12+
it("should render the search input", () => {
13+
render(<RepositoriesSection projects={[]} />);
14+
expect(screen.getByTestId("project-search")).toBeInTheDocument();
15+
});
16+
17+
it("should render filter tags", () => {
18+
renderWithRouter(() => <RepositoriesSection projects={projects} />);
19+
20+
const filterLangs = [
21+
ALL_LANGS,
22+
...new Set(projects.flatMap((p) => p.languages)),
23+
];
24+
25+
filterLangs.forEach((lang) => {
26+
expect(screen.getByTestId(`btn-filter-${lang}`)).toBeInTheDocument();
27+
});
28+
});
29+
30+
it("should render the provided projects", () => {
31+
renderWithRouter(() => <RepositoriesSection projects={projects} />);
32+
projects.forEach((project) => {
33+
expect(screen.getByText(project.repository)).toBeInTheDocument();
34+
expect(screen.getByText(project.description)).toBeInTheDocument();
35+
});
36+
});
37+
38+
it("handles empty projects array gracefully", () => {
39+
render(<RepositoriesSection projects={[]} />);
40+
41+
expect(screen.queryByTestId("learn-more-link")).not.toBeInTheDocument();
42+
});
43+
44+
it("renders 'Learn more' links correctly", () => {
45+
renderWithRouter(() => <RepositoriesSection projects={projects} />);
46+
screen.getAllByTestId("learn-more-link").forEach((link, index) => {
47+
expect(link).toHaveAttribute("href", projects[index].link);
48+
});
49+
});
50+
});

‎src/routes/-components/RepositoriesSection.tsx

+17-13
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ export type Project = {
1212
link: string;
1313
};
1414

15-
const ALL_LANGS = "All";
15+
export const ALL_LANGS = "All";
1616

1717
type Props = {
1818
projects: Project[]; // get 6 projects
@@ -45,22 +45,24 @@ export default function RepositoriesSection({ projects }: Props) {
4545
}, [projects, searchTerm, selectedLanguage]);
4646

4747
return (
48-
<section className="border-brand-green-5 border-y px-5 pb-16 pt-5">
49-
<h2 className="text-brand-gray-8 text-center text-[28px]/8 font-semibold md:text-[32px]/[38px]">
48+
<section className="border-y border-brand-green-5 px-5 pb-16 pt-5">
49+
<h2 className="text-center text-[28px]/8 font-semibold text-brand-gray-8 md:text-[32px]/[38px]">
5050
Access the largest directory of open-source projects
5151
</h2>
52-
<p className="text-sm/base text-brand-gray-1/80 mt-[10px] text-center">
52+
<p className="text-sm/base mt-[10px] text-center text-brand-gray-1/80">
5353
Use advanced filters to find a project you love and make your first
5454
commit message.
5555
</p>
5656

57-
<div className="border-brand-green-5 bg-brand-1/20 mx-auto mt-[30px] w-full rounded-[12px] border px-5 py-10 lg:max-w-[1256px] lg:px-6 lg:py-[55px]">
57+
<div className="mx-auto mt-[30px] w-full rounded-[12px] border border-brand-green-5 bg-brand-1/20 px-5 py-10 lg:max-w-[1256px] lg:px-6 lg:py-[55px]">
5858
<div className="mx-auto max-w-[1030px]">
59-
<div className="border-brand-green-1/80 bg-brand-gray-3/15 mx-auto flex w-full max-w-[690px] items-center gap-3 rounded-[12px] border px-5 py-4 md:px-6">
59+
<div className="mx-auto flex w-full max-w-[690px] items-center gap-3 rounded-[12px] border border-brand-green-1/80 bg-brand-gray-3/15 px-5 py-4 md:px-6">
6060
<CustomIcons.search className="size-4 shrink-0" />
6161
<input
6262
type="text"
63-
className="text-brand-gray-2 flex-1 bg-transparent text-sm/[18px] focus:outline-none"
63+
name="project-search"
64+
data-testid="project-search"
65+
className="flex-1 bg-transparent text-sm/[18px] text-brand-gray-2 focus:outline-none"
6466
placeholder="Search project"
6567
onChange={(e) => {
6668
const value = e.target.value.trim();
@@ -76,8 +78,9 @@ export default function RepositoriesSection({ projects }: Props) {
7678
{filterLanguages.map((lang) => (
7779
<button
7880
key={lang}
81+
data-testid={`btn-filter-${lang}`}
7982
className={cn(
80-
"min-w-[40px] shrink-0 whitespace-nowrap rounded-full border px-2.5 py-1 font-ff-poppins text-sm text-white",
83+
"btn-filter min-w-[40px] shrink-0 whitespace-nowrap rounded-full border px-2.5 py-1 font-ff-poppins text-sm text-white",
8184
lang === selectedLanguage
8285
? "border-brand-green-2/80 bg-brand-gray-7"
8386
: "border-brand-gray-9/80",
@@ -97,27 +100,28 @@ export default function RepositoriesSection({ projects }: Props) {
97100
{searchedProjects.map((project, index) => (
98101
<div
99102
key={index}
100-
className="border-brand-green-5/80 bg-brand-gray-4/[.53] relative isolate overflow-hidden rounded-[12px] border px-4 pb-5 pt-7"
103+
className="project relative isolate overflow-hidden rounded-[12px] border border-brand-green-5/80 bg-brand-gray-4/[.53] px-4 pb-5 pt-7"
101104
>
102105
<CustomIcons.ossIcon className="absolute right-0 top-0 -z-[1]" />
103106
<div className="flex justify-end">
104-
<span className="border-brand-gray-9/80 rounded-full border px-2.5 py-1 font-ff-poppins text-sm text-white">
107+
<span className="rounded-full border border-brand-gray-9/80 px-2.5 py-1 font-ff-poppins text-sm text-white">
105108
{project.issuesCount} Issues
106109
</span>
107110
</div>
108111
<div className="mb-5 mt-[105px] space-y-2">
109-
<p className="text-brand-gray-5 text-lg/[21px] font-semibold">
112+
<p className="text-lg/[21px] font-semibold text-brand-gray-5">
110113
{project.repository}
111114
</p>
112115
<p className="text-sm/4 text-white">{project.description}</p>
113-
<p className="text-brand-gray-6 text-sm/4">
116+
<p className="text-sm/4 text-brand-gray-6">
114117
Lang: {project.languages.join(", ")}
115118
</p>
116119
</div>
117120

118121
<Link
119122
to={project.link}
120-
className="border-brand-green-3 bg-brand-green-4 block w-max rounded-[30px] border px-4 py-2"
123+
data-testid={"learn-more-link"}
124+
className="block w-max rounded-[30px] border border-brand-green-3 bg-brand-green-4 px-4 py-2"
121125
>
122126
Learn more
123127
</Link>

‎tests/setup.ts

+7
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
import "@testing-library/jest-dom/vitest";
2+
import { afterEach } from "vitest";
3+
import { cleanup } from "@testing-library/react";
4+
5+
afterEach(() => {
6+
cleanup();
7+
});

‎tests/utils/index.tsx

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
import {
2+
createRootRoute,
3+
createRoute,
4+
createRouter,
5+
RouterProvider,
6+
} from "@tanstack/react-router";
7+
import { render } from "@testing-library/react";
8+
import React from "react";
9+
10+
/**
11+
* Utility function for rendering a React component with routing context using `@tanstack/react-router`.
12+
* Useful for testing components that rely on routing or are part of a routed application.
13+
*/
14+
export const renderWithRouter = (Comp: React.ComponentType) => {
15+
const rootRoute = createRootRoute();
16+
17+
const indexRoute = createRoute({
18+
getParentRoute: () => rootRoute,
19+
path: "/",
20+
component: () => <Comp />,
21+
});
22+
23+
const router = createRouter({
24+
routeTree: rootRoute.addChildren([indexRoute]),
25+
});
26+
27+
return render(
28+
<RouterProvider
29+
router={
30+
// suppress router type mismatch
31+
router as unknown as Parameters<typeof RouterProvider>[0]["router"]
32+
}
33+
/>,
34+
);
35+
};

‎tsconfig.app.json

+2-1
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@
55
"lib": ["ES2020", "DOM", "DOM.Iterable"],
66
"module": "ESNext",
77
"skipLibCheck": true,
8+
"types": ["vitest/globals"],
89

910
/* Bundler mode */
1011
"moduleResolution": "Bundler",
@@ -26,5 +27,5 @@
2627
"~/*": ["./src/*"]
2728
}
2829
},
29-
"include": ["src"]
30+
"include": ["src", "tests/setup.ts"]
3031
}

‎tsconfig.node.json

+3-2
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@
44
"lib": ["ES2023"],
55
"module": "ESNext",
66
"skipLibCheck": true,
7+
"types": ["vitest/globals"],
78

89
/* Bundler mode */
910
"moduleResolution": "Bundler",
@@ -18,11 +19,11 @@
1819
"noUnusedParameters": true,
1920
"noFallthroughCasesInSwitch": true,
2021
"noUncheckedSideEffectImports": true,
21-
"baseUrl": ".",
22+
"baseUrl": ".",
2223
"paths": {
2324
"@/*": ["./src/*"],
2425
"~/*": ["./src/*"]
2526
}
2627
},
27-
"include": ["vite.config.ts"]
28+
"include": ["vite.config.ts", "tests/setup.ts"]
2829
}

‎vite.config.ts

+3
Original file line numberDiff line numberDiff line change
@@ -25,7 +25,10 @@ export default defineConfig({
2525
host: true,
2626
},
2727
test: {
28+
globals: true,
29+
environment: "jsdom",
2830
include: ["./src/**/*.{test,spec}.?(c|m)[jt]s?(x)"],
2931
exclude: ["e2e-tests", "node_modules"],
32+
setupFiles: "./tests/setup.ts",
3033
},
3134
});

0 commit comments

Comments
 (0)
Please sign in to comment.