/** * Edit Feature E2E Test * * Happy path: Edit an existing feature's description and verify changes persist */ import { test, expect } from '@playwright/test'; import * as fs from 'fs'; import * as path from 'path'; import { createTempDirPath, cleanupTempDir, setupRealProject, waitForNetworkIdle, clickAddFeature, fillAddFeatureDialog, confirmAddFeature, clickElement, } from '../utils'; const TEST_TEMP_DIR = createTempDirPath('edit-feature-test'); test.describe('Edit Feature', () => { let projectPath: string; const projectName = `test-project-${Date.now()}`; test.beforeAll(async () => { if (!fs.existsSync(TEST_TEMP_DIR)) { fs.mkdirSync(TEST_TEMP_DIR, { recursive: true }); } projectPath = path.join(TEST_TEMP_DIR, projectName); fs.mkdirSync(projectPath, { recursive: true }); fs.writeFileSync( path.join(projectPath, 'package.json'), JSON.stringify({ name: projectName, version: '1.0.0' }, null, 2) ); const automakerDir = path.join(projectPath, '.automaker'); fs.mkdirSync(automakerDir, { recursive: true }); fs.mkdirSync(path.join(automakerDir, 'features'), { recursive: true }); fs.mkdirSync(path.join(automakerDir, 'context'), { recursive: true }); fs.writeFileSync( path.join(automakerDir, 'categories.json'), JSON.stringify({ categories: [] }, null, 2) ); fs.writeFileSync( path.join(automakerDir, 'app_spec.txt'), `# ${projectName}\n\nA test project for e2e testing.` ); }); test.afterAll(async () => { cleanupTempDir(TEST_TEMP_DIR); }); test('should edit an existing feature description', async ({ page }) => { const originalDescription = `Original feature ${Date.now()}`; const updatedDescription = `Updated feature ${Date.now()}`; await setupRealProject(page, projectPath, projectName, { setAsCurrent: true }); await page.goto('/board'); await waitForNetworkIdle(page); await expect(page.locator('[data-testid="board-view"]')).toBeVisible({ timeout: 10000 }); await expect(page.locator('[data-testid="kanban-column-backlog"]')).toBeVisible({ timeout: 5000, }); // Create a feature first await clickAddFeature(page); await fillAddFeatureDialog(page, originalDescription); await confirmAddFeature(page); // Wait for the feature to appear in the backlog await expect(async () => { const backlogColumn = page.locator('[data-testid="kanban-column-backlog"]'); const featureCard = backlogColumn.locator('[data-testid^="kanban-card-"]').filter({ hasText: originalDescription, }); expect(await featureCard.count()).toBeGreaterThan(0); }).toPass({ timeout: 10000 }); // Get the feature ID from the card const featureCard = page .locator('[data-testid="kanban-column-backlog"]') .locator('[data-testid^="kanban-card-"]') .filter({ hasText: originalDescription }) .first(); const cardTestId = await featureCard.getAttribute('data-testid'); const featureId = cardTestId?.replace('kanban-card-', ''); // Collapse the sidebar first to avoid it intercepting clicks const collapseSidebarButton = page.locator('button:has-text("Collapse sidebar")'); if (await collapseSidebarButton.isVisible()) { await collapseSidebarButton.click(); await page.waitForTimeout(300); // Wait for sidebar animation } // Click the edit button on the card using JavaScript click to bypass pointer interception const editButton = page.locator(`[data-testid="edit-backlog-${featureId}"]`); await expect(editButton).toBeVisible({ timeout: 5000 }); await editButton.evaluate((el) => (el as HTMLElement).click()); // Wait for edit dialog to appear await expect(page.locator('[data-testid="edit-feature-dialog"]')).toBeVisible({ timeout: 10000, }); // Update the description - the input is inside the DescriptionImageDropZone const descriptionInput = page .locator('[data-testid="edit-feature-dialog"]') .getByPlaceholder('Describe the feature...'); await expect(descriptionInput).toBeVisible({ timeout: 5000 }); await descriptionInput.fill(updatedDescription); // Save changes await clickElement(page, 'confirm-edit-feature'); // Wait for dialog to close await page.waitForFunction( () => !document.querySelector('[data-testid="edit-feature-dialog"]'), { timeout: 5000 } ); // Verify the updated description appears in the card await expect(async () => { const backlogColumn = page.locator('[data-testid="kanban-column-backlog"]'); const updatedCard = backlogColumn.locator('[data-testid^="kanban-card-"]').filter({ hasText: updatedDescription, }); expect(await updatedCard.count()).toBeGreaterThan(0); }).toPass({ timeout: 10000 }); }); });