Files
familienarchiv/frontend/src/routes/DropZone.svelte.test.ts
Marcel 251891fbed test(routes): rewrite DropZone test with behavioral assertions
Replaces 5 setTimeout sleeps with vi.waitFor on the actual class
transition, and converts 6 .not.toThrow smoke tests into assertions
that the validation guard surfaces the expected error message (or
absence thereof). Tightens the dragging-state regex to bg-accent-bg
so it cannot match the idle hover:border-primary substring.

Runtime: faster + deterministic.

Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
2026-05-11 21:50:28 +02:00

210 lines
8.2 KiB
TypeScript

import { describe, it, expect, vi, afterEach } from 'vitest';
import { cleanup, render } from 'vitest-browser-svelte';
import { page } from 'vitest/browser';
vi.mock('$app/navigation', () => ({
beforeNavigate: () => {},
afterNavigate: () => {},
goto: vi.fn(),
invalidate: vi.fn(),
invalidateAll: vi.fn(),
preloadCode: vi.fn(),
preloadData: vi.fn(),
pushState: vi.fn(),
replaceState: vi.fn(),
disableScrollHandling: vi.fn(),
onNavigate: () => () => {}
}));
const { default: DropZone } = await import('./DropZone.svelte');
afterEach(cleanup);
describe('DropZone', () => {
it('renders the drop hint and accepted types by default', async () => {
render(DropZone, { props: {} });
await expect.element(page.getByText(/einzeln oder mehrere/i)).toBeVisible();
await expect.element(page.getByText('PDF, JPEG, PNG, TIFF')).toBeVisible();
});
it('does not render the progress bar by default', async () => {
render(DropZone, { props: {} });
expect(document.querySelector('.bg-primary.h-full')).toBeNull();
});
it('rejects files with unaccepted MIME types and shows an error message', async () => {
render(DropZone, { props: {} });
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
const badFile = new File(['bad'], 'doc.docx', {
type: 'application/vnd.openxmlformats-officedocument.wordprocessingml.document'
});
Object.defineProperty(input, 'files', { value: [badFile], writable: false });
input.dispatchEvent(new Event('change', { bubbles: true }));
await expect.element(page.getByText(/Dateiformat nicht unterstützt/i)).toBeVisible();
});
it('accepts a PDF file as a valid type and renders no "invalid type" message', async () => {
const onUploadComplete = vi.fn();
render(DropZone, { props: { onUploadComplete } });
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
const pdfFile = new File(['%PDF'], 'brief.pdf', { type: 'application/pdf' });
Object.defineProperty(input, 'files', { value: [pdfFile], writable: false });
input.dispatchEvent(new Event('change', { bubbles: true }));
// The validation guard never raises an invalid-type error for application/pdf.
expect(document.body.textContent).not.toMatch(/Dateiformat nicht unterstützt/i);
});
it('returns early when no files are selected', async () => {
render(DropZone, { props: {} });
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
Object.defineProperty(input, 'files', { value: [], writable: false });
input.dispatchEvent(new Event('change', { bubbles: true }));
const errors = document.querySelectorAll('.text-red-600');
expect(errors.length).toBe(0);
});
it('opens the file input when the drop zone is clicked', async () => {
render(DropZone, { props: {} });
const dropZone = document.querySelector('div[role="button"]') as HTMLElement;
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
const clickSpy = vi.spyOn(input, 'click');
dropZone.click();
expect(clickSpy).toHaveBeenCalled();
});
it('opens the file input when Enter is pressed on the drop zone', async () => {
render(DropZone, { props: {} });
const dropZone = document.querySelector('div[role="button"]') as HTMLElement;
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
const clickSpy = vi.spyOn(input, 'click');
dropZone.dispatchEvent(new KeyboardEvent('keydown', { key: 'Enter', bubbles: true }));
expect(clickSpy).toHaveBeenCalled();
});
it('exposes file input as multi-file with accept whitelist', async () => {
render(DropZone, { props: {} });
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
expect(input.multiple).toBe(true);
expect(input.accept).toContain('.pdf');
});
it('applies the dragging style on dragover', async () => {
render(DropZone, { props: {} });
const dropZone = document.querySelector('div[role="button"]') as HTMLElement;
dropZone.dispatchEvent(new DragEvent('dragover', { bubbles: true }));
// isDragging=true switches the class to border-primary bg-accent-bg — wait for the next paint.
await vi.waitFor(() => {
expect(dropZone.className).toMatch(/bg-accent-bg/);
});
});
it('drops dragging style on dragleave', async () => {
render(DropZone, { props: {} });
const dropZone = document.querySelector('div[role="button"]') as HTMLElement;
dropZone.dispatchEvent(new DragEvent('dragover', { bubbles: true }));
await vi.waitFor(() => {
expect(dropZone.className).toMatch(/bg-accent-bg/);
});
dropZone.dispatchEvent(new DragEvent('dragleave', { bubbles: true }));
await vi.waitFor(() => {
expect(dropZone.className).not.toMatch(/bg-accent-bg/);
});
});
it('drop event with no files is a no-op (no error message)', async () => {
render(DropZone, { props: {} });
const dropZone = document.querySelector('div[role="button"]') as HTMLElement;
const dropEvent = new DragEvent('drop', { bubbles: true });
Object.defineProperty(dropEvent, 'dataTransfer', { value: { files: [] }, writable: false });
dropZone.dispatchEvent(dropEvent);
expect(document.querySelectorAll('.text-red-600')).toHaveLength(0);
});
it('rejects multiple invalid files and lists one error message per file', async () => {
render(DropZone, { props: {} });
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
const f1 = new File(['x'], 'a.docx', { type: 'application/x-bad' });
const f2 = new File(['y'], 'b.txt', { type: 'text/plain' });
Object.defineProperty(input, 'files', { value: [f1, f2], writable: false });
input.dispatchEvent(new Event('change', { bubbles: true }));
await vi.waitFor(() => {
expect(document.querySelectorAll('.text-red-600')).toHaveLength(2);
});
});
it('mixed valid+invalid files raises an error only for the invalid one', async () => {
render(DropZone, { props: {} });
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
const f1 = new File(['x'], 'a.docx', { type: 'application/x-bad' });
const f2 = new File(['%PDF'], 'b.pdf', { type: 'application/pdf' });
Object.defineProperty(input, 'files', { value: [f1, f2], writable: false });
input.dispatchEvent(new Event('change', { bubbles: true }));
await expect.element(page.getByText(/Dateiformat nicht unterstützt/i)).toBeVisible();
});
it('Enter handler ignores other keys', async () => {
render(DropZone, { props: {} });
const dropZone = document.querySelector('div[role="button"]') as HTMLElement;
const input = document.querySelector('input[type="file"]') as HTMLInputElement;
const clickSpy = vi.spyOn(input, 'click');
dropZone.dispatchEvent(new KeyboardEvent('keydown', { key: 'a', bubbles: true }));
expect(clickSpy).not.toHaveBeenCalled();
});
it('responds to window-level dragenter only when dataTransfer.types includes "Files"', async () => {
render(DropZone, { props: {} });
const dropZone = document.querySelector('div[role="button"]') as HTMLElement;
// Non-files dragenter should not trigger the windowDragging style.
const evt1 = new DragEvent('dragenter', { bubbles: true });
Object.defineProperty(evt1, 'dataTransfer', { value: { types: ['text/html'] } });
window.dispatchEvent(evt1);
expect(dropZone.className).not.toMatch(/bg-accent-bg/);
// Files dragenter flips windowDragging=true → drop-zone gains the border-primary style.
const evt2 = new DragEvent('dragenter', { bubbles: true });
Object.defineProperty(evt2, 'dataTransfer', { value: { types: ['Files'] } });
window.dispatchEvent(evt2);
await vi.waitFor(() => {
expect(dropZone.className).toMatch(/bg-accent-bg/);
});
// Trailing window drop to clean up.
window.dispatchEvent(new DragEvent('drop', { bubbles: true }));
});
it('window-level dragleave without prior dragenter is safe (counter does not go negative)', async () => {
render(DropZone, { props: {} });
const dropZone = document.querySelector('div[role="button"]') as HTMLElement;
// Two consecutive dragleaves on the window without dragenters should leave the drop-zone
// in its idle (non-highlighted) state.
window.dispatchEvent(new DragEvent('dragleave', { bubbles: true }));
window.dispatchEvent(new DragEvent('dragleave', { bubbles: true }));
expect(dropZone.className).not.toMatch(/bg-accent-bg/);
});
});