Keep/keep-notes/tests/drag-drop.spec.ts

303 lines
12 KiB
TypeScript

import { test, expect, request } from '@playwright/test';
test.describe('Note Grid - Drag and Drop', () => {
test.beforeAll(async ({ request }) => {
console.log('[CLEANUP] beforeAll: Cleaning up any existing test notes...');
// Clean up any existing test notes from previous runs
try {
const response = await request.get('http://localhost:3000/api/notes');
const data = await response.json();
if (data.success && data.data) {
const testNotes = data.data.filter((note: any) =>
note.title?.startsWith('test-') &&
note.content?.startsWith('Content ')
);
console.log(`[CLEANUP] beforeAll: Found ${testNotes.length} test notes to delete`);
for (const note of testNotes) {
try {
await request.delete(`http://localhost:3000/api/notes?id=${note.id}`);
console.log(`[CLEANUP] beforeAll: Deleted note ${note.id}`);
} catch (error) {
console.log(`[CLEANUP] beforeAll: Failed to delete note ${note.id}`, error);
}
}
}
} catch (error) {
console.log('[CLEANUP] beforeAll: Error fetching notes for cleanup', error);
}
});
test.beforeEach(async ({ page }) => {
// Generate unique timestamp for this test run to avoid conflicts
const timestamp = Date.now();
const testId = `test-${timestamp}`;
await page.goto('/');
// Wait for page to fully load
await page.waitForLoadState('networkidle');
// Log initial note count before creating new notes
const initialNotes = page.locator('[data-draggable="true"]');
const initialCount = await initialNotes.count();
console.log(`[DEBUG] [${testId}] Initial note count: ${initialCount}`);
// Create multiple notes for testing drag and drop with unique identifiers
for (let i = 1; i <= 4; i++) {
const noteTitle = `${testId}-Note ${i}`;
const noteContent = `${testId}-Content ${i}`;
console.log(`[DEBUG] [${testId}] Creating ${noteTitle}`);
await page.click('input[placeholder="Take a note..."]');
await page.fill('input[placeholder="Title"]', noteTitle);
await page.fill('textarea[placeholder="Take a note..."]', noteContent);
await page.click('button:has-text("Add")');
await page.waitForTimeout(500);
// Log note count after each creation
const currentCount = await page.locator('[data-draggable="true"]').count();
console.log(`[DEBUG] [${testId}] Note count after creating ${noteTitle}: ${currentCount}`);
}
// Log final note count
const finalCount = await page.locator('[data-draggable="true"]').count();
console.log(`[DEBUG] [${testId}] Final note count after all creations: ${finalCount}`);
});
test('should have draggable notes', async ({ page }) => {
console.log('[DEBUG] Test: should have draggable notes');
// Wait for notes to appear (use a more flexible selector that matches pattern)
await page.waitForSelector('[data-draggable="true"]');
// Check that notes have data-draggable attribute (dnd-kit uses this)
const noteCards = page.locator('[data-draggable="true"]');
const count = await noteCards.count();
console.log(`[DEBUG] Found ${count} notes with data-draggable="true"`);
// Log first few notes details
for (let i = 0; i < Math.min(count, 3); i++) {
const note = noteCards.nth(i);
const text = await note.textContent();
const draggableAttr = await note.getAttribute('data-draggable');
console.log(`[DEBUG] Note ${i}: "${text?.substring(0, 50)}", data-draggable="${draggableAttr}"`);
}
expect(count).toBeGreaterThanOrEqual(4);
});
test('should show cursor-move on note cards', async ({ page }) => {
await page.waitForSelector('[data-draggable="true"]');
// Check CSS class for cursor-move on the note card inside
const firstNote = page.locator('[data-draggable="true"]').first();
const noteCard = firstNote.locator('.note-card-main');
const className = await noteCard.getAttribute('class');
expect(className).toContain('cursor-move');
});
test('should change opacity when dragging', async ({ page }) => {
await page.waitForSelector('[data-draggable="true"]');
const firstNote = page.locator('[data-draggable="true"]').first();
// Start drag
const box = await firstNote.boundingBox();
if (box) {
await page.mouse.move(box.x + box.width / 2, box.y + box.height / 2);
await page.mouse.down();
// Move to trigger drag
await page.mouse.move(box.x + box.width / 2 + 50, box.y + box.height / 2 + 50);
await page.waitForTimeout(200);
// Check if opacity changed (style should have opacity: 0.5)
const style = await firstNote.getAttribute('style');
expect(style).toContain('opacity');
await page.mouse.up();
}
});
test('should reorder notes when dropped on another note', async ({ page }) => {
console.log('[DEBUG] Test: should reorder notes when dropped on another note');
// Wait for notes to be fully loaded and drag-and-drop initialized
await page.waitForSelector('[data-draggable="true"]', { state: 'attached' });
await page.waitForTimeout(500); // Extra wait for dnd-kit to initialize
// Get initial order
const notes = page.locator('[data-draggable="true"]');
const firstNoteText = await notes.first().textContent();
const secondNoteText = await notes.nth(1).textContent();
console.log(`[DEBUG] Initial first note: ${firstNoteText?.substring(0, 30)}`);
console.log(`[DEBUG] Initial second note: ${secondNoteText?.substring(0, 30)}`);
expect(firstNoteText).toMatch(/Note \d+/);
expect(secondNoteText).toMatch(/Note \d+/);
// Use dragTo for more reliable drag and drop
const firstNote = notes.first();
const secondNote = notes.nth(1);
console.log('[DEBUG] Starting drag operation...');
await firstNote.dragTo(secondNote);
console.log('[DEBUG] Drag operation completed');
// Wait for reorder to complete and database to update
await page.waitForTimeout(2000); // Increased wait time for async operations
// Check that order changed
// Note: This depends on order persisting in the database
console.log('[DEBUG] Reloading page to verify persistence...');
await page.reload();
await page.waitForSelector('[data-draggable="true"]', { state: 'attached' });
await page.waitForTimeout(500);
// Verify that order changed (implementation dependent)
console.log('[DEBUG] Page reloaded, checking order...');
});
test('should work with pinned and unpinned notes separately', async ({ page }) => {
await page.waitForSelector('[data-draggable="true"]');
// Pin first note by hovering and clicking pin button
const firstNoteCard = page.locator('[data-draggable="true"]').first();
await firstNoteCard.hover();
// Click the pin button
const pinButton = firstNoteCard.locator('button[title="Pin"]');
await pinButton.click();
await page.waitForTimeout(500);
// Check that "Pinned" section appears
await expect(page.locator('h2:has-text("Pinned")')).toBeVisible();
// Verify note is in pinned section
const pinnedSection = page.locator('h2:has-text("Pinned")').locator('..').locator('div[data-draggable="true"]');
expect(await pinnedSection.count()).toBeGreaterThanOrEqual(1);
});
test('should not mix pinned and unpinned notes when dragging', async ({ page }) => {
await page.waitForSelector('[data-draggable="true"]');
// Pin first note by hovering and clicking pin button
const firstNoteCard = page.locator('[data-draggable="true"]').first();
await firstNoteCard.hover();
// Click the pin button
const pinButton = firstNoteCard.locator('button[title="Pin"]');
await pinButton.click();
await page.waitForTimeout(500);
// Should have both Pinned and Others sections
await expect(page.locator('h2:has-text("Pinned")')).toBeVisible();
await expect(page.locator('h2:has-text("Others")')).toBeVisible();
// Count notes in each section using data-draggable attribute
const pinnedNotes = page.locator('h2:has-text("Pinned") ~ div [data-draggable="true"]');
const unpinnedNotes = page.locator('h2:has-text("Others") ~ div [data-draggable="true"]');
expect(await pinnedNotes.count()).toBeGreaterThanOrEqual(1);
expect(await unpinnedNotes.count()).toBeGreaterThanOrEqual(3);
});
test('should persist note order after page reload', async ({ page }) => {
await page.waitForSelector('[data-draggable="true"]');
// Get initial order
const notes = page.locator('[data-draggable="true"]');
const initialOrder: string[] = [];
const count = await notes.count();
for (let i = 0; i < Math.min(count, 4); i++) {
const text = await notes.nth(i).textContent();
if (text) initialOrder.push(text);
}
// Reload page
await page.reload();
await page.waitForSelector('[data-draggable="true"]');
// Get order after reload
const notesAfterReload = page.locator('[data-draggable="true"]');
const reloadedOrder: string[] = [];
const countAfterReload = await notesAfterReload.count();
for (let i = 0; i < Math.min(countAfterReload, 4); i++) {
const text = await notesAfterReload.nth(i).textContent();
if (text) reloadedOrder.push(text);
}
// Order should be the same
expect(reloadedOrder.length).toBe(initialOrder.length);
});
test.afterEach(async ({ page, request }) => {
console.log('[CLEANUP] Starting cleanup...');
// Clean up created notes via API - more reliable than UI interaction
try {
const response = await request.get('http://localhost:3000/api/notes');
const data = await response.json();
if (data.success && data.data) {
const testNotes = data.data.filter((note: any) =>
note.title?.startsWith('test-') &&
note.content?.startsWith('Content ')
);
console.log(`[CLEANUP] Found ${testNotes.length} test notes to delete via API`);
for (const note of testNotes) {
try {
await request.delete(`http://localhost:3000/api/notes?id=${note.id}`);
console.log(`[CLEANUP] Deleted note ${note.id}`);
} catch (error) {
console.log(`[CLEANUP] Failed to delete note ${note.id}`, error);
}
}
}
} catch (error) {
console.log('[CLEANUP] Error fetching notes for cleanup', error);
}
});
test.afterAll(async ({ request }) => {
console.log('[CLEANUP] afterAll: Final cleanup of any remaining test notes...');
// Final cleanup to ensure no test notes remain
try {
const response = await request.get('http://localhost:3000/api/notes');
const data = await response.json();
if (data.success && data.data) {
const testNotes = data.data.filter((note: any) =>
note.title?.startsWith('test-') &&
note.content?.startsWith('Content ')
);
console.log(`[CLEANUP] afterAll: Found ${testNotes.length} remaining test notes to delete`);
for (const note of testNotes) {
try {
await request.delete(`http://localhost:3000/api/notes?id=${note.id}`);
console.log(`[CLEANUP] afterAll: Deleted note ${note.id}`);
} catch (error) {
console.log(`[CLEANUP] afterAll: Failed to delete note ${note.id}`, error);
}
}
}
} catch (error) {
console.log('[CLEANUP] afterAll: Error fetching notes for final cleanup', error);
}
});
});