mirror of
https://github.com/mauriceboe/TREK.git
synced 2026-06-19 21:31:46 +00:00
d4bb8be86b
Adds ~45 new and updated test files covering Admin, Collab, Dashboard, Map, Memories, PDF, Photos, Planner, Settings, Vacay, Weather components, pages, stores, and a WebSocket integration test.
511 lines
21 KiB
TypeScript
511 lines
21 KiB
TypeScript
// 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(<PackingTemplateManager />);
|
|
expect(document.querySelector('.animate-spin')).toBeInTheDocument();
|
|
});
|
|
|
|
it('FE-ADMIN-PKG-002: shows empty state when no templates', async () => {
|
|
render(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<><ToastContainer /><PackingTemplateManager /></>);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<><ToastContainer /><PackingTemplateManager /></>);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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(<PackingTemplateManager />);
|
|
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()
|
|
);
|
|
});
|
|
});
|