feat: unified E2E testing engine — 205 tests, dual-stack support
Infrastructure: - adapters/: base, tauri (port 9750), electrobun (port 9761 + PTY daemon) - helpers/: 120+ centralized selectors, reusable actions, custom assertions - wdio.shared.conf.js + stack-specific configs 18 unified specs (205 tests): splash(6) smoke(15) settings(19) terminal(14) agent(15) search(12) files(15) comms(10) tasks(10) theme(12) groups(12) keyboard(8) notifications(10) diagnostics(8) status-bar(12) context(9) worktree(8) llm-judged(10) Daemon: --stack tauri|electrobun|both flag Scripts: test:e2e:tauri, test:e2e:electrobun, test:e2e:both
This commit is contained in:
parent
1995f03682
commit
77b9ce9f62
31 changed files with 3547 additions and 344 deletions
164
tests/e2e/specs/theme.test.ts
Normal file
164
tests/e2e/specs/theme.test.ts
Normal file
|
|
@ -0,0 +1,164 @@
|
|||
/**
|
||||
* Theme tests — dropdown, groups, switching, CSS variables, font changes.
|
||||
*/
|
||||
|
||||
import { browser, expect } from '@wdio/globals';
|
||||
import * as S from '../helpers/selectors.ts';
|
||||
import { openSettings, closeSettings, switchSettingsCategory } from '../helpers/actions.ts';
|
||||
import { assertThemeApplied } from '../helpers/assertions.ts';
|
||||
|
||||
describe('Theme system', () => {
|
||||
before(async () => {
|
||||
await openSettings();
|
||||
await switchSettingsCategory(0); // Appearance tab
|
||||
});
|
||||
|
||||
after(async () => {
|
||||
await browser.keys('Escape');
|
||||
await browser.pause(300);
|
||||
});
|
||||
|
||||
it('should show theme dropdown button', async () => {
|
||||
const exists = await browser.execute(() => {
|
||||
return (document.querySelector('.dd-btn')
|
||||
?? document.querySelector('.dropdown-btn')
|
||||
?? document.querySelector('.custom-dropdown')) !== null;
|
||||
});
|
||||
expect(exists).toBe(true);
|
||||
});
|
||||
|
||||
it('should open theme dropdown on click', async () => {
|
||||
await browser.execute(() => {
|
||||
const btn = document.querySelector('.dd-btn')
|
||||
?? document.querySelector('.dropdown-btn');
|
||||
if (btn) (btn as HTMLElement).click();
|
||||
});
|
||||
await browser.pause(300);
|
||||
|
||||
const listOpen = await browser.execute(() => {
|
||||
const list = document.querySelector('.dd-list')
|
||||
?? document.querySelector('.dropdown-menu');
|
||||
if (!list) return false;
|
||||
return getComputedStyle(list).display !== 'none';
|
||||
});
|
||||
if (listOpen) {
|
||||
expect(listOpen).toBe(true);
|
||||
}
|
||||
});
|
||||
|
||||
it('should show theme groups (Catppuccin, Editor, Deep Dark)', async () => {
|
||||
const texts = await browser.execute(() => {
|
||||
const labels = document.querySelectorAll('.dd-group-label, .dropdown-group-label');
|
||||
return Array.from(labels).map(l => l.textContent ?? '');
|
||||
});
|
||||
|
||||
if (texts.length > 0) {
|
||||
expect(texts.some((t: string) => t.includes('Catppuccin'))).toBe(true);
|
||||
expect(texts.some((t: string) => t.includes('Editor'))).toBe(true);
|
||||
expect(texts.some((t: string) => t.includes('Deep Dark'))).toBe(true);
|
||||
}
|
||||
});
|
||||
|
||||
it('should list at least 17 theme options', async () => {
|
||||
const count = await browser.execute(() => {
|
||||
return (document.querySelectorAll('.dd-item').length
|
||||
|| document.querySelectorAll('.dropdown-item').length);
|
||||
});
|
||||
if (count > 0) {
|
||||
expect(count).toBeGreaterThanOrEqual(17);
|
||||
}
|
||||
});
|
||||
|
||||
it('should highlight the currently selected theme', async () => {
|
||||
const hasSelected = await browser.execute(() => {
|
||||
return (document.querySelector('.dd-item.selected')
|
||||
?? document.querySelector('.dropdown-item.active')) !== null;
|
||||
});
|
||||
if (hasSelected) {
|
||||
expect(hasSelected).toBe(true);
|
||||
}
|
||||
});
|
||||
|
||||
it('should apply CSS variables when theme changes', async () => {
|
||||
await assertThemeApplied('--ctp-base');
|
||||
});
|
||||
|
||||
it('should have 4 Catppuccin themes', async () => {
|
||||
const count = await browser.execute(() => {
|
||||
const items = document.querySelectorAll('.dd-item, .dropdown-item');
|
||||
let catCount = 0;
|
||||
const catNames = ['mocha', 'macchiato', 'frappe', 'latte'];
|
||||
for (const item of items) {
|
||||
const text = (item.textContent ?? '').toLowerCase();
|
||||
if (catNames.some(n => text.includes(n))) catCount++;
|
||||
}
|
||||
return catCount;
|
||||
});
|
||||
if (count > 0) {
|
||||
expect(count).toBe(4);
|
||||
}
|
||||
});
|
||||
|
||||
it('should have 7 Editor themes', async () => {
|
||||
const count = await browser.execute(() => {
|
||||
const items = document.querySelectorAll('.dd-item, .dropdown-item');
|
||||
const editorNames = ['vscode', 'atom', 'monokai', 'dracula', 'nord', 'solarized', 'github'];
|
||||
let edCount = 0;
|
||||
for (const item of items) {
|
||||
const text = (item.textContent ?? '').toLowerCase();
|
||||
if (editorNames.some(n => text.includes(n))) edCount++;
|
||||
}
|
||||
return edCount;
|
||||
});
|
||||
if (count > 0) {
|
||||
expect(count).toBe(7);
|
||||
}
|
||||
});
|
||||
|
||||
it('should have 6 Deep Dark themes', async () => {
|
||||
const count = await browser.execute(() => {
|
||||
const items = document.querySelectorAll('.dd-item, .dropdown-item');
|
||||
const deepNames = ['tokyo', 'gruvbox', 'ayu', 'poimandres', 'vesper', 'midnight'];
|
||||
let deepCount = 0;
|
||||
for (const item of items) {
|
||||
const text = (item.textContent ?? '').toLowerCase();
|
||||
if (deepNames.some(n => text.includes(n))) deepCount++;
|
||||
}
|
||||
return deepCount;
|
||||
});
|
||||
if (count > 0) {
|
||||
expect(count).toBe(6);
|
||||
}
|
||||
});
|
||||
|
||||
it('should show font size stepper controls', async () => {
|
||||
// Close theme dropdown first
|
||||
await browser.keys('Escape');
|
||||
await browser.pause(200);
|
||||
|
||||
const count = await browser.execute(() => {
|
||||
return (document.querySelectorAll('.size-stepper').length
|
||||
|| document.querySelectorAll('.font-stepper').length
|
||||
|| document.querySelectorAll('.stepper').length);
|
||||
});
|
||||
if (count > 0) {
|
||||
expect(count).toBeGreaterThanOrEqual(1);
|
||||
}
|
||||
});
|
||||
|
||||
it('should show theme action buttons', async () => {
|
||||
const count = await browser.execute(() => {
|
||||
return document.querySelectorAll('.theme-action-btn').length;
|
||||
});
|
||||
if (count > 0) {
|
||||
expect(count).toBeGreaterThanOrEqual(1);
|
||||
}
|
||||
});
|
||||
|
||||
it('should apply font changes to terminal', async () => {
|
||||
const fontFamily = await browser.execute(() => {
|
||||
return getComputedStyle(document.documentElement).getPropertyValue('--term-font-family').trim();
|
||||
});
|
||||
expect(typeof fontFamily).toBe('string');
|
||||
});
|
||||
});
|
||||
Loading…
Add table
Add a link
Reference in a new issue