import React from 'react';
import { describe, it, expect, beforeEach, vi } from 'vitest';
import { render, screen, waitFor, act } from '../../tests/helpers/render';
import { Route, Routes } from 'react-router-dom';
import { http, HttpResponse } from 'msw';
import { server } from '../../tests/helpers/msw/server';
import { resetAllStores, seedStore } from '../../tests/helpers/store';
import { buildUser, buildTrip, buildTripFile } from '../../tests/helpers/factories';
import { useAuthStore } from '../store/authStore';
import { useTripStore } from '../store/tripStore';
import FilesPage from './FilesPage';
vi.mock('../components/Files/FileManager', () => ({
default: ({ files }: { files: unknown[]; onUpload: unknown; onDelete: unknown }) =>
React.createElement('div', { 'data-testid': 'file-manager' }, `${files.length} files`),
}));
vi.mock('../components/Layout/Navbar', () => ({
default: ({ tripTitle }: { tripTitle?: string }) =>
React.createElement('nav', { 'data-testid': 'navbar' }, tripTitle),
}));
function renderFilesPage(tripId: number | string = 1) {
return render(
} />
,
{ initialEntries: [`/trips/${tripId}/files`] },
);
}
beforeEach(() => {
vi.clearAllMocks();
resetAllStores();
seedStore(useAuthStore, { isAuthenticated: true, user: buildUser() });
seedStore(useTripStore, {
files: [],
loadFiles: vi.fn().mockResolvedValue(undefined),
addFile: vi.fn().mockResolvedValue(undefined),
deleteFile: vi.fn().mockResolvedValue(undefined),
} as any);
});
describe('FilesPage', () => {
describe('FE-PAGE-FILES-001: Loading spinner shown while data fetches', () => {
it('shows a spinner while data is loading', async () => {
server.use(
http.get('/api/trips/:id', async () => {
await new Promise(resolve => setTimeout(resolve, 200));
const trip = buildTrip({ id: 1 });
return HttpResponse.json({ trip });
}),
);
renderFilesPage(1);
expect(document.querySelector('.animate-spin')).toBeInTheDocument();
});
});
describe('FE-PAGE-FILES-002: Trip name displayed in Navbar after load', () => {
it('passes the trip name to Navbar after data loads', async () => {
const trip = buildTrip({ id: 1, name: 'Rome Trip' });
server.use(
http.get('/api/trips/:id', () => HttpResponse.json({ trip })),
);
renderFilesPage(1);
await waitFor(() => {
expect(screen.getByTestId('navbar')).toHaveTextContent('Rome Trip');
});
});
});
describe('FE-PAGE-FILES-003: FileManager renders after load', () => {
it('renders the FileManager after data loads', async () => {
renderFilesPage(1);
await waitFor(() => {
expect(screen.getByTestId('file-manager')).toBeInTheDocument();
});
});
});
describe('FE-PAGE-FILES-004: File count shown in header', () => {
it('shows the correct file count in the header', async () => {
const file1 = buildTripFile();
const file2 = buildTripFile();
seedStore(useTripStore, {
files: [file1, file2],
loadFiles: vi.fn().mockResolvedValue(undefined),
addFile: vi.fn().mockResolvedValue(undefined),
deleteFile: vi.fn().mockResolvedValue(undefined),
} as any);
renderFilesPage(1);
await waitFor(() => {
expect(screen.getByTestId('file-manager')).toBeInTheDocument();
});
expect(screen.getByText(/2 files for/i)).toBeInTheDocument();
});
});
describe('FE-PAGE-FILES-005: Back link navigates to trip planner', () => {
it('back link points to the trip planner page', async () => {
renderFilesPage(1);
await waitFor(() => {
expect(screen.getByTestId('file-manager')).toBeInTheDocument();
});
const backLink = screen.getByRole('link', { name: /back to planning/i });
expect(backLink.getAttribute('href')).toContain('/trips/1');
});
});
describe('FE-PAGE-FILES-006: loadFiles is called with trip ID on mount', () => {
it('calls tripStore.loadFiles with the trip ID from the URL', async () => {
const mockLoadFiles = vi.fn().mockResolvedValue(undefined);
seedStore(useTripStore, {
files: [],
loadFiles: mockLoadFiles,
addFile: vi.fn().mockResolvedValue(undefined),
deleteFile: vi.fn().mockResolvedValue(undefined),
} as any);
renderFilesPage(1);
await waitFor(() => {
expect(mockLoadFiles).toHaveBeenCalledWith('1');
});
});
});
describe('FE-PAGE-FILES-007: Navigation to /dashboard on fetch error', () => {
it('navigates to /dashboard when trip fetch fails', async () => {
server.use(
http.get('/api/trips/:id', () =>
HttpResponse.json({ error: 'Not found' }, { status: 404 }),
),
);
render(
} />
Dashboard} />
,
{ initialEntries: ['/trips/1/files'] },
);
await waitFor(() => {
expect(screen.getByTestId('dashboard')).toBeInTheDocument();
});
});
});
describe('FE-PAGE-FILES-008: Files update when tripStore.files changes', () => {
it('FileManager re-renders when store files change', async () => {
seedStore(useTripStore, {
files: [],
loadFiles: vi.fn().mockResolvedValue(undefined),
addFile: vi.fn().mockResolvedValue(undefined),
deleteFile: vi.fn().mockResolvedValue(undefined),
} as any);
renderFilesPage(1);
await waitFor(() => {
expect(screen.getByTestId('file-manager')).toBeInTheDocument();
});
expect(screen.getByTestId('file-manager')).toHaveTextContent('0 files');
// Simulate store update
act(() => {
useTripStore.setState({ files: [buildTripFile({ id: 99, original_name: 'document.pdf' })] } as any);
});
await waitFor(() => {
expect(screen.getByTestId('file-manager')).toHaveTextContent('1 files');
});
});
});
describe('FE-PAGE-FILES-009: Empty file list renders FileManager with 0 files', () => {
it('renders FileManager with 0 files when files array is empty', async () => {
renderFilesPage(1);
await waitFor(() => {
expect(screen.getByTestId('file-manager')).toBeInTheDocument();
});
expect(screen.getByTestId('file-manager')).toHaveTextContent('0 files');
});
});
describe('FE-PAGE-FILES-010: Page title heading present', () => {
it('renders the "Dateien & Dokumente" heading', async () => {
renderFilesPage(1);
await waitFor(() => {
expect(screen.getByTestId('file-manager')).toBeInTheDocument();
});
expect(screen.getByRole('heading', { name: /Files & Documents/i })).toBeInTheDocument();
});
});
});