import { test, expect } from '@playwright/test'; const BASE = 'http://localhost:3456'; // Auth: first-run signup, blocked signup, login test.describe.serial('auth flow', () => { test.use({ storageState: { cookies: [], origins: [] } }); test('redirects to signup when no users exist', async ({ page }) => { await page.goto(BASE + '/'); await expect(page).toHaveURL(/\/signup$/); await expect(page.getByRole('heading', { name: 'Create account' })).toBeVisible(); }); test('first user can signup', async ({ page }) => { await page.goto(BASE + '/signup'); await page.getByLabel('Full Name').fill('E2E User'); await page.getByLabel('Email').fill('e2e@test.com'); await page.getByLabel('Password').fill('e2e-password-123'); await page.getByRole('button', { name: 'Create Account' }).click(); await expect(page).toHaveURL(BASE + '/'); await expect(page.getByText(/Welcome back/)).toBeVisible(); await page.context().storageState({ path: 'playwright/.auth/state.json' }); }); test('blocks second signup when users exist', async ({ page }) => { await page.goto(BASE + '/signup'); await expect(page).toHaveURL(/\/login$/); }); test('existing user can login', async ({ page }) => { await page.goto(BASE + '/login'); await page.getByLabel('Email').fill('e2e@test.com'); await page.getByLabel('Password').fill('e2e-password-123'); await page.getByRole('button', { name: 'Sign In' }).click(); await expect(page).toHaveURL(BASE + '/'); await expect(page.getByText(/Welcome back/)).toBeVisible(); }); }); // Dashboard: quick actions and stats test.describe.serial('dashboard', () => { test.use({ storageState: 'playwright/.auth/state.json' }); test('shows stats cards', async ({ page }) => { await page.goto(BASE + '/'); await expect(page.getByText('Drawings')).toBeVisible(); await expect(page.getByText('Projects')).toBeVisible(); await expect(page.getByText('Teams')).toBeVisible(); }); test('quick action: New Project navigates to files', async ({ page }) => { await page.goto(BASE + '/'); await page.getByRole('button', { name: 'New Project' }).click(); await expect(page).toHaveURL(/\/files/); await expect(page.getByRole('navigation', { name: 'Project tree' })).toBeVisible(); await expect(page.getByText('All Projects')).toBeVisible(); }); test('quick action: Invite navigates to team', async ({ page }) => { await page.goto(BASE + '/'); await page.getByRole('button', { name: 'Invite' }).click(); await expect(page).toHaveURL(/\/team/); await expect(page.getByRole('heading', { name: 'Team Settings' })).toBeVisible(); }); test('quick action: Library navigates to marketplace', async ({ page }) => { await page.goto(BASE + '/'); await page.getByRole('button', { name: 'Library' }).click(); await expect(page).toHaveURL(/\/library/); await expect(page.getByRole('heading', { name: 'Library Marketplace' })).toBeVisible(); }); test('New Drawing opens template picker', async ({ page }) => { await page.goto(BASE + '/'); await page.getByRole('button', { name: 'New Drawing' }).click(); await expect(page.getByRole('dialog')).toBeVisible(); await expect(page.getByRole('heading', { name: 'Choose a Template' })).toBeVisible(); await expect(page.getByRole('button', { name: 'Blank Canvas' })).toBeVisible(); await expect(page.getByRole('button', { name: 'To-Do List' })).toBeVisible(); await expect(page.getByRole('button', { name: 'Checklist' })).toBeVisible(); await expect(page.getByRole('button', { name: 'Bullet List' })).toBeVisible(); await expect(page.getByRole('button', { name: 'Flow Chart' })).toBeVisible(); }); }); // Projects / FileBrowser test.describe.serial('projects', () => { test.use({ storageState: 'playwright/.auth/state.json' }); test('shows Projects label in sidebar and breadcrumb', async ({ page }) => { await page.goto(BASE + '/files'); await expect(page.getByRole('navigation', { name: 'Main navigation' }).getByText('Projects')).toBeVisible(); await expect(page.getByText('All Projects')).toBeVisible(); }); test('can create a drawing from file browser', async ({ page }) => { await page.goto(BASE + '/files'); await page.getByRole('button', { name: 'Create new drawing' }).click(); await expect(page.getByRole('dialog')).toBeVisible(); await page.getByRole('button', { name: 'Blank Canvas' }).click(); await expect(page).toHaveURL(/\/drawing\//); await expect(page.getByText('Loading Excalidraw')).toBeVisible(); }); }); // Editor / Canvas test.describe.serial('editor', () => { test.use({ storageState: 'playwright/.auth/state.json' }); test('creates drawing with To-Do template', async ({ page }) => { await page.goto(BASE + '/'); await page.getByRole('button', { name: 'New Drawing' }).click(); await page.getByRole('button', { name: 'To-Do List' }).click(); await expect(page).toHaveURL(/\/drawing\//); await expect(page.getByRole('button', { name: /Save Now/i })).toBeVisible({ timeout: 10000 }); }); test('editor shows save controls and back button', async ({ page }) => { await page.goto(BASE + '/'); await page.getByRole('button', { name: 'New Drawing' }).click(); await page.getByRole('button', { name: 'Blank Canvas' }).click(); await expect(page).toHaveURL(/\/drawing\//); await expect(page.getByRole('button', { name: /Save Now/i })).toBeVisible({ timeout: 10000 }); await expect(page.getByRole('button', { name: /Back/i })).toBeVisible(); }); }); // Library Marketplace test.describe.serial('library', () => { test.use({ storageState: 'playwright/.auth/state.json' }); test('loads marketplace with search and categories', async ({ page }) => { await page.goto(BASE + '/library'); await expect(page.getByRole('heading', { name: 'Library Marketplace' })).toBeVisible(); await expect(page.getByPlaceholder('Search libraries...')).toBeVisible(); await expect(page.getByRole('button', { name: 'All' }).first()).toBeVisible(); await expect(page.getByRole('button', { name: 'Open External' })).toBeVisible(); }); test('search filters libraries', async ({ page }) => { await page.goto(BASE + '/library'); await page.getByPlaceholder('Search libraries...').fill('zzzznonexistent'); await expect(page.getByText('No libraries found')).toBeVisible(); }); }); // Team / Invites test.describe.serial('team', () => { test.use({ storageState: 'playwright/.auth/state.json' }); test('shows owner in members list', async ({ page }) => { await page.goto(BASE + '/team'); await expect(page.getByRole('heading', { name: 'Team Settings' })).toBeVisible(); await expect(page.getByText('E2E User')).toBeVisible(); await expect(page.getByText('owner')).toBeVisible(); }); test('can send team invite', async ({ page }) => { await page.goto(BASE + '/team'); await page.getByLabel('Email address').fill('invited@test.com'); await page.locator('select').selectOption('editor'); await page.getByRole('button', { name: 'Send Invite' }).click(); await expect(page.getByText('Invite sent!')).toBeVisible(); await expect(page.getByText('Pending Invites')).toBeVisible(); await expect(page.getByText('invited@test.com')).toBeVisible(); await expect(page.getByText('editor').first()).toBeVisible(); }); });