|
1 | | -import { test, expect, STORAGE_KEY, getThemeAttribute, goToThemedPage, selectors } from '../fixtures/setup'; |
| 1 | +import { test, expect, STORAGE_KEY, getThemeAttribute, goToThemedPage, selectors, getCSSVariable, themeColors } from '../fixtures/setup'; |
2 | 2 |
|
3 | 3 | test.describe('Theme Persistence', () => { |
4 | 4 | test('defaults to light mode when no saved preference', async ({ page }) => { |
@@ -57,3 +57,135 @@ test.describe('Theme Persistence', () => { |
57 | 57 | expect(allKeys).toContain('codex-docs-theme'); |
58 | 58 | }); |
59 | 59 | }); |
| 60 | + |
| 61 | +test.describe('Theme Persistence — Edge Cases (FR-2.2)', () => { |
| 62 | + test('theme persists across page navigation', async ({ page }) => { |
| 63 | + await goToThemedPage(page); |
| 64 | + await page.click(selectors.themeToggleButton); |
| 65 | + expect(await getThemeAttribute(page)).toBe('dark'); |
| 66 | + |
| 67 | + // Navigate to a different page |
| 68 | + await page.goto('/'); |
| 69 | + await page.goto('/auth'); |
| 70 | + await page.waitForSelector(selectors.themeToggleButton, { timeout: 10000 }); |
| 71 | + |
| 72 | + expect(await getThemeAttribute(page)).toBe('dark'); |
| 73 | + }); |
| 74 | + |
| 75 | + test('clearing localStorage resets to default light mode', async ({ page }) => { |
| 76 | + await goToThemedPage(page); |
| 77 | + await page.click(selectors.themeToggleButton); |
| 78 | + expect(await getThemeAttribute(page)).toBe('dark'); |
| 79 | + |
| 80 | + // Clear storage and reload |
| 81 | + await page.evaluate(() => localStorage.clear()); |
| 82 | + await page.reload(); |
| 83 | + await page.waitForSelector(selectors.themeToggleButton, { timeout: 10000 }); |
| 84 | + |
| 85 | + expect(await getThemeAttribute(page)).toBe('light'); |
| 86 | + }); |
| 87 | + |
| 88 | + test('removing only the theme key resets to default', async ({ page }) => { |
| 89 | + await goToThemedPage(page); |
| 90 | + await page.click(selectors.themeToggleButton); |
| 91 | + |
| 92 | + await page.evaluate((key) => localStorage.removeItem(key), STORAGE_KEY); |
| 93 | + await page.reload(); |
| 94 | + await page.waitForSelector(selectors.themeToggleButton, { timeout: 10000 }); |
| 95 | + |
| 96 | + expect(await getThemeAttribute(page)).toBe('light'); |
| 97 | + }); |
| 98 | + |
| 99 | + test('invalid localStorage value does not crash theme initialization', async ({ page }) => { |
| 100 | + await page.goto('/auth'); |
| 101 | + await page.waitForSelector(selectors.themeToggleButton, { timeout: 10000 }); |
| 102 | + |
| 103 | + // Set an invalid theme value |
| 104 | + await page.evaluate((key) => localStorage.setItem(key, 'invalid-theme'), STORAGE_KEY); |
| 105 | + await page.reload(); |
| 106 | + await page.waitForSelector(selectors.themeToggleButton, { timeout: 10000 }); |
| 107 | + |
| 108 | + // Page should still work — toggle should still function |
| 109 | + await page.click(selectors.themeToggleButton); |
| 110 | + const stored = await page.evaluate((key) => localStorage.getItem(key), STORAGE_KEY); |
| 111 | + expect(['light', 'dark']).toContain(stored); |
| 112 | + }); |
| 113 | + |
| 114 | + test('rapid toggles persist the final state', async ({ page }) => { |
| 115 | + await goToThemedPage(page); |
| 116 | + |
| 117 | + // Toggle 7 times — odd count means dark |
| 118 | + for (let i = 0; i < 7; i++) { |
| 119 | + await page.click(selectors.themeToggleButton); |
| 120 | + } |
| 121 | + |
| 122 | + const finalTheme = await getThemeAttribute(page); |
| 123 | + const stored = await page.evaluate((key) => localStorage.getItem(key), STORAGE_KEY); |
| 124 | + expect(stored).toBe(finalTheme); |
| 125 | + |
| 126 | + // Reload and verify persistence |
| 127 | + await page.reload(); |
| 128 | + await page.waitForSelector(selectors.themeToggleButton, { timeout: 10000 }); |
| 129 | + expect(await getThemeAttribute(page)).toBe(finalTheme); |
| 130 | + }); |
| 131 | + |
| 132 | + test('localStorage stores only valid string values (not objects)', async ({ page }) => { |
| 133 | + await goToThemedPage(page); |
| 134 | + await page.click(selectors.themeToggleButton); |
| 135 | + |
| 136 | + const stored = await page.evaluate((key) => { |
| 137 | + const val = localStorage.getItem(key); |
| 138 | + return { |
| 139 | + value: val, |
| 140 | + type: typeof val, |
| 141 | + isJSON: (() => { try { JSON.parse(val!); return typeof JSON.parse(val!) === 'object'; } catch { return false; } })(), |
| 142 | + }; |
| 143 | + }, STORAGE_KEY); |
| 144 | + |
| 145 | + expect(stored.type).toBe('string'); |
| 146 | + expect(stored.isJSON).toBe(false); |
| 147 | + expect(['light', 'dark']).toContain(stored.value); |
| 148 | + }); |
| 149 | + |
| 150 | + test('theme preference does not leak into other storage keys', async ({ page }) => { |
| 151 | + await goToThemedPage(page); |
| 152 | + await page.click(selectors.themeToggleButton); |
| 153 | + |
| 154 | + const themeKeys = await page.evaluate(() => { |
| 155 | + return Object.keys(localStorage).filter(k => k.includes('theme') || k.includes('dark') || k.includes('mode')); |
| 156 | + }); |
| 157 | + |
| 158 | + // Only our key should be theme-related |
| 159 | + expect(themeKeys).toEqual(['codex-docs-theme']); |
| 160 | + }); |
| 161 | + |
| 162 | + test('CSS variables match persisted theme after reload', async ({ page }) => { |
| 163 | + await goToThemedPage(page); |
| 164 | + await page.click(selectors.themeToggleButton); |
| 165 | + expect(await getThemeAttribute(page)).toBe('dark'); |
| 166 | + |
| 167 | + await page.reload(); |
| 168 | + await page.waitForSelector(selectors.themeToggleButton, { timeout: 10000 }); |
| 169 | + |
| 170 | + // Verify CSS variables are dark mode values, not just the attribute |
| 171 | + const bgMain = await getCSSVariable(page, '--color-bg-main'); |
| 172 | + const textMain = await getCSSVariable(page, '--color-text-main'); |
| 173 | + expect(bgMain).toBe(themeColors.dark['--color-bg-main']); |
| 174 | + expect(textMain).toBe(themeColors.dark['--color-text-main']); |
| 175 | + }); |
| 176 | + |
| 177 | + test('toggle icon state matches persisted theme after reload', async ({ page }) => { |
| 178 | + await goToThemedPage(page); |
| 179 | + await page.click(selectors.themeToggleButton); |
| 180 | + expect(await getThemeAttribute(page)).toBe('dark'); |
| 181 | + |
| 182 | + await page.reload(); |
| 183 | + await page.waitForSelector(selectors.themeToggleButton, { timeout: 10000 }); |
| 184 | + |
| 185 | + // In dark mode: sun icon hidden, moon icon visible |
| 186 | + const sunIcon = page.locator(selectors.sunIcon); |
| 187 | + const moonIcon = page.locator(selectors.moonIcon); |
| 188 | + await expect(sunIcon).toHaveCSS('display', 'none'); |
| 189 | + await expect(moonIcon).toHaveCSS('display', 'block'); |
| 190 | + }); |
| 191 | +}); |
0 commit comments