Some checks failed
CI / Unit & Component Tests (pull_request) Failing after 1m36s
CI / Backend Unit Tests (pull_request) Failing after 2m53s
CI / E2E Tests (pull_request) Failing after 1h51m31s
CI / E2E Tests (push) Has been cancelled
CI / Unit & Component Tests (push) Has been cancelled
CI / Backend Unit Tests (push) Has been cancelled
- Change header --c-header dark value from #01335e to #012851 (brand navy): #01335e gave 4.3:1 with ink-3 (WCAG AA fail); #012851 gives 4.99:1 (pass) - Switch header element from bg-surface to bg-header so dark mode uses the independent --c-header token instead of inheriting the surface background - Fix both dark blocks (media query and manual override) to stay in sync Co-Authored-By: Claude Sonnet 4.6 <noreply@anthropic.com>
116 lines
3.7 KiB
TypeScript
116 lines
3.7 KiB
TypeScript
import { test, expect } from '@playwright/test';
|
|
|
|
test.describe('Theme toggle', () => {
|
|
test.beforeEach(async ({ page }) => {
|
|
// Clear any saved theme preference before each test
|
|
await page.goto('/');
|
|
await page.evaluate(() => localStorage.removeItem('theme'));
|
|
});
|
|
|
|
test('toggle button is visible in the header', async ({ page }) => {
|
|
await page.goto('/');
|
|
await expect(
|
|
page.getByRole('banner').getByRole('button', { name: /dark mode|light mode/i })
|
|
).toBeVisible();
|
|
});
|
|
|
|
test('clicking the toggle switches to dark mode', async ({ page }) => {
|
|
await page.goto('/');
|
|
await page.waitForSelector('[data-hydrated]');
|
|
|
|
const html = page.locator('html');
|
|
await expect(html).not.toHaveAttribute('data-theme', 'dark');
|
|
|
|
await page
|
|
.getByRole('banner')
|
|
.getByRole('button', { name: /dark mode/i })
|
|
.click();
|
|
|
|
await expect(html).toHaveAttribute('data-theme', 'dark');
|
|
});
|
|
|
|
test('clicking the toggle again switches back to light mode', async ({ page }) => {
|
|
await page.goto('/');
|
|
await page.waitForSelector('[data-hydrated]');
|
|
|
|
await page
|
|
.getByRole('banner')
|
|
.getByRole('button', { name: /dark mode/i })
|
|
.click();
|
|
await expect(page.locator('html')).toHaveAttribute('data-theme', 'dark');
|
|
|
|
await page
|
|
.getByRole('banner')
|
|
.getByRole('button', { name: /light mode/i })
|
|
.click();
|
|
await expect(page.locator('html')).toHaveAttribute('data-theme', 'light');
|
|
});
|
|
|
|
test('theme persists after page reload', async ({ page }) => {
|
|
await page.goto('/');
|
|
await page.waitForSelector('[data-hydrated]');
|
|
|
|
await page
|
|
.getByRole('banner')
|
|
.getByRole('button', { name: /dark mode/i })
|
|
.click();
|
|
await expect(page.locator('html')).toHaveAttribute('data-theme', 'dark');
|
|
|
|
await page.reload();
|
|
await expect(page.locator('html')).toHaveAttribute('data-theme', 'dark');
|
|
});
|
|
|
|
test('header uses --c-header token background in dark mode', async ({ page }) => {
|
|
await page.goto('/');
|
|
await page.waitForSelector('[data-hydrated]');
|
|
|
|
await page.evaluate(() => document.documentElement.setAttribute('data-theme', 'dark'));
|
|
|
|
const headerBg = await page.evaluate(() => {
|
|
const header = document.querySelector('header');
|
|
return header ? getComputedStyle(header).backgroundColor : null;
|
|
});
|
|
// --c-header in dark mode = #012851 (brand navy) → rgb(1, 40, 81)
|
|
expect(headerBg).toBe('rgb(1, 40, 81)');
|
|
});
|
|
|
|
test('color-scheme is dark when data-theme=dark is set', async ({ page }) => {
|
|
await page.goto('/');
|
|
await page.waitForSelector('[data-hydrated]');
|
|
|
|
await page.evaluate(() => document.documentElement.setAttribute('data-theme', 'dark'));
|
|
|
|
const colorScheme = await page.evaluate(
|
|
() => getComputedStyle(document.documentElement).colorScheme
|
|
);
|
|
expect(colorScheme).toBe('dark');
|
|
});
|
|
|
|
test('color-scheme is dark in prefers-color-scheme: dark media', async ({ browser }) => {
|
|
const context = await browser.newContext({
|
|
colorScheme: 'dark',
|
|
storageState: 'e2e/.auth/user.json'
|
|
});
|
|
const page = await context.newPage();
|
|
await page.goto('/');
|
|
await page.waitForSelector('[data-hydrated]');
|
|
|
|
const colorScheme = await page.evaluate(
|
|
() => getComputedStyle(document.documentElement).colorScheme
|
|
);
|
|
await context.close();
|
|
expect(colorScheme).toBe('dark');
|
|
});
|
|
|
|
test('saved theme is applied before first paint (no flash)', async ({ page }) => {
|
|
// Set dark theme in localStorage before navigating
|
|
await page.goto('/');
|
|
await page.evaluate(() => localStorage.setItem('theme', 'dark'));
|
|
|
|
// Intercept the initial HTML to verify data-theme is set immediately
|
|
await page.goto('/');
|
|
const theme = await page.evaluate(() => document.documentElement.getAttribute('data-theme'));
|
|
expect(theme).toBe('dark');
|
|
});
|
|
});
|