test: improve frontend test coverage (#163)

- Export DashboardPage helper functions for testability
- Add new test files: App, SharedSchedule, AppContext, UnsavedChangesContext, useUnsavedChangesWarning
- Expand existing test coverage for Auth, MedDetailModal, MobileEditModal, DashboardPage, MedicationsPage, PlannerPage, and more
- Add edge case and error handling tests across components, hooks, and pages
This commit is contained in:
Daniel Volz
2026-02-13 18:34:19 +01:00
committed by GitHub
parent 0b0472f2f5
commit 5c09f97cb3
24 changed files with 4482 additions and 45 deletions
@@ -6,6 +6,7 @@ import { AuthProvider } from "../../components/Auth";
// Mock useNavigate
const mockNavigate = vi.fn();
const mockConfirmNavigation = vi.fn();
vi.mock("react-router-dom", async () => {
const actual = await vi.importActual("react-router-dom");
return {
@@ -19,7 +20,7 @@ vi.mock("../../context", () => ({
useUnsavedChanges: () => ({
setHasUnsavedChanges: vi.fn(),
hasUnsavedChanges: false,
confirmNavigation: vi.fn().mockReturnValue(true),
confirmNavigation: mockConfirmNavigation,
}),
}));
@@ -27,6 +28,7 @@ describe("AppHeader", () => {
beforeEach(() => {
vi.clearAllMocks();
mockNavigate.mockClear();
mockConfirmNavigation.mockResolvedValue(true);
// Set up default auth mock - auth disabled
(global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({
@@ -281,4 +283,97 @@ describe("AppHeader", () => {
}
});
});
it("does not navigate when unsaved changes confirmation is denied", async () => {
mockConfirmNavigation.mockResolvedValueOnce(false);
render(
<MemoryRouter initialEntries={["/dashboard"]}>
<AuthProvider>
<AppHeader onOpenProfile={vi.fn()} onOpenAbout={vi.fn()} />
</AuthProvider>
</MemoryRouter>
);
await waitFor(() => {
const buttons = screen.getAllByRole("button");
const plannerBtn = buttons.find((btn) => btn.textContent?.includes("nav.planner"));
expect(plannerBtn).toBeInTheDocument();
if (plannerBtn) fireEvent.click(plannerBtn);
});
await waitFor(() => {
expect(mockConfirmNavigation).toHaveBeenCalled();
expect(mockNavigate).not.toHaveBeenCalledWith("/planner");
});
});
it("renders authenticated user menu and handles profile/about/settings/logout actions", async () => {
const onOpenProfile = vi.fn();
const onOpenAbout = vi.fn();
(global.fetch as ReturnType<typeof vi.fn>).mockReset();
mockNavigate.mockClear();
mockConfirmNavigation.mockResolvedValue(true);
(global.fetch as ReturnType<typeof vi.fn>)
.mockResolvedValueOnce({
ok: true,
json: () =>
Promise.resolve({
authEnabled: true,
registrationEnabled: true,
localAuthEnabled: true,
oidcEnabled: false,
oidcProviderName: "",
hasUsers: true,
needsSetup: false,
}),
})
.mockResolvedValueOnce({
ok: true,
json: () => Promise.resolve({ id: 1, username: "tester", avatarUrl: null }),
})
.mockResolvedValueOnce({
ok: true,
});
const { container } = render(
<MemoryRouter initialEntries={["/dashboard"]}>
<AuthProvider>
<AppHeader onOpenProfile={onOpenProfile} onOpenAbout={onOpenAbout} />
</AuthProvider>
</MemoryRouter>
);
await waitFor(() => {
expect(container.querySelector(".user-menu-btn")).toBeInTheDocument();
});
// Settings icon should not be shown when auth is enabled
expect(screen.queryByTitle(/nav\.settings/i)).not.toBeInTheDocument();
const userMenuBtn = container.querySelector(".user-menu-btn") as HTMLButtonElement;
fireEvent.click(userMenuBtn);
fireEvent.click(screen.getByText(/auth\.profile/i));
expect(onOpenProfile).toHaveBeenCalled();
fireEvent.click(userMenuBtn);
fireEvent.click(screen.getByText(/about\.title/i));
expect(onOpenAbout).toHaveBeenCalled();
fireEvent.click(userMenuBtn);
fireEvent.click(screen.getByText(/^nav\.settings$/i));
await waitFor(() => {
expect(mockNavigate).toHaveBeenCalledWith("/settings");
});
fireEvent.click(userMenuBtn);
fireEvent.click(screen.getByText(/auth\.signOut/i));
await waitFor(() => {
expect(fetch).toHaveBeenCalledWith("/api/auth/logout", {
method: "POST",
credentials: "include",
});
});
});
});