pramod_Bug-#215 : Fixed : Display correct icon for approved check-out regularization requests in employee attendance profile #168
| @ -2,12 +2,27 @@ import React, { useEffect, useState } from "react"; | |||||||
| import { useEmployeeAttendacesLog } from "../../hooks/useAttendance"; | import { useEmployeeAttendacesLog } from "../../hooks/useAttendance"; | ||||||
| import { convertShortTime } from "../../utils/dateUtils"; | import { convertShortTime } from "../../utils/dateUtils"; | ||||||
| import { useNavigate } from "react-router-dom"; | import { useNavigate } from "react-router-dom"; | ||||||
|  | import { THRESH_HOLD } from "../../utils/constants"; | ||||||
| 
 | 
 | ||||||
| const AttendLogs = ({ Id }) => { | const AttendLogs = ({ Id }) => { | ||||||
|   const { logs, loading } = useEmployeeAttendacesLog(Id); |   const { logs, loading } = useEmployeeAttendacesLog(Id); | ||||||
|   const navigate = useNavigate(); |   const navigate = useNavigate(); | ||||||
| 
 | 
 | ||||||
|   const whichActivityPerform = (actvity) => { |   const isCheckoutRegularized = ( | ||||||
|  |     activityTimeStr, | ||||||
|  |     checkoutTimeStr, | ||||||
|  |     threshHours | ||||||
|  |   ) => { | ||||||
|  |     if (!activityTimeStr || !checkoutTimeStr) return false; | ||||||
|  | 
 | ||||||
|  |     const activityTime = new Date(activityTimeStr); | ||||||
|  |     const checkoutTime = new Date(checkoutTimeStr); | ||||||
|  |     const threshTimeMs = threshHours * 60 * 60 * 1000; | ||||||
|  | 
 | ||||||
|  |     return checkoutTime - activityTime > threshTimeMs; | ||||||
|  |   }; | ||||||
|  | 
 | ||||||
|  |   const whichActivityPerform = (actvity, checkOutTime) => { | ||||||
|     switch (actvity) { |     switch (actvity) { | ||||||
|       case 1: |       case 1: | ||||||
|         return ( |         return ( | ||||||
| @ -24,12 +39,12 @@ const AttendLogs = ({ Id }) => { | |||||||
|       case 2: |       case 2: | ||||||
|         return ( |         return ( | ||||||
|           <i |           <i | ||||||
|             className="bx bx-help-circle text-secondary" |             className="bx bx-help-circle text-danger" | ||||||
|             data-bs-toggle="tooltip" |             data-bs-toggle="tooltip" | ||||||
|             data-bs-offset="0,8" |             data-bs-offset="0,8" | ||||||
|             data-bs-placement="top" |             data-bs-placement="top" | ||||||
|             data-bs-custom-class="tooltip" |             data-bs-custom-class="tooltip" | ||||||
|             title="regularize Requested" |             title="Regularize Requested" | ||||||
|           ></i> |           ></i> | ||||||
|         ); |         ); | ||||||
|         break; |         break; | ||||||
| @ -41,11 +56,30 @@ const AttendLogs = ({ Id }) => { | |||||||
|             data-bs-offset="0,8" |             data-bs-offset="0,8" | ||||||
|             data-bs-placement="top" |             data-bs-placement="top" | ||||||
|             data-bs-custom-class="tooltip" |             data-bs-custom-class="tooltip" | ||||||
|             title="Regularized" |             title="Request Deleted!" | ||||||
|           ></i> |           ></i> | ||||||
|         ); |         ); | ||||||
|         break; |         break; | ||||||
|       case 4: |       case 4: | ||||||
|  |         if ( | ||||||
|  |           checkOutTime && | ||||||
|  |           isCheckoutRegularized( | ||||||
|  |             logs[0]?.activityTime, | ||||||
|  |             checkOutTime, | ||||||
|  |             THRESH_HOLD | ||||||
|  |           ) | ||||||
|  |         ) { | ||||||
|  |           return ( | ||||||
|  |             <i | ||||||
|  |               className="bx bx-check-circle text-success" | ||||||
|  |               data-bs-toggle="tooltip" | ||||||
|  |               data-bs-offset="0,8" | ||||||
|  |               data-bs-placement="top" | ||||||
|  |               data-bs-custom-class="tooltip" | ||||||
|  |               title="Regularized" | ||||||
|  |             ></i> | ||||||
|  |           ); | ||||||
|  |         } else { | ||||||
|           return ( |           return ( | ||||||
|             <i |             <i | ||||||
|               className="bx bx-left-arrow-circle text-danger" |               className="bx bx-left-arrow-circle text-danger" | ||||||
| @ -56,6 +90,7 @@ const AttendLogs = ({ Id }) => { | |||||||
|               title="Check-Out" |               title="Check-Out" | ||||||
|             ></i> |             ></i> | ||||||
|           ); |           ); | ||||||
|  |         } | ||||||
| 
 | 
 | ||||||
|         break; |         break; | ||||||
|       case 5: |       case 5: | ||||||
| @ -79,6 +114,7 @@ const AttendLogs = ({ Id }) => { | |||||||
|     const url = `https://www.google.com/maps?q=${lat},${lng}`; |     const url = `https://www.google.com/maps?q=${lat},${lng}`; | ||||||
|     window.open(url, "_blank"); // Open in new tab |     window.open(url, "_blank"); // Open in new tab | ||||||
|   }; |   }; | ||||||
|  | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     const tooltipTriggerList = Array.from( |     const tooltipTriggerList = Array.from( | ||||||
|       document.querySelectorAll('[data-bs-toggle="tooltip"]') |       document.querySelectorAll('[data-bs-toggle="tooltip"]') | ||||||
| @ -122,11 +158,13 @@ const AttendLogs = ({ Id }) => { | |||||||
|                   <tr key={index}> |                   <tr key={index}> | ||||||
|                     <td>{log.activityTime.slice(0, 10)}</td> |                     <td>{log.activityTime.slice(0, 10)}</td> | ||||||
|                     <td>{convertShortTime(log.activityTime)}</td> |                     <td>{convertShortTime(log.activityTime)}</td> | ||||||
|                     <td>{whichActivityPerform(log.activity)}</td> |                     <td> | ||||||
|  |                       {whichActivityPerform(log.activity, log.activityTime)} | ||||||
|  |                     </td> | ||||||
|                     <td> |                     <td> | ||||||
|                       {log?.latitude != 0 ? ( |                       {log?.latitude != 0 ? ( | ||||||
|                         <i |                         <i | ||||||
|                           className="bx bx-location-plus text-danger  cursor-pointer" |                           className="bx bx-map text-danger  cursor-pointer" | ||||||
|                           data-bs-toggle="tooltip" |                           data-bs-toggle="tooltip" | ||||||
|                           data-bs-offset="0,8" |                           data-bs-offset="0,8" | ||||||
|                           data-bs-placement="top" |                           data-bs-placement="top" | ||||||
| @ -146,7 +184,9 @@ const AttendLogs = ({ Id }) => { | |||||||
|                       }`} |                       }`} | ||||||
|                     </td> |                     </td> | ||||||
|                     <td className="text-wrap" colSpan={3}> |                     <td className="text-wrap" colSpan={3}> | ||||||
|                       {log?.comment || "--"} |                       {log?.comment?.length > 50 | ||||||
|  |                         ? `${log.comment.slice(0, 50)}...` | ||||||
|  |                         : log.comment || "--"} | ||||||
|                     </td> |                     </td> | ||||||
|                   </tr> |                   </tr> | ||||||
|                 ))} |                 ))} | ||||||
|  | |||||||
| @ -1,17 +1,16 @@ | |||||||
| import { useEffect, useState } from "react"; | import { useEffect, useState } from 'react'; | ||||||
| import { timeElapsed } from "../utils/dateUtils"; | import { timeElapsed } from '../utils/dateUtils'; // Make sure it calculates in hours
 | ||||||
| import { useSelector } from "react-redux"; | import { THRESH_HOLD } from '../utils/constants'; | ||||||
| import { THRESH_HOLD } from "../utils/constants"; |  | ||||||
| 
 | 
 | ||||||
|  | // THRESH_HOLD in hours (e.g., 12, 48, or 60)
 | ||||||
| export const ACTIONS = { | export const ACTIONS = { | ||||||
|   CHECK_IN: 0, |   CHECK_IN: 0, | ||||||
|   CHECK_OUT: 1, |   CHECK_OUT: 1, | ||||||
|   REGULARIZATION: 2, |   REGULARIZATION: 2, | ||||||
|   REQUESTED: 3, |   REQUESTED: 3, | ||||||
|   APPROVED: 4, |   APPROVED: 4, | ||||||
|   REJECTED: 5, |   REJECTED: 5 | ||||||
| }; | }; | ||||||
| const now = new Date(); |  | ||||||
| 
 | 
 | ||||||
| const useAttendanceStatus = (attendanceData) => { | const useAttendanceStatus = (attendanceData) => { | ||||||
|   const [status, setStatus] = useState({ |   const [status, setStatus] = useState({ | ||||||
| @ -19,133 +18,100 @@ const useAttendanceStatus = (attendanceData) => { | |||||||
|     action: null, |     action: null, | ||||||
|     disabled: true, |     disabled: true, | ||||||
|     text: "Unknown", |     text: "Unknown", | ||||||
|     color: "btn-secondary", |     color: 'btn-secondary', | ||||||
|   }); |   }); | ||||||
| 
 | 
 | ||||||
|   useEffect(() => { |   useEffect(() => { | ||||||
|     const { checkInTime, checkOutTime, activity } = attendanceData; |     const { checkInTime, checkOutTime, activity } = attendanceData; | ||||||
|  |     const now = new Date(); | ||||||
| 
 | 
 | ||||||
|     if (activity === 0 && checkInTime === null && checkOutTime === null) { |     const isSameDay = (date1, date2) => | ||||||
|       setStatus({ |       new Date(date1).toDateString() === new Date(date2).toDateString(); | ||||||
|  | 
 | ||||||
|  |     // 1. No check-in/check-out yet → Allow Check-In
 | ||||||
|  |     if (activity === 0 && !checkInTime && !checkOutTime) { | ||||||
|  |       return setStatus({ | ||||||
|         status: "Check-In", |         status: "Check-In", | ||||||
|         action: ACTIONS.CHECK_IN, |         action: ACTIONS.CHECK_IN, | ||||||
|         disabled: false, |         disabled: false, | ||||||
|         text: "Check In", |         text: "Check In", | ||||||
|         color: "btn-primary", |         color: 'btn-primary', | ||||||
|       }); |       }); | ||||||
|     } else if (activity === 4 && new Date(checkOutTime) < now) { |     } | ||||||
|       setStatus({ | 
 | ||||||
|         status: "Approved", |     // 2. Checked in, no checkout yet
 | ||||||
|         action: ACTIONS.APPROVED, |     if (checkInTime && !checkOutTime) { | ||||||
|         disabled: true, |       if (timeElapsed(checkInTime, THRESH_HOLD)) { | ||||||
|         text: "Approved", |         return setStatus({ | ||||||
|         color: "btn-success", |  | ||||||
|       }); |  | ||||||
|     } else if ( |  | ||||||
|       activity === 0 && |  | ||||||
|       checkInTime === null && |  | ||||||
|       checkOutTime === null && |  | ||||||
|       !timeElapsed(checkInTime, THRESH_HOLD) |  | ||||||
|     ) { |  | ||||||
|       setStatus({ |  | ||||||
|         status: "Check-In", |  | ||||||
|         action: ACTIONS.CHECK_IN, |  | ||||||
|         disabled: false, |  | ||||||
|         text: "Check In", |  | ||||||
|         color: "btn-primary", |  | ||||||
|       }); |  | ||||||
|     } else if ( |  | ||||||
|       activity === 0 && |  | ||||||
|       checkInTime !== null && |  | ||||||
|       checkOutTime === null && |  | ||||||
|       timeElapsed(checkInTime, THRESH_HOLD) |  | ||||||
|     ) { |  | ||||||
|       setStatus({ |  | ||||||
|         status: "Request Regularize", |  | ||||||
|         action: ACTIONS.REGULARIZATION, |  | ||||||
|         disabled: false, |  | ||||||
|         text: "Regularizes", |  | ||||||
|         color: "btn-warning", |  | ||||||
|       }); |  | ||||||
|     } else if ( |  | ||||||
|       activity === 1 && |  | ||||||
|       checkInTime !== null && |  | ||||||
|       checkOutTime === null && |  | ||||||
|       !timeElapsed(checkInTime, THRESH_HOLD) |  | ||||||
|     ) { |  | ||||||
|       setStatus({ |  | ||||||
|         status: "Check-Out", |  | ||||||
|         action: ACTIONS.CHECK_OUT, |  | ||||||
|         disabled: false, |  | ||||||
|         text: "Check Out", |  | ||||||
|         color: "btn-primary", |  | ||||||
|       }); |  | ||||||
|     } else if ( |  | ||||||
|       activity === 1 && |  | ||||||
|       checkInTime !== null && |  | ||||||
|       checkOutTime === null && |  | ||||||
|       timeElapsed(checkInTime, THRESH_HOLD) |  | ||||||
|     ) { |  | ||||||
|       setStatus({ |  | ||||||
|           status: "Request Regularize", |           status: "Request Regularize", | ||||||
|           action: ACTIONS.REGULARIZATION, |           action: ACTIONS.REGULARIZATION, | ||||||
|           disabled: false, |           disabled: false, | ||||||
|           text: "Regularize", |           text: "Regularize", | ||||||
|         color: "btn-warning", |           color: 'btn-warning', | ||||||
|         }); |         }); | ||||||
|     } else if ( |       } else { | ||||||
|       activity === 4 && |         return setStatus({ | ||||||
|       checkInTime !== null && |           status: "Check-Out", | ||||||
|       checkOutTime !== null && |           action: ACTIONS.CHECK_OUT, | ||||||
|       !timeElapsed(checkInTime, THRESH_HOLD) |           disabled: false, | ||||||
|     ) { |           text: "Check Out", | ||||||
|       if ( |           color: 'btn-primary', | ||||||
|         activity === 4 && |         }); | ||||||
|         checkInTime !== null && |       } | ||||||
|         checkOutTime !== null && |     } | ||||||
|         new Date(checkOutTime).toDateString() !== new Date().toDateString() | 
 | ||||||
|       ) { |     // 3. Already checked in and out → Handle activity === 4 (Approved)
 | ||||||
|         setStatus({ |     if (checkInTime && checkOutTime && activity === 4) { | ||||||
|  |       if (!isSameDay(checkOutTime, now) && !timeElapsed(checkOutTime, THRESH_HOLD)) { | ||||||
|  |         return setStatus({ | ||||||
|           status: "Approved", |           status: "Approved", | ||||||
|           action: ACTIONS.APPROVED, |           action: ACTIONS.APPROVED, | ||||||
|           disabled: true, |           disabled: true, | ||||||
|           text: "Approved", |           text: "Approved", | ||||||
|           color: "btn-success", |           color: 'btn-success', | ||||||
|         }); |         }); | ||||||
|       } else { |       } else if (isSameDay(checkOutTime, now)) { | ||||||
|         setStatus({ |         return setStatus({ | ||||||
|           status: "Check-In", |           status: "Check-In", | ||||||
|           action: ACTIONS.CHECK_IN, |           action: ACTIONS.CHECK_IN, | ||||||
|           disabled: false, |           disabled: false, | ||||||
|           text: "Check In", |           text: "Check In", | ||||||
|           color: "btn-primary", |           color: 'btn-primary', | ||||||
|         }); |         }); | ||||||
|       } |       } | ||||||
|     } else if (activity === 2 && checkInTime !== null) { |     } | ||||||
|       setStatus({ | 
 | ||||||
|  |     // 4. Regularization Requested
 | ||||||
|  |     if (activity === 2) { | ||||||
|  |       return setStatus({ | ||||||
|         status: "Requested", |         status: "Requested", | ||||||
|         action: ACTIONS.REQUESTED, |         action: ACTIONS.REQUESTED, | ||||||
|         disabled: true, |         disabled: true, | ||||||
|         text: "Requested", |         text: "Requested", | ||||||
|         color: "btn-info", |         color: 'btn-info', | ||||||
|       }); |       }); | ||||||
|     } else if (activity === 5 && checkInTime !== null) { |     } | ||||||
|       setStatus({ | 
 | ||||||
|  |     // 5. Rejected Regularization
 | ||||||
|  |     if (activity === 5) { | ||||||
|  |       return setStatus({ | ||||||
|         status: "Rejected", |         status: "Rejected", | ||||||
|         action: ACTIONS.REJECTED, |         action: ACTIONS.REJECTED, | ||||||
|         disabled: true, |         disabled: true, | ||||||
|         text: "Rejected", |         text: "Rejected", | ||||||
|         color: "btn-danger", |         color: 'btn-danger', | ||||||
|       }); |       }); | ||||||
|     } else { |     } | ||||||
|       setStatus({ | 
 | ||||||
|  |     // Default to Approved if none of the above apply
 | ||||||
|  |     return setStatus({ | ||||||
|       status: "Approved", |       status: "Approved", | ||||||
|       action: ACTIONS.APPROVED, |       action: ACTIONS.APPROVED, | ||||||
|       disabled: true, |       disabled: true, | ||||||
|       text: "Approved", |       text: "Approved", | ||||||
|         color: "btn-success", |       color: 'btn-success', | ||||||
|     }); |     }); | ||||||
|     } | 
 | ||||||
|   }, [attendanceData]); |   }, [attendanceData]); | ||||||
| 
 | 
 | ||||||
|   return status; |   return status; | ||||||
|  | |||||||
		Loading…
	
	
			
			x
			
			
		
	
		Reference in New Issue
	
	Block a user