Added Document Managment feature #388

Merged
pramod.mahajan merged 124 commits from Document_Manag into main 2025-09-10 14:34:35 +00:00
12 changed files with 347 additions and 203 deletions
Showing only changes of commit bac62e1fa7 - Show all commits

View File

@ -96,8 +96,9 @@ const DocumentFilterPanel = ({ entityTypeId, onApply }) => {
placeholder="DD-MM-YYYY To DD-MM-YYYY" placeholder="DD-MM-YYYY To DD-MM-YYYY"
startField="startDate" startField="startDate"
endField="endDate" endField="endDate"
defaultRange={false} defaultRange={true}
resetSignal={resetKey} resetSignal={resetKey}
maxDate={new Date()}
/> />
</div> </div>

View File

@ -13,6 +13,9 @@ const DocumentVersionList = ({
isPending, isPending,
setOpenDocument, setOpenDocument,
VerifyDocument, VerifyDocument,
RejectDocument,
showVersions,
latestDoc,
}) => { }) => {
const canVerifyDocument = useHasUserPermission(VERIFY_DOCUMENT); const canVerifyDocument = useHasUserPermission(VERIFY_DOCUMENT);
const canDownloadDocument = useHasUserPermission(DOWNLOAD_DOCUMENT); const canDownloadDocument = useHasUserPermission(DOWNLOAD_DOCUMENT);
@ -23,45 +26,149 @@ const DocumentVersionList = ({
} }
}; };
const sortedVersions = versionList?.data
? [...versionList.data].sort((a, b) => b.version - a.version)
: [];
if (versionLoding) { if (versionLoding) {
return <VersionListSkeleton items={2} />; return <VersionListSkeleton items={2} />;
} }
const sortedVersions = versionList?.data
? [...versionList.data].sort((a, b) => b.version - a.version)
: [];
const currentDoc = sortedVersions.length ? sortedVersions[0] : latestDoc;
if (!showVersions) {
if (!currentDoc) {
return <p className="text-muted">No documents available.</p>;
}
return (
<>
<div className="list-group list-group-flush" style={{ marginTop: "-4px", marginLeft: "10px" }}>
<div className="list-group-item list-group-item-action cursor-pointer" style={{ marginLeft: "-30px" }}>
<div className="d-flex justify-content-between">
{/* Left Side: Document Details */}
<div className="d-flex align-items-start" onClick={handleOpenDocument}>
<FileIcon
type={currentDoc.contentType}
size="fs-2"
className="me-2"
/>
<div>
<div className="d-flex align-items-center mb-1">
<span className="fw-semibold" style={{ marginTop: "-6px" }}>
{currentDoc.name}
</span>
</div>
<div className="d-flex align-items-center mb-1">
{getDocuementsStatus(currentDoc.isVerified)}
<span className="text-secondary text-tiny ms-7 ">
File Size: {currentDoc.fileSize} Kb
</span>
</div>
<div className="d-flex align-items-center">
<small className="text-secondary fw-semibold me-2">
Uploaded by
</small>
<Avatar
size="xs"
classAvatar="m-0"
firstName={currentDoc.uploadedBy?.firstName}
lastName={currentDoc.uploadedBy?.lastName}
/>
<span className="ms-1">
<small className="fw-normal" style={{ marginLeft: "-10px" }}>
{`${currentDoc.uploadedBy?.firstName ?? ""} ${currentDoc.uploadedBy?.lastName ?? ""}`.trim() || "N/A"}
</small>
</span>
</div>
<small className="ms-2" style={{ marginRight: "-113px" }}>
{formatUTCToLocalTime(currentDoc?.uploadedAt)}
</small>
</div>
</div>
<div className="d-flex flex-column align-items-end" style={{ marginTop: "-8px", marginRight: "-540px" }}>
<span className="badge bg-success-subtle text-success">
latest
</span>
</div>
{/* Right Side: Status and Info */}
<div className="d-flex flex-column align-items-end" style={{ marginTop: "31px", marginRight: "21px" }}>
{/* <span className="badge bg-success-subtle text-success">
latest
</span> */}
<div className="d-flex align-items-center mt-1">
{currentDoc?.updatedBy && (
<>
<small className="text-secondary me-2 fw-semibold">Updated by</small>
<Avatar
size="xs"
classAvatar="m-0"
firstName={currentDoc.updatedBy?.firstName}
lastName={currentDoc.updatedBy?.lastName}
/>
<span className="ms-1" style={{ marginRight: "-37px" }}>
<small className="fw-normal" style={{ marginLeft: "-10px" }}>
{`${currentDoc.updatedBy?.firstName ?? ""} ${currentDoc.updatedBy?.lastName ?? ""}`.trim() || "N/A"}
</small>
</span>
</>
)}
</div>
{/* Conditionally render updated date */}
{currentDoc?.updatedAt && (
<small className="ms-2" style={{ marginRight: "-75px" }}>
{formatUTCToLocalTime(currentDoc?.updatedAt)}
</small>
)}
</div>
</div>
</div>
</div>
{/* Buttons Div - only for the latest pending document when showVersions is false */}
{currentDoc.isVerified === null && canVerifyDocument && (
<div className="d-flex justify-content-end mb-3 me-4">
{isPending ? (
<span className="text-muted">Please Wait...</span>
) : (
<>
<button
type="button"
onClick={RejectDocument}
className="btn btn-sm btn-danger me-2"
>
Reject
</button>
<button
type="button"
onClick={VerifyDocument}
className="btn btn-sm btn-primary"
>
Verify
</button>
</>
)}
</div>
)}
</>
);
}
// If showVersions is true, display all versions
if (!sortedVersions.length) { if (!sortedVersions.length) {
return <p className="text-muted">No documents available.</p>; return <p className="text-muted">No documents available.</p>;
} }
return ( return (
<>
<div className="d-flex justify-content-between align-items-center mb-3">
<h5 className="mb-0">Documents</h5>
<div className="form-check form-switch">
<input
className="form-check-input"
type="checkbox"
id="showVersionsSwitch"
/>
<label className="form-check-label" htmlFor="showVersionsSwitch">
Show Versions
</label>
</div>
</div>
<div className="accordion" id="docAccordion">
<div className="accordion-item shadow-none">
<div className="accordion-body p-2">
<div className="list-group list-group-flush"> <div className="list-group list-group-flush">
{sortedVersions.map((document, index) => ( {sortedVersions.map((document, index) => (
<div <div
key={document.id} key={document.id}
className="list-group-item list-group-item-action" style={{ marginLeft: "-30px" }} className="list-group-item list-group-item-action cursor-pointer" style={{ marginLeft: "-30px",marginTop:"5px" }}
> >
<div className="d-flex justify-content-between"> <div className="d-flex justify-content-between">
{/* Left Side: Document Details */}
<div className="d-flex align-items-start"> <div className="d-flex align-items-start" onClick={handleOpenDocument}>
<FileIcon <FileIcon
type={document.contentType} type={document.contentType}
size="fs-2" size="fs-2"
@ -69,18 +176,18 @@ const DocumentVersionList = ({
/> />
<div> <div>
<div className="d-flex align-items-center mb-1"> <div className="d-flex align-items-center mb-1">
<span className="fw-normal"> <span className="fw-semibold" style={{ marginTop: "-6px" }}>
{document.name} {document.name}
</span> </span>
</div> </div>
<div className="d-flex align-items-center mb-1"> <div className="d-flex align-items-center mb-1">
{getDocuementsStatus(document.isVerified)} {getDocuementsStatus(document.isVerified)}
<span className="text-secondary ms-7"> <span className="ms-7 ">
File Size: {document.fileSize} Kb File Size: {document.fileSize} Kb
</span> </span>
</div> </div>
<div className="d-flex align-items-center"> <div className="d-flex align-items-center">
<small className="text-secondary me-2"> <small className="text-secondary fw-semibold me-2">
Uploaded by Uploaded by
</small> </small>
<Avatar <Avatar
@ -90,17 +197,17 @@ const DocumentVersionList = ({
lastName={document.uploadedBy?.lastName} lastName={document.uploadedBy?.lastName}
/> />
<span className="ms-1"> <span className="ms-1">
<small className="fw-normal"> <small className="fw-normal" style={{ marginLeft: "-10px" }}>
{`${document.uploadedBy?.firstName ?? ""} ${document.uploadedBy?.lastName ?? ""}`.trim() || "N/A"} {`${document.uploadedBy?.firstName ?? ""} ${document.uploadedBy?.lastName ?? ""}`.trim() || "N/A"}
</small> </small>
</span> </span>
<small className="ms-2"> </div>
<small className="ms-2" style={{ marginRight: "-113px" }}>
{formatUTCToLocalTime(document?.uploadedAt)} {formatUTCToLocalTime(document?.uploadedAt)}
</small> </small>
</div>
{document?.verifiedAt && ( {document?.verifiedAt && (
<div className="d-flex align-items-center mt-1"> <div className="d-flex align-items-center mt-1">
<small className="text-secondary me-2"> <small className="text-secondary me-2 fw-semibold">
{document.isVerified ? "Approved by" : "Rejected by"} {document.isVerified ? "Approved by" : "Rejected by"}
</small> </small>
<Avatar <Avatar
@ -110,59 +217,70 @@ const DocumentVersionList = ({
lastName={document.verifiedBy?.lastName} lastName={document.verifiedBy?.lastName}
/> />
<span className="ms-1"> <span className="ms-1">
<small className="fw-normal"> <small className="fw-normal" style={{ marginLeft: "-10px" }}>
{`${document.verifiedBy?.firstName ?? ""} ${document.verifiedBy?.lastName ?? ""}`.trim() || "N/A"} {`${document.verifiedBy?.firstName ?? ""} ${document.verifiedBy?.lastName ?? ""}`.trim() || "N/A"}
</small> </small>
</span> </span>
<small className="ms-2"> </div>
)}
{/* Conditionally render verified date */}
{document?.verifiedAt && (
<small className="ms-2" style={{ marginRight: "-103px" }}>
{formatUTCToLocalTime(document?.verifiedAt)} {formatUTCToLocalTime(document?.verifiedAt)}
</small> </small>
</div>
)} )}
</div> </div>
</div> </div>
<div className="d-flex flex-column align-items-end" style={{ marginTop: "-13px", marginRight: "46px" }}>
{/* Right Side: Status and Actions */} <small className="badge rounded-pill bg-label-secondary mt-1" style={{ marginRight: "-336px" }}>
<div className="d-flex flex-column align-items-end" style={{ marginTop: "42px", marginRight: "-52px" }}> version {document.version}
</small>
{document.isLatest && (
<span className="badge bg-success-subtle text-success">
latest
</span>
)}
</div>
{/* Right Side: Status and Actions */}
<div className="d-flex flex-column align-items-end" style={{ marginTop: "16px", marginRight: "46px" }}>
{/* <small className="badge rounded-pill bg-label-secondary mt-1" style={{ marginRight: "-105px" }}>
version {document.version}
</small> */}
{document.isLatest && ( {document.isLatest && (
<span className="badge bg-success-subtle text-success"> <span className="badge bg-success-subtle text-success">
latest latest
</span> </span>
)} )}
<div className="d-flex align-items-center mt-1"> <div className="d-flex align-items-center mt-1">
{document?.updatedBy && ( {document?.updatedBy && (
<> <>
<small className="text-secondary me-2">Updated by</small> <small className="text-secondary me-2 fw-semibold">Updated by</small>
<Avatar <Avatar
size="xs" size="xs"
classAvatar="m-0" classAvatar="m-0"
firstName={document.updatedBy?.firstName} firstName={document.updatedBy?.firstName}
lastName={document.updatedBy?.lastName} lastName={document.updatedBy?.lastName}
/> />
<span className="ms-1"> <span className="ms-1" style={{ marginRight: "-37px" }}>
<small className="fw-normal"> <small className="fw-normal" style={{ marginLeft: "-10px" }}>
{`${document.updatedBy?.firstName ?? ""} ${document.updatedBy?.lastName ?? ""}`.trim() || "N/A"} {`${document.updatedBy?.firstName ?? ""} ${document.updatedBy?.lastName ?? ""}`.trim() || "N/A"}
</small> </small>
</span> </span>
<small className="ms-2">
{formatUTCToLocalTime(document?.updatedAt)}
</small>
</> </>
)} )}
</div> </div>
<small className="text-secondary mt-1"> {/* Conditionally render updated date */}
version {document.version} {document?.updatedAt && (
<small className="ms-2" style={{ marginRight: "-75px" }}>
{formatUTCToLocalTime(document?.updatedAt)}
</small> </small>
)}
</div> </div>
</div> </div>
</div> </div>
))} ))}
</div> </div>
</div>
</div>
</div>
</>
); );
}; };

View File

@ -16,6 +16,7 @@ import ManageDocument from "./ManageDocument";
import ViewDocument from "./ViewDocument"; import ViewDocument from "./ViewDocument";
import DocumentViewerModal from "./DocumentViewerModal"; import DocumentViewerModal from "./DocumentViewerModal";
import { useHasUserPermission } from "../../hooks/useHasUserPermission"; import { useHasUserPermission } from "../../hooks/useHasUserPermission";
import { useProfile } from "../../hooks/useProfile";
// Context // Context
export const DocumentContext = createContext(); export const DocumentContext = createContext();
@ -47,6 +48,7 @@ export const getDocuementsStatus = (status) => {
} }
}; };
const Documents = ({ Document_Entity, Entity }) => { const Documents = ({ Document_Entity, Entity }) => {
const [isSelf, setIsSelf] = useState(false);
const [searchText, setSearchText] = useState(""); const [searchText, setSearchText] = useState("");
const [isActive, setIsActive] = useState(true); const [isActive, setIsActive] = useState(true);
const [filters, setFilter] = useState(); const [filters, setFilter] = useState();
@ -63,8 +65,14 @@ const Documents = ({ Document_Entity, Entity }) => {
document: null, document: null,
isOpen: false, isOpen: false,
}); });
const { profile } = useProfile();
const canUploadDocument = useHasUserPermission(UPLOAD_DOCUMENT) useEffect(() => {
if (profile?.employeeInfo?.id) {
setIsSelf(profile.employeeInfo.id === employeeId);
}
}, [profile?.employeeInfo?.id, employeeId]);
const canUploadDocument = useHasUserPermission(UPLOAD_DOCUMENT);
const { setOffcanvasContent, setShowTrigger } = useFab(); const { setOffcanvasContent, setShowTrigger } = useFab();
@ -114,13 +122,16 @@ const Documents = ({ Document_Entity, Entity }) => {
<div className="row align-items-center"> <div className="row align-items-center">
{/* Search */} {/* Search */}
<div className="d-flex col-8 col-md-8 col-lg-4 mb-md-0 align-items-center"> <div className="d-flex col-8 col-md-8 col-lg-4 mb-md-0 align-items-center">
<div className="d-flex"> <input <div className="d-flex">
{" "}
<input
type="search" type="search"
value={searchText} value={searchText}
onChange={(e) => setSearchText(e.target.value)} onChange={(e) => setSearchText(e.target.value)}
className="form-control form-control-sm" className="form-control form-control-sm"
placeholder="Search Document" placeholder="Search Document"
/></div> />
</div>
<label className="switch switch-sm mx-2"> <label className="switch switch-sm mx-2">
<input <input
type="checkbox" type="checkbox"
@ -138,7 +149,6 @@ const Documents = ({ Document_Entity, Entity }) => {
</label> </label>
</div> </div>
{/* Actions */} {/* Actions */}
<div className="col-6 col-md-6 col-lg-8 text-end"> <div className="col-6 col-md-6 col-lg-8 text-end">
{/* <span {/* <span
@ -158,8 +168,8 @@ const Documents = ({ Document_Entity, Entity }) => {
></i> ></i>
</span> */} </span> */}
{(isSelf || canUploadDocument) && (
{canUploadDocument && (<button <button
type="button" type="button"
title="Add New Document" title="Add New Document"
className="p-1 bg-primary rounded-circle cursor-pointer" className="p-1 bg-primary rounded-circle cursor-pointer"
@ -171,7 +181,8 @@ const Documents = ({ Document_Entity, Entity }) => {
} }
> >
<i className="bx bx-plus fs-4 text-white"></i> <i className="bx bx-plus fs-4 text-white"></i>
</button>)} </button>
)}
</div> </div>
</div> </div>
<DocumentsList <DocumentsList

