Remove warnings
This commit is contained in:
parent
83b6c2a307
commit
1418f84976
@ -1,4 +1,10 @@
|
|||||||
import React, { createContext, useContext, useState, useEffect, useRef } from "react";
|
import React, {
|
||||||
|
createContext,
|
||||||
|
useContext,
|
||||||
|
useState,
|
||||||
|
useEffect,
|
||||||
|
useRef,
|
||||||
|
} from "react";
|
||||||
|
|
||||||
const ModalContext = createContext();
|
const ModalContext = createContext();
|
||||||
|
|
||||||
@ -9,12 +15,12 @@ export const ModalProvider = ({ children }) => {
|
|||||||
const [isOpen, setIsOpen] = useState(false);
|
const [isOpen, setIsOpen] = useState(false);
|
||||||
const [modalContent, setModalContent] = useState(null);
|
const [modalContent, setModalContent] = useState(null);
|
||||||
const [onSubmit, setOnSubmit] = useState(null);
|
const [onSubmit, setOnSubmit] = useState(null);
|
||||||
const [modalSize, setModalSize] = useState('lg');
|
const [modalSize, setModalSize] = useState("lg");
|
||||||
|
|
||||||
// Ref to track the modal content element
|
// Ref to track the modal content element
|
||||||
const modalRef = useRef(null);
|
const modalRef = useRef(null);
|
||||||
|
|
||||||
const openModal = (content, onSubmitCallback, size = 'lg') => {
|
const openModal = (content, onSubmitCallback, size = "lg") => {
|
||||||
setModalContent(content); // Set modal content dynamically
|
setModalContent(content); // Set modal content dynamically
|
||||||
setOnSubmit(() => onSubmitCallback); // Set the submit handler dynamically
|
setOnSubmit(() => onSubmitCallback); // Set the submit handler dynamically
|
||||||
setIsOpen(true); // Open the modal
|
setIsOpen(true); // Open the modal
|
||||||
@ -26,9 +32,20 @@ export const ModalProvider = ({ children }) => {
|
|||||||
setIsOpen(false); // Close the modal
|
setIsOpen(false); // Close the modal
|
||||||
setModalContent(null); // Clear modal content
|
setModalContent(null); // Clear modal content
|
||||||
setOnSubmit(null); // Clear the submit callback
|
setOnSubmit(null); // Clear the submit callback
|
||||||
setModalSize('lg'); // Reset modal size
|
setModalSize("lg"); // Reset modal size
|
||||||
};
|
};
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleEscape = (event) => {
|
||||||
|
if (event.key === "Escape") {
|
||||||
|
closeModal();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
document.addEventListener("keydown", handleEscape);
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener("keydown", handleEscape);
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const handleClickOutside = (event) => {
|
const handleClickOutside = (event) => {
|
||||||
@ -43,14 +60,28 @@ export const ModalProvider = ({ children }) => {
|
|||||||
}, []);
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ModalContext.Provider value={{ isOpen, openModal, closeModal, modalContent, modalSize, onSubmit }}>
|
<ModalContext.Provider
|
||||||
|
value={{
|
||||||
|
isOpen,
|
||||||
|
openModal,
|
||||||
|
closeModal,
|
||||||
|
modalContent,
|
||||||
|
modalSize,
|
||||||
|
onSubmit,
|
||||||
|
}}
|
||||||
|
>
|
||||||
{children}
|
{children}
|
||||||
|
|
||||||
{isOpen && (
|
{isOpen && (
|
||||||
<div style={overlayStyles}>
|
<div style={overlayStyles}>
|
||||||
<div ref={modalRef} style={{ ...modalStyles, maxWidth: modalSize === 'sm' ? '400px' : '800px' }}>
|
<div
|
||||||
|
ref={modalRef}
|
||||||
|
style={{
|
||||||
|
...modalStyles,
|
||||||
|
maxWidth: modalSize === "sm" ? "400px" : "800px",
|
||||||
|
}}
|
||||||
|
>
|
||||||
<div>{modalContent}</div>
|
<div>{modalContent}</div>
|
||||||
|
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
|
@ -1,18 +1,18 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from "react";
|
||||||
import Avatar from '../common/Avatar';
|
import Avatar from "../common/Avatar";
|
||||||
import { convertShortTime } from '../../utils/dateUtils';
|
import { convertShortTime } from "../../utils/dateUtils";
|
||||||
import RenderAttendanceStatus from './RenderAttendanceStatus';
|
import RenderAttendanceStatus from "./RenderAttendanceStatus";
|
||||||
import { useSelector, useDispatch } from 'react-redux';
|
import { useSelector, useDispatch } from "react-redux";
|
||||||
import { fetchAttendanceData } from '../../slices/apiSlice/attedanceLogsSlice';
|
import { fetchAttendanceData } from "../../slices/apiSlice/attedanceLogsSlice";
|
||||||
|
|
||||||
const AttendanceLog = ({ attendance, handleModalData, projectId }) => {
|
const AttendanceLog = ({ attendance, handleModalData, projectId }) => {
|
||||||
const [attendances, setAttendnaces] = useState([]);
|
const [attendances, setAttendnaces] = useState([]);
|
||||||
const [selectedDate, setSelectedDate] = useState('');
|
const [selectedDate, setSelectedDate] = useState("");
|
||||||
const dispatch = useDispatch();
|
const dispatch = useDispatch();
|
||||||
const { data, loading, error } = useSelector((store) => store.attendanceLogs);
|
const { data, loading, error } = useSelector((store) => store.attendanceLogs);
|
||||||
|
|
||||||
// Set the default selected date to the current date
|
// Set the default selected date to the current date
|
||||||
const currentDate = new Date().toISOString().split('T')[0]; // "YYYY-MM-DD"
|
const currentDate = new Date().toISOString().split("T")[0]; // "YYYY-MM-DD"
|
||||||
|
|
||||||
const handleDateChange = (e) => {
|
const handleDateChange = (e) => {
|
||||||
const date = e.target.value;
|
const date = e.target.value;
|
||||||
@ -28,11 +28,15 @@ const AttendanceLog = ({ attendance, handleModalData, projectId }) => {
|
|||||||
setSelectedDate(currentDate); // Set default selected date to today
|
setSelectedDate(currentDate); // Set default selected date to today
|
||||||
}, [attendance]);
|
}, [attendance]);
|
||||||
|
|
||||||
const renderAttendanceData = selectedDate === currentDate ? attendances : data;
|
const renderAttendanceData =
|
||||||
|
selectedDate === currentDate ? attendances : data;
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
<div className="dataTables_length text-start py-2" id="DataTables_Table_0_length">
|
<div
|
||||||
|
className="dataTables_length text-start py-2"
|
||||||
|
id="DataTables_Table_0_length"
|
||||||
|
>
|
||||||
<div className="col-md-3">
|
<div className="col-md-3">
|
||||||
<input
|
<input
|
||||||
className="form-control form-control-sm"
|
className="form-control form-control-sm"
|
||||||
@ -50,10 +54,13 @@ const AttendanceLog = ({ attendance, handleModalData, projectId }) => {
|
|||||||
<table className="table mb-0">
|
<table className="table mb-0">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<th className="border-top-1" colSpan={2}>Name</th>
|
<th className="border-top-1" colSpan={2}>
|
||||||
|
Name
|
||||||
|
</th>
|
||||||
<th className="border-top-1">Role</th>
|
<th className="border-top-1">Role</th>
|
||||||
<th>
|
<th>
|
||||||
<i className="bx bxs-down-arrow-alt text-success"></i> Check-In
|
<i className="bx bxs-down-arrow-alt text-success"></i>{" "}
|
||||||
|
Check-In
|
||||||
</th>
|
</th>
|
||||||
<th>
|
<th>
|
||||||
<i className="bx bxs-up-arrow-alt text-danger"></i> Check-Out
|
<i className="bx bxs-up-arrow-alt text-danger"></i> Check-Out
|
||||||
@ -65,14 +72,19 @@ const AttendanceLog = ({ attendance, handleModalData, projectId }) => {
|
|||||||
{loading && <td colSpan={5}>Loading...</td>}
|
{loading && <td colSpan={5}>Loading...</td>}
|
||||||
{error && <td colSpan={5}>{error}</td>}
|
{error && <td colSpan={5}>{error}</td>}
|
||||||
{selectedDate && renderAttendanceData.length === 0 && (
|
{selectedDate && renderAttendanceData.length === 0 && (
|
||||||
|
<tr>
|
||||||
<td colSpan={5}>No Data Found</td>
|
<td colSpan={5}>No Data Found</td>
|
||||||
|
</tr>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{renderAttendanceData?.map((attendance, index) => (
|
{renderAttendanceData?.map((attendance, index) => (
|
||||||
<tr key={index}>
|
<tr key={index}>
|
||||||
<td colSpan={2}>
|
<td colSpan={2}>
|
||||||
<div className="d-flex justify-content-start align-items-center">
|
<div className="d-flex justify-content-start align-items-center">
|
||||||
<Avatar firstName={attendance.firstName} lastName={attendance.lastName} />
|
<Avatar
|
||||||
|
firstName={attendance.firstName}
|
||||||
|
lastName={attendance.lastName}
|
||||||
|
/>
|
||||||
<div className="d-flex flex-column">
|
<div className="d-flex flex-column">
|
||||||
<a href="#" className="text-heading text-truncate">
|
<a href="#" className="text-heading text-truncate">
|
||||||
<span className="fw-medium">
|
<span className="fw-medium">
|
||||||
@ -84,7 +96,11 @@ const AttendanceLog = ({ attendance, handleModalData, projectId }) => {
|
|||||||
</td>
|
</td>
|
||||||
<td>{attendance.jobRoleName}</td>
|
<td>{attendance.jobRoleName}</td>
|
||||||
<td>{convertShortTime(attendance.checkInTime)}</td>
|
<td>{convertShortTime(attendance.checkInTime)}</td>
|
||||||
<td>{attendance.checkOutTime ? convertShortTime(attendance.checkOutTime) : '--'}</td>
|
<td>
|
||||||
|
{attendance.checkOutTime
|
||||||
|
? convertShortTime(attendance.checkOutTime)
|
||||||
|
: "--"}
|
||||||
|
</td>
|
||||||
<td className="text-center">
|
<td className="text-center">
|
||||||
<RenderAttendanceStatus
|
<RenderAttendanceStatus
|
||||||
attendanceData={attendance}
|
attendanceData={attendance}
|
||||||
|
@ -1,23 +1,20 @@
|
|||||||
import React, { useEffect, useState } from 'react';
|
import React, { useEffect, useState } from "react";
|
||||||
import Avatar from '../common/Avatar';
|
import Avatar from "../common/Avatar";
|
||||||
import { convertShortTime } from '../../utils/dateUtils';
|
import { convertShortTime } from "../../utils/dateUtils";
|
||||||
import RegularizationActions from './RegularizationActions';
|
import RegularizationActions from "./RegularizationActions";
|
||||||
import {useSelector} from 'react-redux';
|
import { useSelector } from "react-redux";
|
||||||
import {useRegularizationRequests} from '../../hooks/useAttendance';
|
import { useRegularizationRequests } from "../../hooks/useAttendance";
|
||||||
import moment from 'moment';
|
import moment from "moment";
|
||||||
|
|
||||||
|
const Regularization = ({ handleRequest }) => {
|
||||||
const Regularization = ( { handleRequest} ) =>
|
|
||||||
{
|
|
||||||
var selectedProject = useSelector((store) => store.localVariables.projectId);
|
var selectedProject = useSelector((store) => store.localVariables.projectId);
|
||||||
const [ regularizesList, setregularizedList ] = useState( [] )
|
const [regularizesList, setregularizedList] = useState([]);
|
||||||
const { regularizes, loading,error,refetch} = useRegularizationRequests(selectedProject)
|
const { regularizes, loading, error, refetch } =
|
||||||
|
useRegularizationRequests(selectedProject);
|
||||||
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
setregularizedList(regularizes)
|
setregularizedList(regularizes);
|
||||||
},[regularizes])
|
}, [regularizes]);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="table-responsive text-nowrap">
|
<div className="table-responsive text-nowrap">
|
||||||
@ -26,16 +23,18 @@ const Regularization = ( { handleRequest} ) =>
|
|||||||
<tr>
|
<tr>
|
||||||
<th colSpan={2}>Name</th>
|
<th colSpan={2}>Name</th>
|
||||||
<th>Date</th>
|
<th>Date</th>
|
||||||
<th><i className='bx bxs-down-arrow-alt text-success' ></i>Check-In</th>
|
<th>
|
||||||
<th><i className='bx bxs-up-arrow-alt text-danger' ></i>Check-Out</th>
|
<i className="bx bxs-down-arrow-alt text-success"></i>Check-In
|
||||||
|
</th>
|
||||||
|
<th>
|
||||||
|
<i className="bx bxs-up-arrow-alt text-danger"></i>Check-Out
|
||||||
|
</th>
|
||||||
<th>Action</th>
|
<th>Action</th>
|
||||||
</tr>
|
</tr>
|
||||||
</thead>
|
</thead>
|
||||||
<tbody>
|
<tbody>
|
||||||
|
|
||||||
{loading && <td colSpan={5}>Loading...</td>}
|
{loading && <td colSpan={5}>Loading...</td>}
|
||||||
{
|
{regularizes?.length > 0 ? (
|
||||||
regularizes?.length > 0 ? (
|
|
||||||
regularizes?.map((att, index) => (
|
regularizes?.map((att, index) => (
|
||||||
<tr key={index}>
|
<tr key={index}>
|
||||||
<td colSpan={2}>
|
<td colSpan={2}>
|
||||||
@ -45,10 +44,7 @@ const Regularization = ( { handleRequest} ) =>
|
|||||||
lastName={att.lastName}
|
lastName={att.lastName}
|
||||||
></Avatar>
|
></Avatar>
|
||||||
<div className="d-flex flex-column">
|
<div className="d-flex flex-column">
|
||||||
<a
|
<a href="#" className="text-heading text-truncate">
|
||||||
href="#"
|
|
||||||
className="text-heading text-truncate"
|
|
||||||
>
|
|
||||||
<span className="fw-medium">
|
<span className="fw-medium">
|
||||||
{att.firstName} {att.lastName}
|
{att.firstName} {att.lastName}
|
||||||
</span>
|
</span>
|
||||||
@ -58,22 +54,29 @@ const Regularization = ( { handleRequest} ) =>
|
|||||||
</td>
|
</td>
|
||||||
<td>{moment(att.checkOutTime).format("DD-MMM-YYYY")}</td>
|
<td>{moment(att.checkOutTime).format("DD-MMM-YYYY")}</td>
|
||||||
<td>{convertShortTime(att.checkInTime)}</td>
|
<td>{convertShortTime(att.checkInTime)}</td>
|
||||||
<td>{att.checkOutTime ? convertShortTime(att.checkOutTime):"--"}</td>
|
<td>
|
||||||
<td className='text-center ' >
|
{att.checkOutTime ? convertShortTime(att.checkOutTime) : "--"}
|
||||||
|
</td>
|
||||||
|
<td className="text-center ">
|
||||||
{/* <div className='d-flex justify-content-center align-items-center gap-3'> */}
|
{/* <div className='d-flex justify-content-center align-items-center gap-3'> */}
|
||||||
<RegularizationActions attendanceData={att} handleRequest={handleRequest} refresh={refetch } />
|
<RegularizationActions
|
||||||
|
attendanceData={att}
|
||||||
|
handleRequest={handleRequest}
|
||||||
|
refresh={refetch}
|
||||||
|
/>
|
||||||
{/* </div> */}
|
{/* </div> */}
|
||||||
</td>
|
</td>
|
||||||
</tr>
|
</tr>
|
||||||
))
|
))
|
||||||
) : (
|
) : (
|
||||||
|
<tr>
|
||||||
<td colSpan={5}>No Record Found</td>
|
<td colSpan={5}>No Record Found</td>
|
||||||
)
|
</tr>
|
||||||
}
|
)}
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default Regularization
|
export default Regularization;
|
||||||
|
@ -2,7 +2,7 @@ import React from "react";
|
|||||||
import { useNavigate } from "react-router-dom";
|
import { useNavigate } from "react-router-dom";
|
||||||
|
|
||||||
const Breadcrumb = ({ data }) => {
|
const Breadcrumb = ({ data }) => {
|
||||||
const navigate = useNavigate()
|
const navigate = useNavigate();
|
||||||
return (
|
return (
|
||||||
<nav aria-label="breadcrumb">
|
<nav aria-label="breadcrumb">
|
||||||
<ol className="breadcrumb breadcrumb-style1">
|
<ol className="breadcrumb breadcrumb-style1">
|
||||||
@ -17,7 +17,13 @@ const Breadcrumb = ({ data }) => {
|
|||||||
</a>
|
</a>
|
||||||
</li>
|
</li>
|
||||||
) : (
|
) : (
|
||||||
<li className="breadcrumb-item active "> {item.label}</li>
|
<li
|
||||||
|
className="breadcrumb-item active "
|
||||||
|
key={new Date().getMilliseconds()}
|
||||||
|
>
|
||||||
|
{" "}
|
||||||
|
{item.label}
|
||||||
|
</li>
|
||||||
)
|
)
|
||||||
)}
|
)}
|
||||||
</ol>
|
</ol>
|
||||||
|
@ -37,7 +37,6 @@ const AttendancePage = () => {
|
|||||||
date: new Date().toLocaleDateString(),
|
date: new Date().toLocaleDateString(),
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
const getRole = (roleId) => {
|
const getRole = (roleId) => {
|
||||||
if (!empRoles) return "Unassigned";
|
if (!empRoles) return "Unassigned";
|
||||||
if (!roleId) return "Unassigned";
|
if (!roleId) return "Unassigned";
|
||||||
@ -49,9 +48,7 @@ const AttendancePage = () => {
|
|||||||
setIsCreateModalOpen(true);
|
setIsCreateModalOpen(true);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleModalData = ( employee ) =>
|
const handleModalData = (employee) => {
|
||||||
{
|
|
||||||
|
|
||||||
setModelConfig(employee);
|
setModelConfig(employee);
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -67,24 +64,23 @@ const AttendancePage = () => {
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const handleSubmit = (formData) => {
|
||||||
const handleSubmit = ( formData ) =>
|
dispatch(markCurrentAttendance(formData))
|
||||||
{
|
.then((action) => {
|
||||||
|
const updatedAttendance = attendances.map((item) =>
|
||||||
dispatch( markCurrentAttendance( formData ) ).then( ( action ) =>
|
|
||||||
{
|
|
||||||
const updatedAttendance = attendances.map(item =>
|
|
||||||
item.employeeId === action.payload.employeeId
|
item.employeeId === action.payload.employeeId
|
||||||
? { ...item, ...action.payload }
|
? { ...item, ...action.payload }
|
||||||
: item
|
: item
|
||||||
);
|
);
|
||||||
cacheData("Attendance", { data: updatedAttendance, projectId: selectedProject })
|
cacheData("Attendance", {
|
||||||
setAttendances(updatedAttendance)
|
data: updatedAttendance,
|
||||||
showToast("Attedance Marked Successfully","success")
|
projectId: selectedProject,
|
||||||
|
});
|
||||||
|
setAttendances(updatedAttendance);
|
||||||
|
showToast("Attedance Marked Successfully", "success");
|
||||||
})
|
})
|
||||||
.catch( ( error ) =>
|
.catch((error) => {
|
||||||
{
|
showToast(error.message, "error");
|
||||||
showToast(error.message,"error")
|
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -100,8 +96,6 @@ const AttendancePage = () => {
|
|||||||
dispatch(setProjectId(projects[0]?.id));
|
dispatch(setProjectId(projects[0]?.id));
|
||||||
}, [projects]);
|
}, [projects]);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<>
|
<>
|
||||||
{isCreateModalOpen && modelConfig && (
|
{isCreateModalOpen && modelConfig && (
|
||||||
@ -127,14 +121,14 @@ const AttendancePage = () => {
|
|||||||
{ label: "Attendance", link: null },
|
{ label: "Attendance", link: null },
|
||||||
]}
|
]}
|
||||||
></Breadcrumb>
|
></Breadcrumb>
|
||||||
<div class="nav-align-top nav-tabs-shadow">
|
<div className="nav-align-top nav-tabs-shadow">
|
||||||
<ul class="nav nav-tabs" role="tablist">
|
<ul className="nav nav-tabs" role="tablist">
|
||||||
<div
|
<div
|
||||||
className="dataTables_length text-start py-2 px-2"
|
className="dataTables_length text-start py-2 px-2"
|
||||||
id="DataTables_Table_0_length"
|
id="DataTables_Table_0_length"
|
||||||
>
|
>
|
||||||
{
|
{loginUser && loginUser?.projects?.length > 1 && (
|
||||||
((loginUser && loginUser?.projects?.length > 1) ) && (<label>
|
<label>
|
||||||
<select
|
<select
|
||||||
name="DataTables_Table_0_length"
|
name="DataTables_Table_0_length"
|
||||||
aria-controls="DataTables_Table_0"
|
aria-controls="DataTables_Table_0"
|
||||||
@ -143,42 +137,54 @@ const AttendancePage = () => {
|
|||||||
onChange={(e) => dispatch(setProjectId(e.target.value))}
|
onChange={(e) => dispatch(setProjectId(e.target.value))}
|
||||||
aria-label=""
|
aria-label=""
|
||||||
>
|
>
|
||||||
{!projectLoading && projects?.filter(project =>
|
{!projectLoading &&
|
||||||
loginUser?.projects?.map(Number).includes(project.id)).map((project)=>(
|
projects
|
||||||
<option value={project.id} key={project.id}>{project.name}</option>
|
?.filter((project) =>
|
||||||
|
loginUser?.projects?.map(Number).includes(project.id)
|
||||||
|
)
|
||||||
|
.map((project) => (
|
||||||
|
<option value={project.id} key={project.id}>
|
||||||
|
{project.name}
|
||||||
|
</option>
|
||||||
))}
|
))}
|
||||||
{projectLoading && <option value="Loading..." disabled>Loading...</option> }
|
{projectLoading && (
|
||||||
|
<option value="Loading..." disabled>
|
||||||
|
Loading...
|
||||||
|
</option>
|
||||||
|
)}
|
||||||
</select>
|
</select>
|
||||||
</label>)
|
</label>
|
||||||
}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</ul>
|
</ul>
|
||||||
<ul class="nav nav-tabs" role="tablist">
|
<ul className="nav nav-tabs" role="tablist">
|
||||||
<li class="nav-item">
|
<li className="nav-item">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="nav-link active"
|
className="nav-link active"
|
||||||
role="tab"
|
role="tab"
|
||||||
data-bs-toggle="tab"
|
data-bs-toggle="tab"
|
||||||
data-bs-target="#navs-top-home"
|
data-bs-target="#navs-top-home"
|
||||||
aria-controls="navs-top-home"
|
aria-controls="navs-top-home"
|
||||||
aria-selected="true">
|
aria-selected="true"
|
||||||
|
>
|
||||||
All
|
All
|
||||||
</button>
|
</button>
|
||||||
</li>
|
</li>
|
||||||
<li class="nav-item">
|
<li className="nav-item">
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
class="nav-link"
|
className="nav-link"
|
||||||
role="tab"
|
role="tab"
|
||||||
data-bs-toggle="tab"
|
data-bs-toggle="tab"
|
||||||
data-bs-target="#navs-top-profile"
|
data-bs-target="#navs-top-profile"
|
||||||
aria-controls="navs-top-profile"
|
aria-controls="navs-top-profile"
|
||||||
aria-selected="false">
|
aria-selected="false"
|
||||||
|
>
|
||||||
Logs
|
Logs
|
||||||
</button>
|
</button>
|
||||||
</li>
|
</li>
|
||||||
<li className={`nav-item ${!DoRegularized && 'd-none'}`}>
|
<li className={`nav-item ${!DoRegularized && "d-none"}`}>
|
||||||
<button
|
<button
|
||||||
type="button"
|
type="button"
|
||||||
className="nav-link "
|
className="nav-link "
|
||||||
@ -225,10 +231,7 @@ const AttendancePage = () => {
|
|||||||
id="navs-top-messages"
|
id="navs-top-messages"
|
||||||
role="tabpanel"
|
role="tabpanel"
|
||||||
>
|
>
|
||||||
<Regularization
|
<Regularization handleRequest={handleSubmit} />
|
||||||
|
|
||||||
handleRequest={handleSubmit}
|
|
||||||
/>
|
|
||||||
</div>
|
</div>
|
||||||
</>
|
</>
|
||||||
)}
|
)}
|
||||||
|
@ -12,9 +12,7 @@ import { MANAGE_EMPLOYEES } from "../../utils/constants";
|
|||||||
import { useHasUserPermission } from "../../hooks/useHasUserPermission";
|
import { useHasUserPermission } from "../../hooks/useHasUserPermission";
|
||||||
import SuspendEmp from "../../components/Employee/SuspendEmp";
|
import SuspendEmp from "../../components/Employee/SuspendEmp";
|
||||||
|
|
||||||
const EmployeeList = () =>
|
const EmployeeList = () => {
|
||||||
{
|
|
||||||
|
|
||||||
const { profile: loginUser } = useProfile();
|
const { profile: loginUser } = useProfile();
|
||||||
const [selectedProject, setSelectedProject] = useState("");
|
const [selectedProject, setSelectedProject] = useState("");
|
||||||
const { projects, loading: projectLoading } = useProjects();
|
const { projects, loading: projectLoading } = useProjects();
|
||||||
@ -111,10 +109,10 @@ const EmployeeList = () =>
|
|||||||
)}
|
)}
|
||||||
|
|
||||||
<div
|
<div
|
||||||
className={`modal fade ${showModal ? 'show' : ''}`}
|
className={`modal fade ${showModal ? "show" : ""}`}
|
||||||
tabIndex="-1"
|
tabIndex="-1"
|
||||||
role="dialog"
|
role="dialog"
|
||||||
style={{ display: showModal ? 'block' : 'none' }}
|
style={{ display: showModal ? "block" : "none" }}
|
||||||
aria-hidden={!showModal}
|
aria-hidden={!showModal}
|
||||||
>
|
>
|
||||||
<SuspendEmp onClose={handleClose} />
|
<SuspendEmp onClose={handleClose} />
|
||||||
@ -329,28 +327,31 @@ const EmployeeList = () =>
|
|||||||
</tr>
|
</tr>
|
||||||
)}
|
)}
|
||||||
{!loading && employeeList?.length === 0 && (
|
{!loading && employeeList?.length === 0 && (
|
||||||
|
<tr>
|
||||||
<td
|
<td
|
||||||
colSpan={8}
|
colSpan={8}
|
||||||
style={{ paddingTop: "20px", textAlign: "center" }}
|
style={{ paddingTop: "20px", textAlign: "center" }}
|
||||||
>
|
>
|
||||||
No Data Found
|
No Data Found
|
||||||
</td>
|
</td>
|
||||||
|
</tr>
|
||||||
)}
|
)}
|
||||||
{!loading &&
|
{!loading &&
|
||||||
employeeList &&
|
employeeList &&
|
||||||
currentItems.length === 0 &&
|
currentItems.length === 0 &&
|
||||||
employeeList.length !== 0 && (
|
employeeList.length !== 0 && (
|
||||||
|
<tr>
|
||||||
<td colSpan={8}>
|
<td colSpan={8}>
|
||||||
<small className="muted">
|
<small className="muted">
|
||||||
'{searchText}' employee not found
|
'{searchText}' employee not found
|
||||||
</small>{" "}
|
</small>{" "}
|
||||||
</td>
|
</td>
|
||||||
|
</tr>
|
||||||
)}
|
)}
|
||||||
|
|
||||||
{currentItems &&
|
{currentItems &&
|
||||||
!loading &&
|
!loading &&
|
||||||
currentItems
|
currentItems.map((item) => (
|
||||||
.map((item) => (
|
|
||||||
<tr className="odd" key={item.id}>
|
<tr className="odd" key={item.id}>
|
||||||
<td className="sorting_1" colSpan={2}>
|
<td className="sorting_1" colSpan={2}>
|
||||||
<div className="d-flex justify-content-start align-items-center user-name">
|
<div className="d-flex justify-content-start align-items-center user-name">
|
||||||
@ -361,9 +362,7 @@ const EmployeeList = () =>
|
|||||||
<div className="d-flex flex-column">
|
<div className="d-flex flex-column">
|
||||||
<a
|
<a
|
||||||
onClick={() =>
|
onClick={() =>
|
||||||
navigate(
|
navigate(`/employee/${item.id}?for=account`)
|
||||||
`/employee/${item.id}?for=account`
|
|
||||||
)
|
|
||||||
}
|
}
|
||||||
className="text-heading text-truncate cursor-pointer"
|
className="text-heading text-truncate cursor-pointer"
|
||||||
>
|
>
|
||||||
@ -435,7 +434,10 @@ const EmployeeList = () =>
|
|||||||
>
|
>
|
||||||
Edit
|
Edit
|
||||||
</Link>
|
</Link>
|
||||||
<button className="dropdown-item" onClick={handleShow}>
|
<button
|
||||||
|
className="dropdown-item"
|
||||||
|
onClick={handleShow}
|
||||||
|
>
|
||||||
Suspend
|
Suspend
|
||||||
</button>
|
</button>
|
||||||
<button
|
<button
|
||||||
|
@ -1,92 +1,96 @@
|
|||||||
|
// AppRoutes.jsx
|
||||||
import React from "react";
|
import React from "react";
|
||||||
import { BrowserRouter as Router, Routes, Route } from "react-router-dom";
|
import { createBrowserRouter, RouterProvider, Outlet } from "react-router-dom";
|
||||||
import AuthLayout from "../layouts/AuthLayout"
|
|
||||||
|
// Layouts
|
||||||
|
import AuthLayout from "../layouts/AuthLayout";
|
||||||
import HomeLayout from "../layouts/HomeLayout";
|
import HomeLayout from "../layouts/HomeLayout";
|
||||||
|
|
||||||
|
// Authentication Pages
|
||||||
import LoginPage from "../pages/authentication/LoginPage";
|
import LoginPage from "../pages/authentication/LoginPage";
|
||||||
import RegisterPage from "../pages/authentication/RegisterPage";
|
import RegisterPage from "../pages/authentication/RegisterPage";
|
||||||
import ForgotPasswordPage from "../pages/authentication/ForgotPasswordPage";
|
import ForgotPasswordPage from "../pages/authentication/ForgotPasswordPage";
|
||||||
import ResetPasswordPage from "../pages/authentication/ResetPasswordPage";
|
import ResetPasswordPage from "../pages/authentication/ResetPasswordPage";
|
||||||
|
|
||||||
|
// Home & Protected Pages
|
||||||
import Dashboard from "../components/Dashboard/Dashboard";
|
import Dashboard from "../components/Dashboard/Dashboard";
|
||||||
import ProjectList from "../pages/project/ProjectList";
|
import ProjectList from "../pages/project/ProjectList";
|
||||||
import EmployeeList from "../pages/employee/EmployeeList";
|
|
||||||
import NotFound from "../pages/Home/NotFound";
|
|
||||||
import ProtectedRoute from "./ProtectedRoute";
|
|
||||||
import ProjectDetails from "../pages/project/ProjectDetails";
|
import ProjectDetails from "../pages/project/ProjectDetails";
|
||||||
|
import ManageProject from "../components/Project/ManageProject";
|
||||||
|
import EmployeeList from "../pages/employee/EmployeeList";
|
||||||
|
import ManageEmp from "../pages/employee/ManageEmp";
|
||||||
|
import EmployeeProfile from "../pages/employee/EmployeeProfile";
|
||||||
import Inventory from "../pages/project/Inventory";
|
import Inventory from "../pages/project/Inventory";
|
||||||
|
import AttendancePage from "../pages/Activities/AttendancePage";
|
||||||
|
import DailyTask from "../pages/Activities/DailyTask";
|
||||||
import TaskPlannng from "../pages/Activities/TaskPlannng";
|
import TaskPlannng from "../pages/Activities/TaskPlannng";
|
||||||
import Reports from "../pages/reports/Reports";
|
import Reports from "../pages/reports/Reports";
|
||||||
import ImageGallary from "../pages/Gallary/ImageGallary";
|
import ImageGallary from "../pages/Gallary/ImageGallary";
|
||||||
|
import MasterPage from "../pages/master/MasterPage";
|
||||||
import Support from "../pages/support/Support";
|
import Support from "../pages/support/Support";
|
||||||
import Documentation from "../pages/support/Documentation";
|
import Documentation from "../pages/support/Documentation";
|
||||||
import Connect from "../pages/support/Connect";
|
import Connect from "../pages/support/Connect";
|
||||||
import ManageProject from "../components/Project/ManageProject";
|
|
||||||
import ManageEmp from "../pages/employee/ManageEmp";
|
|
||||||
import EmployeeProfile from "../pages/employee/EmployeeProfile";
|
|
||||||
import { ErrorPage } from "../pages/support/ErrorPage";
|
import { ErrorPage } from "../pages/support/ErrorPage";
|
||||||
import MasterPage from "../pages/master/MasterPage";
|
|
||||||
import DailyTask from "../pages/Activities/DailyTask";
|
|
||||||
import AttendancePage from "../pages/Activities/AttendancePage";
|
|
||||||
|
|
||||||
|
// Protected Route Wrapper
|
||||||
|
import ProtectedRoute from "./ProtectedRoute";
|
||||||
|
|
||||||
|
const router = createBrowserRouter(
|
||||||
|
[
|
||||||
|
{
|
||||||
|
element: <AuthLayout />,
|
||||||
|
children: [
|
||||||
|
{ path: "/auth/login", element: <LoginPage /> },
|
||||||
|
{ path: "/auth/reqest/demo", element: <RegisterPage /> },
|
||||||
|
{ path: "/auth/forgot-password", element: <ForgotPasswordPage /> },
|
||||||
|
{ path: "/reset-password", element: <ResetPasswordPage /> },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
element: <ProtectedRoute />,
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
element: <HomeLayout />,
|
||||||
|
children: [
|
||||||
|
{ path: "/", element: <Dashboard /> },
|
||||||
|
{ path: "/dashboard", element: <Dashboard /> },
|
||||||
|
{ path: "/projects", element: <ProjectList /> },
|
||||||
|
{ path: "/projects/:projectId", element: <ProjectDetails /> },
|
||||||
|
{ path: "/project/manage/:projectId", element: <ManageProject /> },
|
||||||
|
{ path: "/employees", element: <EmployeeList /> },
|
||||||
|
{ path: "/employee/:employeeId", element: <EmployeeProfile /> },
|
||||||
|
{ path: "/employee/manage", element: <ManageEmp /> },
|
||||||
|
{ path: "/employee/manage/:employeeId", element: <ManageEmp /> },
|
||||||
|
{ path: "/inventory", element: <Inventory /> },
|
||||||
|
{ path: "/activities/attendance", element: <AttendancePage /> },
|
||||||
|
{ path: "/activities/records", element: <DailyTask /> },
|
||||||
|
{ path: "/activities/task", element: <TaskPlannng /> },
|
||||||
|
{ path: "/activities/reports", element: <Reports /> },
|
||||||
|
{ path: "/activities/gallary", element: <ImageGallary /> },
|
||||||
|
{ path: "/masters", element: <MasterPage /> },
|
||||||
|
{ path: "/help/support", element: <Support /> },
|
||||||
|
{ path: "/help/docs", element: <Documentation /> },
|
||||||
|
{ path: "/help/connect", element: <Connect /> },
|
||||||
|
],
|
||||||
|
},
|
||||||
|
],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: "*",
|
||||||
|
element: <ErrorPage />,
|
||||||
|
},
|
||||||
|
],
|
||||||
|
{
|
||||||
|
// ✅ Enables the v7 splat path resolution behavior
|
||||||
|
future: {
|
||||||
|
v7_relativeSplatPath: true,
|
||||||
|
v7_startTransition: true,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
const AppRoutes = () => {
|
const AppRoutes = () => {
|
||||||
return (
|
return <RouterProvider router={router} />;
|
||||||
<Router>
|
|
||||||
<Routes>
|
|
||||||
{/* Authentication Routes */}
|
|
||||||
<Route element={<AuthLayout />}>
|
|
||||||
<Route path="/auth/login" element={<LoginPage />} />
|
|
||||||
<Route path="/auth/reqest/demo" element={<RegisterPage />} />
|
|
||||||
<Route
|
|
||||||
path="/auth/forgot-password"
|
|
||||||
element={<ForgotPasswordPage />}
|
|
||||||
/>
|
|
||||||
<Route path="/reset-password" element={<ResetPasswordPage />} />
|
|
||||||
</Route>
|
|
||||||
|
|
||||||
|
|
||||||
{/* Protected Routes */}
|
|
||||||
<Route element={<ProtectedRoute />}>
|
|
||||||
<Route element={<HomeLayout />}>
|
|
||||||
<Route path="/" element={<Dashboard />} />
|
|
||||||
<Route path="/dashboard" element={<Dashboard />} />
|
|
||||||
<Route path="/projects" element={<ProjectList />} />
|
|
||||||
<Route path="/projects/:projectId" element={<ProjectDetails />} />
|
|
||||||
<Route
|
|
||||||
path="/project/manage/:projectId"
|
|
||||||
element={<ManageProject />}
|
|
||||||
/>
|
|
||||||
<Route path="/employees" element={<EmployeeList />} />
|
|
||||||
<Route path="/employee/:employeeId" element={<EmployeeProfile />} />
|
|
||||||
|
|
||||||
<Route path="/employee/manage" element={<ManageEmp />} />
|
|
||||||
<Route
|
|
||||||
path="/employee/manage/:employeeId"
|
|
||||||
element={<ManageEmp />}
|
|
||||||
/>
|
|
||||||
<Route path="/inventory" element={<Inventory />} />
|
|
||||||
|
|
||||||
<Route path="/activities/attendance" element={<AttendancePage />} />
|
|
||||||
|
|
||||||
<Route path="/activities/records" element={<DailyTask />} />
|
|
||||||
<Route path="/activities/task" element={<TaskPlannng />} />
|
|
||||||
|
|
||||||
<Route path="/activities/reports" element={<Reports />} />
|
|
||||||
|
|
||||||
<Route path="/activities/gallary" element={<ImageGallary />} />
|
|
||||||
<Route path="/masters" element={<MasterPage />} />
|
|
||||||
|
|
||||||
<Route path="/help/support" element={<Support />} />
|
|
||||||
<Route path="/help/docs" element={<Documentation />} />
|
|
||||||
<Route path="/help/connect" element={<Connect />} />
|
|
||||||
</Route>
|
|
||||||
</Route>
|
|
||||||
|
|
||||||
{/* Fallback Route */}
|
|
||||||
<Route path="*" element={<ErrorPage />} />
|
|
||||||
</Routes>
|
|
||||||
</Router>
|
|
||||||
);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
export default AppRoutes;
|
export default AppRoutes;
|
||||||
|
@ -3,7 +3,7 @@ import { useNavigate } from "react-router-dom";
|
|||||||
import axiosRetry from "axios-retry";
|
import axiosRetry from "axios-retry";
|
||||||
import showToast from "../services/toastService";
|
import showToast from "../services/toastService";
|
||||||
const base_Url = process.env.VITE_BASE_URL;
|
const base_Url = process.env.VITE_BASE_URL;
|
||||||
// const base_Url = https://api.marcoaiot.com;
|
// const base_Url = "https://api.marcoaiot.com";
|
||||||
export const axiosClient = axios.create({
|
export const axiosClient = axios.create({
|
||||||
baseURL: base_Url, // Your Web API URL
|
baseURL: base_Url, // Your Web API URL
|
||||||
withCredentials: false, // Required if the API uses cookies
|
withCredentials: false, // Required if the API uses cookies
|
||||||
@ -31,11 +31,8 @@ axiosClient.interceptors.request.use(
|
|||||||
// // Response interceptor to handle responses globally (optional)
|
// // Response interceptor to handle responses globally (optional)
|
||||||
// Add an interceptor to handle expired tokens
|
// Add an interceptor to handle expired tokens
|
||||||
axiosClient.interceptors.response.use(
|
axiosClient.interceptors.response.use(
|
||||||
|
|
||||||
|
|
||||||
(response) => response,
|
(response) => response,
|
||||||
async (error) => {
|
async (error) => {
|
||||||
|
|
||||||
const originalRequest = error.config;
|
const originalRequest = error.config;
|
||||||
|
|
||||||
if (!originalRequest) {
|
if (!originalRequest) {
|
||||||
@ -51,46 +48,46 @@ axiosClient.interceptors.response.use(
|
|||||||
"Connection refused. Please ensure the server is running."
|
"Connection refused. Please ensure the server is running."
|
||||||
);
|
);
|
||||||
|
|
||||||
if ( error.config.url.indexOf( 'refresh-token' ) == -1 )
|
if (error.config.url.indexOf("refresh-token") == -1) {
|
||||||
{
|
//showToast("Server is unreachable. Try again later!", "error");
|
||||||
showToast("Server is unreachable. Try again later!", "error");
|
console.log("1 - error fetching refresh token :", error);
|
||||||
}
|
} else {
|
||||||
showToast(
|
showToast(
|
||||||
"Unable to connect to the server. Please try again later.",
|
"Unable to connect to the server. Please try again later.",
|
||||||
"error"
|
"error"
|
||||||
);
|
);
|
||||||
|
}
|
||||||
} else if (error.code === "ERR_NETWORK") {
|
} else if (error.code === "ERR_NETWORK") {
|
||||||
console.error("Network error: Unable to reach the server.");
|
console.error("Network error: Unable to reach the server.");
|
||||||
if ( error.config.url.indexOf( 'refresh-token' ) == -1 )
|
if (error.config.url.indexOf("refresh-token") == -1) {
|
||||||
{
|
//showToast("Server is unreachable. Try again later!", "error");
|
||||||
showToast("Server is unreachable. Try again later!", "error");
|
console.log("2 - error fetching refresh token :", error);
|
||||||
}
|
}
|
||||||
redirectToLogin();
|
redirectToLogin();
|
||||||
} else if (error.code === "ECONNABORTED") {
|
} else if (error.code === "ECONNABORTED") {
|
||||||
console.error("Request timed out.");
|
console.error("Request timed out.");
|
||||||
|
|
||||||
if ( error.config.url.indexOf( 'refresh-token' ) == -1 )
|
if (error.config.url.indexOf("refresh-token") == -1) {
|
||||||
{
|
//showToast("Server is unreachable. Try again later!", "error");
|
||||||
showToast("Server is unreachable. Try again later!", "error");
|
console.log("3 - error fetching refresh token :", error);
|
||||||
}
|
} else {
|
||||||
showToast(
|
showToast(
|
||||||
"The request took too long. Please try again later.",
|
"The request took too long. Please try again later.",
|
||||||
"error"
|
"error"
|
||||||
);
|
);
|
||||||
} else if ( error.response )
|
|
||||||
{
|
|
||||||
|
|
||||||
if ( error.config.url.indexOf( 'refresh-token' ) == -1 )
|
|
||||||
{
|
|
||||||
showToast("Server is unreachable. Try again later!", "error");
|
|
||||||
}
|
}
|
||||||
|
} else if (error.response) {
|
||||||
|
if (error.config.url.indexOf("refresh-token") == -1) {
|
||||||
|
//showToast("Server is unreachable. Try again later!", "error");
|
||||||
|
console.log("4 - error fetching refresh token :", error);
|
||||||
|
} else {
|
||||||
showToast(error.response.data.message, "error");
|
showToast(error.response.data.message, "error");
|
||||||
|
}
|
||||||
|
|
||||||
if (error.response.status === 401 && !originalRequest._retry) {
|
if (error.response.status === 401 && !originalRequest._retry) {
|
||||||
originalRequest._retry = true;
|
originalRequest._retry = true;
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
|
||||||
// Get the refresh token from secure storage
|
// Get the refresh token from secure storage
|
||||||
const refreshToken = localStorage.getItem("refreshToken");
|
const refreshToken = localStorage.getItem("refreshToken");
|
||||||
|
|
||||||
@ -115,11 +112,19 @@ axiosClient.interceptors.response.use(
|
|||||||
// Retry the original request
|
// Retry the original request
|
||||||
return axiosClient(originalRequest);
|
return axiosClient(originalRequest);
|
||||||
} catch (err) {
|
} catch (err) {
|
||||||
|
console.log(
|
||||||
|
"5 - error fetching refresh token during refresh call :",
|
||||||
|
error
|
||||||
|
);
|
||||||
// Redirect to login if token refresh fails
|
// Redirect to login if token refresh fails
|
||||||
redirectToLogin();
|
redirectToLogin();
|
||||||
return Promise.reject(err);
|
return Promise.reject(err);
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
|
if (error.config.url.indexOf("refresh-token") == -1) {
|
||||||
|
//showToast("Server is unreachable. Try again later!", "error");
|
||||||
|
console.log("6 - error fetching refresh token :", error);
|
||||||
|
}
|
||||||
showToast(
|
showToast(
|
||||||
error.response.data?.message ||
|
error.response.data?.message ||
|
||||||
"An error occurred. Please try again.",
|
"An error occurred. Please try again.",
|
||||||
|
Loading…
x
Reference in New Issue
Block a user