code base
This commit is contained in:
246
frontend/src/pages/ApplicationView/ApplicationView.jsx
Normal file
246
frontend/src/pages/ApplicationView/ApplicationView.jsx
Normal file
@@ -0,0 +1,246 @@
|
||||
import React, { useEffect, useState } from "react";
|
||||
import {
|
||||
useNavigate,
|
||||
useParams,
|
||||
useRouteLoaderData,
|
||||
useSubmit,
|
||||
} from "react-router-dom";
|
||||
import ValidationStatus from "./ValidationStatus";
|
||||
import Form from "../ApplicationForm/Form";
|
||||
import RejectionFeedback from "./RejectionFeedback";
|
||||
import { TbLoader3 } from "react-icons/tb";
|
||||
import AcceptChoice from "./AcceptChoice";
|
||||
import { FaCopy } from "react-icons/fa";
|
||||
|
||||
function ApplicationView() {
|
||||
const { role, user } =
|
||||
useRouteLoaderData("Applicant-Root")?.data ||
|
||||
useRouteLoaderData("Validator-Root")?.data;
|
||||
const submit = useSubmit();
|
||||
const navigate = useNavigate();
|
||||
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [applicationDisplay, setApplicationDisplay] = useState(null);
|
||||
const [rejectionFeedbackPopUp, setRejectionFeedbackPopUp] = useState(false);
|
||||
const [acceptChoicePopUp, setAcceptChoicePopUp] = useState(false);
|
||||
const [copySuccess, setCopySuccess] = useState(false);
|
||||
|
||||
const [formValues, setFormValues] = useState({});
|
||||
|
||||
const applicationId = useParams().applicationId;
|
||||
const statusParam = useParams().status;
|
||||
|
||||
const getFullApplication = async (applicationId) => {
|
||||
try {
|
||||
setLoading(true);
|
||||
const response = await fetch(
|
||||
`${
|
||||
import.meta.env.VITE_APP_API_URL
|
||||
}/general/getApplicationData/${applicationId}`,
|
||||
{
|
||||
method: "GET",
|
||||
credentials: "include",
|
||||
}
|
||||
);
|
||||
|
||||
if (!response.ok) {
|
||||
throw new Error(
|
||||
`Failed to fetch application data: ${response.status} ${response.statusText}`
|
||||
);
|
||||
}
|
||||
const fullApplication = await response.json();
|
||||
setApplicationDisplay(fullApplication?.data);
|
||||
} catch (error) {
|
||||
console.error("Error fetching application data:", error);
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
const handleSubmit = (
|
||||
applicationId,
|
||||
action,
|
||||
rejectionFeedback = "",
|
||||
toVC = false,
|
||||
resubmission = false
|
||||
) => {
|
||||
try {
|
||||
setLoading(true);
|
||||
const formData = new FormData();
|
||||
formData.append("applicationId", applicationId);
|
||||
formData.append("action", action);
|
||||
formData.append("rejectionFeedback", rejectionFeedback);
|
||||
formData.append("toVC", toVC);
|
||||
formData.append("resubmission", resubmission);
|
||||
|
||||
if (formValues && formValues?.expenses) {
|
||||
formData.append("expenses", JSON.stringify(formValues.expenses));
|
||||
}
|
||||
|
||||
submit(formData, {
|
||||
method: "PUT",
|
||||
encType: "multipart/form-data", // Specify the encoding type
|
||||
});
|
||||
} catch (error) {
|
||||
console.error("Error during submit:", error);
|
||||
} finally {
|
||||
setLoading(false);
|
||||
}
|
||||
};
|
||||
|
||||
// Navigation for status change
|
||||
let currentStatus = applicationDisplay?.currentStatus?.toLowerCase();
|
||||
|
||||
useEffect(() => {
|
||||
getFullApplication(applicationId);
|
||||
}, [applicationId]);
|
||||
|
||||
useEffect(() => {
|
||||
if (
|
||||
(statusParam !== currentStatus && currentStatus) ||
|
||||
(applicationId !== applicationDisplay?.applicationId &&
|
||||
applicationDisplay?.applicationId)
|
||||
) {
|
||||
const location = window.location.pathname;
|
||||
const newPath = location.split("/").slice(0, -2).join("/");
|
||||
navigate(
|
||||
`${newPath}/${currentStatus}/${applicationDisplay?.applicationId}`
|
||||
);
|
||||
}
|
||||
}, [statusParam, currentStatus, applicationDisplay]);
|
||||
|
||||
if (loading) {
|
||||
return (
|
||||
<div className="flex flex-col justify-center items-center h-full animate-pulse pb-[10%]">
|
||||
<TbLoader3 className="animate-spin text-xl size-24 text-red-700" />
|
||||
<p className="mt-2">Loading...</p>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
let title = applicationDisplay?.formData?.eventName;
|
||||
|
||||
if (!applicationDisplay) return null;
|
||||
|
||||
const copyToClipboard = () => {
|
||||
navigator.clipboard.writeText(applicationId)
|
||||
.then(() => {
|
||||
setCopySuccess(true);
|
||||
setTimeout(() => setCopySuccess(false), 2000);
|
||||
})
|
||||
.catch(err => {
|
||||
console.error("Failed to copy application ID: ", err);
|
||||
alert("Failed to copy application ID. Please try again.");
|
||||
});
|
||||
};
|
||||
|
||||
// Check if this is a Travel Intimation Form
|
||||
const isTravelIntimationForm = applicationDisplay?.formData?.formName === "Travel Intimation Form";
|
||||
|
||||
return (
|
||||
<div className="min-w-min bg-white shadow rounded-lg p-2 sm:p-4 md:p-6 m-4">
|
||||
<div className="flex flex-col sm:flex-row sm:justify-between sm:items-start mb-6 gap-3">
|
||||
<div>
|
||||
<h1 className="text-3xl font-extrabold text-gray-800">{title}</h1>
|
||||
</div>
|
||||
{isTravelIntimationForm && (
|
||||
<button
|
||||
onClick={copyToClipboard}
|
||||
className={`flex items-center ${copySuccess ? 'bg-green-600' : 'bg-red-700 hover:bg-red-800'} text-white font-medium py-2 px-4 rounded-lg shadow-md transition duration-300 transform ${copySuccess ? '' : 'hover:scale-110'} self-start sm:self-auto`}
|
||||
title="Copy Application ID"
|
||||
>
|
||||
{copySuccess ? (
|
||||
<>
|
||||
<span className="mr-2">✓</span>
|
||||
<span className="hidden sm:inline">Copied!</span>
|
||||
</>
|
||||
) : (
|
||||
<>
|
||||
<FaCopy className="mr-2" />
|
||||
<span className="hidden sm:inline">Copy ID</span>
|
||||
</>
|
||||
)}
|
||||
</button>
|
||||
)}
|
||||
</div>
|
||||
|
||||
<ValidationStatus
|
||||
validations={{
|
||||
facultyValidation: applicationDisplay?.facultyValidation,
|
||||
hodValidation: applicationDisplay?.hodValidation,
|
||||
hoiValidation: applicationDisplay?.hoiValidation,
|
||||
vcValidation: applicationDisplay?.vcValidation,
|
||||
accountsValidation: applicationDisplay?.accountsValidation,
|
||||
}}
|
||||
rejectionFeedback={applicationDisplay?.rejectionFeedback}
|
||||
/>
|
||||
|
||||
<Form
|
||||
prefilledData={applicationDisplay?.formData}
|
||||
applicantDesignation={applicationDisplay?.applicant?.designation}
|
||||
resubmission={applicationDisplay?.resubmission || false}
|
||||
onValuesChange={(values) => setFormValues(values)}
|
||||
/>
|
||||
|
||||
{rejectionFeedbackPopUp && (
|
||||
<RejectionFeedback
|
||||
onClose={() => setRejectionFeedbackPopUp(false)}
|
||||
onSubmit={(rejectionFeedback, resubmission) =>
|
||||
handleSubmit(
|
||||
applicationDisplay?.applicationId,
|
||||
"rejected",
|
||||
rejectionFeedback,
|
||||
false,
|
||||
resubmission
|
||||
)
|
||||
}
|
||||
/>
|
||||
)}
|
||||
|
||||
{acceptChoicePopUp && (
|
||||
<AcceptChoice
|
||||
onClose={() => setAcceptChoicePopUp(false)}
|
||||
onSubmit={(toVC) =>
|
||||
handleSubmit(applicationDisplay?.applicationId, "accepted", "", toVC, false)
|
||||
}
|
||||
designation={user.designation}
|
||||
applicantDesignation={applicationDisplay?.applicant?.designation}
|
||||
/>
|
||||
)}
|
||||
|
||||
<div className="flex justify-between items-center my-4 gap-2 mx-2">
|
||||
{role === "Validator" && currentStatus === "pending" && (
|
||||
<div className="flex space-x-2">
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => setAcceptChoicePopUp(true)}
|
||||
className="bg-green-700 text-white font-semibold text-sm sm:text-sm md:text-lg px-4 py-2 rounded-md hover:bg-green-800 focus:outline-double transition duration-200 hover:scale-110 hover:animate-spin"
|
||||
>
|
||||
Accept
|
||||
</button>
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => setRejectionFeedbackPopUp(true)}
|
||||
className="bg-red-700 text-white font-semibold text-sm sm:text-sm md:text-lg px-4 py-2 rounded-md hover:bg-red-800 focus:outline-double transition duration-200 hover:scale-110 hover:animate-spin"
|
||||
>
|
||||
Reject
|
||||
</button>
|
||||
</div>
|
||||
)}
|
||||
<button
|
||||
type="button"
|
||||
onClick={() => {
|
||||
const location = window.location.pathname;
|
||||
const newPath = location.split("/").slice(0, -1).join("/");
|
||||
navigate(newPath);
|
||||
}}
|
||||
className="bg-blue-700 text-white font-semibold text-sm sm:text-sm md:text-lg px-4 py-2 rounded-md hover:bg-blue-800 focus:outline-double transition duration-200 hover:scale-110"
|
||||
>
|
||||
Close
|
||||
</button>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
export default ApplicationView;
|
||||
Reference in New Issue
Block a user