Some checks failed
ci / Validate workspace (push) Successful in 12m32s
landing-page-ci / Validate landing page (push) Successful in 9m41s
landing-page-deploy / Deploy landing page (push) Failing after 5m23s
github-metrics / Generate repository metrics SVG (push) Failing after 2m3s
refresh-contributors-wall / Refresh contributors wall cache bust (push) Failing after 11s
This repository contains the open-design daemon CLI source code, built and packaged at https://helix-mind.ai/cli/open-design/latest.tgz for use by the HelixMind /design slash command. Licenses: Apache-2.0 (root) + MIT (skills/*)
233 lines
8.0 KiB
TypeScript
233 lines
8.0 KiB
TypeScript
import { expect, test } from '@playwright/test';
|
|
import type { Locator, Page } from '@playwright/test';
|
|
|
|
const STORAGE_KEY = 'open-design:config';
|
|
const CHAT_PANEL_WIDTH_STORAGE_KEY = 'open-design.project.chatPanelWidth';
|
|
|
|
test.beforeEach(async ({ page }) => {
|
|
await page.addInitScript((key) => {
|
|
window.localStorage.setItem(
|
|
key,
|
|
JSON.stringify({
|
|
mode: 'daemon',
|
|
apiKey: '',
|
|
baseUrl: 'https://api.anthropic.com',
|
|
model: 'claude-sonnet-4-5',
|
|
agentId: 'mock',
|
|
skillId: null,
|
|
designSystemId: null,
|
|
onboardingCompleted: true,
|
|
agentModels: {},
|
|
}),
|
|
);
|
|
}, STORAGE_KEY);
|
|
|
|
await page.route('**/api/agents', async (route) => {
|
|
await route.fulfill({
|
|
json: {
|
|
agents: [
|
|
{
|
|
id: 'mock',
|
|
name: 'Mock Agent',
|
|
bin: 'mock-agent',
|
|
available: true,
|
|
version: 'test',
|
|
models: [{ id: 'default', label: 'Default' }],
|
|
},
|
|
],
|
|
},
|
|
});
|
|
});
|
|
});
|
|
|
|
test('quick switcher opens from keyboard and activates the selected file', async ({ page }) => {
|
|
await page.goto('/');
|
|
await createProject(page, 'Quick switcher keyboard flow');
|
|
await expectWorkspaceReady(page);
|
|
|
|
await uploadTinyPng(page, 'alpha-file.png');
|
|
await uploadTinyPng(page, 'beta-file.png');
|
|
|
|
const alphaTab = tabBySuffix(page, 'alpha-file.png');
|
|
const betaTab = tabBySuffix(page, 'beta-file.png');
|
|
await expect(alphaTab).toBeVisible();
|
|
await expect(betaTab).toBeVisible();
|
|
await alphaTab.click();
|
|
await expect(alphaTab).toHaveAttribute('aria-selected', 'true');
|
|
|
|
await openQuickSwitcher(page);
|
|
const quickSwitcher = page.locator('.qs-overlay');
|
|
const quickSwitcherInput = page.locator('.qs-input');
|
|
await expect(quickSwitcher).toBeVisible();
|
|
await expect(quickSwitcherInput).toBeVisible();
|
|
|
|
await quickSwitcherInput.fill('beta');
|
|
await expect(page.getByRole('option', { name: /beta-file\.png/i })).toBeVisible();
|
|
await quickSwitcherInput.press('Enter');
|
|
|
|
await expect(quickSwitcher).toBeHidden();
|
|
await expect(betaTab).toHaveAttribute('aria-selected', 'true');
|
|
await expect(alphaTab).toHaveAttribute('aria-selected', 'false');
|
|
|
|
await openQuickSwitcher(page);
|
|
await expect(quickSwitcher).toBeVisible();
|
|
await quickSwitcherInput.press('Escape');
|
|
await expect(quickSwitcher).toBeHidden();
|
|
});
|
|
|
|
test('quick switcher keeps the current file when search has no matches', async ({ page }) => {
|
|
await page.goto('/');
|
|
await createProject(page, 'Quick switcher empty search flow');
|
|
await expectWorkspaceReady(page);
|
|
|
|
await uploadTinyPng(page, 'alpha-empty-search.png');
|
|
await uploadTinyPng(page, 'beta-empty-search.png');
|
|
|
|
const alphaTab = tabBySuffix(page, 'alpha-empty-search.png');
|
|
await expect(alphaTab).toBeVisible();
|
|
await alphaTab.click();
|
|
await expect(alphaTab).toHaveAttribute('aria-selected', 'true');
|
|
|
|
await openQuickSwitcher(page);
|
|
const quickSwitcher = page.locator('.qs-overlay');
|
|
const quickSwitcherInput = page.locator('.qs-input');
|
|
await expect(quickSwitcher).toBeVisible();
|
|
|
|
await quickSwitcherInput.fill('no-file-with-this-name');
|
|
await expect(page.locator('.qs-empty')).toBeVisible();
|
|
await expect(page.getByRole('option')).toHaveCount(0);
|
|
|
|
await quickSwitcherInput.press('Enter');
|
|
await expect(quickSwitcher).toBeVisible();
|
|
await quickSwitcherInput.press('Escape');
|
|
await expect(quickSwitcher).toBeHidden();
|
|
await expect(alphaTab).toHaveAttribute('aria-selected', 'true');
|
|
});
|
|
|
|
test('quick switcher arrow keys move selection before opening a file', async ({ page }) => {
|
|
await page.goto('/');
|
|
await createProject(page, 'Quick switcher arrow navigation flow');
|
|
await expectWorkspaceReady(page);
|
|
|
|
await uploadTinyPng(page, 'arrow-alpha.png');
|
|
await uploadTinyPng(page, 'arrow-beta.png');
|
|
await uploadTinyPng(page, 'arrow-gamma.png');
|
|
|
|
await openQuickSwitcher(page);
|
|
const quickSwitcher = page.locator('.qs-overlay');
|
|
const quickSwitcherInput = page.locator('.qs-input');
|
|
const selectedOption = page.getByRole('option', { selected: true });
|
|
await expect(quickSwitcher).toBeVisible();
|
|
await expect(page.getByRole('option')).toHaveCount(3);
|
|
|
|
const initialSelection = await selectedOption.textContent();
|
|
await quickSwitcherInput.press('ArrowDown');
|
|
const nextSelection = await selectedOption.textContent();
|
|
expect(nextSelection).not.toBe(initialSelection);
|
|
|
|
await quickSwitcherInput.press('Enter');
|
|
await expect(quickSwitcher).toBeHidden();
|
|
|
|
const selectedFileName = selectedBaseName(nextSelection);
|
|
await expect(tabBySuffix(page, selectedFileName)).toHaveAttribute('aria-selected', 'true');
|
|
});
|
|
|
|
test('keyboard chat panel resize persists after reload', async ({ page }) => {
|
|
await page.goto('/');
|
|
await createProject(page, 'Chat panel resize persistence');
|
|
await expectWorkspaceReady(page);
|
|
|
|
await page.evaluate((key) => {
|
|
window.localStorage.removeItem(key);
|
|
}, CHAT_PANEL_WIDTH_STORAGE_KEY);
|
|
await page.reload();
|
|
await expectWorkspaceReady(page);
|
|
|
|
const handle = page.locator('.split-resize-handle');
|
|
await expect(handle).toBeVisible();
|
|
|
|
const initialWidth = await readChatPanelWidth(handle);
|
|
await handle.focus();
|
|
await page.keyboard.press('End');
|
|
let resizedWidth = await readChatPanelWidth(handle);
|
|
if (resizedWidth === initialWidth) {
|
|
await page.keyboard.press('Home');
|
|
resizedWidth = await readChatPanelWidth(handle);
|
|
}
|
|
expect(resizedWidth).not.toBe(initialWidth);
|
|
|
|
const savedWidth = await page.evaluate(
|
|
(key) => window.localStorage.getItem(key),
|
|
CHAT_PANEL_WIDTH_STORAGE_KEY,
|
|
);
|
|
expect(savedWidth).toBe(String(resizedWidth));
|
|
|
|
await page.reload();
|
|
await expectWorkspaceReady(page);
|
|
const restoredWidth = await readChatPanelWidth(handle);
|
|
expect(restoredWidth).toBe(resizedWidth);
|
|
});
|
|
|
|
async function createProject(
|
|
page: Page,
|
|
projectName: string,
|
|
) {
|
|
await expect(page.getByTestId('new-project-panel')).toBeVisible();
|
|
await page.getByTestId('new-project-tab-prototype').click();
|
|
await page.getByTestId('new-project-name').fill(projectName);
|
|
await page.getByTestId('create-project').click();
|
|
}
|
|
|
|
async function expectWorkspaceReady(page: Page) {
|
|
await expect(page).toHaveURL(/\/projects\//);
|
|
await expect(page.getByTestId('chat-composer')).toBeVisible();
|
|
await expect(page.getByTestId('file-workspace')).toBeVisible();
|
|
await expect(page.getByText('Start a conversation')).toBeVisible();
|
|
}
|
|
|
|
async function uploadTinyPng(
|
|
page: Page,
|
|
name: string,
|
|
) {
|
|
const pngBytes = Buffer.from(
|
|
'iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVR42mP8/x8AAwMCAO5W6McAAAAASUVORK5CYII=',
|
|
'base64',
|
|
);
|
|
await page.getByTestId('design-files-upload-input').setInputFiles({
|
|
name,
|
|
mimeType: 'image/png',
|
|
buffer: pngBytes,
|
|
});
|
|
await expect(tabBySuffix(page, name)).toBeVisible();
|
|
}
|
|
|
|
async function readChatPanelWidth(handle: Locator): Promise<number> {
|
|
const raw = await handle.getAttribute('aria-valuenow');
|
|
const parsed = Number.parseInt(raw ?? '', 10);
|
|
expect(Number.isFinite(parsed)).toBeTruthy();
|
|
return parsed;
|
|
}
|
|
|
|
async function openQuickSwitcher(page: Page) {
|
|
const quickSwitcher = page.locator('.qs-overlay');
|
|
await page.keyboard.press('Meta+P');
|
|
if (await quickSwitcher.isVisible()) return;
|
|
await page.keyboard.press('Control+P');
|
|
await expect(quickSwitcher).toBeVisible();
|
|
}
|
|
|
|
function tabBySuffix(page: Page, name: string): Locator {
|
|
return page.getByRole('tab', { name: new RegExp(`${escapeRegExp(name)}$`, 'i') });
|
|
}
|
|
|
|
function selectedBaseName(selectionText: string | null): string {
|
|
const normalized = selectionText?.replace(/\s+/g, ' ').trim() ?? '';
|
|
const match = normalized.match(/arrow-(alpha|beta|gamma)\.png/i);
|
|
expect(match?.[0]).toBeTruthy();
|
|
return match![0];
|
|
}
|
|
|
|
function escapeRegExp(value: string): string {
|
|
return value.replace(/[.*+?^${}()|[\]\\]/g, '\\$&');
|
|
}
|