View File

@ -17,6 +17,9 @@ import Pagination from "../common/Pagination";
import ConfirmModal from "../common/ConfirmModal"; import ConfirmModal from "../common/ConfirmModal";
import { isPending } from "@reduxjs/toolkit"; import { isPending } from "@reduxjs/toolkit";
import { useHasUserPermission } from "../../hooks/useHasUserPermission"; import { useHasUserPermission } from "../../hooks/useHasUserPermission";
import { useProfile } from "../../hooks/useProfile";
import { useParams } from "react-router-dom";
import showToast from "../../services/toastService";
const DocumentsList = ({ const DocumentsList = ({
Document_Entity, Document_Entity,
@ -27,13 +30,22 @@ const DocumentsList = ({
setRefetchFn, setRefetchFn,
isActive, isActive,
}) => { }) => {
const { employeeId } = useParams();
const [isSelf, setIsSelf] = useState(false);
const { profile } = useProfile();
const canDeleteDocument = useHasUserPermission(DELETE_DOCUMENT);
const canModifyDocument = useHasUserPermission(MODIFY_DOCUMENT);
useEffect(() => {
if (profile?.employeeInfo?.id && employeeId) {
setIsSelf(String(profile.employeeInfo.id) === String(employeeId));
}
}, [profile?.employeeInfo?.id, employeeId]);
const [IsDeleteModalOpen, setIsDeleteModalOpen] = useState(false); const [IsDeleteModalOpen, setIsDeleteModalOpen] = useState(false);
const [deletingId, setDeletingId] = useState(null); const [deletingId, setDeletingId] = useState(null);
const [restoringIds, setRestoringIds] = useState([]); const [restoringIds, setRestoringIds] = useState([]);
const debouncedSearch = useDebounce(searchText, 500); const debouncedSearch = useDebounce(searchText, 500);
const [currentPage, setCurrentPage] = useState(1); const [currentPage, setCurrentPage] = useState(1);
const canDeleteDocument = useHasUserPermission(DELETE_DOCUMENT);
const canModifyDocument = useHasUserPermission(MODIFY_DOCUMENT);
const { data, isError, isLoading, error, refetch, isFetching } = const { data, isError, isLoading, error, refetch, isFetching } =
useDocumentListByEntityId( useDocumentListByEntityId(
Document_Entity, Document_Entity,
@ -205,7 +217,7 @@ const DocumentsList = ({
} }
></i> ></i>
{canModifyDocument && ( {(isSelf || canModifyDocument) && (
<i <i
className="bx bx-edit text-secondary cursor-pointer" className="bx bx-edit text-secondary cursor-pointer"
onClick={() => onClick={() =>
@ -214,7 +226,7 @@ const DocumentsList = ({
></i> ></i>
)} )}
{canDeleteDocument && ( {(isSelf || canDeleteDocument) && (
<i <i
className="bx bx-trash text-danger cursor-pointer" className="bx bx-trash text-danger cursor-pointer"
onClick={() => { onClick={() => {

View File

@ -12,9 +12,6 @@ import {
ITEMS_PER_PAGE, ITEMS_PER_PAGE,
VERIFY_DOCUMENT, VERIFY_DOCUMENT,
} from "../../utils/constants"; } from "../../utils/constants";
import Pagination from "../common/Pagination";
import VersionListSkeleton from "./VersionListSkeleton";
// import DocumentDetailsSkeleton from "./DocumentDetailsSkeleton";
import DocumentDetailsSkeleton from "./DocumentDetailsSkeleton "; import DocumentDetailsSkeleton from "./DocumentDetailsSkeleton ";
import { useHasUserPermission } from "../../hooks/useHasUserPermission"; import { useHasUserPermission } from "../../hooks/useHasUserPermission";
import DocumentVersionList from "./DocumentVersionList"; import DocumentVersionList from "./DocumentVersionList";
@ -22,15 +19,20 @@ import DocumentVersionList from "./DocumentVersionList";
const ViewDocument = () => { const ViewDocument = () => {
const { viewDoc, setOpenDocument } = useDocumentContext(); const { viewDoc, setOpenDocument } = useDocumentContext();
const [currentPage, setCurrentPage] = useState(1); const [currentPage, setCurrentPage] = useState(1);
const [showVersions, setShowVersions] = useState(false);
const canVerifyDocument = useHasUserPermission(VERIFY_DOCUMENT); const canVerifyDocument = useHasUserPermission(VERIFY_DOCUMENT);
// Document Details
const { data, isLoading, isError, error } = useDocumentDetails( const { data, isLoading, isError, error } = useDocumentDetails(
viewDoc?.document viewDoc?.document
); );
// Document Versions (fetch only if toggle is ON)
const { const {
data: versionList, data: versionList,
isLoading: versionLoding, isLoading: versionLoading,
} = useDocumentVersionList( } = useDocumentVersionList(
data?.parentAttachmentId, showVersions ? data?.parentAttachmentId : null,
ITEMS_PER_PAGE - 10, ITEMS_PER_PAGE - 10,
currentPage currentPage
); );
@ -41,6 +43,7 @@ const ViewDocument = () => {
} }
}; };
// Verify / Reject
const { mutate: VerifyDoc, isPending } = useVerifyDocument(); const { mutate: VerifyDoc, isPending } = useVerifyDocument();
const VerifyDocument = () => { const VerifyDocument = () => {
VerifyDoc({ documentId: viewDoc?.document, isVerify: true }); VerifyDoc({ documentId: viewDoc?.document, isVerify: true });
@ -57,11 +60,12 @@ const ViewDocument = () => {
<p className="danger-text">{error?.response?.status}</p> <p className="danger-text">{error?.response?.status}</p>
</div> </div>
); );
console.log("kartik", data)
return ( return (
<div className="p-3"> <div className="p-3">
<p className="fw-bold fs-5 mb-3 border-bottom pb-2">Document Details</p> <p className="fw-bold fs-5 mb-3 border-bottom pb-2">Document Details</p>
{/* Document Info Rows */}
<div className="row mb-2 text-start"> <div className="row mb-2 text-start">
<div className="col-12 col-md-6 d-flex"> <div className="col-12 col-md-6 d-flex">
<span className="fw-semibold me-2" style={{ minWidth: "140px" }}> <span className="fw-semibold me-2" style={{ minWidth: "140px" }}>
@ -103,14 +107,7 @@ const ViewDocument = () => {
{formatUTCToLocalTime(data.uploadedAt)} {formatUTCToLocalTime(data.uploadedAt)}
</span> </span>
</div> </div>
<div className="col-12 col-md-6 d-flex">
<span className="fw-semibold me-2" style={{ minWidth: "140px" }}>
Updated At:
</span>
<span className="text-muted">
{formatUTCToLocalTime(data.updatedAt) || "-"}
</span>
</div>
</div> </div>
<div className="row mb-2 text-start"> <div className="row mb-2 text-start">
@ -144,39 +141,32 @@ const ViewDocument = () => {
</div> </div>
</div> </div>
{/* Verify / Reject */} {/* Toggle for Versions */}
{data.isVerified === null && canVerifyDocument && ( <div className="d-flex justify-content-between align-items-center mb-3">
<div className="d-flex justify-content-end mb-3"> <p className="m-0 fw-semibold fs-6">Documents:</p>
{isPending ? ( <div className="form-check form-switch">
<span className="text-muted">Please Wait...</span> <label className="form-check-label" htmlFor="showVersionsSwitch">
) : ( Show Versions
<> </label>
<button <input
type="button" className="form-check-input"
onClick={RejectDocument} type="checkbox"
className="btn btn-sm btn-danger me-2" id="showVersionsSwitch"
> checked={showVersions}
Reject onChange={(e) => setShowVersions(e.target.checked)}
</button> />
<button </div>
type="button"
onClick={VerifyDocument}
className="btn btn-sm btn-primary"
>
Verify
</button>
</>
)}
</div> </div>
)}
<DocumentVersionList <DocumentVersionList
versionLoding={versionLoding} versionLoding={versionLoading}
versionList={versionList} versionList={versionList}
isPending={isPending} isPending={isPending}
setOpenDocument={setOpenDocument} setOpenDocument={setOpenDocument}
VerifyDocument={VerifyDocument} VerifyDocument={VerifyDocument}
RejectDocument={RejectDocument}
showVersions={showVersions}
latestDoc={data}
/> />
</div> </div>
); );

View File

@ -3,6 +3,7 @@ import React, { useState, useEffect } from "react";
import { useChangePassword } from "../../components/Context/ChangePasswordContext"; import { useChangePassword } from "../../components/Context/ChangePasswordContext";
import GlobalModel from "../common/GlobalModel"; import GlobalModel from "../common/GlobalModel";
import ManageEmployee from "./ManageEmployee"; import ManageEmployee from "./ManageEmployee";
import { formatUTCToLocalTime } from "../../utils/dateUtils";
const EmpBanner = ({ profile, loggedInUser }) => { const EmpBanner = ({ profile, loggedInUser }) => {
const { openChangePassword } = useChangePassword(); const { openChangePassword } = useChangePassword();
@ -77,7 +78,7 @@ const EmpBanner = ({ profile, loggedInUser }) => {
{" "} {" "}
Joined on{" "} Joined on{" "}
{profile?.joiningDate ? ( {profile?.joiningDate ? (
new Date(profile.joiningDate).toLocaleDateString() formatUTCToLocalTime(profile.joiningDate)
) : ( ) : (
<em>NA</em> <em>NA</em>
)} )}

View File

@ -1,19 +1,31 @@
import React from "react"; import React, { useEffect, useState } from "react";
import { useHasUserPermission } from "../../hooks/useHasUserPermission"; import { useHasUserPermission } from "../../hooks/useHasUserPermission";
import { VIEW_DOCUMENT } from "../../utils/constants"; import { VIEW_DOCUMENT } from "../../utils/constants";
import { useProfile } from "../../hooks/useProfile";
import { useParams } from "react-router-dom";
const EmployeeNav = ({ onPillClick, activePill }) => { const EmployeeNav = ({ onPillClick, activePill }) => {
const canViewDocuments = useHasUserPermission(VIEW_DOCUMENT) const { employeeId } = useParams();
const [isAbleToViewDocuments, setIsAbleToViewDocuments] = useState(false);
const canViewDocuments = useHasUserPermission(VIEW_DOCUMENT);
const { profile } = useProfile();
useEffect(() => {
if (profile?.employeeInfo?.id) {
setIsAbleToViewDocuments(profile.employeeInfo.id === employeeId);
}
}, [profile?.employeeInfo?.id, employeeId]);
const tabs = [ const tabs = [
{ key: "profile", icon: "bx bx-user", label: "Profile" }, { key: "profile", icon: "bx bx-user", label: "Profile" },
{ key: "attendance", icon: "bx bx-group", label: "Attendances" }, { key: "attendance", icon: "bx bx-group", label: "Attendances" },
canViewDocuments && { (isAbleToViewDocuments || canViewDocuments) && {
key: "documents", key: "documents",
icon: "bx bx-file", icon: "bx bx-file",
label: "Documents", label: "Documents",
}, },
{ key: "activities", icon: "bx bx-grid-alt", label: "Activities" }, { key: "activities", icon: "bx bx-grid-alt", label: "Activities" },
].filter(Boolean); ].filter(Boolean);
return ( return (
<div className="col-md-12"> <div className="col-md-12">
<div className="nav-align-top"> <div className="nav-align-top">

View File

@ -105,7 +105,7 @@ const ProjectPermission = () => {
}; };
return ( return (
<div className="row"> <div className="row px-2 py-1">
<form className="row" onSubmit={handleSubmit(onSubmit)}> <form className="row" onSubmit={handleSubmit(onSubmit)}>
{/* Employee Dropdown */} {/* Employee Dropdown */}
<div className="d-flex align-items-end gap-2"> <div className="d-flex align-items-end gap-2">
@ -142,7 +142,7 @@ const ProjectPermission = () => {
{/* Permissions */} {/* Permissions */}
{ProjectModules.map((feature) => ( {ProjectModules.map((feature) => (
<div key={feature.id} className="row my-2"> <div key={feature.id} className="row my-2 px-3 ">
<div className="col-12 text-start fw-semibold mb-2"> <div className="col-12 text-start fw-semibold mb-2">
{feature.name} {feature.name}
</div> </div>

View File

@ -31,7 +31,7 @@ const ProjectSetting = () => {
return ( return (
<div className="w-100"> <div className="w-100">
<div className="card p-3"> <div className="card py-2 px-5">
<div className="col-4"> <div className="col-4">
<div className="dropdown text-start"> <div className="dropdown text-start">
<button <button

View File

@ -84,6 +84,7 @@ export const DateRangePicker1 = ({
allowText = false, allowText = false,
resetSignal, resetSignal,
defaultRange = true, defaultRange = true,
maxDate = null,
...rest ...rest
}) => { }) => {
const inputRef = useRef(null); const inputRef = useRef(null);
@ -117,6 +118,7 @@ export const DateRangePicker1 = ({
mode: "range", mode: "range",
dateFormat: "d-m-Y", dateFormat: "d-m-Y",
allowInput: allowText, allowInput: allowText,
maxDate ,
onChange: (selectedDates) => { onChange: (selectedDates) => {
if (selectedDates.length === 2) { if (selectedDates.length === 2) {
const [start, end] = selectedDates; const [start, end] = selectedDates;

View File

@ -110,7 +110,7 @@ const SelectMultiple = ({
onChange={() => handleCheckboxChange(valueVal)} onChange={() => handleCheckboxChange(valueVal)}
style={{ marginRight: 8 }} style={{ marginRight: 8 }}
/> />
<label className="text-secondary">{labelVal}</label> <label className="fw-semibold">{labelVal}</label>
</div> </div>
); );
})} })}

View File

@ -185,12 +185,9 @@ export const useActiveInActiveDocument = ()=>{
return useMutation({ return useMutation({
mutationFn:async({documentId,isActive}) => await DocumentRepository.deleteDocument(documentId,isActive), mutationFn:async({documentId,isActive}) => await DocumentRepository.deleteDocument(documentId,isActive),
onSuccess: (data, variables) => { onSuccess: (data, variables) => {
const {isActive} = variables;
queryClient.invalidateQueries({ queryKey: ["DocumentList"] }); queryClient.invalidateQueries({ queryKey: ["DocumentList"] });
showToast( showToast(`Document ${isActive ? "restored":"Deleted"} successfully`,"success")
data.response.data.message ||
"Document Successfully Verified !",
"success"
);
}, },
onError: (error) => { onError: (error) => {
showToast( showToast(