This commit is contained in:
NouhailaAater 2025-04-08 23:36:00 +00:00 committed by GitHub
commit 3347c4bba1
No known key found for this signature in database
GPG key ID: B5690EEEBB952194
8 changed files with 232 additions and 131 deletions

View file

@ -2,6 +2,7 @@ import { Link, useNavigate } from 'react-router-dom';
import * as React from 'react'; import * as React from 'react';
import './header.css'; import './header.css';
import { Button } from '@mui/material'; import { Button } from '@mui/material';
import ExitToAppIcon from '@mui/icons-material/ExitToApp';
interface HeaderProps { interface HeaderProps {
isLoggedIn: boolean; isLoggedIn: boolean;
@ -28,8 +29,9 @@ const Header: React.FC<HeaderProps> = ({ isLoggedIn, handleLogout }) => {
handleLogout(); handleLogout();
navigate('/'); navigate('/');
}} }}
startIcon={<ExitToAppIcon />}
> >
Logout Déconnexion
</Button> </Button>
)} )}

View file

@ -59,7 +59,6 @@ const TrueFalseQuestionDisplay: React.FC<Props> = (props) => {
disabled={disableButton} disabled={disableButton}
> >
{showAnswer ? (<div> {(question.isTrue ? '✅' : '❌')}</div>) : ``} {showAnswer ? (<div> {(question.isTrue ? '✅' : '❌')}</div>) : ``}
<div className={`circle ${selectedTrue}`}>V</div>
<div className={`answer-text ${selectedTrue}`}>Vrai</div> <div className={`answer-text ${selectedTrue}`}>Vrai</div>
{showAnswer && answer && question.trueFormattedFeedback && ( {showAnswer && answer && question.trueFormattedFeedback && (
@ -76,7 +75,6 @@ const TrueFalseQuestionDisplay: React.FC<Props> = (props) => {
> >
{showAnswer ? (<div> {(!question.isTrue ? '✅' : '❌')}</div>) : ``} {showAnswer ? (<div> {(!question.isTrue ? '✅' : '❌')}</div>) : ``}
<div className={`circle ${selectedFalse}`}>F</div>
<div className={`answer-text ${selectedFalse}`}>Faux</div> <div className={`answer-text ${selectedFalse}`}>Faux</div>
{showAnswer && !answer && question.falseFormattedFeedback && ( {showAnswer && !answer && question.falseFormattedFeedback && (

View file

@ -26,8 +26,7 @@ const StudentWaitPage: React.FC<Props> = ({ students, launchQuiz, setQuizMode })
variant="contained" variant="contained"
onClick={handleLaunchClick} onClick={handleLaunchClick}
startIcon={<PlayArrow />} startIcon={<PlayArrow />}
fullWidth sx={{ fontWeight: 600, fontSize: 20, width: 'auto' }}
sx={{ fontWeight: 600, fontSize: 20 }}
> >
Lancer Lancer
</Button> </Button>

View file

@ -116,7 +116,7 @@ const TeacherModeQuiz: React.FC<TeacherModeQuizProps> = ({
</DialogContent> </DialogContent>
<DialogActions> <DialogActions>
<Button onClick={handleFeedbackDialogClose} color="primary"> <Button onClick={handleFeedbackDialogClose} color="primary">
OK Fermer
</Button> </Button>
</DialogActions> </DialogActions>
</Dialog> </Dialog>

View file

@ -44,7 +44,6 @@ const SimpleLogin: React.FC = () => {
variant="outlined" variant="outlined"
value={email} value={email}
onChange={(e) => setEmail(e.target.value)} onChange={(e) => setEmail(e.target.value)}
placeholder="Nom d'utilisateur"
sx={{ marginBottom: '1rem' }} sx={{ marginBottom: '1rem' }}
fullWidth fullWidth
/> />
@ -55,7 +54,6 @@ const SimpleLogin: React.FC = () => {
type="password" type="password"
value={password} value={password}
onChange={(e) => setPassword(e.target.value)} onChange={(e) => setPassword(e.target.value)}
placeholder="Nom de la salle"
sx={{ marginBottom: '1rem' }} sx={{ marginBottom: '1rem' }}
fullWidth fullWidth
/> />

View file

@ -38,7 +38,7 @@ import {
Upload, Upload,
FolderCopy, FolderCopy,
ContentCopy, ContentCopy,
Edit, Edit
} from '@mui/icons-material'; } from '@mui/icons-material';
import ShareQuizModal from 'src/components/ShareQuizModal/ShareQuizModal'; import ShareQuizModal from 'src/components/ShareQuizModal/ShareQuizModal';
@ -65,6 +65,7 @@ const Dashboard: React.FC = () => {
const [selectedRoom, selectRoom] = useState<RoomType>(); // menu const [selectedRoom, selectRoom] = useState<RoomType>(); // menu
const [errorMessage, setErrorMessage] = useState(''); const [errorMessage, setErrorMessage] = useState('');
const [showErrorDialog, setShowErrorDialog] = useState(false); const [showErrorDialog, setShowErrorDialog] = useState(false);
const [isSearchVisible, setIsSearchVisible] = useState(false);
// Filter quizzes based on search term // Filter quizzes based on search term
// const filteredQuizzes = quizzes.filter(quiz => // const filteredQuizzes = quizzes.filter(quiz =>
@ -120,6 +121,10 @@ const Dashboard: React.FC = () => {
} }
}; };
const toggleSearchVisibility = () => {
setIsSearchVisible(!isSearchVisible);
};
// Créer une salle // Créer une salle
const createRoom = async (title: string) => { const createRoom = async (title: string) => {
// Créer la salle et récupérer l'objet complet // Créer la salle et récupérer l'objet complet
@ -133,10 +138,9 @@ const Dashboard: React.FC = () => {
selectRoomByName(newRoom); // Utiliser l'ID de l'objet retourné selectRoomByName(newRoom); // Utiliser l'ID de l'objet retourné
}; };
// Sélectionner une salle // Sélectionner une salle
const selectRoomByName = (roomId: string) => { const selectRoomByName = (roomId: string) => {
const room = rooms.find(r => r._id === roomId); const room = rooms.find((r) => r._id === roomId);
selectRoom(room); selectRoom(room);
localStorage.setItem('selectedRoomId', roomId); localStorage.setItem('selectedRoomId', roomId);
}; };
@ -150,7 +154,7 @@ const Dashboard: React.FC = () => {
setOpenAddRoomDialog(false); setOpenAddRoomDialog(false);
setNewRoomTitle(''); setNewRoomTitle('');
} catch (error) { } catch (error) {
setErrorMessage(error instanceof Error ? error.message : "Erreur inconnue"); setErrorMessage(error instanceof Error ? error.message : 'Erreur inconnue');
setShowErrorDialog(true); setShowErrorDialog(true);
} }
} }
@ -402,30 +406,63 @@ const Dashboard: React.FC = () => {
return ( return (
<div className="dashboard"> <div className="dashboard">
<div className="title">Tableau de bord</div> {/* Conteneur pour le titre et le sélecteur de salle */}
<div
style={{
display: 'flex',
justifyContent: 'space-between',
alignItems: 'center',
marginBottom: '20px'
}}
>
{/* Titre tableau de bord */}
<div className="title" style={{ fontSize: '30px', fontWeight: 'bold' }}>
Tableau de bord
</div>
<div className="roomSelection"> {/* Sélecteur de salle */}
<label htmlFor="select-room">Sélectionner une salle: </label> <div
<select value={selectedRoom?._id || ''} onChange={(e) => handleSelectRoom(e)}> className="roomSelection"
style={{ display: 'flex', justifyContent: 'flex-end', gap: '15px' }}
>
<select
value={selectedRoom?._id || ''}
onChange={(e) => handleSelectRoom(e)}
id="room-select"
style={{
padding: '8px 12px',
fontSize: '14px',
borderRadius: '8px',
border: '1px solid #ccc',
backgroundColor: '#fff',
maxWidth: '200px',
cursor: 'pointer',
fontWeight: '500'
}}
>
<option value="" disabled> <option value="" disabled>
-- Sélectionner une salle -- Sélectionner une salle
</option> </option>
{rooms.map((room) => ( {rooms.map((room) => (
<option key={room._id} value={room._id}> <option key={room._id} value={room._id}>
{room.title} {room.title}
</option> </option>
))} ))}
<option value="add-room">Ajouter salle</option> <option
value="add-room"
style={{
color: 'black',
backgroundColor: '#f0f0f0',
fontWeight: 'bold'
}}
>
Ajouter une salle
</option>
</select> </select>
</div>
</div> </div>
{selectedRoom && ( {/* Dialog pour créer une salle */}
<div className="roomTitle">
<h2>Salle sélectionnée: {selectedRoom.title}</h2>
</div>
)}
<Dialog open={openAddRoomDialog} onClose={() => setOpenAddRoomDialog(false)}> <Dialog open={openAddRoomDialog} onClose={() => setOpenAddRoomDialog(false)}>
<DialogTitle>Créer une nouvelle salle</DialogTitle> <DialogTitle>Créer une nouvelle salle</DialogTitle>
<DialogContent> <DialogContent>
@ -450,24 +487,17 @@ const Dashboard: React.FC = () => {
</DialogActions> </DialogActions>
</Dialog> </Dialog>
<div className="search-bar"> <div
<TextField style={{
onChange={handleSearch} display: 'flex',
value={searchTerm} justifyContent: 'flex-end',
placeholder="Rechercher un quiz par son titre" alignItems: 'center',
fullWidth width: '100%',
InputProps={{ gap: '20px'
endAdornment: (
<InputAdornment position="end">
<IconButton>
<Search />
</IconButton>
</InputAdornment>
)
}} }}
/> ></div>
</div>
{/* Conteneur principal avec les actions et la liste des quiz */}
<div className="folder"> <div className="folder">
<div className="select"> <div className="select">
<NativeSelect <NativeSelect
@ -475,13 +505,18 @@ const Dashboard: React.FC = () => {
color="primary" color="primary"
value={selectedFolderId} value={selectedFolderId}
onChange={handleSelectFolder} onChange={handleSelectFolder}
sx={{
padding: '6px 12px',
maxWidth: '180px',
borderRadius: '8px',
borderColor: '#e0e0e0',
'&:hover': { borderColor: '#5271FF' }
}}
> >
<option value=""> Tous les dossiers... </option> <option value="">Tous les dossiers...</option>
{folders.map((folder) => (
{folders.map((folder: FolderType) => (
<option value={folder._id} key={folder._id}> <option value={folder._id} key={folder._id}>
{' '} {folder.title}
{folder.title}{' '}
</option> </option>
))} ))}
</NativeSelect> </NativeSelect>
@ -537,14 +572,77 @@ const Dashboard: React.FC = () => {
</div> </div>
</div> </div>
<div className="ajouter"> <div
className="search-bar"
style={{
display: 'flex',
justifyContent: 'space-between',
alignItems: 'center',
gap: '20px',
width: '100%'
}}
>
<div style={{ flex: 1 }}>
{!isSearchVisible ? (
<IconButton
onClick={toggleSearchVisibility}
sx={{
borderRadius: '8px',
border: '1px solid #ccc',
padding: '8px 12px',
backgroundColor: '#fff',
color: '#5271FF'
}}
>
<Search />
</IconButton>
) : (
<TextField
onChange={handleSearch}
value={searchTerm}
placeholder="Rechercher un quiz"
fullWidth
autoFocus
sx={{
borderRadius: '8px',
border: '1px solid #ccc',
padding: '8px 12px',
backgroundColor: '#fff',
fontWeight: 500,
width: '100%',
maxWidth: '1000px'
}}
InputProps={{
endAdornment: (
<InputAdornment position="end">
<IconButton
onClick={toggleSearchVisibility}
sx={{
borderRadius: '8px',
border: '1px solid #ccc',
backgroundColor: '#fff',
color: '#5271FF'
}}
>
<Search />
</IconButton>
</InputAdornment>
)
}}
/>
)}
</div>
{/* À droite : les boutons */}
<div style={{ display: 'flex', gap: '12px' }}>
<Button <Button
variant="outlined" variant="outlined"
color="primary" color="primary"
startIcon={<Add />} startIcon={<Add />}
onClick={handleCreateQuiz} onClick={handleCreateQuiz}
sx={{ borderRadius: '8px', minWidth: 'auto', padding: '4px 12px' }}
> >
Ajouter un nouveau quiz Nouveau quiz
</Button> </Button>
<Button <Button
@ -553,9 +651,11 @@ const Dashboard: React.FC = () => {
startIcon={<Upload />} startIcon={<Upload />}
onClick={handleOnImport} onClick={handleOnImport}
> >
Import Importer
</Button> </Button>
</div> </div>
</div>
<div className="list"> <div className="list">
{Object.keys(quizzesByFolder).map((folderName) => ( {Object.keys(quizzesByFolder).map((folderName) => (
<CustomCard key={folderName} className="folder-card"> <CustomCard key={folderName} className="folder-card">
@ -571,7 +671,9 @@ const Dashboard: React.FC = () => {
onClick={() => handleLancerQuiz(quiz)} onClick={() => handleLancerQuiz(quiz)}
disabled={!validateQuiz(quiz.content)} disabled={!validateQuiz(quiz.content)}
> >
{`${quiz.title} (${quiz.content.length} question${ {`${quiz.title} (${
quiz.content.length
} question${
quiz.content.length > 1 ? 's' : '' quiz.content.length > 1 ? 's' : ''
})`} })`}
</Button> </Button>
@ -609,21 +711,20 @@ const Dashboard: React.FC = () => {
<ContentCopy />{' '} <ContentCopy />{' '}
</IconButton> </IconButton>
</Tooltip> </Tooltip>
<div className="quiz-share">
<ShareQuizModal quiz={quiz} />
</div>
<Tooltip title="Supprimer" placement="top"> <Tooltip title="Supprimer" placement="top">
<IconButton <IconButton
aria-label="delete" aria-label="delete"
color="primary" color="error"
onClick={() => handleRemoveQuiz(quiz)} onClick={() => handleRemoveQuiz(quiz)}
> >
{' '} {' '}
<DeleteOutline />{' '} <DeleteOutline />{' '}
</IconButton> </IconButton>
</Tooltip> </Tooltip>
<div className="quiz-share">
<ShareQuizModal quiz={quiz} />
</div>
</div> </div>
</div> </div>
))} ))}

View file

@ -9,7 +9,7 @@ import GiftCheatSheet from 'src/components/GIFTCheatSheet/GiftCheatSheet';
import GIFTTemplatePreview from 'src/components/GiftTemplate/GIFTTemplatePreview'; import GIFTTemplatePreview from 'src/components/GiftTemplate/GIFTTemplatePreview';
import { QuizType } from '../../../Types/QuizType'; import { QuizType } from '../../../Types/QuizType';
import SaveIcon from '@mui/icons-material/Save';
import './editorQuiz.css'; import './editorQuiz.css';
import { Button, TextField, NativeSelect, Divider } from '@mui/material'; import { Button, TextField, NativeSelect, Divider } from '@mui/material';
import ReturnButton from 'src/components/ReturnButton/ReturnButton'; import ReturnButton from 'src/components/ReturnButton/ReturnButton';
@ -214,6 +214,7 @@ const QuizForm: React.FC = () => {
</NativeSelect></label> </NativeSelect></label>
<Button variant="contained" onClick={handleQuizSave}> <Button variant="contained" onClick={handleQuizSave}>
<SaveIcon sx={{ fontSize: 20 }} />
Enregistrer Enregistrer
</Button> </Button>

View file

@ -5,6 +5,7 @@ import './share.css';
import { Button, NativeSelect, Typography, Box } from '@mui/material'; import { Button, NativeSelect, Typography, Box } from '@mui/material';
import ReturnButton from 'src/components/ReturnButton/ReturnButton'; import ReturnButton from 'src/components/ReturnButton/ReturnButton';
import ApiService from '../../../services/ApiService'; import ApiService from '../../../services/ApiService';
import SaveIcon from '@mui/icons-material/Save';
const Share: React.FC = () => { const Share: React.FC = () => {
const navigate = useNavigate(); const navigate = useNavigate();
@ -167,6 +168,7 @@ const Share: React.FC = () => {
</NativeSelect> </NativeSelect>
<Button variant="contained" onClick={handleQuizSave} className="saveButton"> <Button variant="contained" onClick={handleQuizSave} className="saveButton">
{<SaveIcon sx={{ fontSize: 20, marginRight: '8px' }} />}
Enregistrer Enregistrer
</Button> </Button>
</div> </div>