Skip to content

Commit

Permalink
Merge pull request #29 from zakie2003/zak-branch3
Browse files Browse the repository at this point in the history
Ready Question Edit Added
  • Loading branch information
nishant0708 authored Sep 12, 2024
2 parents c4db714 + 25ce5d2 commit 20be6e0
Show file tree
Hide file tree
Showing 3 changed files with 232 additions and 5 deletions.
3 changes: 2 additions & 1 deletion src/App.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@ import QuestionPaperDashboard from './QuestionPaperDashboard/QuestionPaperDashbo
import Editpaper from './Edit_paper/Editpaper';
import ReadyPaperDashboard from './ReadyPaperDashboard/ReadyPaperDashboard';
import ReadyQuestionPaperDashboard from './ReadyQuestionPaperDashboard/ReadyQuestionPaperDashboard';

import EditReadyQuestion from './edit_ready_question/EditReadyQuestion';
import Error404 from './error/error404';
import EditQuestion from './edit_question/EditQuestion';

Expand Down Expand Up @@ -72,6 +72,7 @@ const App = () => {
<Route path="/edit-paper" element={<Editpaper/>} />
<Route path="/add-question/:paperId" element={<Question />} />
<Route path="/edit-question/:paperId/:questionId" element={<EditQuestion/>} />
<Route path='/edit-ready-question/:paperId/:questionId' element={<EditReadyQuestion/>} />
<Route path="/questionPaperDashboard/:paperId" element={<QuestionPaperDashboard />}/>
<Route path="/ready_papers" element={<ReadyPaperDashboard />}/>
<Route path="/ready_questions/:paperId" element={<ReadyQuestionPaperDashboard />} />
Expand Down
23 changes: 19 additions & 4 deletions src/ReadyQuestionPaperDashboard/ReadyQuestionPaperDashboard.jsx
Original file line number Diff line number Diff line change
@@ -1,24 +1,25 @@
import React from 'react'
import "../QuestionPaperDashboard/QuestionPaperDashboard.css";
import Navbar from '../Navbar/Navbar';
import { useParams } from "react-router-dom";
import { useParams,useNavigate } from "react-router-dom";
import axios from "axios";
import Nothing from "../Assets/nothing.svg";
import AlertModal from '../AlertModal/AlertModal';
import { useState,useEffect } from 'react';
import Skeleton from '../Skeleton/Skeleton';

import { CiEdit } from 'react-icons/ci';

const ReadyQuestionPaperDashboard = () => {

// const navigate = useNavigate();
const navigate = useNavigate();
const { paperId } = useParams();
const [questions, setQuestions] = useState([]);
const [loading, setLoading] = useState(true);
const [paperdetails,setpaperdetails]=useState([]);

// State for the modal
const [modalIsOpen, setModalIsOpen] = useState(false);
const [hoveredItem, setHoveredItem] = useState(null);

useEffect(() => {
const fetchQuestions = async () => {
Expand Down Expand Up @@ -52,8 +53,10 @@ const ReadyQuestionPaperDashboard = () => {
fetchpaperdetails();
fetchQuestions();
}, [paperId]);
console.log(paperdetails);

const editReadyQuestion = (question) => {
navigate(`/edit-ready-question/${question.paperId}/${question._id}`, { state: { ...question } });
}
return (
<>
<Navbar />
Expand All @@ -72,7 +75,19 @@ const ReadyQuestionPaperDashboard = () => {
<div className="question-table">
{questions.map((question) => (
<div className="questions-table" key={question._id}
onMouseEnter={() => setHoveredItem(question._id)}
onMouseLeave={() => setHoveredItem(null)}
>
{hoveredItem === question._id && (
<div className="hovered-buttons">
<button onClick={()=>editReadyQuestion(question)}>
<div className="flex-class">
<CiEdit />
<div>Edit</div>
</div>
</button>
</div>
)}
<div className="question-table-data">
<div className="compiler">
Compiler: {question.compilerReq}
Expand Down
211 changes: 211 additions & 0 deletions src/edit_ready_question/EditReadyQuestion.jsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,211 @@
import React, { useState,useEffect } from 'react';
import '../question/question.css';
import { FaTimes } from 'react-icons/fa';
import { useDropzone } from 'react-dropzone';
import { IoCloudUploadOutline } from "react-icons/io5";
import Navbar from '../Navbar/Navbar';
import axios from 'axios';
import { useParams, useLocation, useNavigate } from 'react-router-dom';
import AlertModal from '../AlertModal/AlertModal'; // Import the AlertModal component


const EditReadyQuestion = () => {
const { paperId } = useParams();
const location = useLocation();

const { remainingMarks } = location.state || {}; // Get remaining marks from props
const [questionheading, setQuestionHeading] = useState(location.state.questionheading);
const [questionDescription, setQuestionDescription] = useState(location.state.questionDescription);
const [compilerReq, setCompilerReq] = useState(location.state.compilerReq);
const [marks, setMarks] = useState(location.state.marks);
const [image, setImage] = useState(null);
const [loading, setLoading] = useState(false);
const [loadingSpinner,setLoadingSpinner] = useState(true);
const [modalIsOpen, setModalIsOpen] = useState(false);
const [modalMessage, setModalMessage] = useState('');
const [isError, setIsError] = useState(false);
const navigate = useNavigate();

useEffect(()=>{setTimeout(()=>{setLoadingSpinner(false)},1000);},[]);

const handleEditQuestion = async () => {
if (!questionheading || !questionDescription || !compilerReq || !marks) {
setModalMessage('Please fill in all the required fields.');
setIsError(true);
setModalIsOpen(true);
return;
}

if (parseInt(marks) > remainingMarks) {
setModalMessage(`You can assign a maximum of ${remainingMarks} marks to this question.`);
setIsError(true);
setModalIsOpen(true);
return;
}

setLoading(true);
try {
let imageUrl = '';

if (image) {
const formData = new FormData();
formData.append('file', image);
formData.append('upload_preset', 'question');

const uploadResponse = await axios.post('http://localhost:5000/paper/upload', formData, {
headers: { 'Content-Type': 'multipart/form-data' },
});

imageUrl = uploadResponse.data.url; // Assuming the backend responds with the image URL
}

await editQuestion(imageUrl);
} catch (error) {
console.error('Failed to add question:', error.message);
setModalMessage('Failed to add question. Please try again.');
setIsError(true);
setModalIsOpen(true);
setLoading(false);
}
};

const editQuestion = async (imageUrl) => {
const response = await axios.post('http://localhost:5000/paper/edit-ready-question', {
_id: location.state._id,
paperId,
questionheading,
questionDescription,
compilerReq,
marks,
image: imageUrl, // Include imageUrl even if it's an empty string
});

if (response.status === 200) {
setModalMessage('Question Edited successfully!');
setIsError(false);
setModalIsOpen(true);

setTimeout(() => {
navigate(`/ready_questions/${paperId}`);
}, 2000);
} else {
setModalMessage('Question Edit Failed!');
setIsError(true);
setModalIsOpen(true);
}
setLoading(false);
};

const onDrop = (acceptedFiles) => {
setImage(acceptedFiles[0]);
};

const handleRemoveImage = () => {
setImage(null);
};

const { getRootProps, getInputProps } = useDropzone({
onDrop,
maxSize: 10485760, // 10MB limit
});
console.log(paperId);
return (
<>
<Navbar />
<div className='add_question_container_main'>
{loadingSpinner ? (<></>) : (<>
<div className="add_question_container">
<h2 className="add_question_heading">Edit Ready Question</h2>

<div>
<label className="add_question_label">Question Heading:</label>
<input
type="text"
value={questionheading}
onChange={(e) => setQuestionHeading(e.target.value)}
placeholder="Enter your question heading"
required
className="add_question_input"
/>
</div>

<div>
<label className="add_question_label">Question Description:</label>
<textarea
value={questionDescription}
onChange={(e) => setQuestionDescription(e.target.value)}
placeholder="Enter question description"
required
rows="3"
className="add_question_textarea"
/>
</div>

<div className="add_question_row">
<div className="add_question_column">
<label className="add_question_label">Compiler Requirement:</label>
<select
value={compilerReq}
onChange={(e) => setCompilerReq(e.target.value)}
required
className="add_question_select"
>
<option value="" disabled>Select Compiler</option>
<option value="cpp">C++</option>
<option value="java">Java</option>
<option value="python">Python</option>
<option value="sql">SQL</option>
</select>
</div>
<div className="add_question_column">
<label className="add_question_label">Marks:</label>
<input
type="number"
value={marks}
onChange={(e) => setMarks(e.target.value)}
placeholder="Enter marks"
required
className="add_question_input"
/>
</div>
</div>

<div>
<label className="add_question_label">Upload Image (optional):</label>
<div {...getRootProps({ className: 'add_question_dropbox' })}>
<input {...getInputProps()} />
{image ? (
<div className="add_question_image_preview">
<img src={URL.createObjectURL(image)} alt="Question" className="add_question_preview_image" />
<FaTimes className="add_question_remove_image_icon" onClick={handleRemoveImage} />
</div>
) : (
<p>
<div className='add_question_cloudicon'>
<IoCloudUploadOutline />
</div>
Drag and drop an image here, or click to select one
</p>
)}
</div>
</div>

<button onClick={handleEditQuestion} className="add_question_button" disabled={loading}>
{loading ? 'Editing...' : 'Edit Question'}
</button>
</div>
</>)}
</div>

{/* Alert Modal */}
<AlertModal
isOpen={modalIsOpen}
onClose={() => setModalIsOpen(false)}
message={modalMessage}
iserror={isError}
/>
</>
);
};

export default EditReadyQuestion;

0 comments on commit 20be6e0

Please sign in to comment.