SettingsPage.test.tsx 6.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239
  1. /**
  2. * Tests for the SettingsPage component.
  3. */
  4. import { describe, it, expect, beforeEach } from 'vitest';
  5. import { screen, waitFor } from '@testing-library/react';
  6. import userEvent from '@testing-library/user-event';
  7. import { render } from '../utils';
  8. import { SettingsPage } from '../../pages/SettingsPage';
  9. import { http, HttpResponse } from 'msw';
  10. import { server } from '../mocks/server';
  11. const mockSettings = {
  12. auto_archive: true,
  13. save_thumbnails: true,
  14. capture_finish_photo: true,
  15. default_filament_cost: 25.0,
  16. currency: 'USD',
  17. ams_humidity_good: 40,
  18. ams_humidity_fair: 60,
  19. ams_temp_good: 30,
  20. ams_temp_fair: 35,
  21. time_format: 'system',
  22. date_format: 'system',
  23. mqtt_enabled: false,
  24. mqtt_host: '',
  25. mqtt_port: 1883,
  26. spoolman_enabled: false,
  27. spoolman_url: '',
  28. ha_enabled: false,
  29. ha_url: '',
  30. ha_token: '',
  31. check_updates: false,
  32. check_printer_firmware: false,
  33. bed_cooled_threshold: 35,
  34. };
  35. describe('SettingsPage', () => {
  36. beforeEach(() => {
  37. server.use(
  38. http.get('/api/v1/settings/', () => {
  39. return HttpResponse.json(mockSettings);
  40. }),
  41. http.patch('/api/v1/settings/', async ({ request }) => {
  42. const body = await request.json();
  43. return HttpResponse.json({ ...mockSettings, ...body });
  44. }),
  45. http.get('/api/v1/printers/', () => {
  46. return HttpResponse.json([]);
  47. }),
  48. http.get('/api/v1/smart-plugs/', () => {
  49. return HttpResponse.json([]);
  50. }),
  51. http.get('/api/v1/notifications/', () => {
  52. return HttpResponse.json([]);
  53. }),
  54. http.get('/api/v1/api-keys/', () => {
  55. return HttpResponse.json([]);
  56. }),
  57. http.get('/api/v1/mqtt/status', () => {
  58. return HttpResponse.json({ enabled: false });
  59. }),
  60. http.get('/api/v1/virtual-printer/status', () => {
  61. return HttpResponse.json({ running: false });
  62. }),
  63. http.get('/api/v1/auth/status', () => {
  64. return HttpResponse.json({ auth_enabled: false, requires_setup: false });
  65. })
  66. );
  67. });
  68. describe('rendering', () => {
  69. it('renders the page title', async () => {
  70. render(<SettingsPage />);
  71. await waitFor(() => {
  72. // Use role-based query to avoid conflicts with dropdown options
  73. expect(screen.getByRole('heading', { name: 'Settings' })).toBeInTheDocument();
  74. });
  75. });
  76. it('shows settings tabs', async () => {
  77. render(<SettingsPage />);
  78. await waitFor(() => {
  79. // Use getAllByText since "General" appears both as tab and section heading
  80. expect(screen.getAllByText('General').length).toBeGreaterThan(0);
  81. expect(screen.getByText('Smart Plugs')).toBeInTheDocument();
  82. expect(screen.getByText('Notifications')).toBeInTheDocument();
  83. expect(screen.getAllByText('Filament').length).toBeGreaterThan(0);
  84. expect(screen.getByText('Network')).toBeInTheDocument();
  85. expect(screen.getByText('API Keys')).toBeInTheDocument();
  86. });
  87. });
  88. });
  89. describe('general settings', () => {
  90. it('shows date format setting', async () => {
  91. render(<SettingsPage />);
  92. await waitFor(() => {
  93. expect(screen.getByText('Date Format')).toBeInTheDocument();
  94. });
  95. });
  96. it('shows time format setting', async () => {
  97. render(<SettingsPage />);
  98. await waitFor(() => {
  99. expect(screen.getByText('Time Format')).toBeInTheDocument();
  100. });
  101. });
  102. it('shows default printer setting', async () => {
  103. render(<SettingsPage />);
  104. await waitFor(() => {
  105. expect(screen.getByText('Default Printer')).toBeInTheDocument();
  106. });
  107. });
  108. it('shows preferred slicer setting', async () => {
  109. render(<SettingsPage />);
  110. await waitFor(() => {
  111. expect(screen.getByText('Preferred Slicer')).toBeInTheDocument();
  112. });
  113. });
  114. it('shows slicer dropdown with both options', async () => {
  115. render(<SettingsPage />);
  116. await waitFor(() => {
  117. const slicerSelect = screen.getAllByDisplayValue('Bambu Studio');
  118. expect(slicerSelect.length).toBeGreaterThan(0);
  119. });
  120. });
  121. it('shows appearance section', async () => {
  122. render(<SettingsPage />);
  123. await waitFor(() => {
  124. expect(screen.getByText('Appearance')).toBeInTheDocument();
  125. });
  126. });
  127. it('shows updates section with firmware toggle', async () => {
  128. render(<SettingsPage />);
  129. await waitFor(() => {
  130. expect(screen.getByText('Updates')).toBeInTheDocument();
  131. expect(screen.getByText('Check for updates')).toBeInTheDocument();
  132. expect(screen.getByText('Check printer firmware')).toBeInTheDocument();
  133. });
  134. });
  135. });
  136. describe('tabs navigation', () => {
  137. it('can switch to Network tab', async () => {
  138. const user = userEvent.setup();
  139. render(<SettingsPage />);
  140. // Wait for settings to load first
  141. await waitFor(() => {
  142. expect(screen.getByText('Date Format')).toBeInTheDocument();
  143. });
  144. await user.click(screen.getByText('Network'));
  145. await waitFor(() => {
  146. // Network tab contains MQTT Publishing section
  147. expect(screen.getByText('MQTT Publishing')).toBeInTheDocument();
  148. });
  149. });
  150. it('can switch to Smart Plugs tab', async () => {
  151. const user = userEvent.setup();
  152. render(<SettingsPage />);
  153. await waitFor(() => {
  154. expect(screen.getByText('Smart Plugs')).toBeInTheDocument();
  155. });
  156. await user.click(screen.getByText('Smart Plugs'));
  157. await waitFor(() => {
  158. expect(screen.getByText('Add Smart Plug')).toBeInTheDocument();
  159. });
  160. });
  161. it('can switch to Notifications tab', async () => {
  162. const user = userEvent.setup();
  163. render(<SettingsPage />);
  164. await waitFor(() => {
  165. expect(screen.getByText('Notifications')).toBeInTheDocument();
  166. });
  167. await user.click(screen.getByText('Notifications'));
  168. await waitFor(() => {
  169. expect(screen.getByText('Add Provider')).toBeInTheDocument();
  170. });
  171. });
  172. it('can switch to Filament tab', async () => {
  173. const user = userEvent.setup();
  174. render(<SettingsPage />);
  175. await waitFor(() => {
  176. expect(screen.getAllByText('Filament').length).toBeGreaterThan(0);
  177. });
  178. await user.click(screen.getAllByText('Filament')[0]);
  179. await waitFor(() => {
  180. expect(screen.getByText('AMS Display Thresholds')).toBeInTheDocument();
  181. });
  182. });
  183. });
  184. describe('API Keys tab', () => {
  185. it('can switch to API Keys tab', async () => {
  186. const user = userEvent.setup();
  187. render(<SettingsPage />);
  188. await waitFor(() => {
  189. expect(screen.getByText('API Keys')).toBeInTheDocument();
  190. });
  191. await user.click(screen.getByText('API Keys'));
  192. await waitFor(() => {
  193. // Button text is "Create Key"
  194. expect(screen.getByText('Create Key')).toBeInTheDocument();
  195. });
  196. });
  197. });
  198. });