| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368 |
- /**
- * Tests for the StreamOverlayPage component.
- */
- import { describe, it, expect, beforeEach, afterEach, vi } from 'vitest';
- import { screen, waitFor, render as rtlRender } from '@testing-library/react';
- import { StreamOverlayPage } from '../../pages/StreamOverlayPage';
- import { http, HttpResponse } from 'msw';
- import { server } from '../mocks/server';
- import { MemoryRouter, Route, Routes } from 'react-router-dom';
- import { QueryClient, QueryClientProvider } from '@tanstack/react-query';
- import { ThemeProvider } from '../../contexts/ThemeContext';
- import { ToastProvider } from '../../contexts/ToastContext';
- const mockPrinter = {
- id: 1,
- name: 'X1 Carbon',
- ip_address: '192.168.1.100',
- serial_number: '00M09A350100001',
- access_code: '12345678',
- model: 'X1C',
- enabled: true,
- };
- const mockStatusIdle = {
- id: 1,
- name: 'X1 Carbon',
- connected: true,
- state: 'IDLE',
- progress: 0,
- current_print: null,
- remaining_time: null,
- layer_num: null,
- total_layers: null,
- stg_cur_name: null,
- };
- const mockStatusPrinting = {
- id: 1,
- name: 'X1 Carbon',
- connected: true,
- state: 'RUNNING',
- progress: 45,
- current_print: 'Benchy.gcode.3mf',
- remaining_time: 82,
- layer_num: 150,
- total_layers: 300,
- stg_cur_name: null,
- };
- // Custom render for StreamOverlayPage
- function renderOverlayPage(printerId: number, queryParams = '') {
- const queryClient = new QueryClient({
- defaultOptions: {
- queries: { retry: false, gcTime: 0 },
- mutations: { retry: false },
- },
- });
- return rtlRender(
- <QueryClientProvider client={queryClient}>
- <MemoryRouter initialEntries={[`/overlay/${printerId}${queryParams}`]}>
- <ThemeProvider>
- <ToastProvider>
- <Routes>
- <Route path="/overlay/:printerId" element={<StreamOverlayPage />} />
- </Routes>
- </ToastProvider>
- </ThemeProvider>
- </MemoryRouter>
- </QueryClientProvider>
- );
- }
- describe('StreamOverlayPage', () => {
- const originalTitle = document.title;
- beforeEach(() => {
- // Mock WebSocket. vitest 4 dropped support for arrow-function constructor
- // mocks (`new (() => ...)` throws "is not a constructor"); use a plain
- // function so `new WebSocket(...)` resolves correctly.
- vi.stubGlobal(
- 'WebSocket',
- vi.fn().mockImplementation(function (this: { close: () => void; onmessage: null; onerror: null }) {
- this.close = vi.fn();
- this.onmessage = null;
- this.onerror = null;
- }),
- );
- server.use(
- http.get('/api/v1/printers/:id', () => {
- return HttpResponse.json(mockPrinter);
- }),
- http.get('/api/v1/printers/:id/status', () => {
- return HttpResponse.json(mockStatusIdle);
- })
- );
- });
- afterEach(() => {
- document.title = originalTitle;
- vi.unstubAllGlobals();
- });
- describe('rendering', () => {
- it('renders overlay page for printer', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- expect(screen.getByText('Printer is idle')).toBeInTheDocument();
- });
- });
- it('shows Bambuddy logo', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- expect(screen.getByAltText('Bambuddy')).toBeInTheDocument();
- });
- });
- it('logo links to GitHub', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- const logo = screen.getByAltText('Bambuddy');
- const link = logo.closest('a');
- expect(link).toHaveAttribute('href', 'https://github.com/maziggy/bambuddy');
- });
- });
- });
- describe('printing state', () => {
- beforeEach(() => {
- server.use(
- http.get('/api/v1/printers/:id/status', () => {
- return HttpResponse.json(mockStatusPrinting);
- })
- );
- });
- it('shows filename when printing', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- expect(screen.getByText('Benchy')).toBeInTheDocument();
- });
- });
- it('shows progress percentage', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- expect(screen.getByText('45%')).toBeInTheDocument();
- });
- });
- it('shows layer count', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- expect(screen.getByText('150')).toBeInTheDocument();
- expect(screen.getByText('300')).toBeInTheDocument();
- });
- });
- it('shows status text', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- expect(screen.getByText('Printing')).toBeInTheDocument();
- });
- });
- });
- describe('invalid printer', () => {
- it('shows invalid printer message for ID 0', async () => {
- renderOverlayPage(0);
- await waitFor(() => {
- expect(screen.getByText('Invalid printer ID')).toBeInTheDocument();
- });
- });
- });
- describe('query parameters', () => {
- it('respects size parameter', async () => {
- renderOverlayPage(1, '?size=large');
- await waitFor(() => {
- // Just verify it renders without error
- expect(screen.getByAltText('Bambuddy')).toBeInTheDocument();
- });
- });
- it('respects show parameter to hide elements', async () => {
- server.use(
- http.get('/api/v1/printers/:id/status', () => {
- return HttpResponse.json(mockStatusPrinting);
- })
- );
- renderOverlayPage(1, '?show=progress');
- await waitFor(() => {
- // Progress should be visible
- expect(screen.getByText('45%')).toBeInTheDocument();
- // Status text should be hidden when not in show list
- expect(screen.queryByText('Printing')).not.toBeInTheDocument();
- });
- });
- });
- describe('FPS configuration', () => {
- it('uses default FPS of 15 when not specified', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- const img = screen.getByAltText('Camera stream') as HTMLImageElement;
- expect(img.src).toContain('fps=15');
- });
- });
- it('uses custom FPS when specified in query params', async () => {
- renderOverlayPage(1, '?fps=30');
- await waitFor(() => {
- const img = screen.getByAltText('Camera stream') as HTMLImageElement;
- expect(img.src).toContain('fps=30');
- });
- });
- it('clamps FPS to maximum of 30', async () => {
- renderOverlayPage(1, '?fps=60');
- await waitFor(() => {
- const img = screen.getByAltText('Camera stream') as HTMLImageElement;
- expect(img.src).toContain('fps=30');
- });
- });
- it('clamps FPS to minimum of 1', async () => {
- renderOverlayPage(1, '?fps=0');
- await waitFor(() => {
- const img = screen.getByAltText('Camera stream') as HTMLImageElement;
- expect(img.src).toContain('fps=1');
- });
- });
- it('handles invalid FPS value gracefully', async () => {
- renderOverlayPage(1, '?fps=invalid');
- await waitFor(() => {
- const img = screen.getByAltText('Camera stream') as HTMLImageElement;
- // Should fall back to default of 15
- expect(img.src).toContain('fps=15');
- });
- });
- });
- describe('camera toggle (status-only mode)', () => {
- it('shows camera by default', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- expect(screen.getByAltText('Camera stream')).toBeInTheDocument();
- });
- });
- it('hides camera when camera=false', async () => {
- renderOverlayPage(1, '?camera=false');
- await waitFor(() => {
- // Status should still be visible
- expect(screen.getByText('Printer is idle')).toBeInTheDocument();
- });
- // Camera should not be rendered
- expect(screen.queryByAltText('Camera stream')).not.toBeInTheDocument();
- });
- it('hides camera when camera=0', async () => {
- renderOverlayPage(1, '?camera=0');
- await waitFor(() => {
- expect(screen.getByText('Printer is idle')).toBeInTheDocument();
- });
- expect(screen.queryByAltText('Camera stream')).not.toBeInTheDocument();
- });
- it('shows camera when camera=true', async () => {
- renderOverlayPage(1, '?camera=true');
- await waitFor(() => {
- expect(screen.getByAltText('Camera stream')).toBeInTheDocument();
- });
- });
- it('shows camera when camera=1', async () => {
- renderOverlayPage(1, '?camera=1');
- await waitFor(() => {
- expect(screen.getByAltText('Camera stream')).toBeInTheDocument();
- });
- });
- });
- describe('combined parameters', () => {
- it('supports fps and camera together', async () => {
- renderOverlayPage(1, '?fps=25&camera=true');
- await waitFor(() => {
- const img = screen.getByAltText('Camera stream') as HTMLImageElement;
- expect(img.src).toContain('fps=25');
- });
- });
- it('supports status-only with custom size', async () => {
- renderOverlayPage(1, '?camera=false&size=large');
- await waitFor(() => {
- expect(screen.getByText('Printer is idle')).toBeInTheDocument();
- });
- expect(screen.queryByAltText('Camera stream')).not.toBeInTheDocument();
- });
- it('supports show parameter with fps', async () => {
- server.use(
- http.get('/api/v1/printers/:id/status', () => {
- return HttpResponse.json(mockStatusPrinting);
- })
- );
- renderOverlayPage(1, '?fps=20&show=progress');
- await waitFor(() => {
- const img = screen.getByAltText('Camera stream') as HTMLImageElement;
- expect(img.src).toContain('fps=20');
- expect(screen.getByText('45%')).toBeInTheDocument();
- });
- });
- });
- describe('offline state', () => {
- beforeEach(() => {
- server.use(
- http.get('/api/v1/printers/:id/status', () => {
- return HttpResponse.json({
- ...mockStatusIdle,
- connected: false,
- });
- })
- );
- });
- it('shows offline message when printer disconnected', async () => {
- renderOverlayPage(1);
- await waitFor(() => {
- expect(screen.getByText('Printer offline')).toBeInTheDocument();
- });
- });
- });
- });
|