import { describe, it, expect, beforeEach, vi } from 'vitest'; import { http, HttpResponse } from 'msw'; import { useTripStore } from '../../../src/store/tripStore'; import { resetAllStores, seedStore } from '../../helpers/store'; import { buildDay, buildDayNote } from '../../helpers/factories'; import { server } from '../../helpers/msw/server'; vi.mock('../../../src/api/websocket', () => ({ connect: vi.fn(), disconnect: vi.fn(), getSocketId: vi.fn(() => null), joinTrip: vi.fn(), leaveTrip: vi.fn(), addListener: vi.fn(), removeListener: vi.fn(), setRefetchCallback: vi.fn(), setPreReconnectHook: vi.fn(), })); beforeEach(() => { resetAllStores(); }); describe('dayNotesSlice', () => { describe('addDayNote', () => { it('FE-DAYNOTES-001: addDayNote inserts temp note immediately, replaces on success', async () => { seedStore(useTripStore, { dayNotes: { '1': [] } }); let tempAdded = false; const realNote = buildDayNote({ id: 500, day_id: 1, text: 'New note' }); server.use( http.post('/api/trips/1/days/1/notes', async () => { const state = useTripStore.getState(); const notes = state.dayNotes['1']; if (notes.some(n => n.id < 0)) { tempAdded = true; } return HttpResponse.json({ note: realNote }); }), ); const result = await useTripStore.getState().addDayNote(1, 1, { text: 'New note', sort_order: 0 }); expect(tempAdded).toBe(true); expect(result.id).toBe(500); const notes = useTripStore.getState().dayNotes['1']; expect(notes).toHaveLength(1); expect(notes[0].id).toBe(500); }); it('FE-DAYNOTES-002: addDayNote on failure rolls back — temp note removed', async () => { seedStore(useTripStore, { dayNotes: { '1': [] } }); server.use( http.post('/api/trips/1/days/1/notes', () => HttpResponse.json({ message: 'Error' }, { status: 500 }) ), ); await expect( useTripStore.getState().addDayNote(1, 1, { text: 'Fail note', sort_order: 0 }) ).rejects.toThrow(); expect(useTripStore.getState().dayNotes['1']).toHaveLength(0); }); }); describe('updateDayNote', () => { it('FE-DAYNOTES-003: updateDayNote replaces note in map by id', async () => { const note = buildDayNote({ id: 10, day_id: 1, text: 'Old text' }); seedStore(useTripStore, { dayNotes: { '1': [note] } }); const updated = { ...note, text: 'Updated text' }; server.use( http.put('/api/trips/1/days/1/notes/10', () => HttpResponse.json({ note: updated }) ), ); const result = await useTripStore.getState().updateDayNote(1, 1, 10, { text: 'Updated text' }); expect(result.text).toBe('Updated text'); expect(useTripStore.getState().dayNotes['1'][0].text).toBe('Updated text'); }); }); describe('deleteDayNote', () => { it('FE-DAYNOTES-004: deleteDayNote optimistically removes note, restores on failure', async () => { const note = buildDayNote({ id: 10, day_id: 1 }); seedStore(useTripStore, { dayNotes: { '1': [note] } }); server.use( http.delete('/api/trips/1/days/1/notes/10', () => HttpResponse.json({ message: 'Error' }, { status: 500 }) ), ); await expect(useTripStore.getState().deleteDayNote(1, 1, 10)).rejects.toThrow(); // Rolled back expect(useTripStore.getState().dayNotes['1']).toHaveLength(1); expect(useTripStore.getState().dayNotes['1'][0].id).toBe(10); }); it('FE-DAYNOTES-004b: deleteDayNote success removes note from correct day', async () => { const note1 = buildDayNote({ id: 10, day_id: 1 }); const note2 = buildDayNote({ id: 20, day_id: 1 }); seedStore(useTripStore, { dayNotes: { '1': [note1, note2] } }); await useTripStore.getState().deleteDayNote(1, 1, 10); const notes = useTripStore.getState().dayNotes['1']; expect(notes).toHaveLength(1); expect(notes[0].id).toBe(20); }); }); describe('moveDayNote', () => { it('FE-DAYNOTES-005: moveDayNote removes from source, adds to target (delete+create)', async () => { const note = buildDayNote({ id: 10, day_id: 1, text: 'Move me' }); const newNote = buildDayNote({ id: 99, day_id: 2, text: 'Move me' }); seedStore(useTripStore, { dayNotes: { '1': [note], '2': [] } }); server.use( http.delete('/api/trips/1/days/1/notes/10', () => HttpResponse.json({ success: true })), http.post('/api/trips/1/days/2/notes', () => HttpResponse.json({ note: newNote })), ); await useTripStore.getState().moveDayNote(1, 1, 2, 10); expect(useTripStore.getState().dayNotes['1']).toHaveLength(0); expect(useTripStore.getState().dayNotes['2']).toHaveLength(1); expect(useTripStore.getState().dayNotes['2'][0].id).toBe(99); }); it('FE-DAYNOTES-006: moveDayNote rolls back to source day on failure', async () => { const note = buildDayNote({ id: 10, day_id: 1, text: 'Move me' }); seedStore(useTripStore, { dayNotes: { '1': [note], '2': [] } }); server.use( http.delete('/api/trips/1/days/1/notes/10', () => HttpResponse.json({ message: 'Error' }, { status: 500 }) ), ); await expect(useTripStore.getState().moveDayNote(1, 1, 2, 10)).rejects.toThrow(); expect(useTripStore.getState().dayNotes['1']).toHaveLength(1); expect(useTripStore.getState().dayNotes['1'][0].id).toBe(10); }); }); describe('updateDayNotes', () => { it('FE-DAYNOTES-007: updateDayNotes persists notes text and updates days array', async () => { const day = buildDay({ id: 1, trip_id: 1, notes: null }); seedStore(useTripStore, { days: [day] }); await useTripStore.getState().updateDayNotes(1, 1, 'My travel notes'); const updatedDay = useTripStore.getState().days.find(d => d.id === 1); expect(updatedDay?.notes).toBe('My travel notes'); }); }); describe('updateDayTitle', () => { it('FE-DAYNOTES-008: updateDayTitle persists title and updates days array', async () => { const day = buildDay({ id: 1, trip_id: 1, title: null }); seedStore(useTripStore, { days: [day] }); await useTripStore.getState().updateDayTitle(1, 1, 'Day at the Beach'); const updatedDay = useTripStore.getState().days.find(d => d.id === 1); expect(updatedDay?.title).toBe('Day at the Beach'); }); }); });