// FE-ADMIN-PKG-001 to FE-ADMIN-PKG-020 import { render, screen, waitFor } from '../../../tests/helpers/render'; import userEvent from '@testing-library/user-event'; import { http, HttpResponse } from 'msw'; import { server } from '../../../tests/helpers/msw/server'; import { resetAllStores } from '../../../tests/helpers/store'; import PackingTemplateManager from './PackingTemplateManager'; import { ToastContainer } from '../shared/Toast'; const tmpl1 = { id: 1, name: 'Beach Trip', item_count: 5, category_count: 2, created_by_name: 'admin' } const tmpl2 = { id: 2, name: 'City Break', item_count: 3, category_count: 1, created_by_name: 'admin' } const cat1 = { id: 10, template_id: 1, name: 'Clothing', sort_order: 0 } const item1 = { id: 100, category_id: 10, name: 'T-shirt', sort_order: 0 } const item2 = { id: 101, category_id: 10, name: 'Shorts', sort_order: 1 } beforeEach(() => { resetAllStores(); }); describe('PackingTemplateManager', () => { it('FE-ADMIN-PKG-001: shows loading spinner on mount', async () => { server.use( http.get('/api/admin/packing-templates', async () => { await new Promise(r => setTimeout(r, 100)); return HttpResponse.json({ templates: [] }); }) ); render(); expect(document.querySelector('.animate-spin')).toBeInTheDocument(); }); it('FE-ADMIN-PKG-002: shows empty state when no templates', async () => { render(); await screen.findByText('No templates created yet'); expect(screen.queryAllByRole('button', { name: /chevron/i })).toHaveLength(0); }); it('FE-ADMIN-PKG-003: template list renders names and counts', async () => { server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1, tmpl2] }) ) ); render(); await screen.findByText('Beach Trip'); expect(screen.getByText('City Break')).toBeInTheDocument(); // tmpl1 has 2 categories and 5 items expect(screen.getByText(/2 categories · 5 items/i)).toBeInTheDocument(); }); it('FE-ADMIN-PKG-004: clicking "+" shows create input', async () => { const user = userEvent.setup(); render(); await screen.findByText('No templates created yet'); const createBtn = screen.getByRole('button', { name: /new template/i }); await user.click(createBtn); expect(screen.getByPlaceholderText('Template name (e.g. Beach Holiday)')).toBeInTheDocument(); }); it('FE-ADMIN-PKG-005: creates template on Enter and shows success toast', async () => { const user = userEvent.setup(); let postCalled = false; server.use( http.post('/api/admin/packing-templates', async () => { postCalled = true; return HttpResponse.json({ template: { id: 99, name: 'New Template' } }); }) ); render(<>); await screen.findByText('No templates created yet'); await user.click(screen.getByRole('button', { name: /new template/i })); const input = screen.getByPlaceholderText('Template name (e.g. Beach Holiday)'); await user.type(input, 'New Template{Enter}'); await waitFor(() => expect(postCalled).toBe(true)); // "New Template" may appear both as the button label and the new list item await waitFor(() => expect(screen.getAllByText('New Template').length).toBeGreaterThanOrEqual(1)); await screen.findByText('Template created'); }); it('FE-ADMIN-PKG-006: Escape dismisses create input without API call', async () => { const user = userEvent.setup(); let postCalled = false; server.use( http.post('/api/admin/packing-templates', async () => { postCalled = true; return HttpResponse.json({ template: { id: 99, name: 'Should Not Appear' } }); }) ); render(); await screen.findByText('No templates created yet'); await user.click(screen.getByRole('button', { name: /new template/i })); const input = screen.getByPlaceholderText('Template name (e.g. Beach Holiday)'); await user.type(input, 'Test{Escape}'); await waitFor(() => { expect(screen.queryByPlaceholderText('Template name (e.g. Beach Holiday)')).not.toBeInTheDocument(); }); expect(postCalled).toBe(false); }); it('FE-ADMIN-PKG-007: expanding a template loads and displays its categories and items', async () => { const user = userEvent.setup(); server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [cat1], items: [item1, item2] }) ) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('Clothing'); expect(screen.getByText('T-shirt')).toBeInTheDocument(); expect(screen.getByText('Shorts')).toBeInTheDocument(); }); it('FE-ADMIN-PKG-008: collapsing an expanded template hides its content', async () => { const user = userEvent.setup(); server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [cat1], items: [item1, item2] }) ) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('Clothing'); // Collapse by clicking again await user.click(screen.getByText('Beach Trip')); await waitFor(() => { expect(screen.queryByText('Clothing')).not.toBeInTheDocument(); expect(screen.queryByText('T-shirt')).not.toBeInTheDocument(); }); }); it('FE-ADMIN-PKG-009: deleting a template removes it from the list and shows toast', async () => { const user = userEvent.setup(); let deleteCalled = false; server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1, tmpl2] }) ), http.delete('/api/admin/packing-templates/1', () => { deleteCalled = true; return HttpResponse.json({ success: true }); }) ); render(<>); await screen.findByText('Beach Trip'); expect(screen.getByText('City Break')).toBeInTheDocument(); // Find all Trash2 (delete) buttons — there are 2 (one per template) const deleteButtons = screen.getAllByRole('button').filter(b => b.className.includes('hover:bg-red-50') || b.querySelector('svg') ); // Click the delete button for "Beach Trip" (first template row's trash button) // The buttons layout in each row: [chevron, edit, delete] // We find rows first const beachTripRow = screen.getByText('Beach Trip').closest('div'); const trashBtn = beachTripRow!.parentElement!.querySelector('button.hover\\:bg-red-50') as HTMLElement | null; if (trashBtn) { await user.click(trashBtn); } else { // Fallback: find all red-hover buttons and click first const allBtns = screen.getAllByRole('button'); const redBtns = allBtns.filter(b => b.className.includes('hover:bg-red-50')); await user.click(redBtns[0]); } await waitFor(() => expect(deleteCalled).toBe(true)); await waitFor(() => expect(screen.queryByText('Beach Trip')).not.toBeInTheDocument()); expect(screen.getByText('City Break')).toBeInTheDocument(); await screen.findByText('Template deleted'); }); it('FE-ADMIN-PKG-010: renaming a template inline updates the list', async () => { const user = userEvent.setup(); let putCalled = false; server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.put('/api/admin/packing-templates/1', async () => { putCalled = true; return HttpResponse.json({ success: true }); }) ); render(); await screen.findByText('Beach Trip'); // Find the Edit2 button on the template row const beachTripText = screen.getByText('Beach Trip'); const row = beachTripText.closest('div')!.parentElement!; const editBtn = row.querySelector('button.hover\\:bg-slate-100') as HTMLElement | null; if (editBtn) { await user.click(editBtn); } else { // Fallback: find all slate-100-hover buttons const allBtns = screen.getAllByRole('button'); const editBtns = allBtns.filter(b => b.className.includes('hover:bg-slate-100')); await user.click(editBtns[0]); } const input = screen.getByDisplayValue('Beach Trip'); await user.clear(input); await user.type(input, 'Summer Packing{Enter}'); await waitFor(() => expect(putCalled).toBe(true)); await screen.findByText('Summer Packing'); }); it('FE-ADMIN-PKG-011: adding a category to an expanded template', async () => { const user = userEvent.setup(); server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [], items: [] }) ), http.post('/api/admin/packing-templates/1/categories', async () => HttpResponse.json({ category: { id: 20, template_id: 1, name: 'Electronics', sort_order: 1 } }) ) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); // Wait for expanded state (Add category button should appear) await screen.findByText('Add category'); await user.click(screen.getByText('Add category')); const catInput = screen.getByPlaceholderText('Category name (e.g. Clothing)'); await user.type(catInput, 'Electronics{Enter}'); await screen.findByText('Electronics'); }); it('FE-ADMIN-PKG-012: adding an item to a category', async () => { const user = userEvent.setup(); server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [cat1], items: [] }) ), http.post('/api/admin/packing-templates/1/categories/10/items', async () => HttpResponse.json({ item: { id: 102, category_id: 10, name: 'Sandals', sort_order: 2 } }) ) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('Clothing'); // Click the "+" button on the Clothing category row const clothingHeader = screen.getByText('Clothing').closest('div')!; const addItemBtn = clothingHeader.querySelector('button') as HTMLElement; await user.click(addItemBtn); const itemInput = screen.getByPlaceholderText('Item name'); await user.type(itemInput, 'Sandals'); // Submit via Enter key (the input's onKeyDown handler triggers handleAddItem) await user.type(itemInput, '{Enter}'); await screen.findByText('Sandals'); }); it('FE-ADMIN-PKG-013: renaming a category inline updates its name', async () => { const user = userEvent.setup(); server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [cat1], items: [] }) ), http.put('/api/admin/packing-templates/1/categories/10', async () => HttpResponse.json({ success: true }) ) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('Clothing'); // Find the Edit2 button in the Clothing category header const clothingHeader = screen.getByText('Clothing').closest('div')!; const editBtns = Array.from(clothingHeader.querySelectorAll('button')).filter( b => b.className.includes('hover:text-slate-700') ); // Second button (after Plus) is Edit2 await user.click(editBtns[1]); const catInput = screen.getByDisplayValue('Clothing'); await user.clear(catInput); await user.type(catInput, 'Shoes{Enter}'); await screen.findByText('Shoes'); }); it('FE-ADMIN-PKG-014: deleting a category removes it and its items', async () => { const user = userEvent.setup(); server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [cat1], items: [item1, item2] }) ), http.delete('/api/admin/packing-templates/1/categories/10', () => HttpResponse.json({ success: true }) ) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('Clothing'); expect(screen.getByText('T-shirt')).toBeInTheDocument(); // Find the Trash2 button in the Clothing category header const clothingHeader = screen.getByText('Clothing').closest('div')!; const trashBtn = clothingHeader.querySelector('button.hover\\:text-red-500') as HTMLElement; await user.click(trashBtn); await waitFor(() => { expect(screen.queryByText('Clothing')).not.toBeInTheDocument(); expect(screen.queryByText('T-shirt')).not.toBeInTheDocument(); }); }); it('FE-ADMIN-PKG-015: renaming an item inline updates its name', async () => { const user = userEvent.setup(); server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [cat1], items: [item1] }) ), http.put('/api/admin/packing-templates/1/items/100', async () => HttpResponse.json({ success: true }) ) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('T-shirt'); // Find the Edit2 button in the T-shirt item row (opacity-0 group-hover buttons) const itemRow = screen.getByText('T-shirt').closest('div')!; const editBtn = Array.from(itemRow.querySelectorAll('button')).find( b => b.className.includes('opacity-0') ) as HTMLElement | undefined; if (editBtn) { await user.click(editBtn); } else { // Directly click the first button in the item row const btns = itemRow.querySelectorAll('button'); await user.click(btns[0] as HTMLElement); } const input = screen.getByDisplayValue('T-shirt'); await user.clear(input); await user.type(input, 'Tank Top{Enter}'); await screen.findByText('Tank Top'); }); it('FE-ADMIN-PKG-016: deleting an item removes it from the list', async () => { const user = userEvent.setup(); server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [cat1], items: [item1, item2] }) ), http.delete('/api/admin/packing-templates/1/items/100', () => HttpResponse.json({ success: true }) ) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('T-shirt'); expect(screen.getByText('Shorts')).toBeInTheDocument(); // Find the Trash2 button in the T-shirt row const itemRow = screen.getByText('T-shirt').closest('div')!; const trashBtns = Array.from(itemRow.querySelectorAll('button')).filter( b => b.className.includes('opacity-0') ); // Second opacity-0 button is the delete (trash) button const trashBtn = trashBtns[1] || trashBtns[0]; await user.click(trashBtn as HTMLElement); await waitFor(() => expect(screen.queryByText('T-shirt')).not.toBeInTheDocument()); expect(screen.getByText('Shorts')).toBeInTheDocument(); }); it('FE-ADMIN-PKG-017: Escape cancels add category without saving', async () => { const user = userEvent.setup(); let postCalled = false; server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [], items: [] }) ), http.post('/api/admin/packing-templates/1/categories', async () => { postCalled = true; return HttpResponse.json({ category: { id: 20, template_id: 1, name: 'Ignored', sort_order: 1 } }); }) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('Add category'); await user.click(screen.getByText('Add category')); const catInput = screen.getByPlaceholderText('Category name (e.g. Clothing)'); await user.type(catInput, 'Test{Escape}'); await waitFor(() => expect(screen.queryByPlaceholderText('Category name (e.g. Clothing)')).not.toBeInTheDocument() ); expect(postCalled).toBe(false); }); it('FE-ADMIN-PKG-018: Escape cancels add item without saving', async () => { const user = userEvent.setup(); let postCalled = false; server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.get('/api/admin/packing-templates/1', () => HttpResponse.json({ categories: [cat1], items: [] }) ), http.post('/api/admin/packing-templates/1/categories/10/items', async () => { postCalled = true; return HttpResponse.json({ item: { id: 102, category_id: 10, name: 'Ignored', sort_order: 2 } }); }) ); render(); await screen.findByText('Beach Trip'); await user.click(screen.getByText('Beach Trip')); await screen.findByText('Clothing'); const clothingHeader = screen.getByText('Clothing').closest('div')!; const addItemBtn = clothingHeader.querySelector('button') as HTMLElement; await user.click(addItemBtn); const itemInput = screen.getByPlaceholderText('Item name'); await user.type(itemInput, 'Test{Escape}'); await waitFor(() => expect(screen.queryByPlaceholderText('Item name')).not.toBeInTheDocument() ); expect(postCalled).toBe(false); }); it('FE-ADMIN-PKG-019: Escape cancels template rename without saving', async () => { const user = userEvent.setup(); let putCalled = false; server.use( http.get('/api/admin/packing-templates', () => HttpResponse.json({ templates: [tmpl1] }) ), http.put('/api/admin/packing-templates/1', async () => { putCalled = true; return HttpResponse.json({ success: true }); }) ); render(); await screen.findByText('Beach Trip'); const beachTripText = screen.getByText('Beach Trip'); const row = beachTripText.closest('div')!.parentElement!; const editBtn = row.querySelector('button.hover\\:bg-slate-100') as HTMLElement | null; if (editBtn) { await user.click(editBtn); } else { const allBtns = screen.getAllByRole('button'); const editBtns = allBtns.filter(b => b.className.includes('hover:bg-slate-100')); await user.click(editBtns[0]); } const input = screen.getByDisplayValue('Beach Trip'); await user.type(input, '{Escape}'); await waitFor(() => expect(screen.queryByDisplayValue('Beach Trip')).not.toBeInTheDocument()); expect(putCalled).toBe(false); // Original name should be restored expect(screen.getByText('Beach Trip')).toBeInTheDocument(); }); it('FE-ADMIN-PKG-020: X button on create template input dismisses it', async () => { const user = userEvent.setup(); render(); await screen.findByText('No templates created yet'); await user.click(screen.getByRole('button', { name: /new template/i })); expect(screen.getByPlaceholderText('Template name (e.g. Beach Holiday)')).toBeInTheDocument(); // Find the X (cancel) button in the create row — it's the last button in the create row const createRow = screen.getByPlaceholderText('Template name (e.g. Beach Holiday)').closest('div')!; const cancelBtn = Array.from(createRow.querySelectorAll('button')).at(-1) as HTMLElement; await user.click(cancelBtn); await waitFor(() => expect(screen.queryByPlaceholderText('Template name (e.g. Beach Holiday)')).not.toBeInTheDocument() ); }); });