SettingsPage.test.tsx 7.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  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.getAllByText('Notifications').length).toBeGreaterThan(0);
  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.getAllByText('Notifications').length).toBeGreaterThan(0);
  166. });
  167. // Click the tab button (not the mobile dropdown option)
  168. const notificationButtons = screen.getAllByText('Notifications');
  169. const tabButton = notificationButtons.find(el => el.tagName === 'BUTTON') || notificationButtons[0];
  170. await user.click(tabButton);
  171. await waitFor(() => {
  172. expect(screen.getByText('Add Provider')).toBeInTheDocument();
  173. });
  174. });
  175. it('can switch to Filament tab', async () => {
  176. const user = userEvent.setup();
  177. render(<SettingsPage />);
  178. await waitFor(() => {
  179. expect(screen.getAllByText('Filament').length).toBeGreaterThan(0);
  180. });
  181. await user.click(screen.getAllByText('Filament')[0]);
  182. await waitFor(() => {
  183. expect(screen.getByText('AMS Display Thresholds')).toBeInTheDocument();
  184. });
  185. });
  186. });
  187. describe('API Keys tab', () => {
  188. it('can switch to API Keys tab', async () => {
  189. const user = userEvent.setup();
  190. render(<SettingsPage />);
  191. await waitFor(() => {
  192. expect(screen.getByText('API Keys')).toBeInTheDocument();
  193. });
  194. await user.click(screen.getByText('API Keys'));
  195. await waitFor(() => {
  196. // Button text is "Create Key"
  197. expect(screen.getByText('Create Key')).toBeInTheDocument();
  198. });
  199. });
  200. });
  201. });