EvalueTonSavoir/client/src/__tests__/pages/ManageRoom/ManageRoom.test.tsx

383 lines
13 KiB
TypeScript
Raw Normal View History

import React from 'react';
import { render, screen, fireEvent, waitFor, act } from '@testing-library/react';
import '@testing-library/jest-dom';
import { MemoryRouter, useNavigate, useParams } from 'react-router-dom';
import ManageRoom from 'src/pages/Teacher/ManageRoom/ManageRoom';
import { StudentType } from 'src/Types/StudentType';
import { QuizType } from 'src/Types/QuizType';
import webSocketService, { AnswerReceptionFromBackendType } from 'src/services/WebsocketService';
import ApiService from 'src/services/ApiService';
import { Socket } from 'socket.io-client';
import { RoomProvider } from 'src/pages/Teacher/ManageRoom/RoomContext';
jest.mock('src/services/WebsocketService');
jest.mock('src/services/ApiService');
jest.mock('react-router-dom', () => ({
...jest.requireActual('react-router-dom'),
useNavigate: jest.fn(),
useParams: jest.fn(),
}));
2025-02-24 14:32:08 -05:00
jest.mock('src/pages/Teacher/ManageRoom/RoomContext');
2025-03-12 14:25:17 -04:00
jest.mock('qrcode.react', () => ({
__esModule: true,
2025-03-13 16:09:04 -04:00
QRCodeCanvas: ({ value }: { value: string }) => <div data-testid="qr-code">{value}</div>,
2025-03-12 14:25:17 -04:00
}));
const mockSocket = {
on: jest.fn(),
off: jest.fn(),
emit: jest.fn(),
connect: jest.fn(),
disconnect: jest.fn(),
} as unknown as Socket;
const mockQuiz: QuizType = {
_id: 'test-quiz-id',
title: 'Test Quiz',
content: ['::Q1:: Question 1 { =Answer1 ~Answer2 }', '::Q2:: Question 2 { =Answer1 ~Answer2 }'],
folderId: 'folder-id',
folderName: 'folder-name',
userId: 'user-id',
created_at: new Date(),
2025-02-27 02:17:54 -05:00
updated_at: new Date(),
};
const mockStudents: StudentType[] = [
{ id: '1', name: 'Student 1', answers: [] },
{ id: '2', name: 'Student 2', answers: [] },
];
const mockAnswerData: AnswerReceptionFromBackendType = {
2025-03-21 00:25:25 -04:00
answer: ['Answer1'],
idQuestion: 1,
idUser: '1',
username: 'Student 1',
};
describe('ManageRoom', () => {
const navigate = jest.fn();
const useParamsMock = useParams as jest.Mock;
2025-02-24 14:32:08 -05:00
const mockSetSelectedRoom = jest.fn();
beforeEach(() => {
jest.clearAllMocks();
(useNavigate as jest.Mock).mockReturnValue(navigate);
2025-02-27 02:17:54 -05:00
useParamsMock.mockReturnValue({ quizId: 'test-quiz-id', roomName: 'Test Room' });
(ApiService.getQuiz as jest.Mock).mockResolvedValue(mockQuiz);
(webSocketService.connect as jest.Mock).mockReturnValue(mockSocket);
(RoomProvider as jest.Mock).mockReturnValue({
2025-02-24 14:32:08 -05:00
selectedRoom: { id: '1', title: 'Test Room' },
2025-02-27 02:17:54 -05:00
setSelectedRoom: mockSetSelectedRoom,
2025-02-24 14:32:08 -05:00
});
});
test('prepares to launch quiz and fetches quiz data', async () => {
await act(async () => {
render(
<MemoryRouter>
<ManageRoom />
</MemoryRouter>
);
});
2025-02-24 14:32:08 -05:00
await act(async () => {
const createSuccessCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'create-success')[1];
2025-02-24 14:32:08 -05:00
createSuccessCallback('Test Room');
});
2025-02-24 14:32:08 -05:00
await waitFor(() => {
expect(ApiService.getQuiz).toHaveBeenCalledWith('test-quiz-id');
});
2025-02-27 02:17:54 -05:00
const launchButton = screen.getByText('Lancer');
fireEvent.click(launchButton);
2025-02-24 14:32:08 -05:00
const rythmeButton = screen.getByText('Rythme du professeur');
fireEvent.click(rythmeButton);
2025-02-24 14:32:08 -05:00
const secondLaunchButton = screen.getAllByText('Lancer');
fireEvent.click(secondLaunchButton[1]);
2025-02-24 14:32:08 -05:00
await waitFor(() => {
expect(screen.getByText('Test Quiz')).toBeInTheDocument();
2025-02-27 02:17:54 -05:00
const roomHeader = document.querySelector('h1');
expect(roomHeader).toHaveTextContent('Salle : TEST ROOM');
2025-02-27 02:17:54 -05:00
expect(screen.getByText('0/60')).toBeInTheDocument();
expect(screen.getByText('Question 1/2')).toBeInTheDocument();
});
});
2025-02-27 02:17:54 -05:00
test('handles create-success event', async () => {
await act(async () => {
render(
<MemoryRouter>
<ManageRoom />
</MemoryRouter>
);
});
2025-02-27 02:17:54 -05:00
await act(async () => {
const createSuccessCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'create-success')[1];
2025-02-27 02:17:54 -05:00
createSuccessCallback('Test Room');
});
2025-02-27 02:17:54 -05:00
await waitFor(() => {
2025-02-27 02:17:54 -05:00
expect(screen.getByText(/Salle\s*:\s*Test Room/i)).toBeInTheDocument();
});
});
test('handles user-joined event', async () => {
await act(async () => {
render(
<MemoryRouter>
<ManageRoom />
</MemoryRouter>
);
});
await act(async () => {
const createSuccessCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'create-success')[1];
2025-02-27 02:17:54 -05:00
createSuccessCallback('Test Room');
});
await act(async () => {
const userJoinedCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'user-joined')[1];
userJoinedCallback(mockStudents[0]);
});
await waitFor(() => {
expect(screen.getByText('Student 1')).toBeInTheDocument();
});
const launchButton = screen.getByText('Lancer');
fireEvent.click(launchButton);
const rythmeButton = screen.getByText('Rythme du professeur');
fireEvent.click(rythmeButton);
const secondLaunchButton = screen.getAllByText('Lancer');
fireEvent.click(secondLaunchButton[1]);
await waitFor(() => {
expect(screen.getByText('1/60')).toBeInTheDocument();
});
});
test('handles next question', async () => {
await act(async () => {
render(
<MemoryRouter>
<ManageRoom />
</MemoryRouter>
);
});
2025-02-27 02:17:54 -05:00
await act(async () => {
const createSuccessCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'create-success')[1];
2025-02-27 02:17:54 -05:00
createSuccessCallback('Test Room');
});
2025-02-27 02:17:54 -05:00
fireEvent.click(screen.getByText('Lancer'));
fireEvent.click(screen.getByText('Rythme du professeur'));
fireEvent.click(screen.getAllByText('Lancer')[1]);
await waitFor(() => {
screen.debug();
});
const nextQuestionButton = await screen.findByRole('button', { name: /Prochaine question/i });
expect(nextQuestionButton).toBeInTheDocument();
fireEvent.click(nextQuestionButton);
2025-02-27 02:17:54 -05:00
await waitFor(() => {
expect(screen.getByText('Question 2/2')).toBeInTheDocument();
});
});
2025-02-27 02:17:54 -05:00
test('handles disconnect', async () => {
await act(async () => {
render(
<MemoryRouter>
<ManageRoom />
</MemoryRouter>
);
});
await act(async () => {
const createSuccessCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'create-success')[1];
2025-02-27 02:17:54 -05:00
createSuccessCallback('Test Room');
});
const disconnectButton = screen.getByText('Quitter');
fireEvent.click(disconnectButton);
const confirmButton = screen.getAllByText('Confirmer');
fireEvent.click(confirmButton[1]);
await waitFor(() => {
expect(webSocketService.disconnect).toHaveBeenCalled();
expect(navigate).toHaveBeenCalledWith('/teacher/dashboard');
});
});
2025-02-24 14:32:08 -05:00
test('handles submit-answer-room event', async () => {
const consoleSpy = jest.spyOn(console, 'log');
await act(async () => {
render(
<MemoryRouter>
<ManageRoom />
</MemoryRouter>
);
});
await act(async () => {
const createSuccessCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'create-success')[1];
2025-03-21 00:25:25 -04:00
createSuccessCallback('Test Room');
2025-02-24 14:32:08 -05:00
});
const launchButton = screen.getByText('Lancer');
fireEvent.click(launchButton);
const rythmeButton = screen.getByText('Rythme du professeur');
fireEvent.click(rythmeButton);
const secondLaunchButton = screen.getAllByText('Lancer');
fireEvent.click(secondLaunchButton[1]);
await act(async () => {
const userJoinedCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'user-joined')[1];
userJoinedCallback(mockStudents[0]);
});
await act(async () => {
const submitAnswerCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'submit-answer-room')[1];
submitAnswerCallback(mockAnswerData);
});
await waitFor(() => {
2025-03-21 00:25:25 -04:00
// console.info(consoleSpy.mock.calls);
2025-02-24 14:32:08 -05:00
expect(consoleSpy).toHaveBeenCalledWith(
'Received answer from Student 1 for question 1: Answer1'
);
});
consoleSpy.mockRestore();
});
2025-02-27 02:17:54 -05:00
2025-02-24 14:32:08 -05:00
test('vide la liste des étudiants après déconnexion', async () => {
await act(async () => {
render(
<MemoryRouter>
<ManageRoom />
</MemoryRouter>
);
});
await act(async () => {
const createSuccessCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'create-success')[1];
2025-02-27 02:17:54 -05:00
createSuccessCallback('Test Room');
2025-02-24 14:32:08 -05:00
});
await act(async () => {
const userJoinedCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'user-joined')[1];
userJoinedCallback(mockStudents[0]);
});
const disconnectButton = screen.getByText('Quitter');
fireEvent.click(disconnectButton);
const confirmButton = screen.getAllByText('Confirmer');
fireEvent.click(confirmButton[1]);
await waitFor(() => {
expect(screen.queryByText('Student 1')).not.toBeInTheDocument();
});
});
2025-03-24 03:00:05 -04:00
test('terminates the quiz and navigates to teacher dashboard when the "Terminer le quiz" button is clicked', async () => {
await act(async () => {
render(
<MemoryRouter>
<ManageRoom />
</MemoryRouter>
);
});
await act(async () => {
const createSuccessCallback = (mockSocket.on as jest.Mock).mock.calls.find(call => call[0] === 'create-success')[1];
createSuccessCallback('Test Room');
});
fireEvent.click(screen.getByText('Lancer'));
fireEvent.click(screen.getByText('Rythme du professeur'));
fireEvent.click(screen.getAllByText('Lancer')[1]);
await waitFor(() => {
expect(screen.getByText('Test Quiz')).toBeInTheDocument();
});
const finishQuizButton = screen.getByText('Terminer le quiz');
fireEvent.click(finishQuizButton);
await waitFor(() => {
expect(navigate).toHaveBeenCalledWith('/teacher/dashboard');
});
});
2025-03-19 14:41:59 -04:00
test("Affiche la modale QR Code lorsquon clique sur le bouton", async () => {
2025-03-12 14:25:17 -04:00
render(<MemoryRouter><ManageRoom /></MemoryRouter>);
2025-03-19 14:41:59 -04:00
2025-03-12 14:25:17 -04:00
const button = screen.getByRole('button', { name: /lien de participation/i });
fireEvent.click(button);
2025-03-19 14:41:59 -04:00
await waitFor(() => {
expect(screen.getByRole('dialog')).toBeInTheDocument();
});
expect(screen.getByRole('heading', { name: /Rejoindre la salle/i })).toBeInTheDocument();
expect(screen.getByText(/Scannez ce QR code ou partagez le lien ci-dessous/i)).toBeInTheDocument();
2025-03-12 14:25:17 -04:00
expect(screen.getByTestId('qr-code')).toBeInTheDocument();
});
2025-03-19 14:41:59 -04:00
test("Ferme la modale QR Code lorsquon clique sur le bouton Fermer", async () => {
2025-03-12 14:25:17 -04:00
render(<MemoryRouter><ManageRoom /></MemoryRouter>);
2025-03-19 14:41:59 -04:00
2025-03-12 14:25:17 -04:00
fireEvent.click(screen.getByRole('button', { name: /lien de participation/i }));
2025-03-19 14:41:59 -04:00
await waitFor(() => {
expect(screen.getByRole('dialog')).toBeInTheDocument();
});
2025-03-12 14:25:17 -04:00
fireEvent.click(screen.getByRole('button', { name: /fermer/i }));
2025-03-19 14:41:59 -04:00
await waitFor(() => {
expect(screen.queryByRole('dialog')).not.toBeInTheDocument();
});
2025-03-12 14:25:17 -04:00
});
2025-03-19 14:41:59 -04:00
2025-03-12 14:25:17 -04:00
test('Affiche le bon lien de participation', () => {
render(<MemoryRouter><ManageRoom /></MemoryRouter>);
fireEvent.click(screen.getByRole('button', { name: /lien de participation/i }));
const roomUrl = `${window.location.origin}/student/join-room?roomName=Test Room`;
expect(screen.getByTestId('qr-code')).toHaveTextContent(roomUrl);
});
test('Vérifie que le QR code contient la bonne URL', () => {
render(<MemoryRouter><ManageRoom /></MemoryRouter>);
fireEvent.click(screen.getByRole('button', { name: /lien de participation/i }));
const roomUrl = `${window.location.origin}/student/join-room?roomName=Test Room`;
2025-03-13 15:56:02 -04:00
expect(screen.getByTestId('qr-code')).toHaveTextContent(roomUrl);
2025-03-12 14:25:17 -04:00
});
2025-02-24 14:32:08 -05:00
});
2025-03-21 00:25:25 -04:00